Configures Python LSP support for Emacs using Eglot, including support for PEP-723 script metadata and automatic project detection.
This package configures Eglot to work with Python files using ty, basedpyright, or rassumfrassum as the language server frontend. It automatically handles environment synchronization for uv-managed scripts with inline dependencies.
- Emacs 30.2 or later
- uv - fast Python package installer and resolver
- One of the following language servers:
- ty (>= v0.0.8) - Astral's new Python type
checker. It can be installed globally or in a project-root
.venv. - basedpyright - fork of pyright
with additional features. It can be installed globally or in a project-root
.venv. - rassumfrassum (>= v0.3.3) - optional stdio multiplexer for combining multiple Python tools.
- ty (>= v0.0.8) - Astral's new Python type
checker. It can be installed globally or in a project-root
Choose one of the following ways to install. After that, opening Python files will automatically start the LSP server using Eglot.
(use-package eglot-python-preset
:ensure t
:after eglot
:custom
(eglot-python-preset-lsp-server 'ty) ; or 'basedpyright or 'rass
:config
(eglot-python-preset-setup))Clone this repository and add to your Emacs configuration:
mkdir -p ~/devel
git clone https://github.com/mwolson/eglot-python-preset ~/devel/eglot-python-preset(add-to-list 'load-path (expand-file-name "~/devel/eglot-python-preset"))
(require 'eglot-python-preset)
(setopt eglot-python-preset-lsp-server 'ty) ; or 'basedpyright or 'rass
(eglot-python-preset-setup)For standard Python projects (those with pyproject.toml or
requirements.txt), the package automatically detects the project root and
starts Eglot with appropriate configuration.
PEP-723 allows embedding dependency metadata directly in Python scripts using special comments:
# /// script
# requires-python = ">=3.10"
# dependencies = [
# "requests>=2.31.0",
# ]
# ///
import requests
response = requests.get("https://example.com")
print(response.status_code)When you open a file containing PEP-723 metadata:
- The package detects the script metadata automatically
- It locates or allows you to manually create an isolated environment via
uv - The LSP server is configured to use that environment for type checking
-
M-x eglot-python-preset-sync-environment- Sync dependencies for the current PEP-723 script usinguv sync --script, then restart Eglot. Use this after adding or modifying dependencies in your script's metadata block. -
M-x eglot-python-preset-remove-environment- Remove the cached uv environment for the current PEP-723 script. Useful when you want to force a clean reinstall of dependencies or troubleshoot environment issues. After removal, runeglot-python-preset-sync-environmentto recreate it. -
M-x eglot-python-preset-run-script- Run the current PEP-723 script usinguv run. Opens a compilation buffer with the output.
Choose which language server to use:
(setopt eglot-python-preset-lsp-server 'ty) ; default
;; or
(setopt eglot-python-preset-lsp-server 'basedpyright)
;; or
(setopt eglot-python-preset-lsp-server 'rass)When using the rass backend, this list controls the generated preset.
Supported symbolic tools get local .venv executable resolution and special
handling for PEP-723 where available.
Literal commands in a vector are passed through to rass:
(setopt eglot-python-preset-rass-tools
'(ty
ruff
["custom-lsp" "--stdio"]))If you want to pass arguments to a built-in tool, spell that entry out as a
literal vector. For example, Ruff supports top-level --isolated before the
server subcommand, which can be useful when you want to ignore ambient Ruff
configuration files:
(setopt eglot-python-preset-rass-tools
'(ty
["ruff" "--isolated" "server"]))If you want to bypass the generated preset entirely, set an exact rass command
vector. When this is non-nil, it is used verbatim and
eglot-python-preset-rass-tools is ignored:
(setopt eglot-python-preset-rass-command ["rass" "python"])Or with a custom preset path:
(setopt eglot-python-preset-rass-command
["rass" "/path/to/custom-preset.py"])Note that this will remove support for PEP-723 scripts unless the preset is updated to handle it.
Files that indicate a Python project root:
(setopt eglot-python-preset-python-project-markers
'("pyproject.toml" "requirements.txt")) ; defaultContextual rass presets are the ones that embed per-script or project-local
state, such as PEP-723 Ty configuration or a project-local .venv executable
path. Older contextual presets are pruned automatically to keep the generated
directory from growing without bound:
(setopt eglot-python-preset-rass-max-contextual-presets 50) ; defaultTo customize basedpyright settings, set eglot-workspace-configuration before
calling eglot-python-preset-setup. Your settings will be merged with PEP-723
script configurations (which add the Python interpreter path).
Example to disable auto-import completions and set type checking mode:
;; With use-package
(use-package eglot-python-preset
:ensure t
:after eglot
:custom
(eglot-python-preset-lsp-server 'basedpyright)
(eglot-workspace-configuration
'(:basedpyright.analysis
(:autoImportCompletions :json-false
:typeCheckingMode "basic")))
:config
(eglot-python-preset-setup))
;; Or manually (can be spread across multiple init.el sections)
(setopt eglot-python-preset-lsp-server 'basedpyright)
(setopt eglot-workspace-configuration
(plist-put eglot-workspace-configuration
:basedpyright.analysis
'(:autoImportCompletions :json-false
:typeCheckingMode "basic")))
(eglot-python-preset-setup)- Eglot publishes diagnostics through Flymake. If you are using Flycheck, you will need separate bridge or integration configuration in your Emacs setup.
- If
tyorbasedpyright-langserveris installed only in a project-local.venv, make sure you are using v0.3.0 or later so this package can prefer that executable automatically. - If you use the
rassbackend, the package generates a preset under your Emacs directory and updates it as needed. Context-free presets are reused across buffers, while PEP-723 and project-local.venvcases keep separate generated files when the preset content depends on that local context.
- The package uses
uvfor all Python environment management. Ensureuvis installed and in your PATH. - For standard Python projects, the package prefers
tyorbasedpyright-langserverfrom a project-root.venvand otherwise falls back to PATH. The same resolution is used for supported tools in generatedrasspresets. - For PEP-723 scripts, environments are cached by
uvand shared across sessions. - If you see a warning about the environment not being synced, run
M-x eglot-python-preset-sync-environment.