Spaces:
Runtime error
Runtime error
/** internal | |
* class ActionContainer | |
* | |
* Action container. Parent for [[ArgumentParser]] and [[ArgumentGroup]] | |
**/ | |
; | |
var format = require('util').format; | |
// Constants | |
var c = require('./const'); | |
var $$ = require('./utils'); | |
//Actions | |
var ActionHelp = require('./action/help'); | |
var ActionAppend = require('./action/append'); | |
var ActionAppendConstant = require('./action/append/constant'); | |
var ActionCount = require('./action/count'); | |
var ActionStore = require('./action/store'); | |
var ActionStoreConstant = require('./action/store/constant'); | |
var ActionStoreTrue = require('./action/store/true'); | |
var ActionStoreFalse = require('./action/store/false'); | |
var ActionVersion = require('./action/version'); | |
var ActionSubparsers = require('./action/subparsers'); | |
// Errors | |
var argumentErrorHelper = require('./argument/error'); | |
/** | |
* new ActionContainer(options) | |
* | |
* Action container. Parent for [[ArgumentParser]] and [[ArgumentGroup]] | |
* | |
* ##### Options: | |
* | |
* - `description` -- A description of what the program does | |
* - `prefixChars` -- Characters that prefix optional arguments | |
* - `argumentDefault` -- The default value for all arguments | |
* - `conflictHandler` -- The conflict handler to use for duplicate arguments | |
**/ | |
var ActionContainer = module.exports = function ActionContainer(options) { | |
options = options || {}; | |
this.description = options.description; | |
this.argumentDefault = options.argumentDefault; | |
this.prefixChars = options.prefixChars || ''; | |
this.conflictHandler = options.conflictHandler; | |
// set up registries | |
this._registries = {}; | |
// register actions | |
this.register('action', null, ActionStore); | |
this.register('action', 'store', ActionStore); | |
this.register('action', 'storeConst', ActionStoreConstant); | |
this.register('action', 'storeTrue', ActionStoreTrue); | |
this.register('action', 'storeFalse', ActionStoreFalse); | |
this.register('action', 'append', ActionAppend); | |
this.register('action', 'appendConst', ActionAppendConstant); | |
this.register('action', 'count', ActionCount); | |
this.register('action', 'help', ActionHelp); | |
this.register('action', 'version', ActionVersion); | |
this.register('action', 'parsers', ActionSubparsers); | |
// raise an exception if the conflict handler is invalid | |
this._getHandler(); | |
// action storage | |
this._actions = []; | |
this._optionStringActions = {}; | |
// groups | |
this._actionGroups = []; | |
this._mutuallyExclusiveGroups = []; | |
// defaults storage | |
this._defaults = {}; | |
// determines whether an "option" looks like a negative number | |
// -1, -1.5 -5e+4 | |
this._regexpNegativeNumber = new RegExp('^[-]?[0-9]*\\.?[0-9]+([eE][-+]?[0-9]+)?$'); | |
// whether or not there are any optionals that look like negative | |
// numbers -- uses a list so it can be shared and edited | |
this._hasNegativeNumberOptionals = []; | |
}; | |
// Groups must be required, then ActionContainer already defined | |
var ArgumentGroup = require('./argument/group'); | |
var MutuallyExclusiveGroup = require('./argument/exclusive'); | |
// | |
// Registration methods | |
// | |
/** | |
* ActionContainer#register(registryName, value, object) -> Void | |
* - registryName (String) : object type action|type | |
* - value (string) : keyword | |
* - object (Object|Function) : handler | |
* | |
* Register handlers | |
**/ | |
ActionContainer.prototype.register = function (registryName, value, object) { | |
this._registries[registryName] = this._registries[registryName] || {}; | |
this._registries[registryName][value] = object; | |
}; | |
ActionContainer.prototype._registryGet = function (registryName, value, defaultValue) { | |
if (arguments.length < 3) { | |
defaultValue = null; | |
} | |
return this._registries[registryName][value] || defaultValue; | |
}; | |
// | |
// Namespace default accessor methods | |
// | |
/** | |
* ActionContainer#setDefaults(options) -> Void | |
* - options (object):hash of options see [[Action.new]] | |
* | |
* Set defaults | |
**/ | |
ActionContainer.prototype.setDefaults = function (options) { | |
options = options || {}; | |
for (var property in options) { | |
if ($$.has(options, property)) { | |
this._defaults[property] = options[property]; | |
} | |
} | |
// if these defaults match any existing arguments, replace the previous | |
// default on the object with the new one | |
this._actions.forEach(function (action) { | |
if ($$.has(options, action.dest)) { | |
action.defaultValue = options[action.dest]; | |
} | |
}); | |
}; | |
/** | |
* ActionContainer#getDefault(dest) -> Mixed | |
* - dest (string): action destination | |
* | |
* Return action default value | |
**/ | |
ActionContainer.prototype.getDefault = function (dest) { | |
var result = $$.has(this._defaults, dest) ? this._defaults[dest] : null; | |
this._actions.forEach(function (action) { | |
if (action.dest === dest && $$.has(action, 'defaultValue')) { | |
result = action.defaultValue; | |
} | |
}); | |
return result; | |
}; | |
// | |
// Adding argument actions | |
// | |
/** | |
* ActionContainer#addArgument(args, options) -> Object | |
* - args (String|Array): argument key, or array of argument keys | |
* - options (Object): action objects see [[Action.new]] | |
* | |
* #### Examples | |
* - addArgument([ '-f', '--foo' ], { action: 'store', defaultValue: 1, ... }) | |
* - addArgument([ 'bar' ], { action: 'store', nargs: 1, ... }) | |
* - addArgument('--baz', { action: 'store', nargs: 1, ... }) | |
**/ | |
ActionContainer.prototype.addArgument = function (args, options) { | |
args = args; | |
options = options || {}; | |
if (typeof args === 'string') { | |
args = [ args ]; | |
} | |
if (!Array.isArray(args)) { | |
throw new TypeError('addArgument first argument should be a string or an array'); | |
} | |
if (typeof options !== 'object' || Array.isArray(options)) { | |
throw new TypeError('addArgument second argument should be a hash'); | |
} | |
// if no positional args are supplied or only one is supplied and | |
// it doesn't look like an option string, parse a positional argument | |
if (!args || args.length === 1 && this.prefixChars.indexOf(args[0][0]) < 0) { | |
if (args && !!options.dest) { | |
throw new Error('dest supplied twice for positional argument'); | |
} | |
options = this._getPositional(args, options); | |
// otherwise, we're adding an optional argument | |
} else { | |
options = this._getOptional(args, options); | |
} | |
// if no default was supplied, use the parser-level default | |
if (typeof options.defaultValue === 'undefined') { | |
var dest = options.dest; | |
if ($$.has(this._defaults, dest)) { | |
options.defaultValue = this._defaults[dest]; | |
} else if (typeof this.argumentDefault !== 'undefined') { | |
options.defaultValue = this.argumentDefault; | |
} | |
} | |
// create the action object, and add it to the parser | |
var ActionClass = this._popActionClass(options); | |
if (typeof ActionClass !== 'function') { | |
throw new Error(format('Unknown action "%s".', ActionClass)); | |
} | |
var action = new ActionClass(options); | |
// throw an error if the action type is not callable | |
var typeFunction = this._registryGet('type', action.type, action.type); | |
if (typeof typeFunction !== 'function') { | |
throw new Error(format('"%s" is not callable', typeFunction)); | |
} | |
return this._addAction(action); | |
}; | |
/** | |
* ActionContainer#addArgumentGroup(options) -> ArgumentGroup | |
* - options (Object): hash of options see [[ArgumentGroup.new]] | |
* | |
* Create new arguments groups | |
**/ | |
ActionContainer.prototype.addArgumentGroup = function (options) { | |
var group = new ArgumentGroup(this, options); | |
this._actionGroups.push(group); | |
return group; | |
}; | |
/** | |
* ActionContainer#addMutuallyExclusiveGroup(options) -> ArgumentGroup | |
* - options (Object): {required: false} | |
* | |
* Create new mutual exclusive groups | |
**/ | |
ActionContainer.prototype.addMutuallyExclusiveGroup = function (options) { | |
var group = new MutuallyExclusiveGroup(this, options); | |
this._mutuallyExclusiveGroups.push(group); | |
return group; | |
}; | |
ActionContainer.prototype._addAction = function (action) { | |
var self = this; | |
// resolve any conflicts | |
this._checkConflict(action); | |
// add to actions list | |
this._actions.push(action); | |
action.container = this; | |
// index the action by any option strings it has | |
action.optionStrings.forEach(function (optionString) { | |
self._optionStringActions[optionString] = action; | |
}); | |
// set the flag if any option strings look like negative numbers | |
action.optionStrings.forEach(function (optionString) { | |
if (optionString.match(self._regexpNegativeNumber)) { | |
if (!self._hasNegativeNumberOptionals.some(Boolean)) { | |
self._hasNegativeNumberOptionals.push(true); | |
} | |
} | |
}); | |
// return the created action | |
return action; | |
}; | |
ActionContainer.prototype._removeAction = function (action) { | |
var actionIndex = this._actions.indexOf(action); | |
if (actionIndex >= 0) { | |
this._actions.splice(actionIndex, 1); | |
} | |
}; | |
ActionContainer.prototype._addContainerActions = function (container) { | |
// collect groups by titles | |
var titleGroupMap = {}; | |
this._actionGroups.forEach(function (group) { | |
if (titleGroupMap[group.title]) { | |
throw new Error(format('Cannot merge actions - two groups are named "%s".', group.title)); | |
} | |
titleGroupMap[group.title] = group; | |
}); | |
// map each action to its group | |
var groupMap = {}; | |
function actionHash(action) { | |
// unique (hopefully?) string suitable as dictionary key | |
return action.getName(); | |
} | |
container._actionGroups.forEach(function (group) { | |
// if a group with the title exists, use that, otherwise | |
// create a new group matching the container's group | |
if (!titleGroupMap[group.title]) { | |
titleGroupMap[group.title] = this.addArgumentGroup({ | |
title: group.title, | |
description: group.description | |
}); | |
} | |
// map the actions to their new group | |
group._groupActions.forEach(function (action) { | |
groupMap[actionHash(action)] = titleGroupMap[group.title]; | |
}); | |
}, this); | |
// add container's mutually exclusive groups | |
// NOTE: if add_mutually_exclusive_group ever gains title= and | |
// description= then this code will need to be expanded as above | |
var mutexGroup; | |
container._mutuallyExclusiveGroups.forEach(function (group) { | |
mutexGroup = this.addMutuallyExclusiveGroup({ | |
required: group.required | |
}); | |
// map the actions to their new mutex group | |
group._groupActions.forEach(function (action) { | |
groupMap[actionHash(action)] = mutexGroup; | |
}); | |
}, this); // forEach takes a 'this' argument | |
// add all actions to this container or their group | |
container._actions.forEach(function (action) { | |
var key = actionHash(action); | |
if (groupMap[key]) { | |
groupMap[key]._addAction(action); | |
} else { | |
this._addAction(action); | |
} | |
}); | |
}; | |
ActionContainer.prototype._getPositional = function (dest, options) { | |
if (Array.isArray(dest)) { | |
dest = dest[0]; | |
} | |
// make sure required is not specified | |
if (options.required) { | |
throw new Error('"required" is an invalid argument for positionals.'); | |
} | |
// mark positional arguments as required if at least one is | |
// always required | |
if (options.nargs !== c.OPTIONAL && options.nargs !== c.ZERO_OR_MORE) { | |
options.required = true; | |
} | |
if (options.nargs === c.ZERO_OR_MORE && typeof options.defaultValue === 'undefined') { | |
options.required = true; | |
} | |
// return the keyword arguments with no option strings | |
options.dest = dest; | |
options.optionStrings = []; | |
return options; | |
}; | |
ActionContainer.prototype._getOptional = function (args, options) { | |
var prefixChars = this.prefixChars; | |
var optionStrings = []; | |
var optionStringsLong = []; | |
// determine short and long option strings | |
args.forEach(function (optionString) { | |
// error on strings that don't start with an appropriate prefix | |
if (prefixChars.indexOf(optionString[0]) < 0) { | |
throw new Error(format('Invalid option string "%s": must start with a "%s".', | |
optionString, | |
prefixChars | |
)); | |
} | |
// strings starting with two prefix characters are long options | |
optionStrings.push(optionString); | |
if (optionString.length > 1 && prefixChars.indexOf(optionString[1]) >= 0) { | |
optionStringsLong.push(optionString); | |
} | |
}); | |
// infer dest, '--foo-bar' -> 'foo_bar' and '-x' -> 'x' | |
var dest = options.dest || null; | |
delete options.dest; | |
if (!dest) { | |
var optionStringDest = optionStringsLong.length ? optionStringsLong[0] : optionStrings[0]; | |
dest = $$.trimChars(optionStringDest, this.prefixChars); | |
if (dest.length === 0) { | |
throw new Error( | |
format('dest= is required for options like "%s"', optionStrings.join(', ')) | |
); | |
} | |
dest = dest.replace(/-/g, '_'); | |
} | |
// return the updated keyword arguments | |
options.dest = dest; | |
options.optionStrings = optionStrings; | |
return options; | |
}; | |
ActionContainer.prototype._popActionClass = function (options, defaultValue) { | |
defaultValue = defaultValue || null; | |
var action = (options.action || defaultValue); | |
delete options.action; | |
var actionClass = this._registryGet('action', action, action); | |
return actionClass; | |
}; | |
ActionContainer.prototype._getHandler = function () { | |
var handlerString = this.conflictHandler; | |
var handlerFuncName = '_handleConflict' + $$.capitalize(handlerString); | |
var func = this[handlerFuncName]; | |
if (typeof func === 'undefined') { | |
var msg = 'invalid conflict resolution value: ' + handlerString; | |
throw new Error(msg); | |
} else { | |
return func; | |
} | |
}; | |
ActionContainer.prototype._checkConflict = function (action) { | |
var optionStringActions = this._optionStringActions; | |
var conflictOptionals = []; | |
// find all options that conflict with this option | |
// collect pairs, the string, and an existing action that it conflicts with | |
action.optionStrings.forEach(function (optionString) { | |
var conflOptional = optionStringActions[optionString]; | |
if (typeof conflOptional !== 'undefined') { | |
conflictOptionals.push([ optionString, conflOptional ]); | |
} | |
}); | |
if (conflictOptionals.length > 0) { | |
var conflictHandler = this._getHandler(); | |
conflictHandler.call(this, action, conflictOptionals); | |
} | |
}; | |
ActionContainer.prototype._handleConflictError = function (action, conflOptionals) { | |
var conflicts = conflOptionals.map(function (pair) { return pair[0]; }); | |
conflicts = conflicts.join(', '); | |
throw argumentErrorHelper( | |
action, | |
format('Conflicting option string(s): %s', conflicts) | |
); | |
}; | |
ActionContainer.prototype._handleConflictResolve = function (action, conflOptionals) { | |
// remove all conflicting options | |
var self = this; | |
conflOptionals.forEach(function (pair) { | |
var optionString = pair[0]; | |
var conflictingAction = pair[1]; | |
// remove the conflicting option string | |
var i = conflictingAction.optionStrings.indexOf(optionString); | |
if (i >= 0) { | |
conflictingAction.optionStrings.splice(i, 1); | |
} | |
delete self._optionStringActions[optionString]; | |
// if the option now has no option string, remove it from the | |
// container holding it | |
if (conflictingAction.optionStrings.length === 0) { | |
conflictingAction.container._removeAction(conflictingAction); | |
} | |
}); | |
}; | |