You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

537 lines
18 KiB

4 years ago
  1. /**
  2. * @fileoverview `FileEnumerator` class.
  3. *
  4. * `FileEnumerator` class has two responsibilities:
  5. *
  6. * 1. Find target files by processing glob patterns.
  7. * 2. Tie each target file and appropriate configuration.
  8. *
  9. * It provides a method:
  10. *
  11. * - `iterateFiles(patterns)`
  12. * Iterate files which are matched by given patterns together with the
  13. * corresponded configuration. This is for `CLIEngine#executeOnFiles()`.
  14. * While iterating files, it loads the configuration file of each directory
  15. * before iterate files on the directory, so we can use the configuration
  16. * files to determine target files.
  17. *
  18. * @example
  19. * const enumerator = new FileEnumerator();
  20. * const linter = new Linter();
  21. *
  22. * for (const { config, filePath } of enumerator.iterateFiles(["*.js"])) {
  23. * const code = fs.readFileSync(filePath, "utf8");
  24. * const messages = linter.verify(code, config, filePath);
  25. *
  26. * console.log(messages);
  27. * }
  28. *
  29. * @author Toru Nagashima <https://github.com/mysticatea>
  30. */
  31. "use strict";
  32. //------------------------------------------------------------------------------
  33. // Requirements
  34. //------------------------------------------------------------------------------
  35. const fs = require("fs");
  36. const path = require("path");
  37. const getGlobParent = require("glob-parent");
  38. const isGlob = require("is-glob");
  39. const { escapeRegExp } = require("lodash");
  40. const { Minimatch } = require("minimatch");
  41. const {
  42. Legacy: {
  43. IgnorePattern,
  44. CascadingConfigArrayFactory
  45. }
  46. } = require("@eslint/eslintrc");
  47. const debug = require("debug")("eslint:file-enumerator");
  48. //------------------------------------------------------------------------------
  49. // Helpers
  50. //------------------------------------------------------------------------------
  51. const minimatchOpts = { dot: true, matchBase: true };
  52. const dotfilesPattern = /(?:(?:^\.)|(?:[/\\]\.))[^/\\.].*/u;
  53. const NONE = 0;
  54. const IGNORED_SILENTLY = 1;
  55. const IGNORED = 2;
  56. // For VSCode intellisense
  57. /** @typedef {ReturnType<CascadingConfigArrayFactory["getConfigArrayForFile"]>} ConfigArray */
  58. /**
  59. * @typedef {Object} FileEnumeratorOptions
  60. * @property {CascadingConfigArrayFactory} [configArrayFactory] The factory for config arrays.
  61. * @property {string} [cwd] The base directory to start lookup.
  62. * @property {string[]} [extensions] The extensions to match files for directory patterns.
  63. * @property {boolean} [globInputPaths] Set to false to skip glob resolution of input file paths to lint (default: true). If false, each input file paths is assumed to be a non-glob path to an existing file.
  64. * @property {boolean} [ignore] The flag to check ignored files.
  65. * @property {string[]} [rulePaths] The value of `--rulesdir` option.
  66. */
  67. /**
  68. * @typedef {Object} FileAndConfig
  69. * @property {string} filePath The path to a target file.
  70. * @property {ConfigArray} config The config entries of that file.
  71. * @property {boolean} ignored If `true` then this file should be ignored and warned because it was directly specified.
  72. */
  73. /**
  74. * @typedef {Object} FileEntry
  75. * @property {string} filePath The path to a target file.
  76. * @property {ConfigArray} config The config entries of that file.
  77. * @property {NONE|IGNORED_SILENTLY|IGNORED} flag The flag.
  78. * - `NONE` means the file is a target file.
  79. * - `IGNORED_SILENTLY` means the file should be ignored silently.
  80. * - `IGNORED` means the file should be ignored and warned because it was directly specified.
  81. */
  82. /**
  83. * @typedef {Object} FileEnumeratorInternalSlots
  84. * @property {CascadingConfigArrayFactory} configArrayFactory The factory for config arrays.
  85. * @property {string} cwd The base directory to start lookup.
  86. * @property {RegExp|null} extensionRegExp The RegExp to test if a string ends with specific file extensions.
  87. * @property {boolean} globInputPaths Set to false to skip glob resolution of input file paths to lint (default: true). If false, each input file paths is assumed to be a non-glob path to an existing file.
  88. * @property {boolean} ignoreFlag The flag to check ignored files.
  89. * @property {(filePath:string, dot:boolean) => boolean} defaultIgnores The default predicate function to ignore files.
  90. */
  91. /** @type {WeakMap<FileEnumerator, FileEnumeratorInternalSlots>} */
  92. const internalSlotsMap = new WeakMap();
  93. /**
  94. * Check if a string is a glob pattern or not.
  95. * @param {string} pattern A glob pattern.
  96. * @returns {boolean} `true` if the string is a glob pattern.
  97. */
  98. function isGlobPattern(pattern) {
  99. return isGlob(path.sep === "\\" ? pattern.replace(/\\/gu, "/") : pattern);
  100. }
  101. /**
  102. * Get stats of a given path.
  103. * @param {string} filePath The path to target file.
  104. * @returns {fs.Stats|null} The stats.
  105. * @private
  106. */
  107. function statSafeSync(filePath) {
  108. try {
  109. return fs.statSync(filePath);
  110. } catch (error) {
  111. /* istanbul ignore next */
  112. if (error.code !== "ENOENT") {
  113. throw error;
  114. }
  115. return null;
  116. }
  117. }
  118. /**
  119. * Get filenames in a given path to a directory.
  120. * @param {string} directoryPath The path to target directory.
  121. * @returns {import("fs").Dirent[]} The filenames.
  122. * @private
  123. */
  124. function readdirSafeSync(directoryPath) {
  125. try {
  126. return fs.readdirSync(directoryPath, { withFileTypes: true });
  127. } catch (error) {
  128. /* istanbul ignore next */
  129. if (error.code !== "ENOENT") {
  130. throw error;
  131. }
  132. return [];
  133. }
  134. }
  135. /**
  136. * Create a `RegExp` object to detect extensions.
  137. * @param {string[] | null} extensions The extensions to create.
  138. * @returns {RegExp | null} The created `RegExp` object or null.
  139. */
  140. function createExtensionRegExp(extensions) {
  141. if (extensions) {
  142. const normalizedExts = extensions.map(ext => escapeRegExp(
  143. ext.startsWith(".")
  144. ? ext.slice(1)
  145. : ext
  146. ));
  147. return new RegExp(
  148. `.\\.(?:${normalizedExts.join("|")})$`,
  149. "u"
  150. );
  151. }
  152. return null;
  153. }
  154. /**
  155. * The error type when no files match a glob.
  156. */
  157. class NoFilesFoundError extends Error {
  158. // eslint-disable-next-line jsdoc/require-description
  159. /**
  160. * @param {string} pattern The glob pattern which was not found.
  161. * @param {boolean} globDisabled If `true` then the pattern was a glob pattern, but glob was disabled.
  162. */
  163. constructor(pattern, globDisabled) {
  164. super(`No files matching '${pattern}' were found${globDisabled ? " (glob was disabled)" : ""}.`);
  165. this.messageTemplate = "file-not-found";
  166. this.messageData = { pattern, globDisabled };
  167. }
  168. }
  169. /**
  170. * The error type when there are files matched by a glob, but all of them have been ignored.
  171. */
  172. class AllFilesIgnoredError extends Error {
  173. // eslint-disable-next-line jsdoc/require-description
  174. /**
  175. * @param {string} pattern The glob pattern which was not found.
  176. */
  177. constructor(pattern) {
  178. super(`All files matched by '${pattern}' are ignored.`);
  179. this.messageTemplate = "all-files-ignored";
  180. this.messageData = { pattern };
  181. }
  182. }
  183. /**
  184. * This class provides the functionality that enumerates every file which is
  185. * matched by given glob patterns and that configuration.
  186. */
  187. class FileEnumerator {
  188. /**
  189. * Initialize this enumerator.
  190. * @param {FileEnumeratorOptions} options The options.
  191. */
  192. constructor({
  193. cwd = process.cwd(),
  194. configArrayFactory = new CascadingConfigArrayFactory({
  195. cwd,
  196. eslintRecommendedPath: path.resolve(__dirname, "../../conf/eslint-recommended.js"),
  197. eslintAllPath: path.resolve(__dirname, "../../conf/eslint-all.js")
  198. }),
  199. extensions = null,
  200. globInputPaths = true,
  201. errorOnUnmatchedPattern = true,
  202. ignore = true
  203. } = {}) {
  204. internalSlotsMap.set(this, {
  205. configArrayFactory,
  206. cwd,
  207. defaultIgnores: IgnorePattern.createDefaultIgnore(cwd),
  208. extensionRegExp: createExtensionRegExp(extensions),
  209. globInputPaths,
  210. errorOnUnmatchedPattern,
  211. ignoreFlag: ignore
  212. });
  213. }
  214. /**
  215. * Check if a given file is target or not.
  216. * @param {string} filePath The path to a candidate file.
  217. * @param {ConfigArray} [providedConfig] Optional. The configuration for the file.
  218. * @returns {boolean} `true` if the file is a target.
  219. */
  220. isTargetPath(filePath, providedConfig) {
  221. const {
  222. configArrayFactory,
  223. extensionRegExp
  224. } = internalSlotsMap.get(this);
  225. // If `--ext` option is present, use it.
  226. if (extensionRegExp) {
  227. return extensionRegExp.test(filePath);
  228. }
  229. // `.js` file is target by default.
  230. if (filePath.endsWith(".js")) {
  231. return true;
  232. }
  233. // use `overrides[].files` to check additional targets.
  234. const config =
  235. providedConfig ||
  236. configArrayFactory.getConfigArrayForFile(
  237. filePath,
  238. { ignoreNotFoundError: true }
  239. );
  240. return config.isAdditionalTargetPath(filePath);
  241. }
  242. /**
  243. * Iterate files which are matched by given glob patterns.
  244. * @param {string|string[]} patternOrPatterns The glob patterns to iterate files.
  245. * @returns {IterableIterator<FileAndConfig>} The found files.
  246. */
  247. *iterateFiles(patternOrPatterns) {
  248. const { globInputPaths, errorOnUnmatchedPattern } = internalSlotsMap.get(this);
  249. const patterns = Array.isArray(patternOrPatterns)
  250. ? patternOrPatterns
  251. : [patternOrPatterns];
  252. debug("Start to iterate files: %o", patterns);
  253. // The set of paths to remove duplicate.
  254. const set = new Set();
  255. for (const pattern of patterns) {
  256. let foundRegardlessOfIgnored = false;
  257. let found = false;
  258. // Skip empty string.
  259. if (!pattern) {
  260. continue;
  261. }
  262. // Iterate files of this pattern.
  263. for (const { config, filePath, flag } of this._iterateFiles(pattern)) {
  264. foundRegardlessOfIgnored = true;
  265. if (flag === IGNORED_SILENTLY) {
  266. continue;
  267. }
  268. found = true;
  269. // Remove duplicate paths while yielding paths.
  270. if (!set.has(filePath)) {
  271. set.add(filePath);
  272. yield {
  273. config,
  274. filePath,
  275. ignored: flag === IGNORED
  276. };
  277. }
  278. }
  279. // Raise an error if any files were not found.
  280. if (errorOnUnmatchedPattern) {
  281. if (!foundRegardlessOfIgnored) {
  282. throw new NoFilesFoundError(
  283. pattern,
  284. !globInputPaths && isGlob(pattern)
  285. );
  286. }
  287. if (!found) {
  288. throw new AllFilesIgnoredError(pattern);
  289. }
  290. }
  291. }
  292. debug(`Complete iterating files: ${JSON.stringify(patterns)}`);
  293. }
  294. /**
  295. * Iterate files which are matched by a given glob pattern.
  296. * @param {string} pattern The glob pattern to iterate files.
  297. * @returns {IterableIterator<FileEntry>} The found files.
  298. */
  299. _iterateFiles(pattern) {
  300. const { cwd, globInputPaths } = internalSlotsMap.get(this);
  301. const absolutePath = path.resolve(cwd, pattern);
  302. const isDot = dotfilesPattern.test(pattern);
  303. const stat = statSafeSync(absolutePath);
  304. if (stat && stat.isDirectory()) {
  305. return this._iterateFilesWithDirectory(absolutePath, isDot);
  306. }
  307. if (stat && stat.isFile()) {
  308. return this._iterateFilesWithFile(absolutePath);
  309. }
  310. if (globInputPaths && isGlobPattern(pattern)) {
  311. return this._iterateFilesWithGlob(absolutePath, isDot);
  312. }
  313. return [];
  314. }
  315. /**
  316. * Iterate a file which is matched by a given path.
  317. * @param {string} filePath The path to the target file.
  318. * @returns {IterableIterator<FileEntry>} The found files.
  319. * @private
  320. */
  321. _iterateFilesWithFile(filePath) {
  322. debug(`File: ${filePath}`);
  323. const { configArrayFactory } = internalSlotsMap.get(this);
  324. const config = configArrayFactory.getConfigArrayForFile(filePath);
  325. const ignored = this._isIgnoredFile(filePath, { config, direct: true });
  326. const flag = ignored ? IGNORED : NONE;
  327. return [{ config, filePath, flag }];
  328. }
  329. /**
  330. * Iterate files in a given path.
  331. * @param {string} directoryPath The path to the target directory.
  332. * @param {boolean} dotfiles If `true` then it doesn't skip dot files by default.
  333. * @returns {IterableIterator<FileEntry>} The found files.
  334. * @private
  335. */
  336. _iterateFilesWithDirectory(directoryPath, dotfiles) {
  337. debug(`Directory: ${directoryPath}`);
  338. return this._iterateFilesRecursive(
  339. directoryPath,
  340. { dotfiles, recursive: true, selector: null }
  341. );
  342. }
  343. /**
  344. * Iterate files which are matched by a given glob pattern.
  345. * @param {string} pattern The glob pattern to iterate files.
  346. * @param {boolean} dotfiles If `true` then it doesn't skip dot files by default.
  347. * @returns {IterableIterator<FileEntry>} The found files.
  348. * @private
  349. */
  350. _iterateFilesWithGlob(pattern, dotfiles) {
  351. debug(`Glob: ${pattern}`);
  352. const directoryPath = path.resolve(getGlobParent(pattern));
  353. const globPart = pattern.slice(directoryPath.length + 1);
  354. /*
  355. * recursive if there are `**` or path separators in the glob part.
  356. * Otherwise, patterns such as `src/*.js`, it doesn't need recursive.
  357. */
  358. const recursive = /\*\*|\/|\\/u.test(globPart);
  359. const selector = new Minimatch(pattern, minimatchOpts);
  360. debug(`recursive? ${recursive}`);
  361. return this._iterateFilesRecursive(
  362. directoryPath,
  363. { dotfiles, recursive, selector }
  364. );
  365. }
  366. /**
  367. * Iterate files in a given path.
  368. * @param {string} directoryPath The path to the target directory.
  369. * @param {Object} options The options to iterate files.
  370. * @param {boolean} [options.dotfiles] If `true` then it doesn't skip dot files by default.
  371. * @param {boolean} [options.recursive] If `true` then it dives into sub directories.
  372. * @param {InstanceType<Minimatch>} [options.selector] The matcher to choose files.
  373. * @returns {IterableIterator<FileEntry>} The found files.
  374. * @private
  375. */
  376. *_iterateFilesRecursive(directoryPath, options) {
  377. debug(`Enter the directory: ${directoryPath}`);
  378. const { configArrayFactory } = internalSlotsMap.get(this);
  379. /** @type {ConfigArray|null} */
  380. let config = null;
  381. // Enumerate the files of this directory.
  382. for (const entry of readdirSafeSync(directoryPath)) {
  383. const filePath = path.join(directoryPath, entry.name);
  384. // Check if the file is matched.
  385. if (entry.isFile()) {
  386. if (!config) {
  387. config = configArrayFactory.getConfigArrayForFile(
  388. filePath,
  389. /*
  390. * We must ignore `ConfigurationNotFoundError` at this
  391. * point because we don't know if target files exist in
  392. * this directory.
  393. */
  394. { ignoreNotFoundError: true }
  395. );
  396. }
  397. const matched = options.selector
  398. // Started with a glob pattern; choose by the pattern.
  399. ? options.selector.match(filePath)
  400. // Started with a directory path; choose by file extensions.
  401. : this.isTargetPath(filePath, config);
  402. if (matched) {
  403. const ignored = this._isIgnoredFile(filePath, { ...options, config });
  404. const flag = ignored ? IGNORED_SILENTLY : NONE;
  405. debug(`Yield: ${entry.name}${ignored ? " but ignored" : ""}`);
  406. yield {
  407. config: configArrayFactory.getConfigArrayForFile(filePath),
  408. filePath,
  409. flag
  410. };
  411. } else {
  412. debug(`Didn't match: ${entry.name}`);
  413. }
  414. // Dive into the sub directory.
  415. } else if (options.recursive && entry.isDirectory()) {
  416. if (!config) {
  417. config = configArrayFactory.getConfigArrayForFile(
  418. filePath,
  419. { ignoreNotFoundError: true }
  420. );
  421. }
  422. const ignored = this._isIgnoredFile(
  423. filePath + path.sep,
  424. { ...options, config }
  425. );
  426. if (!ignored) {
  427. yield* this._iterateFilesRecursive(filePath, options);
  428. }
  429. }
  430. }
  431. debug(`Leave the directory: ${directoryPath}`);
  432. }
  433. /**
  434. * Check if a given file should be ignored.
  435. * @param {string} filePath The path to a file to check.
  436. * @param {Object} options Options
  437. * @param {ConfigArray} [options.config] The config for this file.
  438. * @param {boolean} [options.dotfiles] If `true` then this is not ignore dot files by default.
  439. * @param {boolean} [options.direct] If `true` then this is a direct specified file.
  440. * @returns {boolean} `true` if the file should be ignored.
  441. * @private
  442. */
  443. _isIgnoredFile(filePath, {
  444. config: providedConfig,
  445. dotfiles = false,
  446. direct = false
  447. }) {
  448. const {
  449. configArrayFactory,
  450. defaultIgnores,
  451. ignoreFlag
  452. } = internalSlotsMap.get(this);
  453. if (ignoreFlag) {
  454. const config =
  455. providedConfig ||
  456. configArrayFactory.getConfigArrayForFile(
  457. filePath,
  458. { ignoreNotFoundError: true }
  459. );
  460. const ignores =
  461. config.extractConfig(filePath).ignores || defaultIgnores;
  462. return ignores(filePath, dotfiles);
  463. }
  464. return !direct && defaultIgnores(filePath, dotfiles);
  465. }
  466. }
  467. //------------------------------------------------------------------------------
  468. // Public Interface
  469. //------------------------------------------------------------------------------
  470. module.exports = { FileEnumerator };