⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 clargsparser.java

📁 测试工具
💻 JAVA
📖 第 1 页 / 共 2 页
字号:
/* 
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed  under the  License is distributed on an "AS IS" BASIS,
 * WITHOUT  WARRANTIES OR CONDITIONS  OF ANY KIND, either  express  or
 * implied.
 * 
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.commons.cli.avalon;

// Renamed from org.apache.avalon.excalibur.cli

import java.text.ParseException;
import java.util.Hashtable;
import java.util.Vector;

/**
 * Parser for command line arguments.
 * 
 * This parses command lines according to the standard (?) of GNU utilities.
 * 
 * Note: This is still used in 1.1 libraries so do not add 1.2+ dependencies.
 * 
 * Note that CLArgs uses a backing hashtable for the options index and so
 * duplicate arguments are only returned by getArguments().
 * 
 * @see ParserControl
 * @see CLOption
 * @see CLOptionDescriptor
 */
public final class CLArgsParser {
	// cached character == Integer.MAX_VALUE when invalid
	private static final int INVALID = Integer.MAX_VALUE;

	private static final int STATE_NORMAL = 0;

	private static final int STATE_REQUIRE_2ARGS = 1;

	private static final int STATE_REQUIRE_ARG = 2;

	private static final int STATE_OPTIONAL_ARG = 3;

	private static final int STATE_NO_OPTIONS = 4;

	private static final int STATE_OPTION_MODE = 5;

	// Values for creating tokens
	private static final int TOKEN_SEPARATOR = 0;

	private static final int TOKEN_STRING = 1;

	private static final char[] ARG_SEPARATORS = new char[] { (char) 0, '=' };

	private static final char[] NULL_SEPARATORS = new char[] { (char) 0 };

	private final CLOptionDescriptor[] m_optionDescriptors;

	private final Vector m_options;

	private Hashtable m_optionIndex;

	private final ParserControl m_control;

	private String m_errorMessage;

	private String[] m_unparsedArgs = new String[] {};

	// variables used while parsing options.
	private char m_ch;

	private String[] m_args;

	private boolean m_isLong;

	private int m_argIndex;

	private int m_stringIndex;

	private int m_stringLength;

	private int m_lastChar = INVALID;

	private int m_lastOptionId;

	private CLOption m_option;

	private int m_state = STATE_NORMAL;

	/**
	 * Retrieve an array of arguments that have not been parsed due to the
	 * parser halting.
	 * 
	 * @return an array of unparsed args
	 */
	public final String[] getUnparsedArgs() {
		return m_unparsedArgs;
	}

	/**
	 * Retrieve a list of options that were parsed from command list.
	 * 
	 * @return the list of options
	 */
	public final Vector getArguments() {
		// System.out.println( "Arguments: " + m_options );
		return m_options;
	}

	/**
	 * Retrieve the {@link CLOption} with specified id, or <code>null</code>
	 * if no command line option is found.
	 * 
	 * @param id
	 *            the command line option id
	 * @return the {@link CLOption} with the specified id, or <code>null</code>
	 *         if no CLOption is found.
	 * @see CLOption
	 */
	public final CLOption getArgumentById(final int id) {
		return (CLOption) m_optionIndex.get(new Integer(id));
	}

	/**
	 * Retrieve the {@link CLOption} with specified name, or <code>null</code>
	 * if no command line option is found.
	 * 
	 * @param name
	 *            the command line option name
	 * @return the {@link CLOption} with the specified name, or
	 *         <code>null</code> if no CLOption is found.
	 * @see CLOption
	 */
	public final CLOption getArgumentByName(final String name) {
		return (CLOption) m_optionIndex.get(name);
	}

	/**
	 * Get Descriptor for option id.
	 * 
	 * @param id
	 *            the id
	 * @return the descriptor
	 */
	private final CLOptionDescriptor getDescriptorFor(final int id) {
		for (int i = 0; i < m_optionDescriptors.length; i++) {
			if (m_optionDescriptors[i].getId() == id) {
				return m_optionDescriptors[i];
			}
		}

		return null;
	}

	/**
	 * Retrieve a descriptor by name.
	 * 
	 * @param name
	 *            the name
	 * @return the descriptor
	 */
	private final CLOptionDescriptor getDescriptorFor(final String name) {
		for (int i = 0; i < m_optionDescriptors.length; i++) {
			if (m_optionDescriptors[i].getName().equals(name)) {
				return m_optionDescriptors[i];
			}
		}

		return null;
	}

	/**
	 * Retrieve an error message that occured during parsing if one existed.
	 * 
	 * @return the error string
	 */
	public final String getErrorString() {
		// System.out.println( "ErrorString: " + m_errorMessage );
		return m_errorMessage;
	}

	/**
	 * Require state to be placed in for option.
	 * 
	 * @param descriptor
	 *            the Option Descriptor
	 * @return the state
	 */
	private final int getStateFor(final CLOptionDescriptor descriptor) {
		final int flags = descriptor.getFlags();
		if ((flags & CLOptionDescriptor.ARGUMENTS_REQUIRED_2) == CLOptionDescriptor.ARGUMENTS_REQUIRED_2) {
			return STATE_REQUIRE_2ARGS;
		} else if ((flags & CLOptionDescriptor.ARGUMENT_REQUIRED) == CLOptionDescriptor.ARGUMENT_REQUIRED) {
			return STATE_REQUIRE_ARG;
		} else if ((flags & CLOptionDescriptor.ARGUMENT_OPTIONAL) == CLOptionDescriptor.ARGUMENT_OPTIONAL) {
			return STATE_OPTIONAL_ARG;
		} else {
			return STATE_NORMAL;
		}
	}

	/**
	 * Create a parser that can deal with options and parses certain args.
	 * 
	 * @param args
	 *            the args, typically that passed to the
	 *            <code>public static void main(String[] args)</code> method.
	 * @param optionDescriptors
	 *            the option descriptors
	 * @param control
	 *            the parser control used determine behaviour of parser
	 */
	public CLArgsParser(final String[] args, final CLOptionDescriptor[] optionDescriptors, final ParserControl control) {
		m_optionDescriptors = optionDescriptors;
		m_control = control;
		m_options = new Vector();
		m_args = args;

		try {
			parse();
			checkIncompatibilities(m_options);
			buildOptionIndex();
		} catch (final ParseException pe) {
			m_errorMessage = pe.getMessage();
		}

		// System.out.println( "Built : " + m_options );
		// System.out.println( "From : " + Arrays.asList( args ) );
	}

	/**
	 * Check for duplicates of an option. It is an error to have duplicates
	 * unless appropriate flags is set in descriptor.
	 * 
	 * @param arguments
	 *            the arguments
	 */
	private final void checkIncompatibilities(final Vector arguments) throws ParseException {
		final int size = arguments.size();

		for (int i = 0; i < size; i++) {
			final CLOption option = (CLOption) arguments.elementAt(i);
			final int id = option.getDescriptor().getId();
			final CLOptionDescriptor descriptor = getDescriptorFor(id);

			// this occurs when id == 0 and user has not supplied a descriptor
			// for arguments
			if (null == descriptor) {
				continue;
			}

			final int[] incompatible = descriptor.getIncompatible();

			checkIncompatible(arguments, incompatible, i);
		}
	}

	private final void checkIncompatible(final Vector arguments, final int[] incompatible, final int original)
			throws ParseException {
		final int size = arguments.size();

		for (int i = 0; i < size; i++) {
			if (original == i) {
				continue;
			}

			final CLOption option = (CLOption) arguments.elementAt(i);
			final int id = option.getDescriptor().getId();

			for (int j = 0; j < incompatible.length; j++) {
				if (id == incompatible[j]) {
					final CLOption originalOption = (CLOption) arguments.elementAt(original);
					final int originalId = originalOption.getDescriptor().getId();

					String message = null;

					if (id == originalId) {
						message = "Duplicate options for " + describeDualOption(originalId) + " found.";
					} else {
						message = "Incompatible options -" + describeDualOption(id) + " and "
								+ describeDualOption(originalId) + " found.";
					}
					throw new ParseException(message, 0);
				}
			}
		}
	}

	private final String describeDualOption(final int id) {
		final CLOptionDescriptor descriptor = getDescriptorFor(id);
		if (null == descriptor) {
			return "<parameter>";
		} else {
			final StringBuffer sb = new StringBuffer();
			boolean hasCharOption = false;

			if (Character.isLetter((char) id)) {
				sb.append('-');
				sb.append((char) id);
				hasCharOption = true;
			}

			final String longOption = descriptor.getName();
			if (null != longOption) {
				if (hasCharOption) {
					sb.append('/');
				}
				sb.append("--");
				sb.append(longOption);
			}

			return sb.toString();
		}
	}

	/**
	 * Create a parser that deals with options and parses certain args.
	 * 
	 * @param args
	 *            the args
	 * @param optionDescriptors
	 *            the option descriptors
	 */
	public CLArgsParser(final String[] args, final CLOptionDescriptor[] optionDescriptors) {
		this(args, optionDescriptors, null);
	}

	/**
	 * Create a string array that is subset of input array. The sub-array should
	 * start at array entry indicated by index. That array element should only
	 * include characters from charIndex onwards.
	 * 

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -