Home

Awesome

nvim-lint

An asynchronous linter plugin for Neovim (>= 0.6.0) complementary to the built-in Language Server Protocol support.

Motivation & Goals

With ale we already got an asynchronous linter, why write yet another one?

Because ale also includes its own language server client.

nvim-lint instead has a more narrow scope: It spawns linters, parses their output, and reports the results via the vim.diagnostic module.

nvim-lint complements the built-in language server client for languages where there are no language servers, or where standalone linters provide better results.

Installation

For example:

git clone \
    https://github.com/mfussenegger/nvim-lint.git
    ~/.config/nvim/pack/plugins/start/nvim-lint

Usage

Configure the linters you want to run per file type. For example:

require('lint').linters_by_ft = {
  markdown = {'vale',}
}

Then setup a autocmd to trigger linting. For example:

au BufWritePost * lua require('lint').try_lint()

or with Lua autocmds (requires 0.7):

vim.api.nvim_create_autocmd({ "BufWritePost" }, {
  callback = function()

    -- try_lint without arguments runs the linters defined in `linters_by_ft`
    -- for the current filetype
    require("lint").try_lint()

    -- You can call `try_lint` with a linter name or a list of names to always
    -- run specific linters, independent of the `linters_by_ft` configuration
    require("lint").try_lint("cspell")
  end,
})

Some linters require a file to be saved to disk, others support linting stdin input. For such linters you could also define a more aggressive autocmd, for example on the InsertLeave or TextChanged events.

If you want to customize how the diagnostics are displayed, read :help vim.diagnostic.config.

Available Linters

There is a generic linter called compiler that uses the makeprg and errorformat options of the current buffer.

Other dedicated linters that are built-in are:

ToolLinter name
Set via makeprgcompiler
actionlintactionlint
alexalex
amebaameba
ansible-lintansible_lint
banditbandit
bean-checkbean_check
biomejsbiomejs
blocklintblocklint
buf_lintbuf_lint
buildifierbuildifier
cfn-lintcfn_lint
cfn_nagcfn_nag
checkmakecheckmake
checkpatch.plcheckpatch
checkstylecheckstyle
chktexchktex
clang-tidyclangtidy
clazyclazy
clj-kondoclj-kondo
cmakelintcmakelint
codespellcodespell
commitlintcommitlint
cppcheckcppcheck
cpplintcpplint
credocredo
cspellcspell
cuecue
curlylintcurlylint
dashdash
deadnixdeadnix
denodeno
DirectX Shader Compilerdxc
djlintdjlint
dotenv-linterdotenv_linter
editorconfig-checkereditorconfig-checker
erb-linterb_lint
ESLinteslint
eslint_deslint_d
fennelfennel
fishfish
Flake8flake8
flawfinderflawfinder
gdlint (gdtoolkit)gdlint
GHDLghdl
gitlintgitlint
glslcglslc
Golangci-lintgolangcilint
hadolinthadolint
hlinthlint
htmlhinthtmlhint
HTML Tidytidy
Inkoinko
janetjanet
jokerjoker
jshintjshint
jsonlintjsonlint
ktlintktlint
lachecklacheck
Languagetoollanguagetool
luacheckluacheck
markdownlintmarkdownlint
markdownlint-cli2markdownlint-cli2
markuplintmarkuplint
mlintmlint
Mypymypy
Nagelfarnagelfar
Nixnix
npm-groovy-lintnpm-groovy-lint
oelint-advoelint-adv
opa_checkopa_check
oxlintoxlint
perlcriticperlcritic
perlimportsperlimports
phpcsphpcs
phpinsightsphpinsights
phpmdphpmd
phpphp
phpstanphpstan
ponycpony
prisma-lintprisma-lint
proselintproselint
protolintprotolint
psalmpsalm
puppet-lintpuppet-lint
pycodestylepycodestyle
pydocstylepydocstyle
Pylintpylint
pyproject-flake8pflake8
quick-lint-jsquick-lint-js
regalregal
Reviverevive
rflintrflint
robocoprobocop
RPMrpmspec
rstcheckrstcheck
rstlintrstlint
RuboCoprubocop
Rubyruby
Ruffruff
salt-lintsaltlint
Seleneselene
ShellCheckshellcheck
snyksnyk_iac
Solhintsolhint
Spectralspectral
sqlfluffsqlfluff
standardjsstandardjs
StandardRBstandardrb
statix checkstatix
stylelintstylelint
SwiftLintswiftlint
systemdlintsystemdlint
tflinttflint
tfsectfsec
tlinttlint
trivytrivy
typostypos
Valavala_lint
Valevale
Verilatorverilator
vintvint
VSGvsg
vulturevulture
wokewoke
write-goodwrite_good
yamllintyamllint
zshzsh

