index.js 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. 'use strict';
  2. function hasKey(obj, keys) {
  3. var o = obj;
  4. keys.slice(0, -1).forEach(function (key) {
  5. o = o[key] || {};
  6. });
  7. var key = keys[keys.length - 1];
  8. return key in o;
  9. }
  10. function isNumber(x) {
  11. if (typeof x === 'number') { return true; }
  12. if ((/^0x[0-9a-f]+$/i).test(x)) { return true; }
  13. return (/^[-+]?(?:\d+(?:\.\d*)?|\.\d+)(e[-+]?\d+)?$/).test(x);
  14. }
  15. function isConstructorOrProto(obj, key) {
  16. return (key === 'constructor' && typeof obj[key] === 'function') || key === '__proto__';
  17. }
  18. module.exports = function (args, opts) {
  19. if (!opts) { opts = {}; }
  20. var flags = {
  21. bools: {},
  22. strings: {},
  23. unknownFn: null,
  24. };
  25. if (typeof opts.unknown === 'function') {
  26. flags.unknownFn = opts.unknown;
  27. }
  28. if (typeof opts.boolean === 'boolean' && opts.boolean) {
  29. flags.allBools = true;
  30. } else {
  31. [].concat(opts.boolean).filter(Boolean).forEach(function (key) {
  32. flags.bools[key] = true;
  33. });
  34. }
  35. var aliases = {};
  36. function aliasIsBoolean(key) {
  37. return aliases[key].some(function (x) {
  38. return flags.bools[x];
  39. });
  40. }
  41. Object.keys(opts.alias || {}).forEach(function (key) {
  42. aliases[key] = [].concat(opts.alias[key]);
  43. aliases[key].forEach(function (x) {
  44. aliases[x] = [key].concat(aliases[key].filter(function (y) {
  45. return x !== y;
  46. }));
  47. });
  48. });
  49. [].concat(opts.string).filter(Boolean).forEach(function (key) {
  50. flags.strings[key] = true;
  51. if (aliases[key]) {
  52. [].concat(aliases[key]).forEach(function (k) {
  53. flags.strings[k] = true;
  54. });
  55. }
  56. });
  57. var defaults = opts.default || {};
  58. var argv = { _: [] };
  59. function argDefined(key, arg) {
  60. return (flags.allBools && (/^--[^=]+$/).test(arg))
  61. || flags.strings[key]
  62. || flags.bools[key]
  63. || aliases[key];
  64. }
  65. function setKey(obj, keys, value) {
  66. var o = obj;
  67. for (var i = 0; i < keys.length - 1; i++) {
  68. var key = keys[i];
  69. if (isConstructorOrProto(o, key)) { return; }
  70. if (o[key] === undefined) { o[key] = {}; }
  71. if (
  72. o[key] === Object.prototype
  73. || o[key] === Number.prototype
  74. || o[key] === String.prototype
  75. ) {
  76. o[key] = {};
  77. }
  78. if (o[key] === Array.prototype) { o[key] = []; }
  79. o = o[key];
  80. }
  81. var lastKey = keys[keys.length - 1];
  82. if (isConstructorOrProto(o, lastKey)) { return; }
  83. if (
  84. o === Object.prototype
  85. || o === Number.prototype
  86. || o === String.prototype
  87. ) {
  88. o = {};
  89. }
  90. if (o === Array.prototype) { o = []; }
  91. if (o[lastKey] === undefined || flags.bools[lastKey] || typeof o[lastKey] === 'boolean') {
  92. o[lastKey] = value;
  93. } else if (Array.isArray(o[lastKey])) {
  94. o[lastKey].push(value);
  95. } else {
  96. o[lastKey] = [o[lastKey], value];
  97. }
  98. }
  99. function setArg(key, val, arg) {
  100. if (arg && flags.unknownFn && !argDefined(key, arg)) {
  101. if (flags.unknownFn(arg) === false) { return; }
  102. }
  103. var value = !flags.strings[key] && isNumber(val)
  104. ? Number(val)
  105. : val;
  106. setKey(argv, key.split('.'), value);
  107. (aliases[key] || []).forEach(function (x) {
  108. setKey(argv, x.split('.'), value);
  109. });
  110. }
  111. Object.keys(flags.bools).forEach(function (key) {
  112. setArg(key, defaults[key] === undefined ? false : defaults[key]);
  113. });
  114. var notFlags = [];
  115. if (args.indexOf('--') !== -1) {
  116. notFlags = args.slice(args.indexOf('--') + 1);
  117. args = args.slice(0, args.indexOf('--'));
  118. }
  119. for (var i = 0; i < args.length; i++) {
  120. var arg = args[i];
  121. var key;
  122. var next;
  123. if ((/^--.+=/).test(arg)) {
  124. // Using [\s\S] instead of . because js doesn't support the
  125. // 'dotall' regex modifier. See:
  126. // http://stackoverflow.com/a/1068308/13216
  127. var m = arg.match(/^--([^=]+)=([\s\S]*)$/);
  128. key = m[1];
  129. var value = m[2];
  130. if (flags.bools[key]) {
  131. value = value !== 'false';
  132. }
  133. setArg(key, value, arg);
  134. } else if ((/^--no-.+/).test(arg)) {
  135. key = arg.match(/^--no-(.+)/)[1];
  136. setArg(key, false, arg);
  137. } else if ((/^--.+/).test(arg)) {
  138. key = arg.match(/^--(.+)/)[1];
  139. next = args[i + 1];
  140. if (
  141. next !== undefined
  142. && !(/^(-|--)[^-]/).test(next)
  143. && !flags.bools[key]
  144. && !flags.allBools
  145. && (aliases[key] ? !aliasIsBoolean(key) : true)
  146. ) {
  147. setArg(key, next, arg);
  148. i += 1;
  149. } else if ((/^(true|false)$/).test(next)) {
  150. setArg(key, next === 'true', arg);
  151. i += 1;
  152. } else {
  153. setArg(key, flags.strings[key] ? '' : true, arg);
  154. }
  155. } else if ((/^-[^-]+/).test(arg)) {
  156. var letters = arg.slice(1, -1).split('');
  157. var broken = false;
  158. for (var j = 0; j < letters.length; j++) {
  159. next = arg.slice(j + 2);
  160. if (next === '-') {
  161. setArg(letters[j], next, arg);
  162. continue;
  163. }
  164. if ((/[A-Za-z]/).test(letters[j]) && next[0] === '=') {
  165. setArg(letters[j], next.slice(1), arg);
  166. broken = true;
  167. break;
  168. }
  169. if (
  170. (/[A-Za-z]/).test(letters[j])
  171. && (/-?\d+(\.\d*)?(e-?\d+)?$/).test(next)
  172. ) {
  173. setArg(letters[j], next, arg);
  174. broken = true;
  175. break;
  176. }
  177. if (letters[j + 1] && letters[j + 1].match(/\W/)) {
  178. setArg(letters[j], arg.slice(j + 2), arg);
  179. broken = true;
  180. break;
  181. } else {
  182. setArg(letters[j], flags.strings[letters[j]] ? '' : true, arg);
  183. }
  184. }
  185. key = arg.slice(-1)[0];
  186. if (!broken && key !== '-') {
  187. if (
  188. args[i + 1]
  189. && !(/^(-|--)[^-]/).test(args[i + 1])
  190. && !flags.bools[key]
  191. && (aliases[key] ? !aliasIsBoolean(key) : true)
  192. ) {
  193. setArg(key, args[i + 1], arg);
  194. i += 1;
  195. } else if (args[i + 1] && (/^(true|false)$/).test(args[i + 1])) {
  196. setArg(key, args[i + 1] === 'true', arg);
  197. i += 1;
  198. } else {
  199. setArg(key, flags.strings[key] ? '' : true, arg);
  200. }
  201. }
  202. } else {
  203. if (!flags.unknownFn || flags.unknownFn(arg) !== false) {
  204. argv._.push(flags.strings._ || !isNumber(arg) ? arg : Number(arg));
  205. }
  206. if (opts.stopEarly) {
  207. argv._.push.apply(argv._, args.slice(i + 1));
  208. break;
  209. }
  210. }
  211. }
  212. Object.keys(defaults).forEach(function (k) {
  213. if (!hasKey(argv, k.split('.'))) {
  214. setKey(argv, k.split('.'), defaults[k]);
  215. (aliases[k] || []).forEach(function (x) {
  216. setKey(argv, x.split('.'), defaults[k]);
  217. });
  218. }
  219. });
  220. if (opts['--']) {
  221. argv['--'] = notFlags.slice();
  222. } else {
  223. notFlags.forEach(function (k) {
  224. argv._.push(k);
  225. });
  226. }
  227. return argv;
  228. };