org.jboss.as.cli.operation.impl.DefaultCallbackHandler Maven / Gradle / Ivy
The newest version!
/*
* JBoss, Home of Professional Open Source.
* Copyright 2011, Red Hat, Inc., and individual contributors
* as indicated by the @author tags. See the copyright.txt file in the
* distribution for a full listing of individual contributors.
*
* This is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation; either version 2.1 of
* the License, or (at your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this software; if not, write to the Free
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
*/
package org.jboss.as.cli.operation.impl;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.jboss.as.cli.ArgumentValueConverter;
import org.jboss.as.cli.CommandContext;
import org.jboss.as.cli.CommandFormatException;
import org.jboss.as.cli.CommandLineFormat;
import org.jboss.as.cli.Util;
import org.jboss.as.cli.operation.OperationFormatException;
import org.jboss.as.cli.operation.ParsedOperationRequestHeader;
import org.jboss.as.cli.operation.ParsedCommandLine;
import org.jboss.as.cli.operation.OperationRequestAddress;
import org.jboss.as.cli.operation.OperationRequestAddress.Node;
import org.jboss.as.cli.parsing.ParserUtil;
import org.jboss.as.cli.parsing.ParsingStateCallbackHandler;
import org.jboss.as.cli.parsing.operation.OperationFormat;
import org.jboss.as.cli.parsing.operation.header.RolloutPlanHeaderCallbackHandler;
import org.jboss.dmr.ModelNode;
/**
*
* @author Alexey Loubyansky
*/
public class DefaultCallbackHandler extends ValidatingCallbackHandler implements ParsedCommandLine {
private static final int SEPARATOR_NONE = 0;
private static final int SEPARATOR_NODE_TYPE_NAME = 1;
private static final int SEPARATOR_NODE = 2;
private static final int SEPARATOR_ADDRESS_OPERATION = 3;
private static final int SEPARATOR_OPERATION_ARGUMENTS = 4;
private static final int SEPARATOR_ARG_NAME_VALUE = 5;
private static final int SEPARATOR_ARG = 6;
private static final int SEPARATOR_ARG_LIST_END = 7;
private static final int SEPARATOR_HEADERS_START = 8;
private static final int SEPARATOR_HEADER = 9;
private static final DefaultOperationRequestAddress EMPTY_ADDRESS = new DefaultOperationRequestAddress();
private int separator = SEPARATOR_NONE;
private int lastSeparatorIndex = -1;
private int lastChunkIndex = 0;
private boolean operationComplete;
private String operationName;
private OperationRequestAddress address;
private Map props = Collections.emptyMap();
private List otherArgs = Collections.emptyList();
private String outputTarget;
private String lastPropName;
private String lastPropValue;
private String lastHeaderName;
private CommandLineFormat format;
private boolean validation;
private String originalLine;
private String substitutedLine;
private LinkedHashMap headers;
private ParsedOperationRequestHeader lastHeader;
public DefaultCallbackHandler() {
this(true);
}
public DefaultCallbackHandler(boolean validate) {
this.validation = validate;
}
public DefaultCallbackHandler(OperationRequestAddress prefix) {
address = prefix;
}
@Deprecated
public void parse(OperationRequestAddress initialAddress, String argsStr) throws CommandFormatException {
parse(initialAddress, argsStr, null);
}
public void parse(OperationRequestAddress initialAddress, String argsStr, CommandContext ctx) throws CommandFormatException {
reset();
if(initialAddress != null) {
address = new DefaultOperationRequestAddress(initialAddress);
}
this.originalLine = argsStr;
substitutedLine = ParserUtil.parse(argsStr, this, validation, ctx);
}
@Deprecated
public void parse(OperationRequestAddress initialAddress, String argsStr, boolean validation) throws CommandFormatException {
parse(initialAddress, argsStr, validation, null);
}
public void parse(OperationRequestAddress initialAddress, String argsStr, boolean validation, CommandContext ctx) throws CommandFormatException {
final boolean defaultValidation = this.validation;
this.validation = validation;
try {
parse(initialAddress, argsStr, ctx);
} finally {
this.validation = defaultValidation;
}
}
public void parseOperation(OperationRequestAddress prefix, String argsStr) throws CommandFormatException {
reset();
this.setFormat(OperationFormat.INSTANCE);
if(prefix != null) {
address = new DefaultOperationRequestAddress(prefix);
}
this.setFormat(OperationFormat.INSTANCE);
this.originalLine = argsStr;
substitutedLine = ParserUtil.parseOperationRequest(argsStr, this);
}
public void parseHeaders(String argsStr) throws CommandFormatException {
reset();
this.originalLine = argsStr;
substitutedLine = ParserUtil.parseHeaders(argsStr, this);
}
public void reset() {
operationComplete = false;
operationName = null;
address = null;
props = Collections.emptyMap();
otherArgs = Collections.emptyList();
outputTarget = null;
lastPropName = null;
lastPropValue = null;
lastSeparatorIndex = -1;
lastChunkIndex = 0;
format = null;
originalLine = null;
substitutedLine = null;
headers = null;
lastHeaderName = null;
lastHeader = null;
}
@Override
public String getOriginalLine() {
return originalLine;
}
@Override
public String getSubstitutedLine() {
return substitutedLine;
}
public List getOtherProperties() {
return otherArgs;
}
@Override
public boolean isRequestComplete() {
return operationComplete;
}
@Override
public boolean endsOnPropertySeparator() {
return separator == SEPARATOR_ARG;
}
@Override
public boolean endsOnPropertyValueSeparator() {
return separator == SEPARATOR_ARG_NAME_VALUE;
}
@Override
public boolean endsOnPropertyListStart() {
return separator == SEPARATOR_OPERATION_ARGUMENTS;
}
@Override
public boolean endsOnPropertyListEnd() {
return separator == SEPARATOR_ARG_LIST_END;
}
@Override
public boolean endsOnHeaderListStart() {
return separator == SEPARATOR_HEADERS_START;
}
@Override
public boolean endsOnAddressOperationNameSeparator() {
return separator == SEPARATOR_ADDRESS_OPERATION;
}
@Override
public boolean endsOnNodeSeparator() {
return separator == SEPARATOR_NODE;
}
@Override
public boolean endsOnNodeTypeNameSeparator() {
return separator == SEPARATOR_NODE_TYPE_NAME;
}
@Override
public boolean endsOnSeparator() {
return separator != SEPARATOR_NONE;
}
@Override
public boolean hasAddress() {
return address != null;
}
@Override
public OperationRequestAddress getAddress() {
return address == null ? EMPTY_ADDRESS : address;
}
@Override
public boolean hasOperationName() {
return operationName != null;
}
@Override
public String getOperationName() {
return operationName;
}
@Override
public boolean hasProperties() {
return !props.isEmpty() || !otherArgs.isEmpty();
}
@Override
public boolean hasProperty(String propertyName) {
return props.containsKey(propertyName);
}
@Override
public void validatedNodeType(int index, String nodeType) throws OperationFormatException {
if(address == null) {
address = new DefaultOperationRequestAddress();
} else if (address.endsOnType()) {
throw new OperationFormatException(
"Can't proceed with node type '"
+ nodeType
+ "' until the node name for the previous node type has been specified.");
}
address.toNodeType(nodeType);
separator = SEPARATOR_NONE;
lastChunkIndex = index;
}
@Override
public void nodeTypeNameSeparator(int index) {
separator = SEPARATOR_NODE_TYPE_NAME;
this.lastSeparatorIndex = index;
}
@Override
public void validatedNodeName(int index, String nodeName) throws OperationFormatException {
if(address == null) {
address = new DefaultOperationRequestAddress();
}
if(!address.endsOnType()) {
throw new OperationFormatException("Node path format is wrong around '" + nodeName + "' (index=" + index + ").");
}
address.toNode(nodeName);
separator = SEPARATOR_NONE;
lastChunkIndex = index;
}
@Override
public void nodeSeparator(int index) {
separator = SEPARATOR_NODE;
this.lastSeparatorIndex = index;
}
@Override
public void addressOperationSeparator(int index) {
separator = SEPARATOR_ADDRESS_OPERATION;
this.lastSeparatorIndex = index;
}
@Override
public void operationName(int index, String operationName) throws OperationFormatException {
if(validation) {
super.operationName(index, operationName);
} else {
validatedOperationName(index, operationName);
}
}
public void validatedOperationName(int index, String operationName) throws OperationFormatException {
this.operationName = operationName;
separator = SEPARATOR_NONE;
lastChunkIndex = index;
}
@Override
public void propertyListStart(int index) {
separator = SEPARATOR_OPERATION_ARGUMENTS;
this.lastSeparatorIndex = index;
}
@Override
public void propertyName(int index, String propertyName) throws OperationFormatException {
if(validation) {
super.propertyName(index, propertyName);
} else {
validatedPropertyName(index, propertyName);
}
}
@Override
protected void validatedPropertyName(int index, String propertyName) throws OperationFormatException {
putProperty(propertyName, null);
lastPropName = propertyName;
lastPropValue = null;
separator = SEPARATOR_NONE;
lastChunkIndex = index;
}
@Override
public void propertyNameValueSeparator(int index) {
separator = SEPARATOR_ARG_NAME_VALUE;
this.lastSeparatorIndex = index;
}
@Override
public void property(String name, String value, int nameValueSeparatorIndex)
throws OperationFormatException {
/* if (value.isEmpty()) {
throw new OperationFormatException(
"The argument value is missing or the format is wrong for argument '"
+ value + "'");
}
*/
if(validation) {
super.property(name, value, nameValueSeparatorIndex);
} else {
validatedProperty(name, value, nameValueSeparatorIndex);
}
}
@Override
protected void validatedProperty(String name, String value, int nameValueSeparatorIndex) throws OperationFormatException {
if(name == null) {
addArgument(value);
} else {
putProperty(name, value);
}
lastPropName = name;
lastPropValue = value;
separator = SEPARATOR_NONE;
if(nameValueSeparatorIndex >= 0) {
this.lastSeparatorIndex = nameValueSeparatorIndex;
}
lastChunkIndex = nameValueSeparatorIndex;
}
@Override
public void propertySeparator(int index) {
separator = SEPARATOR_ARG;
this.lastSeparatorIndex = index;
this.lastPropName = null;
this.lastPropValue = null;
}
@Override
public void propertyListEnd(int index) {
separator = SEPARATOR_ARG_LIST_END;
this.lastSeparatorIndex = index;
this.lastPropName = null;
this.lastPropValue = null;
}
@Override
public void headerListStart(int index) {
separator = SEPARATOR_HEADERS_START;
this.lastSeparatorIndex = index;
}
@Override
public void headerListEnd(int index) {
separator = SEPARATOR_NONE;
//operationComplete = true;
this.lastSeparatorIndex = index;
//this.lastPropName = null;
//this.lastPropValue = null;
operationComplete = true;
}
public void headerSeparator(int index) {
this.separator = SEPARATOR_HEADER;
this.lastSeparatorIndex = index;
}
public void headerNameValueSeparator(int index) {
this.separator = SEPARATOR_ARG_NAME_VALUE;
this.lastSeparatorIndex = index;
}
@Override
public ParsingStateCallbackHandler headerName(int index, String headerName) throws CommandFormatException {
this.separator = SEPARATOR_NONE;
lastChunkIndex = index;
this.lastHeaderName = headerName;
lastHeader = null;
if(headerName.equals("rollout")) {
return new RolloutPlanHeaderCallbackHandler(this);
}
return null;
}
@Override
public void header(String name, String value, int valueIndex) throws CommandFormatException {
if(headers == null) {
headers = new LinkedHashMap();
}
lastHeader = new SimpleParsedOperationRequestHeader(name, value);
headers.put(name, lastHeader);
separator = SEPARATOR_NONE;
this.lastSeparatorIndex = valueIndex -1;
this.lastChunkIndex = valueIndex;
this.lastHeaderName = null;
}
public void header(ParsedOperationRequestHeader header) {
if(headers == null) {
headers = new LinkedHashMap();
}
lastHeader = header;
headers.put(header.getName(), header);
separator = SEPARATOR_NONE;
this.lastHeaderName = null;
}
@Override
public boolean hasHeaders() {
return headers != null || lastHeaderName != null;
}
@Override
public boolean hasHeader(String name) {
return headers != null && headers.containsKey(name);
}
@Override
public String getLastHeaderName() {
return lastHeaderName;
}
@Override
public Collection getHeaders() {
return headers == null ? Collections.emptyList() : headers.values();
}
@Override
public ParsedOperationRequestHeader getLastHeader() {
return lastHeader;
}
@Override
public void rootNode(int index) {
if(address == null) {
address = new DefaultOperationRequestAddress();
} else {
address.reset();
}
separator = SEPARATOR_NONE;
lastChunkIndex = index;
}
@Override
public void parentNode(int index) {
if(address == null) {
throw new IllegalStateException("The address hasn't been initialized yet.");
}
address.toParentNode();
separator = SEPARATOR_NONE;
lastChunkIndex = index;
}
@Override
public void nodeType(int index) {
if(address == null) {
throw new IllegalStateException("The address hasn't been initialized yet.");
}
address.toNodeType();
separator = SEPARATOR_NONE;
lastChunkIndex = index;
}
@Override
public void nodeName(int index, String nodeName) throws OperationFormatException {
if(validation) {
super.nodeName(index, nodeName);
} else {
this.validatedNodeName(index, nodeName);
}
}
@Override
public void nodeType(int index, String nodeType) throws OperationFormatException {
if(validation) {
super.nodeType(index, nodeType);
} else {
this.validatedNodeType(index, nodeType);
}
}
@Override
public void nodeTypeOrName(int index, String typeOrName) throws OperationFormatException {
if(address == null) {
address = new DefaultOperationRequestAddress();
}
if(address.endsOnType()) {
nodeName(index, typeOrName);
} else {
nodeType(index, typeOrName);
}
separator = SEPARATOR_NONE;
}
@Override
public Set getPropertyNames() {
return props.keySet();
}
@Override
public String getPropertyValue(String name) {
return props.get(name);
}
@Override
public int getLastSeparatorIndex() {
return lastSeparatorIndex;
}
@Override
public int getLastChunkIndex() {
return lastChunkIndex;
}
@Override
public void outputTarget(int index, String outputTarget) {
this.outputTarget = outputTarget;
lastChunkIndex = index;
}
public String getOutputTarget() {
return outputTarget;
}
@Override
public String getLastParsedPropertyName() {
return lastPropName;
}
@Override
public String getLastParsedPropertyValue() {
return lastPropValue;
}
public ModelNode toOperationRequest(CommandContext ctx) throws CommandFormatException {
ModelNode request = new ModelNode();
ModelNode addressNode = request.get(Util.ADDRESS);
if(address.isEmpty()) {
addressNode.setEmptyList();
} else {
Iterator iterator = address.iterator();
while (iterator.hasNext()) {
OperationRequestAddress.Node node = iterator.next();
if (node.getName() != null) {
addressNode.add(node.getType(), node.getName());
} else if (iterator.hasNext()) {
throw new OperationFormatException(
"The node name is not specified for type '"
+ node.getType() + "'");
}
}
}
if(operationName == null || operationName.isEmpty()) {
throw new OperationFormatException("The operation name is missing or the format of the operation request is wrong.");
}
request.get(Util.OPERATION).set(operationName);
for(String propName : props.keySet()) {
final String value = props.get(propName);
if(propName == null || propName.trim().isEmpty())
throw new OperationFormatException("The argument name is not specified: '" + propName + "'");
if(value == null || value.trim().isEmpty())
throw new OperationFormatException("The argument value is not specified for " + propName + ": '" + value + "'");
final ModelNode toSet = ArgumentValueConverter.DEFAULT.fromString(ctx, value);
request.get(propName).set(toSet);
}
if(this.lastHeaderName != null) {
throw new OperationFormatException("Header '" + this.lastHeaderName + "' is not complete.");
}
if(headers != null) {
final ModelNode headersNode = request.get(Util.OPERATION_HEADERS);
for(ParsedOperationRequestHeader header : headers.values()) {
header.addTo(ctx, headersNode);
}
}
return request;
}
@Override
public void setFormat(CommandLineFormat format) {
this.format = format;
}
@Override
public CommandLineFormat getFormat() {
return format;
}
@Override
public boolean endsOnHeaderSeparator() {
return separator == SEPARATOR_HEADER;
}
private void putProperty(String key, String name) {
if(props.isEmpty()) {
props = new HashMap();
}
props.put(key, name);
}
private void addArgument(String value) {
if(otherArgs.isEmpty()) {
otherArgs = new ArrayList();
}
otherArgs.add(value);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy