blob: 86e4a204e4489b4842d7d773a0ca6dde926eec9c [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one or more contributor license
* agreements. See the NOTICE file distributed with this work for additional information regarding
* copyright ownership. The ASF licenses this file to You under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the License. You may obtain a
* copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License
* is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
* or implied. See the License for the specific language governing permissions and limitations under
* the License.
*/
package org.apache.geode.management.cli;
import java.util.Collections;
import java.util.List;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import org.apache.shiro.subject.Subject;
import org.springframework.shell.core.CommandMarker;
import org.apache.geode.annotations.Experimental;
import org.apache.geode.cache.Cache;
import org.apache.geode.cache.execute.Execution;
import org.apache.geode.cache.execute.Function;
import org.apache.geode.cache.execute.FunctionService;
import org.apache.geode.cache.execute.ResultCollector;
import org.apache.geode.distributed.ConfigurationPersistenceService;
import org.apache.geode.distributed.DistributedMember;
import org.apache.geode.distributed.internal.InternalLocator;
import org.apache.geode.internal.cache.InternalCache;
import org.apache.geode.internal.serialization.KnownVersion;
import org.apache.geode.management.ManagementService;
import org.apache.geode.management.api.ClusterManagementService;
import org.apache.geode.management.internal.cli.shell.Gfsh;
import org.apache.geode.management.internal.exceptions.EntityNotFoundException;
import org.apache.geode.management.internal.functions.CliFunctionResult;
import org.apache.geode.management.internal.i18n.CliStrings;
import org.apache.geode.management.internal.util.ManagementUtils;
import org.apache.geode.security.ResourcePermission;
@Experimental
public abstract class GfshCommand implements CommandMarker {
public static final String EXPERIMENTAL = "(Experimental) ";
private InternalCache cache;
public boolean isOnlineCommandAvailable() {
Gfsh gfsh = Gfsh.getCurrentInstance();
// command should always be available on the server
if (gfsh == null) {
return true;
}
// if in gfshVM, only when gfsh is connected and ready
return gfsh.isConnectedAndReady();
}
/**
* For those commands that could possibly change the cluster configuration, they need to
* override this method to return true.
*
* @return whether the command may change the cluster configuration
*/
public boolean affectsClusterConfiguration() {
return false;
}
public void authorize(ResourcePermission.Resource resource,
ResourcePermission.Operation operation, ResourcePermission.Target target) {
cache.getSecurityService().authorize(resource, operation, target);
}
public void authorize(ResourcePermission.Resource resource,
ResourcePermission.Operation operation, String target) {
cache.getSecurityService().authorize(resource, operation, target);
}
public void authorize(ResourcePermission.Resource resource,
ResourcePermission.Operation operation, String target, String key) {
cache.getSecurityService().authorize(resource, operation, target, key);
}
public Cache getCache() {
if (cache == null) {
return null;
}
return cache.getCacheForProcessingClientRequests();
}
@SuppressWarnings("unchecked")
public <T extends ManagementService> T getManagementService() {
return (T) ManagementService.getExistingManagementService(cache);
}
@SuppressWarnings("unchecked")
public <T extends ConfigurationPersistenceService> T getConfigurationPersistenceService() {
InternalLocator locator = InternalLocator.getLocator();
return locator == null ? null : (T) locator.getConfigurationPersistenceService();
}
public ClusterManagementService getClusterManagementService() {
InternalLocator locator = InternalLocator.getLocator();
return locator == null ? null : locator.getClusterManagementService();
}
public void setCache(Cache cache) {
this.cache = (InternalCache) cache;
}
public boolean isSharedConfigurationRunning() {
InternalLocator locator = InternalLocator.getLocator();
return locator != null && locator.isSharedConfigurationRunning();
}
public Subject getSubject() {
return cache.getSecurityService().getSubject();
}
/**
* this either returns a non-null member or throw an exception if member is not found.
*
* @param memberName the member name of the member to find
* @return the member
*/
public DistributedMember getMember(final String memberName) {
DistributedMember member = findMember(memberName);
if (member == null) {
throw new EntityNotFoundException(
CliStrings.format(CliStrings.MEMBER_NOT_FOUND_ERROR_MESSAGE, memberName));
}
return member;
}
/**
* this will return the member found or null if no member with that name
*
* @param memberName the member name of the member to find
* @return the member found or null if no member with that name exists
*/
public DistributedMember findMember(final String memberName) {
return ManagementUtils.getDistributedMemberByNameOrId(memberName, (InternalCache) getCache());
}
/**
* Gets all members in the GemFire distributed system/cache, including locators
*
* @return a set of all members
*/
public Set<DistributedMember> getAllMembers() {
return ManagementUtils.getAllMembers(cache);
}
/**
* Get All members, excluding locators
*
* @return a set of all non-locator members
*/
public Set<DistributedMember> getAllNormalMembers() {
return ManagementUtils.getAllNormalMembers(cache);
}
/**
* Get All members &gt;= a specific version, excluding locators
*
* @param version a {@link KnownVersion} to compare with the version of members
* @return a set of non-locator members with version equal to or newer than the given version
*/
public Set<DistributedMember> getNormalMembersWithSameOrNewerVersion(KnownVersion version) {
return ManagementUtils.getNormalMembersWithSameOrNewerVersion(cache, version);
}
@SuppressWarnings("rawtypes")
public Execution getMembersFunctionExecutor(final Set<DistributedMember> members) {
return FunctionService.onMembers(members);
}
/**
* if no members matches these names, an empty set would return, this does not include locators
*
* @param groups names of groups to which returned members belong
* @param members member names or IDs which returned members match
* @return a set of matching members
*/
public Set<DistributedMember> findMembers(String[] groups, String[] members) {
return ManagementUtils.findMembers(groups, members, cache);
}
public Set<DistributedMember> findAllOtherLocators() {
Set<DistributedMember> allLocators = findAllLocators();
String thisId = getCache().getDistributedSystem().getDistributedMember().getId();
return allLocators.stream().filter(m -> !(m.getId().equals(thisId)))
.collect(Collectors.toSet());
}
public Set<DistributedMember> findAllLocators() {
return ManagementUtils.getAllLocators(cache);
}
/**
* if no members matches these names, a UserErrorException will be thrown
*
* @param groups names of groups to which returned members belong
* @param members member names or IDs which returned members match
* @return a set of matching members
*/
public Set<DistributedMember> getMembers(String[] groups, String[] members) {
Set<DistributedMember> matchingMembers = findMembers(groups, members);
if (matchingMembers.size() == 0) {
throw new EntityNotFoundException(CliStrings.NO_MEMBERS_FOUND_MESSAGE);
}
return matchingMembers;
}
/**
* if no members matches these names, an empty set would return
*
* @param groups names of groups to which returned members belong
* @param members member names or IDs which returned members match
* @return a set of matching members
*/
public Set<DistributedMember> findMembersIncludingLocators(String[] groups, String[] members) {
return ManagementUtils.findMembersIncludingLocators(groups, members,
(InternalCache) getCache());
}
/**
* if no members matches these names, a UserErrorException will be thrown
*
* @param groups names of groups to which returned members belong
* @param members member names or IDs which returned members match
* @return a set of matching members
*/
public Set<DistributedMember> getMembersIncludingLocators(String[] groups, String[] members) {
Set<DistributedMember> matchingMembers = findMembersIncludingLocators(groups, members);
if (matchingMembers.size() == 0) {
throw new EntityNotFoundException(CliStrings.NO_MEMBERS_FOUND_MESSAGE);
}
return matchingMembers;
}
public Set<DistributedMember> findMembersForRegion(String regionPath) {
return ManagementUtils.getRegionAssociatedMembers(regionPath, cache, true);
}
public Set<DistributedMember> findAnyMembersForRegion(String regionPath) {
return ManagementUtils.getRegionAssociatedMembers(regionPath, cache, false);
}
public ResultCollector<?, ?> executeFunction(Function<?> function, Object args,
final Set<DistributedMember> targetMembers) {
return ManagementUtils.executeFunction(function, args, targetMembers);
}
public ResultCollector<?, ?> executeFunction(Function<?> function, Object args,
final DistributedMember targetMember) {
return executeFunction(function, args, Collections.singleton(targetMember));
}
public CliFunctionResult executeFunctionAndGetFunctionResult(Function<?> function, Object args,
final DistributedMember targetMember) {
ResultCollector<?, ?> rc = executeFunction(function, args, Collections.singleton(targetMember));
List<CliFunctionResult> results = CliFunctionResult.cleanResults((List<?>) rc.getResult());
return results.size() > 0 ? results.get(0) : null;
}
public List<CliFunctionResult> executeAndGetFunctionResult(Function<?> function, Object args,
Set<DistributedMember> targetMembers) {
ResultCollector<?, ?> rc = executeFunction(function, args, targetMembers);
return CliFunctionResult.cleanResults((List<?>) rc.getResult());
}
/**
* Very basic polling functionality that executes a function until it returns true or the timeout
* is reached. The polling call is performed on the calling thread. Do not use it with a
* function that may have an unbounded runtime. The timeout is very coarse and will not account
* for the function overrunning the given time.
*
* @param timeout the maximum amount of time to wait for the function to return
* @param unit the {@link TimeUnit} for the timeout
* @param function a {@link Supplier Supplier&lt;Boolean&gt;} function that will poll for the
* condition
* @return true if the function returns true within the timeout period; false otherwise
*/
public boolean poll(long timeout, TimeUnit unit, Supplier<Boolean> function) {
long startWaitTime = System.currentTimeMillis();
long waitTime = unit.toMillis(timeout);
do {
try {
if (function.get()) {
return true;
}
Thread.sleep(100);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
return false;
}
} while (System.currentTimeMillis() - startWaitTime < waitTime);
return false;
}
}