Custom Linters

You can register custom linters by adding them to the linters table, but please consider contributing a linter if it is missing.

require('lint').linters.your_linter_name = {
  cmd = 'linter_cmd',
  stdin = true, -- or false if it doesn't support content input via stdin. In that case the filename is automatically added to the arguments.
  append_fname = true, -- Automatically append the file name to `args` if `stdin = false` (default: true)
  args = {}, -- list of arguments. Can contain functions with zero arguments that will be evaluated once the linter is used.
  stream = nil, -- ('stdout' | 'stderr' | 'both') configure the stream to which the linter outputs the linting result.
  ignore_exitcode = false, -- set this to true if the linter exits with a code != 0 and that's considered normal.
  env = nil, -- custom environment table to use with the external process. Note that this replaces the *entire* environment, it is not additive.
  parser = your_parse_function
}

Instead of declaring the linter as a table, you can also declare it as a function which returns the linter table in case you want to dynamically generate some of the properties.

your_parse_function can be a function which takes three arguments:

The output is the output generated by the linter command. The function must return a list of diagnostics as specified in :help diagnostic-structure.

You can override the environment that the linting process runs in by setting the env key, e.g.

env = { ["FOO"] = "bar" }

Note that this completely overrides the environment, it does not add new environment variables. The one exception is that the PATH variable will be preserved if it is not explicitly set.

You can generate a parse function from a Lua pattern or from an errorformat using the function in the lint.parser module:

from_errorformat

parser = require('lint.parser').from_errorformat(errorformat)

The function takes two arguments: errorformat and skeleton (optional).

from_pattern

parser = require('lint.parser').from_pattern(pattern, groups, severity_map, defaults, opts)

The function allows to parse the linter's output using a Lua regular expression pattern.

Available groups:

The order of the groups must match the order of the captures within the pattern. An example:

local pattern = '[^:]+:(%d+):(%d+):(%w+):(.+)'
local groups = { 'lnum', 'col', 'code', 'message' }
default_severity = {
['error'] = vim.diagnostic.severity.ERROR,
['warning'] = vim.diagnostic.severity.WARN,
['information'] = vim.diagnostic.severity.INFO,
['hint'] = vim.diagnostic.severity.HINT,
}
defaults = {["source"] = "mylint-name"}

Customize built-in linters

You can import a linter and modify its properties. An example:

local phpcs = require('lint').linters.phpcs
phpcs.args = {
  '-q',
  -- <- Add a new parameter here
  '--report=json',
  '-'
}

You can also post-process the diagnostics produced by a linter by wrapping it. For example, to change the severity of all diagnostics created by cspell:

local lint = require("lint")
lint.linters.cspell = require("lint.util").wrap(lint.linters.cspell, function(diagnostic)
  diagnostic.severity = vim.diagnostic.severity.HINT
  return diagnostic
end)

Display configuration

See :help vim.diagnostic.config.

If you want to have different settings per linter, you can get the namespace for a linter via require("lint").get_namespace("linter_name"). An example:

local ns = require("lint").get_namespace("my_linter_name")
vim.diagnostic.config({ virtual_text = true }, ns)

Get the current running linters for your buffer

You can see which linters are running with require("lint").get_running(). To include the running linters in the status line you could format them like this:

local lint_progress = function()
  local linters = require("lint").get_running()
  if #linters == 0 then
      return "󰦕"
  end
  return "󱉶 " .. table.concat(linters, ", ")
end

Alternatives

Development ☢️

Run tests

Running tests requires busted.

See neorocks or Using Neovim as Lua interpreter with Luarocks for installation instructions.

busted tests/