YARN-11235. Refactor Policy Code and Define getReservationHomeSubcluster (#4656)
This commit is contained in:
parent
0aa08ef543
commit
6f7c4c74ea
|
@ -20,7 +20,7 @@ package org.apache.hadoop.yarn.server.federation.policies;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.Collection;
|
||||||
import java.util.Random;
|
import java.util.Random;
|
||||||
|
|
||||||
import org.apache.hadoop.classification.InterfaceAudience.Private;
|
import org.apache.hadoop.classification.InterfaceAudience.Private;
|
||||||
|
@ -188,8 +188,8 @@ public final class FederationPolicyUtils {
|
||||||
* @throws FederationPolicyException if there are no usable subclusters.
|
* @throws FederationPolicyException if there are no usable subclusters.
|
||||||
*/
|
*/
|
||||||
public static void validateSubClusterAvailability(
|
public static void validateSubClusterAvailability(
|
||||||
List<SubClusterId> activeSubClusters,
|
Collection<SubClusterId> activeSubClusters,
|
||||||
List<SubClusterId> blackListSubClusters)
|
Collection<SubClusterId> blackListSubClusters)
|
||||||
throws FederationPolicyException {
|
throws FederationPolicyException {
|
||||||
if (activeSubClusters != null && !activeSubClusters.isEmpty()) {
|
if (activeSubClusters != null && !activeSubClusters.isEmpty()) {
|
||||||
if (blackListSubClusters == null) {
|
if (blackListSubClusters == null) {
|
||||||
|
|
|
@ -25,6 +25,7 @@ import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
|
||||||
import org.apache.hadoop.conf.Configuration;
|
import org.apache.hadoop.conf.Configuration;
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
||||||
|
import org.apache.hadoop.yarn.api.protocolrecords.ReservationSubmissionRequest;
|
||||||
import org.apache.hadoop.yarn.conf.YarnConfiguration;
|
import org.apache.hadoop.yarn.conf.YarnConfiguration;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
|
@ -136,7 +137,7 @@ public class RouterPolicyFacade {
|
||||||
|
|
||||||
if (appSubmissionContext == null) {
|
if (appSubmissionContext == null) {
|
||||||
throw new FederationPolicyException(
|
throw new FederationPolicyException(
|
||||||
"The ApplicationSubmissionContext " + "cannot be null.");
|
"The ApplicationSubmissionContext cannot be null.");
|
||||||
}
|
}
|
||||||
|
|
||||||
String queue = appSubmissionContext.getQueue();
|
String queue = appSubmissionContext.getQueue();
|
||||||
|
@ -148,51 +149,7 @@ public class RouterPolicyFacade {
|
||||||
queue = YarnConfiguration.DEFAULT_QUEUE_NAME;
|
queue = YarnConfiguration.DEFAULT_QUEUE_NAME;
|
||||||
}
|
}
|
||||||
|
|
||||||
// the facade might cache this request, based on its parameterization
|
FederationRouterPolicy policy = getFederationRouterPolicy(cachedConfs, policyMap, queue);
|
||||||
SubClusterPolicyConfiguration configuration = null;
|
|
||||||
|
|
||||||
try {
|
|
||||||
configuration = federationFacade.getPolicyConfiguration(queue);
|
|
||||||
} catch (YarnException e) {
|
|
||||||
String errMsg = "There is no policy configured for the queue: " + queue
|
|
||||||
+ ", falling back to defaults.";
|
|
||||||
LOG.warn(errMsg, e);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If there is no policy configured for this queue, fallback to the baseline
|
|
||||||
// policy that is configured either in the store or via XML config (and
|
|
||||||
// cached)
|
|
||||||
if (configuration == null) {
|
|
||||||
LOG.warn("There is no policies configured for queue: " + queue + " we"
|
|
||||||
+ " fallback to default policy for: "
|
|
||||||
+ YarnConfiguration.DEFAULT_FEDERATION_POLICY_KEY);
|
|
||||||
|
|
||||||
queue = YarnConfiguration.DEFAULT_FEDERATION_POLICY_KEY;
|
|
||||||
try {
|
|
||||||
configuration = federationFacade.getPolicyConfiguration(queue);
|
|
||||||
} catch (YarnException e) {
|
|
||||||
String errMsg = "Cannot retrieve policy configured for the queue: "
|
|
||||||
+ queue + ", falling back to defaults.";
|
|
||||||
LOG.warn(errMsg, e);
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// the fallback is not configure via store, but via XML, using
|
|
||||||
// previously loaded configuration.
|
|
||||||
if (configuration == null) {
|
|
||||||
configuration =
|
|
||||||
cachedConfs.get(YarnConfiguration.DEFAULT_FEDERATION_POLICY_KEY);
|
|
||||||
}
|
|
||||||
|
|
||||||
// if the configuration has changed since last loaded, reinit the policy
|
|
||||||
// based on current configuration
|
|
||||||
if (!cachedConfs.containsKey(queue)
|
|
||||||
|| !cachedConfs.get(queue).equals(configuration)) {
|
|
||||||
singlePolicyReinit(policyMap, cachedConfs, queue, configuration);
|
|
||||||
}
|
|
||||||
|
|
||||||
FederationRouterPolicy policy = policyMap.get(queue);
|
|
||||||
if (policy == null) {
|
if (policy == null) {
|
||||||
// this should never happen, as the to maps are updated together
|
// this should never happen, as the to maps are updated together
|
||||||
throw new FederationPolicyException("No FederationRouterPolicy found "
|
throw new FederationPolicyException("No FederationRouterPolicy found "
|
||||||
|
@ -262,4 +219,92 @@ public class RouterPolicyFacade {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method provides a wrapper of all policy functionalities for routing a
|
||||||
|
* reservation. Internally it manages configuration changes, and policy
|
||||||
|
* init/reinit.
|
||||||
|
*
|
||||||
|
* @param request the reservation to route.
|
||||||
|
*
|
||||||
|
* @return the id of the subcluster that will be the "home" for this
|
||||||
|
* reservation.
|
||||||
|
*
|
||||||
|
* @throws YarnException if there are issues initializing policies, or no
|
||||||
|
* valid sub-cluster id could be found for this reservation.
|
||||||
|
*/
|
||||||
|
public SubClusterId getReservationHomeSubCluster(
|
||||||
|
ReservationSubmissionRequest request) throws YarnException {
|
||||||
|
|
||||||
|
// the maps are concurrent, but we need to protect from reset()
|
||||||
|
// reinitialization mid-execution by creating a new reference local to this
|
||||||
|
// method.
|
||||||
|
Map<String, SubClusterPolicyConfiguration> cachedConfs = globalConfMap;
|
||||||
|
Map<String, FederationRouterPolicy> policyMap = globalPolicyMap;
|
||||||
|
|
||||||
|
if (request == null) {
|
||||||
|
throw new FederationPolicyException(
|
||||||
|
"The ReservationSubmissionRequest cannot be null.");
|
||||||
|
}
|
||||||
|
|
||||||
|
String queue = request.getQueue();
|
||||||
|
FederationRouterPolicy policy = getFederationRouterPolicy(cachedConfs, policyMap, queue);
|
||||||
|
|
||||||
|
if (policy == null) {
|
||||||
|
// this should never happen, as the to maps are updated together
|
||||||
|
throw new FederationPolicyException("No FederationRouterPolicy found "
|
||||||
|
+ "for queue: " + request.getQueue() + " (while routing "
|
||||||
|
+ "reservation: " + request.getReservationId() + ") "
|
||||||
|
+ "and no default specified.");
|
||||||
|
}
|
||||||
|
|
||||||
|
return policy.getReservationHomeSubcluster(request);
|
||||||
|
}
|
||||||
|
|
||||||
|
private FederationRouterPolicy getFederationRouterPolicy(
|
||||||
|
Map<String, SubClusterPolicyConfiguration> cachedConfiguration,
|
||||||
|
Map<String, FederationRouterPolicy> policyMap, String queue)
|
||||||
|
throws FederationPolicyInitializationException {
|
||||||
|
|
||||||
|
// the facade might cache this request, based on its parameterization
|
||||||
|
SubClusterPolicyConfiguration configuration = null;
|
||||||
|
String copyQueue = queue;
|
||||||
|
|
||||||
|
try {
|
||||||
|
configuration = federationFacade.getPolicyConfiguration(copyQueue);
|
||||||
|
} catch (YarnException e) {
|
||||||
|
LOG.warn("There is no policy configured for the queue: {}, falling back to defaults.",
|
||||||
|
copyQueue, e);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If there is no policy configured for this queue, fallback to the baseline
|
||||||
|
// policy that is configured either in the store or via XML config (and cached)
|
||||||
|
if (configuration == null) {
|
||||||
|
final String policyKey = YarnConfiguration.DEFAULT_FEDERATION_POLICY_KEY;
|
||||||
|
LOG.warn("There is no policies configured for queue: {} " +
|
||||||
|
"we fallback to default policy for: {}. ", copyQueue, policyKey);
|
||||||
|
copyQueue = YarnConfiguration.DEFAULT_FEDERATION_POLICY_KEY;
|
||||||
|
try {
|
||||||
|
configuration = federationFacade.getPolicyConfiguration(copyQueue);
|
||||||
|
} catch (YarnException e) {
|
||||||
|
LOG.warn("Cannot retrieve policy configured for the queue: {}, falling back to defaults.",
|
||||||
|
copyQueue, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// the fallback is not configure via store, but via XML, using
|
||||||
|
// previously loaded configuration.
|
||||||
|
if (configuration == null) {
|
||||||
|
configuration = cachedConfiguration.get(YarnConfiguration.DEFAULT_FEDERATION_POLICY_KEY);
|
||||||
|
}
|
||||||
|
|
||||||
|
// if the configuration has changed since last loaded, reinit the policy
|
||||||
|
// based on current configuration
|
||||||
|
SubClusterPolicyConfiguration policyConfiguration =
|
||||||
|
cachedConfiguration.getOrDefault(copyQueue, null);
|
||||||
|
if (policyConfiguration == null || !policyConfiguration.equals(configuration)) {
|
||||||
|
singlePolicyReinit(policyMap, cachedConfiguration, copyQueue, configuration);
|
||||||
|
}
|
||||||
|
|
||||||
|
return policyMap.get(copyQueue);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,15 +18,22 @@
|
||||||
|
|
||||||
package org.apache.hadoop.yarn.server.federation.policies.router;
|
package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.apache.hadoop.yarn.api.protocolrecords.ReservationSubmissionRequest;
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
||||||
|
import org.apache.hadoop.yarn.api.records.ReservationId;
|
||||||
import org.apache.hadoop.yarn.conf.YarnConfiguration;
|
import org.apache.hadoop.yarn.conf.YarnConfiguration;
|
||||||
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.AbstractConfigurableFederationPolicy;
|
import org.apache.hadoop.yarn.server.federation.policies.AbstractConfigurableFederationPolicy;
|
||||||
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyUtils;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
||||||
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
||||||
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Base abstract class for {@link FederationRouterPolicy} implementations, that
|
* Base abstract class for {@link FederationRouterPolicy} implementations, that
|
||||||
|
@ -63,4 +70,107 @@ public abstract class AbstractRouterPolicy extends
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method is implemented by the specific policy, and it is used to route
|
||||||
|
* both reservations, and applications among a given set of
|
||||||
|
* sub-clusters.
|
||||||
|
*
|
||||||
|
* @param queue the queue for this application/reservation
|
||||||
|
* @param preSelectSubClusters a pre-filter set of sub-clusters
|
||||||
|
* @return the chosen sub-cluster
|
||||||
|
*
|
||||||
|
* @throws YarnException if the policy fails to choose a sub-cluster
|
||||||
|
*/
|
||||||
|
protected abstract SubClusterId chooseSubCluster(String queue,
|
||||||
|
Map<SubClusterId, SubClusterInfo> preSelectSubClusters) throws YarnException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Filter chosen SubCluster based on reservationId.
|
||||||
|
*
|
||||||
|
* @param reservationId the globally unique identifier for a reservation.
|
||||||
|
* @param activeSubClusters the map of ids to info for all active subclusters.
|
||||||
|
* @return the chosen sub-cluster
|
||||||
|
* @throws YarnException if the policy fails to choose a sub-cluster
|
||||||
|
*/
|
||||||
|
protected Map<SubClusterId, SubClusterInfo> prefilterSubClusters(
|
||||||
|
ReservationId reservationId, Map<SubClusterId, SubClusterInfo> activeSubClusters)
|
||||||
|
throws YarnException {
|
||||||
|
|
||||||
|
// if a reservation exists limit scope to the sub-cluster this
|
||||||
|
// reservation is mapped to
|
||||||
|
// TODO: Implemented in YARN-11236
|
||||||
|
return activeSubClusters;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Simply picks from alphabetically-sorted active subclusters based on the
|
||||||
|
* hash of quey name. Jobs of the same queue will all be routed to the same
|
||||||
|
* sub-cluster, as far as the number of active sub-cluster and their names
|
||||||
|
* remain the same.
|
||||||
|
*
|
||||||
|
* @param appContext the {@link ApplicationSubmissionContext} that
|
||||||
|
* has to be routed to an appropriate subCluster for execution.
|
||||||
|
*
|
||||||
|
* @param blackLists the list of subClusters as identified by
|
||||||
|
* {@link SubClusterId} to blackList from the selection of the home
|
||||||
|
* subCluster.
|
||||||
|
*
|
||||||
|
* @return a hash-based chosen {@link SubClusterId} that will be the "home"
|
||||||
|
* for this application.
|
||||||
|
*
|
||||||
|
* @throws YarnException if there are no active subclusters.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public SubClusterId getHomeSubcluster(ApplicationSubmissionContext appContext,
|
||||||
|
List<SubClusterId> blackLists) throws YarnException {
|
||||||
|
|
||||||
|
// null checks and default-queue behavior
|
||||||
|
validate(appContext);
|
||||||
|
|
||||||
|
// apply filtering based on reservation location and active sub-clusters
|
||||||
|
Map<SubClusterId, SubClusterInfo> filteredSubClusters = prefilterSubClusters(
|
||||||
|
appContext.getReservationID(), getActiveSubclusters());
|
||||||
|
|
||||||
|
FederationPolicyUtils.validateSubClusterAvailability(filteredSubClusters.keySet(), blackLists);
|
||||||
|
|
||||||
|
// remove black SubCluster
|
||||||
|
if (blackLists != null) {
|
||||||
|
blackLists.forEach(filteredSubClusters::remove);
|
||||||
|
}
|
||||||
|
|
||||||
|
// pick the chosen subCluster from the active ones
|
||||||
|
return chooseSubCluster(appContext.getQueue(), filteredSubClusters);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method provides a wrapper of all policy functionalities for routing a
|
||||||
|
* reservation. Internally it manages configuration changes, and policy
|
||||||
|
* init/reinit.
|
||||||
|
*
|
||||||
|
* @param request the reservation to route.
|
||||||
|
*
|
||||||
|
* @return the id of the subcluster that will be the "home" for this
|
||||||
|
* reservation.
|
||||||
|
*
|
||||||
|
* @throws YarnException if there are issues initializing policies, or no
|
||||||
|
* valid sub-cluster id could be found for this reservation.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public SubClusterId getReservationHomeSubcluster(ReservationSubmissionRequest request)
|
||||||
|
throws YarnException {
|
||||||
|
if (request == null) {
|
||||||
|
throw new FederationPolicyException("The ReservationSubmissionRequest cannot be null.");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (request.getQueue() == null) {
|
||||||
|
request.setQueue(YarnConfiguration.DEFAULT_QUEUE_NAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
// apply filtering based on reservation location and active sub-clusters
|
||||||
|
Map<SubClusterId, SubClusterInfo> filteredSubClusters = prefilterSubClusters(
|
||||||
|
request.getReservationId(), getActiveSubclusters());
|
||||||
|
|
||||||
|
// pick the chosen subCluster from the active ones
|
||||||
|
return chooseSubCluster(request.getQueue(), filteredSubClusters);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,7 @@ package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.apache.hadoop.yarn.api.protocolrecords.ReservationSubmissionRequest;
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.ConfigurableFederationPolicy;
|
import org.apache.hadoop.yarn.server.federation.policies.ConfigurableFederationPolicy;
|
||||||
|
@ -49,4 +50,16 @@ public interface FederationRouterPolicy extends ConfigurableFederationPolicy {
|
||||||
SubClusterId getHomeSubcluster(
|
SubClusterId getHomeSubcluster(
|
||||||
ApplicationSubmissionContext appSubmissionContext,
|
ApplicationSubmissionContext appSubmissionContext,
|
||||||
List<SubClusterId> blackListSubClusters) throws YarnException;
|
List<SubClusterId> blackListSubClusters) throws YarnException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determines the sub-cluster where a ReservationSubmissionRequest should be
|
||||||
|
* sent to.
|
||||||
|
*
|
||||||
|
* @param request the original request
|
||||||
|
* @return a mapping of sub-clusters and the requests
|
||||||
|
*
|
||||||
|
* @throws YarnException if the policy fails to choose a sub-cluster
|
||||||
|
*/
|
||||||
|
SubClusterId getReservationHomeSubcluster(
|
||||||
|
ReservationSubmissionRequest request) throws YarnException;
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,11 +22,9 @@ import java.util.Collections;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContext;
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContext;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContextValidator;
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContextValidator;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyUtils;
|
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
||||||
|
@ -50,53 +48,12 @@ public class HashBasedRouterPolicy extends AbstractRouterPolicy {
|
||||||
setPolicyContext(federationPolicyContext);
|
setPolicyContext(federationPolicyContext);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Simply picks from alphabetically-sorted active subclusters based on the
|
|
||||||
* hash of quey name. Jobs of the same queue will all be routed to the same
|
|
||||||
* sub-cluster, as far as the number of active sub-cluster and their names
|
|
||||||
* remain the same.
|
|
||||||
*
|
|
||||||
* @param appSubmissionContext the {@link ApplicationSubmissionContext} that
|
|
||||||
* has to be routed to an appropriate subCluster for execution.
|
|
||||||
*
|
|
||||||
* @param blackListSubClusters the list of subClusters as identified by
|
|
||||||
* {@link SubClusterId} to blackList from the selection of the home
|
|
||||||
* subCluster.
|
|
||||||
*
|
|
||||||
* @return a hash-based chosen {@link SubClusterId} that will be the "home"
|
|
||||||
* for this application.
|
|
||||||
*
|
|
||||||
* @throws YarnException if there are no active subclusters.
|
|
||||||
*/
|
|
||||||
@Override
|
@Override
|
||||||
public SubClusterId getHomeSubcluster(
|
protected SubClusterId chooseSubCluster(String queue,
|
||||||
ApplicationSubmissionContext appSubmissionContext,
|
Map<SubClusterId, SubClusterInfo> preSelectSubclusters) throws YarnException {
|
||||||
List<SubClusterId> blackListSubClusters) throws YarnException {
|
int chosenPosition = Math.abs(queue.hashCode() % preSelectSubclusters.size());
|
||||||
|
List<SubClusterId> list = new ArrayList<>(preSelectSubclusters.keySet());
|
||||||
// throws if no active subclusters available
|
|
||||||
Map<SubClusterId, SubClusterInfo> activeSubclusters =
|
|
||||||
getActiveSubclusters();
|
|
||||||
|
|
||||||
FederationPolicyUtils.validateSubClusterAvailability(
|
|
||||||
new ArrayList<SubClusterId>(activeSubclusters.keySet()),
|
|
||||||
blackListSubClusters);
|
|
||||||
|
|
||||||
if (blackListSubClusters != null) {
|
|
||||||
|
|
||||||
// Remove from the active SubClusters from StateStore the blacklisted ones
|
|
||||||
for (SubClusterId scId : blackListSubClusters) {
|
|
||||||
activeSubclusters.remove(scId);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
validate(appSubmissionContext);
|
|
||||||
|
|
||||||
int chosenPosition = Math.abs(
|
|
||||||
appSubmissionContext.getQueue().hashCode() % activeSubclusters.size());
|
|
||||||
|
|
||||||
List<SubClusterId> list = new ArrayList<>(activeSubclusters.keySet());
|
|
||||||
Collections.sort(list);
|
Collections.sort(list);
|
||||||
return list.get(chosenPosition);
|
return list.get(chosenPosition);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,14 +17,10 @@
|
||||||
|
|
||||||
package org.apache.hadoop.yarn.server.federation.policies.router;
|
package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContext;
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContext;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyUtils;
|
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
||||||
|
@ -65,28 +61,12 @@ public class LoadBasedRouterPolicy extends AbstractRouterPolicy {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public SubClusterId getHomeSubcluster(
|
protected SubClusterId chooseSubCluster(
|
||||||
ApplicationSubmissionContext appSubmissionContext,
|
String queue, Map<SubClusterId, SubClusterInfo> preSelectSubclusters) throws YarnException {
|
||||||
List<SubClusterId> blacklist) throws YarnException {
|
Map<SubClusterIdInfo, Float> weights = getPolicyInfo().getRouterPolicyWeights();
|
||||||
|
|
||||||
// null checks and default-queue behavior
|
|
||||||
validate(appSubmissionContext);
|
|
||||||
|
|
||||||
Map<SubClusterId, SubClusterInfo> activeSubclusters =
|
|
||||||
getActiveSubclusters();
|
|
||||||
|
|
||||||
FederationPolicyUtils.validateSubClusterAvailability(
|
|
||||||
new ArrayList<SubClusterId>(activeSubclusters.keySet()), blacklist);
|
|
||||||
|
|
||||||
Map<SubClusterIdInfo, Float> weights =
|
|
||||||
getPolicyInfo().getRouterPolicyWeights();
|
|
||||||
SubClusterIdInfo chosen = null;
|
SubClusterIdInfo chosen = null;
|
||||||
long currBestMem = -1;
|
long currBestMem = -1;
|
||||||
for (Map.Entry<SubClusterId, SubClusterInfo> entry : activeSubclusters
|
for (Map.Entry<SubClusterId, SubClusterInfo> entry : preSelectSubclusters.entrySet()) {
|
||||||
.entrySet()) {
|
|
||||||
if (blacklist != null && blacklist.contains(entry.getKey())) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
SubClusterIdInfo id = new SubClusterIdInfo(entry.getKey());
|
SubClusterIdInfo id = new SubClusterIdInfo(entry.getKey());
|
||||||
if (weights.containsKey(id) && weights.get(id) > 0) {
|
if (weights.containsKey(id) && weights.get(id) > 0) {
|
||||||
long availableMemory = getAvailableMemory(entry.getValue());
|
long availableMemory = getAvailableMemory(entry.getValue());
|
||||||
|
@ -110,7 +90,7 @@ public class LoadBasedRouterPolicy extends AbstractRouterPolicy {
|
||||||
mem = obj.getJSONObject("clusterMetrics").getLong("availableMB");
|
mem = obj.getJSONObject("clusterMetrics").getLong("availableMB");
|
||||||
return mem;
|
return mem;
|
||||||
} catch (JSONException j) {
|
} catch (JSONException j) {
|
||||||
throw new YarnException("FederationSubCluserInfo cannot be parsed", j);
|
throw new YarnException("FederationSubClusterInfo cannot be parsed", j);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -21,6 +21,7 @@ package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
|
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
||||||
|
@ -78,7 +79,7 @@ public class LocalityRouterPolicy extends WeightedRandomRouterPolicy {
|
||||||
resolver = policyContext.getFederationSubclusterResolver();
|
resolver = policyContext.getFederationSubclusterResolver();
|
||||||
Map<SubClusterIdInfo, Float> weights =
|
Map<SubClusterIdInfo, Float> weights =
|
||||||
getPolicyInfo().getRouterPolicyWeights();
|
getPolicyInfo().getRouterPolicyWeights();
|
||||||
enabledSCs = new ArrayList<SubClusterId>();
|
enabledSCs = new ArrayList<>();
|
||||||
for (Map.Entry<SubClusterIdInfo, Float> entry : weights.entrySet()) {
|
for (Map.Entry<SubClusterIdInfo, Float> entry : weights.entrySet()) {
|
||||||
if (entry != null && entry.getValue() > 0) {
|
if (entry != null && entry.getValue() > 0) {
|
||||||
enabledSCs.add(entry.getKey().toId());
|
enabledSCs.add(entry.getKey().toId());
|
||||||
|
@ -100,8 +101,7 @@ public class LocalityRouterPolicy extends WeightedRandomRouterPolicy {
|
||||||
// Fast path for FailForward to WeightedRandomRouterPolicy
|
// Fast path for FailForward to WeightedRandomRouterPolicy
|
||||||
if (rrList == null || rrList.isEmpty() || (rrList.size() == 1
|
if (rrList == null || rrList.isEmpty() || (rrList.size() == 1
|
||||||
&& ResourceRequest.isAnyLocation(rrList.get(0).getResourceName()))) {
|
&& ResourceRequest.isAnyLocation(rrList.get(0).getResourceName()))) {
|
||||||
return super
|
return super.getHomeSubcluster(appSubmissionContext, blackListSubClusters);
|
||||||
.getHomeSubcluster(appSubmissionContext, blackListSubClusters);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (rrList.size() != 3) {
|
if (rrList.size() != 3) {
|
||||||
|
@ -109,12 +109,11 @@ public class LocalityRouterPolicy extends WeightedRandomRouterPolicy {
|
||||||
"Invalid number of resource requests: " + rrList.size());
|
"Invalid number of resource requests: " + rrList.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
Map<SubClusterId, SubClusterInfo> activeSubClusters =
|
Map<SubClusterId, SubClusterInfo> activeSubClusters = getActiveSubclusters();
|
||||||
getActiveSubclusters();
|
Set<SubClusterId> validSubClusters = activeSubClusters.keySet();
|
||||||
List<SubClusterId> validSubClusters =
|
FederationPolicyUtils.validateSubClusterAvailability(activeSubClusters.keySet(),
|
||||||
new ArrayList<>(activeSubClusters.keySet());
|
blackListSubClusters);
|
||||||
FederationPolicyUtils
|
|
||||||
.validateSubClusterAvailability(validSubClusters, blackListSubClusters);
|
|
||||||
if (blackListSubClusters != null) {
|
if (blackListSubClusters != null) {
|
||||||
// Remove from the active SubClusters from StateStore the blacklisted ones
|
// Remove from the active SubClusters from StateStore the blacklisted ones
|
||||||
validSubClusters.removeAll(blackListSubClusters);
|
validSubClusters.removeAll(blackListSubClusters);
|
||||||
|
@ -128,20 +127,21 @@ public class LocalityRouterPolicy extends WeightedRandomRouterPolicy {
|
||||||
ResourceRequest nodeRequest = null;
|
ResourceRequest nodeRequest = null;
|
||||||
ResourceRequest rackRequest = null;
|
ResourceRequest rackRequest = null;
|
||||||
ResourceRequest anyRequest = null;
|
ResourceRequest anyRequest = null;
|
||||||
|
|
||||||
for (ResourceRequest rr : rrList) {
|
for (ResourceRequest rr : rrList) {
|
||||||
// Handle "node" requests
|
// Handle "node" requests
|
||||||
try {
|
try {
|
||||||
targetId = resolver.getSubClusterForNode(rr.getResourceName());
|
targetId = resolver.getSubClusterForNode(rr.getResourceName());
|
||||||
nodeRequest = rr;
|
nodeRequest = rr;
|
||||||
} catch (YarnException e) {
|
} catch (YarnException e) {
|
||||||
LOG.error("Cannot resolve node : {}", e.getLocalizedMessage());
|
LOG.error("Cannot resolve node : {}.", e.getMessage());
|
||||||
}
|
}
|
||||||
// Handle "rack" requests
|
// Handle "rack" requests
|
||||||
try {
|
try {
|
||||||
resolver.getSubClustersForRack(rr.getResourceName());
|
resolver.getSubClustersForRack(rr.getResourceName());
|
||||||
rackRequest = rr;
|
rackRequest = rr;
|
||||||
} catch (YarnException e) {
|
} catch (YarnException e) {
|
||||||
LOG.error("Cannot resolve rack : {}", e.getLocalizedMessage());
|
LOG.error("Cannot resolve rack : {}.", e.getMessage());
|
||||||
}
|
}
|
||||||
// Handle "ANY" requests
|
// Handle "ANY" requests
|
||||||
if (ResourceRequest.isAnyLocation(rr.getResourceName())) {
|
if (ResourceRequest.isAnyLocation(rr.getResourceName())) {
|
||||||
|
@ -149,32 +149,33 @@ public class LocalityRouterPolicy extends WeightedRandomRouterPolicy {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (nodeRequest == null) {
|
if (nodeRequest == null) {
|
||||||
throw new YarnException("Missing node request");
|
throw new YarnException("Missing node request.");
|
||||||
}
|
}
|
||||||
if (rackRequest == null) {
|
if (rackRequest == null) {
|
||||||
throw new YarnException("Missing rack request");
|
throw new YarnException("Missing rack request.");
|
||||||
}
|
}
|
||||||
if (anyRequest == null) {
|
if (anyRequest == null) {
|
||||||
throw new YarnException("Missing any request");
|
throw new YarnException("Missing any request.");
|
||||||
}
|
}
|
||||||
LOG.info(
|
|
||||||
"Node request: " + nodeRequest.getResourceName() + ", Rack request: "
|
LOG.info("Node request: {} , Rack request: {} , Any request: {}.",
|
||||||
+ rackRequest.getResourceName() + ", Any request: " + anyRequest
|
nodeRequest.getResourceName(), rackRequest.getResourceName(),
|
||||||
.getResourceName());
|
anyRequest.getResourceName());
|
||||||
|
|
||||||
// Handle "node" requests
|
// Handle "node" requests
|
||||||
if (validSubClusters.contains(targetId) && enabledSCs
|
if (validSubClusters.contains(targetId) && enabledSCs
|
||||||
.contains(targetId)) {
|
.contains(targetId)) {
|
||||||
LOG.info("Node {} is in SubCluster: {}", nodeRequest.getResourceName(),
|
LOG.info("Node {} is in SubCluster: {}.", nodeRequest.getResourceName(), targetId);
|
||||||
targetId);
|
|
||||||
return targetId;
|
return targetId;
|
||||||
} else {
|
} else {
|
||||||
throw new YarnException("The node " + nodeRequest.getResourceName()
|
throw new YarnException("The node " + nodeRequest.getResourceName()
|
||||||
+ " is in a blacklist SubCluster or not active. ");
|
+ " is in a blacklist SubCluster or not active. ");
|
||||||
}
|
}
|
||||||
} catch (YarnException e) {
|
} catch (YarnException e) {
|
||||||
LOG.error("Validating resource requests failed, Falling back to "
|
LOG.error("Validating resource requests failed, " +
|
||||||
+ "WeightedRandomRouterPolicy placement: " + e.getMessage());
|
"Falling back to WeightedRandomRouterPolicy placement : {}.", e.getMessage());
|
||||||
// FailForward to WeightedRandomRouterPolicy
|
// FailForward to WeightedRandomRouterPolicy
|
||||||
// Overwrite request to use a default ANY
|
// Overwrite request to use a default ANY
|
||||||
ResourceRequest amReq = Records.newRecord(ResourceRequest.class);
|
ResourceRequest amReq = Records.newRecord(ResourceRequest.class);
|
||||||
|
@ -183,14 +184,10 @@ public class LocalityRouterPolicy extends WeightedRandomRouterPolicy {
|
||||||
amReq.setCapability(appSubmissionContext.getResource());
|
amReq.setCapability(appSubmissionContext.getResource());
|
||||||
amReq.setNumContainers(1);
|
amReq.setNumContainers(1);
|
||||||
amReq.setRelaxLocality(true);
|
amReq.setRelaxLocality(true);
|
||||||
amReq.setNodeLabelExpression(
|
amReq.setNodeLabelExpression(appSubmissionContext.getNodeLabelExpression());
|
||||||
appSubmissionContext.getNodeLabelExpression());
|
amReq.setExecutionTypeRequest(ExecutionTypeRequest.newInstance(ExecutionType.GUARANTEED));
|
||||||
amReq.setExecutionTypeRequest(
|
appSubmissionContext.setAMContainerResourceRequests(Collections.singletonList(amReq));
|
||||||
ExecutionTypeRequest.newInstance(ExecutionType.GUARANTEED));
|
return super.getHomeSubcluster(appSubmissionContext, blackListSubClusters);
|
||||||
appSubmissionContext
|
|
||||||
.setAMContainerResourceRequests(Collections.singletonList(amReq));
|
|
||||||
return super
|
|
||||||
.getHomeSubcluster(appSubmissionContext, blackListSubClusters);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,13 +17,9 @@
|
||||||
|
|
||||||
package org.apache.hadoop.yarn.server.federation.policies.router;
|
package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyUtils;
|
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
||||||
|
@ -37,30 +33,15 @@ import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
||||||
public class PriorityRouterPolicy extends AbstractRouterPolicy {
|
public class PriorityRouterPolicy extends AbstractRouterPolicy {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public SubClusterId getHomeSubcluster(
|
protected SubClusterId chooseSubCluster(
|
||||||
ApplicationSubmissionContext appSubmissionContext,
|
String queue, Map<SubClusterId, SubClusterInfo> preSelectSubclusters) throws YarnException {
|
||||||
List<SubClusterId> blacklist) throws YarnException {
|
|
||||||
|
|
||||||
// null checks and default-queue behavior
|
|
||||||
validate(appSubmissionContext);
|
|
||||||
|
|
||||||
Map<SubClusterId, SubClusterInfo> activeSubclusters =
|
|
||||||
getActiveSubclusters();
|
|
||||||
|
|
||||||
FederationPolicyUtils.validateSubClusterAvailability(
|
|
||||||
new ArrayList<SubClusterId>(activeSubclusters.keySet()), blacklist);
|
|
||||||
|
|
||||||
// This finds the sub-cluster with the highest weight among the
|
// This finds the sub-cluster with the highest weight among the
|
||||||
// currently active ones.
|
// currently active ones.
|
||||||
Map<SubClusterIdInfo, Float> weights =
|
Map<SubClusterIdInfo, Float> weights = getPolicyInfo().getRouterPolicyWeights();
|
||||||
getPolicyInfo().getRouterPolicyWeights();
|
|
||||||
SubClusterId chosen = null;
|
SubClusterId chosen = null;
|
||||||
Float currentBest = Float.MIN_VALUE;
|
Float currentBest = Float.MIN_VALUE;
|
||||||
for (SubClusterId id : activeSubclusters.keySet()) {
|
for (SubClusterId id : preSelectSubclusters.keySet()) {
|
||||||
SubClusterIdInfo idInfo = new SubClusterIdInfo(id);
|
SubClusterIdInfo idInfo = new SubClusterIdInfo(id);
|
||||||
if (blacklist != null && blacklist.contains(id)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (weights.containsKey(idInfo) && weights.get(idInfo) > currentBest) {
|
if (weights.containsKey(idInfo) && weights.get(idInfo) > currentBest) {
|
||||||
currentBest = weights.get(idInfo);
|
currentBest = weights.get(idInfo);
|
||||||
chosen = id;
|
chosen = id;
|
||||||
|
@ -68,10 +49,8 @@ public class PriorityRouterPolicy extends AbstractRouterPolicy {
|
||||||
}
|
}
|
||||||
if (chosen == null) {
|
if (chosen == null) {
|
||||||
throw new FederationPolicyException(
|
throw new FederationPolicyException(
|
||||||
"No Active Subcluster with weight vector greater than zero");
|
"No Active Subcluster with weight vector greater than zero.");
|
||||||
}
|
}
|
||||||
|
|
||||||
return chosen;
|
return chosen;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
|
@ -17,15 +17,15 @@
|
||||||
|
|
||||||
package org.apache.hadoop.yarn.server.federation.policies.router;
|
package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContext;
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContext;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContextValidator;
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContextValidator;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This {@link FederationRouterPolicy} simply rejects all incoming requests.
|
* This {@link FederationRouterPolicy} simply rejects all incoming requests.
|
||||||
|
@ -43,34 +43,12 @@ public class RejectRouterPolicy extends AbstractRouterPolicy {
|
||||||
setPolicyContext(federationPolicyContext);
|
setPolicyContext(federationPolicyContext);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* The policy always reject requests.
|
|
||||||
*
|
|
||||||
* @param appSubmissionContext the {@link ApplicationSubmissionContext} that
|
|
||||||
* has to be routed to an appropriate subCluster for execution.
|
|
||||||
*
|
|
||||||
* @param blackListSubClusters the list of subClusters as identified by
|
|
||||||
* {@link SubClusterId} to blackList from the selection of the home
|
|
||||||
* subCluster.
|
|
||||||
*
|
|
||||||
* @return (never).
|
|
||||||
*
|
|
||||||
* @throws YarnException (always) to prevent applications in this queue to be
|
|
||||||
* run anywhere in the federated cluster.
|
|
||||||
*/
|
|
||||||
@Override
|
@Override
|
||||||
public SubClusterId getHomeSubcluster(
|
protected SubClusterId chooseSubCluster(
|
||||||
ApplicationSubmissionContext appSubmissionContext,
|
String queue, Map<SubClusterId, SubClusterInfo> preSelectSubclusters) throws YarnException {
|
||||||
List<SubClusterId> blackListSubClusters) throws YarnException {
|
throw new FederationPolicyException(
|
||||||
|
"The policy configured for this queue (" + queue + ") "
|
||||||
// run standard validation, as error might differ
|
+ "reject all routing requests by construction. Application in "
|
||||||
validate(appSubmissionContext);
|
+ queue + " cannot be routed to any RM.");
|
||||||
|
|
||||||
throw new FederationPolicyException("The policy configured for this queue"
|
|
||||||
+ " (" + appSubmissionContext.getQueue() + ") reject all routing "
|
|
||||||
+ "requests by construction. Application "
|
|
||||||
+ appSubmissionContext.getApplicationId()
|
|
||||||
+ " cannot be routed to any RM.");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,11 +22,10 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Random;
|
import java.util.Random;
|
||||||
|
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContext;
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContext;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContextValidator;
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyInitializationContextValidator;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyUtils;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
||||||
|
@ -55,50 +54,16 @@ public class UniformRandomRouterPolicy extends AbstractRouterPolicy {
|
||||||
this.getClass().getCanonicalName());
|
this.getClass().getCanonicalName());
|
||||||
|
|
||||||
// note: this overrides AbstractRouterPolicy and ignores the weights
|
// note: this overrides AbstractRouterPolicy and ignores the weights
|
||||||
|
|
||||||
setPolicyContext(policyContext);
|
setPolicyContext(policyContext);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Simply picks a random active subCluster to start the AM (this does NOT
|
|
||||||
* depend on the weights in the policy).
|
|
||||||
*
|
|
||||||
* @param appSubmissionContext the {@link ApplicationSubmissionContext} that
|
|
||||||
* has to be routed to an appropriate subCluster for execution.
|
|
||||||
*
|
|
||||||
* @param blackListSubClusters the list of subClusters as identified by
|
|
||||||
* {@link SubClusterId} to blackList from the selection of the home
|
|
||||||
* subCluster.
|
|
||||||
*
|
|
||||||
* @return a randomly chosen subcluster.
|
|
||||||
*
|
|
||||||
* @throws YarnException if there are no active subclusters.
|
|
||||||
*/
|
|
||||||
@Override
|
@Override
|
||||||
public SubClusterId getHomeSubcluster(
|
protected SubClusterId chooseSubCluster(
|
||||||
ApplicationSubmissionContext appSubmissionContext,
|
String queue, Map<SubClusterId, SubClusterInfo> preSelectSubclusters) throws YarnException {
|
||||||
List<SubClusterId> blackListSubClusters) throws YarnException {
|
if (preSelectSubclusters == null || preSelectSubclusters.isEmpty()) {
|
||||||
|
throw new FederationPolicyException("No available subcluster to choose from.");
|
||||||
// null checks and default-queue behavior
|
|
||||||
validate(appSubmissionContext);
|
|
||||||
|
|
||||||
Map<SubClusterId, SubClusterInfo> activeSubclusters =
|
|
||||||
getActiveSubclusters();
|
|
||||||
|
|
||||||
List<SubClusterId> list = new ArrayList<>(activeSubclusters.keySet());
|
|
||||||
|
|
||||||
FederationPolicyUtils.validateSubClusterAvailability(list,
|
|
||||||
blackListSubClusters);
|
|
||||||
|
|
||||||
if (blackListSubClusters != null) {
|
|
||||||
|
|
||||||
// Remove from the active SubClusters from StateStore the blacklisted ones
|
|
||||||
for (SubClusterId scId : blackListSubClusters) {
|
|
||||||
list.remove(scId);
|
|
||||||
}
|
}
|
||||||
}
|
List<SubClusterId> list = new ArrayList<>(preSelectSubclusters.keySet());
|
||||||
|
|
||||||
return list.get(rand.nextInt(list.size()));
|
return list.get(rand.nextInt(list.size()));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,10 +19,8 @@
|
||||||
package org.apache.hadoop.yarn.server.federation.policies.router;
|
package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyUtils;
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyUtils;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
|
@ -35,47 +33,30 @@ import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
||||||
* sub-clusters.
|
* sub-clusters.
|
||||||
*/
|
*/
|
||||||
public class WeightedRandomRouterPolicy extends AbstractRouterPolicy {
|
public class WeightedRandomRouterPolicy extends AbstractRouterPolicy {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public SubClusterId getHomeSubcluster(
|
protected SubClusterId chooseSubCluster(
|
||||||
ApplicationSubmissionContext appSubmissionContext,
|
String queue, Map<SubClusterId, SubClusterInfo> preSelectSubclusters) throws YarnException {
|
||||||
List<SubClusterId> blacklist) throws YarnException {
|
|
||||||
|
|
||||||
// null checks and default-queue behavior
|
// note: we cannot pre-compute the weights, as the set of activeSubCluster
|
||||||
validate(appSubmissionContext);
|
|
||||||
|
|
||||||
Map<SubClusterId, SubClusterInfo> activeSubclusters =
|
|
||||||
getActiveSubclusters();
|
|
||||||
|
|
||||||
FederationPolicyUtils.validateSubClusterAvailability(
|
|
||||||
new ArrayList<SubClusterId>(activeSubclusters.keySet()), blacklist);
|
|
||||||
|
|
||||||
// note: we cannot pre-compute the weights, as the set of activeSubcluster
|
|
||||||
// changes dynamically (and this would unfairly spread the load to
|
// changes dynamically (and this would unfairly spread the load to
|
||||||
// sub-clusters adjacent to an inactive one), hence we need to count/scan
|
// sub-clusters adjacent to an inactive one), hence we need to count/scan
|
||||||
// the list and based on weight pick the next sub-cluster.
|
// the list and based on weight pick the next sub-cluster.
|
||||||
Map<SubClusterIdInfo, Float> weights =
|
Map<SubClusterIdInfo, Float> weights = getPolicyInfo().getRouterPolicyWeights();
|
||||||
getPolicyInfo().getRouterPolicyWeights();
|
|
||||||
|
|
||||||
ArrayList<Float> weightList = new ArrayList<>();
|
ArrayList<Float> weightList = new ArrayList<>();
|
||||||
ArrayList<SubClusterId> scIdList = new ArrayList<>();
|
ArrayList<SubClusterId> scIdList = new ArrayList<>();
|
||||||
for (Map.Entry<SubClusterIdInfo, Float> entry : weights.entrySet()) {
|
for (Map.Entry<SubClusterIdInfo, Float> entry : weights.entrySet()) {
|
||||||
if (blacklist != null && blacklist.contains(entry.getKey().toId())) {
|
SubClusterIdInfo key = entry.getKey();
|
||||||
continue;
|
if (key != null && preSelectSubclusters.containsKey(key.toId())) {
|
||||||
}
|
|
||||||
if (entry.getKey() != null
|
|
||||||
&& activeSubclusters.containsKey(entry.getKey().toId())) {
|
|
||||||
weightList.add(entry.getValue());
|
weightList.add(entry.getValue());
|
||||||
scIdList.add(entry.getKey().toId());
|
scIdList.add(key.toId());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int pickedIndex = FederationPolicyUtils.getWeightedRandom(weightList);
|
int pickedIndex = FederationPolicyUtils.getWeightedRandom(weightList);
|
||||||
if (pickedIndex == -1) {
|
if (pickedIndex == -1) {
|
||||||
throw new FederationPolicyException(
|
throw new FederationPolicyException("No positive weight found on active subclusters");
|
||||||
"No positive weight found on active subclusters");
|
|
||||||
}
|
}
|
||||||
return scIdList.get(pickedIndex);
|
return scIdList.get(pickedIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
package org.apache.hadoop.yarn.server.federation.store.records;
|
package org.apache.hadoop.yarn.server.federation.store.records;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Collection;
|
||||||
|
|
||||||
import org.apache.hadoop.classification.InterfaceAudience.Private;
|
import org.apache.hadoop.classification.InterfaceAudience.Private;
|
||||||
import org.apache.hadoop.classification.InterfaceAudience.Public;
|
import org.apache.hadoop.classification.InterfaceAudience.Public;
|
||||||
|
@ -36,7 +37,7 @@ public abstract class GetSubClustersInfoResponse {
|
||||||
@Public
|
@Public
|
||||||
@Unstable
|
@Unstable
|
||||||
public static GetSubClustersInfoResponse newInstance(
|
public static GetSubClustersInfoResponse newInstance(
|
||||||
List<SubClusterInfo> subClusters) {
|
Collection<SubClusterInfo> subClusters) {
|
||||||
GetSubClustersInfoResponse subClusterInfos =
|
GetSubClustersInfoResponse subClusterInfos =
|
||||||
Records.newRecord(GetSubClustersInfoResponse.class);
|
Records.newRecord(GetSubClustersInfoResponse.class);
|
||||||
subClusterInfos.setSubClusters(subClusters);
|
subClusterInfos.setSubClusters(subClusters);
|
||||||
|
@ -61,6 +62,5 @@ public abstract class GetSubClustersInfoResponse {
|
||||||
*/
|
*/
|
||||||
@Private
|
@Private
|
||||||
@Unstable
|
@Unstable
|
||||||
public abstract void setSubClusters(List<SubClusterInfo> subClusters);
|
public abstract void setSubClusters(Collection<SubClusterInfo> subClusters);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,8 +19,10 @@
|
||||||
package org.apache.hadoop.yarn.server.federation.store.records.impl.pb;
|
package org.apache.hadoop.yarn.server.federation.store.records.impl.pb;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import org.apache.hadoop.classification.InterfaceAudience.Private;
|
import org.apache.hadoop.classification.InterfaceAudience.Private;
|
||||||
import org.apache.hadoop.classification.InterfaceStability.Unstable;
|
import org.apache.hadoop.classification.InterfaceStability.Unstable;
|
||||||
|
@ -93,12 +95,12 @@ public class GetSubClustersInfoResponsePBImpl
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void setSubClusters(List<SubClusterInfo> subClusters) {
|
public void setSubClusters(Collection<SubClusterInfo> subClusters) {
|
||||||
if (subClusters == null) {
|
if (subClusters == null) {
|
||||||
builder.clearSubClusterInfos();
|
builder.clearSubClusterInfos();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this.subClusterInfos = subClusters;
|
this.subClusterInfos = subClusters.stream().collect(Collectors.toList());
|
||||||
}
|
}
|
||||||
|
|
||||||
private void initSubClustersInfoList() {
|
private void initSubClustersInfoList() {
|
||||||
|
|
|
@ -19,7 +19,6 @@
|
||||||
package org.apache.hadoop.yarn.server.federation.policies;
|
package org.apache.hadoop.yarn.server.federation.policies;
|
||||||
|
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
import static org.mockito.Mockito.when;
|
|
||||||
|
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
@ -28,20 +27,26 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Random;
|
import java.util.Random;
|
||||||
|
|
||||||
|
import org.apache.hadoop.yarn.api.protocolrecords.ReservationSubmissionRequest;
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
||||||
import org.apache.hadoop.yarn.api.records.ResourceRequest;
|
import org.apache.hadoop.yarn.api.records.ResourceRequest;
|
||||||
|
import org.apache.hadoop.yarn.conf.YarnConfiguration;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.amrmproxy.FederationAMRMProxyPolicy;
|
import org.apache.hadoop.yarn.server.federation.policies.amrmproxy.FederationAMRMProxyPolicy;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyInitializationException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.router.FederationRouterPolicy;
|
import org.apache.hadoop.yarn.server.federation.policies.router.FederationRouterPolicy;
|
||||||
|
import org.apache.hadoop.yarn.server.federation.store.FederationStateStore;
|
||||||
|
import org.apache.hadoop.yarn.server.federation.store.impl.MemoryFederationStateStore;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterPolicyConfiguration;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterPolicyConfiguration;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterState;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterState;
|
||||||
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterRegisterRequest;
|
||||||
import org.apache.hadoop.yarn.server.federation.utils.FederationPoliciesTestUtil;
|
import org.apache.hadoop.yarn.server.federation.utils.FederationPoliciesTestUtil;
|
||||||
|
import org.apache.hadoop.yarn.server.federation.utils.FederationStateStoreFacade;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -58,6 +63,9 @@ public abstract class BaseFederationPoliciesTest {
|
||||||
private Random rand = new Random();
|
private Random rand = new Random();
|
||||||
private SubClusterId homeSubCluster;
|
private SubClusterId homeSubCluster;
|
||||||
|
|
||||||
|
private ReservationSubmissionRequest reservationSubmissionRequest =
|
||||||
|
mock(ReservationSubmissionRequest.class);
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testReinitilialize() throws YarnException {
|
public void testReinitilialize() throws YarnException {
|
||||||
FederationPolicyInitializationContext fpc =
|
FederationPolicyInitializationContext fpc =
|
||||||
|
@ -177,11 +185,60 @@ public abstract class BaseFederationPoliciesTest {
|
||||||
public void setMockActiveSubclusters(int numSubclusters) {
|
public void setMockActiveSubclusters(int numSubclusters) {
|
||||||
for (int i = 1; i <= numSubclusters; i++) {
|
for (int i = 1; i <= numSubclusters; i++) {
|
||||||
SubClusterIdInfo sc = new SubClusterIdInfo("sc" + i);
|
SubClusterIdInfo sc = new SubClusterIdInfo("sc" + i);
|
||||||
SubClusterInfo sci = mock(SubClusterInfo.class);
|
SubClusterInfo sci = SubClusterInfo.newInstance(
|
||||||
when(sci.getState()).thenReturn(SubClusterState.SC_RUNNING);
|
sc.toId(), "dns1:80", "dns1:81", "dns1:82", "dns1:83", SubClusterState.SC_RUNNING,
|
||||||
when(sci.getSubClusterId()).thenReturn(sc.toId());
|
System.currentTimeMillis(), "something");
|
||||||
getActiveSubclusters().put(sc.toId(), sci);
|
getActiveSubclusters().put(sc.toId(), sci);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String generateClusterMetricsInfo(int id) {
|
||||||
|
long mem = 1024 * getRand().nextInt(277 * 100 - 1);
|
||||||
|
// plant a best cluster
|
||||||
|
if (id == 5) {
|
||||||
|
mem = 1024 * 277 * 100;
|
||||||
|
}
|
||||||
|
String clusterMetrics =
|
||||||
|
"{\"clusterMetrics\":{\"appsSubmitted\":65, \"appsCompleted\":64,\"appsPending\":0,"
|
||||||
|
+ "\"appsRunning\":0, \"appsFailed\":0, \"appsKilled\":1,\"reservedMB\":0,\"availableMB\":"
|
||||||
|
+ mem + ", \"allocatedMB\":0,\"reservedVirtualCores\":0, \"availableVirtualCores\":2216,"
|
||||||
|
+ "\"allocatedVirtualCores\":0, \"containersAllocated\":0,\"containersReserved\":0,"
|
||||||
|
+ "\"containersPending\":0,\"totalMB\":28364800, \"totalVirtualCores\":2216,"
|
||||||
|
+ "\"totalNodes\":278, \"lostNodes\":1,\"unhealthyNodes\":0,\"decommissionedNodes\":0, "
|
||||||
|
+ "\"rebootedNodes\":0, \"activeNodes\":277}}";
|
||||||
|
return clusterMetrics;
|
||||||
|
}
|
||||||
|
|
||||||
|
public FederationStateStoreFacade getMemoryFacade() throws YarnException {
|
||||||
|
|
||||||
|
// setting up a store and its facade (with caching off)
|
||||||
|
FederationStateStoreFacade fedFacade = FederationStateStoreFacade.getInstance();
|
||||||
|
YarnConfiguration conf = new YarnConfiguration();
|
||||||
|
conf.setInt(YarnConfiguration.FEDERATION_CACHE_TIME_TO_LIVE_SECS, 0);
|
||||||
|
FederationStateStore store = new MemoryFederationStateStore();
|
||||||
|
store.init(conf);
|
||||||
|
fedFacade.reinitialize(store, conf);
|
||||||
|
|
||||||
|
for (SubClusterInfo sinfo : getActiveSubclusters().values()) {
|
||||||
|
store.registerSubCluster(SubClusterRegisterRequest.newInstance(sinfo));
|
||||||
|
}
|
||||||
|
|
||||||
|
return fedFacade;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ReservationSubmissionRequest getReservationSubmissionRequest() {
|
||||||
|
return reservationSubmissionRequest;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setReservationSubmissionRequest(
|
||||||
|
ReservationSubmissionRequest reservationSubmissionRequest) {
|
||||||
|
this.reservationSubmissionRequest = reservationSubmissionRequest;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setupContext() throws YarnException {
|
||||||
|
FederationPolicyInitializationContext context =
|
||||||
|
FederationPoliciesTestUtil.initializePolicyContext2(getPolicy(),
|
||||||
|
getPolicyInfo(), getActiveSubclusters(), getMemoryFacade());
|
||||||
|
this.setFederationPolicyContext(context);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,6 +23,7 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Random;
|
import java.util.Random;
|
||||||
|
|
||||||
|
import org.apache.hadoop.test.LambdaTestUtils;
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.BaseFederationPoliciesTest;
|
import org.apache.hadoop.yarn.server.federation.policies.BaseFederationPoliciesTest;
|
||||||
|
@ -115,4 +116,14 @@ public abstract class BaseRouterPoliciesTest
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testNullReservationContext() throws Exception {
|
||||||
|
FederationRouterPolicy policy = ((FederationRouterPolicy) getPolicy());
|
||||||
|
|
||||||
|
LambdaTestUtils.intercept(FederationPolicyException.class,
|
||||||
|
"The ReservationSubmissionRequest cannot be null.",
|
||||||
|
() -> policy.getReservationHomeSubcluster(null));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,7 +27,6 @@ import java.util.concurrent.atomic.AtomicLong;
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
import org.apache.hadoop.yarn.server.federation.utils.FederationPoliciesTestUtil;
|
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
@ -50,8 +49,7 @@ public class TestHashBasedRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
setMockActiveSubclusters(numSubclusters);
|
setMockActiveSubclusters(numSubclusters);
|
||||||
|
|
||||||
// initialize policy with context
|
// initialize policy with context
|
||||||
FederationPoliciesTestUtil.initializePolicyContext(getPolicy(),
|
setupContext();
|
||||||
getPolicyInfo(), getActiveSubclusters());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
|
@ -17,12 +17,13 @@
|
||||||
|
|
||||||
package org.apache.hadoop.yarn.server.federation.policies.router;
|
package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
|
|
||||||
import static org.junit.Assert.fail;
|
|
||||||
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.apache.hadoop.test.LambdaTestUtils;
|
||||||
|
import org.apache.hadoop.util.Time;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
|
import org.apache.hadoop.yarn.server.federation.policies.ConfigurableFederationPolicy;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
||||||
|
@ -46,12 +47,14 @@ public class TestLoadBasedRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
Map<SubClusterIdInfo, Float> routerWeights = new HashMap<>();
|
Map<SubClusterIdInfo, Float> routerWeights = new HashMap<>();
|
||||||
Map<SubClusterIdInfo, Float> amrmWeights = new HashMap<>();
|
Map<SubClusterIdInfo, Float> amrmWeights = new HashMap<>();
|
||||||
|
|
||||||
|
long now = Time.now();
|
||||||
|
|
||||||
// simulate 20 active subclusters
|
// simulate 20 active subclusters
|
||||||
for (int i = 0; i < 20; i++) {
|
for (int i = 0; i < 20; i++) {
|
||||||
SubClusterIdInfo sc = new SubClusterIdInfo(String.format("sc%02d", i));
|
SubClusterIdInfo sc = new SubClusterIdInfo(String.format("sc%02d", i));
|
||||||
SubClusterInfo federationSubClusterInfo =
|
SubClusterInfo federationSubClusterInfo = SubClusterInfo.newInstance(
|
||||||
SubClusterInfo.newInstance(sc.toId(), null, null, null, null, -1,
|
sc.toId(), "dns1:80", "dns1:81", "dns1:82", "dns1:83",
|
||||||
SubClusterState.SC_RUNNING, -1, generateClusterMetricsInfo(i));
|
now - 1000, SubClusterState.SC_RUNNING, now - 2000, generateClusterMetricsInfo(i));
|
||||||
getActiveSubclusters().put(sc.toId(), federationSubClusterInfo);
|
getActiveSubclusters().put(sc.toId(), federationSubClusterInfo);
|
||||||
float weight = getRand().nextInt(2);
|
float weight = getRand().nextInt(2);
|
||||||
if (i == 5) {
|
if (i == 5) {
|
||||||
|
@ -67,12 +70,11 @@ public class TestLoadBasedRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
getPolicyInfo().setRouterPolicyWeights(routerWeights);
|
getPolicyInfo().setRouterPolicyWeights(routerWeights);
|
||||||
getPolicyInfo().setAMRMPolicyWeights(amrmWeights);
|
getPolicyInfo().setAMRMPolicyWeights(amrmWeights);
|
||||||
|
|
||||||
FederationPoliciesTestUtil.initializePolicyContext(getPolicy(),
|
// initialize policy with context
|
||||||
getPolicyInfo(), getActiveSubclusters());
|
setupContext();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private String generateClusterMetricsInfo(int id) {
|
public String generateClusterMetricsInfo(int id) {
|
||||||
|
|
||||||
long mem = 1024 * getRand().nextInt(277 * 100 - 1);
|
long mem = 1024 * getRand().nextInt(277 * 100 - 1);
|
||||||
// plant a best cluster
|
// plant a best cluster
|
||||||
|
@ -106,7 +108,7 @@ public class TestLoadBasedRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testIfNoSubclustersWithWeightOne() {
|
public void testIfNoSubclustersWithWeightOne() throws Exception {
|
||||||
setPolicy(new LoadBasedRouterPolicy());
|
setPolicy(new LoadBasedRouterPolicy());
|
||||||
setPolicyInfo(new WeightedPolicyInfo());
|
setPolicyInfo(new WeightedPolicyInfo());
|
||||||
Map<SubClusterIdInfo, Float> routerWeights = new HashMap<>();
|
Map<SubClusterIdInfo, Float> routerWeights = new HashMap<>();
|
||||||
|
@ -123,15 +125,13 @@ public class TestLoadBasedRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
getPolicyInfo().setRouterPolicyWeights(routerWeights);
|
getPolicyInfo().setRouterPolicyWeights(routerWeights);
|
||||||
getPolicyInfo().setAMRMPolicyWeights(amrmWeights);
|
getPolicyInfo().setAMRMPolicyWeights(amrmWeights);
|
||||||
|
|
||||||
try {
|
|
||||||
FederationPoliciesTestUtil.initializePolicyContext(getPolicy(),
|
ConfigurableFederationPolicy policy = getPolicy();
|
||||||
|
FederationPoliciesTestUtil.initializePolicyContext(policy,
|
||||||
getPolicyInfo(), getActiveSubclusters());
|
getPolicyInfo(), getActiveSubclusters());
|
||||||
((FederationRouterPolicy) getPolicy())
|
|
||||||
.getHomeSubcluster(getApplicationSubmissionContext(), null);
|
LambdaTestUtils.intercept(YarnException.class, "Zero Active Subcluster with weight 1.",
|
||||||
fail();
|
() -> ((FederationRouterPolicy) policy).
|
||||||
} catch (YarnException ex) {
|
getHomeSubcluster(getApplicationSubmissionContext(), null));
|
||||||
Assert.assertTrue(
|
|
||||||
ex.getMessage().contains("Zero Active Subcluster with weight 1"));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -73,6 +73,7 @@ public class TestLocalityRouterPolicy extends TestWeightedRandomRouterPolicy {
|
||||||
|
|
||||||
configureWeights(4);
|
configureWeights(4);
|
||||||
|
|
||||||
|
// initialize policy with context
|
||||||
initializePolicy(new YarnConfiguration());
|
initializePolicy(new YarnConfiguration());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -86,9 +87,7 @@ public class TestLocalityRouterPolicy extends TestWeightedRandomRouterPolicy {
|
||||||
.newInstance("queue1", getPolicy().getClass().getCanonicalName(),
|
.newInstance("queue1", getPolicy().getClass().getCanonicalName(),
|
||||||
buf));
|
buf));
|
||||||
getFederationPolicyContext().setHomeSubcluster(getHomeSubCluster());
|
getFederationPolicyContext().setHomeSubcluster(getHomeSubCluster());
|
||||||
FederationPoliciesTestUtil
|
setupContext();
|
||||||
.initializePolicyContext(getFederationPolicyContext(), getPolicy(),
|
|
||||||
getPolicyInfo(), getActiveSubclusters(), conf);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -23,6 +23,7 @@ import static org.mockito.Mockito.when;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.apache.hadoop.util.Time;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
|
@ -54,10 +55,11 @@ public class TestPriorityRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
|
|
||||||
// with 5% omit a subcluster
|
// with 5% omit a subcluster
|
||||||
if (getRand().nextFloat() < 0.95f || i == 5) {
|
if (getRand().nextFloat() < 0.95f || i == 5) {
|
||||||
SubClusterInfo sci = mock(SubClusterInfo.class);
|
long now = Time.now();
|
||||||
when(sci.getState()).thenReturn(SubClusterState.SC_RUNNING);
|
SubClusterInfo federationSubClusterInfo = SubClusterInfo.newInstance(
|
||||||
when(sci.getSubClusterId()).thenReturn(sc.toId());
|
sc.toId(), "dns1:80", "dns1:81", "dns1:82", "dns1:83",
|
||||||
getActiveSubclusters().put(sc.toId(), sci);
|
now - 1000, SubClusterState.SC_RUNNING, now - 2000, generateClusterMetricsInfo(i));
|
||||||
|
getActiveSubclusters().put(sc.toId(), federationSubClusterInfo);
|
||||||
}
|
}
|
||||||
float weight = getRand().nextFloat();
|
float weight = getRand().nextFloat();
|
||||||
if (i == 5) {
|
if (i == 5) {
|
||||||
|
@ -105,7 +107,7 @@ public class TestPriorityRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
getPolicyInfo(), getActiveSubclusters());
|
getPolicyInfo(), getActiveSubclusters());
|
||||||
|
|
||||||
intercept(FederationPolicyException.class,
|
intercept(FederationPolicyException.class,
|
||||||
"No Active Subcluster with weight vector greater than zero",
|
"No Active Subcluster with weight vector greater than zero.",
|
||||||
() -> ((FederationRouterPolicy) getPolicy())
|
() -> ((FederationRouterPolicy) getPolicy())
|
||||||
.getHomeSubcluster(getApplicationSubmissionContext(), null));
|
.getHomeSubcluster(getApplicationSubmissionContext(), null));
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,7 +20,6 @@ package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
import org.apache.hadoop.yarn.server.federation.policies.exceptions.FederationPolicyException;
|
||||||
import org.apache.hadoop.yarn.server.federation.utils.FederationPoliciesTestUtil;
|
|
||||||
import org.apache.hadoop.yarn.util.resource.Resources;
|
import org.apache.hadoop.yarn.util.resource.Resources;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
@ -39,8 +38,7 @@ public class TestRejectRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
setMockActiveSubclusters(2);
|
setMockActiveSubclusters(2);
|
||||||
|
|
||||||
// initialize policy with context
|
// initialize policy with context
|
||||||
FederationPoliciesTestUtil.initializePolicyContext(getPolicy(),
|
setupContext();
|
||||||
getPolicyInfo(), getActiveSubclusters());
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -59,5 +57,4 @@ public class TestRejectRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
false, false, 0, Resources.none(), null, false, null, null);
|
false, false, 0, Resources.none(), null, false, null, null);
|
||||||
localPolicy.getHomeSubcluster(applicationSubmissionContext, null);
|
localPolicy.getHomeSubcluster(applicationSubmissionContext, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,15 +18,14 @@
|
||||||
package org.apache.hadoop.yarn.server.federation.policies.router;
|
package org.apache.hadoop.yarn.server.federation.policies.router;
|
||||||
|
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
import static org.mockito.Mockito.when;
|
|
||||||
|
|
||||||
|
import org.apache.hadoop.util.Time;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
import org.apache.hadoop.yarn.server.federation.policies.dao.WeightedPolicyInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterState;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterState;
|
||||||
import org.apache.hadoop.yarn.server.federation.utils.FederationPoliciesTestUtil;
|
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
@ -44,14 +43,14 @@ public class TestUniformRandomRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
setPolicyInfo(mock(WeightedPolicyInfo.class));
|
setPolicyInfo(mock(WeightedPolicyInfo.class));
|
||||||
for (int i = 1; i <= 2; i++) {
|
for (int i = 1; i <= 2; i++) {
|
||||||
SubClusterIdInfo sc = new SubClusterIdInfo("sc" + i);
|
SubClusterIdInfo sc = new SubClusterIdInfo("sc" + i);
|
||||||
SubClusterInfo sci = mock(SubClusterInfo.class);
|
long now = Time.now();
|
||||||
when(sci.getState()).thenReturn(SubClusterState.SC_RUNNING);
|
SubClusterInfo federationSubClusterInfo = SubClusterInfo.newInstance(
|
||||||
when(sci.getSubClusterId()).thenReturn(sc.toId());
|
sc.toId(), "dns1:80", "dns1:81", "dns1:82", "dns1:83",
|
||||||
getActiveSubclusters().put(sc.toId(), sci);
|
now - 1000, SubClusterState.SC_RUNNING, now - 2000, generateClusterMetricsInfo(i));
|
||||||
|
getActiveSubclusters().put(sc.toId(), federationSubClusterInfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
FederationPoliciesTestUtil.initializePolicyContext(getPolicy(),
|
setupContext();
|
||||||
mock(WeightedPolicyInfo.class), getActiveSubclusters());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
|
@ -24,6 +24,7 @@ import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.atomic.AtomicLong;
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
|
|
||||||
|
import org.apache.hadoop.util.Time;
|
||||||
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
import org.apache.hadoop.yarn.api.records.ApplicationSubmissionContext;
|
||||||
import org.apache.hadoop.yarn.exceptions.YarnException;
|
import org.apache.hadoop.yarn.exceptions.YarnException;
|
||||||
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyUtils;
|
import org.apache.hadoop.yarn.server.federation.policies.FederationPolicyUtils;
|
||||||
|
@ -32,7 +33,6 @@ import org.apache.hadoop.yarn.server.federation.store.records.SubClusterId;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterIdInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterInfo;
|
||||||
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterState;
|
import org.apache.hadoop.yarn.server.federation.store.records.SubClusterState;
|
||||||
import org.apache.hadoop.yarn.server.federation.utils.FederationPoliciesTestUtil;
|
|
||||||
import org.junit.Assert;
|
import org.junit.Assert;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
@ -51,8 +51,7 @@ public class TestWeightedRandomRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
|
|
||||||
configureWeights(20);
|
configureWeights(20);
|
||||||
|
|
||||||
FederationPoliciesTestUtil.initializePolicyContext(getPolicy(),
|
setupContext();
|
||||||
getPolicyInfo(), getActiveSubclusters());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void configureWeights(float numSubClusters) {
|
public void configureWeights(float numSubClusters) {
|
||||||
|
@ -68,10 +67,11 @@ public class TestWeightedRandomRouterPolicy extends BaseRouterPoliciesTest {
|
||||||
SubClusterIdInfo sc = new SubClusterIdInfo("sc" + i);
|
SubClusterIdInfo sc = new SubClusterIdInfo("sc" + i);
|
||||||
// with 5% omit a subcluster
|
// with 5% omit a subcluster
|
||||||
if (getRand().nextFloat() < 0.95f) {
|
if (getRand().nextFloat() < 0.95f) {
|
||||||
SubClusterInfo sci = mock(SubClusterInfo.class);
|
long now = Time.now();
|
||||||
when(sci.getState()).thenReturn(SubClusterState.SC_RUNNING);
|
SubClusterInfo federationSubClusterInfo = SubClusterInfo.newInstance(
|
||||||
when(sci.getSubClusterId()).thenReturn(sc.toId());
|
sc.toId(), "dns1:80", "dns1:81", "dns1:82", "dns1:83",
|
||||||
getActiveSubclusters().put(sc.toId(), sci);
|
now - 1000, SubClusterState.SC_RUNNING, now - 2000, generateClusterMetricsInfo(i));
|
||||||
|
getActiveSubclusters().put(sc.toId(), federationSubClusterInfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
// 80% of the weight is evenly spread, 20% is randomly generated
|
// 80% of the weight is evenly spread, 20% is randomly generated
|
||||||
|
|
|
@ -159,6 +159,51 @@ public final class FederationPoliciesTestUtil {
|
||||||
new Configuration());
|
new Configuration());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static FederationPolicyInitializationContext initializePolicyContext2(
|
||||||
|
ConfigurableFederationPolicy policy, WeightedPolicyInfo policyInfo,
|
||||||
|
Map<SubClusterId, SubClusterInfo> activeSubClusters,
|
||||||
|
FederationStateStoreFacade facade) throws YarnException {
|
||||||
|
FederationPolicyInitializationContext context =
|
||||||
|
new FederationPolicyInitializationContext(null, initResolver(), facade,
|
||||||
|
SubClusterId.newInstance("homesubcluster"));
|
||||||
|
return initializePolicyContext2(context, policy, policyInfo, activeSubClusters);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static FederationPolicyInitializationContext initializePolicyContext2(
|
||||||
|
ConfigurableFederationPolicy policy, WeightedPolicyInfo policyInfo,
|
||||||
|
Map<SubClusterId, SubClusterInfo> activeSubClusters)
|
||||||
|
throws YarnException {
|
||||||
|
return initializePolicyContext2(policy, policyInfo, activeSubClusters, initFacade());
|
||||||
|
}
|
||||||
|
|
||||||
|
public static FederationPolicyInitializationContext initializePolicyContext2(
|
||||||
|
FederationPolicyInitializationContext fpc,
|
||||||
|
ConfigurableFederationPolicy policy, WeightedPolicyInfo policyInfo,
|
||||||
|
Map<SubClusterId, SubClusterInfo> activeSubClusters)
|
||||||
|
throws YarnException {
|
||||||
|
ByteBuffer buf = policyInfo.toByteBuffer();
|
||||||
|
fpc.setSubClusterPolicyConfiguration(SubClusterPolicyConfiguration
|
||||||
|
.newInstance("queue1", policy.getClass().getCanonicalName(), buf));
|
||||||
|
|
||||||
|
if (fpc.getFederationStateStoreFacade() == null) {
|
||||||
|
FederationStateStoreFacade facade = FederationStateStoreFacade.getInstance();
|
||||||
|
FederationStateStore fss = mock(FederationStateStore.class);
|
||||||
|
|
||||||
|
if (activeSubClusters == null) {
|
||||||
|
activeSubClusters = new HashMap<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
GetSubClustersInfoResponse response = GetSubClustersInfoResponse.newInstance(
|
||||||
|
activeSubClusters.values());
|
||||||
|
|
||||||
|
when(fss.getSubClusters(any())).thenReturn(response);
|
||||||
|
facade.reinitialize(fss, new Configuration());
|
||||||
|
fpc.setFederationStateStoreFacade(facade);
|
||||||
|
}
|
||||||
|
policy.reinitialize(fpc);
|
||||||
|
return fpc;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Initialize a {@link SubClusterResolver}.
|
* Initialize a {@link SubClusterResolver}.
|
||||||
*
|
*
|
||||||
|
|
|
@ -34,6 +34,7 @@ import java.util.concurrent.CompletionService;
|
||||||
import java.util.concurrent.ExecutorCompletionService;
|
import java.util.concurrent.ExecutorCompletionService;
|
||||||
import java.util.concurrent.ExecutorService;
|
import java.util.concurrent.ExecutorService;
|
||||||
import java.util.concurrent.Future;
|
import java.util.concurrent.Future;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import javax.servlet.http.HttpServletRequest;
|
import javax.servlet.http.HttpServletRequest;
|
||||||
import javax.servlet.http.HttpServletRequestWrapper;
|
import javax.servlet.http.HttpServletRequestWrapper;
|
||||||
|
@ -179,19 +180,13 @@ public class FederationInterceptorREST extends AbstractRESTRequestInterceptor {
|
||||||
RouterServerUtil.logAndThrowException(
|
RouterServerUtil.logAndThrowException(
|
||||||
FederationPolicyUtils.NO_ACTIVE_SUBCLUSTER_AVAILABLE, null);
|
FederationPolicyUtils.NO_ACTIVE_SUBCLUSTER_AVAILABLE, null);
|
||||||
}
|
}
|
||||||
List<SubClusterId> list = new ArrayList<>(activeSubclusters.keySet());
|
Collection<SubClusterId> keySet = activeSubclusters.keySet();
|
||||||
|
FederationPolicyUtils.validateSubClusterAvailability(keySet, blackListSubClusters);
|
||||||
FederationPolicyUtils.validateSubClusterAvailability(
|
|
||||||
list, blackListSubClusters);
|
|
||||||
|
|
||||||
if (blackListSubClusters != null) {
|
if (blackListSubClusters != null) {
|
||||||
|
keySet.removeAll(blackListSubClusters);
|
||||||
// Remove from the active SubClusters from StateStore the blacklisted ones
|
|
||||||
for (SubClusterId scId : blackListSubClusters) {
|
|
||||||
list.remove(scId);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
List<SubClusterId> list = keySet.stream().collect(Collectors.toList());
|
||||||
return list.get(rand.nextInt(list.size()));
|
return list.get(rand.nextInt(list.size()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue