Skip to main content
  1. Posts/

Pylsp setup for Neovim in 2023

··1020 words·5 mins·
Table of Contents

TL;DR: my config for pylsp can be found here.

In this post, I would like to share my latest setup for Pylsp in Neovim.

Installation
#

  • pylsp install: pip install "python-lsp-server[all]"
  • 3rd party plugins:

I also tried with pylsp-rope, but it works like sh*t, so I won’t waste my time on it.

Neovim lsp configuration
#

Use your favorite plugin manager to install nvim-lspconfig. Here is my configuration for nvim-lsp with the help of nvim-lspconfig.

lspconfig = require("lspconfig")
lspconfig.pylsp.setup {
on_attach = custom_attach,
settings = {
    pylsp = {
    plugins = {
        -- formatter options
        black = { enabled = true },
        autopep8 = { enabled = false },
        yapf = { enabled = false },
        -- linter options
        pylint = { enabled = true, executable = "pylint" },
        pyflakes = { enabled = false },
        pycodestyle = { enabled = false },
        -- type checker
        pylsp_mypy = { enabled = true },
        -- auto-completion options
        jedi_completion = { fuzzy = true },
        -- import sorting
        pyls_isort = { enabled = true },
    },
    },
},
flags = {
    debounce_text_changes = 200,
},
capabilities = capabilities,
}

Note that for some of the plugins, you can pass other configuration to the execute, but I prefer to control their behavior using in their configuration files, not using nvim-lsp.

ref:

Configuration pylint and black in pyproject.toml
#

pyproject.toml
#

For project-specific settings, it is convenient to configure all this in the file pyproject.toml.

Ref:

Pylint
#

Pylint recognizes configurations in pyproject.toml file. However, configuring pylint in pyproject.toml requires a bit understanding. First, we can generate a full pylint configuration for later reference:

pylint --generate-rcfile > pylintrc

We can see that pylint configuration is separated into multiple sections (or table by TOML’s jargon). When we want to configure a certain option for Pylint, we must know its table in pylintrc. For example, suppose we want to set max-line-length to 100, first check which table it belongs to in the original pylintrc. After checking, we know it belongs to FORMAT table, then we can use either two confs in pyproject.toml:

[tool.pylint.format]
max-line-length = 100

or

[tool.pylint.'FORMAT']
max-line-length = 100

Note that for values which are a list, we must use a list notation in pyproject.toml. For example, if we want to disable both missing-module-docstring and missing-function-docstring, we can write the config as this:

[tool.pylint.'MESSAGES CONTROL']
disable = ["missing-module-docstring", "missing-function-docstring"]

The following also works:

[tool.pylint.messages_control]
disable = ["missing-module-docstring", "missing-function-docstring"]

In this case, the syntax is not the same as pylintrc, where you only need to separated the warnings with comma.

As another way, we can also use pylint --generate-toml-config to generate a copy of pylintrc options in toml format. This would be much easier for hand-picking the options that we want to change.

There are a lot of examples on using pylint in pyproject.toml in GitHub, e.g., in AiiDA, jax, example conf from pylint offical.

ref:

Black
#

Configuration can be inside pyproject.toml, the section of black is [tool.black]. For example, your configuration may look like the following:

[tool.black]
line-length = 100

You can also check how the black project configures itself here.

ref:

Make Pylsp work inside a virtual env
#

In my computer, pylsp, mypy, black, pylint and isort are all installed globally. I do not want to install all of these tools separately for each virtual env I use. For black and isort, they seem to work fine without any additional settings.

For Pylsp, there seem to be not much mature solution1. I experimented with different configuration and find what is working for me.

Change PYTHONPATH
#

We can also update the PYTHONPATH variable to make the global pylint and mypy be aware of the packages installed inside the virtual environment. That is we can set variable to the package directory for the virtual environment:

export PYTHONPATH=$PYTHONPATH:/path/to/project/venv/lib/<python-version>/site-packages

After this setup, pylint and mypy seem to work fine.

ref:

https://docs.python.org/3/using/cmdline.html#envvar-PYTHONPATH

Change config for individual tool
#

Pylint
#

For pylint installed globally, it can not recognize the packages installed inside the virtual environment, even if we are inside the virtual env. First, we can install pylint-venv in the global level (outside the virtual env):

pip install pylint-venv

Then inside the pyproject.toml file for this project, we can configure a init-hook for pylint:

[tool.pylint.main]
init-hook ="""
try:
  import pylint_venv
except ImportError:
  pass
else:
  pylint_venv.inithook()
"""

This seems to work pretty good to prevent pylint from complaining missing package errors.

mypy
#

To use global mypy, we need to instruct mypy the python path inside the virtual environment.

[tool.mypy]
python_executable="../venv/bin/python"

Another way to do this is to directly change the options passed to mypy in neovim-lsp configuration:

  local venv_path = os.getenv('VIRTUAL_ENV')
  local py_path = nil
  -- decide which python executable to use for mypy
  if venv_path ~= nil then
    py_path = venv_path .. "/bin/python3"
  else
    py_path = vim.g.python3_host_prog
  end

  lspconfig.pylsp.setup {
    on_attach = custom_attach,
    settings = {
      pylsp = {
        plugins = {
          -- formatter options
          black = { enabled = true },
          autopep8 = { enabled = false },
          yapf = { enabled = false },
          -- linter options
          pylint = { enabled = true, executable = "pylint" },
          ruff = { enabled = false },
          pyflakes = { enabled = false },
          pycodestyle = { enabled = false },
          -- type checker
          pylsp_mypy = {
            enabled = true,
            overrides = { "--python-executable", py_path, true },
            report_progress = true,
            live_mode = false
          },
          -- auto-completion options
          jedi_completion = { fuzzy = true },
          -- import sorting
          isort = { enabled = true },
        },
      },
    },
    flags = {
      debounce_text_changes = 200,
    },
    capabilities = capabilities,
  }

ref:

References
#


  1. For pyright, there seems to be a simple solution using pyrightconfig.json, as discussed here↩︎

Related

Setting up Sumneko Lua Language Server for Nvim-lsp
··360 words·2 mins
Setting up Neovim for C++ Development with LSP
··1254 words·6 mins
Set up Fuzzy Completion for Vim-lsp
··267 words·2 mins