TestNodeAttributesCLI.java

/**
 * 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.client.cli;

import org.apache.hadoop.thirdparty.com.google.common.collect.ImmutableList;
import org.apache.hadoop.thirdparty.com.google.common.collect.ImmutableMap;
import org.apache.hadoop.thirdparty.com.google.common.collect.ImmutableSet;
import org.apache.hadoop.yarn.api.ApplicationClientProtocol;
import org.apache.hadoop.yarn.api.protocolrecords.GetAttributesToNodesRequest;
import org.apache.hadoop.yarn.api.protocolrecords.GetAttributesToNodesResponse;
import org.apache.hadoop.yarn.api.protocolrecords.GetClusterNodeAttributesRequest;
import org.apache.hadoop.yarn.api.protocolrecords.GetClusterNodeAttributesResponse;

import org.apache.hadoop.yarn.api.protocolrecords.GetNodesToAttributesRequest;
import org.apache.hadoop.yarn.api.protocolrecords.GetNodesToAttributesResponse;
import org.apache.hadoop.yarn.api.records.NodeAttributeInfo;
import org.apache.hadoop.yarn.api.records.NodeAttributeKey;
import org.apache.hadoop.yarn.api.records.NodeToAttributeValue;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.PrintStream;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;

import org.apache.hadoop.yarn.api.records.NodeAttribute;
import org.apache.hadoop.yarn.api.records.NodeAttributeType;
import org.apache.hadoop.yarn.exceptions.YarnException;
import org.apache.hadoop.yarn.server.api.ResourceManagerAdministrationProtocol;
import org.apache.hadoop.yarn.server.api.protocolrecords.AttributeMappingOperationType;
import org.apache.hadoop.yarn.server.api.protocolrecords.NodeToAttributes;
import org.apache.hadoop.yarn.server.api.protocolrecords.NodesToAttributesMappingRequest;
import org.apache.hadoop.yarn.server.api.protocolrecords.NodesToAttributesMappingResponse;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.mockito.invocation.InvocationOnMock;
import org.mockito.stubbing.Answer;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import org.apache.hadoop.thirdparty.com.google.common.base.Joiner;

/**
 * Test class for TestNodeAttributesCLI.
 */
public class TestNodeAttributesCLI {
  private static final Logger LOG =
      LoggerFactory.getLogger(TestNodeAttributesCLI.class);
  private ResourceManagerAdministrationProtocol admin;
  private ApplicationClientProtocol client;
  private NodesToAttributesMappingRequest nodeToAttrRequest;
  private NodeAttributesCLI nodeAttributesCLI;
  private ByteArrayOutputStream errOutBytes = new ByteArrayOutputStream();
  private ByteArrayOutputStream sysOutBytes = new ByteArrayOutputStream();
  private String errOutput;
  private String sysOutput;

  @BeforeEach
  public void configure() throws IOException, YarnException {

    admin = mock(ResourceManagerAdministrationProtocol.class);
    client = mock(ApplicationClientProtocol.class);

    when(admin.mapAttributesToNodes(any(NodesToAttributesMappingRequest.class)))
        .thenAnswer(new Answer<NodesToAttributesMappingResponse>() {
          @Override
          public NodesToAttributesMappingResponse answer(
              InvocationOnMock invocation) throws Throwable {
            nodeToAttrRequest =
                (NodesToAttributesMappingRequest) invocation.getArguments()[0];
            return NodesToAttributesMappingResponse.newInstance();
          }
        });

    nodeAttributesCLI = new NodeAttributesCLI() {
      @Override
      protected AdminCommandHandler getAdminCommandHandler() {
        return new AdminCommandHandler() {
          @Override
          protected ResourceManagerAdministrationProtocol createAdminProtocol()
              throws IOException {
            return admin;
          }
        };
      }

      @Override
      protected ClientCommandHandler getClientCommandHandler() {
        ClientCommandHandler handler = new ClientCommandHandler() {
          @Override
          protected ApplicationClientProtocol createApplicationProtocol()
              throws IOException {
            return client;
          }
        };
        handler.setSysOut(new PrintStream(sysOutBytes));
        return handler;
      }
    };
    nodeAttributesCLI.setErrOut(new PrintStream(errOutBytes));
  }

