* $optionHandler = new ezcConsoleInput();
*
* // Register simple parameter -h/--help
* $optionHandler->registerOption( new ezcConsoleOption( 'h', 'help' ) );
*
* // Register complex parameter -f/--file
* $file = new ezcConsoleOption(
* 'f',
* 'file',
* ezcConsoleInput::TYPE_STRING,
* null,
* false,
* 'Process a file.',
* 'Processes a single file.'
* );
* $optionHandler->registerOption( $file );
*
* // Manipulate parameter -f/--file after registration
* $file->multiple = true;
*
* // Register another complex parameter that depends on -f and excludes -h
* $dir = new ezcConsoleOption(
* 'd',
* 'dir',
* ezcConsoleInput::TYPE_STRING,
* null,
* true,
* 'Process a directory.',
* 'Processes a complete directory.',
* array( new ezcConsoleOptionRule( $optionHandler->getOption( 'f' ) ) ),
* array( new ezcConsoleOptionRule( $optionHandler->getOption( 'h' ) ) )
* );
* $optionHandler->registerOption( $dir );
*
* // Register an alias for this parameter
* $optionHandler->registerAlias( 'e', 'extended-dir', $dir );
*
* // Process registered parameters and handle errors
* try
* {
* $optionHandler->process( array( 'example_input.php', '-h' ) );
* }
* catch ( ezcConsoleInputException $e )
* {
* if ( $e->getCode() === ezcConsoleInputException::PARAMETER_DEPENDENCY_RULE_NOT_MET )
* {
* $consoleOut->outputText(
* 'Parameter ' . isset( $e->param ) ? $e->param->name : 'unknown' . " may not occur here.\n", 'error'
* );
* }
* exit( 1 );
* }
*
* // Process a single parameter
* $file = $optionHandler->getOption( 'f' );
* if ( $file->value === false )
* {
* echo "Parameter -{$file->short}/--{$file->long} was not submitted.\n";
* }
* elseif ( $file->value === true )
* {
* echo "Parameter -{$file->short}/--{$file->long} was submitted without value.\n";
* }
* else
* {
* echo "Parameter -{$file->short}/--{$file->long} was submitted with value <".var_export($file->value, true).">.\n";
* }
*
* // Process all parameters at once:
* foreach ( $optionHandler->getOptionValues() as $paramShort => $val )
* {
* switch ( true )
* {
* case $val === false:
* echo "Parameter $paramShort was not submitted.\n";
* break;
* case $val === true:
* echo "Parameter $paramShort was submitted without a value.\n";
* break;
* case is_array( $val ):
* echo "Parameter $paramShort was submitted multiple times with value: <".implode(', ', $val).">.\n";
* break;
* default:
* echo "Parameter $paramShort was submitted with value: <$val>.\n";
* break;
* }
* }
*
*
* @package ConsoleTools
* @version 1.0
*/
class ezcConsoleInput
{
/**
* Option does not cary a value.
*/
const TYPE_NONE = 1;
/**
* Option takes an int value.
*/
const TYPE_INT = 2;
/**
* Option takes a string value.
*/
const TYPE_STRING = 3;
/**
* Array of option definitions, indexed by number.
* This array stores the ezcConsoleOption objects representing
* the options.
*
* For lookup of a option after it's short or long values the attributes
* @link ezcConsoleInput::$optionShort
* @link ezcConsoleInput::$optionLong
* are used.
*
* @var array(int => array)
*/
private $options = array();
/**
* Short option names. Each references a key in
* {@link ezcConsoleInput::$options}.
*
* @var array(string => int)
*/
private $optionShort = array();
/**
* Long option names. Each references a key in
* {@link ezcConsoleInput::$options}.
*
* @var array(string => int)
*/
private $optionLong = array();
/**
* Arguments, if submitted, are stored here.
*
* @var array
*/
private $arguments = array();
/**
* Create input handler
*/
public function __construct()
{
}
/**
* Register a new option.
* This method adds a new option to your option collection. If allready a
* option with the assigned short or long value exists, an exception will
* be thrown.
*
* @see ezcConsoleInput::unregisterOption()
*
* @param ezcConsoleOption $option The option to register.
*
* @return ezcConsoleOption The recently registered option.
*/
public function registerOption( ezcConsoleOption $option )
{
foreach ( $this->optionShort as $short => $ref )
{
if ( $short === $option->short )
{
throw new ezcConsoleOptionAlreadyRegisteredException( $short );
}
}
foreach ( $this->optionLong as $long => $ref )
{
if ( $long === $option->long )
{
throw new ezcConsoleOptionAlreadyRegisteredException( $long );
}
}
$this->options[] = $option;
$this->optionLong[$option->long] = $option;
$this->optionShort[$option->short] = $option;
return $option;
}
/**
* Register an alias to a option.
* Registers a new alias for an existing option. Aliases may
* then be used as if they were real option.
*
* @see ezcConsoleInput::unregisterAlias()
*
* @param string $short Shortcut of the alias
* @param string $long Long version of the alias
* @param ezcConsoleOption $option Reference to an existing option
*
*
* @throws ezcConsoleOptionNotExistsException
* If the referenced option is not registered.
* @throws ezcConsoleOptionAlreadyRegisteredException
* If another option/alias has taken the provided short or long name.
* @return void
*/
public function registerAlias( $short, $long, ezcConsoleOption $option )
{
$short = $short;
$long = $long;
if ( !isset( $this->optionShort[$option->short] ) || !isset( $this->optionLong[$option->long] ) )
{
throw new ezcConsoleOptionNotExistsException( $option->long );
}
if ( isset( $this->optionShort[$short] ) || isset( $optionLong[$long] ) )
{
throw new ezcConsoleOptionAlreadyRegisteredException( isset( $this->optionShort[$short] ) ? $this->optionShort[$short] : $this->optionLong[$long] );
}
$this->shortParam[$short] = $option;
$this->longParam[$long] = $option;
}
/**
* Registeres options according to a string specification.
* Accepts a string like used in eZ publis 3.x to define parameters and
* registeres all parameters as options accordingly. String definitions look like
* this:
*
*
* [s:|size:][u:|user:][a:|all:]
*
*
* This string will result in 3 parameters:
* -s / --size
* -u / --user
* -a / --all
*
* @param string $optionDef Option definition string.
* @return void
*
* @throws ezcConsoleOptionStringNotWellformedException
* If string provided is not wellformed.
*/
public function registerOptionString( $optionDef )
{
$regex = '/\[([a-z0-9-]+)([:?*+])?([^|]*)\|([a-z0-9-]+)([:?*+])?\]/';
if ( preg_match_all( $regex, $optionDef, $matches ) )
{
foreach ( $matches[1] as $id => $short )
{
$option = null;
if ( empty( $matches[4][$id] ) )
{
throw new ezcConsoleOptionStringNotWellformedException( "Missing long parameter name for short parameter <-{$short}>" );
}
$option = new ezcConsoleOption( $short, $matches[4][$id] );
if ( !empty( $matches[2][$id] ) || !empty( $matches[5][$id] ) )
{
switch ( !empty( $matches[2][$id] ) ? $matches[2][$id] : $matches[5][$id] )
{
case '*':
// Allows 0 or more occurances
$option->multiple = true;
break;
case '+':
// Allows 1 or more occurances
$option->multiple = true;
$option->type = self::TYPE_STRING;
break;
case '?':
$option->type = self::TYPE_STRING;
$option->default = '';
break;
default:
break;
}
}
if ( !empty( $matches[3][$id] ) )
{
$option->default = $matches[3][$id];
}
$this->registerOption( $option );
}
}
}
/**
* Remove a option to be no more supported.
* Using this function you will remove a option. All dependencies to that
* specific option are removed completly from every other registered
* option.
*
* @see ezcConsoleInput::registerOption()
*
* @param ezcConsoleOption $option The option object to unregister.
*
* @throws ezcConsoleOptionNotExistsException
* If requesting a not registered option.
* @return void
*/
public function unregisterOption( ezcConsoleOption $option )
{
$found = false;
foreach ( $this->options as $id => $existParam )
{
if ( $existParam === $option )
{
$found = true;
unset( $this->options[$id] );
continue;
}
$existParam->removeAllExclusions( $option );
$existParam->removeAllDependencies( $option );
}
if ( $found === false )
{
throw new ezcConsoleOptionNotExistsException( $option->long );
}
foreach ( $this->optionLong as $name => $existParam )
{
if ( $existParam === $option )
{
unset( $this->optionLong[$name] );
}
}
foreach ( $this->optionShort as $name => $existParam )
{
if ( $existParam === $option )
{
unset( $this->optionShort[$name] );
}
}
}
/**
* Remove a alias to be no more supported.
* Using this function you will remove an alias.
*
* @see ezcConsoleInput::registerAlias()
*
* @throws ezcConsoleOptionNoAliasException
* If the requested short/long name belongs to a real parameter instead.
*
* @param string $short Short name of the alias
* @param string $long Long name of the alias.
* @return void
*
* @todo Check if $short and $long refer to the same option!
*/
public function unregisterAlias( $short, $long )
{
$short = $short;
$long = $long;
foreach ( $this->options as $id => $option )
{
if ( $option->short === $short )
{
throw new ezcConsoleOptionNoAliasException( $short );
}
if ( $option->long === $long )
{
throw new ezcConsoleOptionNoAliasException( $long );
}
}
if ( isset( $this->optionShort[$short] ) )
{
unset( $this->optionShort[$short] );
}
if ( isset( $this->optionLong[$short] ) )
{
unset( $this->optionLong[$long] );
}
}
/**
* Returns the definition object for a specific option.
* This method receives the long or short name of a option and
* returns the ezcConsoleOption object.
*
* @param string $name Short or long name of the option (without - or --).
* @return ezcConsoleOption The requested option.
*
* @throws ezcConsoleOptionNotExistsException
* If requesting a not registered parameter.
*/
public function getOption( $name )
{
$name = $name;
if ( isset( $this->optionShort[$name] ) )
{
return $this->optionShort[$name];
}
if ( isset( $this->optionLong[$name] ) )
{
return $this->optionLong[$name];
}
throw new ezcConsoleOptionNotExistsException( $name );
}
/**
* Process the input parameters.
* Actually process the input options and arguments according to the actual
* settings.
*
* Per default this method uses $argc and $argv for processing. You can
* override this setting with your own input, if necessary, using the
* parameters of this method. (Attention, first argument is always the pro
* gram name itself!)
*
* All exceptions thrown by this method contain an additional attribute "option"
* which specifies the parameter on which the error occured.
*
* @param array(int=>string) $args The arguments
* @return void
*
* @throws ezcConsoleOptionNotExistsException
* If an option that was submitted does not exist.
* @throws ezcConsoleOptionDependencyViolationException
* If a dependency rule was violated.
* @throws ezcConsoleOptionExclusionViolationException
* If an exclusion rule was violated.
* @throws ezcConsoleOptionTypeViolationException
* If the type of a submitted value violates the options type rule.
* @throws ezcConsoleOptionArgumentsViolationException
* If arguments are passed although a parameter dissallowed them.
*
* @see ezcConsoleOptionException
*/
public function process( array $args = null )
{
if ( !isset( $args ) )
{
$args = isset( $argv ) ? $argv : isset( $_SERVER['argv'] ) ? $_SERVER['argv'] : array();
}
$i = 1;
while ( $i < count( $args ) )
{
// Equalize parameter handling (long params with =)
if ( substr( $args[$i], 0, 2 ) == '--' )
{
$this->preprocessLongOption( $args, $i );
}
// Check for parameter
if ( substr( $args[$i], 0, 1) === '-' && $this->hasOption( preg_replace( '/^-*/', '', $args[$i] ) ) !== false )
{
$this->processOptions( $args, $i );
}
// Looks like parameter, but is not available??
elseif ( substr( $args[$i], 0, 1) === '-' && trim( $args[$i] ) !== '--' )
{
throw new ezcConsoleOptionNotExistsException( $args[$i] );
}
// Must be the arguments
else
{
$args[$i] == '--' ? ++$i : $i;
$this->processArguments( $args, $i );
break;
}
}
$this->checkRules();
}
/**
* Returns if an option with the given name exists.
* Checks if an option with the given name is registered.
*
* @param string $name Short or long name of the option.
* @return bool True if option exists, otherwise false.
*/
public function hasOption( $name )
{
try
{
$param = $this->getOption( $name );
}
catch ( ezcConsoleOptionNotExistsException $e )
{
return false;
}
return true;
}
/**
* Returns an array of all registered options.
* Returns an array of all registered options in the following format:
*
* array(
* 0 => ezcConsoleOption,
* 1 => ezcConsoleOption,
* 2 => ezcConsoleOption,
* ...
* );
*
*
* @return array(string=>ezcConsoleOption) Registered options.
*/
public function getOptions()
{
return $this->options;
}
/**
* Returns the values of all submitted options.
* Returns an array of all values submitted to the options. The array is
* indexed by the parameters short name (excluding the '-' prefix). The array
* does not contain any parameter, whiches value is 'false' (meaning: the
* parameter was not submitted).
*
* @return array(string=>mixed)
*/
public function getOptionValues()
{
$res = array();
foreach ( $this->options as $param )
{
if ( $param->value !== false )
{
$res[$param->short] = $param->value;
}
}
return $res;
}
/**
* Returns arguments provided to the program.
* This method returns all arguments provided to a program in an
* int indexed array. Arguments are sorted in the way
* they are submitted to the program. You can disable arguments
* through the 'arguments' flag of a parameter, if you want
* to disallow arguments.
*
* Arguments are either the last part of the program call (if the
* last parameter is not a 'multiple' one) or divided via the '--'
* method which is commonly used on Unix (if the last parameter
* accepts multiple values this is required).
*
* @return array(int=>string) Arguments.
*/
public function getArguments()
{
return $this->arguments;
}
/**
* Get help information for your options.
* This method returns an array of help information for your options,
* indexed by int. Each helo info has 2 fields:
*
* 0 => The options names (" / ")
* 1 => The help text (depending on the $long parameter)
*
* The $long options determines if you want to get the short- or longhelp
* texts. The array returned can be used by {@link ezcConsoleTable}.
*
* If using the second options, you can filter the options shown in the
* help output (e.g. to show short help for related options). Provide
* as simple number indexed array of short and/or long values to set a filter.
*
* @param bool $long Set this to true for getting the long help version.
* @param array(int=>string) $params Set of option names to generate help for, default is all.
* @return array(int=>array(int=>string)) Table structure as explained.
*/
public function getHelp( $long = false, array $params = array() )
{
$help = array();
foreach ( $this->options as $id => $param )
{
if ( count( $params ) === 0 || in_array( $param->short, $params ) || in_array( $param->long, $params ) )
{
$help[] = array(
'-' . $param->short . ' / ' . '--' . $param->long,
$long == false ? $param->shorthelp : $param->longhelp,
);
}
}
return $help;
}
/**
* Returns a synopsis string for the program.
* This gives you a synopsis definition for the options and arguments
* defined with this instance of ezcConsoleInput. You can filter the
* options named in the synopsis by submitting their short names in an
* array as the parameter of this method. If the parameter $options
* is set, only the option names listed in this array are listed in the
* synopsis.
*
* @param array(int=>string) $optionNames Names of options to include.
* @return string The generated synopsis
*/
public function getSynopsis( array $optionNames = null )
{
$usedOptions = array();
$allowsArgs = true;
$synopsis = '$ ' . ( isset( $argv ) && sizeoff( $argv ) > 0 ? $argv[0] : $_SERVER['argv'][0] ) . ' ';
foreach ( $this->getOptions() as $option )
{
if ( $optionNames === null || is_array( $optionNames ) && ( in_array( $optionNames, $option->short ) || in_array( $optionNames, $option->long ) ) )
{
$synopsis .= $this->createOptionSynopsis( $option, $usedOptions, $allowsArgs );
}
}
$synopsis .= " [[--] ]";
return $synopsis;
}
/**
* Returns the synopsis string for a single option and it's dependencies.
* This method returns a part of the program synopsis, specifically for a
* certain parameter. The method recursively adds depending parameters up
* to the 2nd depth level to the synopsis. The second parameter is used
* to store the short names of all options that have already been used in
* the synopsis (to avoid adding an option twice). The 3rd parameter
* determines the actual deps in the option dependency recursion to
* terminate that after 2 recursions.
*
* @param ezcConsoleOption $option The option to include.
* @param array(int=>string) $usedOptions Array of used option short names.
* @param int $depth Current recursion depth.
* @return string The synopsis for this parameter.
*/
protected function createOptionSynopsis( ezcConsoleOption $option, &$usedOptions, $depth = 0 )
{
$synopsis = '';
// Break after a nesting level of 2
if ( $depth++ > 2 || in_array( $option->short, $usedOptions ) ) return $synopsis;
$usedOptions[] = $option->short;
$synopsis .= "-{$option->short}";
if ( isset( $option->default ) )
{
$synopsis .= " " . ( $option->type === ezcConsoleInput::TYPE_STRING ? '"' : '' ) . $option->default . ( $option->type === ezcConsoleInput::TYPE_STRING ? '"' : '' );
}
else if ( $option->type !== ezcConsoleInput::TYPE_NONE )
{
$synopsis .= " ";
switch ( $option->type )
{
case ezcConsoleInput::TYPE_STRING:
$synopsis .= "";
break;
case ezcConsoleInput::TYPE_INT:
$synopsis .= "";
break;
default:
$synopsis .= "";
break;
}
}
foreach ( $option->getDependencies() as $rule )
{
$deeperSynopsis = $this->createOptionSynopsis( $rule->option, $usedOptions, $depth );
$synopsis .= strlen( trim( $deeperSynopsis ) ) > 0 ? ' ' . $deeperSynopsis : '';
}
if ( $option->arguments === false )
{
$allowsArgs = false;
}
// Make the whole thing optional?
if ( $option->mandatory === false )
{
$synopsis = "[$synopsis]";
}
return $synopsis . ' ';
}
/**
* Process an option.
* This method does the processing of a single option.
*
* @param array(int=>string) $args The arguments array.
* @param int $i The current position in the arguments array.
* @return void
*
* @throws ezcConsoleOptionTooManyValuesException
* If an option that expects only a single value was submitted
* with multiple values.
* @throws ezcConsoleOptionTypeViolationException
* If an option was submitted with a value of the wrong type.
* @throws ezcConsoleOptionMissingValueException
* If an option thats expects a value was submitted without.
*/
private function processOptions( array $args, &$i )
{
$option = $this->getOption( preg_replace( '/^-+/', '', $args[$i++] ) );
// No value expected
if ( $option->type === ezcConsoleInput::TYPE_NONE )
{
// No value expected
if ( isset( $args[$i] ) && substr( $args[$i], 0, 1 ) !== '-' )
{
// But one found
throw new ezcConsoleOptionTypeViolationException( $option, $args[$i] );
}
// Multiple occurance possible
if ( $option->multiple === true )
{
$option->value[] = true;
}
else
{
$option->value = true;
}
// Everything fine, nothing to do
return $i;
}
// Value expected, check for it
if ( isset( $args[$i] ) && substr( $args[$i], 0, 1 ) !== '-' )
{
// Type check
if ( $this->isCorrectType( $option, $args[$i] ) === false )
{
throw new ezcConsoleOptionTypeViolationException( $option, $args[$i] );
}
// Multiple values possible
if ( $option->multiple === true )
{
$option->value[] = $args[$i];
}
// Only single value expected, check for multiple
elseif ( isset( $option->value ) && $option->value !== false )
{
throw new ezcConsoleOptionTooManyValuesException( $option );
}
else
{
$option->value = $args[$i];
}
$i++;
}
// Value found? If not, use default, if available
if ( !isset( $option->value ) || $option->value === false || ( is_array( $option->value ) && count( $option->value ) === 0) )
{
throw new ezcConsoleOptionMissingValueException( $option );
}
return $i;
}
/**
* Process arguments given to the program.
*
* @param array(int=>string) $args The arguments array.
* @param int $i Current index in arguments array.
* @return void
*/
private function processArguments( array $args, &$i )
{
while ( $i < count( $args ) )
{
$this->arguments[] = $args[$i++];
}
}
/**
* Check the rules that may be associated with an option.
*
* Options are allowed to have rules associated for
* dependencies to other options and exclusion of other options or
* arguments. This method processes the checks.
*
* @throws ezcConsoleOptionDependencyViolationException
* If a dependency was violated.
* @throws ezcConsoleOptionExclusionViolationException
* If an exclusion rule was violated.
* @throws ezcConsoleOptionArgumentsViolationException
* If arguments are passed although a parameter dissallowed them.
* @throws ezcConsoleOptionMandatoryViolationException
* If an option that was marked mandatory was not submitted.
* @throws ezcConsoleOptionMissingValueException
* If an option that expects a value was submitted without one.
* @return void
*/
private function checkRules()
{
$values = $this->getOptionValues();
foreach ( $this->options as $id => $option )
{
// Mandatory
if ( $option->mandatory === true && $option->value === false )
{
throw new ezcConsoleOptionMandatoryViolationException( $option );
}
// Not set and not mandatory? No checking.
if ( $option->value === false || is_array( $option->value ) && count( $option->value ) === 0 )
{
// Parameter was not set so ignore it's rules.
continue;
}
// Option was set, so check further on
// Dependencies
foreach ( $option->getDependencies() as $dep )
{
if ( !isset( $values[$dep->option->short] ) || $values[$dep->option->short] === false )
{
throw new ezcConsoleOptionDependencyViolationException( $option, $dep->option );
}
$depVals = $dep->values;
if ( count( $depVals ) > 0 )
{
if ( !in_array( $values[$dep->option->short], $depVals ) )
{
throw new ezcConsoleOptionDependencyViolationException( $option, $dep->option, implode( ', ', $depVals ) );
}
}
}
// Exclusions
foreach ( $option->getExclusions() as $exc )
{
if ( isset( $values[$exc->option->short] ) && $values[$exc->option->short] !== false )
{
throw new ezcConsoleOptionExclusionViolationException( $option, $exc->option );
}
$excVals = $exc->values;
if ( count( $excVals ) > 0 )
{
if ( in_array( $values[$exc->option->short], $excVals ) )
{
throw new ezcConsoleOptionExclusionViolationException( $option, $exc->option, $option->value );
}
}
}
// Arguments
if ( $option->arguments === false && is_array( $this->arguments ) && count( $this->arguments ) > 0 )
{
throw new ezcConsoleOptionArgumentsViolationException( $option );
}
}
}
/**
* Checks if a value is of a given type. Converts the value to the
* correct PHP type on success.
*
* @param ezcConsoleOption $option The option.
* @param string $val The value to check.
* @return bool True on succesful check, otherwise false.
*/
private function isCorrectType( ezcConsoleOption $option, &$val )
{
$res = false;
switch ( $option->type )
{
case ezcConsoleInput::TYPE_STRING:
$res = true;
$val = preg_replace( '/^(["\'])(.*)\1$/', '\2', $val );
break;
case ezcConsoleInput::TYPE_INT:
$res = preg_match( '/^[0-9]+$/', $val ) ? true : false;
if ( $res )
{
$val = ( int ) $val;
}
break;
}
return $res;
}
/**
* Split parameter and value for long option names. This method checks
* for long options, if the value is passed using =. If this is the case
* parameter and value get split and replaced in the arguments array.
*
* @param array(int=>string) $args The arguments array
* @param int $i Current arguments array position
* @return void
*/
private function preprocessLongOption( array &$args, $i )
{
// Value given?
if ( preg_match( '/^--\w+\=[^ ]/i', $args[$i] ) )
{
// Split param and value and replace current param
$parts = explode( '=', $args[$i], 2 );
array_splice( $args, $i, 1, $parts );
}
}
}
?>