Skip to main content

Workspace symbols, inlay hints, code lens, semantic tokens, call/type hierarchy, document links and colors for python-lsp-server via Jedi

Project description

pylsp-workspace-symbols

CI PyPI PyPI - Python Version Downloads License: MIT

A python-lsp-server plugin that adds workspace symbol search, inlay hints, code lens, semantic tokens, call/type hierarchy, document links and document colors via Jedi.

Why? pylsp does not implement several LSP features natively. This plugin fills those gaps — enabling "Go to Symbol in Workspace", rich type inference hints, code lens overlays, semantic token highlighting, call/type hierarchy navigation, clickable import links, and inline color previews in any LSP client — including CudaText, Neovim, Emacs, and others — with broad client compatibility out of the box.


✨ Features

  • 🔍 Workspace-wide symbol search — find functions, classes, and modules across all files in the project
  • 💡 Inlay hints — inline type annotations inferred by Jedi for assignments, return types, raised exceptions, and parameter names at call sites
  • 🔭 Code lens — per-definition overlays showing reference counts (👥 N references), subclass/override counts (🔗 N implementations), run entry points (▶ Run), and test markers (🧪 Run test)
  • 🎨 Semantic tokens — Jedi-backed token classification for editors that support textDocument/semanticTokens (opt-in, disabled by default)
  • 🌳 Call hierarchy — navigate callers and callees of any function via callHierarchy/incomingCalls and callHierarchy/outgoingCalls
  • 🧬 Type hierarchy — explore supertypes and subtypes of any class via typeHierarchy/supertypes and typeHierarchy/subtypes
  • 🔗 Document links — clickable links for URLs in comments/strings and import statements (resolves to stdlib source when available)
  • 🎨 Document colors — inline color previews for CSS/hex/RGB/HSL color literals in source files
  • 🔌 Broad client compatibility — capabilities announced via proper LSP channel (works with Neovim, eglot, and any client that does not support experimental capabilities), with automatic fallback to the experimental channel
  • Fast — results in ~130ms after the first call (Jedi cache warm)
  • 🔤 Case-insensitive substring matcharea finds calculate_area, Cal finds Calculator
  • 📁 Smart folder exclusion — automatically skips .git, __pycache__, node_modules, .venv, dist, build, and more
  • ⚙️ Configurable — tune all options via pylsp settings
  • 🐍 Python 3.8+ — compatible with all modern Python versions

📦 Installation

pip install pylsp-workspace-symbols

The plugin is discovered automatically by pylsp via its entry point — no manual configuration needed.

⚙️ Configuration

Add to your LSP client's pylsp settings (e.g. in settings.json or equivalent):

{
  "pylsp": {
    "plugins": {
      "jedi_workspace_symbols": {
        "enabled": true,
        "max_symbols": 500,
        "ignore_folders": []
      },
      "inlay_hints": {
        "enabled": true,
        "show_assign_types": true,
        "show_return_types": true,
        "show_raises": true,
        "show_parameter_hints": true,
        "max_hints_per_file": 200
      },
      "code_lens": {
        "enabled": true,
        "show_references": true,
        "show_implementations": true,
        "cross_file_implementations": false,
        "show_run": true,
        "show_tests": true,
        "max_definitions": 150
      },
      "semantic_tokens": {
        "enabled": false
      },
      "call_hierarchy": {
        "enabled": true
      },
      "type_hierarchy": {
        "enabled": true
      },
      "document_links": {
        "enabled": true
      },
      "document_colors": {
        "enabled": true
      }
    }
  }
}

Workspace symbol options

Option Type Default Description
enabled bool true Enable/disable workspace symbol search
max_symbols int 500 Maximum symbols returned. 0 means no limit
ignore_folders list [] Extra folder names to skip (merged with built-in list)

Inlay hint options

Option Type Default Description
enabled bool true Enable/disable all inlay hints
show_assign_types bool true Show inferred types for unannotated assignments (x = 42: int)
show_return_types bool true Show inferred return types for unannotated functions (def f():-> str)
show_raises bool true Show raised exception types (raise ValueError(...)Raises: ValueError)
show_parameter_hints bool true Show parameter names at call sites (f(1, 2)a=1, b=2)
max_hints_per_file int 200 Maximum hints per file. 0 means no limit

Code lens options

Option Type Default Description
enabled bool true Enable/disable all code lenses
show_references bool true Show 👥 N references above every function, method, and class
show_implementations bool true Show 🔗 N implementations on classes with subclasses and methods with overrides
cross_file_implementations bool false Extend 🔗 counts to subclasses/overrides in other files. Opt-in — adds one get_references() call + file I/O per class/method
show_run bool true Show ▶ Run above if __name__ == "__main__": blocks
show_tests bool true Show 🧪 Run test above test_* functions and Test* classes
max_definitions int 150 Maximum number of definitions to process per file

Semantic token options

Option Type Default Description
enabled bool false Enable/disable semantic token highlighting. Opt-in — can be slow on very large files

Call hierarchy options

Option Type Default Description
enabled bool true Enable/disable call hierarchy (callHierarchy/incomingCalls, callHierarchy/outgoingCalls)

Type hierarchy options

Option Type Default Description
enabled bool true Enable/disable type hierarchy (typeHierarchy/supertypes, typeHierarchy/subtypes)

Document links options

Option Type Default Description
enabled bool true Enable/disable document links (URLs in comments/strings and import resolution)

Document colors options

Option Type Default Description
enabled bool true Enable/disable document color previews (hex, RGB, HSL, CSS named colors)

Built-in ignored folders

.git, .hg, .svn, __pycache__, .mypy_cache, .ruff_cache, .pytest_cache, node_modules, .venv, venv, .env, env, dist, build, .eggs, egg-info

🚀 Usage

Workspace symbol search

Once installed, your LSP client will receive workspaceSymbolProvider: true in the server capabilities. Use your client's "Go to Symbol in Workspace" command (typically Ctrl+T or # in the symbol picker).

Inlay hints

Your LSP client will receive inlayHintProvider: true in the server capabilities. Hints are rendered inline by the client automatically. The following hint types are supported:

  • Assignment hints — unannotated variable assignments, including self.attr in __init__
  • Return hints — unannotated def and async def functions, inferred from the first return statement
  • Raise hintsraise ExceptionType(...) statements
  • Parameter hints — positional argument names at call sites (keyword args are skipped as self-documenting)

Inlay hints respect type annotations already present in the source — annotated functions and variables are never hinted twice.

Code lens

Your LSP client will receive codeLensProvider: true. The following overlays are shown above definitions:

  • 👥 N references — every top-level function, method, and class. Inheritance usages (class Dog(Animal)) are excluded from the reference count and counted as implementations instead. Cross-file inheritance usages are also excluded when cross_file_implementations=true.
  • 🔗 N implementations — classes with direct subclasses; methods overridden in at least one subclass. Intra-file always; cross-file when cross_file_implementations=true (opt-in).
  • ▶ Runif __name__ == "__main__": entry point blocks. Fires workspace/executeCommand with command pylsp_workspace_symbols.run_file.
  • 🧪 Run testtest_* functions and Test* / unittest.TestCase subclasses. Fires workspace/executeCommand with command pylsp_workspace_symbols.run_test, passing {"path": ..., "name": ..., "kind": "function"|"class"} as arguments.

Semantic tokens

Your LSP client will receive semanticTokensProvider when semantic_tokens.enabled is true. Token types follow an extended LSP legend: namespace, type, class, enum, interface, struct, typeParameter, parameter, variable, property, enumMember, function, method, macro, keyword, comment, string, number, regexp, operator, decorator, plus Python-specific selfParameter and clsParameter. Modifiers include declaration, readonly, static, deprecated, async, modification, documentation, defaultLibrary, builtin, classMember, and parameter. Disabled by default — enable explicitly if your client supports it and you want Jedi-backed token classification in addition to your editor's built-in lexer.

Call hierarchy

