All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.hadoop.yarn.api.resource.PlacementConstraint Maven / Gradle / Ivy

The newest version!
/**
 * 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.hadoop.yarn.api.resource;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.Iterator;

import org.apache.hadoop.classification.InterfaceAudience.Private;
import org.apache.hadoop.classification.InterfaceAudience.Public;
import org.apache.hadoop.classification.InterfaceStability.Unstable;
import org.apache.hadoop.yarn.api.records.NodeAttributeOpCode;

/**
 * {@code PlacementConstraint} represents a placement constraint for a resource
 * allocation.
 */
@Public
@Unstable
public class PlacementConstraint {

  /**
   * The constraint expression tree.
   */
  private AbstractConstraint constraintExpr;

  public PlacementConstraint(AbstractConstraint constraintExpr) {
    this.constraintExpr = constraintExpr;
  }

  @Override
  public String toString() {
    return this.constraintExpr.toString();
  }

  /**
   * Get the constraint expression of the placement constraint.
   *
   * @return the constraint expression
   */
  public AbstractConstraint getConstraintExpr() {
    return constraintExpr;
  }

  @Override
  public boolean equals(Object o) {
    if (this == o) {
      return true;
    }
    if (!(o instanceof PlacementConstraint)) {
      return false;
    }

    PlacementConstraint that = (PlacementConstraint) o;

    return getConstraintExpr() != null ? getConstraintExpr().equals(that
        .getConstraintExpr()) : that.getConstraintExpr() == null;
  }

  @Override
  public int hashCode() {
    return getConstraintExpr() != null ? getConstraintExpr().hashCode() : 0;
  }

  /**
   * Interface used to enable the elements of the constraint tree to be visited.
   */
  @Private
  public interface Visitable {
    /**
     * Visitor pattern.
     *
     * @param visitor visitor to be used
     * @param  defines the type that the visitor will use and the return type
     *          of the accept.
     * @return the result of visiting a given object.
     */
     T accept(Visitor visitor);

  }

  /**
   * Visitor API for a constraint tree.
   *
   * @param  determines the return type of the visit methods.
   */
  @Private
  public interface Visitor {
    T visit(SingleConstraint constraint);

    T visit(TargetExpression target);

    T visit(TargetConstraint constraint);

    T visit(CardinalityConstraint constraint);

    T visit(And constraint);

    T visit(Or constraint);

    T visit(DelayedOr constraint);

    T visit(TimedPlacementConstraint constraint);
  }

  /**
   * Abstract class that acts as the superclass of all placement constraint
   * classes.
   */
  public abstract static class AbstractConstraint implements Visitable {
    public PlacementConstraint build() {
      return new PlacementConstraint(this);
    }

    @Override
    public String toString() {
      return super.toString();
    }
  }

  static final String NODE_SCOPE = "node";
  static final String RACK_SCOPE = "rack";

  /**
   * Consider a set of nodes N that belongs to the scope specified in the
   * constraint. If the target expressions are satisfied at least minCardinality
   * times and at most max-cardinality times in the node set N, then the
   * constraint is satisfied.
   *
   * For example, a constraint of the form {@code {RACK, 2, 10,
   * allocationTag("zk")}}, requires an allocation to be placed within a rack
   * that has at least 2 and at most 10 other allocations with tag "zk".
   */
  public static class SingleConstraint extends AbstractConstraint {
    private String scope;
    private int minCardinality;
    private int maxCardinality;
    private Set targetExpressions;
    private NodeAttributeOpCode attributeOpCode;

    public SingleConstraint(String scope, int minCardinality,
        int maxCardinality, NodeAttributeOpCode opCode,
        Set targetExpressions) {
      this.scope = scope;
      this.minCardinality = minCardinality;
      this.maxCardinality = maxCardinality;
      this.targetExpressions = targetExpressions;
      this.attributeOpCode = opCode;
    }

    public SingleConstraint(String scope, int minCardinality,
        int maxCardinality, Set targetExpressions) {
      this(scope, minCardinality, maxCardinality, NodeAttributeOpCode.NO_OP,
          targetExpressions);
    }

    public SingleConstraint(String scope, int minC, int maxC,
        TargetExpression... targetExpressions) {
      this(scope, minC, maxC, new HashSet<>(Arrays.asList(targetExpressions)));
    }

    public SingleConstraint(String scope, int minC, int maxC,
        NodeAttributeOpCode opCode,
        TargetExpression... targetExpressions) {
      this(scope, minC, maxC, opCode,
          new HashSet<>(Arrays.asList(targetExpressions)));
    }

