YARN-8103. Add CLI interface to query node attributes. Contributed by Bibin A Chundatt.

This commit is contained in:
Naganarasimha 2018-06-28 08:13:09 +08:00 committed by Sunil G
parent 76183428b7
commit eb08543c7a
23 changed files with 1067 additions and 470 deletions

View File

@ -19,6 +19,7 @@
package org.apache.hadoop.yarn.sls.nodemanager; package org.apache.hadoop.yarn.sls.nodemanager;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Set; import java.util.Set;
@ -220,16 +221,9 @@ public class NodeInfo {
return null; return null;
} }
@Override @Override
public void setNodeAttributes(String prefix, public Set<NodeAttribute> getAllNodeAttributes() {
Set<NodeAttribute> nodeAttributes) { return Collections.emptySet();
}
@Override
public Map<String, Set<NodeAttribute>> getAllNodeAttributes() {
return null;
} }
@Override @Override

View File

@ -209,13 +209,7 @@ public class RMNodeWrapper implements RMNode {
} }
@Override @Override
public void setNodeAttributes(String prefix, public Set<NodeAttribute> getAllNodeAttributes() {
Set<NodeAttribute> nodeAttributes) {
node.setNodeAttributes(prefix, nodeAttributes);
}
@Override
public Map<String, Set<NodeAttribute>> getAllNodeAttributes() {
return node.getAllNodeAttributes(); return node.getAllNodeAttributes();
} }

View File

@ -55,7 +55,7 @@ function hadoop_usage
hadoop_add_subcommand "timelinereader" client "run the timeline reader server" hadoop_add_subcommand "timelinereader" client "run the timeline reader server"
hadoop_add_subcommand "timelineserver" daemon "run the timeline server" hadoop_add_subcommand "timelineserver" daemon "run the timeline server"
hadoop_add_subcommand "top" client "view cluster information" hadoop_add_subcommand "top" client "view cluster information"
hadoop_add_subcommand "node-attributes" "map node to attibutes" hadoop_add_subcommand "nodeattributes" client "node attributes cli client"
hadoop_add_subcommand "version" client "print the version" hadoop_add_subcommand "version" client "print the version"
hadoop_generate_usage "${HADOOP_SHELL_EXECNAME}" true hadoop_generate_usage "${HADOOP_SHELL_EXECNAME}" true
} }
@ -187,7 +187,7 @@ ${HADOOP_COMMON_HOME}/${HADOOP_COMMON_LIB_JARS_DIR}"
hadoop_add_classpath "$HADOOP_YARN_HOME/$YARN_DIR/timelineservice/lib/*" hadoop_add_classpath "$HADOOP_YARN_HOME/$YARN_DIR/timelineservice/lib/*"
HADOOP_CLASSNAME='org.apache.hadoop.yarn.server.timelineservice.reader.TimelineReaderServer' HADOOP_CLASSNAME='org.apache.hadoop.yarn.server.timelineservice.reader.TimelineReaderServer'
;; ;;
node-attributes) nodeattributes)
HADOOP_SUBCMD_SUPPORTDAEMONIZATION="false" HADOOP_SUBCMD_SUPPORTDAEMONIZATION="false"
HADOOP_CLASSNAME='org.apache.hadoop.yarn.client.cli.NodeAttributesCLI' HADOOP_CLASSNAME='org.apache.hadoop.yarn.client.cli.NodeAttributesCLI'
;; ;;

View File

@ -258,4 +258,17 @@ public abstract class NodeReport {
* Set the node update type (null indicates absent node update type). * Set the node update type (null indicates absent node update type).
* */ * */
public void setNodeUpdateType(NodeUpdateType nodeUpdateType) {} public void setNodeUpdateType(NodeUpdateType nodeUpdateType) {}
/**
* Set the node attributes of node.
*
* @param nodeAttributes set of node attributes.
*/
public abstract void setNodeAttributes(Set<NodeAttribute> nodeAttributes);
/**
* Get node attributes of node.
* @return the set of node attributes.
*/
public abstract Set<NodeAttribute> getNodeAttributes();
} }

View File

@ -355,6 +355,7 @@ message NodeReportProto {
optional ResourceUtilizationProto node_utilization = 12; optional ResourceUtilizationProto node_utilization = 12;
optional uint32 decommissioning_timeout = 13; optional uint32 decommissioning_timeout = 13;
optional NodeUpdateTypeProto node_update_type = 14; optional NodeUpdateTypeProto node_update_type = 14;
repeated NodeAttributeProto node_attributes = 15;
} }
message NodeIdToLabelsProto { message NodeIdToLabelsProto {

View File

@ -36,6 +36,7 @@ import org.apache.commons.lang3.StringUtils;
import org.apache.hadoop.classification.InterfaceAudience.Private; import org.apache.hadoop.classification.InterfaceAudience.Private;
import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.util.ToolRunner; import org.apache.hadoop.util.ToolRunner;
import org.apache.hadoop.yarn.api.records.NodeAttributeInfo;
import org.apache.hadoop.yarn.api.records.NodeLabel; import org.apache.hadoop.yarn.api.records.NodeLabel;
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;
@ -52,6 +53,7 @@ public class ClusterCLI extends YarnCLI {
public static final String LIST_LABELS_CMD = "list-node-labels"; public static final String LIST_LABELS_CMD = "list-node-labels";
public static final String DIRECTLY_ACCESS_NODE_LABEL_STORE = public static final String DIRECTLY_ACCESS_NODE_LABEL_STORE =
"directly-access-node-label-store"; "directly-access-node-label-store";
public static final String LIST_CLUSTER_ATTRIBUTES="list-node-attributes";
public static final String CMD = "cluster"; public static final String CMD = "cluster";
private boolean accessLocal = false; private boolean accessLocal = false;
static CommonNodeLabelsManager localNodeLabelsManager = null; static CommonNodeLabelsManager localNodeLabelsManager = null;
@ -71,6 +73,8 @@ public class ClusterCLI extends YarnCLI {
opts.addOption("lnl", LIST_LABELS_CMD, false, opts.addOption("lnl", LIST_LABELS_CMD, false,
"List cluster node-label collection"); "List cluster node-label collection");
opts.addOption("lna", LIST_CLUSTER_ATTRIBUTES, false,
"List cluster node-attribute collection");
opts.addOption("h", HELP_CMD, false, "Displays help for all commands."); opts.addOption("h", HELP_CMD, false, "Displays help for all commands.");
opts.addOption("dnl", DIRECTLY_ACCESS_NODE_LABEL_STORE, false, opts.addOption("dnl", DIRECTLY_ACCESS_NODE_LABEL_STORE, false,
"This is DEPRECATED, will be removed in future releases. Directly access node label store, " "This is DEPRECATED, will be removed in future releases. Directly access node label store, "
@ -102,6 +106,8 @@ public class ClusterCLI extends YarnCLI {
if (parsedCli.hasOption(LIST_LABELS_CMD)) { if (parsedCli.hasOption(LIST_LABELS_CMD)) {
printClusterNodeLabels(); printClusterNodeLabels();
} else if(parsedCli.hasOption(LIST_CLUSTER_ATTRIBUTES)){
printClusterNodeAttributes();
} else if (parsedCli.hasOption(HELP_CMD)) { } else if (parsedCli.hasOption(HELP_CMD)) {
printUsage(opts); printUsage(opts);
return 0; return 0;
@ -112,6 +118,17 @@ public class ClusterCLI extends YarnCLI {
return 0; return 0;
} }
private void printClusterNodeAttributes() throws IOException, YarnException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
PrintWriter pw = new PrintWriter(
new OutputStreamWriter(baos, Charset.forName("UTF-8")));
for (NodeAttributeInfo attribute : client.getClusterAttributes()) {
pw.println(attribute.toString());
}
pw.close();
sysout.println(baos.toString("UTF-8"));
}
void printClusterNodeLabels() throws YarnException, IOException { void printClusterNodeLabels() throws YarnException, IOException {
List<NodeLabel> nodeLabels = null; List<NodeLabel> nodeLabels = null;
if (accessLocal) { if (accessLocal) {

View File

@ -18,29 +18,30 @@
package org.apache.hadoop.yarn.client.cli; package org.apache.hadoop.yarn.client.cli;
import java.io.IOException; import com.google.common.base.Preconditions;
import java.io.PrintStream; import com.google.common.collect.Lists;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.apache.commons.cli.CommandLine; import org.apache.commons.cli.CommandLine;
import org.apache.commons.cli.GnuParser; import org.apache.commons.cli.GnuParser;
import org.apache.commons.cli.MissingArgumentException; import org.apache.commons.cli.MissingArgumentException;
import org.apache.commons.cli.Option;
import org.apache.commons.cli.OptionGroup;
import org.apache.commons.cli.Options; import org.apache.commons.cli.Options;
import org.apache.commons.cli.ParseException; import org.apache.commons.cli.UnrecognizedOptionException;
import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.conf.Configured; import org.apache.hadoop.conf.Configured;
import org.apache.hadoop.fs.CommonConfigurationKeys; import org.apache.hadoop.fs.CommonConfigurationKeys;
import org.apache.hadoop.ha.HAAdmin.UsageInfo;
import org.apache.hadoop.ipc.RemoteException;
import org.apache.hadoop.util.Tool; import org.apache.hadoop.util.Tool;
import org.apache.hadoop.util.ToolRunner; import org.apache.hadoop.util.ToolRunner;
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.NodeAttribute; import org.apache.hadoop.yarn.api.records.NodeAttribute;
import org.apache.hadoop.yarn.api.records.NodeAttributeInfo;
import org.apache.hadoop.yarn.api.records.NodeAttributeKey;
import org.apache.hadoop.yarn.api.records.NodeAttributeType; import org.apache.hadoop.yarn.api.records.NodeAttributeType;
import org.apache.hadoop.yarn.client.ClientRMProxy; import org.apache.hadoop.yarn.client.ClientRMProxy;
import org.apache.hadoop.yarn.conf.YarnConfiguration; import org.apache.hadoop.yarn.conf.YarnConfiguration;
@ -50,13 +51,24 @@ import org.apache.hadoop.yarn.server.api.protocolrecords.AttributeMappingOperati
import org.apache.hadoop.yarn.server.api.protocolrecords.NodeToAttributes; 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.NodesToAttributesMappingRequest;
import com.google.common.base.Preconditions; import java.io.ByteArrayOutputStream;
import com.google.common.collect.ImmutableMap; import java.io.IOException;
import com.google.common.collect.Lists; import java.io.OutputStreamWriter;
import java.io.PrintStream;
import java.io.PrintWriter;
import java.io.UnsupportedEncodingException;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
/** /**
* CLI to map attributes to Nodes. * CLI to map attributes to Nodes.
*
*/ */
public class NodeAttributesCLI extends Configured implements Tool { public class NodeAttributesCLI extends Configured implements Tool {
@ -64,262 +76,529 @@ public class NodeAttributesCLI extends Configured implements Tool {
"Invalid Node to attribute mapping : "; "Invalid Node to attribute mapping : ";
protected static final String USAGE_YARN_NODE_ATTRIBUTES = protected static final String USAGE_YARN_NODE_ATTRIBUTES =
"Usage: yarn node-attributes "; "Usage: yarn nodeattributes ";
protected static final String MISSING_ARGUMENT =
"Missing argument for command";
protected static final String NO_MAPPING_ERR_MSG = protected static final String NO_MAPPING_ERR_MSG =
"No node-to-attributes mappings are specified"; "No node-to-attributes mappings are specified";
protected final static Map<String, UsageInfo> NODE_ATTRIB_USAGE = private static final String DEFAULT_SEPARATOR = System.lineSeparator();
ImmutableMap.<String, UsageInfo>builder() public static final String INVALID_COMMAND_USAGE = "Invalid Command Usage : ";
.put("-replace", /**
new UsageInfo( * Output stream for errors, for use in tests.
"<\"node1:attribute[(type)][=value],attribute1[=value]," */
+ "attribute2 node2:attribute2[=value],attribute3\">",
" 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."))
.put("-add",
new UsageInfo(
"<\"node1:attribute[(type)][=value],attribute1[=value],"
+ "attribute2 node2:attribute2[=value],attribute3\">",
" 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."))
.put("-remove",
new UsageInfo("<\"node1:attribute,attribute1 node2:attribute2\">",
" Removes the specified node to attributes mapping"
+ " information at the ResourceManager"))
.put("-failOnUnknownNodes",
new UsageInfo("",
"Can be used optionally along with other options. When its"
+ " set, it will fail if specified nodes are unknown."))
.build();
/** Output stream for errors, for use in tests. */
private PrintStream errOut = System.err; private PrintStream errOut = System.err;
public NodeAttributesCLI() { public NodeAttributesCLI() {
super(); super();
} }
public NodeAttributesCLI(Configuration conf) {
super(conf);
}
protected void setErrOut(PrintStream errOut) { protected void setErrOut(PrintStream errOut) {
this.errOut = errOut; this.errOut = errOut;
} }
private void printHelpMsg(String cmd) { protected AdminCommandHandler getAdminCommandHandler() {
StringBuilder builder = new StringBuilder(); return new AdminCommandHandler();
UsageInfo usageInfo = null;
if (cmd != null && !(cmd.trim().isEmpty())) {
usageInfo = NODE_ATTRIB_USAGE.get(cmd);
}
if (usageInfo != null) {
if (usageInfo.args == null) {
builder.append(" " + cmd + ":\n" + usageInfo.help);
} else {
String space = (usageInfo.args == "") ? "" : " ";
builder.append(
" " + cmd + space + usageInfo.args + " :\n" + usageInfo.help);
}
} else {
// help for all commands
builder.append("Usage: yarn node-attributes\n");
for (Map.Entry<String, UsageInfo> cmdEntry : NODE_ATTRIB_USAGE
.entrySet()) {
usageInfo = cmdEntry.getValue();
builder.append(" " + cmdEntry.getKey() + " " + usageInfo.args
+ " :\n " + usageInfo.help + "\n");
}
builder.append(" -help" + " [cmd]\n");
}
errOut.println(builder);
} }
private static void buildIndividualUsageMsg(String cmd, protected ClientCommandHandler getClientCommandHandler() {
StringBuilder builder) { return new ClientCommandHandler();
UsageInfo usageInfo = NODE_ATTRIB_USAGE.get(cmd);
if (usageInfo == null) {
return;
} }
if (usageInfo.args == null) {
builder.append(USAGE_YARN_NODE_ATTRIBUTES + cmd + "\n"); void printUsage(String cmd, boolean desc, CommandHandler... handlers)
throws UnsupportedEncodingException {
StringBuilder usageBuilder = new StringBuilder();
usageBuilder.append(USAGE_YARN_NODE_ATTRIBUTES);
boolean satisfied = false;
for (CommandHandler cmdHandlers : handlers) {
satisfied |= cmdHandlers.getHelp(cmd, usageBuilder, desc);
}
if (!satisfied) {
printUsage(desc, handlers);
} else { } else {
String space = (usageInfo.args == "") ? "" : " "; print(usageBuilder);
builder.append(
USAGE_YARN_NODE_ATTRIBUTES + cmd + space + usageInfo.args + "\n");
} }
} }
private static void buildUsageMsgForAllCmds(StringBuilder builder) { private void printUsage(boolean desc, CommandHandler... handlers)
builder.append("Usage: yarn node-attributes\n"); throws UnsupportedEncodingException {
for (Map.Entry<String, UsageInfo> cmdEntry : NODE_ATTRIB_USAGE.entrySet()) { StringBuilder usageBuilder = new StringBuilder();
UsageInfo usageInfo = cmdEntry.getValue(); usageBuilder.append(USAGE_YARN_NODE_ATTRIBUTES);
builder.append(" " + cmdEntry.getKey() + " " + usageInfo.args + "\n"); for (CommandHandler cmdHandlers : handlers) {
cmdHandlers.getHelp(usageBuilder, desc);
} }
builder.append(" -help" + " [cmd]\n");
// append help with usage
usageBuilder.append(DEFAULT_SEPARATOR)
.append(" -help [cmd] List help of commands");
print(usageBuilder);
}
private void print(StringBuilder usageBuilder)
throws UnsupportedEncodingException {
ByteArrayOutputStream baos = new ByteArrayOutputStream();
PrintWriter pw =
new PrintWriter(new OutputStreamWriter(baos, Charset.forName("UTF-8")));
pw.write(usageBuilder.toString());
pw.close();
errOut.println(baos.toString("UTF-8"));
}
private Options buildOptions(CommandHandler... handlers) {
Options opts = new Options();
for (CommandHandler handler : handlers) {
Options handlerOpts = handler.getOptions();
handlerOpts.getOptions().iterator()
.forEachRemaining(option -> opts.addOption((Option) option));
}
return opts;
}
public int run(String[] args) throws Exception {
int exitCode = -1;
AdminCommandHandler adminCmdHandler = getAdminCommandHandler();
ClientCommandHandler clientCmdHandler = getClientCommandHandler();
// Build options
Options opts = buildOptions(adminCmdHandler, clientCmdHandler);
if (args.length < 1) {
printUsage(false, adminCmdHandler, clientCmdHandler);
return -1;
}
// Handle command separate
if (handleHelpCommand(args, adminCmdHandler, clientCmdHandler)) {
return 0;
}
CommandLine cliParser;
CommandHandler handler = null;
try {
cliParser = new GnuParser().parse(opts, args);
handler = adminCmdHandler.canHandleCommand(cliParser) ?
adminCmdHandler :
clientCmdHandler.canHandleCommand(cliParser) ?
clientCmdHandler :
null;
if (handler == null) {
errOut.println(INVALID_COMMAND_USAGE);
printUsage(false, adminCmdHandler, clientCmdHandler);
return exitCode;
} else {
return handler.handleCommand(cliParser);
}
} catch (UnrecognizedOptionException e) {
errOut.println(INVALID_COMMAND_USAGE);
printUsage(false, adminCmdHandler, clientCmdHandler);
return exitCode;
} catch (MissingArgumentException ex) {
errOut.println(MISSING_ARGUMENT);
printUsage(true, adminCmdHandler, clientCmdHandler);
return exitCode;
} catch (IllegalArgumentException arge) {
errOut.println(arge.getLocalizedMessage());
// print admin command detail
printUsage(true, handler);
return exitCode;
} catch (Exception e) {
errOut.println(e.toString());
printUsage(true, handler);
return exitCode;
}
}
private boolean handleHelpCommand(String[] args, CommandHandler... handlers)
throws UnsupportedEncodingException {
if (args[0].equals("-help")) {
if (args.length == 2) {
printUsage(args[1], true, handlers);
} else {
printUsage(true, handlers);
}
return true;
}
return false;
}
public static void main(String[] args) throws Exception {
int result = ToolRunner.run(new NodeAttributesCLI(), args);
System.exit(result);
} }
/** /**
* Displays format of commands. * Abstract class for command handler.
*
* @param cmd The command that is being executed.
*/ */
private void printUsage(String cmd) { public static abstract class CommandHandler extends Configured {
StringBuilder usageBuilder = new StringBuilder();
if (NODE_ATTRIB_USAGE.containsKey(cmd)) { private Options options;
buildIndividualUsageMsg(cmd, usageBuilder);
} else { private LinkedList<String> order = new LinkedList<>();
buildUsageMsgForAllCmds(usageBuilder); private String header;
}
errOut.println(usageBuilder); protected CommandHandler(String header) {
this(new YarnConfiguration());
this.header = header;
} }
private void printUsage() { protected CommandHandler(Configuration conf) {
printUsage(""); super(conf);
options = buildOptions();
}
public boolean canHandleCommand(CommandLine parse) {
ArrayList<Option> arrayList = new ArrayList<Option>(options.getOptions());
return arrayList.stream().anyMatch(opt -> parse.hasOption(opt.getOpt()));
}
public abstract int handleCommand(CommandLine parse)
throws IOException, YarnException;
public abstract Options buildOptions();
public Options getOptions() {
return options;
}
public boolean getHelp(String cmd, StringBuilder strcnd, boolean addDesc) {
Option opt = options.getOption(cmd);
if (opt != null) {
strcnd.append(DEFAULT_SEPARATOR).append(" -").append(opt.getOpt());
if (opt.hasArg()) {
strcnd.append(" <").append(opt.getArgName()).append(">");
}
if (addDesc) {
strcnd.append(DEFAULT_SEPARATOR).append("\t")
.append(opt.getDescription());
}
}
return opt == null;
}
public void getHelp(StringBuilder builder, boolean description) {
builder.append(DEFAULT_SEPARATOR).append(DEFAULT_SEPARATOR)
.append(header);
for (String option : order) {
getHelp(option, builder, description);
}
}
protected void addOrder(String key){
order.add(key);
}
}
/**
* Client commands handler.
*/
public static class ClientCommandHandler extends CommandHandler {
private static final String LIST_ALL_ATTRS = "list";
private static final String NODESTOATTR = "nodestoattributes";
private static final String NODES = "nodes";
private static final String ATTRTONODES = "attributestonodes";
private static final String ATTRIBUTES = "attributes";
public static final String SPLITPATTERN = "/";
private static final String NODEATTRIBUTE =
"%40s\t%10s\t%20s" + DEFAULT_SEPARATOR;
private static final String NODEATTRIBUTEINFO =
"%40s\t%15s" + DEFAULT_SEPARATOR;
private static final String HOSTNAMEVAL = "%40s\t%15s" + DEFAULT_SEPARATOR;
private PrintStream sysOut = System.out;
public ClientCommandHandler() {
super("Client Commands:");
}
public void setSysOut(PrintStream out) {
this.sysOut = out;
}
@Override
public int handleCommand(CommandLine parse)
throws IOException, YarnException {
if (parse.hasOption(LIST_ALL_ATTRS)) {
return printClusterAttributes();
} else if (parse.hasOption(NODESTOATTR)) {
String[] nodes = new String[0];
if (parse.hasOption(NODES)) {
nodes = parse.getOptionValues(NODES);
}
return printAttributesByNode(nodes);
} else if (parse.hasOption(ATTRTONODES)) {
String[] attrKeys = {};
if (parse.hasOption(ATTRIBUTES)) {
attrKeys = parse.getOptionValues(ATTRIBUTES);
}
return printNodesByAttributes(attrKeys);
}
return 0;
}
protected ApplicationClientProtocol createApplicationProtocol()
throws IOException {
// Get the current configuration
final YarnConfiguration conf = new YarnConfiguration(getConf());
return ClientRMProxy.createRMProxy(conf, ApplicationClientProtocol.class);
}
public int printNodesByAttributes(String[] attrs)
throws YarnException, IOException {
ApplicationClientProtocol protocol = createApplicationProtocol();
HashSet<NodeAttributeKey> set = new HashSet<>();
for (String attr : attrs) {
String[] attrFields = attr.split(SPLITPATTERN);
if (attrFields.length == 1) {
set.add(NodeAttributeKey.newInstance(attrFields[0]));
} else if (attrFields.length == 2) {
set.add(NodeAttributeKey.newInstance(attrFields[0], attrFields[1]));
} else {
throw new IllegalArgumentException(
" Attribute format not correct. Should be <[prefix]/[name]> :"
+ attr);
}
}
GetAttributesToNodesRequest request =
GetAttributesToNodesRequest.newInstance(set);
GetAttributesToNodesResponse response =
protocol.getAttributesToNodes(request);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
PrintWriter writer = new PrintWriter(
new OutputStreamWriter(baos, Charset.forName("UTF-8")));
writer.format(HOSTNAMEVAL, "Hostname", "Attribute-value");
response.getAttributesToNodes().forEach((attributeKey, v) -> {
writer.println(getKeyString(attributeKey) + " :");
v.iterator().forEachRemaining(attrVal -> writer
.format(HOSTNAMEVAL, attrVal.getHostname(),
attrVal.getAttributeValue()));
});
writer.close();
sysOut.println(baos.toString("UTF-8"));
return 0;
}
private int printAttributesByNode(String[] nodeArray)
throws YarnException, IOException {
ApplicationClientProtocol protocol = createApplicationProtocol();
HashSet<String> nodes = new HashSet<>(Arrays.asList(nodeArray));
GetNodesToAttributesRequest request =
GetNodesToAttributesRequest.newInstance(nodes);
GetNodesToAttributesResponse response =
protocol.getNodesToAttributes(request);
Map<String, Set<NodeAttribute>> nodeToAttrs =
response.getNodeToAttributes();
ByteArrayOutputStream baos = new ByteArrayOutputStream();
PrintWriter writer = new PrintWriter(
new OutputStreamWriter(baos, Charset.forName("UTF-8")));
writer.printf(NODEATTRIBUTE, "Attribute", "Type", "Value");
nodeToAttrs.forEach((node, v) -> {
// print node header
writer.println(node + ":");
v.iterator().forEachRemaining(attr -> writer
.format(NODEATTRIBUTE, getKeyString(attr.getAttributeKey()),
attr.getAttributeType().name(), attr.getAttributeValue()));
});
writer.close();
sysOut.println(baos.toString("UTF-8"));
return 0;
}
private int printClusterAttributes() throws IOException, YarnException {
ApplicationClientProtocol protocol = createApplicationProtocol();
GetClusterNodeAttributesRequest request =
GetClusterNodeAttributesRequest.newInstance();
GetClusterNodeAttributesResponse response =
protocol.getClusterNodeAttributes(request);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
PrintWriter writer = new PrintWriter(
new OutputStreamWriter(baos, Charset.forName("UTF-8")));
writer.format(NODEATTRIBUTEINFO, "Attribute", "Type");
for (NodeAttributeInfo attr : response.getNodeAttributes()) {
writer.format(NODEATTRIBUTEINFO, getKeyString(attr.getAttributeKey()),
attr.getAttributeType().name());
}
writer.close();
sysOut.println(baos.toString("UTF-8"));
return 0;
}
private String getKeyString(NodeAttributeKey key) {
StringBuilder sb = new StringBuilder();
sb.append(key.getAttributePrefix()).append("/")
.append(key.getAttributeName());
return sb.toString();
}
@Override
public Options buildOptions() {
Options clientOptions = new Options();
clientOptions.addOption(
new Option(LIST_ALL_ATTRS, false, "List all attributes in cluster"));
// group by command
OptionGroup nodeToAttr = new OptionGroup();
Option attrtonodes = new Option(NODESTOATTR, false,
"Lists all mapping to nodes to attributes");
Option nodes = new Option(NODES,
"Works with [" + LIST_ALL_ATTRS + "] to specify node hostnames "
+ "whose mappings are required to be displayed.");
nodes.setValueSeparator(',');
nodes.setArgName("Host Names");
nodes.setArgs(Option.UNLIMITED_VALUES);
nodeToAttr.addOption(attrtonodes);
nodeToAttr.addOption(nodes);
clientOptions.addOptionGroup(nodeToAttr);
// Defines as groups to add extendability for later
OptionGroup attrToNodes = new OptionGroup();
attrToNodes.addOption(new Option(ATTRTONODES, false,
"Displays mapping of "
+ "attributes to nodes and attribute values grouped by "
+ "attributes"));
Option attrs = new Option(ATTRIBUTES, "Works with [" + ATTRTONODES
+ "] to specify attributes whose mapping "
+ "are required to be displayed.");
attrs.setValueSeparator(',');
attrs.setArgName("Attributes");
attrs.setArgs(Option.UNLIMITED_VALUES);
attrToNodes.addOption(attrs);
clientOptions.addOptionGroup(attrToNodes);
// DEFINE ORDER
addOrder(LIST_ALL_ATTRS);
addOrder(NODESTOATTR);
addOrder(NODES);
addOrder(ATTRTONODES);
addOrder(ATTRIBUTES);
return clientOptions;
}
}
/**
* Admin commands handler.
*/
public static class AdminCommandHandler extends CommandHandler {
private static final String ADD = "add";
private static final String REMOVE = "remove";
private static final String REPLACE = "replace";
private static final String FAILUNKNOWNNODES = "failOnUnknownNodes";
AdminCommandHandler() {
super("Admin Commands:");
}
@Override
public Options buildOptions() {
Options adminOptions = new Options();
Option replace = new Option(REPLACE, true,
"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.");
replace.setArgName("\"node1:attribute[(type)][=value],attribute1[=value],"
+ "attribute2 node2:attribute2[=value],attribute3\"");
replace.setArgs(1);
adminOptions.addOption(replace);
Option add = new Option(ADD, true,
"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.");
add.setArgName("\"node1:attribute[(type)][=value],attribute1[=value],"
+ "attribute2 node2:attribute2[=value],attribute3\"");
add.setArgs(1);
adminOptions.addOption(add);
Option remove = new Option(REMOVE, true,
"Removes the specified node to attributes mapping"
+ " information at the ResourceManager");
remove.setArgName("\"node1:attribute,attribute1 node2:attribute2\"");
remove.setArgs(1);
adminOptions.addOption(remove);
adminOptions.addOption(new Option(FAILUNKNOWNNODES, false,
"Can be used optionally along with [add,remove,replace] options. "
+ "When set, command will fail if specified nodes are unknown."));
// DEFINE ORDER
addOrder(REPLACE);
addOrder(ADD);
addOrder(REMOVE);
addOrder(FAILUNKNOWNNODES);
return adminOptions;
} }
protected ResourceManagerAdministrationProtocol createAdminProtocol() protected ResourceManagerAdministrationProtocol createAdminProtocol()
throws IOException { throws IOException {
// Get the current configuration // Get the current configuration
final YarnConfiguration conf = new YarnConfiguration(getConf()); final YarnConfiguration conf = new YarnConfiguration(getConf());
return ClientRMProxy.createRMProxy(conf, return ClientRMProxy
ResourceManagerAdministrationProtocol.class); .createRMProxy(conf, ResourceManagerAdministrationProtocol.class);
} }
@Override public int handleCommand(CommandLine cliParser)
public void setConf(Configuration conf) { throws IOException, YarnException {
if (conf != null) { String operation = null;
conf = addSecurityConfiguration(conf); if (cliParser.hasOption(ADD)) {
operation = ADD;
} else if (cliParser.hasOption(REMOVE)) {
operation = REMOVE;
} else if (cliParser.hasOption(REPLACE)) {
operation = REPLACE;
} }
super.setConf(conf); if (operation != null) {
} List<NodeToAttributes> buildNodeLabelsListFromStr =
buildNodeLabelsListFromStr(cliParser.getOptionValue(operation),
/** !operation.equals(REPLACE), operation);
* Add the requisite security principal settings to the given Configuration, NodesToAttributesMappingRequest request =
* returning a copy. NodesToAttributesMappingRequest.newInstance(
* AttributeMappingOperationType.valueOf(operation.toUpperCase()),
* @param conf the original config buildNodeLabelsListFromStr,
* @return a copy with the security settings added cliParser.hasOption(FAILUNKNOWNNODES));
*/ ResourceManagerAdministrationProtocol adminProtocol =
private static Configuration addSecurityConfiguration(Configuration conf) { createAdminProtocol();
// Make a copy so we don't mutate it. Also use an YarnConfiguration to
// force loading of yarn-site.xml.
conf = new YarnConfiguration(conf);
conf.set(CommonConfigurationKeys.HADOOP_SECURITY_SERVICE_USER_NAME_KEY,
conf.get(YarnConfiguration.RM_PRINCIPAL, ""));
return conf;
}
@Override
public int run(String[] args) throws Exception {
if (args.length < 1) {
printUsage();
return -1;
}
int exitCode = -1;
int i = 0;
String cmd = args[i++];
if ("-help".equals(cmd)) {
exitCode = 0;
if (args.length >= 2) {
printHelpMsg(args[i]);
} else {
printHelpMsg("");
}
return exitCode;
}
try {
if ("-replace".equals(cmd)) {
exitCode = handleNodeAttributeMapping(args,
AttributeMappingOperationType.REPLACE);
} else if ("-add".equals(cmd)) {
exitCode =
handleNodeAttributeMapping(args, AttributeMappingOperationType.ADD);
} else if ("-remove".equals(cmd)) {
exitCode = handleNodeAttributeMapping(args,
AttributeMappingOperationType.REMOVE);
} else {
exitCode = -1;
errOut.println(cmd.substring(1) + ": Unknown command");
printUsage();
}
} catch (IllegalArgumentException arge) {
exitCode = -1;
errOut.println(cmd.substring(1) + ": " + arge.getLocalizedMessage());
printUsage(cmd);
} catch (RemoteException e) {
//
// This is a error returned by hadoop server. Print
// out the first line of the error message, ignore the stack trace.
exitCode = -1;
try {
String[] content;
content = e.getLocalizedMessage().split("\n");
errOut.println(cmd.substring(1) + ": " + content[0]);
} catch (Exception ex) {
errOut.println(cmd.substring(1) + ": " + ex.getLocalizedMessage());
}
} catch (Exception e) {
exitCode = -1;
errOut.println(cmd.substring(1) + ": " + e.getLocalizedMessage());
}
return exitCode;
}
private int handleNodeAttributeMapping(String args[],
AttributeMappingOperationType operation)
throws IOException, YarnException, ParseException {
Options opts = new Options();
opts.addOption(operation.name().toLowerCase(), true,
operation.name().toLowerCase());
opts.addOption("failOnUnknownNodes", false, "Fail on unknown nodes.");
int exitCode = -1;
CommandLine cliParser = null;
try {
cliParser = new GnuParser().parse(opts, args);
} catch (MissingArgumentException ex) {
errOut.println(NO_MAPPING_ERR_MSG);
printUsage(args[0]);
return exitCode;
}
List<NodeToAttributes> buildNodeLabelsMapFromStr =
buildNodeLabelsMapFromStr(
cliParser.getOptionValue(operation.name().toLowerCase()),
operation != AttributeMappingOperationType.REPLACE, operation);
NodesToAttributesMappingRequest request = NodesToAttributesMappingRequest
.newInstance(operation, buildNodeLabelsMapFromStr,
cliParser.hasOption("failOnUnknownNodes"));
ResourceManagerAdministrationProtocol adminProtocol = createAdminProtocol();
adminProtocol.mapAttributesToNodes(request); adminProtocol.mapAttributesToNodes(request);
} else {
// Handle case for only failOnUnknownNodes passed
throw new IllegalArgumentException(
getOptions().getOption(FAILUNKNOWNNODES).getDescription());
}
return 0; return 0;
} }
/** /**
* args are expected to be of the format * args are expected to be of the format
* node1:java(string)=8,ssd(boolean)=false node2:ssd(boolean)=true * node1:java(string)=8,ssd(boolean)=false node2:ssd(boolean)=true.
*/ */
private List<NodeToAttributes> buildNodeLabelsMapFromStr(String args, private List<NodeToAttributes> buildNodeLabelsListFromStr(String args,
boolean validateForAttributes, AttributeMappingOperationType operation) { boolean validateForAttributes, String operation) {
Map<String, NodeToAttributes> nodeToAttributesMap = new HashMap<>(); Map<String, NodeToAttributes> nodeToAttributesMap = new HashMap<>();
for (String nodeToAttributesStr : args.split("[ \n]")) { for (String nodeToAttributesStr : args.split("[ \n]")) {
// for each node to attribute mapping // for each node to attribute mapping
nodeToAttributesStr = nodeToAttributesStr.trim(); nodeToAttributesStr = nodeToAttributesStr.trim();
if (nodeToAttributesStr.isEmpty() if (nodeToAttributesStr.isEmpty() || nodeToAttributesStr
|| nodeToAttributesStr.startsWith("#")) { .startsWith("#")) {
continue; continue;
} }
if (nodeToAttributesStr.indexOf(":") == -1) { if (nodeToAttributesStr.indexOf(":") == -1) {
@ -337,7 +616,7 @@ public class NodeAttributesCLI extends Configured implements Tool {
String attributeName; String attributeName;
Set<String> attributeNamesMapped = new HashSet<>(); Set<String> attributeNamesMapped = new HashSet<>();
String attributesStr[]; String[] attributesStr;
if (nodeToAttributes.length == 2) { if (nodeToAttributes.length == 2) {
// fetching multiple attributes for a node // fetching multiple attributes for a node
attributesStr = nodeToAttributes[1].split(","); attributesStr = nodeToAttributes[1].split(",");
@ -346,11 +625,11 @@ public class NodeAttributesCLI extends Configured implements Tool {
attributeNameValueType = attributeStr.split("="); // to find name attributeNameValueType = attributeStr.split("="); // to find name
// value // value
Preconditions.checkArgument( Preconditions.checkArgument(
!(attributeNameValueType[0] == null !(attributeNameValueType[0] == null || attributeNameValueType[0]
|| attributeNameValueType[0].isEmpty()), .isEmpty()), "Attribute name cannot be null or empty");
"Attribute name cannot be null or empty"); attributeValue = attributeNameValueType.length > 1 ?
attributeValue = attributeNameValueType.length > 1 attributeNameValueType[1] :
? attributeNameValueType[1] : ""; "";
int indexOfOpenBracket = attributeNameValueType[0].indexOf("("); int indexOfOpenBracket = attributeNameValueType[0].indexOf("(");
if (indexOfOpenBracket == -1) { if (indexOfOpenBracket == -1) {
attributeName = attributeNameValueType[0]; attributeName = attributeNameValueType[0];
@ -387,15 +666,16 @@ public class NodeAttributesCLI extends Configured implements Tool {
// TODO when we support different type of attribute type we need to // TODO when we support different type of attribute type we need to
// cross verify whether input attributes itself is not violating // cross verify whether input attributes itself is not violating
// attribute Name to Type mapping. // attribute Name to Type mapping.
attributesList attributesList.add(NodeAttribute
.add(NodeAttribute.newInstance(NodeAttribute.PREFIX_CENTRALIZED, .newInstance(NodeAttribute.PREFIX_CENTRALIZED,
attributeName.trim(), attributeType, attributeValue.trim())); attributeName.trim(), attributeType,
attributeValue.trim()));
} }
} }
if (validateForAttributes) { if (validateForAttributes) {
Preconditions.checkArgument((attributesList.size() > 0), Preconditions.checkArgument((attributesList.size() > 0),
"Attributes cannot be null or empty for Operation " "Attributes cannot be null or empty for Operation [" + operation
+ operation.name() + " on the node " + node); + "] on the node " + node);
} }
nodeToAttributesMap nodeToAttributesMap
.put(node, NodeToAttributes.newInstance(node, attributesList)); .put(node, NodeToAttributes.newInstance(node, attributesList));
@ -407,8 +687,29 @@ public class NodeAttributesCLI extends Configured implements Tool {
return Lists.newArrayList(nodeToAttributesMap.values()); return Lists.newArrayList(nodeToAttributesMap.values());
} }
public static void main(String[] args) throws Exception { @Override
int result = ToolRunner.run(new NodeAttributesCLI(), args); public void setConf(Configuration conf) {
System.exit(result); if (conf != null) {
conf = addSecurityConfiguration(conf);
}
super.setConf(conf);
}
/**
* Add the requisite security principal settings to the given Configuration,
* returning a copy.
*
* @param conf the original config
* @return a copy with the security settings added
*/
private Configuration addSecurityConfiguration(Configuration conf) {
// Make a copy so we don't mutate it. Also use an YarnConfiguration to
// force loading of yarn-site.xml.
conf = new YarnConfiguration(conf);
conf.set(CommonConfigurationKeys.HADOOP_SECURITY_SERVICE_USER_NAME_KEY,
conf.get(YarnConfiguration.RM_PRINCIPAL, ""));
return conf;
}
} }
} }

View File

@ -44,7 +44,6 @@ import org.apache.hadoop.yarn.api.records.NodeId;
import org.apache.hadoop.yarn.api.records.NodeReport; import org.apache.hadoop.yarn.api.records.NodeReport;
import org.apache.hadoop.yarn.api.records.NodeState; import org.apache.hadoop.yarn.api.records.NodeState;
import org.apache.hadoop.yarn.exceptions.YarnException; import org.apache.hadoop.yarn.exceptions.YarnException;
import org.apache.hadoop.yarn.util.ConverterUtils;
@Private @Private
@Unstable @Unstable
@ -307,6 +306,18 @@ public class NodeCLI extends YarnCLI {
Collections.sort(nodeLabelsList); Collections.sort(nodeLabelsList);
nodeReportStr.println(StringUtils.join(nodeLabelsList.iterator(), ',')); nodeReportStr.println(StringUtils.join(nodeLabelsList.iterator(), ','));
if (nodeReport.getNodeAttributes().size() > 0) {
ArrayList nodeAtrs = new ArrayList<>(nodeReport.getNodeAttributes());
nodeReportStr.print("\tNode Attributes : ");
nodeReportStr.println(nodeAtrs.get(0).toString());
for (int index = 1; index < nodeAtrs.size(); index++) {
nodeReportStr.println(
String.format("\t%18s%s", "", nodeAtrs.get(index).toString()));
}
} else {
nodeReportStr.println("\tNode Attributes : ");
}
nodeReportStr.print("\tResource Utilization by Node : "); nodeReportStr.print("\tResource Utilization by Node : ");
if (nodeReport.getNodeUtilization() != null) { if (nodeReport.getNodeUtilization() != null) {
nodeReportStr.print("PMem:" nodeReportStr.print("PMem:"

View File

@ -18,6 +18,9 @@
package org.apache.hadoop.yarn.client.cli; package org.apache.hadoop.yarn.client.cli;
import org.apache.hadoop.yarn.api.records.NodeAttributeInfo;
import org.apache.hadoop.yarn.api.records.NodeAttributeKey;
import org.apache.hadoop.yarn.api.records.NodeAttributeType;
import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertEquals;
import static org.mockito.Mockito.mock; import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.spy; import static org.mockito.Mockito.spy;
@ -75,6 +78,31 @@ public class TestClusterCLI {
verify(sysOut).println(baos.toString("UTF-8")); verify(sysOut).println(baos.toString("UTF-8"));
} }
@Test
public void testGetClusterNodeAttributes() throws Exception {
YarnClient client = mock(YarnClient.class);
when(client.getClusterAttributes()).thenReturn(ImmutableSet
.of(NodeAttributeInfo.newInstance(NodeAttributeKey.newInstance("GPU"),
NodeAttributeType.STRING), NodeAttributeInfo
.newInstance(NodeAttributeKey.newInstance("CPU"),
NodeAttributeType.STRING)));
ClusterCLI cli = new ClusterCLI();
cli.setClient(client);
cli.setSysOutPrintStream(sysOut);
cli.setSysErrPrintStream(sysErr);
int rc = cli.run(new String[] {ClusterCLI.CMD,
"-" + ClusterCLI.LIST_CLUSTER_ATTRIBUTES});
assertEquals(0, rc);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
PrintWriter pw = new PrintWriter(baos);
pw.println("rm.yarn.io/GPU(STRING)");
pw.println("rm.yarn.io/CPU(STRING)");
pw.close();
verify(sysOut).println(baos.toString("UTF-8"));
}
@Test @Test
public void testGetClusterNodeLabelsWithLocalAccess() throws Exception { public void testGetClusterNodeLabelsWithLocalAccess() throws Exception {
YarnClient client = mock(YarnClient.class); YarnClient client = mock(YarnClient.class);
@ -157,6 +185,8 @@ public class TestClusterCLI {
pw.println(" option is UNSTABLE, could be"); pw.println(" option is UNSTABLE, could be");
pw.println(" removed in future releases."); pw.println(" removed in future releases.");
pw.println(" -h,--help Displays help for all commands."); pw.println(" -h,--help Displays help for all commands.");
pw.println(" -lna,--list-node-attributes List cluster node-attribute");
pw.println(" collection");
pw.println(" -lnl,--list-node-labels List cluster node-label"); pw.println(" -lnl,--list-node-labels List cluster node-label");
pw.println(" collection"); pw.println(" collection");
pw.close(); pw.close();

View File

@ -18,6 +18,20 @@
package org.apache.hadoop.yarn.client.cli; package org.apache.hadoop.yarn.client.cli;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import 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.Assert.assertTrue; import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail; import static org.junit.Assert.fail;
import static org.mockito.Matchers.any; import static org.mockito.Matchers.any;
@ -29,8 +43,8 @@ import java.io.IOException;
import java.io.PrintStream; import java.io.PrintStream;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
import java.util.Set;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.yarn.api.records.NodeAttribute; import org.apache.hadoop.yarn.api.records.NodeAttribute;
import org.apache.hadoop.yarn.api.records.NodeAttributeType; import org.apache.hadoop.yarn.api.records.NodeAttributeType;
import org.apache.hadoop.yarn.exceptions.YarnException; import org.apache.hadoop.yarn.exceptions.YarnException;
@ -56,34 +70,56 @@ public class TestNodeAttributesCLI {
private static final Logger LOG = private static final Logger LOG =
LoggerFactory.getLogger(TestNodeAttributesCLI.class); LoggerFactory.getLogger(TestNodeAttributesCLI.class);
private ResourceManagerAdministrationProtocol admin; private ResourceManagerAdministrationProtocol admin;
private NodesToAttributesMappingRequest request; private ApplicationClientProtocol client;
private NodesToAttributesMappingRequest nodeToAttrRequest;
private NodeAttributesCLI nodeAttributesCLI; private NodeAttributesCLI nodeAttributesCLI;
private ByteArrayOutputStream errOutBytes = new ByteArrayOutputStream(); private ByteArrayOutputStream errOutBytes = new ByteArrayOutputStream();
private ByteArrayOutputStream sysOutBytes = new ByteArrayOutputStream();
private String errOutput; private String errOutput;
private String sysOutput;
@Before @Before
public void configure() throws IOException, YarnException { public void configure() throws IOException, YarnException {
admin = mock(ResourceManagerAdministrationProtocol.class); admin = mock(ResourceManagerAdministrationProtocol.class);
client = mock(ApplicationClientProtocol.class);
when(admin.mapAttributesToNodes(any(NodesToAttributesMappingRequest.class))) when(admin.mapAttributesToNodes(any(NodesToAttributesMappingRequest.class)))
.thenAnswer(new Answer<NodesToAttributesMappingResponse>() { .thenAnswer(new Answer<NodesToAttributesMappingResponse>() {
@Override @Override
public NodesToAttributesMappingResponse answer( public NodesToAttributesMappingResponse answer(
InvocationOnMock invocation) throws Throwable { InvocationOnMock invocation) throws Throwable {
request = nodeToAttrRequest =
(NodesToAttributesMappingRequest) invocation.getArguments()[0]; (NodesToAttributesMappingRequest) invocation.getArguments()[0];
return NodesToAttributesMappingResponse.newInstance(); return NodesToAttributesMappingResponse.newInstance();
} }
}); });
nodeAttributesCLI = new NodeAttributesCLI(new Configuration()) { nodeAttributesCLI = new NodeAttributesCLI() {
@Override
protected AdminCommandHandler getAdminCommandHandler() {
return new AdminCommandHandler() {
@Override @Override
protected ResourceManagerAdministrationProtocol createAdminProtocol() protected ResourceManagerAdministrationProtocol createAdminProtocol()
throws IOException { throws IOException {
return admin; 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)); nodeAttributesCLI.setErrOut(new PrintStream(errOutBytes));
} }
@ -91,10 +127,9 @@ public class TestNodeAttributesCLI {
public void testHelp() throws Exception { public void testHelp() throws Exception {
String[] args = new String[] {"-help", "-replace"}; String[] args = new String[] {"-help", "-replace"};
assertTrue("It should have succeeded help for replace", 0 == runTool(args)); assertTrue("It should have succeeded help for replace", 0 == runTool(args));
assertOutputContains( assertErrorContains("-replace <\"node1:attribute[(type)][=value],attribute1"
"-replace <\"node1:attribute[(type)][=value],attribute1" + "[=value],attribute2 node2:attribute2[=value],attribute3\">");
+ "[=value],attribute2 node2:attribute2[=value],attribute3\"> :"); assertErrorContains("Replace the node to attributes mapping information at"
assertOutputContains("Replace the node to attributes mapping information at"
+ " the ResourceManager with the new mapping. Currently supported" + " the ResourceManager with the new mapping. Currently supported"
+ " attribute type. And string is the default type too. Attribute value" + " attribute type. And string is the default type too. Attribute value"
+ " if not specified for string type value will be considered as empty" + " if not specified for string type value will be considered as empty"
@ -103,17 +138,17 @@ public class TestNodeAttributesCLI {
args = new String[] {"-help", "-remove"}; args = new String[] {"-help", "-remove"};
assertTrue("It should have succeeded help for replace", 0 == runTool(args)); assertTrue("It should have succeeded help for replace", 0 == runTool(args));
assertOutputContains( assertErrorContains(
"-remove <\"node1:attribute,attribute1" + " node2:attribute2\"> :"); "-remove <\"node1:attribute,attribute1" + " node2:attribute2\">");
assertOutputContains("Removes the specified node to attributes mapping" assertErrorContains("Removes the specified node to attributes mapping"
+ " information at the ResourceManager"); + " information at the ResourceManager");
args = new String[] {"-help", "-add"}; args = new String[] {"-help", "-add"};
assertTrue("It should have succeeded help for replace", 0 == runTool(args)); assertTrue("It should have succeeded help for replace", 0 == runTool(args));
assertOutputContains("-add <\"node1:attribute[(type)][=value]," assertErrorContains("-add <\"node1:attribute[(type)][=value],"
+ "attribute1[=value],attribute2 node2:attribute2[=value],attribute3\">" + "attribute1[=value],attribute2 node2:attribute2[=value],"
+ " :"); + "attribute3\">");
assertOutputContains("Adds or updates the node to attributes mapping" assertErrorContains("Adds or updates the node to attributes mapping"
+ " information at the ResourceManager. Currently supported attribute" + " information at the ResourceManager. Currently supported attribute"
+ " type is string. And string is the default type too. Attribute value" + " type is string. And string is the default type too. Attribute value"
+ " if not specified for string type value will be considered as empty" + " if not specified for string type value will be considered as empty"
@ -122,9 +157,35 @@ public class TestNodeAttributesCLI {
args = new String[] {"-help", "-failOnUnknownNodes"}; args = new String[] {"-help", "-failOnUnknownNodes"};
assertTrue("It should have succeeded help for replace", 0 == runTool(args)); assertTrue("It should have succeeded help for replace", 0 == runTool(args));
assertOutputContains("-failOnUnknownNodes :"); assertErrorContains("-failOnUnknownNodes");
assertOutputContains("Can be used optionally along with other options. When" assertErrorContains("Can be used optionally along with [add,remove,"
+ " its set, it will fail if specified nodes are unknown."); + "replace] options. When set, command will fail if specified nodes "
+ "are unknown.");
args = new String[] {"-help", "-list"};
assertTrue("It should have succeeded help for replace", 0 == runTool(args));
assertErrorContains("-list");
assertErrorContains("List all attributes in cluster");
args = new String[] {"-help", "-nodes"};
assertTrue("It should have succeeded help for replace", 0 == runTool(args));
assertErrorContains("-nodes");
assertErrorContains(
"Works with [list] to specify node hostnames whose mappings "
+ "are required to be displayed.");
args = new String[] {"-help", "-attributes"};
assertTrue("It should have succeeded help for replace", 0 == runTool(args));
assertErrorContains("-attributes");
assertErrorContains(
"Works with [attributestonodes] to specify attributes whose mapping "
+ "are required to be displayed.");
args = new String[] {"-help", "-attributestonodes"};
assertTrue("It should have succeeded help for replace", 0 == runTool(args));
assertErrorContains("-attributestonodes");
assertErrorContains("Displays mapping of attributes to nodes and attribute "
+ "values grouped by attributes");
} }
@Test @Test
@ -167,14 +228,14 @@ public class TestNodeAttributesCLI {
args = new String[] {"-replace"}; args = new String[] {"-replace"};
assertTrue("Should fail as no attribute mappings specified", assertTrue("Should fail as no attribute mappings specified",
0 != runTool(args)); 0 != runTool(args));
assertFailureMessageContains(NodeAttributesCLI.NO_MAPPING_ERR_MSG); assertFailureMessageContains(NodeAttributesCLI.MISSING_ARGUMENT);
// no labels, should fail // no labels, should fail
args = new String[] {"-replace", "-failOnUnknownNodes", args = new String[] {"-replace", "-failOnUnknownNodes",
"x:key(string)=value,key2=val2"}; "x:key(string)=value,key2=val2"};
assertTrue("Should fail as no attribute mappings specified for replace", assertTrue("Should fail as no attribute mappings specified for replace",
0 != runTool(args)); 0 != runTool(args));
assertFailureMessageContains(NodeAttributesCLI.NO_MAPPING_ERR_MSG); assertFailureMessageContains(NodeAttributesCLI.MISSING_ARGUMENT);
// no labels, should fail // no labels, should fail
args = new String[] {"-replace", " "}; args = new String[] {"-replace", " "};
@ -221,10 +282,10 @@ public class TestNodeAttributesCLI {
.add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, "")); .add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, ""));
nodeAttributesList.add(NodeToAttributes.newInstance("z", attributes)); nodeAttributesList.add(NodeToAttributes.newInstance("z", attributes));
NodesToAttributesMappingRequest expected = NodesToAttributesMappingRequest expected = NodesToAttributesMappingRequest
NodesToAttributesMappingRequest.newInstance( .newInstance(AttributeMappingOperationType.REPLACE, nodeAttributesList,
AttributeMappingOperationType.REPLACE, nodeAttributesList, false); false);
assertTrue(request.equals(expected)); assertTrue(nodeToAttrRequest.equals(expected));
} }
@Test @Test
@ -237,7 +298,8 @@ public class TestNodeAttributesCLI {
assertTrue("It should have failed as no node is specified", assertTrue("It should have failed as no node is specified",
0 != runTool(args)); 0 != runTool(args));
assertFailureMessageContains( assertFailureMessageContains(
"Attributes cannot be null or empty for Operation REMOVE on the node x"); "Attributes cannot be null or empty for Operation [remove] on the "
+ "node x");
// -------------------------------- // --------------------------------
// success scenarios // success scenarios
// -------------------------------- // --------------------------------
@ -259,10 +321,10 @@ public class TestNodeAttributesCLI {
.add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, "")); .add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, ""));
nodeAttributesList.add(NodeToAttributes.newInstance("z", attributes)); nodeAttributesList.add(NodeToAttributes.newInstance("z", attributes));
NodesToAttributesMappingRequest expected = NodesToAttributesMappingRequest expected = NodesToAttributesMappingRequest
NodesToAttributesMappingRequest.newInstance( .newInstance(AttributeMappingOperationType.REMOVE, nodeAttributesList,
AttributeMappingOperationType.REMOVE, nodeAttributesList, true); true);
assertTrue(request.equals(expected)); assertTrue(nodeToAttrRequest.equals(expected));
} }
@Test @Test
@ -275,7 +337,7 @@ public class TestNodeAttributesCLI {
assertTrue("It should have failed as no node is specified", assertTrue("It should have failed as no node is specified",
0 != runTool(args)); 0 != runTool(args));
assertFailureMessageContains( assertFailureMessageContains(
"Attributes cannot be null or empty for Operation ADD on the node x"); "Attributes cannot be null or empty for Operation [add] on the node x");
// -------------------------------- // --------------------------------
// success scenarios // success scenarios
// -------------------------------- // --------------------------------
@ -297,10 +359,10 @@ public class TestNodeAttributesCLI {
.add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, "")); .add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, ""));
nodeAttributesList.add(NodeToAttributes.newInstance("z", attributes)); nodeAttributesList.add(NodeToAttributes.newInstance("z", attributes));
NodesToAttributesMappingRequest expected = NodesToAttributesMappingRequest expected = NodesToAttributesMappingRequest
NodesToAttributesMappingRequest.newInstance( .newInstance(AttributeMappingOperationType.ADD, nodeAttributesList,
AttributeMappingOperationType.ADD, nodeAttributesList, true); true);
assertTrue(request.equals(expected)); assertTrue(nodeToAttrRequest.equals(expected));
// -------------------------------- // --------------------------------
// with Duplicate mappings for a host // with Duplicate mappings for a host
@ -315,32 +377,161 @@ public class TestNodeAttributesCLI {
.add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, "")); .add(NodeAttribute.newInstance("key4", NodeAttributeType.STRING, ""));
nodeAttributesList.add(NodeToAttributes.newInstance("x", attributes)); nodeAttributesList.add(NodeToAttributes.newInstance("x", attributes));
expected = expected = NodesToAttributesMappingRequest
NodesToAttributesMappingRequest.newInstance( .newInstance(AttributeMappingOperationType.ADD, nodeAttributesList,
AttributeMappingOperationType.ADD, nodeAttributesList, true); true);
assertTrue(request.equals(expected)); 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("It should be success since it list all attributes",
0 == runTool(args));
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("It should not success since nodes are not specified",
0 != runTool(args));
assertErrorContains(NodeAttributesCLI.INVALID_COMMAND_USAGE);
// Missing argument for nodes
args = new String[] {"-nodestoattributes", "-nodes"};
assertTrue("It should not success since nodes are not specified",
0 != runTool(args));
assertErrorContains(NodeAttributesCLI.MISSING_ARGUMENT);
// --------------------------------
// Success with hostname param
// --------------------------------
args = new String[] {"-nodestoattributes", "-nodes", "hostname"};
assertTrue("Should return hostname to attributed list", 0 == runTool(args));
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("It should be success since it list all attributes",
0 == runTool(args));
assertSysOutContains("Hostname\tAttribute-value", "rm.yarn.io/GPU :",
"host1\t ARM");
// --------------------------------
// fail scenario argument filter missing
// --------------------------------
args = new String[] {"-attributestonodes", "-attributes"};
assertTrue(
"It should not success since attributes for filter are not specified",
0 != runTool(args));
assertErrorContains(NodeAttributesCLI.MISSING_ARGUMENT);
// --------------------------------
// fail scenario argument filter missing
// --------------------------------
args = new String[] {"-attributestonodes", "-attributes", "fail/da/fail"};
assertTrue("It should not success since attributes format is not correct",
0 != runTool(args));
assertErrorContains(
"Attribute format not correct. Should be <[prefix]/[name]> "
+ ":fail/da/fail");
} }
private void assertFailureMessageContains(String... messages) { private void assertFailureMessageContains(String... messages) {
assertOutputContains(messages); assertErrorContains(messages);
assertOutputContains(NodeAttributesCLI.USAGE_YARN_NODE_ATTRIBUTES); assertErrorContains(NodeAttributesCLI.USAGE_YARN_NODE_ATTRIBUTES);
} }
private void assertOutputContains(String... messages) { private void assertErrorContains(String... messages) {
for (String message : messages) { for (String message : messages) {
if (!errOutput.contains(message)) { if (!errOutput.contains(message)) {
fail("Expected output to contain '" + message fail(
+ "' but err_output was:\n" + errOutput); "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 { private int runTool(String... args) throws Exception {
errOutBytes.reset(); errOutBytes.reset();
sysOutBytes.reset();
LOG.info("Running: NodeAttributesCLI " + Joiner.on(" ").join(args)); LOG.info("Running: NodeAttributesCLI " + Joiner.on(" ").join(args));
int ret = nodeAttributesCLI.run(args); int ret = nodeAttributesCLI.run(args);
errOutput = new String(errOutBytes.toByteArray(), Charsets.UTF_8); errOutput = new String(errOutBytes.toByteArray(), Charsets.UTF_8);
sysOutput = new String(sysOutBytes.toByteArray(), Charsets.UTF_8);
LOG.info("Err_output:\n" + errOutput); LOG.info("Err_output:\n" + errOutput);
LOG.info("Sys_output:\n" + sysOutput);
return ret; return ret;
} }
} }

View File

@ -17,6 +17,8 @@
*/ */
package org.apache.hadoop.yarn.client.cli; package org.apache.hadoop.yarn.client.cli;
import org.apache.hadoop.yarn.api.records.NodeAttribute;
import org.apache.hadoop.yarn.api.records.NodeAttributeType;
import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertEquals;
import static org.mockito.Matchers.any; import static org.mockito.Matchers.any;
import static org.mockito.Matchers.anyInt; import static org.mockito.Matchers.anyInt;
@ -1544,8 +1546,8 @@ public class TestYarnCLI {
public void testNodeStatus() throws Exception { public void testNodeStatus() throws Exception {
NodeId nodeId = NodeId.newInstance("host0", 0); NodeId nodeId = NodeId.newInstance("host0", 0);
NodeCLI cli = new NodeCLI(); NodeCLI cli = new NodeCLI();
when(client.getNodeReports()).thenReturn( when(client.getNodeReports())
getNodeReports(3, NodeState.RUNNING, false)); .thenReturn(getNodeReports(3, NodeState.RUNNING, false, false, false));
cli.setClient(client); cli.setClient(client);
cli.setSysOutPrintStream(sysOut); cli.setSysOutPrintStream(sysOut);
cli.setSysErrPrintStream(sysErr); cli.setSysErrPrintStream(sysErr);
@ -1568,6 +1570,8 @@ public class TestYarnCLI {
pw.println("\tCPU-Used : 0 vcores"); pw.println("\tCPU-Used : 0 vcores");
pw.println("\tCPU-Capacity : 0 vcores"); pw.println("\tCPU-Capacity : 0 vcores");
pw.println("\tNode-Labels : a,b,c,x,y,z"); pw.println("\tNode-Labels : a,b,c,x,y,z");
pw.println("\tNode Attributes : rm.yarn.io/GPU(STRING)=ARM");
pw.println("\t rm.yarn.io/CPU(STRING)=ARM");
pw.println("\tResource Utilization by Node : PMem:2048 MB, VMem:4096 MB, VCores:8.0"); pw.println("\tResource Utilization by Node : PMem:2048 MB, VMem:4096 MB, VCores:8.0");
pw.println("\tResource Utilization by Containers : PMem:1024 MB, VMem:2048 MB, VCores:4.0"); pw.println("\tResource Utilization by Containers : PMem:1024 MB, VMem:2048 MB, VCores:4.0");
pw.close(); pw.close();
@ -1604,6 +1608,7 @@ public class TestYarnCLI {
pw.println("\tCPU-Used : 0 vcores"); pw.println("\tCPU-Used : 0 vcores");
pw.println("\tCPU-Capacity : 0 vcores"); pw.println("\tCPU-Capacity : 0 vcores");
pw.println("\tNode-Labels : "); pw.println("\tNode-Labels : ");
pw.println("\tNode Attributes : ");
pw.println("\tResource Utilization by Node : PMem:2048 MB, VMem:4096 MB, VCores:8.0"); pw.println("\tResource Utilization by Node : PMem:2048 MB, VMem:4096 MB, VCores:8.0");
pw.println("\tResource Utilization by Containers : PMem:1024 MB, VMem:2048 MB, VCores:4.0"); pw.println("\tResource Utilization by Containers : PMem:1024 MB, VMem:2048 MB, VCores:4.0");
pw.close(); pw.close();
@ -1616,8 +1621,8 @@ public class TestYarnCLI {
public void testNodeStatusWithEmptyResourceUtilization() throws Exception { public void testNodeStatusWithEmptyResourceUtilization() throws Exception {
NodeId nodeId = NodeId.newInstance("host0", 0); NodeId nodeId = NodeId.newInstance("host0", 0);
NodeCLI cli = new NodeCLI(); NodeCLI cli = new NodeCLI();
when(client.getNodeReports()).thenReturn( when(client.getNodeReports())
getNodeReports(3, NodeState.RUNNING, false, true)); .thenReturn(getNodeReports(3, NodeState.RUNNING, false, true, true));
cli.setClient(client); cli.setClient(client);
cli.setSysOutPrintStream(sysOut); cli.setSysOutPrintStream(sysOut);
cli.setSysErrPrintStream(sysErr); cli.setSysErrPrintStream(sysErr);
@ -1640,6 +1645,7 @@ public class TestYarnCLI {
pw.println("\tCPU-Used : 0 vcores"); pw.println("\tCPU-Used : 0 vcores");
pw.println("\tCPU-Capacity : 0 vcores"); pw.println("\tCPU-Capacity : 0 vcores");
pw.println("\tNode-Labels : a,b,c,x,y,z"); pw.println("\tNode-Labels : a,b,c,x,y,z");
pw.println("\tNode Attributes : ");
pw.println("\tResource Utilization by Node : "); pw.println("\tResource Utilization by Node : ");
pw.println("\tResource Utilization by Containers : "); pw.println("\tResource Utilization by Containers : ");
pw.close(); pw.close();
@ -2051,16 +2057,18 @@ public class TestYarnCLI {
} }
private List<NodeReport> getNodeReports(int noOfNodes, NodeState state) { private List<NodeReport> getNodeReports(int noOfNodes, NodeState state) {
return getNodeReports(noOfNodes, state, true, false); return getNodeReports(noOfNodes, state, true, false, true);
} }
private List<NodeReport> getNodeReports(int noOfNodes, NodeState state, private List<NodeReport> getNodeReports(int noOfNodes, NodeState state,
boolean emptyNodeLabel) { boolean emptyNodeLabel, boolean emptyAttributes) {
return getNodeReports(noOfNodes, state, emptyNodeLabel, false); return getNodeReports(noOfNodes, state, emptyNodeLabel, false,
emptyAttributes);
} }
private List<NodeReport> getNodeReports(int noOfNodes, NodeState state, private List<NodeReport> getNodeReports(int noOfNodes, NodeState state,
boolean emptyNodeLabel, boolean emptyResourceUtilization) { boolean emptyNodeLabel, boolean emptyResourceUtilization,
boolean emptyAttributes) {
List<NodeReport> nodeReports = new ArrayList<NodeReport>(); List<NodeReport> nodeReports = new ArrayList<NodeReport>();
for (int i = 0; i < noOfNodes; i++) { for (int i = 0; i < noOfNodes; i++) {
@ -2082,6 +2090,11 @@ public class TestYarnCLI {
nodeReport.setAggregatedContainersUtilization(containersUtilization); nodeReport.setAggregatedContainersUtilization(containersUtilization);
nodeReport.setNodeUtilization(nodeUtilization); nodeReport.setNodeUtilization(nodeUtilization);
} }
if (!emptyAttributes) {
nodeReport.setNodeAttributes(ImmutableSet.of(NodeAttribute
.newInstance("GPU", NodeAttributeType.STRING, "ARM"),
NodeAttribute.newInstance("CPU", NodeAttributeType.STRING, "ARM")));
}
nodeReports.add(nodeReport); nodeReports.add(nodeReport);
} }
return nodeReports; return nodeReports;

View File

@ -130,14 +130,18 @@ public class NodeAttributeInfoPBImpl extends NodeAttributeInfo {
} }
if (obj instanceof NodeAttributeInfo) { if (obj instanceof NodeAttributeInfo) {
NodeAttributeInfo other = (NodeAttributeInfo) obj; NodeAttributeInfo other = (NodeAttributeInfo) obj;
getAttributeKey().equals(other.getAttributeKey()); return getAttributeKey().equals(other.getAttributeKey());
return true;
} }
return false; return false;
} }
@Override @Override
public String toString() { public String toString() {
return getAttributeKey().toString() + ":Type-" + getAttributeType(); StringBuilder strBuilder = new StringBuilder();
NodeAttributeKey key = this.getAttributeKey();
strBuilder.append(key.getAttributePrefix()).append("/")
.append(key.getAttributeName()).append("(")
.append(this.getAttributeType()).append(")");
return strBuilder.toString();
} }
} }

View File

@ -152,15 +152,19 @@ public class NodeAttributePBImpl extends NodeAttribute {
} }
if (obj instanceof NodeAttribute) { if (obj instanceof NodeAttribute) {
NodeAttribute other = (NodeAttribute) obj; NodeAttribute other = (NodeAttribute) obj;
getAttributeKey().equals(other.getAttributeKey()); return getAttributeKey().equals(other.getAttributeKey());
return true;
} }
return false; return false;
} }
@Override @Override
public String toString() { public String toString() {
return getAttributeKey().toString() + ":Value-" + getAttributeValue() StringBuilder strBuilder = new StringBuilder();
+ ":Type-" + getAttributeType(); NodeAttributeKey key = this.getAttributeKey();
strBuilder.append(key.getAttributePrefix()).append("/")
.append(key.getAttributeName()).append("(")
.append(this.getAttributeType()).append(")=")
.append(this.getAttributeValue());
return strBuilder.toString();
} }
} }

View File

@ -18,17 +18,21 @@
package org.apache.hadoop.yarn.api.records.impl.pb; package org.apache.hadoop.yarn.api.records.impl.pb;
import java.util.ArrayList;
import java.util.HashSet; import java.util.HashSet;
import java.util.List;
import java.util.Set; import java.util.Set;
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;
import org.apache.hadoop.yarn.api.records.NodeAttribute;
import org.apache.hadoop.yarn.api.records.NodeId; import org.apache.hadoop.yarn.api.records.NodeId;
import org.apache.hadoop.yarn.api.records.NodeReport; import org.apache.hadoop.yarn.api.records.NodeReport;
import org.apache.hadoop.yarn.api.records.NodeState; import org.apache.hadoop.yarn.api.records.NodeState;
import org.apache.hadoop.yarn.api.records.NodeUpdateType; import org.apache.hadoop.yarn.api.records.NodeUpdateType;
import org.apache.hadoop.yarn.api.records.Resource; import org.apache.hadoop.yarn.api.records.Resource;
import org.apache.hadoop.yarn.api.records.ResourceUtilization; import org.apache.hadoop.yarn.api.records.ResourceUtilization;
import org.apache.hadoop.yarn.proto.YarnProtos.NodeAttributeProto;
import org.apache.hadoop.yarn.proto.YarnProtos.NodeIdProto; import org.apache.hadoop.yarn.proto.YarnProtos.NodeIdProto;
import org.apache.hadoop.yarn.proto.YarnProtos.NodeReportProto; import org.apache.hadoop.yarn.proto.YarnProtos.NodeReportProto;
import org.apache.hadoop.yarn.proto.YarnProtos.NodeReportProtoOrBuilder; import org.apache.hadoop.yarn.proto.YarnProtos.NodeReportProtoOrBuilder;
@ -50,6 +54,7 @@ public class NodeReportPBImpl extends NodeReport {
private ResourceUtilization containersUtilization = null; private ResourceUtilization containersUtilization = null;
private ResourceUtilization nodeUtilization = null; private ResourceUtilization nodeUtilization = null;
Set<String> labels; Set<String> labels;
private Set<NodeAttribute> nodeAttributes;
public NodeReportPBImpl() { public NodeReportPBImpl() {
builder = NodeReportProto.newBuilder(); builder = NodeReportProto.newBuilder();
@ -268,6 +273,14 @@ public class NodeReportPBImpl extends NodeReport {
builder.clearNodeLabels(); builder.clearNodeLabels();
builder.addAllNodeLabels(this.labels); builder.addAllNodeLabels(this.labels);
} }
if (this.nodeAttributes != null) {
builder.clearNodeAttributes();
List<NodeAttributeProto> attrList = new ArrayList<>();
for (NodeAttribute attr : this.nodeAttributes) {
attrList.add(convertToProtoFormat(attr));
}
builder.addAllNodeAttributes(attrList);
}
if (this.nodeUtilization != null if (this.nodeUtilization != null
&& !((ResourceUtilizationPBImpl) this.nodeUtilization).getProto() && !((ResourceUtilizationPBImpl) this.nodeUtilization).getProto()
.equals(builder.getNodeUtilization())) { .equals(builder.getNodeUtilization())) {
@ -307,6 +320,15 @@ public class NodeReportPBImpl extends NodeReport {
return ((NodeIdPBImpl) nodeId).getProto(); return ((NodeIdPBImpl) nodeId).getProto();
} }
private NodeAttributeProto convertToProtoFormat(NodeAttribute nodeAttr) {
return ((NodeAttributePBImpl) nodeAttr).getProto();
}
private NodeAttributePBImpl convertFromProtoFormat(
NodeAttributeProto nodeAttr) {
return new NodeAttributePBImpl(nodeAttr);
}
private ResourcePBImpl convertFromProtoFormat(ResourceProto p) { private ResourcePBImpl convertFromProtoFormat(ResourceProto p) {
return new ResourcePBImpl(p); return new ResourcePBImpl(p);
} }
@ -427,4 +449,24 @@ public class NodeReportPBImpl extends NodeReport {
} }
builder.setNodeUpdateType(ProtoUtils.convertToProtoFormat(nodeUpdateType)); builder.setNodeUpdateType(ProtoUtils.convertToProtoFormat(nodeUpdateType));
} }
@Override
public void setNodeAttributes(Set<NodeAttribute> nodeAttrs) {
maybeInitBuilder();
builder.clearNodeAttributes();
this.nodeAttributes = nodeAttrs;
}
@Override
public Set<NodeAttribute> getNodeAttributes() {
if (nodeAttributes != null) {
return nodeAttributes;
}
NodeReportProtoOrBuilder p = viaProto ? proto : builder;
this.nodeAttributes = new HashSet<>();
for (NodeAttributeProto nattrProto : p.getNodeAttributesList()) {
nodeAttributes.add(convertFromProtoFormat(nattrProto));
}
return nodeAttributes;
}
} }

View File

@ -54,6 +54,7 @@ import org.apache.hadoop.yarn.api.records.FinalApplicationStatus;
import org.apache.hadoop.yarn.api.records.LocalResource; import org.apache.hadoop.yarn.api.records.LocalResource;
import org.apache.hadoop.yarn.api.records.LocalResourceType; import org.apache.hadoop.yarn.api.records.LocalResourceType;
import org.apache.hadoop.yarn.api.records.LocalResourceVisibility; import org.apache.hadoop.yarn.api.records.LocalResourceVisibility;
import org.apache.hadoop.yarn.api.records.NodeAttribute;
import org.apache.hadoop.yarn.api.records.NodeId; import org.apache.hadoop.yarn.api.records.NodeId;
import org.apache.hadoop.yarn.api.records.NodeReport; import org.apache.hadoop.yarn.api.records.NodeReport;
import org.apache.hadoop.yarn.api.records.NodeState; import org.apache.hadoop.yarn.api.records.NodeState;
@ -199,7 +200,7 @@ public class BuilderUtils {
NodeUpdateType nodeUpdateType) { NodeUpdateType nodeUpdateType) {
return newNodeReport(nodeId, nodeState, httpAddress, rackName, used, return newNodeReport(nodeId, nodeState, httpAddress, rackName, used,
capability, numContainers, healthReport, lastHealthReportTime, capability, numContainers, healthReport, lastHealthReportTime,
nodeLabels, null, null, decommissioningTimeout, nodeUpdateType); nodeLabels, null, null, decommissioningTimeout, nodeUpdateType, null);
} }
public static NodeReport newNodeReport(NodeId nodeId, NodeState nodeState, public static NodeReport newNodeReport(NodeId nodeId, NodeState nodeState,
@ -207,7 +208,7 @@ public class BuilderUtils {
int numContainers, String healthReport, long lastHealthReportTime, int numContainers, String healthReport, long lastHealthReportTime,
Set<String> nodeLabels, ResourceUtilization containersUtilization, Set<String> nodeLabels, ResourceUtilization containersUtilization,
ResourceUtilization nodeUtilization, Integer decommissioningTimeout, ResourceUtilization nodeUtilization, Integer decommissioningTimeout,
NodeUpdateType nodeUpdateType) { NodeUpdateType nodeUpdateType, Set<NodeAttribute> attrs) {
NodeReport nodeReport = recordFactory.newRecordInstance(NodeReport.class); NodeReport nodeReport = recordFactory.newRecordInstance(NodeReport.class);
nodeReport.setNodeId(nodeId); nodeReport.setNodeId(nodeId);
nodeReport.setNodeState(nodeState); nodeReport.setNodeState(nodeState);
@ -223,6 +224,7 @@ public class BuilderUtils {
nodeReport.setNodeUtilization(nodeUtilization); nodeReport.setNodeUtilization(nodeUtilization);
nodeReport.setDecommissioningTimeout(decommissioningTimeout); nodeReport.setDecommissioningTimeout(decommissioningTimeout);
nodeReport.setNodeUpdateType(nodeUpdateType); nodeReport.setNodeUpdateType(nodeUpdateType);
nodeReport.setNodeAttributes(attrs);
return nodeReport; return nodeReport;
} }

View File

@ -983,12 +983,11 @@ public class AdminService extends CompositeService implements
List<NodeToAttributes> nodesToAttributes = request.getNodesToAttributes(); List<NodeToAttributes> nodesToAttributes = request.getNodesToAttributes();
boolean failOnUnknownNodes = request.getFailOnUnknownNodes(); boolean failOnUnknownNodes = request.getFailOnUnknownNodes();
Map<String, Set<NodeAttribute>> nodeAttributeMapping =
validateAndFetch(nodesToAttributes, failOnUnknownNodes);
NodeAttributesManager nodeAttributesManager = NodeAttributesManager nodeAttributesManager =
rm.getRMContext().getNodeAttributesManager(); rm.getRMContext().getNodeAttributesManager();
try { try {
Map<String, Set<NodeAttribute>> nodeAttributeMapping =
validateAndFetch(nodesToAttributes, failOnUnknownNodes);
switch (request.getOperation()) { switch (request.getOperation()) {
case ADD: case ADD:
nodeAttributesManager.addNodeAttributes(nodeAttributeMapping); nodeAttributesManager.addNodeAttributes(nodeAttributeMapping);

View File

@ -1052,6 +1052,7 @@ public class ClientRMService extends AbstractService implements
numContainers = schedulerNodeReport.getNumContainers(); numContainers = schedulerNodeReport.getNumContainers();
} }
Set<NodeAttribute> attrs = rmNode.getAllNodeAttributes();
NodeReport report = NodeReport report =
BuilderUtils.newNodeReport(rmNode.getNodeID(), rmNode.getState(), BuilderUtils.newNodeReport(rmNode.getNodeID(), rmNode.getState(),
rmNode.getHttpAddress(), rmNode.getRackName(), used, rmNode.getHttpAddress(), rmNode.getRackName(), used,
@ -1059,7 +1060,7 @@ public class ClientRMService extends AbstractService implements
rmNode.getHealthReport(), rmNode.getLastHealthReportTime(), rmNode.getHealthReport(), rmNode.getLastHealthReportTime(),
rmNode.getNodeLabels(), rmNode.getAggregatedContainersUtilization(), rmNode.getNodeLabels(), rmNode.getAggregatedContainersUtilization(),
rmNode.getNodeUtilization(), rmNode.getDecommissioningTimeout(), rmNode.getNodeUtilization(), rmNode.getDecommissioningTimeout(),
null); null, attrs);
return report; return report;
} }

View File

@ -673,10 +673,6 @@ public class ResourceTrackerService extends AbstractService implements
this.rmContext.getNodeAttributesManager() this.rmContext.getNodeAttributesManager()
.replaceNodeAttributes(NodeAttribute.PREFIX_DISTRIBUTED, .replaceNodeAttributes(NodeAttribute.PREFIX_DISTRIBUTED,
ImmutableMap.of(nodeId.getHost(), nodeAttributes)); ImmutableMap.of(nodeId.getHost(), nodeAttributes));
// Update node attributes to RMNode
rmNode.setNodeAttributes(NodeAttribute.PREFIX_DISTRIBUTED,
nodeAttributes);
} }
} }

View File

@ -198,14 +198,7 @@ public interface RMNode {
RMContext getRMContext(); RMContext getRMContext();
/** /**
* Sets node attributes per prefix. * @return all node attributes as a Set.
* @param prefix node attribute prefix
* @param nodeAttributes node attributes
*/ */
void setNodeAttributes(String prefix, Set<NodeAttribute> nodeAttributes); Set<NodeAttribute> getAllNodeAttributes();
/**
* @return all node attributes grouped by their prefix as a map.
*/
Map<String, Set<NodeAttribute>> getAllNodeAttributes();
} }

View File

@ -59,7 +59,9 @@ import org.apache.hadoop.yarn.api.records.ResourceUtilization;
import org.apache.hadoop.yarn.event.EventHandler; import org.apache.hadoop.yarn.event.EventHandler;
import org.apache.hadoop.yarn.factories.RecordFactory; import org.apache.hadoop.yarn.factories.RecordFactory;
import org.apache.hadoop.yarn.factory.providers.RecordFactoryProvider; import org.apache.hadoop.yarn.factory.providers.RecordFactoryProvider;
import org.apache.hadoop.yarn.nodelabels.AttributeValue;
import org.apache.hadoop.yarn.nodelabels.CommonNodeLabelsManager; import org.apache.hadoop.yarn.nodelabels.CommonNodeLabelsManager;
import org.apache.hadoop.yarn.nodelabels.NodeAttributesManager;
import org.apache.hadoop.yarn.server.api.protocolrecords.LogAggregationReport; import org.apache.hadoop.yarn.server.api.protocolrecords.LogAggregationReport;
import org.apache.hadoop.yarn.server.api.protocolrecords.NMContainerStatus; import org.apache.hadoop.yarn.server.api.protocolrecords.NMContainerStatus;
import org.apache.hadoop.yarn.server.api.protocolrecords.NodeHeartbeatResponse; import org.apache.hadoop.yarn.server.api.protocolrecords.NodeHeartbeatResponse;
@ -186,9 +188,6 @@ public class RMNodeImpl implements RMNode, EventHandler<RMNodeEvent> {
private NodeHeartbeatResponse latestNodeHeartBeatResponse = recordFactory private NodeHeartbeatResponse latestNodeHeartBeatResponse = recordFactory
.newRecordInstance(NodeHeartbeatResponse.class); .newRecordInstance(NodeHeartbeatResponse.class);
// Node attributes, store by prefix
private Map<String, Set<NodeAttribute>> nodeAttributes = new HashMap<>();
private static final StateMachineFactory<RMNodeImpl, private static final StateMachineFactory<RMNodeImpl,
NodeState, NodeState,
RMNodeEventType, RMNodeEventType,
@ -1552,13 +1551,10 @@ public class RMNodeImpl implements RMNode, EventHandler<RMNodeEvent> {
} }
@Override @Override
public void setNodeAttributes(String prefix, public Set<NodeAttribute> getAllNodeAttributes() {
Set<NodeAttribute> nodeAttributeSet) { NodeAttributesManager attrMgr = context.getNodeAttributesManager();
this.nodeAttributes.put(prefix, nodeAttributeSet); Map<NodeAttribute, AttributeValue> nodeattrs =
} attrMgr.getAttributesForNode(hostName);
return nodeattrs.keySet();
@Override
public Map<String, Set<NodeAttribute>> getAllNodeAttributes() {
return this.nodeAttributes;
} }
} }

View File

@ -116,17 +116,12 @@ public class NodeInfo {
} }
// add attributes // add attributes
Map<String, Set<NodeAttribute>> nodeAttributes = Set<NodeAttribute> attrs = ni.getAllNodeAttributes();
ni.getAllNodeAttributes();
nodeAttributesInfo = new NodeAttributesInfo(); nodeAttributesInfo = new NodeAttributesInfo();
if (nodeAttributes != null) {
for (Set<NodeAttribute> attrs : nodeAttributes.values()) {
for (NodeAttribute attribute : attrs) { for (NodeAttribute attribute : attrs) {
NodeAttributeInfo info = new NodeAttributeInfo(attribute); NodeAttributeInfo info = new NodeAttributeInfo(attribute);
this.nodeAttributesInfo.addNodeAttributeInfo(info); this.nodeAttributesInfo.addNodeAttributeInfo(info);
} }
}
}
// add allocation tags // add allocation tags
allocationTags = new AllocationTagsInfo(); allocationTags = new AllocationTagsInfo();

View File

@ -292,8 +292,8 @@ public class MockNodes {
} }
@Override @Override
public Map<String, Set<NodeAttribute>> getAllNodeAttributes() { public Set<NodeAttribute> getAllNodeAttributes() {
return null; return Collections.emptySet();
} }
@Override @Override

View File

@ -1611,9 +1611,10 @@ public class TestRMAdminService {
try { try {
rm.adminService.mapAttributesToNodes(request); rm.adminService.mapAttributesToNodes(request);
fail("host5 is not a valid node, It should have failed"); fail("host5 is not a valid node, It should have failed");
} catch (Exception ex) { } catch (YarnException ex) {
Assert.assertEquals("Exception Message is not as desired", Assert.assertEquals("Exception Message is not as desired",
" Following nodes does not exist : [host5]", ex.getMessage()); " Following nodes does not exist : [host5]",
ex.getCause().getMessage());
} }
request = request =
@ -1633,10 +1634,10 @@ public class TestRMAdminService {
// against hostname hence the message as : nodes does not exist. // against hostname hence the message as : nodes does not exist.
rm.adminService.mapAttributesToNodes(request); rm.adminService.mapAttributesToNodes(request);
fail("host with the port should fail as only hostnames are validated"); fail("host with the port should fail as only hostnames are validated");
} catch (Exception ex) { } catch (YarnException ex) {
Assert.assertEquals("Exception Message is not as desired", Assert.assertEquals("Exception Message is not as desired",
" Following nodes does not exist : [host4:8889, host2:8889]", " Following nodes does not exist : [host4:8889, host2:8889]",
ex.getMessage()); ex.getCause().getMessage());
} }
request = request =
@ -1669,11 +1670,10 @@ public class TestRMAdminService {
try { try {
rm.adminService.mapAttributesToNodes(request); rm.adminService.mapAttributesToNodes(request);
fail("This operation should fail as prefix should be \"nm.yarn.io\"."); fail("This operation should fail as prefix should be \"nm.yarn.io\".");
} catch (Exception ex) { } catch (YarnException ex) {
Assert.assertEquals("Exception Message is not as desired", Assert.assertEquals("Exception Message is not as desired",
"Invalid Attribute Mapping for the node host5. Prefix should be " "Invalid Attribute Mapping for the node host5. Prefix should be "
+ "rm.yarn.io", + "rm.yarn.io", ex.getCause().getMessage());
ex.getMessage());
} }
rm.close(); rm.close();