  @Test
  public void testHelp() throws Exception {
    String[] args = new String[] {"-help", "-replace"};
    assertTrue(0 == runTool(args), "It should have succeeded help for replace");
    assertErrorContains("-replace <\"node1:attribute[(type)][=value],attribute1"
        + "[=value],attribute2  node2:attribute2[=value],attribute3\">");
    assertErrorContains("Replace the node to attributes mapping information at"
        + " the ResourceManager with the new mapping. Currently supported"
        + " attribute type. And string is the default type too. Attribute value"
        + " if not specified for string type value will be considered as empty"
        + " string. Replaced node-attributes should not violate the existing"
        + " attribute to attribute type mapping.");

    args = new String[] {"-help", "-remove"};
    assertTrue(0 == runTool(args), "It should have succeeded help for replace");
    assertErrorContains(
        "-remove <\"node1:attribute,attribute1" + " node2:attribute2\">");
    assertErrorContains("Removes the specified node to attributes mapping"
        + " information at the ResourceManager");

    args = new String[] {"-help", "-add"};
    assertTrue(0 == runTool(args), "It should have succeeded help for replace");
    assertErrorContains("-add <\"node1:attribute[(type)][=value],"
        + "attribute1[=value],attribute2  node2:attribute2[=value],"
        + "attribute3\">");
    assertErrorContains("Adds or updates the node to attributes mapping"
        + " information at the ResourceManager. Currently supported attribute"
        + " type is string. And string is the default type too. Attribute value"
        + " if not specified for string type value will be considered as empty"
        + " string. Added or updated node-attributes should not violate the"
        + " existing attribute to attribute type mapping.");

    args = new String[] {"-help", "-failOnUnknownNodes"};
    assertTrue(0 == runTool(args), "It should have succeeded help for replace");
    assertErrorContains("-failOnUnknownNodes");
    assertErrorContains("Can be used optionally along with [add,remove,"
        + "replace] options. When set, command will fail if specified nodes "
        + "are unknown.");

    args = new String[] {"-help", "-list"};
    assertTrue(0 == runTool(args), "It should have succeeded help for replace");
    assertErrorContains("-list");
    assertErrorContains("List all attributes in cluster");

    args = new String[] {"-help", "-nodes"};
    assertTrue(0 == runTool(args), "It should have succeeded help for replace");
    assertErrorContains("-nodes");
    assertErrorContains(
        "Works with [list] to specify node hostnames whose mappings "
            + "are required to be displayed.");

    args = new String[] {"-help", "-attributes"};
    assertTrue(0 == runTool(args), "It should have succeeded help for replace");
    assertErrorContains("-attributes");
    assertErrorContains(
        "Works with [attributestonodes] to specify attributes whose mapping "
            + "are required to be displayed.");

    args = new String[] {"-help", "-attributestonodes"};
    assertTrue(0 == runTool(args), "It should have succeeded help for replace");
    assertErrorContains("-attributestonodes");
    assertErrorContains("Displays mapping of attributes to nodes and attribute "
        + "values grouped by attributes");
  }

