blob: 944b4b79353b3dcc635827cbee1edb2dd3278d1e [file] [log] [blame]
Yang Guo4fd355c2019-09-19 10:59:03 +02001/**
2 * @fileoverview Main CLI object.
3 * @author Nicholas C. Zakas
4 */
5
6"use strict";
7
8/*
9 * The CLI object should *not* call process.exit() directly. It should only return
10 * exit codes. This allows other programs to use the CLI object and still control
11 * when the program exits.
12 */
13
14//------------------------------------------------------------------------------
15// Requirements
16//------------------------------------------------------------------------------
17
18const fs = require("fs"),
19 path = require("path"),
20 mkdirp = require("mkdirp"),
21 { CLIEngine } = require("./cli-engine"),
22 options = require("./options"),
Tim van der Lippec8f6ffd2020-04-06 13:42:00 +010023 log = require("./shared/logging"),
24 RuntimeInfo = require("./shared/runtime-info");
Yang Guo4fd355c2019-09-19 10:59:03 +020025
26const debug = require("debug")("eslint:cli");
27
28//------------------------------------------------------------------------------
29// Helpers
30//------------------------------------------------------------------------------
31
32/**
33 * Predicate function for whether or not to apply fixes in quiet mode.
34 * If a message is a warning, do not apply a fix.
35 * @param {LintResult} lintResult The lint result.
36 * @returns {boolean} True if the lint message is an error (and thus should be
37 * autofixed), false otherwise.
38 */
39function quietFixPredicate(lintResult) {
40 return lintResult.severity === 2;
41}
42
43/**
44 * Translates the CLI options into the options expected by the CLIEngine.
45 * @param {Object} cliOptions The CLI options to translate.
46 * @returns {CLIEngineOptions} The options object for the CLIEngine.
47 * @private
48 */
49function translateOptions(cliOptions) {
50 return {
51 envs: cliOptions.env,
52 extensions: cliOptions.ext,
53 rules: cliOptions.rule,
54 plugins: cliOptions.plugin,
55 globals: cliOptions.global,
56 ignore: cliOptions.ignore,
57 ignorePath: cliOptions.ignorePath,
58 ignorePattern: cliOptions.ignorePattern,
59 configFile: cliOptions.config,
60 rulePaths: cliOptions.rulesdir,
61 useEslintrc: cliOptions.eslintrc,
62 parser: cliOptions.parser,
63 parserOptions: cliOptions.parserOptions,
64 cache: cliOptions.cache,
65 cacheFile: cliOptions.cacheFile,
66 cacheLocation: cliOptions.cacheLocation,
67 fix: (cliOptions.fix || cliOptions.fixDryRun) && (cliOptions.quiet ? quietFixPredicate : true),
68 fixTypes: cliOptions.fixType,
69 allowInlineConfig: cliOptions.inlineConfig,
70 reportUnusedDisableDirectives: cliOptions.reportUnusedDisableDirectives,
Tim van der Lippec8f6ffd2020-04-06 13:42:00 +010071 resolvePluginsRelativeTo: cliOptions.resolvePluginsRelativeTo,
72 errorOnUnmatchedPattern: cliOptions.errorOnUnmatchedPattern
Yang Guo4fd355c2019-09-19 10:59:03 +020073 };
74}
75
76/**
77 * Outputs the results of the linting.
78 * @param {CLIEngine} engine The CLIEngine to use.
79 * @param {LintResult[]} results The results to print.
80 * @param {string} format The name of the formatter to use or the path to the formatter.
81 * @param {string} outputFile The path for the output file.
82 * @returns {boolean} True if the printing succeeds, false if not.
83 * @private
84 */
85function printResults(engine, results, format, outputFile) {
86 let formatter;
87 let rulesMeta;
88
89 try {
90 formatter = engine.getFormatter(format);
91 } catch (e) {
92 log.error(e.message);
93 return false;
94 }
95
96 const output = formatter(results, {
97 get rulesMeta() {
98 if (!rulesMeta) {
99 rulesMeta = {};
100 for (const [ruleId, rule] of engine.getRules()) {
101 rulesMeta[ruleId] = rule.meta;
102 }
103 }
104 return rulesMeta;
105 }
106 });
107
108 if (output) {
109 if (outputFile) {
110 const filePath = path.resolve(process.cwd(), outputFile);
111
112 if (fs.existsSync(filePath) && fs.statSync(filePath).isDirectory()) {
113 log.error("Cannot write to output file path, it is a directory: %s", outputFile);
114 return false;
115 }
116
117 try {
118 mkdirp.sync(path.dirname(filePath));
119 fs.writeFileSync(filePath, output);
120 } catch (ex) {
121 log.error("There was a problem writing the output file:\n%s", ex);
122 return false;
123 }
124 } else {
125 log.info(output);
126 }
127 }
128
129 return true;
130
131}
132
133//------------------------------------------------------------------------------
134// Public Interface
135//------------------------------------------------------------------------------
136
137/**
138 * Encapsulates all CLI behavior for eslint. Makes it easier to test as well as
139 * for other Node.js programs to effectively run the CLI.
140 */
141const cli = {
142
143 /**
144 * Executes the CLI based on an array of arguments that is passed in.
145 * @param {string|Array|Object} args The arguments to process.
146 * @param {string} [text] The text to lint (used for TTY).
147 * @returns {int} The exit code for the operation.
148 */
149 execute(args, text) {
150 if (Array.isArray(args)) {
151 debug("CLI args: %o", args.slice(2));
152 }
153
154 let currentOptions;
155
156 try {
157 currentOptions = options.parse(args);
158 } catch (error) {
159 log.error(error.message);
160 return 2;
161 }
162
163 const files = currentOptions._;
Yang Guo4fd355c2019-09-19 10:59:03 +0200164 const useStdin = typeof text === "string";
165
Tim van der Lippec8f6ffd2020-04-06 13:42:00 +0100166 if (currentOptions.version) {
167 log.info(RuntimeInfo.version());
168 } else if (currentOptions.envInfo) {
169 try {
170 log.info(RuntimeInfo.environment());
171 return 0;
172 } catch (err) {
173 log.error(err.message);
174 return 2;
175 }
Yang Guo4fd355c2019-09-19 10:59:03 +0200176 } else if (currentOptions.printConfig) {
177 if (files.length) {
178 log.error("The --print-config option must be used with exactly one file name.");
179 return 2;
180 }
181 if (useStdin) {
182 log.error("The --print-config option is not available for piped-in code.");
183 return 2;
184 }
185
186 const engine = new CLIEngine(translateOptions(currentOptions));
Yang Guo4fd355c2019-09-19 10:59:03 +0200187 const fileConfig = engine.getConfigForFile(currentOptions.printConfig);
188
189 log.info(JSON.stringify(fileConfig, null, " "));
190 return 0;
191 } else if (currentOptions.help || (!files.length && !useStdin)) {
Yang Guo4fd355c2019-09-19 10:59:03 +0200192 log.info(options.generateHelp());
Yang Guo4fd355c2019-09-19 10:59:03 +0200193 } else {
Yang Guo4fd355c2019-09-19 10:59:03 +0200194 debug(`Running on ${useStdin ? "text" : "files"}`);
195
196 if (currentOptions.fix && currentOptions.fixDryRun) {
197 log.error("The --fix option and the --fix-dry-run option cannot be used together.");
198 return 2;
199 }
200
201 if (useStdin && currentOptions.fix) {
202 log.error("The --fix option is not available for piped-in code; use --fix-dry-run instead.");
203 return 2;
204 }
205
206 if (currentOptions.fixType && !currentOptions.fix && !currentOptions.fixDryRun) {
207 log.error("The --fix-type option requires either --fix or --fix-dry-run.");
208 return 2;
209 }
210
211 const engine = new CLIEngine(translateOptions(currentOptions));
212 const report = useStdin ? engine.executeOnText(text, currentOptions.stdinFilename, true) : engine.executeOnFiles(files);
213
214 if (currentOptions.fix) {
215 debug("Fix mode enabled - applying fixes");
216 CLIEngine.outputFixes(report);
217 }
218
219 if (currentOptions.quiet) {
220 debug("Quiet mode enabled - filtering out warnings");
221 report.results = CLIEngine.getErrorResults(report.results);
222 }
223
224 if (printResults(engine, report.results, currentOptions.format, currentOptions.outputFile)) {
225 const tooManyWarnings = currentOptions.maxWarnings >= 0 && report.warningCount > currentOptions.maxWarnings;
226
227 if (!report.errorCount && tooManyWarnings) {
228 log.error("ESLint found too many warnings (maximum: %s).", currentOptions.maxWarnings);
229 }
230
231 return (report.errorCount || tooManyWarnings) ? 1 : 0;
232 }
Tim van der Lippec8f6ffd2020-04-06 13:42:00 +0100233
Yang Guo4fd355c2019-09-19 10:59:03 +0200234 return 2;
Yang Guo4fd355c2019-09-19 10:59:03 +0200235 }
236
237 return 0;
238 }
239};
240
241module.exports = cli;