    /**
     * Get the scope of the constraint.
     *
     * @return the scope of the constraint
     */
    public String getScope() {
      return scope;
    }

    /**
     * Get the minimum cardinality of the constraint.
     *
     * @return the minimum cardinality of the constraint
     */
    public int getMinCardinality() {
      return minCardinality;
    }

    /**
     * Get the maximum cardinality of the constraint.
     *
     * @return the maximum cardinality of the constraint
     */
    public int getMaxCardinality() {
      return maxCardinality;
    }

    /**
     * Get the target expressions of the constraint.
     *
     * @return the set of target expressions
     */
    public Set getTargetExpressions() {
      return targetExpressions;
    }

    /**
     * Get the NodeAttributeOpCode of the constraint.
     *
     * @return nodeAttribute Op Code
     */
    public NodeAttributeOpCode getNodeAttributeOpCode() {
      return attributeOpCode;
    }

    @Override
    public boolean equals(Object o) {
      if (this == o) {
        return true;
      }
      if (!(o instanceof SingleConstraint)) {
        return false;
      }

      SingleConstraint that = (SingleConstraint) o;

      if (getMinCardinality() != that.getMinCardinality()) {
        return false;
      }
      if (getMaxCardinality() != that.getMaxCardinality()) {
        return false;
      }
      if (!getScope().equals(that.getScope())) {
        return false;
      }
      if (getNodeAttributeOpCode() != null && !getNodeAttributeOpCode()
          .equals(that.getNodeAttributeOpCode())) {
        return false;
      }
      return getTargetExpressions().equals(that.getTargetExpressions());
    }

    @Override
    public int hashCode() {
      int result = getScope().hashCode();
      result = 31 * result + getMinCardinality();
      result = 31 * result + getMaxCardinality();
      result = 31 * result + getNodeAttributeOpCode().hashCode();
      result = 31 * result + getTargetExpressions().hashCode();
      return result;
    }

    @Override
    public String toString() {
      int max = getMaxCardinality();
      int min = getMinCardinality();
      List targetExprList = getTargetExpressions().stream()
          .map(TargetExpression::toString).collect(Collectors.toList());
      List targetConstraints = new ArrayList<>();
      for (String targetExpr : targetExprList) {
        if (min == 0 && max == 0) {
          // anti-affinity
          targetConstraints.add(new StringBuilder()
              .append("notin").append(",")
              .append(getScope()).append(",")
              .append(targetExpr)
              .toString());
        } else if (min == 1 && max == Integer.MAX_VALUE) {
          // affinity
          targetConstraints.add(new StringBuilder()
              .append("in").append(",")
              .append(getScope()).append(",")
              .append(targetExpr)
              .toString());
        } else if (min == -1 && max == -1) {
          // node attribute
          targetConstraints.add(new StringBuilder()
              .append(getScope()).append(",")
              .append(getNodeAttributeOpCode()).append(",")
              .append(targetExpr)
              .toString());
        } else {
          // cardinality
          targetConstraints.add(new StringBuilder()
              .append("cardinality").append(",")
              .append(getScope()).append(",")
              .append(targetExpr).append(",")
              .append(min).append(",")
              .append(max)
              .toString());
        }
      }
      return String.join(":", targetConstraints);
    }

    @Override
    public  T accept(Visitor visitor) {
      return visitor.visit(this);
    }
  }

  /**
   * Class representing the target expressions that are used in placement
   * constraints. They might refer to expressions on node attributes, allocation
   * tags, or be self-targets (referring to the allocation to which the
   * constraint is attached).
   */
  public static class TargetExpression implements Visitable {
    /**
     * Enum specifying the type of the target expression.
     */
    public enum TargetType {
      NODE_ATTRIBUTE, ALLOCATION_TAG
    }

    private TargetType targetType;
    private String targetKey;
    private Set targetValues;

    public TargetExpression(TargetType targetType, String targetKey,
        Set targetValues) {
      this.targetType = targetType;
      this.targetKey = targetKey;
      this.targetValues = targetValues;
    }

    public TargetExpression(TargetType targetType) {
      this(targetType, null, new HashSet<>());
    }

    public TargetExpression(TargetType targetType, String targetKey,
        String... targetValues) {
      this(targetType, targetKey, new HashSet<>(Arrays.asList(targetValues)));
    }

