-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathcommands.ts
143 lines (122 loc) · 4.36 KB
/
commands.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
import {CliUx, Command, Flags, toConfiguredId} from '@oclif/core'
import * as _ from 'lodash'
import {EOL} from 'os'
import createCommandTree from '../utils/tree'
type Dictionary = {[index: string]: object}
export default class Commands extends Command {
static description = 'list all the commands'
static enableJsonFlag = true
static flags: any = {
help: Flags.help({char: 'h'}),
hidden: Flags.boolean({description: 'show hidden commands'}),
tree: Flags.boolean({description: 'show tree of commands'}),
...CliUx.ux.table.flags(),
}
async run() {
const {flags} = await this.parse(Commands)
let commands = this.getCommands()
if (!flags.hidden) {
commands = commands.filter(c => !c.hidden)
}
const config = this.config
commands = _.sortBy(commands, 'id').map(command => {
// Template supported fields.
command.description = (typeof command.description === 'string' && _.template(command.description)({command, config})) || undefined
command.summary = (typeof command.summary === 'string' && _.template(command.summary)({command, config})) || undefined
command.usage = (typeof command.usage === 'string' && _.template(command.usage)({command, config})) || undefined
command.id = toConfiguredId(command.id, this.config)
return command
})
if (this.jsonEnabled() && !flags.tree) {
const formatted = await Promise.all(commands.map(async cmd => {
let commandClass = await cmd.load()
const obj = {...cmd, ...commandClass}
// Load all properties on all extending classes.
while (commandClass !== undefined) {
commandClass = Object.getPrototypeOf(commandClass) || undefined
Object.assign(obj, commandClass)
}
// The plugin property on the loaded class contains a LOT of information including all the commands again. Remove it.
delete obj.plugin
// If Command classes have circular references, don't break the commands command.
return this.removeCycles(obj)
}))
return formatted
}
if (flags.tree) {
const tree = createCommandTree(commands, this.config.topicSeparator)
if (!this.jsonEnabled()) {
tree.display()
}
return tree
}
CliUx.ux.table(commands.map(command => {
// Massage some fields so it looks good in the table
command.description = (command.description || '').split(EOL)[0]
command.summary = (command.summary || (command.description || '').split(EOL)[0])
command.hidden = Boolean(command.hidden)
command.usage = (command.usage || '')
return command as unknown as Record<string, unknown>
}), {
id: {
header: 'Command',
},
summary: {},
description: {
extended: true,
},
usage: {
extended: true,
},
pluginName: {
extended: true,
header: 'Plugin',
},
pluginType: {
extended: true,
header: 'Type',
},
hidden: {
extended: true,
},
}, {
// to-do: investigate this oclif/core error when printLine is enabled
// printLine: this.log,
...flags, // parsed flags
})
}
private getCommands() {
return this.config.commands
}
private removeCycles(object: unknown) {
// Keep track of seen objects.
const seenObjects = new WeakMap<Dictionary, undefined>()
const _removeCycles = (obj: unknown) => {
// Use object prototype to get around type and null checks
if (Object.prototype.toString.call(obj) === '[object Object]') {
// We know it is a "Dictionary" because of the conditional
const dictionary = obj as Dictionary
if (seenObjects.has(dictionary)) {
// Seen, return undefined to remove.
return undefined
}
seenObjects.set(dictionary, undefined)
for (const key in dictionary) {
// Delete the duplicate object if cycle found.
if (_removeCycles(dictionary[key]) === undefined) {
delete dictionary[key]
}
}
} else if (Array.isArray(obj)) {
for (const i in obj) {
if (_removeCycles(obj[i]) === undefined) {
// We don't want to delete the array, but we can replace the element with null.
obj[i] = null
}
}
}
return obj
}
return _removeCycles(object)
}
}