| /** | 
|  * 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; | 
| }; |