-
-
Notifications
You must be signed in to change notification settings - Fork 289
/
cli.js
executable file
·224 lines (201 loc) · 5.71 KB
/
cli.js
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
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
#!/usr/bin/env node
/* eslint-disable unicorn/prefer-top-level-await -- TODO: Use top-level await */
import process from 'node:process';
import getStdin from 'get-stdin';
import meow from 'meow';
import formatterPretty from 'eslint-formatter-pretty';
import semver from 'semver';
import openReport from './lib/open-report.js';
import xo from './index.js';
const cli = meow(`
Usage
$ xo [<file|glob> ...]
Options
--fix Automagically fix issues
--reporter Reporter to use
--env Environment preset [Can be set multiple times]
--global Global variable [Can be set multiple times]
--ignore Additional paths to ignore [Can be set multiple times]
--space Use space indent instead of tabs [Default: 2]
--no-semicolon Prevent use of semicolons
--prettier Conform to Prettier code style
--node-version Range of Node.js version to support
--plugin Include third-party plugins [Can be set multiple times]
--extend Extend defaults with a custom config [Can be set multiple times]
--open Open files with issues in your editor
--quiet Show only errors and no warnings
--extension Additional extension to lint [Can be set multiple times]
--cwd=<dir> Working directory for files
--stdin Validate/fix code from stdin
--stdin-filename Specify a filename for the --stdin option
--print-config Print the effective ESLint config for the given file
Examples
$ xo
$ xo index.js
$ xo *.js !foo.js
$ xo --space
$ xo --env=node --env=mocha
$ xo --plugin=react
$ xo --plugin=html --extension=html
$ echo 'const x=true' | xo --stdin --fix
$ xo --print-config=index.js
Tips
- Add XO to your project with \`npm init xo\`.
- Put options in package.json instead of using flags so other tools can read it.
`, {
importMeta: import.meta,
autoVersion: false,
booleanDefault: undefined,
flags: {
fix: {
type: 'boolean',
},
reporter: {
type: 'string',
},
env: {
type: 'string',
isMultiple: true,
},
global: {
type: 'string',
isMultiple: true,
},
ignore: {
type: 'string',
isMultiple: true,
},
space: {
type: 'string',
},
semicolon: {
type: 'boolean',
},
prettier: {
type: 'boolean',
},
nodeVersion: {
type: 'string',
},
plugin: {
type: 'string',
isMultiple: true,
},
extend: {
type: 'string',
isMultiple: true,
},
open: {
type: 'boolean',
},
quiet: {
type: 'boolean',
},
extension: {
type: 'string',
isMultiple: true,
},
cwd: {
type: 'string',
},
printConfig: {
type: 'string',
},
stdin: {
type: 'boolean',
},
stdinFilename: {
type: 'string',
},
},
});
const {input, flags: options, showVersion} = cli;
// TODO: Fix this properly instead of the below workaround.
// Revert behavior of meow >8 to pre-8 (7.1.1) for flags using `isMultiple: true`.
// Otherwise, options defined in package.json can't be merged by lib/options-manager.js `mergeOptions()`.
for (const key in options) {
if (Array.isArray(options[key]) && options[key].length === 0) {
delete options[key];
}
}
// Make data types for `options.space` match those of the API
// Check for string type because `xo --no-space` sets `options.space` to `false`
if (typeof options.space === 'string') {
if (/^\d+$/u.test(options.space)) {
options.space = Number.parseInt(options.space, 10);
} else if (options.space === 'true') {
options.space = true;
} else if (options.space === 'false') {
options.space = false;
} else {
if (options.space !== '') {
// Assume `options.space` was set to a filename when run as `xo --space file.js`
input.push(options.space);
}
options.space = true;
}
}
if (process.env.GITHUB_ACTIONS && !options.fix && !options.reporter) {
options.quiet = true;
}
const log = async report => {
const reporter = options.reporter || process.env.GITHUB_ACTIONS ? await xo.getFormatter(options.reporter || 'compact') : formatterPretty;
process.stdout.write(reporter(report.results, {rulesMeta: report.rulesMeta}));
process.exitCode = report.errorCount === 0 ? 0 : 1;
};
// `xo -` => `xo --stdin`
if (input[0] === '-') {
options.stdin = true;
input.shift();
}
if (options.version) {
showVersion();
}
if (options.nodeVersion) {
if (options.nodeVersion === 'false') {
options.nodeVersion = false;
} else if (!semver.validRange(options.nodeVersion)) {
console.error('The `--node-engine` flag must be a valid semver range (for example `>=6`)');
process.exit(1);
}
}
(async () => {
if (typeof options.printConfig === 'string') {
if (input.length > 0 || options.printConfig === '') {
console.error('The `--print-config` flag must be used with exactly one filename');
process.exit(1);
}
if (options.stdin) {
console.error('The `--print-config` flag is not supported on stdin');
process.exit(1);
}
options.filePath = options.printConfig;
const config = await xo.getConfig(options);
console.log(JSON.stringify(config, undefined, '\t'));
} else if (options.stdin) {
const stdin = await getStdin();
if (options.stdinFilename) {
options.filePath = options.stdinFilename;
}
if (options.fix) {
const {results: [result]} = await xo.lintText(stdin, options);
// If there is no output, pass the stdin back out
process.stdout.write((result && result.output) || stdin);
return;
}
if (options.open) {
console.error('The `--open` flag is not supported on stdin');
process.exit(1);
}
await log(await xo.lintText(stdin, options));
} else {
const report = await xo.lintFiles(input, options);
if (options.fix) {
await xo.outputFixes(report);
}
if (options.open) {
openReport(report);
}
await log(report);
}
})();