Skip to main content

A lint running tool and framework.

Project description

Lintrunner

Overview

lintrunner is a tool that runs linters. It is responsible for:

  • Deciding which files need to be linted.
  • Invoking linters according to a common protocol.
  • Gathering results and presenting them to users.

The intention is to provide a universal way to configure and invoke linters, which is useful on large polyglot projects.

The design of lintrunner is heavily inspired by linttool, a project that exists internally at Meta.

Installation

pip install lintrunner

Usage

First, you need to add a configuration file to your repo. See the Linter configuration section for more info.

Then, simply run lintrunner to lint your changes!

How to control what paths to lint lintrunner

When run with no arguments, lintrunner will check:

  • The files changed in the HEAD commit.
  • The files changed in the user’s working tree.

It does not check:

  • Any files not tracked by git; git add them to lint them.

There are multiple ways to customize how paths are checked:

Pass paths as positional arguments

For example:

lintrunner foo.py bar.cpp

This naturally composes with xargs, for example the canonical way to check every path in the repo is:

git grep -Il . | xargs lintrunner

--configs/ --config

"Comma-separated paths to lintrunner configuration files. Multiple files are merged, with later definitions overriding earlier ones. ONLY THE FIRST is required to be present on your machine. Defaults to lintrunner.toml, lintrunner.private.toml. Extra configs like lintrunner.private.toml are useful for combining project-wide and local configs."

--paths-cmd

Some ways to invoke xargs will cause multiple lintrunner processes to be run, increasing lint time (especially on huge path sets). As an alternative that gives lintrunner control of parallelization, you can use --paths-cmd. If --paths-cmd is specified lintrunner will execute that command and consider each line of its stdout to be a file to lint.

For example, the same command above would be:

lintrunner --paths-cmd='git grep -Il .'

--paths-file

If this is specified, lintrunner will read paths from the given file, one per line, and check those. This can be useful if you have some really complex logic to determine which paths to check.

--revision

This value can be any <tree-ish> accepted by git diff-tree, like a commit hash or revspec. If this is specified, lintrunner will check:

  • All paths changed from <tree-ish> to HEAD
  • All paths changed in the user's working tree.

--merge-base-with

Like --revision, except the revision is determined by computing the merge-base of HEAD and the provided <tree-ish>. This is useful for linting all commits in a specific pull request. For example, for a pull request targeting master, you can run:

lintrunner -m master

--all-files

This will run lint on all files specified in .lintrunner.toml.

--only-lint-under-config-dir

If set, will only lint files under the directory where the configuration file is located and its subdirectories.

Linter configuration

lintrunner knows which linters to run and how by looking at a configuration file, conventionally named .lintrunner.toml.

Here is an example linter configuration:

merge_base_with = 'main'

[[linter]]
name = 'FLAKE8'
include_patterns = [
  'src/**/*.py',  # unix-style globs supported
  'test/**/*.py',
]
exclude_patterns = ['src/my_bad_file.py']
command = [
  'python3',
  'flake8_linter.py',
  '—-',
  # {{PATHSFILE}} gets rewritten to a tmpfile containing all paths to lint
  '@{{PATHSFILE}}',
]

A complete description of the configuration schema can be found here.

Linter protocol

Most linters have their own output format and arguments. In order to impose consistency on linter invocation and outputs, lintrunner implements a protocol that it expects linters to fulfill. In most cases, a small script (called a linter adapter) is required to implement the protocol for a given external linter. You can see some example adapters in examples/ .

Invocation

Linters will be invoked according to the command specified by their configuration. They will be called once per lint run.

If a linter needs to know which paths to run on, it should take a {{PATHSFILE}} argument. During invocation, the string {{PATHSFILE}} will be replaced with the name of a temporary file containing which paths the linter should run on, one path per line.

A common way to implement this in a linter adapter is to use argparse’s fromfile_prefix_chars feature. In the Flake8 example above, we use @ as the fromfile_prefix_chars argument, so argparse will automatically read the {{PATHSFILE}} and supply its contents as a list of arguments.

Output

Any lint messages a linter would like to communicate the user must be represented as a LintMessage. The linter, must print LintMessages as JSON Lines to stdout, one message per line. Output to stderr will be ignored.

A complete description of the LintMessage schema can be found here.

Exiting

Linters should always exit with code 0. This is true even if lint errors are reported; lintrunner itself will determine how to exit based on what linters report.

To signal a general linter failure (which should ideally never happen!), linters can return a LintMessage with path = None.

In the event a linter exits non-zero, it will be caught by lintrunnerand presented as a “general linter failure” with stdout/stderr shown to the user. This should be considered a bug in the linter’s implementation of this protocol.

