|
|
/** * class HelpFormatter * * Formatter for generating usage messages and argument help strings. Only the * name of this class is considered a public API. All the methods provided by * the class are considered an implementation detail. * * Do not call in your code, use this class only for inherits your own forvatter * * ToDo add [additonal formatters][1] * * [1]:http://docs.python.org/dev/library/argparse.html#formatter-class
**/ 'use strict';
var sprintf = require('sprintf-js').sprintf;
// Constants
var c = require('../const');
var $$ = require('../utils');
/*:nodoc:* internal * new Support(parent, heding) * - parent (object): parent section * - heading (string): header string * **/ function Section(parent, heading) { this._parent = parent; this._heading = heading; this._items = []; }
/*:nodoc:* internal * Section#addItem(callback) -> Void * - callback (array): tuple with function and args * * Add function for single element **/ Section.prototype.addItem = function (callback) { this._items.push(callback); };
/*:nodoc:* internal * Section#formatHelp(formatter) -> string * - formatter (HelpFormatter): current formatter * * Form help section string * **/ Section.prototype.formatHelp = function (formatter) { var itemHelp, heading;
// format the indented section
if (this._parent) { formatter._indent(); }
itemHelp = this._items.map(function (item) { var obj, func, args;
obj = formatter; func = item[0]; args = item[1]; return func.apply(obj, args); }); itemHelp = formatter._joinParts(itemHelp);
if (this._parent) { formatter._dedent(); }
// return nothing if the section was empty
if (!itemHelp) { return ''; }
// add the heading if the section was non-empty
heading = ''; if (this._heading && this._heading !== c.SUPPRESS) { var currentIndent = formatter.currentIndent; heading = $$.repeat(' ', currentIndent) + this._heading + ':' + c.EOL; }
// join the section-initialize newline, the heading and the help
return formatter._joinParts([ c.EOL, heading, itemHelp, c.EOL ]); };
/** * new HelpFormatter(options) * * #### Options: * - `prog`: program name * - `indentIncriment`: indent step, default value 2 * - `maxHelpPosition`: max help position, default value = 24 * - `width`: line width * **/ var HelpFormatter = module.exports = function HelpFormatter(options) { options = options || {};
this._prog = options.prog;
this._maxHelpPosition = options.maxHelpPosition || 24; this._width = (options.width || ((process.env.COLUMNS || 80) - 2));
this._currentIndent = 0; this._indentIncriment = options.indentIncriment || 2; this._level = 0; this._actionMaxLength = 0;
this._rootSection = new Section(null); this._currentSection = this._rootSection;
this._whitespaceMatcher = new RegExp('\\s+', 'g'); this._longBreakMatcher = new RegExp(c.EOL + c.EOL + c.EOL + '+', 'g'); };
HelpFormatter.prototype._indent = function () { this._currentIndent += this._indentIncriment; this._level += 1; };
HelpFormatter.prototype._dedent = function () { this._currentIndent -= this._indentIncriment; this._level -= 1; if (this._currentIndent < 0) { throw new Error('Indent decreased below 0.'); } };
HelpFormatter.prototype._addItem = function (func, args) { this._currentSection.addItem([ func, args ]); };
//
// Message building methods
//
/** * HelpFormatter#startSection(heading) -> Void * - heading (string): header string * * Start new help section * * See alse [code example][1] * * ##### Example * * formatter.startSection(actionGroup.title); * formatter.addText(actionGroup.description); * formatter.addArguments(actionGroup._groupActions); * formatter.endSection(); * **/ HelpFormatter.prototype.startSection = function (heading) { this._indent(); var section = new Section(this._currentSection, heading); var func = section.formatHelp.bind(section); this._addItem(func, [ this ]); this._currentSection = section; };
/** * HelpFormatter#endSection -> Void * * End help section * * ##### Example * * formatter.startSection(actionGroup.title); * formatter.addText(actionGroup.description); * formatter.addArguments(actionGroup._groupActions); * formatter.endSection(); **/ HelpFormatter.prototype.endSection = function () { this._currentSection = this._currentSection._parent; this._dedent(); };
/** * HelpFormatter#addText(text) -> Void * - text (string): plain text * * Add plain text into current section * * ##### Example * * formatter.startSection(actionGroup.title); * formatter.addText(actionGroup.description); * formatter.addArguments(actionGroup._groupActions); * formatter.endSection(); * **/ HelpFormatter.prototype.addText = function (text) { if (text && text !== c.SUPPRESS) { this._addItem(this._formatText, [ text ]); } };
/** * HelpFormatter#addUsage(usage, actions, groups, prefix) -> Void * - usage (string): usage text * - actions (array): actions list * - groups (array): groups list * - prefix (string): usage prefix * * Add usage data into current section * * ##### Example * * formatter.addUsage(this.usage, this._actions, []); * return formatter.formatHelp(); * **/ HelpFormatter.prototype.addUsage = function (usage, actions, groups, prefix) { if (usage !== c.SUPPRESS) { this._addItem(this._formatUsage, [ usage, actions, groups, prefix ]); } };
/** * HelpFormatter#addArgument(action) -> Void * - action (object): action * * Add argument into current section * * Single variant of [[HelpFormatter#addArguments]] **/ HelpFormatter.prototype.addArgument = function (action) { if (action.help !== c.SUPPRESS) { var self = this;
// find all invocations
var invocations = [ this._formatActionInvocation(action) ]; var invocationLength = invocations[0].length;
var actionLength;
if (action._getSubactions) { this._indent(); action._getSubactions().forEach(function (subaction) {
var invocationNew = self._formatActionInvocation(subaction); invocations.push(invocationNew); invocationLength = Math.max(invocationLength, invocationNew.length);
}); this._dedent(); }
// update the maximum item length
actionLength = invocationLength + this._currentIndent; this._actionMaxLength = Math.max(this._actionMaxLength, actionLength);
// add the item to the list
this._addItem(this._formatAction, [ action ]); } };
/** * HelpFormatter#addArguments(actions) -> Void * - actions (array): actions list * * Mass add arguments into current section * * ##### Example * * formatter.startSection(actionGroup.title); * formatter.addText(actionGroup.description); * formatter.addArguments(actionGroup._groupActions); * formatter.endSection(); * **/ HelpFormatter.prototype.addArguments = function (actions) { var self = this; actions.forEach(function (action) { self.addArgument(action); }); };
//
// Help-formatting methods
//
/** * HelpFormatter#formatHelp -> string * * Format help * * ##### Example * * formatter.addText(this.epilog); * return formatter.formatHelp(); * **/ HelpFormatter.prototype.formatHelp = function () { var help = this._rootSection.formatHelp(this); if (help) { help = help.replace(this._longBreakMatcher, c.EOL + c.EOL); help = $$.trimChars(help, c.EOL) + c.EOL; } return help; };
HelpFormatter.prototype._joinParts = function (partStrings) { return partStrings.filter(function (part) { return (part && part !== c.SUPPRESS); }).join(''); };
HelpFormatter.prototype._formatUsage = function (usage, actions, groups, prefix) { if (!prefix && typeof prefix !== 'string') { prefix = 'usage: '; }
actions = actions || []; groups = groups || [];
// if usage is specified, use that
if (usage) { usage = sprintf(usage, { prog: this._prog });
// if no optionals or positionals are available, usage is just prog
} else if (!usage && actions.length === 0) { usage = this._prog;
// if optionals and positionals are available, calculate usage
} else if (!usage) { var prog = this._prog; var optionals = []; var positionals = []; var actionUsage; var textWidth;
// split optionals from positionals
actions.forEach(function (action) { if (action.isOptional()) { optionals.push(action); } else { positionals.push(action); } });
// build full usage string
actionUsage = this._formatActionsUsage([].concat(optionals, positionals), groups); usage = [ prog, actionUsage ].join(' ');
// wrap the usage parts if it's too long
textWidth = this._width - this._currentIndent; if ((prefix.length + usage.length) > textWidth) {
// break usage into wrappable parts
var regexpPart = new RegExp('\\(.*?\\)+|\\[.*?\\]+|\\S+', 'g'); var optionalUsage = this._formatActionsUsage(optionals, groups); var positionalUsage = this._formatActionsUsage(positionals, groups);
var optionalParts = optionalUsage.match(regexpPart); var positionalParts = positionalUsage.match(regexpPart) || [];
if (optionalParts.join(' ') !== optionalUsage) { throw new Error('assert "optionalParts.join(\' \') === optionalUsage"'); } if (positionalParts.join(' ') !== positionalUsage) { throw new Error('assert "positionalParts.join(\' \') === positionalUsage"'); }
// helper for wrapping lines
/*eslint-disable func-style*/ // node 0.10 compat
var _getLines = function (parts, indent, prefix) { var lines = []; var line = [];
var lineLength = prefix ? prefix.length - 1 : indent.length - 1;
parts.forEach(function (part) { if (lineLength + 1 + part.length > textWidth) { lines.push(indent + line.join(' ')); line = []; lineLength = indent.length - 1; } line.push(part); lineLength += part.length + 1; });
if (line) { lines.push(indent + line.join(' ')); } if (prefix) { lines[0] = lines[0].substr(indent.length); } return lines; };
var lines, indent, parts; // if prog is short, follow it with optionals or positionals
if (prefix.length + prog.length <= 0.75 * textWidth) { indent = $$.repeat(' ', (prefix.length + prog.length + 1)); if (optionalParts) { lines = [].concat( _getLines([ prog ].concat(optionalParts), indent, prefix), _getLines(positionalParts, indent) ); } else if (positionalParts) { lines = _getLines([ prog ].concat(positionalParts), indent, prefix); } else { lines = [ prog ]; }
// if prog is long, put it on its own line
} else { indent = $$.repeat(' ', prefix.length); parts = optionalParts.concat(positionalParts); lines = _getLines(parts, indent); if (lines.length > 1) { lines = [].concat( _getLines(optionalParts, indent), _getLines(positionalParts, indent) ); } lines = [ prog ].concat(lines); } // join lines into usage
usage = lines.join(c.EOL); } }
// prefix with 'usage:'
return prefix + usage + c.EOL + c.EOL; };
HelpFormatter.prototype._formatActionsUsage = function (actions, groups) { // find group indices and identify actions in groups
var groupActions = []; var inserts = []; var self = this;
groups.forEach(function (group) { var end; var i;
var start = actions.indexOf(group._groupActions[0]); if (start >= 0) { end = start + group._groupActions.length;
//if (actions.slice(start, end) === group._groupActions) {
if ($$.arrayEqual(actions.slice(start, end), group._groupActions)) { group._groupActions.forEach(function (action) { groupActions.push(action); });
if (!group.required) { if (inserts[start]) { inserts[start] += ' ['; } else { inserts[start] = '['; } inserts[end] = ']'; } else { if (inserts[start]) { inserts[start] += ' ('; } else { inserts[start] = '('; } inserts[end] = ')'; } for (i = start + 1; i < end; i += 1) { inserts[i] = '|'; } } } });
// collect all actions format strings
var parts = [];
actions.forEach(function (action, actionIndex) { var part; var optionString; var argsDefault; var argsString;
// suppressed arguments are marked with None
// remove | separators for suppressed arguments
if (action.help === c.SUPPRESS) { parts.push(null); if (inserts[actionIndex] === '|') { inserts.splice(actionIndex, actionIndex); } else if (inserts[actionIndex + 1] === '|') { inserts.splice(actionIndex + 1, actionIndex + 1); }
// produce all arg strings
} else if (!action.isOptional()) { part = self._formatArgs(action, action.dest);
// if it's in a group, strip the outer []
if (groupActions.indexOf(action) >= 0) { if (part[0] === '[' && part[part.length - 1] === ']') { part = part.slice(1, -1); } } // add the action string to the list
parts.push(part);
// produce the first way to invoke the option in brackets
} else { optionString = action.optionStrings[0];
// if the Optional doesn't take a value, format is: -s or --long
if (action.nargs === 0) { part = '' + optionString;
// if the Optional takes a value, format is: -s ARGS or --long ARGS
} else { argsDefault = action.dest.toUpperCase(); argsString = self._formatArgs(action, argsDefault); part = optionString + ' ' + argsString; } // make it look optional if it's not required or in a group
if (!action.required && groupActions.indexOf(action) < 0) { part = '[' + part + ']'; } // add the action string to the list
parts.push(part); } });
// insert things at the necessary indices
for (var i = inserts.length - 1; i >= 0; --i) { if (inserts[i] !== null) { parts.splice(i, 0, inserts[i]); } }
// join all the action items with spaces
var text = parts.filter(function (part) { return !!part; }).join(' ');
// clean up separators for mutually exclusive groups
text = text.replace(/([\[(]) /g, '$1'); // remove spaces
text = text.replace(/ ([\])])/g, '$1'); text = text.replace(/\[ *\]/g, ''); // remove empty groups
text = text.replace(/\( *\)/g, ''); text = text.replace(/\(([^|]*)\)/g, '$1'); // remove () from single action groups
text = text.trim();
// return the text
return text; };
HelpFormatter.prototype._formatText = function (text) { text = sprintf(text, { prog: this._prog }); var textWidth = this._width - this._currentIndent; var indentIncriment = $$.repeat(' ', this._currentIndent); return this._fillText(text, textWidth, indentIncriment) + c.EOL + c.EOL; };
HelpFormatter.prototype._formatAction = function (action) { var self = this;
var helpText; var helpLines; var parts; var indentFirst;
// determine the required width and the entry label
var helpPosition = Math.min(this._actionMaxLength + 2, this._maxHelpPosition); var helpWidth = this._width - helpPosition; var actionWidth = helpPosition - this._currentIndent - 2; var actionHeader = this._formatActionInvocation(action);
// no help; start on same line and add a final newline
if (!action.help) { actionHeader = $$.repeat(' ', this._currentIndent) + actionHeader + c.EOL;
// short action name; start on the same line and pad two spaces
} else if (actionHeader.length <= actionWidth) { actionHeader = $$.repeat(' ', this._currentIndent) + actionHeader + ' ' + $$.repeat(' ', actionWidth - actionHeader.length); indentFirst = 0;
// long action name; start on the next line
} else { actionHeader = $$.repeat(' ', this._currentIndent) + actionHeader + c.EOL; indentFirst = helpPosition; }
// collect the pieces of the action help
parts = [ actionHeader ];
// if there was help for the action, add lines of help text
if (action.help) { helpText = this._expandHelp(action); helpLines = this._splitLines(helpText, helpWidth); parts.push($$.repeat(' ', indentFirst) + helpLines[0] + c.EOL); helpLines.slice(1).forEach(function (line) { parts.push($$.repeat(' ', helpPosition) + line + c.EOL); });
// or add a newline if the description doesn't end with one
} else if (actionHeader.charAt(actionHeader.length - 1) !== c.EOL) { parts.push(c.EOL); } // if there are any sub-actions, add their help as well
if (action._getSubactions) { this._indent(); action._getSubactions().forEach(function (subaction) { parts.push(self._formatAction(subaction)); }); this._dedent(); } // return a single string
return this._joinParts(parts); };
HelpFormatter.prototype._formatActionInvocation = function (action) { if (!action.isOptional()) { var format_func = this._metavarFormatter(action, action.dest); var metavars = format_func(1); return metavars[0]; }
var parts = []; var argsDefault; var argsString;
// if the Optional doesn't take a value, format is: -s, --long
if (action.nargs === 0) { parts = parts.concat(action.optionStrings);
// if the Optional takes a value, format is: -s ARGS, --long ARGS
} else { argsDefault = action.dest.toUpperCase(); argsString = this._formatArgs(action, argsDefault); action.optionStrings.forEach(function (optionString) { parts.push(optionString + ' ' + argsString); }); } return parts.join(', '); };
HelpFormatter.prototype._metavarFormatter = function (action, metavarDefault) { var result;
if (action.metavar || action.metavar === '') { result = action.metavar; } else if (action.choices) { var choices = action.choices;
if (typeof choices === 'string') { choices = choices.split('').join(', '); } else if (Array.isArray(choices)) { choices = choices.join(','); } else { choices = Object.keys(choices).join(','); } result = '{' + choices + '}'; } else { result = metavarDefault; }
return function (size) { if (Array.isArray(result)) { return result; }
var metavars = []; for (var i = 0; i < size; i += 1) { metavars.push(result); } return metavars; }; };
HelpFormatter.prototype._formatArgs = function (action, metavarDefault) { var result; var metavars;
var buildMetavar = this._metavarFormatter(action, metavarDefault);
switch (action.nargs) { /*eslint-disable no-undefined*/ case undefined: case null: metavars = buildMetavar(1); result = '' + metavars[0]; break; case c.OPTIONAL: metavars = buildMetavar(1); result = '[' + metavars[0] + ']'; break; case c.ZERO_OR_MORE: metavars = buildMetavar(2); result = '[' + metavars[0] + ' [' + metavars[1] + ' ...]]'; break; case c.ONE_OR_MORE: metavars = buildMetavar(2); result = '' + metavars[0] + ' [' + metavars[1] + ' ...]'; break; case c.REMAINDER: result = '...'; break; case c.PARSER: metavars = buildMetavar(1); result = metavars[0] + ' ...'; break; default: metavars = buildMetavar(action.nargs); result = metavars.join(' '); } return result; };
HelpFormatter.prototype._expandHelp = function (action) { var params = { prog: this._prog };
Object.keys(action).forEach(function (actionProperty) { var actionValue = action[actionProperty];
if (actionValue !== c.SUPPRESS) { params[actionProperty] = actionValue; } });
if (params.choices) { if (typeof params.choices === 'string') { params.choices = params.choices.split('').join(', '); } else if (Array.isArray(params.choices)) { params.choices = params.choices.join(', '); } else { params.choices = Object.keys(params.choices).join(', '); } }
return sprintf(this._getHelpString(action), params); };
HelpFormatter.prototype._splitLines = function (text, width) { var lines = []; var delimiters = [ ' ', '.', ',', '!', '?' ]; var re = new RegExp('[' + delimiters.join('') + '][^' + delimiters.join('') + ']*$');
text = text.replace(/[\n\|\t]/g, ' ');
text = text.trim(); text = text.replace(this._whitespaceMatcher, ' ');
// Wraps the single paragraph in text (a string) so every line
// is at most width characters long.
text.split(c.EOL).forEach(function (line) { if (width >= line.length) { lines.push(line); return; }
var wrapStart = 0; var wrapEnd = width; var delimiterIndex = 0; while (wrapEnd <= line.length) { if (wrapEnd !== line.length && delimiters.indexOf(line[wrapEnd] < -1)) { delimiterIndex = (re.exec(line.substring(wrapStart, wrapEnd)) || {}).index; wrapEnd = wrapStart + delimiterIndex + 1; } lines.push(line.substring(wrapStart, wrapEnd)); wrapStart = wrapEnd; wrapEnd += width; } if (wrapStart < line.length) { lines.push(line.substring(wrapStart, wrapEnd)); } });
return lines; };
HelpFormatter.prototype._fillText = function (text, width, indent) { var lines = this._splitLines(text, width); lines = lines.map(function (line) { return indent + line; }); return lines.join(c.EOL); };
HelpFormatter.prototype._getHelpString = function (action) { return action.help; };
|