    /**
     * Get the type of the target expression.
     *
     * @return the type of the target expression
     */
    public TargetType getTargetType() {
      return targetType;
    }

    /**
     * Get the key of the target expression.
     *
     * @return the key of the target expression
     */
    public String getTargetKey() {
      return targetKey;
    }

    /**
     * Get the set of values of the target expression.
     *
     * @return the set of values of the target expression
     */
    public Set getTargetValues() {
      return targetValues;
    }

    @Override
    public int hashCode() {
      int result = targetType != null ? targetType.hashCode() : 0;
      result = 31 * result + (targetKey != null ? targetKey.hashCode() : 0);
      result =
          31 * result + (targetValues != null ? targetValues.hashCode() : 0);
      return result;
    }

    @Override
    public boolean equals(Object o) {
      if (this == o) {
        return true;
      }
      if (o == null) {
        return false;
      }
      if (!(o instanceof TargetExpression)) {
        return false;
      }

      TargetExpression that = (TargetExpression) o;
      if (targetType != that.targetType) {
        return false;
      }
      if (targetKey != null ? !targetKey.equals(that.targetKey)
          : that.targetKey != null) {
        return false;
      }
      return targetValues != null ? targetValues.equals(that.targetValues)
          : that.targetValues == null;
    }

    @Override
    public String toString() {
      StringBuffer sb = new StringBuffer();
      if (TargetType.ALLOCATION_TAG == this.targetType) {
        // following by a comma separated tags
        sb.append(String.join(",", getTargetValues()));
      } else if (TargetType.NODE_ATTRIBUTE == this.targetType) {
        // following by a comma separated key value pairs
        if (this.getTargetValues() != null) {
          String attributeName = this.getTargetKey();
          String attributeValues = String.join(":", this.getTargetValues());
          sb.append(attributeName + "=[" + attributeValues + "]");
        }
      }
      return sb.toString();
    }

    @Override
    public  T accept(Visitor visitor) {
      return visitor.visit(this);
    }
  }

  /**
   * Class that represents a target constraint. Such a constraint requires an
   * allocation to be placed within a scope that satisfies some specified
   * expressions on node attributes and allocation tags.
   *
   * It is a specialized version of the {@link SingleConstraint}, where the
   * minimum and the maximum cardinalities take specific values based on the
   * {@link TargetOperator} used.
   */
  public static class TargetConstraint extends AbstractConstraint {
    /**
     * TargetOperator enum helps to specify type.
     */
    enum TargetOperator {
      IN("in"), NOT_IN("notin");

      private String operator;
      TargetOperator(String op) {
        this.operator = op;
      }

      String getOperator() {
        return this.operator;
      }
    }

    private TargetOperator op;
    private String scope;
    private Set targetExpressions;

    public TargetConstraint(TargetOperator op, String scope,
        Set targetExpressions) {
      this.op = op;
      this.scope = scope;
      this.targetExpressions = targetExpressions;
    }

    /**
     * Get the target operator of the constraint.
     *
     * @return the target operator
     */
    public TargetOperator getOp() {
      return op;
    }

    /**
     * Get the scope of the constraint.
     *
     * @return the scope of the constraint
     */
    public String getScope() {
      return scope;
    }

    /**
     * Get the set of target expressions.
     *
     * @return the set of target expressions
     */
    public Set getTargetExpressions() {
      return targetExpressions;
    }

    @Override
    public boolean equals(Object o) {
      if (this == o) {
        return true;
      }
      if (!(o instanceof TargetConstraint)) {
        return false;
      }

      TargetConstraint that = (TargetConstraint) o;

      if (getOp() != that.getOp()) {
        return false;
      }
      if (!getScope().equals(that.getScope())) {
        return false;
      }
      return getTargetExpressions().equals(that.getTargetExpressions());
    }

    @Override
    public int hashCode() {
      int result = getOp().hashCode();
      result = 31 * result + getScope().hashCode();
      result = 31 * result + getTargetExpressions().hashCode();
      return result;
    }

    @Override
    public String toString() {
      List targetExprs = getTargetExpressions().stream().map(
          targetExpression -> new StringBuilder()
              .append(op.getOperator()).append(",")
              .append(scope).append(",")
              .append(targetExpression.toString())
              .toString()).collect(Collectors.toList());
      return String.join(":", targetExprs);
    }

    @Override
    public  T accept(Visitor visitor) {
      return visitor.visit(this);
    }
  }