Your LSP client will receive callHierarchyProvider: true. Place the cursor on any function name and invoke "Show Call Hierarchy" to see incoming callers and outgoing callees. Compatible with any LSP client that supports the standard callHierarchy/* requests (Neovim, eglot, CudaText, etc.).

Type hierarchy

Your LSP client will receive typeHierarchyProvider: true. Place the cursor on any class name and invoke "Show Type Hierarchy" to explore supertypes and subtypes. Compatible with any LSP client that supports the standard typeHierarchy/* requests (Neovim, eglot, CudaText, etc.).

Document links

Your LSP client will receive documentLinkProvider: true. The following are turned into clickable links:

  • URLshttp:// and https:// links in comments and strings
  • Import statements — resolved to the corresponding .py source file in the system Python's Lib/ directory (when Python is installed and available on PATH); modules without a .py source (C extensions, frozen modules, embedded-only .pyc) are silently skipped
  • Workspace path literals — relative path strings (e.g. "./config.json", "../data/file.csv") and open()/Path() calls that reference files present in the workspace

Document colors

Your LSP client will receive colorProvider: true. Inline color swatches are shown for: hex (#rgb, #rrggbb, #rrggbbaa), rgb()/rgba(), hsl()/hsla(), and CSS named colors.

🔍 How it works

Workspace symbols

pylsp does not define a pylsp_workspace_symbols hookspec, so this plugin uses a two-pronged approach:

  1. Capability injection (preferred) — at import time, monkey-patches PythonLSPServer.capabilities() to insert workspaceSymbolProvider: true and inlayHintProvider directly into the proper LSP capabilities dict. This makes the plugin work out-of-the-box with clients that require proper capabilities, such as Neovim and eglot.
  2. Experimental fallback — if the injection fails (e.g. pylsp changes its internal API), capabilities are announced via pylsp_experimental_capabilities instead. Clients that honour the experimental channel (CudaText, VSCode with pylsp, etc.) will still work.
  3. pylsp_dispatchers — registers a custom JSON-RPC handler for workspace/symbol that calls Jedi's project.complete_search() and filters results client-side by case-insensitive substring match.

Results are limited to files inside the known workspace folders. All open workspace roots are read from the live server at query time via server.workspaces, so folders added after startup (workspace/didChangeWorkspaceFolders) are included correctly. Each root is searched with sys_path=[root], restricting Jedi's indexing to that folder only — avoiding the full Python environment (stdlib + site-packages). This yields an ~80x speedup on complete_search compared to the default Jedi project. A _is_relative_to() guard (Python 3.8-compatible replacement for Path.is_relative_to, which requires 3.9+) provides a second layer of filtering.

Note: workspace/symbol returns module-level definitions (functions, classes, modules). Local variables inside functions are not indexed — this is standard LSP behaviour, consistent with pyright and other Python language servers.

Inlay hints

The plugin handles the textDocument/inlayHint request using a hybrid approach:

  1. Regex scan — fast pass over the source to locate def, assignment, raise, and call patterns.
  2. _literal_type fast-path — resolves common literals ("str", 42, True, [...], etc.) without calling Jedi.
  3. Jedi inference — for non-literal expressions, script.infer() and script.get_signatures() are used to resolve types.
  4. Signature fallback — for self.attr = param assignments, the enclosing def signature is inspected for type annotations or default values.

Code lens

Handled via the native pylsp_code_lens hookspec. Uses a two-pass approach:

  1. AST pass — single ast.walk over the file to collect all definitions and build intra-file maps of subclass relationships (class_subclasses) and method overrides (method_overrides). Also pre-computes inheritance usage positions (intra-file) to correctly separate reference counts from implementation counts.
  2. Jedi reference pass — one script.get_references() call per definition to count non-definition references. Cross-file inheritance positions are excluded from the reference count using _find_cross_file_subclasses (only when cross_file_implementations=True). Results are cached by (uri, hash(source)) so repeated requests on an unchanged file skip all work.

When cross_file_implementations=True, _find_cross_file_subclasses uses script.get_references() with the workspace project to find subclasses in other files, verifying each ref against the AST of the referenced file. A per-request _cf_subclass_cache ensures the I/O is shared between the references filter and the implementations count.

Semantic tokens

Handled via textDocument/semanticTokens/full, full/delta, and range dispatchers. Uses a two-phase O(n) approach — no per-token goto calls:

  1. AST pass — single ast.parse() walk to build lookup tables for token types and modifiers that Jedi alone cannot determine: enumMember, typeParameter, decorator, @classmethod/@staticmethodstatic, ClassVar/Finalreadonly, deprecated (via decorator or docstring), async, and augmented assignments → modification.
  2. Jedi passjedi.Script.get_names(all_scopes=True) provides the base token stream. A two-sub-pass strategy resolves statement references via lookup dicts built in phase 1, avoiding any per-token inference calls.

Beyond the two main passes, a token injection stage handles symbols that neither Jedi nor the AST walk emit directly: @ decorator markers, names inside forward-reference annotation strings, type parameter attributes (e.g. ParamSpec.args), and regexp patterns in re.compile() calls. Each injected token receives the correct type and modifiers using the same lookup tables built in the AST pass. Delta computation uses SequenceMatcher to produce minimal SemanticTokensEdit[] arrays.

Call hierarchy

Handled via callHierarchy/incomingCalls and callHierarchy/outgoingCalls dispatchers. Uses Jedi's script.goto() and script.get_references() to resolve callers and callees, building LSP-compliant CallHierarchyItem structures with correct range information. Functions decorated with @overload are handled correctly — Jedi returns all overload stubs as references to the real implementation (shared qualified name); incoming calls filters them out via ref.is_definition(), which is True for all definition sites and False for actual call sites.

Type hierarchy

Handled via typeHierarchy/supertypes and typeHierarchy/subtypes dispatchers. Uses Jedi's script.goto() and static class inheritance analysis to build the type tree.

Document links

Three-pass collection over the source:

  1. URL pass — regex scan for http:// and https:// URLs in comments, docstrings, and string literals.
  2. Import pass — AST parse to find all import and from ... import statements; resolves each module name to a .py file by querying the system Python's sys.prefix via a single cached subprocess call.
  3. Path literal pass — detects relative path strings, open() calls and Path() calls whose argument resolves to an existing file inside the workspace root.

Modules without a .py source (C extensions, frozen modules, .pyc-only embedded builds) are silently skipped.

Document colors

Regex-based scan over the source for color literals: hex (#rgb, #rrggbb, #rrggbbaa), rgb()/rgba(), hsl()/hsla(), and the full set of CSS named colors. Each match is returned as an LSP ColorInformation with normalised [0.0, 1.0] RGBA components.

🧪 Tests

pip install -e ".[dev]"
pytest

🤝 Contributing

Issues and pull requests are welcome! Please open an issue before submitting a large change.

📚 References

👤 Author

Bruno Eduardo — github.com/Hanatarou

📄 License

MIT

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

pylsp_workspace_symbols-0.6.2.tar.gz (66.7 kB view details)

Uploaded Source

Built Distribution

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

pylsp_workspace_symbols-0.6.2-py3-none-any.whl (54.6 kB view details)

Uploaded Python 3

File details

Details for the file pylsp_workspace_symbols-0.6.2.tar.gz.

File metadata

  • Download URL: pylsp_workspace_symbols-0.6.2.tar.gz
  • Upload date:
  • Size: 66.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for pylsp_workspace_symbols-0.6.2.tar.gz
Algorithm Hash digest
SHA256 f683d7521ad386a388b896e24b79f04c51a229662976653f89a072d0552c2cb1
MD5 8562682f1c3a340505ae0cd65db9b79f
BLAKE2b-256 c7887bede355836e103d927fe693aaffa3d24920f55b694366df6d2807e0fb36

See more details on using hashes here.

Provenance

The following attestation bundles were made for pylsp_workspace_symbols-0.6.2.tar.gz:

Publisher: publish.yml on Hanatarou/pylsp-workspace-symbols

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file pylsp_workspace_symbols-0.6.2-py3-none-any.whl.

File metadata

File hashes

Hashes for pylsp_workspace_symbols-0.6.2-py3-none-any.whl
Algorithm Hash digest
SHA256 970db709db0d76342b0091d759f23f0d41a0fbb00c87d665c288a263a5b17502
MD5 f6987e0119eb156bb3220dbfbcc13f8e
BLAKE2b-256 118d2ee8fe6b8e60d7be125ce3f701b27fe50c5c94970ab3ad3b07d450422fce

See more details on using hashes here.

Provenance

The following attestation bundles were made for pylsp_workspace_symbols-0.6.2-py3-none-any.whl:

Publisher: publish.yml on Hanatarou/pylsp-workspace-symbols

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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