  @Test
  public void testReplace() throws Exception {
    // --------------------------------
    // failure scenarios
    // --------------------------------
    // parenthesis not match
    String[] args = new String[] {"-replace", "x("};
    assertTrue(0 != runTool(args), "It should have failed as no node is specified");
    assertFailureMessageContains(NodeAttributesCLI.INVALID_MAPPING_ERR_MSG);

    // parenthesis not match
    args = new String[] {"-replace", "x:(=abc"};
    assertTrue(0 != runTool(args),
        "It should have failed as no closing parenthesis is not specified");
    assertFailureMessageContains(
        "Attribute for node x is not properly configured : (=abc");

    args = new String[] {"-replace", "x:()=abc"};
    assertTrue(0 != runTool(args),
        "It should have failed as no type specified inside parenthesis");
    assertFailureMessageContains(
        "Attribute for node x is not properly configured : ()=abc");

    args = new String[] {"-replace", ":x(string)"};
    assertTrue(0 != runTool(args), "It should have failed as no node is specified");
    assertFailureMessageContains("Node name cannot be empty");

    // Not expected key=value specifying inner parenthesis
    args = new String[] {"-replace", "x:(key=value)"};
    assertTrue(0 != runTool(args));
    assertFailureMessageContains(
        "Attribute for node x is not properly configured : (key=value)");

    // Should fail as no attributes specified
    args = new String[] {"-replace"};
    assertTrue(0 != runTool(args), "Should fail as no attribute mappings specified");
    assertFailureMessageContains(NodeAttributesCLI.MISSING_ARGUMENT);

    // no labels, should fail
    args = new String[] {"-replace", "-failOnUnknownNodes",
        "x:key(string)=value,key2=val2"};
    assertTrue(0 != runTool(args),
        "Should fail as no attribute mappings specified for replace");
    assertFailureMessageContains(NodeAttributesCLI.MISSING_ARGUMENT);

    // no labels, should fail
    args = new String[] {"-replace", " "};
    assertTrue(0 != runTool(args));
    assertFailureMessageContains(NodeAttributesCLI.NO_MAPPING_ERR_MSG);

    args = new String[] {"-replace", ", "};
    assertTrue(0 != runTool(args));
    assertFailureMessageContains(NodeAttributesCLI.INVALID_MAPPING_ERR_MSG);
    // --------------------------------
    // success scenarios
    // --------------------------------
    args = new String[] {"-replace",
        "x:key(string)=value,key2=val2 y:key2=val23,key3 z:key4"};
    assertTrue(0 == runTool(args),
        "Should not fail as attribute has been properly mapped");
    List<NodeToAttributes> nodeAttributesList = new ArrayList<>();
    List<NodeAttribute> attributes = new ArrayList<>();
    attributes.add(
        NodeAttribute.newInstance("key", NodeAttributeType.STRING, "value"));
    attributes.add(
        NodeAttribute.newInstance("key2", NodeAttributeType.STRING, "val2"));
    nodeAttributesList.add(NodeToAttributes.newInstance("x", attributes));

    // for node y
    attributes = new ArrayList<>();
    attributes.add(
        NodeAttribute.newInstance("key2", NodeAttributeType.STRING, "val23"));
    attributes
        .add(NodeAttribute.newInstance("key3", NodeAttributeType.STRING, ""));
    nodeAttributesList.add(NodeToAttributes.newInstance("y", attributes));

    // for node y
    attributes = new ArrayList<>();
    attributes.add(
        NodeAttribute.newInstance("key2", NodeAttributeType.STRING, "val23"));
    attributes
        .add(NodeAttribute.newInstance("key3", NodeAttributeType.STRING, ""));
    nodeAttributesList.add(NodeToAttributes.newInstance("y", attributes));

    // for node z
    attributes = new ArrayList<>();
    attributes
        .add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, ""));
    nodeAttributesList.add(NodeToAttributes.newInstance("z", attributes));

    NodesToAttributesMappingRequest expected = NodesToAttributesMappingRequest
        .newInstance(AttributeMappingOperationType.REPLACE, nodeAttributesList,
            false);
    assertTrue(nodeToAttrRequest.equals(expected));
  }

  @Test
  public void testRemove() throws Exception {
    // --------------------------------
    // failure scenarios
    // --------------------------------
    // parenthesis not match
    String[] args = new String[] {"-remove", "x:"};
    assertTrue(0 != runTool(args),
        "It should have failed as no node is specified");
    assertFailureMessageContains(
        "Attributes cannot be null or empty for Operation [remove] on the "
            + "node x");
    // --------------------------------
    // success scenarios
    // --------------------------------
    args =
        new String[] {"-remove", "x:key2,key3 z:key4", "-failOnUnknownNodes"};
    assertTrue(0 == runTool(args),
        "Should not fail as attribute has been properly mapped");
    List<NodeToAttributes> nodeAttributesList = new ArrayList<>();
    List<NodeAttribute> attributes = new ArrayList<>();
    attributes
        .add(NodeAttribute.newInstance("key2", NodeAttributeType.STRING, ""));
    attributes
        .add(NodeAttribute.newInstance("key3", NodeAttributeType.STRING, ""));
    nodeAttributesList.add(NodeToAttributes.newInstance("x", attributes));

    // for node z
    attributes = new ArrayList<>();
    attributes
        .add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, ""));
    nodeAttributesList.add(NodeToAttributes.newInstance("z", attributes));

    NodesToAttributesMappingRequest expected = NodesToAttributesMappingRequest
        .newInstance(AttributeMappingOperationType.REMOVE, nodeAttributesList,
            true);
    assertTrue(nodeToAttrRequest.equals(expected));
  }

  @Test
  public void testAdd() throws Exception {
    // --------------------------------
    // failure scenarios
    // --------------------------------
    // parenthesis not match
    String[] args = new String[] {"-add", "x:"};
    assertTrue(0 != runTool(args),
        "It should have failed as no node is specified");
    assertFailureMessageContains(
        "Attributes cannot be null or empty for Operation [add] on the node x");
    // --------------------------------
    // success scenarios
    // --------------------------------
    args = new String[] {"-add", "x:key2=123,key3=abc z:key4(string)",
        "-failOnUnknownNodes"};
    assertTrue(0 == runTool(args),
        "Should not fail as attribute has been properly mapped");
    List<NodeToAttributes> nodeAttributesList = new ArrayList<>();
    List<NodeAttribute> attributes = new ArrayList<>();
    attributes.add(
        NodeAttribute.newInstance("key2", NodeAttributeType.STRING, "123"));
    attributes.add(
        NodeAttribute.newInstance("key3", NodeAttributeType.STRING, "abc"));
    nodeAttributesList.add(NodeToAttributes.newInstance("x", attributes));

    // for node z
    attributes = new ArrayList<>();
    attributes
        .add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, ""));
    nodeAttributesList.add(NodeToAttributes.newInstance("z", attributes));

    NodesToAttributesMappingRequest expected = NodesToAttributesMappingRequest
        .newInstance(AttributeMappingOperationType.ADD, nodeAttributesList,
            true);
    assertTrue(nodeToAttrRequest.equals(expected));

    // --------------------------------
    // with Duplicate mappings for a host
    // --------------------------------
    args = new String[] {"-add", "x:key2=123,key3=abc x:key4(string)",
        "-failOnUnknownNodes"};
    assertTrue(0 == runTool(args),
        "Should not fail as attribute has been properly mapped");
    nodeAttributesList = new ArrayList<>();
    attributes = new ArrayList<>();
    attributes
        .add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, ""));
    nodeAttributesList.add(NodeToAttributes.newInstance("x", attributes));

    expected = NodesToAttributesMappingRequest
        .newInstance(AttributeMappingOperationType.ADD, nodeAttributesList,
            true);
    assertTrue(nodeToAttrRequest.equals(expected));
  }

  @Test
  public void testListAttributes() throws Exception {

    // GetClusterNodeAttributesRequest
    when(client
        .getClusterNodeAttributes(any(GetClusterNodeAttributesRequest.class)))
        .thenAnswer(new Answer<GetClusterNodeAttributesResponse>() {
          @Override
          public GetClusterNodeAttributesResponse answer(
              InvocationOnMock invocation) throws Throwable {
            GetClusterNodeAttributesRequest nodeAttrReq =
                (GetClusterNodeAttributesRequest) invocation.getArguments()[0];
            return GetClusterNodeAttributesResponse.newInstance(ImmutableSet
                .of(NodeAttributeInfo
                    .newInstance(NodeAttributeKey.newInstance("GPU"),
                        NodeAttributeType.STRING)));
          }
        });

    // --------------------------------
    // Success scenarios
    // --------------------------------
    String[] args = new String[] {"-list"};
    assertTrue(0 == runTool(args),
        "It should be success since it list all attributes");
    assertSysOutContains("Attribute\t           Type",
        "rm.yarn.io/GPU\t         STRING");
  }

  @Test
  public void testNodeToAttributes() throws Exception {
    // GetNodesToAttributesRequest response
    when(client.getNodesToAttributes(any(GetNodesToAttributesRequest.class)))
        .thenAnswer(new Answer<GetNodesToAttributesResponse>() {
          @Override
          public GetNodesToAttributesResponse answer(
              InvocationOnMock invocation) throws Throwable {
            GetNodesToAttributesRequest nodeToAttributes =
                (GetNodesToAttributesRequest) invocation.getArguments()[0];
            return GetNodesToAttributesResponse.newInstance(
                ImmutableMap.<String, Set<NodeAttribute>>builder()
                    .put("hostname", ImmutableSet.of(NodeAttribute
                        .newInstance("GPU", NodeAttributeType.STRING, "ARM")))
                    .build());
          }
        });
    // --------------------------------
    // Failure scenarios
    // --------------------------------
    String[] args = new String[] {"-nodetoattributes", "-nodes"};
    assertTrue(0 != runTool(args),
        "It should not success since nodes are not specified");
    assertErrorContains(NodeAttributesCLI.INVALID_COMMAND_USAGE);

    // Missing argument for nodes
    args = new String[] {"-nodestoattributes", "-nodes"};
    assertTrue(0 != runTool(args),
        "It should not success since nodes are not specified");
    assertErrorContains(NodeAttributesCLI.MISSING_ARGUMENT);

    // --------------------------------
    // Success with hostname param
    // --------------------------------
    args = new String[] {"-nodestoattributes", "-nodes", "hostname"};
    assertTrue(0 == runTool(args), "Should return hostname to attributed list");
    assertSysOutContains("hostname");
  }

  @Test
  public void testAttributesToNodes() throws Exception {
    // GetAttributesToNodesResponse response
    when(client.getAttributesToNodes(any(GetAttributesToNodesRequest.class)))
        .thenAnswer(new Answer<GetAttributesToNodesResponse>() {
          @Override
          public GetAttributesToNodesResponse answer(
              InvocationOnMock invocation) throws Throwable {
            GetAttributesToNodesRequest attrToNodes =
                (GetAttributesToNodesRequest) invocation.getArguments()[0];
            return GetAttributesToNodesResponse.newInstance(
                ImmutableMap.<NodeAttributeKey,
                    List<NodeToAttributeValue>>builder()
                    .put(NodeAttributeKey.newInstance("GPU"), ImmutableList
                        .of(NodeToAttributeValue.newInstance("host1", "ARM")))
                    .build());
          }
        });
    // --------------------------------
    // Success scenarios
    // --------------------------------
    String[] args = new String[] {"-attributestonodes"};
    assertTrue(0 == runTool(args),
        "It should be success since it list all attributes");
    assertSysOutContains("Hostname\tAttribute-value", "rm.yarn.io/GPU :",
        "host1\t            ARM");

    // --------------------------------
    // fail scenario argument filter missing
    // --------------------------------
    args = new String[] {"-attributestonodes", "-attributes"};
    assertTrue(0 != runTool(args),
        "It should not success since attributes for filter are not specified");
    assertErrorContains(NodeAttributesCLI.MISSING_ARGUMENT);

    // --------------------------------
    // fail scenario argument filter missing
    // --------------------------------
    args = new String[] {"-attributestonodes", "-attributes", "fail/da/fail"};
    assertTrue(0 != runTool(args),
        "It should not success since attributes format is not correct");
    assertErrorContains(
        "Attribute format not correct. Should be <[prefix]/[name]> "
            + ":fail/da/fail");
  }

  private void assertFailureMessageContains(String... messages) {
    assertErrorContains(messages);
    assertErrorContains(NodeAttributesCLI.USAGE_YARN_NODE_ATTRIBUTES);
  }

  private void assertErrorContains(String... messages) {
    for (String message : messages) {
      if (!errOutput.contains(message)) {
        fail(
            "Expected output to contain '" + message + "' but err_output was:\n"
                + errOutput);
      }
    }
  }

  private void assertSysOutContains(String... messages) {
    for (String message : messages) {
      if (!sysOutput.contains(message)) {
        fail(
            "Expected output to contain '" + message + "' but sys_output was:\n"
                + sysOutput);
      }
    }
  }

  private int runTool(String... args) throws Exception {
    errOutBytes.reset();
    sysOutBytes.reset();
    LOG.info("Running: NodeAttributesCLI " + Joiner.on(" ").join(args));
    int ret = nodeAttributesCLI.run(args);
    errOutput = new String(errOutBytes.toByteArray(), StandardCharsets.UTF_8);
    sysOutput = new String(sysOutBytes.toByteArray(), StandardCharsets.UTF_8);
    LOG.info("Err_output:\n" + errOutput);
    LOG.info("Sys_output:\n" + sysOutput);
    return ret;
  }
}