Initial commit: SCIP indexer for Clojure

Features:
- Generate SCIP index from clojure-lsp dump
- Namespace definitions and usages
- Var definitions and usages
- Namespace alias definitions and usage references
- External symbol documentation for hover

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
2026-02-03 16:49:10 -05:00
commit 3f669b422a
6 changed files with 23482 additions and 0 deletions
+5
View File
@@ -0,0 +1,5 @@
.cpcache/
classes/
*.jar
.lsp/
.clj-kondo/
+14
View File
@@ -0,0 +1,14 @@
(ns build
(:require [clojure.tools.build.api :as b]))
(def class-dir "classes")
(def java-src "java-src")
(defn compile-java [_]
(b/javac {:src-dirs [java-src]
:class-dir class-dir
:basis (b/create-basis {:project "deps.edn"})
:javac-opts ["--release" "11"]}))
(defn clean [_]
(b/delete {:path class-dir}))
+16
View File
@@ -0,0 +1,16 @@
{:paths ["src" "classes"]
:deps {org.clojure/clojure {:mvn/version "1.12.0"}
com.google.protobuf/protobuf-java {:mvn/version "3.25.3"}
org.clojure/tools.cli {:mvn/version "1.1.230"}}
:aliases
{:build {:extra-deps {io.github.clojure/tools.build {:mvn/version "0.10.5"}}
:ns-default build}
:run {:main-opts ["-m" "scip-clojure.core"]}
:uberjar {:extra-deps {com.github.seancorfield/depstar {:mvn/version "2.1.303"}}
:exec-fn hf.depstar/uberjar
:exec-args {:jar "scip-clojure.jar"
:aot true
:main-class scip_clojure.core}}}}
File diff suppressed because it is too large Load Diff
+885
View File
@@ -0,0 +1,885 @@
// An index contains one or more pieces of information about a given piece of
// source code or software artifact. Complementary information can be merged
// together from multiple sources to provide a unified code intelligence
// experience.
//
// Programs producing a file of this format is an "indexer" and may operate
// somewhere on the spectrum between precision, such as indexes produced by
// compiler-backed indexers, and heurstics, such as indexes produced by local
// syntax-directed analysis for scope rules.
syntax = "proto3";
package scip;
option go_package = "github.com/sourcegraph/scip/bindings/go/scip/";
// Index represents a complete SCIP index for a workspace this is rooted at a
// single directory. An Index message payload can have a large memory footprint
// and it's therefore recommended to emit and consume an Index payload one field
// value at a time. To permit streaming consumption of an Index payload, the
// `metadata` field must appear at the start of the stream and must only appear
// once in the stream. Other field values may appear in any order.
message Index {
// Metadata about this index.
Metadata metadata = 1;
// Documents that belong to this index.
repeated Document documents = 2;
// (optional) Symbols that are referenced from this index but are defined in
// an external package (a separate `Index` message). Leave this field empty
// if you assume the external package will get indexed separately. If the
// external package won't get indexed for some reason then you can use this
// field to provide hover documentation for those external symbols.
repeated SymbolInformation external_symbols = 3;
// IMPORTANT: When adding a new field to `Index` here, add a matching
// function in `IndexVisitor` and update `ParseStreaming`.
}
message Metadata {
// Which version of this protocol was used to generate this index?
ProtocolVersion version = 1;
// Information about the tool that produced this index.
ToolInfo tool_info = 2;
// URI-encoded absolute path to the root directory of this index. All
// documents in this index must appear in a subdirectory of this root
// directory.
string project_root = 3;
// Text encoding of the source files on disk that are referenced from
// `Document.relative_path`. This value is unrelated to the `Document.text`
// field, which is a Protobuf string and hence must be UTF-8 encoded.
TextEncoding text_document_encoding = 4;
}
enum ProtocolVersion {
UnspecifiedProtocolVersion = 0;
}
enum TextEncoding {
UnspecifiedTextEncoding = 0;
UTF8 = 1;
UTF16 = 2;
}
message ToolInfo {
// Name of the indexer that produced this index.
string name = 1;
// Version of the indexer that produced this index.
string version = 2;
// Command-line arguments that were used to invoke this indexer.
repeated string arguments = 3;
}
// Document defines the metadata about a source file on disk.
message Document {
// The string ID for the programming language this file is written in.
// The `Language` enum contains the names of most common programming languages.
// This field is typed as a string to permit any programming language, including
// ones that are not specified by the `Language` enum.
string language = 4;
// (Required) Unique path to the text document.
//
// 1. The path must be relative to the directory supplied in the associated
// `Metadata.project_root`.
// 2. The path must not begin with a leading '/'.
// 3. The path must point to a regular file, not a symbolic link.
// 4. The path must use '/' as the separator, including on Windows.
// 5. The path must be canonical; it cannot include empty components ('//'),
// or '.' or '..'.
string relative_path = 1;
// Occurrences that appear in this file.
repeated Occurrence occurrences = 2;
// Symbols that are "defined" within this document.
//
// This should include symbols which technically do not have any definition,
// but have a reference and are defined by some other symbol (see
// Relationship.is_definition).
repeated SymbolInformation symbols = 3;
// (optional) Text contents of the this document. Indexers are not expected to
// include the text by default. It's preferrable that clients read the text
// contents from the file system by resolving the absolute path from joining
// `Index.metadata.project_root` and `Document.relative_path`. This field was
// introduced to support `SymbolInformation.signature_documentation`, but it
// can be used for other purposes as well, for example testing or when working
// with virtual/in-memory documents.
string text = 5;
// Specifies the encoding used for source ranges in this Document.
//
// Usually, this will match the type used to index the string type
// in the indexer's implementation language in O(1) time.
// - For an indexer implemented in JVM/.NET language or JavaScript/TypeScript,
// use UTF16CodeUnitOffsetFromLineStart.
// - For an indexer implemented in Python,
// use UTF32CodeUnitOffsetFromLineStart.
// - For an indexer implemented in Go, Rust or C++,
// use UTF8ByteOffsetFromLineStart.
PositionEncoding position_encoding = 6;
}
// Encoding used to interpret the 'character' value in source ranges.
enum PositionEncoding {
// Default value. This value should not be used by new SCIP indexers
// so that a consumer can process the SCIP index without ambiguity.
UnspecifiedPositionEncoding = 0;
// The 'character' value is interpreted as an offset in terms
// of UTF-8 code units (i.e. bytes).
//
// Example: For the string "🚀 Woo" in UTF-8, the bytes are
// [240, 159, 154, 128, 32, 87, 111, 111], so the offset for 'W'
// would be 5.
UTF8CodeUnitOffsetFromLineStart = 1;
// The 'character' value is interpreted as an offset in terms
// of UTF-16 code units (each is 2 bytes).
//
// Example: For the string "🚀 Woo", the UTF-16 code units are
// ['\ud83d', '\ude80', ' ', 'W', 'o', 'o'], so the offset for 'W'
// would be 3.
UTF16CodeUnitOffsetFromLineStart = 2;
// The 'character' value is interpreted as an offset in terms
// of UTF-32 code units (each is 4 bytes).
//
// Example: For the string "🚀 Woo", the UTF-32 code units are
// ['🚀', ' ', 'W', 'o', 'o'], so the offset for 'W' would be 2.
UTF32CodeUnitOffsetFromLineStart = 3;
}
// Symbol is similar to a URI, it identifies a class, method, or a local
// variable. `SymbolInformation` contains rich metadata about symbols such as
// the docstring.
//
// Symbol has a standardized string representation, which can be used
// interchangeably with `Symbol`. The syntax for Symbol is the following:
// ```
// # (<x>)+ stands for one or more repetitions of <x>
// # (<x>)? stands for zero or one occurrence of <x>
// <symbol> ::= <scheme> ' ' <package> ' ' (<descriptor>)+ | 'local ' <local-id>
// <package> ::= <manager> ' ' <package-name> ' ' <version>
// <scheme> ::= any UTF-8, escape spaces with double space. Must not be empty nor start with 'local'
// <manager> ::= any UTF-8, escape spaces with double space. Use the placeholder '.' to indicate an empty value
// <package-name> ::= same as above
// <version> ::= same as above
// <descriptor> ::= <namespace> | <type> | <term> | <method> | <type-parameter> | <parameter> | <meta> | <macro>
// <namespace> ::= <name> '/'
// <type> ::= <name> '#'
// <term> ::= <name> '.'
// <meta> ::= <name> ':'
// <macro> ::= <name> '!'
// <method> ::= <name> '(' (<method-disambiguator>)? ').'
// <type-parameter> ::= '[' <name> ']'
// <parameter> ::= '(' <name> ')'
// <name> ::= <identifier>
// <method-disambiguator> ::= <simple-identifier>
// <identifier> ::= <simple-identifier> | <escaped-identifier>
// <simple-identifier> ::= (<identifier-character>)+
// <identifier-character> ::= '_' | '+' | '-' | '$' | ASCII letter or digit
// <escaped-identifier> ::= '`' (<escaped-character>)+ '`', must contain at least one non-<identifier-character>
// <escaped-characters> ::= any UTF-8, escape backticks with double backtick.
// <local-id> ::= <simple-identifier>
// ```
//
// The list of descriptors for a symbol should together form a fully
// qualified name for the symbol. That is, it should serve as a unique
// identifier across the package. Typically, it will include one descriptor
// for every node in the AST (along the ancestry path) between the root of
// the file and the node corresponding to the symbol.
//
// Local symbols MUST only be used for entities which are local to a Document,
// and cannot be accessed from outside the Document.
message Symbol {
string scheme = 1;
Package package = 2;
repeated Descriptor descriptors = 3;
}
// Unit of packaging and distribution.
//
// NOTE: This corresponds to a module in Go and JVM languages.
message Package {
string manager = 1;
string name = 2;
string version = 3;
}
message Descriptor {
enum Suffix {
option allow_alias = true;
UnspecifiedSuffix = 0;
// Unit of code abstraction and/or namespacing.
//
// NOTE: This corresponds to a package in Go and JVM languages.
Namespace = 1;
// Use Namespace instead.
Package = 1 [deprecated=true];
Type = 2;
Term = 3;
Method = 4;
TypeParameter = 5;
Parameter = 6;
// Can be used for any purpose.
Meta = 7;
Local = 8;
Macro = 9;
}
string name = 1;
string disambiguator = 2;
Suffix suffix = 3;
// NOTE: If you add new fields here, make sure to update the prepareSlot()
// function responsible for parsing symbols.
}
// SymbolInformation defines metadata about a symbol, such as the symbol's
// docstring or what package it's defined it.
message SymbolInformation {
// Identifier of this symbol, which can be referenced from `Occurence.symbol`.
// The string must be formatted according to the grammar in `Symbol`.
string symbol = 1;
// (optional, but strongly recommended) The markdown-formatted documentation
// for this symbol. Use `SymbolInformation.signature_documentation` to
// document the method/class/type signature of this symbol.
// Due to historical reasons, indexers may include signature documentation in
// this field by rendering markdown code blocks. New indexers should only
// include non-code documentation in this field, for example docstrings.
repeated string documentation = 3;
// (optional) Relationships to other symbols (e.g., implements, type definition).
repeated Relationship relationships = 4;
// The kind of this symbol. Use this field instead of
// `SymbolDescriptor.Suffix` to determine whether something is, for example, a
// class or a method.
Kind kind = 5;
// (optional) Kind represents the fine-grained category of a symbol, suitable for presenting
// information about the symbol's meaning in the language.
//
// For example:
// - A Java method would have the kind `Method` while a Go function would
// have the kind `Function`, even if the symbols for these use the same
// syntax for the descriptor `SymbolDescriptor.Suffix.Method`.
// - A Go struct has the symbol kind `Struct` while a Java class has
// the symbol kind `Class` even if they both have the same descriptor:
// `SymbolDescriptor.Suffix.Type`.
//
// Since Kind is more fine-grained than Suffix:
// - If two symbols have the same Kind, they should share the same Suffix.
// - If two symbols have different Suffixes, they should have different Kinds.
enum Kind {
UnspecifiedKind = 0;
// A method which may or may not have a body. For Java, Kotlin etc.
AbstractMethod = 66;
// For Ruby's attr_accessor
Accessor = 72;
Array = 1;
// For Alloy
Assertion = 2;
AssociatedType = 3;
// For C++
Attribute = 4;
// For Lean
Axiom = 5;
Boolean = 6;
Class = 7;
// For C++
Concept = 86;
Constant = 8;
Constructor = 9;
// For Solidity
Contract = 62;
// For Haskell
DataFamily = 10;
// For C# and F#
Delegate = 73;
Enum = 11;
EnumMember = 12;
Error = 63;
Event = 13;
// For Dart
Extension = 84;
// For Alloy
Fact = 14;
Field = 15;
File = 16;
Function = 17;
// For 'get' in Swift, 'attr_reader' in Ruby
Getter = 18;
// For Raku
Grammar = 19;
// For Purescript and Lean
Instance = 20;
Interface = 21;
Key = 22;
// For Racket
Lang = 23;
// For Lean
Lemma = 24;
// For solidity
Library = 64;
Macro = 25;
Method = 26;
// For Ruby
MethodAlias = 74;
// Analogous to 'ThisParameter' and 'SelfParameter', but for languages
// like Go where the receiver doesn't have a conventional name.
MethodReceiver = 27;
// Analogous to 'AbstractMethod', for Go.
MethodSpecification = 67;
// For Protobuf
Message = 28;
// For Dart
Mixin = 85;
// For Solidity
Modifier = 65;
Module = 29;
Namespace = 30;
Null = 31;
Number = 32;
Object = 33;
Operator = 34;
Package = 35;
PackageObject = 36;
Parameter = 37;
ParameterLabel = 38;
// For Haskell's PatternSynonyms
Pattern = 39;
// For Alloy
Predicate = 40;
Property = 41;
// Analogous to 'Trait' and 'TypeClass', for Swift and Objective-C
Protocol = 42;
// Analogous to 'AbstractMethod', for Swift and Objective-C.
ProtocolMethod = 68;
// Analogous to 'AbstractMethod', for C++.
PureVirtualMethod = 69;
// For Haskell
Quasiquoter = 43;
// 'self' in Python, Rust, Swift etc.
SelfParameter = 44;
// For 'set' in Swift, 'attr_writer' in Ruby
Setter = 45;
// For Alloy, analogous to 'Struct'.
Signature = 46;
// For Ruby
SingletonClass = 75;
// Analogous to 'StaticMethod', for Ruby.
SingletonMethod = 76;
// Analogous to 'StaticField', for C++
StaticDataMember = 77;
// For C#
StaticEvent = 78;
// For C#
StaticField = 79;
// For Java, C#, C++ etc.
StaticMethod = 80;
// For C#, TypeScript etc.
StaticProperty = 81;
// For C, C++
StaticVariable = 82;
String = 48;
Struct = 49;
// For Swift
Subscript = 47;
// For Lean
Tactic = 50;
// For Lean
Theorem = 51;
// Method receiver for languages
// 'this' in JavaScript, C++, Java etc.
ThisParameter = 52;
// Analogous to 'Protocol' and 'TypeClass', for Rust, Scala etc.
Trait = 53;
// Analogous to 'AbstractMethod', for Rust, Scala etc.
TraitMethod = 70;
// Data type definition for languages like OCaml which use `type`
// rather than separate keywords like `struct` and `enum`.
Type = 54;
TypeAlias = 55;
// Analogous to 'Trait' and 'Protocol', for Haskell, Purescript etc.
TypeClass = 56;
// Analogous to 'AbstractMethod', for Haskell, Purescript etc.
TypeClassMethod = 71;
// For Haskell
TypeFamily = 57;
TypeParameter = 58;
// For C, C++, Capn Proto
Union = 59;
Value = 60;
Variable = 61;
// Next = 87;
// Feel free to open a PR proposing new language-specific kinds.
}
// (optional) The name of this symbol as it should be displayed to the user.
// For example, the symbol "com/example/MyClass#myMethod(+1)." should have the
// display name "myMethod". The `symbol` field is not a reliable source of
// the display name for several reasons:
//
// - Local symbols don't encode the name.
// - Some languages have case-insensitive names, so the symbol is all-lowercase.
// - The symbol may encode names with special characters that should not be
// displayed to the user.
string display_name = 6;
// (optional) The signature of this symbol as it's displayed in API
// documentation or in hover tooltips. For example, a Java method that adds
// two numbers this would have `Document.language = "java"` and `Document.text
// = "void add(int a, int b)". The `language` and `text` fields are required
// while other fields such as `Documentation.occurrences` can be optionally
// included to support hyperlinking referenced symbols in the signature.
Document signature_documentation = 7;
// (optional) The enclosing symbol if this is a local symbol. For non-local
// symbols, the enclosing symbol should be parsed from the `symbol` field
// using the `Descriptor` grammar.
//
// The primary use-case for this field is to allow local symbol to be displayed
// in a symbol hierarchy for API documentation. It's OK to leave this field
// empty for local variables since local variables usually don't belong in API
// documentation. However, in the situation that you wish to include a local
// symbol in the hierarchy, then you can use `enclosing_symbol` to locate the
// "parent" or "owner" of this local symbol. For example, a Java indexer may
// choose to use local symbols for private class fields while providing an
// `enclosing_symbol` to reference the enclosing class to allow the field to
// be part of the class documentation hierarchy. From the perspective of an
// author of an indexer, the decision to use a local symbol or global symbol
// should exclusively be determined whether the local symbol is accessible
// outside the document, not by the capability to find the enclosing
// symbol.
string enclosing_symbol = 8;
}
message Relationship {
string symbol = 1;
// When resolving "Find references", this field documents what other symbols
// should be included together with this symbol. For example, consider the
// following TypeScript code that defines two symbols `Animal#sound()` and
// `Dog#sound()`:
// ```ts
// interface Animal {
// ^^^^^^ definition Animal#
// sound(): string
// ^^^^^ definition Animal#sound()
// }
// class Dog implements Animal {
// ^^^ definition Dog#, relationships = [{symbol: "Animal#", is_implementation: true}]
// public sound(): string { return "woof" }
// ^^^^^ definition Dog#sound(), references_symbols = Animal#sound(), relationships = [{symbol: "Animal#sound()", is_implementation:true, is_reference: true}]
// }
// const animal: Animal = new Dog()
// ^^^^^^ reference Animal#
// console.log(animal.sound())
// ^^^^^ reference Animal#sound()
// ```
// Doing "Find references" on the symbol `Animal#sound()` should return
// references to the `Dog#sound()` method as well. Vice-versa, doing "Find
// references" on the `Dog#sound()` method should include references to the
// `Animal#sound()` method as well.
bool is_reference = 2;
// Similar to `is_reference` but for "Find implementations".
// It's common for `is_implementation` and `is_reference` to both be true but
// it's not always the case.
// In the TypeScript example above, observe that `Dog#` has an
// `is_implementation` relationship with `"Animal#"` but not `is_reference`.
// This is because "Find references" on the "Animal#" symbol should not return
// "Dog#". We only want "Dog#" to return as a result for "Find
// implementations" on the "Animal#" symbol.
bool is_implementation = 3;
// Similar to `references_symbols` but for "Go to type definition".
bool is_type_definition = 4;
// Allows overriding the behavior of "Go to definition" and "Find references"
// for symbols which do not have a definition of their own or could
// potentially have multiple definitions.
//
// For example, in a language with single inheritance and no field overriding,
// inherited fields can reuse the same symbol as the ancestor which declares
// the field. In such a situation, is_definition is not needed.
//
// On the other hand, in languages with single inheritance and some form
// of mixins, you can use is_definition to relate the symbol to the
// matching symbol in ancestor classes, and is_reference to relate the
// symbol to the matching symbol in mixins.
//
// NOTE: At the moment, due to limitations of the SCIP to LSIF conversion,
// only global symbols in an index are allowed to use is_definition.
// The relationship may not get recorded if either symbol is local.
bool is_definition = 5;
// Update registerInverseRelationships on adding a new field here.
}
// SymbolRole declares what "role" a symbol has in an occurrence. A role is
// encoded as a bitset where each bit represents a different role. For example,
// to determine if the `Import` role is set, test whether the second bit of the
// enum value is defined. In pseudocode, this can be implemented with the
// logic: `const isImportRole = (role.value & SymbolRole.Import.value) > 0`.
enum SymbolRole {
// This case is not meant to be used; it only exists to avoid an error
// from the Protobuf code generator.
UnspecifiedSymbolRole = 0;
// Is the symbol defined here? If not, then this is a symbol reference.
Definition = 0x1;
// Is the symbol imported here?
Import = 0x2;
// Is the symbol written here?
WriteAccess = 0x4;
// Is the symbol read here?
ReadAccess = 0x8;
// Is the symbol in generated code?
Generated = 0x10;
// Is the symbol in test code?
Test = 0x20;
// Is this a signature for a symbol that is defined elsewhere?
//
// Applies to forward declarations for languages like C, C++
// and Objective-C, as well as `val` declarations in interface
// files in languages like SML and OCaml.
ForwardDefinition = 0x40;
}
enum SyntaxKind {
option allow_alias = true;
UnspecifiedSyntaxKind = 0;
// Comment, including comment markers and text
Comment = 1;
// `;` `.` `,`
PunctuationDelimiter = 2;
// (), {}, [] when used syntactically
PunctuationBracket = 3;
// `if`, `else`, `return`, `class`, etc.
Keyword = 4;
IdentifierKeyword = 4 [deprecated=true];
// `+`, `*`, etc.
IdentifierOperator = 5;
// non-specific catch-all for any identifier not better described elsewhere
Identifier = 6;
// Identifiers builtin to the language: `min`, `print` in Python.
IdentifierBuiltin = 7;
// Identifiers representing `null`-like values: `None` in Python, `nil` in Go.
IdentifierNull = 8;
// `xyz` in `const xyz = "hello"`
IdentifierConstant = 9;
// `var X = "hello"` in Go
IdentifierMutableGlobal = 10;
// Parameter definition and references
IdentifierParameter = 11;
// Identifiers for variable definitions and references within a local scope
IdentifierLocal = 12;
// Identifiers that shadow other identifiers in an outer scope
IdentifierShadowed = 13;
// Identifier representing a unit of code abstraction and/or namespacing.
//
// NOTE: This corresponds to a package in Go and JVM languages,
// and a module in languages like Python and JavaScript.
IdentifierNamespace = 14;
IdentifierModule = 14 [deprecated=true];
// Function references, including calls
IdentifierFunction = 15;
// Function definition only
IdentifierFunctionDefinition = 16;
// Macro references, including invocations
IdentifierMacro = 17;
// Macro definition only
IdentifierMacroDefinition = 18;
// non-builtin types
IdentifierType = 19;
// builtin types only, such as `str` for Python or `int` in Go
IdentifierBuiltinType = 20;
// Python decorators, c-like __attribute__
IdentifierAttribute = 21;
// `\b`
RegexEscape = 22;
// `*`, `+`
RegexRepeated = 23;
// `.`
RegexWildcard = 24;
// `(`, `)`, `[`, `]`
RegexDelimiter = 25;
// `|`, `-`
RegexJoin = 26;
// Literal strings: "Hello, world!"
StringLiteral = 27;
// non-regex escapes: "\t", "\n"
StringLiteralEscape = 28;
// datetimes within strings, special words within a string, `{}` in format strings
StringLiteralSpecial = 29;
// "key" in { "key": "value" }, useful for example in JSON
StringLiteralKey = 30;
// 'c' or similar, in languages that differentiate strings and characters
CharacterLiteral = 31;
// Literal numbers, both floats and integers
NumericLiteral = 32;
// `true`, `false`
BooleanLiteral = 33;
// Used for XML-like tags
Tag = 34;
// Attribute name in XML-like tags
TagAttribute = 35;
// Delimiters for XML-like tags
TagDelimiter = 36;
}
// Occurrence associates a source position with a symbol and/or highlighting
// information.
//
// If possible, indexers should try to bundle logically related information
// across occurrences into a single occurrence to reduce payload sizes.
message Occurrence {
// Half-open [start, end) range of this occurrence. Must be exactly three or four
// elements:
//
// - Four elements: `[startLine, startCharacter, endLine, endCharacter]`
// - Three elements: `[startLine, startCharacter, endCharacter]`. The end line
// is inferred to have the same value as the start line.
//
// It is allowed for the range to be empty (i.e. start==end).
//
// Line numbers and characters are always 0-based. Make sure to increment the
// line/character values before displaying them in an editor-like UI because
// editors conventionally use 1-based numbers.
//
// The 'character' value is interpreted based on the PositionEncoding for
// the Document.
//
// Historical note: the original draft of this schema had a `Range` message
// type with `start` and `end` fields of type `Position`, mirroring LSP.
// Benchmarks revealed that this encoding was inefficient and that we could
// reduce the total payload size of an index by 50% by using `repeated int32`
// instead. The `repeated int32` encoding is admittedly more embarrassing to
// work with in some programming languages but we hope the performance
// improvements make up for it.
repeated int32 range = 1;
// (optional) The symbol that appears at this position. See
// `SymbolInformation.symbol` for how to format symbols as strings.
string symbol = 2;
// (optional) Bitset containing `SymbolRole`s in this occurrence.
// See `SymbolRole`'s documentation for how to read and write this field.
int32 symbol_roles = 3;
// (optional) CommonMark-formatted documentation for this specific range. If
// empty, the `Symbol.documentation` field is used instead. One example
// where this field might be useful is when the symbol represents a generic
// function (with abstract type parameters such as `List<T>`) and at this
// occurrence we know the exact values (such as `List<String>`).
//
// This field can also be used for dynamically or gradually typed languages,
// which commonly allow for type-changing assignment.
repeated string override_documentation = 4;
// (optional) What syntax highlighting class should be used for this range?
SyntaxKind syntax_kind = 5;
// (optional) Diagnostics that have been reported for this specific range.
repeated Diagnostic diagnostics = 6;
// (optional) Using the same encoding as the sibling `range` field, half-open
// source range of the nearest non-trivial enclosing AST node. This range must
// enclose the `range` field. Example applications that make use of the
// enclosing_range field:
//
// - Call hierarchies: to determine what symbols are references from the body
// of a function
// - Symbol outline: to display breadcrumbs from the cursor position to the
// root of the file
// - Expand selection: to select the nearest enclosing AST node.
// - Highlight range: to indicate the AST expression that is associated with a
// hover popover
//
// For definition occurrences, the enclosing range should indicate the
// start/end bounds of the entire definition AST node, including
// documentation.
// ```
// const n = 3
// ^ range
// ^^^^^^^^^^^ enclosing_range
//
// /** Parses the string into something */
// ^ enclosing_range start --------------------------------------|
// function parse(input string): string { |
// ^^^^^ range |
// return input.slice(n) |
// } |
// ^ enclosing_range end <---------------------------------------|
// ```
//
// Any attributes/decorators/attached macros should also be part of the
// enclosing range.
//
// ```python
// @cache
// ^ enclosing_range start---------------------|
// def factorial(n): |
// return n * factorial(n-1) if n else 1 |
// < enclosing_range end-----------------------|
//
// ```
//
// For reference occurrences, the enclosing range should indicate the start/end
// bounds of the parent expression.
// ```
// const a = a.b
// ^ range
// ^^^ enclosing_range
// const b = a.b(41).f(42).g(43)
// ^ range
// ^^^^^^^^^^^^^ enclosing_range
// ```
repeated int32 enclosing_range = 7;
}
// Represents a diagnostic, such as a compiler error or warning, which should be
// reported for a document.
message Diagnostic {
// Should this diagnostic be reported as an error, warning, info, or hint?
Severity severity = 1;
// (optional) Code of this diagnostic, which might appear in the user interface.
string code = 2;
// Message of this diagnostic.
string message = 3;
// (optional) Human-readable string describing the source of this diagnostic, e.g.
// 'typescript' or 'super lint'.
string source = 4;
repeated DiagnosticTag tags = 5;
}
enum Severity {
UnspecifiedSeverity = 0;
Error = 1;
Warning = 2;
Information = 3;
Hint = 4;
}
enum DiagnosticTag {
UnspecifiedDiagnosticTag = 0;
Unnecessary = 1;
Deprecated = 2;
}
// Language standardises names of common programming languages that can be used
// for the `Document.language` field. The primary purpose of this enum is to
// prevent a situation where we have a single programming language ends up with
// multiple string representations. For example, the C++ language uses the name
// "CPP" in this enum and other names such as "cpp" are incompatible.
// Feel free to send a pull-request to add missing programming languages.
enum Language {
UnspecifiedLanguage = 0;
ABAP = 60;
Apex = 96;
APL = 49;
Ada = 39;
Agda = 45;
AsciiDoc = 86;
Assembly = 58;
Awk = 66;
Bat = 68;
BibTeX = 81;
C = 34;
COBOL = 59;
CPP = 35; // C++ (the name "CPP" was chosen for consistency with LSP)
CSS = 26;
CSharp = 1;
Clojure = 8;
Coffeescript = 21;
CommonLisp = 9;
Coq = 47;
CUDA = 97;
Dart = 3;
Delphi = 57;
Diff = 88;
Dockerfile = 80;
Dyalog = 50;
Elixir = 17;
Erlang = 18;
FSharp = 42;
Fish = 65;
Flow = 24;
Fortran = 56;
Git_Commit = 91;
Git_Config = 89;
Git_Rebase = 92;
Go = 33;
GraphQL = 98;
Groovy = 7;
HTML = 30;
Hack = 20;
Handlebars = 90;
Haskell = 44;
Idris = 46;
Ini = 72;
J = 51;
JSON = 75;
Java = 6;
JavaScript = 22;
JavaScriptReact = 93;
Jsonnet = 76;
Julia = 55;
Justfile = 109;
Kotlin = 4;
LaTeX = 83;
Lean = 48;
Less = 27;
Lua = 12;
Luau = 108;
Makefile = 79;
Markdown = 84;
Matlab = 52;
Nickel = 110; // https://nickel-lang.org/
Nix = 77;
OCaml = 41;
Objective_C = 36;
Objective_CPP = 37;
Pascal = 99;
PHP = 19;
PLSQL = 70;
Perl = 13;
PowerShell = 67;
Prolog = 71;
Protobuf = 100;
Python = 15;
R = 54;
Racket = 11;
Raku = 14;
Razor = 62;
Repro = 102; // Internal language for testing SCIP
ReST = 85;
Ruby = 16;
Rust = 40;
SAS = 61;
SCSS = 29;
SML = 43;
SQL = 69;
Sass = 28;
Scala = 5;
Scheme = 10;
ShellScript = 64; // Bash
Skylark = 78;
Slang = 107;
Solidity = 95;
Svelte = 106;
Swift = 2;
Tcl = 101;
TOML = 73;
TeX = 82;
Thrift = 103;
TypeScript = 23;
TypeScriptReact = 94;
Verilog = 104;
VHDL = 105;
VisualBasic = 63;
Vue = 25;
Wolfram = 53;
XML = 31;
XSL = 32;
YAML = 74;
Zig = 38;
// NextLanguage = 111;
// Steps add a new language:
// 1. Copy-paste the "NextLanguage = N" line above
// 2. Increment "NextLanguage = N" to "NextLanguage = N+1"
// 3. Replace "NextLanguage = N" with the name of the new language.
// 4. Move the new language to the correct line above using alphabetical order
// 5. (optional) Add a brief comment behind the language if the name is not self-explanatory
}
+388
View File
@@ -0,0 +1,388 @@
(ns scip-clojure.core
(:require [clojure.edn :as edn]
[clojure.java.io :as io]
[clojure.java.shell :refer [sh]]
[clojure.string :as str]
[clojure.tools.cli :refer [parse-opts]]
[clojure.repl :as repl])
(:import [scip Scip$Index Scip$Document Scip$Occurrence Scip$SymbolInformation
Scip$Metadata Scip$Metadata$Builder Scip$ToolInfo Scip$SymbolRole
Scip$PositionEncoding Scip$ProtocolVersion]
[java.io FileOutputStream StringWriter])
(:gen-class))
(def cli-options
[["-p" "--project-root PATH" "Path to the Clojure project root"
:default "."]
["-o" "--output FILE" "Output SCIP index file"
:default "index.scip"]
["-h" "--help" "Show help"]])
(defn escape-identifier
"Escape an identifier for SCIP symbol format.
Identifiers containing characters other than [_+-$a-zA-Z0-9] must be
wrapped in backticks. Backticks within the identifier are doubled."
[s]
(let [s (str s)]
(if (re-matches #"[_+\-$a-zA-Z0-9]+" s)
s
(str "`" (str/replace s "`" "``") "`"))))
(defn make-symbol
"Create a SCIP symbol identifier from namespace and name.
Format: scip-clojure clojure clojure . namespace/name.
No spaces around descriptor suffixes (/ and .)"
[ns-sym var-name]
(format "scip-clojure clojure clojure . %s/%s."
(escape-identifier ns-sym)
(escape-identifier var-name)))
(defn make-ns-symbol
"Create a SCIP symbol identifier for a namespace."
[ns-sym]
(format "scip-clojure clojure clojure . %s/"
(escape-identifier ns-sym)))
(defn make-alias-symbol
"Create a SCIP symbol identifier for a namespace alias.
The alias is scoped to the namespace where it's defined.
Format: scip-clojure clojure clojure . from-ns/alias."
[from-ns alias-name]
(format "scip-clojure clojure clojure . %s/%s."
(escape-identifier from-ns)
(escape-identifier alias-name)))
(defn position->range
"Convert row/col positions to SCIP range format.
SCIP uses 0-indexed [startLine, startChar, endLine, endChar].
For single-line occurrences, uses compact [startLine, startChar, endChar].
Uses name-row/name-col when available for precise symbol positioning."
[{:keys [row col name-row name-col name-end-row name-end-col]}]
(let [;; Prefer name-row/name-col for precise symbol location
start-line (int (dec (or name-row row 1)))
start-char (int (dec (or name-col col 1)))
end-line (int (dec (or name-end-row name-row row 1)))
end-char (int (dec (or name-end-col (+ (or name-col col 1) 1))))]
(if (= start-line end-line)
;; Compact format for single-line
[(int start-line) (int start-char) (int end-char)]
;; Full format for multi-line
[(int start-line) (int start-char) (int end-line) (int end-char)])))
(defn var-definition->occurrence
"Convert a clojure-lsp var-definition to SCIP Occurrence."
[{:keys [ns name doc] :as var-def}]
(let [builder (Scip$Occurrence/newBuilder)]
(.addAllRange builder (position->range var-def))
(.setSymbol builder (make-symbol ns name))
;; Definition role = 1
(.setSymbolRoles builder (int Scip$SymbolRole/Definition_VALUE))
(.build builder)))
(defn var-definition->symbol-info
"Convert a clojure-lsp var-definition to SCIP SymbolInformation."
[{:keys [ns name doc defined-by fixed-arities]}]
(let [builder (Scip$SymbolInformation/newBuilder)]
(.setSymbol builder (make-symbol ns name))
(when doc
(.addDocumentation builder doc))
(when defined-by
(.addDocumentation builder (str "Defined by: " defined-by)))
(when (seq fixed-arities)
(.addDocumentation builder (str "Arities: " (str/join ", " fixed-arities))))
(.build builder)))
(defn var-usage->occurrence
"Convert a clojure-lsp var-usage to SCIP Occurrence (reference)."
[{:keys [to name] :as var-usage}]
(let [builder (Scip$Occurrence/newBuilder)]
(.addAllRange builder (position->range var-usage))
(.setSymbol builder (make-symbol to name))
;; No role bits = reference
(.build builder)))
(defn ns-definition->occurrence
"Convert a namespace definition to SCIP Occurrence."
[{:keys [name row col name-end-row name-end-col] :as ns-def}]
(let [builder (Scip$Occurrence/newBuilder)]
(.addAllRange builder (position->range ns-def))
(.setSymbol builder (make-ns-symbol name))
(.setSymbolRoles builder (int Scip$SymbolRole/Definition_VALUE))
(.build builder)))
(defn ns-definition->symbol-info
"Convert a namespace definition to SCIP SymbolInformation."
[{:keys [name doc]}]
(let [builder (Scip$SymbolInformation/newBuilder)]
(.setSymbol builder (make-ns-symbol name))
(when doc
(.addDocumentation builder doc))
(.build builder)))
(defn ns-usage->occurrence
"Convert a namespace usage (require/import) to SCIP Occurrence."
[{:keys [name] :as ns-usage}]
(let [builder (Scip$Occurrence/newBuilder)]
(.addAllRange builder (position->range ns-usage))
(.setSymbol builder (make-ns-symbol name))
(.build builder)))
(defn ns-alias->occurrence
"Convert a namespace alias definition (e.g., :as ansi) to SCIP Occurrence.
Creates a Definition for the alias symbol, scoped to the defining namespace."
[{:keys [from alias] :as ns-alias}]
(let [builder (Scip$Occurrence/newBuilder)]
(.addAllRange builder (position->range ns-alias))
(.setSymbol builder (make-alias-symbol from alias))
(.setSymbolRoles builder (int Scip$SymbolRole/Definition_VALUE))
(.build builder)))
(defn alias-usage->occurrence
"Convert a var-usage with an alias to SCIP Occurrence for the alias part.
E.g., for (ansi/style ...), creates an occurrence for 'ansi' referencing the alias."
[{:keys [from alias row col name-col]}]
(when alias
(let [builder (Scip$Occurrence/newBuilder)
;; alias spans from col to name-col - 2 (excluding the /)
alias-end-col (- name-col 1)
start-line (int (dec row))
start-char (int (dec col))
end-char (int (dec alias-end-col))]
(.addAllRange builder [(int start-line) (int start-char) (int end-char)])
(.setSymbol builder (make-alias-symbol from alias))
(.build builder))))
;; --- External symbol support for hover documentation ---
(def ^:private external-ns-prefixes
"Namespace prefixes considered external (not project-internal)."
#{"clojure.core" "clojure.string" "clojure.set" "clojure.walk"
"clojure.zip" "clojure.data" "clojure.edn" "clojure.java.io"
"clojure.java.shell" "clojure.pprint" "clojure.test" "clojure.repl"
"clojure.spec.alpha" "clojure.spec.gen.alpha"})
(defn- get-special-form-doc
"Get documentation for a special form from clojure.repl/special-doc-map.
This is the canonical source of truth for special form documentation."
[sym-name]
(when-let [doc-map (get @#'clojure.repl/special-doc-map (symbol sym-name))]
{:doc (:doc doc-map)
:arglists (:forms doc-map) ;; special forms use :forms instead of :arglists
:special-form true}))
(defn external-ns?
"Check if a namespace is external (standard library or dependency)."
[ns-sym internal-namespaces]
(let [ns-str (str ns-sym)]
(and (not (contains? internal-namespaces ns-sym))
(or (str/starts-with? ns-str "clojure.")
(str/starts-with? ns-str "java.")
(contains? external-ns-prefixes ns-str)))))
(defn get-var-doc
"Get documentation for a var. Returns a map with :doc, :arglists, :macro.
Handles special forms which don't have var metadata by consulting
clojure.repl/special-doc-map (the canonical source of truth)."
[ns-sym var-name]
;; First check if it's a special form (only in clojure.core)
(if-let [special-doc (and (= (str ns-sym) "clojure.core")
(get-special-form-doc var-name))]
(assoc special-doc :name var-name :ns ns-sym)
;; Otherwise try to resolve the var and get its metadata
(try
(require (symbol ns-sym))
(when-let [v (ns-resolve (symbol ns-sym) (symbol var-name))]
(let [m (meta v)]
{:doc (:doc m)
:arglists (:arglists m)
:macro (:macro m)
:name var-name
:ns ns-sym}))
(catch Exception _ nil))))
(defn format-arglists
"Format arglists for display."
[arglists]
(when (seq arglists)
(str/join "\n" (map #(str " " (pr-str %)) arglists))))
(defn external-symbol->symbol-info
"Create SCIP SymbolInformation for an external symbol with documentation."
[{:keys [ns name doc arglists macro special-form]}]
(let [builder (Scip$SymbolInformation/newBuilder)
sig (str (cond special-form "special form "
macro "macro "
:else "")
ns "/" name)]
(.setSymbol builder (make-symbol ns name))
;; Add signature as first doc line
(.addDocumentation builder (str "```clojure\n" sig "\n```"))
;; Add arglists
(when (seq arglists)
(.addDocumentation builder (str "```clojure\n" (format-arglists arglists) "\n```")))
;; Add docstring
(when doc
(.addDocumentation builder doc))
(.build builder)))
(defn collect-external-symbols
"Collect all unique external symbol references from analysis."
[analysis internal-namespaces]
(->> (vals analysis)
(mapcat :var-usages)
(filter #(external-ns? (:to %) internal-namespaces))
(map (fn [{:keys [to name]}] [to name]))
(distinct)
(keep (fn [[ns-sym var-name]]
(get-var-doc ns-sym var-name)))
(filter :doc))) ; Only include symbols with documentation
(defn uri->relative-path
"Convert file:// URI to relative path from project root."
[uri project-root]
(let [path (str/replace-first uri "file://" "")]
(str/replace-first path (str project-root "/") "")))
(defn file-analysis->document
"Convert a single file's analysis to a SCIP Document."
[uri file-analysis project-root]
(let [builder (Scip$Document/newBuilder)
relative-path (uri->relative-path uri project-root)
{:keys [var-definitions var-usages namespace-definitions namespace-usages namespace-alias]} file-analysis
;; Create occurrences for definitions
def-occurrences (map var-definition->occurrence (or var-definitions []))
;; Create symbol info for definitions
def-symbols (map var-definition->symbol-info (or var-definitions []))
;; Create occurrences for usages (references)
usage-occurrences (map var-usage->occurrence (or var-usages []))
;; Namespace definitions
ns-def-occurrences (map ns-definition->occurrence (or namespace-definitions []))
ns-def-symbols (map ns-definition->symbol-info (or namespace-definitions []))
;; Namespace usages
ns-usage-occurrences (map ns-usage->occurrence (or namespace-usages []))
;; Namespace alias definitions (e.g., :as ansi)
ns-alias-occurrences (map ns-alias->occurrence (or namespace-alias []))
;; Alias usages from var-usages (e.g., ansi in ansi/style)
alias-usage-occs (keep alias-usage->occurrence (or var-usages []))]
(.setLanguage builder "clojure")
(.setRelativePath builder relative-path)
(.setPositionEncoding builder Scip$PositionEncoding/UTF8CodeUnitOffsetFromLineStart)
;; Add all occurrences
(doseq [occ (concat def-occurrences usage-occurrences
ns-def-occurrences ns-usage-occurrences
ns-alias-occurrences alias-usage-occs)]
(.addOccurrences builder occ))
;; Add symbol information
(doseq [sym (concat def-symbols ns-def-symbols)]
(.addSymbols builder sym))
(.build builder)))
(defn dump->scip-index
"Convert clojure-lsp dump data to a SCIP Index."
[{:keys [analysis project-root] :as dump}]
(let [builder (Scip$Index/newBuilder)
metadata-builder (Scip$Metadata/newBuilder)
tool-info-builder (Scip$ToolInfo/newBuilder)
;; Collect internal namespaces from namespace-definitions
internal-namespaces (->> (vals analysis)
(mapcat :namespace-definitions)
(map :name)
(set))]
;; Set tool info
(.setName tool-info-builder "scip-clojure")
(.setVersion tool-info-builder "0.2.0")
;; Set metadata
(.setVersion metadata-builder Scip$ProtocolVersion/UnspecifiedProtocolVersion)
(.setToolInfo metadata-builder (.build tool-info-builder))
(.setProjectRoot metadata-builder (str "file://" project-root))
(.setMetadata builder (.build metadata-builder))
;; Convert each file
(doseq [[uri file-analysis] analysis]
(let [doc (file-analysis->document (str uri) file-analysis project-root)]
(.addDocuments builder doc)))
;; Add external symbols with documentation for hover
(println "Collecting external symbol documentation...")
(let [external-syms (collect-external-symbols analysis internal-namespaces)]
(println "Found" (count external-syms) "external symbols with documentation")
(doseq [sym external-syms]
(.addExternalSymbols builder (external-symbol->symbol-info sym))))
(.build builder)))
(defn run-clojure-lsp-dump
"Run clojure-lsp dump command and return parsed EDN."
[project-root]
(println "Running clojure-lsp dump...")
(let [lsp-path (or (System/getenv "CLOJURE_LSP_PATH")
(str (System/getProperty "user.home") "/.local/bin/clojure-lsp")
"clojure-lsp")
result (sh lsp-path "dump"
"--project-root" project-root
"--output" "{:format :edn}")]
(if (zero? (:exit result))
(do
(println "Parsing clojure-lsp output...")
(edn/read-string (:out result)))
(throw (ex-info "clojure-lsp dump failed"
{:exit (:exit result)
:err (:err result)})))))
(defn write-scip-index
"Write SCIP Index to a file."
[index output-path]
(with-open [out (FileOutputStream. output-path)]
(.writeTo index out))
(println "Wrote SCIP index to" output-path))
(defn -main
[& args]
(let [{:keys [options errors summary]} (parse-opts args cli-options)]
(cond
(:help options)
(do
(println "scip-clojure - Generate SCIP index from Clojure project")
(println)
(println "Usage: scip-clojure [options]")
(println)
(println "Options:")
(println summary))
errors
(do
(doseq [err errors]
(println "Error:" err))
(System/exit 1))
:else
(let [project-root (-> (:project-root options)
io/file
.getCanonicalPath)
output-file (:output options)]
(println "Generating SCIP index for" project-root)
(try
(let [dump (run-clojure-lsp-dump project-root)
index (dump->scip-index dump)]
(write-scip-index index output-file)
(println "Done!"))
(catch Exception e
(println "Error:" (.getMessage e))
(when-let [data (ex-data e)]
(when (:err data)
(println "stderr:" (:err data))))
(System/exit 1)))))))