UNPKG

47.6 kBJavaScriptView Raw
1'use strict';
2
3Object.defineProperty(exports, "__esModule", {
4 value: true
5});
6exports.Plugin = undefined;
7
8var _keys = require('babel-runtime/core-js/object/keys');
9
10var _keys2 = _interopRequireDefault(_keys);
11
12var _regenerator = require('babel-runtime/regenerator');
13
14var _regenerator2 = _interopRequireDefault(_regenerator);
15
16var _stringify = require('babel-runtime/core-js/json/stringify');
17
18var _stringify2 = _interopRequireDefault(_stringify);
19
20var _asyncToGenerator2 = require('babel-runtime/helpers/asyncToGenerator');
21
22var _asyncToGenerator3 = _interopRequireDefault(_asyncToGenerator2);
23
24var _getIterator2 = require('babel-runtime/core-js/get-iterator');
25
26var _getIterator3 = _interopRequireDefault(_getIterator2);
27
28var _getPrototypeOf = require('babel-runtime/core-js/object/get-prototype-of');
29
30var _getPrototypeOf2 = _interopRequireDefault(_getPrototypeOf);
31
32var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck');
33
34var _classCallCheck3 = _interopRequireDefault(_classCallCheck2);
35
36var _createClass2 = require('babel-runtime/helpers/createClass');
37
38var _createClass3 = _interopRequireDefault(_createClass2);
39
40var _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn');
41
42var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2);
43
44var _inherits2 = require('babel-runtime/helpers/inherits');
45
46var _inherits3 = _interopRequireDefault(_inherits2);
47
48var _cliEngineCommand = require('cli-engine-command');
49
50var _cliEngineCommand2 = _interopRequireDefault(_cliEngineCommand);
51
52var _path = require('path');
53
54var _path2 = _interopRequireDefault(_path);
55
56var _yarn = require('./yarn');
57
58var _yarn2 = _interopRequireDefault(_yarn);
59
60function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
61
62var Cache = function (_Base) {
63 (0, _inherits3.default)(Cache, _Base);
64
65 function Cache() {
66 (0, _classCallCheck3.default)(this, Cache);
67 return (0, _possibleConstructorReturn3.default)(this, (Cache.__proto__ || (0, _getPrototypeOf2.default)(Cache)).apply(this, arguments));
68 }
69
70 (0, _createClass3.default)(Cache, [{
71 key: 'plugin',
72 value: function plugin(path) {
73 return this.cache.plugins[path];
74 }
75 }, {
76 key: 'updatePlugin',
77 value: function updatePlugin(path, plugin) {
78 this.constructor.updated = true;
79 this.cache.plugins[path] = plugin;
80 }
81 }, {
82 key: 'save',
83 value: function save() {
84 if (!this.constructor.updated) return;
85 try {
86 this.fs.writeJSONSync(this.file, this.cache);
87 } catch (err) {
88 this.warn(err);
89 }
90 }
91 }, {
92 key: 'file',
93 get: function get() {
94 return _path2.default.join(this.config.dirs.cache, 'plugins.json');
95 }
96 }, {
97 key: 'cache',
98 get: function get() {
99 if (this._cache) return this._cache;
100 var initial = { version: this.config.version, plugins: {} };
101 try {
102 this._cache = this.fs.readJSONSync(this.file);
103 } catch (err) {
104 if (err.code !== 'ENOENT') throw err;
105 this._cache = initial;
106 }
107 if (this._cache.version !== this.config.version) this._cache = initial;
108 return this._cache;
109 }
110 }]);
111 return Cache;
112}(_cliEngineCommand.Base);
113/* globals
114 Class
115*/
116
117Cache.updated = false;
118
119
120function undefaultTopic(t) {
121 if (t.default) return t.default;
122 return t;
123}
124
125function undefaultCommand(c) {
126 if (c.default && typeof c.default !== 'boolean') return c.default;
127 return c;
128}
129
130var Plugin = exports.Plugin = function (_Base2) {
131 (0, _inherits3.default)(Plugin, _Base2);
132
133 function Plugin(type, path, config, cache) {
134 (0, _classCallCheck3.default)(this, Plugin);
135
136 var _this2 = (0, _possibleConstructorReturn3.default)(this, (Plugin.__proto__ || (0, _getPrototypeOf2.default)(Plugin)).call(this, config));
137
138 _this2.cache = cache;
139 _this2.type = type;
140 _this2.path = path;
141 var p = _this2.fetch();
142 _this2.name = p.name;
143 _this2.version = p.version;
144 _this2.commands = p.commands;
145 _this2.topics = p.topics;
146 return _this2;
147 }
148
149 (0, _createClass3.default)(Plugin, [{
150 key: 'findCommand',
151 value: function findCommand(cmd) {
152 var c = this.commands.find(function (c) {
153 return c.id === cmd || (c.aliases || []).includes(cmd);
154 });
155 if (!c) return;
156 var topic = c.topic,
157 command = c.command;
158
159 var p = this.require();
160 var Command = (p.commands || []).map(undefaultCommand).find(function (d) {
161 return topic === d.topic && command === d.command;
162 });
163 return typeof Command === 'function' ? Command : this.buildCommand(Command);
164 }
165 }, {
166 key: 'findTopic',
167 value: function findTopic(name) {
168 var t = this.topics.find(function (t) {
169 return t.topic === name;
170 });
171 if (!t) return;
172 var Topic = (this.require().topics || []).find(function (t) {
173 return [t.topic, t.name].includes(name);
174 });
175 return typeof Topic === 'function' ? Topic : this.buildTopic(t);
176 }
177 }, {
178 key: 'buildTopic',
179 value: function buildTopic(t) {
180 var _class, _temp;
181
182 return _temp = _class = function (_Topic) {
183 (0, _inherits3.default)(_class, _Topic);
184
185 function _class() {
186 (0, _classCallCheck3.default)(this, _class);
187 return (0, _possibleConstructorReturn3.default)(this, (_class.__proto__ || (0, _getPrototypeOf2.default)(_class)).apply(this, arguments));
188 }
189
190 return _class;
191 }(_cliEngineCommand.Topic), _class.topic = t.topic, _class.description = t.description, _class.hidden = t.hidden, _temp;
192 }
193 }, {
194 key: 'buildCommand',
195 value: function buildCommand(c) {
196 var _class2, _temp2;
197
198 if (!c.topic) throw new Error('command has no topic');
199 var Base = c.needsApp || c.wantsApp ? _cliEngineCommand.mixins.app(_cliEngineCommand2.default, { required: !!c.needsApp }) : _cliEngineCommand2.default;
200 return _temp2 = _class2 = function (_Base3) {
201 (0, _inherits3.default)(_class2, _Base3);
202
203 function _class2() {
204 (0, _classCallCheck3.default)(this, _class2);
205 return (0, _possibleConstructorReturn3.default)(this, (_class2.__proto__ || (0, _getPrototypeOf2.default)(_class2)).apply(this, arguments));
206 }
207
208 (0, _createClass3.default)(_class2, [{
209 key: 'run',
210 value: function run() {
211 var ctx = {
212 supportsColor: this.color.enabled,
213 auth: {},
214 debug: this.config.debug,
215 flags: this.flags,
216 args: this.args,
217 // flow$ignore
218 app: this.app
219 };
220 if (c.needsAuth) {
221 ctx.auth.password = process.env.HEROKU_API_KEY;
222 if (!ctx.auth.password) {
223 var netrc = require('netrc')();
224 var host = netrc['api.heroku.com'];
225 if (host) ctx.auth.password = host.password;
226 }
227 if (!ctx.auth.password) throw new Error('Not logged in');
228 }
229 return c.run(ctx);
230 }
231 }]);
232 return _class2;
233 }(Base), _class2.topic = c.topic, _class2.command = c.command, _class2.description = c.description, _class2.hidden = c.hidden, _class2.args = c.args || [], _class2.flags = c.flags || [], _temp2;
234 }
235 }, {
236 key: 'fetch',
237 value: function fetch() {
238 var c = this.cache.plugin(this.path);
239 if (c) return c;
240 try {
241 return this.updatePlugin(this.require());
242 } catch (err) {
243 if (this.type === 'builtin') throw err;
244 this.warn(err);
245 return {
246 name: this.path,
247 version: '',
248 commands: [],
249 topics: []
250 };
251 }
252 }
253 }, {
254 key: 'updatePlugin',
255 value: function updatePlugin(plugin) {
256 var name = this.type === 'builtin' ? 'builtin' : this.pjson().name;
257 var version = this.type === 'builtin' ? this.config.version : this.pjson().version;
258 if (!plugin.commands) throw new Error('no commands found');
259 var commands = plugin.commands.map(undefaultCommand).map(function (c) {
260 return {
261 id: c.command ? c.topic + ':' + c.command : c.topic,
262 topic: c.topic,
263 command: c.command,
264 description: c.description,
265 args: c.args,
266 flags: c.flags,
267 help: c.help,
268 usage: c.usage,
269 hidden: !!c.hidden,
270 aliases: c.aliases
271 };
272 });
273 var topics = (plugin.topics || (plugin.topic ? [plugin.topic] : [])).map(undefaultTopic).map(function (t) {
274 return {
275 topic: t.topic || t.name || '',
276 description: t.description,
277 hidden: !!t.hidden
278 };
279 });
280
281 var _iteratorNormalCompletion = true;
282 var _didIteratorError = false;
283 var _iteratorError = undefined;
284
285 try {
286 var _loop = function _loop() {
287 var command = _step.value;
288
289 if (topics.find(function (t) {
290 return t.topic === command.topic;
291 })) return 'continue';
292 topics.push({
293 topic: command.topic,
294 hidden: true
295 });
296 };
297
298 for (var _iterator = (0, _getIterator3.default)(commands), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
299 var _ret = _loop();
300
301 if (_ret === 'continue') continue;
302 }
303 } catch (err) {
304 _didIteratorError = true;
305 _iteratorError = err;
306 } finally {
307 try {
308 if (!_iteratorNormalCompletion && _iterator.return) {
309 _iterator.return();
310 }
311 } finally {
312 if (_didIteratorError) {
313 throw _iteratorError;
314 }
315 }
316 }
317
318 var cachedPlugin = { name: name, version: version, commands: commands, topics: topics };
319 this.cache.updatePlugin(this.path, cachedPlugin);
320 return cachedPlugin;
321 }
322
323 // flow$ignore
324
325 }, {
326 key: 'require',
327 value: function (_require) {
328 function require() {
329 return _require.apply(this, arguments);
330 }
331
332 require.toString = function () {
333 return _require.toString();
334 };
335
336 return require;
337 }(function () {
338 return require(this.path);
339 })
340 // flow$ignore
341
342 }, {
343 key: 'pjson',
344 value: function pjson() {
345 return require(_path2.default.join(this.path, 'package.json'));
346 }
347 }]);
348 return Plugin;
349}(_cliEngineCommand.Base);
350
351var Plugins = function (_Base4) {
352 (0, _inherits3.default)(Plugins, _Base4);
353
354 function Plugins(config) {
355 (0, _classCallCheck3.default)(this, Plugins);
356
357 var _this5 = (0, _possibleConstructorReturn3.default)(this, (Plugins.__proto__ || (0, _getPrototypeOf2.default)(Plugins)).call(this, config));
358
359 _this5.config = config;
360 _this5.cache = new Cache(config);
361 _this5.plugins = [new Plugin('builtin', './commands', config, _this5.cache)].concat(_this5.corePlugins).concat(_this5.userPlugins);
362 _this5.cache.save();
363 _this5.yarn = new _yarn2.default(_this5.config);
364 return _this5;
365 }
366
367 (0, _createClass3.default)(Plugins, [{
368 key: 'list',
369 value: function list() {
370 return this.plugins;
371 }
372 }, {
373 key: 'findCommand',
374 value: function findCommand(cmd) {
375 var _iteratorNormalCompletion2 = true;
376 var _didIteratorError2 = false;
377 var _iteratorError2 = undefined;
378
379 try {
380 for (var _iterator2 = (0, _getIterator3.default)(this.plugins), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
381 var plugin = _step2.value;
382
383 var c = plugin.findCommand(cmd);
384 if (c) return c;
385 }
386 } catch (err) {
387 _didIteratorError2 = true;
388 _iteratorError2 = err;
389 } finally {
390 try {
391 if (!_iteratorNormalCompletion2 && _iterator2.return) {
392 _iterator2.return();
393 }
394 } finally {
395 if (_didIteratorError2) {
396 throw _iteratorError2;
397 }
398 }
399 }
400 }
401 }, {
402 key: 'commandsForTopic',
403 value: function commandsForTopic(topic) {
404 return this.plugins.reduce(function (t, p) {
405 return t.concat(p.commands.filter(function (c) {
406 return c.topic === topic;
407 }).map(function (c) {
408 return p.findCommand(c.id);
409 }));
410 }, []);
411 }
412 }, {
413 key: 'findTopic',
414 value: function findTopic(cmd) {
415 var name = cmd.split(':')[0];
416 var _iteratorNormalCompletion3 = true;
417 var _didIteratorError3 = false;
418 var _iteratorError3 = undefined;
419
420 try {
421 for (var _iterator3 = (0, _getIterator3.default)(this.plugins), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) {
422 var plugin = _step3.value;
423
424 var t = plugin.findTopic(name);
425 if (t) return t;
426 }
427 } catch (err) {
428 _didIteratorError3 = true;
429 _iteratorError3 = err;
430 } finally {
431 try {
432 if (!_iteratorNormalCompletion3 && _iterator3.return) {
433 _iterator3.return();
434 }
435 } finally {
436 if (_didIteratorError3) {
437 throw _iteratorError3;
438 }
439 }
440 }
441 }
442 }, {
443 key: 'setupUserPlugins',
444 value: function () {
445 var _ref = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee() {
446 var pjson;
447 return _regenerator2.default.wrap(function _callee$(_context) {
448 while (1) {
449 switch (_context.prev = _context.next) {
450 case 0:
451 pjson = _path2.default.join(this.userPluginsDir, 'package.json');
452
453 this.fs.mkdirpSync(this.userPluginsDir);
454 if (!this.fs.existsSync(pjson)) this.fs.writeFileSync(pjson, (0, _stringify2.default)({ private: true }));
455 _context.next = 5;
456 return this.yarn.exec();
457
458 case 5:
459 case 'end':
460 return _context.stop();
461 }
462 }
463 }, _callee, this);
464 }));
465
466 function setupUserPlugins() {
467 return _ref.apply(this, arguments);
468 }
469
470 return setupUserPlugins;
471 }()
472 }, {
473 key: 'install',
474 value: function () {
475 var _ref2 = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee2(name) {
476 var plugin;
477 return _regenerator2.default.wrap(function _callee2$(_context2) {
478 while (1) {
479 switch (_context2.prev = _context2.next) {
480 case 0:
481 _context2.next = 2;
482 return this.setupUserPlugins();
483
484 case 2:
485 if (!this.config.debug) this.action.start('Installing plugin ' + name);
486 _context2.next = 5;
487 return this.yarn.exec('add', name);
488
489 case 5:
490 this.clearCache(name);
491 _context2.prev = 6;
492
493 // flow$ignore
494 plugin = require(this.userPluginPath(name));
495
496 if (plugin.commands) {
497 _context2.next = 10;
498 break;
499 }
500
501 throw new Error(name + ' does not appear to be a Heroku CLI plugin');
502
503 case 10:
504 _context2.next = 18;
505 break;
506
507 case 12:
508 _context2.prev = 12;
509 _context2.t0 = _context2['catch'](6);
510
511 this.error(_context2.t0, false);
512 _context2.next = 17;
513 return this.uninstall(name);
514
515 case 17:
516 this.exit(1);
517
518 case 18:
519 this.action.stop();
520
521 case 19:
522 case 'end':
523 return _context2.stop();
524 }
525 }
526 }, _callee2, this, [[6, 12]]);
527 }));
528
529 function install(_x) {
530 return _ref2.apply(this, arguments);
531 }
532
533 return install;
534 }()
535 }, {
536 key: 'uninstall',
537 value: function () {
538 var _ref3 = (0, _asyncToGenerator3.default)(_regenerator2.default.mark(function _callee3(name) {
539 return _regenerator2.default.wrap(function _callee3$(_context3) {
540 while (1) {
541 switch (_context3.prev = _context3.next) {
542 case 0:
543 if (!this.config.debug) this.action.start('Uninstalling plugin ' + name);
544 _context3.next = 3;
545 return this.yarn.exec('remove', name);
546
547 case 3:
548 case 'end':
549 return _context3.stop();
550 }
551 }
552 }, _callee3, this);
553 }));
554
555 function uninstall(_x2) {
556 return _ref3.apply(this, arguments);
557 }
558
559 return uninstall;
560 }()
561 }, {
562 key: 'clearCache',
563 value: function clearCache(name) {
564 var _iteratorNormalCompletion4 = true;
565 var _didIteratorError4 = false;
566 var _iteratorError4 = undefined;
567
568 try {
569 for (var _iterator4 = (0, _getIterator3.default)((0, _keys2.default)(this.cache.cache.plugins)), _step4; !(_iteratorNormalCompletion4 = (_step4 = _iterator4.next()).done); _iteratorNormalCompletion4 = true) {
570 var k = _step4.value;
571
572 if (this.cache.cache.plugins[k].name === name) delete this.cache.cache[k];
573 }
574 } catch (err) {
575 _didIteratorError4 = true;
576 _iteratorError4 = err;
577 } finally {
578 try {
579 if (!_iteratorNormalCompletion4 && _iterator4.return) {
580 _iterator4.return();
581 }
582 } finally {
583 if (_didIteratorError4) {
584 throw _iteratorError4;
585 }
586 }
587 }
588
589 this.cache.save();
590 }
591 }, {
592 key: 'userPluginPath',
593 value: function userPluginPath(name) {
594 return _path2.default.join(this.userPluginsDir, 'node_modules', name);
595 }
596 }, {
597 key: 'corePlugins',
598 get: function get() {
599 var _this6 = this;
600
601 return (this.config._cli.plugins || []).map(function (name) {
602 return new Plugin('core', _path2.default.join(_this6.config.root, 'node_modules', name), _this6.config, _this6.cache);
603 });
604 }
605 }, {
606 key: 'userPlugins',
607 get: function get() {
608 var _this7 = this;
609
610 var pjson = this.userPluginsPJSON;
611 return (0, _keys2.default)(pjson.dependencies || {}).map(function (name) {
612 return new Plugin('user', _this7.userPluginPath(name), _this7.config, _this7.cache);
613 });
614 }
615 }, {
616 key: 'userPluginsPJSON',
617 get: function get() {
618 try {
619 return this.fs.readJSONSync(_path2.default.join(this.userPluginsDir, 'package.json'));
620 } catch (err) {
621 return { dependencies: {} };
622 }
623 }
624 }, {
625 key: 'userPluginsDir',
626 get: function get() {
627 return _path2.default.join(this.config.dirs.data, 'plugins');
628 }
629 }, {
630 key: 'topics',
631 get: function get() {
632 return this.plugins.reduce(function (t, p) {
633 return t.concat(p.topics);
634 }, []);
635 }
636 }]);
637 return Plugins;
638}(_cliEngineCommand.Base);
639
640exports.default = Plugins;
641//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/plugins.js"],"names":["Cache","path","cache","plugins","plugin","constructor","updated","fs","writeJSONSync","file","err","warn","join","config","dirs","_cache","initial","version","readJSONSync","code","undefaultTopic","t","default","undefaultCommand","c","Plugin","type","p","fetch","name","commands","topics","cmd","find","id","aliases","includes","topic","command","require","Command","map","d","buildCommand","Topic","buildTopic","description","hidden","Error","Base","needsApp","wantsApp","app","required","ctx","supportsColor","color","enabled","auth","debug","flags","args","needsAuth","password","process","env","HEROKU_API_KEY","netrc","host","run","updatePlugin","pjson","help","usage","push","cachedPlugin","Plugins","concat","corePlugins","userPlugins","save","yarn","findCommand","reduce","filter","split","findTopic","userPluginsDir","mkdirpSync","existsSync","writeFileSync","private","exec","setupUserPlugins","action","start","clearCache","userPluginPath","error","uninstall","exit","stop","k","_cli","root","userPluginsPJSON","dependencies","data"],"mappings":"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","file":"plugins.js","sourcesContent":["// @flow\n/* globals\n   Class\n*/\n\nimport Command, {Config, Base, Topic, mixins, type Flag, type Arg} from 'cli-engine-command'\nimport path from 'path'\nimport Yarn from './yarn'\n\ntype PluginType = | \"builtin\" | \"core\" | \"user\"\n\ntype LegacyCommand = {\n  topic: string,\n  command?: string,\n  aliases?: string[],\n  args: Arg[],\n  flags: Flag[],\n  description?: ?string,\n  help?: ?string,\n  usage?: ?string,\n  needsApp?: ?boolean,\n  needsAuth?: ?boolean,\n  hidden?: ?boolean,\n  default?: ?boolean,\n  run: (ctx: LegacyContext) => Promise<any>\n}\n\ntype LegacyContext = {\n  supportsColor: boolean\n}\n\ntype ParsedTopic = | {\n  name?: ?string,\n  topic?: ?string,\n  description?: ?string,\n  hidden?: ?boolean\n} | Class<Topic>\n\ntype ParsedCommand = | LegacyCommand | Class<Command>\n\ntype ParsedPlugin = {\n  topic?: ParsedTopic,\n  topics?: ParsedTopic[],\n  commands?: (ParsedCommand | {default: ParsedCommand})[]\n}\n\ntype CachedCommand = {\n  id: string,\n  topic: string,\n  command?: ?string,\n  aliases?: string[],\n  args: Arg[],\n  flags: Flag[],\n  description: ?string,\n  help?: ?string,\n  usage?: ?string,\n  hidden: boolean\n}\n\ntype CachedTopic = {\n  topic: string,\n  description?: ?string,\n  hidden: boolean\n}\n\ntype CachedPlugin = {\n  name: string,\n  version: string,\n  commands: CachedCommand[],\n  topics: CachedTopic[]\n}\n\ntype CacheData = {\n  version: string,\n  plugins: {[path: string]: CachedPlugin}\n}\n\ntype PJSON = {\n  dependencies?: { [name: string]: string }\n}\n\nclass Cache extends Base {\n  static updated = false\n  _cache: CacheData\n\n  get file (): string { return path.join(this.config.dirs.cache, 'plugins.json') }\n  get cache (): CacheData {\n    if (this._cache) return this._cache\n    let initial = {version: this.config.version, plugins: {}}\n    try {\n      this._cache = this.fs.readJSONSync(this.file)\n    } catch (err) {\n      if (err.code !== 'ENOENT') throw err\n      this._cache = initial\n    }\n    if (this._cache.version !== this.config.version) this._cache = initial\n    return this._cache\n  }\n\n  plugin (path: string): ?CachedPlugin { return this.cache.plugins[path] }\n  updatePlugin (path: string, plugin: CachedPlugin) {\n    this.constructor.updated = true\n    this.cache.plugins[path] = plugin\n  }\n\n  save () {\n    if (!this.constructor.updated) return\n    try {\n      this.fs.writeJSONSync(this.file, this.cache)\n    } catch (err) {\n      this.warn(err)\n    }\n  }\n}\n\nfunction undefaultTopic (t: (ParsedTopic | {default: ParsedTopic})): ParsedTopic {\n  if (t.default) return (t.default: any)\n  return t\n}\n\nfunction undefaultCommand (c: (ParsedCommand | {default: ParsedCommand})): ParsedCommand {\n  if (c.default && typeof c.default !== 'boolean') return (c.default: any)\n  return (c: any)\n}\n\nexport class Plugin extends Base {\n  constructor (type: PluginType, path: string, config: Config, cache: Cache) {\n    super(config)\n    this.cache = cache\n    this.type = type\n    this.path = path\n    let p = this.fetch()\n    this.name = p.name\n    this.version = p.version\n    this.commands = p.commands\n    this.topics = p.topics\n  }\n\n  type: PluginType\n  path: string\n  cache: Cache\n  name: string\n  version: string\n  commands: CachedCommand[]\n  topics: CachedTopic[]\n\n  findCommand (cmd: string): ?Class<Command> {\n    let c = this.commands.find(c => c.id === cmd || (c.aliases || []).includes(cmd))\n    if (!c) return\n    let {topic, command} = c\n    let p = this.require()\n    let Command = (p.commands || [])\n      .map(undefaultCommand)\n      .find(d => topic === d.topic && command === d.command)\n    return typeof Command === 'function' ? Command : this.buildCommand((Command: any))\n  }\n\n  findTopic (name: string): ?Class<Topic> {\n    let t = this.topics.find(t => t.topic === name)\n    if (!t) return\n    let Topic = (this.require().topics || [])\n      .find(t => [t.topic, t.name].includes(name))\n    return typeof Topic === 'function' ? Topic : this.buildTopic(t)\n  }\n\n  buildTopic (t: CachedTopic): Class<Topic> {\n    return class extends Topic {\n      static topic = t.topic\n      static description = t.description\n      static hidden = t.hidden\n    }\n  }\n\n  buildCommand (c: LegacyCommand): Class<Command> {\n    if (!c.topic) throw new Error('command has no topic')\n    let Base = (c.needsApp || c.wantsApp)\n    ? mixins.app(Command, {required: !!c.needsApp})\n    : Command\n    return class extends Base {\n      static topic = c.topic\n      static command = c.command\n      static description = c.description\n      static hidden = c.hidden\n      static args = c.args || []\n      static flags = c.flags || []\n\n      run () {\n        const ctx = {\n          supportsColor: this.color.enabled,\n          auth: {},\n          debug: this.config.debug,\n          flags: this.flags,\n          args: this.args,\n          // flow$ignore\n          app: this.app\n        }\n        if (c.needsAuth) {\n          ctx.auth.password = process.env.HEROKU_API_KEY\n          if (!ctx.auth.password) {\n            const netrc = require('netrc')()\n            const host = netrc['api.heroku.com']\n            if (host) ctx.auth.password = host.password\n          }\n          if (!ctx.auth.password) throw new Error('Not logged in')\n        }\n        return c.run(ctx)\n      }\n    }\n  }\n\n  fetch (): CachedPlugin {\n    let c = this.cache.plugin(this.path)\n    if (c) return c\n    try {\n      return this.updatePlugin(this.require())\n    } catch (err) {\n      if (this.type === 'builtin') throw err\n      this.warn(err)\n      return {\n        name: this.path,\n        version: '',\n        commands: [],\n        topics: []\n      }\n    }\n  }\n\n  updatePlugin (plugin: ParsedPlugin): CachedPlugin {\n    const name = this.type === 'builtin' ? 'builtin' : this.pjson().name\n    const version = this.type === 'builtin' ? this.config.version : this.pjson().version\n    if (!plugin.commands) throw new Error('no commands found')\n    const commands: CachedCommand[] = plugin.commands\n    .map(undefaultCommand)\n    .map(c => ({\n      id: c.command ? `${c.topic}:${c.command}` : c.topic,\n      topic: c.topic,\n      command: c.command,\n      description: c.description,\n      args: c.args,\n      flags: c.flags,\n      help: c.help,\n      usage: c.usage,\n      hidden: !!c.hidden,\n      aliases: c.aliases\n    }))\n    const topics: CachedTopic[] = (plugin.topics || (plugin.topic ? [plugin.topic] : []))\n    .map(undefaultTopic)\n    .map(t => ({\n      topic: t.topic || t.name || '',\n      description: t.description,\n      hidden: !!t.hidden\n    }))\n\n    for (let command of commands) {\n      if (topics.find(t => t.topic === command.topic)) continue\n      topics.push({\n        topic: command.topic,\n        hidden: true\n      })\n    }\n\n    const cachedPlugin: CachedPlugin = {name, version, commands, topics}\n    this.cache.updatePlugin(this.path, cachedPlugin)\n    return cachedPlugin\n  }\n\n  // flow$ignore\n  require (): ParsedPlugin { return require(this.path) }\n  // flow$ignore\n  pjson (): {name: string, version: string} { return require(path.join(this.path, 'package.json')) }\n}\n\nexport default class Plugins extends Base {\n  constructor (config: Config) {\n    super(config)\n    this.config = config\n    this.cache = new Cache(config)\n    this.plugins = [new Plugin('builtin', './commands', config, this.cache)]\n    .concat(this.corePlugins)\n    .concat(this.userPlugins)\n    this.cache.save()\n    this.yarn = new Yarn(this.config)\n  }\n\n  plugins: Plugin[]\n  cache: Cache\n  yarn: Yarn\n\n  get corePlugins (): Plugin[] {\n    return (this.config._cli.plugins || []).map(name => {\n      return new Plugin('core', path.join(this.config.root, 'node_modules', name), this.config, this.cache)\n    })\n  }\n\n  get userPlugins (): Plugin[] {\n    const pjson = this.userPluginsPJSON\n    return Object.keys(pjson.dependencies || {}).map(name => {\n      return new Plugin('user', this.userPluginPath(name), this.config, this.cache)\n    })\n  }\n\n  get userPluginsPJSON (): PJSON {\n    try {\n      return this.fs.readJSONSync(path.join(this.userPluginsDir, 'package.json'))\n    } catch (err) {\n      return { dependencies: {} }\n    }\n  }\n\n  list () {\n    return this.plugins\n  }\n\n  findCommand (cmd: string): ?Class<Command> {\n    for (let plugin of this.plugins) {\n      let c = plugin.findCommand(cmd)\n      if (c) return c\n    }\n  }\n\n  commandsForTopic (topic: string): Class<Command>[] {\n    return this.plugins.reduce((t, p) => {\n      return t.concat(p.commands\n        .filter(c => c.topic === topic)\n        .map(c => (p.findCommand(c.id): any)))\n    }, [])\n  }\n\n  findTopic (cmd: string): ?Class<Topic> {\n    let name = cmd.split(':')[0]\n    for (let plugin of this.plugins) {\n      let t = plugin.findTopic(name)\n      if (t) return t\n    }\n  }\n\n  async setupUserPlugins () {\n    const pjson = path.join(this.userPluginsDir, 'package.json')\n    this.fs.mkdirpSync(this.userPluginsDir)\n    if (!this.fs.existsSync(pjson)) this.fs.writeFileSync(pjson, JSON.stringify({private: true}))\n    await this.yarn.exec()\n  }\n\n  async install (name: string) {\n    await this.setupUserPlugins()\n    if (!this.config.debug) this.action.start(`Installing plugin ${name}`)\n    await this.yarn.exec('add', name)\n    this.clearCache(name)\n    try {\n      // flow$ignore\n      let plugin = (require(this.userPluginPath(name)): ParsedPlugin)\n      if (!plugin.commands) throw new Error(`${name} does not appear to be a Heroku CLI plugin`)\n    } catch (err) {\n      this.error(err, false)\n      await this.uninstall(name)\n      this.exit(1)\n    }\n    this.action.stop()\n  }\n\n  async uninstall (name: string) {\n    if (!this.config.debug) this.action.start(`Uninstalling plugin ${name}`)\n    await this.yarn.exec('remove', name)\n  }\n\n  clearCache (name: string) {\n    for (let k of Object.keys(this.cache.cache.plugins)) {\n      if (this.cache.cache.plugins[k].name === name) delete this.cache.cache[k]\n    }\n    this.cache.save()\n  }\n\n  get userPluginsDir (): string { return path.join(this.config.dirs.data, 'plugins') }\n  userPluginPath (name: string): string { return path.join(this.userPluginsDir, 'node_modules', name) }\n\n  get topics (): CachedTopic[] {\n    return this.plugins.reduce((t, p) => t.concat(p.topics), [])\n  }\n\n  config: Config\n}\n"]}
\No newline at end of file