Tips for adopting lintrunner in a new project

When adopting lintrunner in a previously un-linted project, it may generate a lot of lint messages. You can use the --output oneline option to make lintrunner display each lint message in its separate line to quickly navigate through them.

Additionally, you can selectively run specific linters with the --take option, like --take RUFF,CLANGFORMAT, to focus on resolving specific lint errors, or use --skip to skip a long running linter like MYPY.

GitHub Action

To use lintrunner in a GitHub workflow, you can consider lintrunner-action.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

lintrunner-0.12.4.tar.gz (62.1 kB view details)

Uploaded Source

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

lintrunner-0.12.4-py3-none-win32.whl (1.6 MB view details)

Uploaded Python 3Windows x86

lintrunner-0.12.4-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (2.9 MB view details)

Uploaded Python 3manylinux: glibc 2.17+ x86-64

lintrunner-0.12.4-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl (2.8 MB view details)

Uploaded Python 3manylinux: glibc 2.17+ ARM64

lintrunner-0.12.4-py3-none-macosx_11_0_arm64.whl (1.9 MB view details)

Uploaded Python 3macOS 11.0+ ARM64

lintrunner-0.12.4-py3-none-macosx_10_7_x86_64.whl (2.0 MB view details)

Uploaded Python 3macOS 10.7+ x86-64

File details

Details for the file lintrunner-0.12.4.tar.gz.

File metadata

  • Download URL: lintrunner-0.12.4.tar.gz
  • Upload date:
  • Size: 62.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: maturin/1.5.0

File hashes

Hashes for lintrunner-0.12.4.tar.gz
Algorithm Hash digest
SHA256 de030dfc6de23b561f98be731d12b505a40290450ba8d15e6a611a18c3ad95a8
MD5 1586b846c18707e4899ba6a397cc7428
BLAKE2b-256 dfb00852067cdf743bb80519841e8ba62242bd17e1338b3559ffd1375666a890

See more details on using hashes here.

File details

Details for the file lintrunner-0.12.4-py3-none-win32.whl.

File metadata

  • Download URL: lintrunner-0.12.4-py3-none-win32.whl
  • Upload date:
  • Size: 1.6 MB
  • Tags: Python 3, Windows x86
  • Uploaded using Trusted Publishing? No
  • Uploaded via: maturin/1.5.0

File hashes

Hashes for lintrunner-0.12.4-py3-none-win32.whl
Algorithm Hash digest
SHA256 f8ccdd244860a486accb4d48030777c83037b3303e2271c36f97827e14b99fe7
MD5 3f67a9612dd8c3f4611edf5fb0b42781
BLAKE2b-256 14b25af5f9a9f1506cfd1479439ac431ac16e6ed422301bca3c269908a5dc505

See more details on using hashes here.

File details

Details for the file lintrunner-0.12.4-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for lintrunner-0.12.4-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 a3fabbd842f4c7669036346eea8e44fb9c8f1dddc1f8fcb532288a570d927e87
MD5 675e8a3ec7fbf258c3e2e43e2c193061
BLAKE2b-256 8497419618c12f570824a0f3c5acc7a26275d9c276930a16b671a97972ff325f

See more details on using hashes here.

File details

Details for the file lintrunner-0.12.4-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl.

File metadata

File hashes

Hashes for lintrunner-0.12.4-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl
Algorithm Hash digest
SHA256 e70cb422e9368af24ed81f4c46ee308d28fd0b65d30111a389f1f200ce91b4f7
MD5 79018362dbaab0a2c25d8907f98c050c
BLAKE2b-256 1e5ffccf0987d40e49974e72d700f6a8b1ebf5608aa601e0dd82f4d9fcbf21e5

See more details on using hashes here.

File details

Details for the file lintrunner-0.12.4-py3-none-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for lintrunner-0.12.4-py3-none-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 952edde74054d86d12e32237d346266772a5acd787d3228eead0ae262f6c65df
MD5 797d55e6f3b203afe5c4dab5726e57b2
BLAKE2b-256 cdba62d61c78a8bf8affa789b3545f045d3f9424018708562287a6299324f29d

See more details on using hashes here.

File details

Details for the file lintrunner-0.12.4-py3-none-macosx_10_7_x86_64.whl.

File metadata

File hashes

Hashes for lintrunner-0.12.4-py3-none-macosx_10_7_x86_64.whl
Algorithm Hash digest
SHA256 15f25af3dc93e20c9f0b0dd86159518b81ed70b68103bb5a2c5433ad8be7e524
MD5 00ec71306899736b44914e3821422386
BLAKE2b-256 7c29148aa614d5f7684393ec27870f64dc10bd181a447500bd803b7b4b5fd678

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page