Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

eslint-remote-tester

AriPerkkio11.9kMIT4.0.1TypeScript support: included

Tool for running ESLint on multiple repositories

eslint, eslint-test

readme

eslint-remote-tester

version

Installation | Configuration | How and when to use | Github CI Actions | Examples

eslint-remote-tester is a CLI tool for testing given ESlint rules against multiple repositories at once. It's designed to be used when validating regression of new rules. It can be used to spot whether a new rule flags false positives or crashes ESLint completely. CIs can be configured to verify regression of large set of rules so that possible null pointers or any unexpected errors are caught immediately.

Installation

# In your project directory
$ npm install --save-dev eslint-remote-tester

# eslint@>=9 is also required as peer dependency
$ npm install --save-dev eslint

The underlying git integration is done via simple-git. It requires git to be installed and that it can be called using the command git.

Configuration

Linking your local eslint plugin as a dependency

In order for eslint-remote-tester's eslint to know where the definitions for the new rules or modified existing rules are, your local eslint plugin folder needs to be symlinked. In your plugin folder, dependending on which package manager you use, run:

# Using npm package manager
npm link
npm link [your-plugin-name]

package.json

Add new script to your package.json file.

"scripts": {
    "lint:remote": "eslint-remote-tester",

eslint-remote-tester.config.js

Create new configuration file eslint-remote-tester.config.js in the root of your project. This is used as configuration file by default. Use -c or --config argument for custom configuration file, e.g. --config path/custom.config.js.

import js from '@eslint/js';

/** @type {import('eslint-remote-tester').Config} */
const config = {
    repositories: ['mui-org/material-ui', 'reach/reach-ui'],
    extensions: ['js', 'jsx', 'ts', 'tsx'],
    eslintConfig: [js.configs.recommended],
    pathIgnorePattern: `(${[
        'node_modules',
        '\\/\\.', // Any file or directory starting with dot, e.g. ".git"
        'test-results',
        'docs',
    ].join('|')})`,
};

export default config;

Configuration file can also be written in TypeScript if jiti or importx is installed. Use --config argument for TypeScript configuration file, e.g. --config eslint-remote-tester.config.ts.

import type { Config } from 'eslint-remote-tester';
import js from '@eslint/js';

const config: Config = {
    repositories: ['mui-org/material-ui', 'reach/reach-ui'],
    extensions: ['js', 'jsx', 'ts', 'tsx'],
    eslintConfig: [js.configs.recommended],
};

export default config;

Configuration options

Name Description                               Type Required Default Example
repositories Repositories to scan in format of owner/project. See eslint-remote-tester-repositories for shared list of repositories. string[] :white_check_mark: :x: ['mui-org/material-ui', 'reach/reach-ui']
extensions Extensions of files under scanning string[] :white_check_mark: :x: ['js', 'jsx', 'ts', 'tsx']
eslintConfig ESLint configuration. Supports lazy initialization based on currently tested repository when a function is passed. Function is called with current repository and its location on filesystem. object | ({ location, repository }) => object See Configuring ESLint :white_check_mark: :x: [js.configs.recommended] (options) => options.repository === 'my-repo' ? [js.configs.recommended] : []
pathIgnorePattern Regexp pattern string used to exclude paths string :x: :x: (node_modules|docs|\\/\\.git)
maxFileSizeBytes Max file size used to exclude bigger files number :x: 2000000 1500000
rulesUnderTesting Array of rules or a filter method used to filter out results. Use undefined or empty array when ESLint crashes are the only interest. Filter method is called with ruleId and options. string[] | (ruleId, { repository }) => boolean :x: [] ['no-empty', 'react/sort-prop-types'] (ruleId, options) => ruleId === 'no-undef' && options.repository === 'owner/repo'
resultParser Syntax for the result parser plaintext|markdown :x: markdown on CLI. plaintext on CI markdown
concurrentTasks Maximum amount of tasks run concurrently number :x: 5 3
CI Flag used to set CI mode. process.env.CI is used when not set. boolean :x: value of process.env.CI === 'true' true
logLevel Filter log messages based on their priority verbose|info|warn|error :x: verbose warn
slowLintTimeLimit Time limit before linting of a single file is considered as slow, and logged as warning. Disabled by default. number|null :x: :x: 5 // 5 seconds
cache Flag used to enable caching of cloned repositories. For CIs it's ideal to disable caching due to limited disk space. boolean :x: true true
timeLimit Time limit before scan is interrupted and exited successfully. Ideal for avoiding CI timeouts in regression tests. number :x: 5.5 * 60 * 60, // 5 hours 30 minutes 5 * 60 * 60 // 5 hours
compare Flag used to enable result comparison mode. Compares results of two scans and output the diff. Ideal for identifying new false positives when fixing existing rules. See Fixing existing rules. boolean :x: false true
updateComparisonReference Flag used to enable result comparison reference updating. Indicates whether comparison base should be updated after scan has finished. Ideal to be turned off once initial comparison base has been collected. boolean :x: true true
onComplete Callback invoked once scan is completed. Asynchronous functions are supported. Ideal for extending the process with custom features. (results, comparisonResults, repositoryCount) => void|Promise<void>. See onComplete example. :x: :x: async (results, comparisonResults, repositoryCount) => {}

Execution

Run npm run lint:remote. Results are written into ./eslint-remote-tester-results folder.

How and when to use

Disclaimer: This section is still work in progress.

Developing new ESLint rules

When developing new ESLint rules you'll be most interested in results reported by the rule. As the AST of Javascript and Typescript can cause very unexpected results it is not enough to test the rule only against unit tests and a small amount of repositories. Testing the rule against 1000's of repositories can reveal unexpected results:

  • Does the rule report the intended patterns?
  • Does the rule falsely mark valid patterns as errors?
  • Are the error messages set as expected?

Follow these steps to test the new rule easily:

  1. Include new rule in rulesUnderTesting
  2. Configure eslintConfig with the new rule and its options
  3. Run eslint-remote-tester locally
  4. Check generated results at ./eslint-remote-tester-results

Fixing existing rules

Fixing existing rules without breaking old functionality or introducing new bugs can be difficult. By testing the rule against 1000's of repositories and comparing these reports against previous build's reports it is easily seen whether anything broke.

Running eslint-remote-tester in comparison mode allows developers to see the exact changes in ESLint reports their code changes introduced. Example of comparison results can be found at Examples section.

Follow these steps to test code changes of existing rules easily:

  1. Include given rule in rulesUnderTesting
  2. Configure eslintConfig with the rule and its options
  3. Stash code changes of the fix so that the current setup matches the latest build (typically master branch)
  4. Enable compare in the configuration file. This will generate comparison reference results.
  5. Run eslint-remote-tester locally
  6. Apply code changes of the fix back
  7. Disable updateComparisonReference in the configuration file. This will disable comparison reference updates - meaning that all next comparison runs are compared against the previously generated results (e.g. master branch).
  8. Run eslint-remote-tester locally again
  9. Check generated results at ./eslint-remote-tester-results/comparison-results. New reports introduced by these code changes are saved in added.md. Reports not seen with latest changes are saved in removed.md.
  10. Repeat steps 8-9 and keep fixing the rule until added.md does not contain new false positives and removed.md does not contain unintended reports.

Plugin maintainer making sure new PRs don't introduce new false positives

Reviewing PRs of ESLint plugin projects can be slow if maintainers are expected to test every change manually. Plugin projects can set their CI to run eslint-remote-tester on comparison mode for each new PR. The CI will compare ESLint reports between the master and PR branches, and report all new and removed reports. This way the reviewers can easily see whether PR actually fixed the issue and if it introduced new bugs.

Projects using GitHub CIs can utilize eslint-remote-tester-compare-action for setting up PR comparisons.

Plugin maintainer making sure all existing rules do not crash

Plugin projects can use eslint-remote-tester for smoke testing. All the rules of given project can be run against 1000's of repositories so that only crashing rules are detected. The job can even be configured to run for a limited time so that it will test as many repositories as it can in a specific time. This can be done by setting timeLimit configuration. For Github CI the maximum time limit can be as long as 6 hours.

This project is set to run 6 hour long weekly scheduled tests using some of the most well-known ESLint community plugins. Typically the fastest plugins can test more than 10K repositories in the 6 hour time limit.

Projects using GitHub CIs can utilize eslint-remote-tester-run-action for setting up smoke tests.

Configuration maintainer making sure all repositories follow the rules

Configuration maintainers (eslint-config-*) can check all their repositories follow rules of the configuration. By using an all-inclusive filter in rulesUnderTesting all ESLint reports are picked, e.g. rulesUnderTesting: () => true.

The arguments of filter callback can be used to exclude specific rules from specific repositories:

rulesUnderTesting: function ruleFilter(ruleId, options) {
    if (ruleId === 'node/no-process-env' && options.repository === 'username/my-cli-package') {
        // my-cli-package is allowed to use process.env
        return false;
    }

    // All other rules & repositories
    return true;
}

Github CI Actions

Examples

Results:

Comparison results:

Configuration

Repositories

For easiest setup eslint-remote-tester-repositories npm package is provided. See documentation.

Target repositories can easily be found using libraries.io. They provide an API for querying repositories which depend on certain project. Example usage at repository-query-script.

changelog

(2024-07-16)

4.0.1 (2024-07-16)

Bug Fixes

4.0.0 (2024-06-09)

3.0.1 (2023-10-01)

Bug Fixes

  • parse ruleId from loading error message (#475) (95c89de)

3.0.0 (2022-06-12)

  • feat!: never create configuration files (a26563b)
  • feat!: resolve slow lint time from configuration (5b33d89)

Features

  • config: adds slowLintTimeLimit option (67f759c)

BREAKING CHANGES

  • Previously a default configuration file was created when given configuration file was not found. Removed configuration file creation completely. It was completely useless.
  • Previously a warning was logged if linting took more than 5s. Now by default no such warnings are logged. Timelimit for such logs can be configured via config.slowLintTimeLimit

2.1.4 (2022-04-21)

Bug Fixes

2.1.3 (2022-04-20)

2.1.2 (2022-04-20)

2.1.1 (2021-12-31)

Bug Fixes

  • include loadConfig in private actions API (#324) (ecb4633)

2.1.0 (2021-12-31)

Bug Fixes

  • minor typo and grammar adjustments (#319) (ac263a7)
  • repositories: prevent exiting script too early (aaeba18)

Features

  • export configuration types (#321) (3d8bee0)
  • load typescript config by default if present (126033c)
  • support typescript configs using ts-node (#320) (b5f1252)

2.0.1 (2021-10-31)

Bug Fixes

  • deps: match react version requirement with ink (#302) (21824eb)

2.0.0 (2021-10-23)

  • feat!: improve ruleId parsing (69a14c6)

BREAKING CHANGES

  • Reported rules now may include plugin name, e.g. no-unstable-components -> react/no-unstable-components These are used in reported results and in some callbacks of eslint-remote-tester.config.js.

  • parse ruleId from stack traces provided by ESLint v8

1.3.1 (2021-09-11)

Bug Fixes

  • node-16: avoid deprecation warnings of fs.rmdirSync (#277) (3a84361)

1.3.0 (2021-07-10)

Bug Fixes

  • pass main thread's environment variables to worker threads (bd294f5)

Features

  • compile to target the actual feature set of Node >=12 (#173) (65ec3d8)
  • support config.eslintrc as function (6079cab)

1.2.0 (2021-05-15)

Bug Fixes

  • compare: do not "git pull" when in comparison mode (e3cbc0f)
  • engine: ignore eslint inline configurations (93616dd), closes #38
  • engine: report worker crashes correctly (#43) (82cb301)
  • results: limit rows of results into 1000 characters (#101) (14c170e)

Features

  • ci: include count of errors in ci status messages (#126) (d34660f)
  • cli: log status of cached repositories (dc5580c)
  • move cache under node_modules (83cb6e5)
  • pass count of scanned repositories to config.onComplete (#102) (814f6f0)

1.1.0 (2021-02-10)

Features

  • enable checking all rules (9d1a299)

1.0.1 (2021-02-08)

Bug Fixes

  • engine: support large amount of JSONs (668adb2)
  • handle results as streams (562538e)
  • include error message in write failure logs (84100b7)

Performance Improvements

  • result-comparator: add internal hash for comparing results via map (44daa5d)

1.0.0 (2021-02-01)

Bug Fixes

  • auto-generated config contains incorrectly escaped regex (ea52f15)

Features

  • use @babel/code-frame for result.source (e28c06d)

BREAKING CHANGES

  • output and onComplete args result.source is now formatted with @babel/code-frame

0.6.0 (2021-01-28)

Bug Fixes

  • exclude unnecessary sourcemaps from builds (9822364)

Features

  • add private api for eslint-remote-tester-compare-action (52c790b)

0.5.0 (2021-01-16)

Bug Fixes

  • results: markdown template formatting (c0f62fb)
  • results: prevent unnecessary text wrap on ci (a836913)
  • small error (318d512)
  • validator: resolve result parser properly (a4af9bb)

Features

  • config: adds compare option (acb97c5)
  • config: adds updateComparisonReference option (d544578)
  • file-client: initialize compaison results directory (19f6136)
  • result-comparator: generate comparison results (d8af9c8)
  • result-comparator: include comparison results in config.onComplete (378a2b3)
  • result-comparator: render comparison results on ui (e36d4c9)
  • result-comparator: update comparison reference based on config.updateComparisonReference (f5d9b04)
  • validator: improve configuration validation (09c23ce)

0.4.0 (2020-12-17)

Bug Fixes

  • engine: pass process.env.CI to workers (74c7831)
  • file-client: handle symlinks (5c4144f)

Features

  • config: adds timeLimit (5cb133d)
  • config: validate duplicate repositories (485fd53)
  • config: validate unknown options (5c2cc53)

0.3.5 (2020-12-06)

Bug Fixes

  • progress-logger: add info logLevel for ci-keep-alive messages (ecacd30)

0.3.4 (2020-12-05)

Bug Fixes

  • config: include eslintrc.rules to validation (556a09f)
  • repository-client: prevent process hang caused by git prompt (ebdd8c6)

Features

  • config: adds logLevel (f538ff7)
  • log-level: decrease log level of multiple warnings (f83ca73)

Performance Improvements

  • repository-client: clone only a single branch without history (07ac720)

0.3.3 (2020-11-29)

Features

  • config: add maxFileSizeBytes option (364f26d)

0.3.2 (2020-11-28)

Bug Fixes

  • config: exit with error code when config is invalid (f4f3ab6)
  • results: exit with status code when lint contains errors (a6c7c51)

Features

  • ci-runner: add separate ci-runner package (525ab3e)
  • ci-runner: run ci on schedule (a51bebe)
  • ci: ci-keep-alive message (35c79c6)
  • ci: github actions (926c717)
  • config: add cache flag (5d8d21d)
  • config: add onComplete hook (d053d2f)

0.3.1 (2020-11-08)

Bug Fixes

  • linter-crash: add missing file count increase on crashes (74e75de)
  • linter-crash: prevent displaying Rule: null when unable to parse ruleId (10c4ac9)
  • results: source window size resolving (caa6236)

Features

  • cli: handle non-tty streams (b08c2e6)
  • results: include erroneous line and source to linter crashes (2737019)

0.3.0 (2020-11-05)

Features

  • cli: ink CLI support (a7c2253)
  • config: config.CI to override env.CI (377a869)
  • results: include repository owner in results filenames (21e45a3)
  • tests: integration tests update and split (0c6135c)

BREAKING CHANGES

  • config: Prioritize config over environment variables when resolving value for CI

0.2.1 (2020-10-17)

0.2.0 (2020-10-11)

0.1.3 (2020-10-07)

0.1.2 (2020-10-01)

0.1.1 (2020-09-25)

0.1.0 (2020-09-25)

0.0.4 (2020-09-18)

0.0.3 (2020-09-17)

0.0.2 (2020-09-13)