  /**
   * Class that represents a cardinality constraint. Such a constraint allows
   * the number of allocations with a specific set of tags and within a given
   * scope to be between some minimum and maximum values.
   *
   * It is a specialized version of the {@link SingleConstraint}, where the
   * target is a set of allocation tags.
   */
  public static class CardinalityConstraint extends AbstractConstraint {
    private String scope;
    private int minCardinality;
    private int maxCardinality;
    private Set allocationTags;

    public CardinalityConstraint(String scope, int minCardinality,
        int maxCardinality, Set allocationTags) {
      this.scope = scope;
      this.minCardinality = minCardinality;
      this.maxCardinality = maxCardinality;
      this.allocationTags = allocationTags;
    }

    /**
     * Get the scope of the constraint.
     *
     * @return the scope of the constraint
     */
    public String getScope() {
      return scope;
    }

    /**
     * Get the minimum cardinality of the constraint.
     *
     * @return the minimum cardinality of the constraint
     */
    public int getMinCardinality() {
      return minCardinality;
    }

    /**
     * Get the maximum cardinality of the constraint.
     *
     * @return the maximum cardinality of the constraint
     */
    public int getMaxCardinality() {
      return maxCardinality;
    }

    /**
     * Get the allocation tags of the constraint.
     *
     * @return the allocation tags of the constraint
     */
    public Set getAllocationTags() {
      return allocationTags;
    }

    @Override
    public  T accept(Visitor visitor) {
      return visitor.visit(this);
    }


    @Override
    public boolean equals(Object o) {
      if (this == o) {
        return true;
      }
      if (o == null || getClass() != o.getClass()) {
        return false;
      }

      CardinalityConstraint that = (CardinalityConstraint) o;

      if (minCardinality != that.minCardinality) {
        return false;
      }
      if (maxCardinality != that.maxCardinality) {
        return false;
      }
      if (scope != null ? !scope.equals(that.scope) : that.scope != null) {
        return false;
      }
      return allocationTags != null ? allocationTags.equals(that.allocationTags)
          : that.allocationTags == null;
    }

    @Override
    public int hashCode() {
      int result = scope != null ? scope.hashCode() : 0;
      result = 31 * result + minCardinality;
      result = 31 * result + maxCardinality;
      result = 31 * result
          + (allocationTags != null ? allocationTags.hashCode() : 0);
      return result;
    }

    @Override
    public String toString() {
      StringBuffer sb = new StringBuffer();
      sb.append("cardinality").append(",").append(getScope()).append(",");
      for (String tag : getAllocationTags()) {
        sb.append(tag).append(",");
      }
      sb.append(minCardinality).append(",").append(maxCardinality);
      return sb.toString();
    }
  }

  /**
   * Class that represents composite constraints, which comprise other
   * constraints, forming a constraint tree.
   *
   * @param  the type of constraints that are used as children of the
   *          specific composite constraint
   */
  public abstract static class CompositeConstraint
      extends AbstractConstraint {

    /**
     * Get the children of this composite constraint.
     *
     * @return the children of the composite constraint
     */
    public abstract List getChildren();

    @Override
    public boolean equals(Object o) {
      if (this == o) {
        return true;
      }
      if (o == null || getClass() != o.getClass()) {
        return false;
      }

      return getChildren() != null ? getChildren().equals(
          ((CompositeConstraint)o).getChildren()) :
          ((CompositeConstraint)o).getChildren() == null;
    }

    @Override
    public int hashCode() {
      return getChildren() != null ? getChildren().hashCode() : 0;
    }
  }

  /**
   * Class that represents a composite constraint that is a conjunction of other
   * constraints.
   */
  public static class And extends CompositeConstraint {
    private List children;

    public And(List children) {
      this.children = children;
    }

    public And(AbstractConstraint... children) {
      this(Arrays.asList(children));
    }

    @Override
    public List getChildren() {
      return children;
    }

    @Override
    public  T accept(Visitor visitor) {
      return visitor.visit(this);
    }

    @Override
    public String toString() {
      StringBuffer sb = new StringBuffer();
      sb.append("and(");
      Iterator it = getChildren().iterator();
      while (it.hasNext()) {
        AbstractConstraint child = it.next();
        sb.append(child.toString());
        if (it.hasNext()) {
          sb.append(":");
        }
      }
      sb.append(")");
      return sb.toString();
    }
  }

  /**
   * Class that represents a composite constraint that is a disjunction of other
   * constraints.
   */
  public static class Or extends CompositeConstraint {
    private List children;

    public Or(List children) {
      this.children = children;
    }

    public Or(AbstractConstraint... children) {
      this(Arrays.asList(children));
    }

    @Override
    public List getChildren() {
      return children;
    }

    @Override
    public  T accept(Visitor visitor) {
      return visitor.visit(this);
    }

    @Override
    public String toString() {
      StringBuffer sb = new StringBuffer();
      sb.append("or(");
      Iterator it = getChildren().iterator();
      while (it.hasNext()) {
        AbstractConstraint child = it.next();
        sb.append(child.toString());
        if (it.hasNext()) {
          sb.append(":");
        }
      }
      sb.append(")");
      return sb.toString();
    }
  }

  /**
   * Class that represents a composite constraint that comprises a list of timed
   * placement constraints (see {@link TimedPlacementConstraint}). The scheduler
   * should try to satisfy first the first timed child constraint within the
   * specified time window. If this is not possible, it should attempt to
   * satisfy the second, and so on.
   */
  public static class DelayedOr
      extends CompositeConstraint {
    private List children = new ArrayList<>();

    public DelayedOr(List children) {
      this.children = children;
    }

    public DelayedOr(TimedPlacementConstraint... children) {
      this(Arrays.asList(children));
    }

    @Override
    public List getChildren() {
      return children;
    }

    @Override
    public  T accept(Visitor visitor) {
      return visitor.visit(this);
    }

    @Override
    public String toString() {
      StringBuffer sb = new StringBuffer();
      sb.append("DelayedOr(");
      Iterator it = getChildren().iterator();
      while (it.hasNext()) {
        TimedPlacementConstraint child = it.next();
        sb.append(child.toString());
        if (it.hasNext()) {
          sb.append(",");
        }
      }
      sb.append(")");
      return sb.toString();
    }
  }

  /**
   * Represents a timed placement constraint that has to be satisfied within a
   * time window.
   */
  public static class TimedPlacementConstraint implements Visitable {
    /**
     * The unit of scheduling delay.
     */
    public enum DelayUnit {
      MILLISECONDS, OPPORTUNITIES
    }

    private AbstractConstraint constraint;
    private long schedulingDelay;
    private DelayUnit delayUnit;

    public TimedPlacementConstraint(AbstractConstraint constraint,
        long schedulingDelay, DelayUnit delayUnit) {
      this.constraint = constraint;
      this.schedulingDelay = schedulingDelay;
      this.delayUnit = delayUnit;
    }

    public TimedPlacementConstraint(AbstractConstraint constraint,
        long schedulingDelay) {
      this(constraint, schedulingDelay, DelayUnit.MILLISECONDS);
    }

    public TimedPlacementConstraint(AbstractConstraint constraint) {
      this(constraint, Long.MAX_VALUE, DelayUnit.MILLISECONDS);
    }

    /**
     * Get the constraint that has to be satisfied within the time window.
     *
     * @return the constraint to be satisfied
     */
    public AbstractConstraint getConstraint() {
      return constraint;
    }

    /**
     * Sets the constraint that has to be satisfied within the time window.
     *
     * @param constraint the constraint to be satisfied
     */
    public void setConstraint(AbstractConstraint constraint) {
      this.constraint = constraint;
    }

    /**
     * Get the scheduling delay value that determines the time window within
     * which the constraint has to be satisfied.
     *
     * @return the value of the scheduling delay
     */
    public long getSchedulingDelay() {
      return schedulingDelay;
    }

    /**
     * The unit of the scheduling delay.
     *
     * @return the unit of the delay
     */
    public DelayUnit getDelayUnit() {
      return delayUnit;
    }

    @Override
    public  T accept(Visitor visitor) {
      return visitor.visit(this);
    }

    @Override
    public boolean equals(Object o) {
      if (this == o) {
        return true;
      }
      if (o == null || getClass() != o.getClass()) {
        return false;
      }

      TimedPlacementConstraint that = (TimedPlacementConstraint) o;

      if (schedulingDelay != that.schedulingDelay) {
        return false;
      }
      if (constraint != null ? !constraint.equals(that.constraint) :
          that.constraint != null) {
        return false;
      }
      return delayUnit == that.delayUnit;
    }

    @Override
    public int hashCode() {
      int result = constraint != null ? constraint.hashCode() : 0;
      result = 31 * result + (int) (schedulingDelay ^ (schedulingDelay >>> 32));
      result = 31 * result + (delayUnit != null ? delayUnit.hashCode() : 0);
      return result;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy