elixir: Extract to zed-extensions/elixir repository (#26167)
This PR extracts the Elixir extension to the [zed-extensions/elixir](https://github.com/zed-extensions/elixir) repository. Release Notes: - N/A
This commit is contained in:
parent
5daadc0d30
commit
4db9ab15a7
26 changed files with 1 additions and 1231 deletions
18
Cargo.lock
generated
18
Cargo.lock
generated
|
@ -16996,13 +16996,6 @@ dependencies = [
|
||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "zed_elixir"
|
|
||||||
version = "0.1.4"
|
|
||||||
dependencies = [
|
|
||||||
"zed_extension_api 0.2.0",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "zed_emmet"
|
name = "zed_emmet"
|
||||||
version = "0.0.3"
|
version = "0.0.3"
|
||||||
|
@ -17028,17 +17021,6 @@ dependencies = [
|
||||||
"wit-bindgen",
|
"wit-bindgen",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "zed_extension_api"
|
|
||||||
version = "0.2.0"
|
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
|
||||||
checksum = "9fd16b8b30a9dc920fc1678ff852f696b5bdf5b5843bc745a128be0aac29859e"
|
|
||||||
dependencies = [
|
|
||||||
"serde",
|
|
||||||
"serde_json",
|
|
||||||
"wit-bindgen",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "zed_extension_api"
|
name = "zed_extension_api"
|
||||||
version = "0.3.0"
|
version = "0.3.0"
|
||||||
|
|
|
@ -168,7 +168,6 @@ members = [
|
||||||
# Extensions
|
# Extensions
|
||||||
#
|
#
|
||||||
|
|
||||||
"extensions/elixir",
|
|
||||||
"extensions/emmet",
|
"extensions/emmet",
|
||||||
"extensions/erlang",
|
"extensions/erlang",
|
||||||
"extensions/glsl",
|
"extensions/glsl",
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Elixir
|
# Elixir
|
||||||
|
|
||||||
Elixir support is available through the [Elixir extension](https://github.com/zed-industries/zed/tree/main/extensions/elixir).
|
Elixir support is available through the [Elixir extension](https://github.com/zed-extensions/elixir).
|
||||||
|
|
||||||
- Tree-sitter:
|
- Tree-sitter:
|
||||||
- [elixir-lang/tree-sitter-elixir](https://github.com/elixir-lang/tree-sitter-elixir)
|
- [elixir-lang/tree-sitter-elixir](https://github.com/elixir-lang/tree-sitter-elixir)
|
||||||
|
|
|
@ -1,16 +0,0 @@
|
||||||
[package]
|
|
||||||
name = "zed_elixir"
|
|
||||||
version = "0.1.4"
|
|
||||||
edition.workspace = true
|
|
||||||
publish.workspace = true
|
|
||||||
license = "Apache-2.0"
|
|
||||||
|
|
||||||
[lints]
|
|
||||||
workspace = true
|
|
||||||
|
|
||||||
[lib]
|
|
||||||
path = "src/elixir.rs"
|
|
||||||
crate-type = ["cdylib"]
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
zed_extension_api = "0.2.0"
|
|
|
@ -1 +0,0 @@
|
||||||
../../LICENSE-APACHE
|
|
|
@ -1,27 +0,0 @@
|
||||||
id = "elixir"
|
|
||||||
name = "Elixir"
|
|
||||||
description = "Elixir support."
|
|
||||||
version = "0.1.4"
|
|
||||||
schema_version = 1
|
|
||||||
authors = ["Marshall Bowers <elliott.codes@gmail.com>"]
|
|
||||||
repository = "https://github.com/zed-industries/zed"
|
|
||||||
|
|
||||||
[language_servers.elixir-ls]
|
|
||||||
name = "ElixirLS"
|
|
||||||
languages = ["Elixir", "HEEX"]
|
|
||||||
|
|
||||||
[language_servers.next-ls]
|
|
||||||
name = "Next LS"
|
|
||||||
languages = ["Elixir", "HEEX"]
|
|
||||||
|
|
||||||
[language_servers.lexical]
|
|
||||||
name = "Lexical"
|
|
||||||
languages = ["Elixir", "HEEX"]
|
|
||||||
|
|
||||||
[grammars.elixir]
|
|
||||||
repository = "https://github.com/elixir-lang/tree-sitter-elixir"
|
|
||||||
commit = "a2861e88a730287a60c11ea9299c033c7d076e30"
|
|
||||||
|
|
||||||
[grammars.heex]
|
|
||||||
repository = "https://github.com/phoenixframework/tree-sitter-heex"
|
|
||||||
commit = "2e1348c3cf2c9323e87c2744796cf3f3868aa82a"
|
|
|
@ -1,5 +0,0 @@
|
||||||
("(" @open ")" @close)
|
|
||||||
("[" @open "]" @close)
|
|
||||||
("{" @open "}" @close)
|
|
||||||
("\"" @open "\"" @close)
|
|
||||||
("do" @open "end" @close)
|
|
|
@ -1,18 +0,0 @@
|
||||||
name = "Elixir"
|
|
||||||
grammar = "elixir"
|
|
||||||
path_suffixes = ["ex", "exs"]
|
|
||||||
line_comments = ["# "]
|
|
||||||
autoclose_before = ";:.,=}])>"
|
|
||||||
brackets = [
|
|
||||||
{ start = "{", end = "}", close = true, newline = true },
|
|
||||||
{ start = "[", end = "]", close = true, newline = true },
|
|
||||||
{ start = "(", end = ")", close = true, newline = true },
|
|
||||||
{ start = "\"", end = "\"", close = true, newline = false, not_in = ["string", "comment"] },
|
|
||||||
{ start = "'", end = "'", close = true, newline = false, not_in = ["string", "comment"] },
|
|
||||||
]
|
|
||||||
tab_size = 2
|
|
||||||
scope_opt_in_language_servers = ["tailwindcss-language-server"]
|
|
||||||
|
|
||||||
[overrides.string]
|
|
||||||
word_characters = ["-"]
|
|
||||||
opt_into_language_servers = ["tailwindcss-language-server"]
|
|
|
@ -1,27 +0,0 @@
|
||||||
(
|
|
||||||
(unary_operator
|
|
||||||
operator: "@"
|
|
||||||
operand: (call
|
|
||||||
target: (identifier) @unary
|
|
||||||
(#match? @unary "^(doc)$"))
|
|
||||||
) @context
|
|
||||||
.
|
|
||||||
(call
|
|
||||||
target: (identifier) @name
|
|
||||||
(arguments
|
|
||||||
[
|
|
||||||
(identifier) @name
|
|
||||||
(call
|
|
||||||
target: (identifier) @name)
|
|
||||||
(binary_operator
|
|
||||||
left: (call
|
|
||||||
target: (identifier) @name)
|
|
||||||
operator: "when")
|
|
||||||
])
|
|
||||||
(#match? @name "^(def|defp|defdelegate|defguard|defguardp|defmacro|defmacrop|defn|defnp)$")) @item
|
|
||||||
)
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: (identifier) @name
|
|
||||||
(arguments (alias) @name)
|
|
||||||
(#match? @name "^(defmodule|defprotocol)$")) @item
|
|
|
@ -1,155 +0,0 @@
|
||||||
["when" "and" "or" "not" "in" "not in" "fn" "do" "end" "catch" "rescue" "after" "else"] @keyword
|
|
||||||
|
|
||||||
(unary_operator
|
|
||||||
operator: "&"
|
|
||||||
operand: (integer) @operator)
|
|
||||||
|
|
||||||
(operator_identifier) @operator
|
|
||||||
|
|
||||||
(unary_operator
|
|
||||||
operator: _ @operator)
|
|
||||||
|
|
||||||
(binary_operator
|
|
||||||
operator: _ @operator)
|
|
||||||
|
|
||||||
(dot
|
|
||||||
operator: _ @operator)
|
|
||||||
|
|
||||||
(stab_clause
|
|
||||||
operator: _ @operator)
|
|
||||||
|
|
||||||
[
|
|
||||||
(boolean)
|
|
||||||
(nil)
|
|
||||||
] @constant
|
|
||||||
|
|
||||||
[
|
|
||||||
(integer)
|
|
||||||
(float)
|
|
||||||
] @number
|
|
||||||
|
|
||||||
(alias) @type
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: (dot
|
|
||||||
left: (atom) @type))
|
|
||||||
|
|
||||||
(char) @constant
|
|
||||||
|
|
||||||
(escape_sequence) @string.escape
|
|
||||||
|
|
||||||
[
|
|
||||||
(atom)
|
|
||||||
(quoted_atom)
|
|
||||||
(keyword)
|
|
||||||
(quoted_keyword)
|
|
||||||
] @string.special.symbol
|
|
||||||
|
|
||||||
[
|
|
||||||
(string)
|
|
||||||
(charlist)
|
|
||||||
] @string
|
|
||||||
|
|
||||||
(sigil
|
|
||||||
(sigil_name) @__name__
|
|
||||||
quoted_start: _ @string
|
|
||||||
quoted_end: _ @string
|
|
||||||
(#match? @__name__ "^[sS]$")) @string
|
|
||||||
|
|
||||||
(sigil
|
|
||||||
(sigil_name) @__name__
|
|
||||||
quoted_start: _ @string.regex
|
|
||||||
quoted_end: _ @string.regex
|
|
||||||
(#match? @__name__ "^[rR]$")) @string.regex
|
|
||||||
|
|
||||||
(sigil
|
|
||||||
(sigil_name) @__name__
|
|
||||||
quoted_start: _ @string.special
|
|
||||||
quoted_end: _ @string.special) @string.special
|
|
||||||
|
|
||||||
(identifier) @variable
|
|
||||||
|
|
||||||
(
|
|
||||||
(identifier) @comment.unused
|
|
||||||
(#match? @comment.unused "^_")
|
|
||||||
)
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: [
|
|
||||||
(identifier) @function
|
|
||||||
(dot
|
|
||||||
right: (identifier) @function)
|
|
||||||
])
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: (identifier) @keyword
|
|
||||||
(arguments
|
|
||||||
[
|
|
||||||
(identifier) @function
|
|
||||||
(binary_operator
|
|
||||||
left: (identifier) @function
|
|
||||||
operator: "when")
|
|
||||||
(binary_operator
|
|
||||||
operator: "|>"
|
|
||||||
right: (identifier))
|
|
||||||
])
|
|
||||||
(#match? @keyword "^(def|defdelegate|defguard|defguardp|defmacro|defmacrop|defn|defnp|defp)$"))
|
|
||||||
|
|
||||||
(binary_operator
|
|
||||||
operator: "|>"
|
|
||||||
right: (identifier) @function)
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: (identifier) @keyword
|
|
||||||
(#match? @keyword "^(def|defdelegate|defexception|defguard|defguardp|defimpl|defmacro|defmacrop|defmodule|defn|defnp|defoverridable|defp|defprotocol|defstruct)$"))
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: (identifier) @keyword
|
|
||||||
(#match? @keyword "^(alias|case|cond|else|for|if|import|quote|raise|receive|require|reraise|super|throw|try|unless|unquote|unquote_splicing|use|with)$"))
|
|
||||||
|
|
||||||
(
|
|
||||||
(identifier) @constant.builtin
|
|
||||||
(#match? @constant.builtin "^(__MODULE__|__DIR__|__ENV__|__CALLER__|__STACKTRACE__)$")
|
|
||||||
)
|
|
||||||
|
|
||||||
(unary_operator
|
|
||||||
operator: "@" @comment.doc
|
|
||||||
operand: (call
|
|
||||||
target: (identifier) @__attribute__ @comment.doc
|
|
||||||
(arguments
|
|
||||||
[
|
|
||||||
(string)
|
|
||||||
(charlist)
|
|
||||||
(sigil)
|
|
||||||
(boolean)
|
|
||||||
] @comment.doc))
|
|
||||||
(#match? @__attribute__ "^(moduledoc|typedoc|doc)$"))
|
|
||||||
|
|
||||||
(comment) @comment
|
|
||||||
|
|
||||||
[
|
|
||||||
"%"
|
|
||||||
] @punctuation
|
|
||||||
|
|
||||||
[
|
|
||||||
","
|
|
||||||
";"
|
|
||||||
] @punctuation.delimiter
|
|
||||||
|
|
||||||
[
|
|
||||||
"("
|
|
||||||
")"
|
|
||||||
"["
|
|
||||||
"]"
|
|
||||||
"{"
|
|
||||||
"}"
|
|
||||||
"<<"
|
|
||||||
">>"
|
|
||||||
] @punctuation.bracket
|
|
||||||
|
|
||||||
(interpolation "#{" @punctuation.special "}" @punctuation.special) @embedded
|
|
||||||
|
|
||||||
((sigil
|
|
||||||
(sigil_name) @_sigil_name
|
|
||||||
(quoted_content) @embedded)
|
|
||||||
(#eq? @_sigil_name "H"))
|
|
|
@ -1,6 +0,0 @@
|
||||||
(call) @indent
|
|
||||||
|
|
||||||
(_ "[" "]" @end) @indent
|
|
||||||
(_ "{" "}" @end) @indent
|
|
||||||
(_ "(" ")" @end) @indent
|
|
||||||
(_ "do" "end" @end) @indent
|
|
|
@ -1,7 +0,0 @@
|
||||||
; Phoenix HTML template
|
|
||||||
|
|
||||||
((sigil
|
|
||||||
(sigil_name) @_sigil_name
|
|
||||||
(quoted_content) @injection.content)
|
|
||||||
(#eq? @_sigil_name "H")
|
|
||||||
(#set! injection.language "heex"))
|
|
|
@ -1,56 +0,0 @@
|
||||||
(call
|
|
||||||
target: (identifier) @context
|
|
||||||
(arguments (alias) @name)
|
|
||||||
(#match? @context "^(defmodule|defprotocol)$")) @item
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: (identifier) @context
|
|
||||||
(arguments (_) @name)?
|
|
||||||
(#match? @context "^(setup|setup_all)$")) @item
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: (identifier) @context
|
|
||||||
(arguments (string) @name)
|
|
||||||
(#match? @context "^(describe|test)$")) @item
|
|
||||||
|
|
||||||
(unary_operator
|
|
||||||
operator: "@" @name
|
|
||||||
operand: (call
|
|
||||||
target: (identifier) @context
|
|
||||||
(arguments
|
|
||||||
[
|
|
||||||
(binary_operator
|
|
||||||
left: (identifier) @name)
|
|
||||||
(binary_operator
|
|
||||||
left: (call
|
|
||||||
target: (identifier) @name
|
|
||||||
(arguments
|
|
||||||
"(" @context.extra
|
|
||||||
_* @context.extra
|
|
||||||
")" @context.extra)))
|
|
||||||
]
|
|
||||||
)
|
|
||||||
)
|
|
||||||
(#match? @context "^(callback|type|typep)$")) @item
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: (identifier) @context
|
|
||||||
(arguments
|
|
||||||
[
|
|
||||||
(identifier) @name
|
|
||||||
(call
|
|
||||||
target: (identifier) @name
|
|
||||||
(arguments
|
|
||||||
"(" @context.extra
|
|
||||||
_* @context.extra
|
|
||||||
")" @context.extra))
|
|
||||||
(binary_operator
|
|
||||||
left: (call
|
|
||||||
target: (identifier) @name
|
|
||||||
(arguments
|
|
||||||
"(" @context.extra
|
|
||||||
_* @context.extra
|
|
||||||
")" @context.extra))
|
|
||||||
operator: "when")
|
|
||||||
])
|
|
||||||
(#match? @context "^(def|defp|defdelegate|defguard|defguardp|defmacro|defmacrop|defn|defnp)$")) @item
|
|
|
@ -1,2 +0,0 @@
|
||||||
(comment) @comment.inclusive
|
|
||||||
[(string) (charlist)] @string
|
|
|
@ -1,20 +0,0 @@
|
||||||
; Macros `describe`, `test` and `property`.
|
|
||||||
; This matches the ExUnit test style.
|
|
||||||
(
|
|
||||||
(call
|
|
||||||
target: (identifier) @run (#any-of? @run "describe" "test" "property")
|
|
||||||
) @_elixir-test
|
|
||||||
(#set! tag elixir-test)
|
|
||||||
)
|
|
||||||
|
|
||||||
; Modules containing at least one `describe`, `test` and `property`.
|
|
||||||
; This matches the ExUnit test style.
|
|
||||||
(
|
|
||||||
(call
|
|
||||||
target: (identifier) @run (#eq? @run "defmodule")
|
|
||||||
(do_block
|
|
||||||
(call target: (identifier) @_keyword (#any-of? @_keyword "describe" "test" "property"))
|
|
||||||
)
|
|
||||||
) @_elixir-module-test
|
|
||||||
(#set! tag elixir-module-test)
|
|
||||||
)
|
|
|
@ -1,30 +0,0 @@
|
||||||
// Taken from https://gist.github.com/josevalim/2e4f60a14ccd52728e3256571259d493#gistcomment-4995881
|
|
||||||
[
|
|
||||||
{
|
|
||||||
"label": "mix test",
|
|
||||||
"command": "mix",
|
|
||||||
"args": ["test"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "mix test --failed",
|
|
||||||
"command": "mix",
|
|
||||||
"args": ["test", "--failed"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "mix test $ZED_RELATIVE_FILE",
|
|
||||||
"command": "mix",
|
|
||||||
"args": ["test", "$ZED_RELATIVE_FILE"],
|
|
||||||
"tags": ["elixir-module-test"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "mix test $ZED_RELATIVE_FILE:$ZED_ROW",
|
|
||||||
"command": "mix",
|
|
||||||
"args": ["test", "$ZED_RELATIVE_FILE:$ZED_ROW"],
|
|
||||||
"tags": ["elixir-test"]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"label": "Elixir: break line",
|
|
||||||
"command": "iex",
|
|
||||||
"args": ["-S", "mix", "test", "-b", "$ZED_RELATIVE_FILE:$ZED_ROW"]
|
|
||||||
}
|
|
||||||
]
|
|
|
@ -1,46 +0,0 @@
|
||||||
(call
|
|
||||||
target: ((identifier) @_identifier
|
|
||||||
(#any-of? @_identifier "defmodule" "defprotocol" "defimpl"))
|
|
||||||
(do_block
|
|
||||||
"do"
|
|
||||||
(_)* @class.inside
|
|
||||||
"end")) @class.around
|
|
||||||
|
|
||||||
(anonymous_function
|
|
||||||
(stab_clause
|
|
||||||
right: (body) @function.inside)) @function.around
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: ((identifier) @_identifier
|
|
||||||
(#any-of? @_identifier "def" "defmacro" "defmacrop" "defn" "defnp" "defp"))
|
|
||||||
(do_block
|
|
||||||
"do"
|
|
||||||
(_)* @function.inside
|
|
||||||
"end")) @function.around
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: ((identifier) @_identifier
|
|
||||||
(#any-of? @_identifier "def" "defmacro" "defmacrop" "defn" "defnp" "defp"))
|
|
||||||
(arguments
|
|
||||||
(_)
|
|
||||||
(keywords
|
|
||||||
(pair
|
|
||||||
value: (_) @function.inside)))) @function.around
|
|
||||||
|
|
||||||
(call
|
|
||||||
target: ((identifier) @_identifier
|
|
||||||
(#any-of? @_identifier "defdelegate" "defguard" "defguardp"))) @function.around
|
|
||||||
|
|
||||||
(comment) @comment.around
|
|
||||||
|
|
||||||
(unary_operator
|
|
||||||
operator: "@"
|
|
||||||
operand: (call
|
|
||||||
target: ((identifier) @_identifier
|
|
||||||
(#any-of? @_identifier "moduledoc" "typedoc" "shortdoc" "doc"))
|
|
||||||
(arguments
|
|
||||||
[
|
|
||||||
(keywords) @comment.inside
|
|
||||||
(string
|
|
||||||
(quoted_content) @comment.inside)
|
|
||||||
]))) @comment.around
|
|
|
@ -1,13 +0,0 @@
|
||||||
name = "HEEX"
|
|
||||||
grammar = "heex"
|
|
||||||
path_suffixes = ["heex"]
|
|
||||||
autoclose_before = ">})"
|
|
||||||
brackets = [
|
|
||||||
{ start = "<", end = ">", close = true, newline = true },
|
|
||||||
]
|
|
||||||
block_comment = ["<%!-- ", " --%>"]
|
|
||||||
scope_opt_in_language_servers = ["tailwindcss-language-server"]
|
|
||||||
|
|
||||||
[overrides.string]
|
|
||||||
word_characters = ["-"]
|
|
||||||
opt_into_language_servers = ["tailwindcss-language-server"]
|
|
|
@ -1,57 +0,0 @@
|
||||||
; HEEx delimiters
|
|
||||||
[
|
|
||||||
"/>"
|
|
||||||
"<!"
|
|
||||||
"<"
|
|
||||||
"</"
|
|
||||||
"</:"
|
|
||||||
"<:"
|
|
||||||
">"
|
|
||||||
"{"
|
|
||||||
"}"
|
|
||||||
] @punctuation.bracket
|
|
||||||
|
|
||||||
[
|
|
||||||
"<%!--"
|
|
||||||
"<%"
|
|
||||||
"<%#"
|
|
||||||
"<%%="
|
|
||||||
"<%="
|
|
||||||
"%>"
|
|
||||||
"--%>"
|
|
||||||
"-->"
|
|
||||||
"<!--"
|
|
||||||
] @keyword
|
|
||||||
|
|
||||||
; HEEx operators are highlighted as such
|
|
||||||
"=" @operator
|
|
||||||
|
|
||||||
; HEEx inherits the DOCTYPE tag from HTML
|
|
||||||
(doctype) @tag.doctype
|
|
||||||
|
|
||||||
(comment) @comment
|
|
||||||
|
|
||||||
; HEEx tags and slots are highlighted as HTML
|
|
||||||
[
|
|
||||||
(tag_name)
|
|
||||||
(slot_name)
|
|
||||||
] @tag
|
|
||||||
|
|
||||||
; HEEx attributes are highlighted as HTML attributes
|
|
||||||
(attribute_name) @attribute
|
|
||||||
|
|
||||||
; HEEx special attributes are highlighted as keywords
|
|
||||||
(special_attribute_name) @keyword
|
|
||||||
|
|
||||||
[
|
|
||||||
(attribute_value)
|
|
||||||
(quoted_attribute_value)
|
|
||||||
] @string
|
|
||||||
|
|
||||||
; HEEx components are highlighted as Elixir modules and functions
|
|
||||||
(component_name
|
|
||||||
[
|
|
||||||
(module) @module
|
|
||||||
(function) @function
|
|
||||||
"." @punctuation.delimiter
|
|
||||||
])
|
|
|
@ -1,13 +0,0 @@
|
||||||
(
|
|
||||||
(directive
|
|
||||||
[
|
|
||||||
(partial_expression_value)
|
|
||||||
(expression_value)
|
|
||||||
(ending_expression_value)
|
|
||||||
] @injection.content)
|
|
||||||
(#set! injection.language "elixir")
|
|
||||||
(#set! injection.combined)
|
|
||||||
)
|
|
||||||
|
|
||||||
((expression (expression_value) @injection.content)
|
|
||||||
(#set! injection.language "elixir"))
|
|
|
@ -1,4 +0,0 @@
|
||||||
[
|
|
||||||
(attribute_value)
|
|
||||||
(quoted_attribute_value)
|
|
||||||
] @string
|
|
|
@ -1,123 +0,0 @@
|
||||||
mod language_servers;
|
|
||||||
|
|
||||||
use zed::lsp::{Completion, Symbol};
|
|
||||||
use zed::{serde_json, CodeLabel, LanguageServerId};
|
|
||||||
use zed_extension_api::{self as zed, Result};
|
|
||||||
|
|
||||||
use crate::language_servers::{ElixirLs, Lexical, NextLs};
|
|
||||||
|
|
||||||
struct ElixirExtension {
|
|
||||||
elixir_ls: Option<ElixirLs>,
|
|
||||||
next_ls: Option<NextLs>,
|
|
||||||
lexical: Option<Lexical>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl zed::Extension for ElixirExtension {
|
|
||||||
fn new() -> Self {
|
|
||||||
Self {
|
|
||||||
elixir_ls: None,
|
|
||||||
next_ls: None,
|
|
||||||
lexical: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn language_server_command(
|
|
||||||
&mut self,
|
|
||||||
language_server_id: &LanguageServerId,
|
|
||||||
worktree: &zed::Worktree,
|
|
||||||
) -> Result<zed::Command> {
|
|
||||||
match language_server_id.as_ref() {
|
|
||||||
ElixirLs::LANGUAGE_SERVER_ID => {
|
|
||||||
let elixir_ls = self.elixir_ls.get_or_insert_with(ElixirLs::new);
|
|
||||||
|
|
||||||
Ok(zed::Command {
|
|
||||||
command: elixir_ls.language_server_binary_path(language_server_id, worktree)?,
|
|
||||||
args: vec![],
|
|
||||||
env: Default::default(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
NextLs::LANGUAGE_SERVER_ID => {
|
|
||||||
let next_ls = self.next_ls.get_or_insert_with(NextLs::new);
|
|
||||||
|
|
||||||
Ok(zed::Command {
|
|
||||||
command: next_ls.language_server_binary_path(language_server_id, worktree)?,
|
|
||||||
args: vec!["--stdio".to_string()],
|
|
||||||
env: Default::default(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
Lexical::LANGUAGE_SERVER_ID => {
|
|
||||||
let lexical = self.lexical.get_or_insert_with(Lexical::new);
|
|
||||||
let lexical_binary =
|
|
||||||
lexical.language_server_binary(language_server_id, worktree)?;
|
|
||||||
|
|
||||||
Ok(zed::Command {
|
|
||||||
command: lexical_binary.path,
|
|
||||||
args: lexical_binary.args.unwrap_or_default(),
|
|
||||||
env: Default::default(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
language_server_id => Err(format!("unknown language server: {language_server_id}")),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn label_for_completion(
|
|
||||||
&self,
|
|
||||||
language_server_id: &LanguageServerId,
|
|
||||||
completion: Completion,
|
|
||||||
) -> Option<CodeLabel> {
|
|
||||||
match language_server_id.as_ref() {
|
|
||||||
ElixirLs::LANGUAGE_SERVER_ID => {
|
|
||||||
self.elixir_ls.as_ref()?.label_for_completion(completion)
|
|
||||||
}
|
|
||||||
NextLs::LANGUAGE_SERVER_ID => self.next_ls.as_ref()?.label_for_completion(completion),
|
|
||||||
Lexical::LANGUAGE_SERVER_ID => self.lexical.as_ref()?.label_for_completion(completion),
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn label_for_symbol(
|
|
||||||
&self,
|
|
||||||
language_server_id: &LanguageServerId,
|
|
||||||
symbol: Symbol,
|
|
||||||
) -> Option<CodeLabel> {
|
|
||||||
match language_server_id.as_ref() {
|
|
||||||
ElixirLs::LANGUAGE_SERVER_ID => self.elixir_ls.as_ref()?.label_for_symbol(symbol),
|
|
||||||
NextLs::LANGUAGE_SERVER_ID => self.next_ls.as_ref()?.label_for_symbol(symbol),
|
|
||||||
Lexical::LANGUAGE_SERVER_ID => self.lexical.as_ref()?.label_for_symbol(symbol),
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn language_server_initialization_options(
|
|
||||||
&mut self,
|
|
||||||
language_server_id: &LanguageServerId,
|
|
||||||
_worktree: &zed::Worktree,
|
|
||||||
) -> Result<Option<serde_json::Value>> {
|
|
||||||
match language_server_id.as_ref() {
|
|
||||||
NextLs::LANGUAGE_SERVER_ID => Ok(Some(serde_json::json!({
|
|
||||||
"experimental": {
|
|
||||||
"completions": {
|
|
||||||
"enable": true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}))),
|
|
||||||
_ => Ok(None),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn language_server_workspace_configuration(
|
|
||||||
&mut self,
|
|
||||||
language_server_id: &LanguageServerId,
|
|
||||||
worktree: &zed::Worktree,
|
|
||||||
) -> Result<Option<serde_json::Value>> {
|
|
||||||
if language_server_id.as_ref() == ElixirLs::LANGUAGE_SERVER_ID {
|
|
||||||
if let Some(elixir_ls) = self.elixir_ls.as_mut() {
|
|
||||||
return elixir_ls.language_server_workspace_configuration(worktree);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
zed::register_extension!(ElixirExtension);
|
|
|
@ -1,7 +0,0 @@
|
||||||
mod elixir_ls;
|
|
||||||
mod lexical;
|
|
||||||
mod next_ls;
|
|
||||||
|
|
||||||
pub use elixir_ls::*;
|
|
||||||
pub use lexical::*;
|
|
||||||
pub use next_ls::*;
|
|
|
@ -1,231 +0,0 @@
|
||||||
use std::fs;
|
|
||||||
|
|
||||||
use zed::lsp::{Completion, CompletionKind, Symbol, SymbolKind};
|
|
||||||
use zed::settings::LspSettings;
|
|
||||||
use zed::{serde_json, CodeLabel, CodeLabelSpan, LanguageServerId};
|
|
||||||
use zed_extension_api::{self as zed, Result};
|
|
||||||
|
|
||||||
pub struct ElixirLs {
|
|
||||||
cached_binary_path: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ElixirLs {
|
|
||||||
pub const LANGUAGE_SERVER_ID: &'static str = "elixir-ls";
|
|
||||||
|
|
||||||
pub fn new() -> Self {
|
|
||||||
Self {
|
|
||||||
cached_binary_path: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn language_server_binary_path(
|
|
||||||
&mut self,
|
|
||||||
language_server_id: &LanguageServerId,
|
|
||||||
worktree: &zed::Worktree,
|
|
||||||
) -> Result<String> {
|
|
||||||
if let Some(path) = worktree.which("elixir-ls") {
|
|
||||||
return Ok(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(path) = &self.cached_binary_path {
|
|
||||||
if fs::metadata(path).map_or(false, |stat| stat.is_file()) {
|
|
||||||
return Ok(path.clone());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
zed::set_language_server_installation_status(
|
|
||||||
language_server_id,
|
|
||||||
&zed::LanguageServerInstallationStatus::CheckingForUpdate,
|
|
||||||
);
|
|
||||||
let release = zed::latest_github_release(
|
|
||||||
"elixir-lsp/elixir-ls",
|
|
||||||
zed::GithubReleaseOptions {
|
|
||||||
require_assets: true,
|
|
||||||
pre_release: false,
|
|
||||||
},
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let asset_name = format!("elixir-ls-{version}.zip", version = release.version,);
|
|
||||||
|
|
||||||
let asset = release
|
|
||||||
.assets
|
|
||||||
.iter()
|
|
||||||
.find(|asset| asset.name == asset_name)
|
|
||||||
.ok_or_else(|| format!("no asset found matching {:?}", asset_name))?;
|
|
||||||
|
|
||||||
let (platform, _arch) = zed::current_platform();
|
|
||||||
let version_dir = format!("elixir-ls-{}", release.version);
|
|
||||||
let extension = match platform {
|
|
||||||
zed::Os::Mac | zed::Os::Linux => "sh",
|
|
||||||
zed::Os::Windows => "bat",
|
|
||||||
};
|
|
||||||
let binary_path = format!("{version_dir}/language_server.{extension}");
|
|
||||||
|
|
||||||
if !fs::metadata(&binary_path).map_or(false, |stat| stat.is_file()) {
|
|
||||||
zed::set_language_server_installation_status(
|
|
||||||
language_server_id,
|
|
||||||
&zed::LanguageServerInstallationStatus::Downloading,
|
|
||||||
);
|
|
||||||
|
|
||||||
zed::download_file(
|
|
||||||
&asset.download_url,
|
|
||||||
&version_dir,
|
|
||||||
zed::DownloadedFileType::Zip,
|
|
||||||
)
|
|
||||||
.map_err(|e| format!("failed to download file: {e}"))?;
|
|
||||||
|
|
||||||
zed::make_file_executable(&binary_path)?;
|
|
||||||
zed::make_file_executable(&format!("{version_dir}/launch.{extension}"))?;
|
|
||||||
zed::make_file_executable(&format!("{version_dir}/debug_adapter.{extension}"))?;
|
|
||||||
|
|
||||||
let entries =
|
|
||||||
fs::read_dir(".").map_err(|e| format!("failed to list working directory {e}"))?;
|
|
||||||
for entry in entries {
|
|
||||||
let entry = entry.map_err(|e| format!("failed to load directory entry {e}"))?;
|
|
||||||
if entry.file_name().to_str() != Some(&version_dir) {
|
|
||||||
fs::remove_dir_all(entry.path()).ok();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
self.cached_binary_path = Some(binary_path.clone());
|
|
||||||
Ok(binary_path)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn language_server_workspace_configuration(
|
|
||||||
&mut self,
|
|
||||||
worktree: &zed::Worktree,
|
|
||||||
) -> Result<Option<serde_json::Value>> {
|
|
||||||
let settings = LspSettings::for_worktree("elixir-ls", worktree)
|
|
||||||
.ok()
|
|
||||||
.and_then(|lsp_settings| lsp_settings.settings.clone())
|
|
||||||
.unwrap_or_default();
|
|
||||||
|
|
||||||
Ok(Some(serde_json::json!({
|
|
||||||
"elixirLS": settings
|
|
||||||
})))
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn label_for_completion(&self, completion: Completion) -> Option<CodeLabel> {
|
|
||||||
let name = &completion.label;
|
|
||||||
let detail = completion
|
|
||||||
.detail
|
|
||||||
.filter(|detail| detail != "alias")
|
|
||||||
.map(|detail| format!(": {detail}"))
|
|
||||||
.unwrap_or("".to_string());
|
|
||||||
|
|
||||||
let detail_span = CodeLabelSpan::literal(detail, Some("comment.unused".to_string()));
|
|
||||||
|
|
||||||
match completion.kind? {
|
|
||||||
CompletionKind::Module | CompletionKind::Class | CompletionKind::Struct => {
|
|
||||||
let defmodule = "defmodule ";
|
|
||||||
let alias = completion
|
|
||||||
.label_details
|
|
||||||
.and_then(|details| details.description)
|
|
||||||
.filter(|description| description.starts_with("alias"))
|
|
||||||
.map(|description| format!(" ({description})"))
|
|
||||||
.unwrap_or("".to_string());
|
|
||||||
|
|
||||||
let code = format!("{defmodule}{name}{alias}");
|
|
||||||
let name_start = defmodule.len();
|
|
||||||
let name_end = name_start + name.len();
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
code,
|
|
||||||
spans: vec![
|
|
||||||
CodeLabelSpan::code_range(name_start..name_end),
|
|
||||||
detail_span,
|
|
||||||
CodeLabelSpan::code_range(name_end..(name_end + alias.len())),
|
|
||||||
],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
CompletionKind::Interface => Some(CodeLabel {
|
|
||||||
code: name.to_string(),
|
|
||||||
spans: vec![CodeLabelSpan::code_range(0..name.len()), detail_span],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
}),
|
|
||||||
CompletionKind::Field => Some(CodeLabel {
|
|
||||||
code: name.to_string(),
|
|
||||||
spans: vec![
|
|
||||||
CodeLabelSpan::literal(name, Some("function".to_string())),
|
|
||||||
detail_span,
|
|
||||||
],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
}),
|
|
||||||
CompletionKind::Function | CompletionKind::Constant => {
|
|
||||||
let detail = completion
|
|
||||||
.label_details
|
|
||||||
.clone()
|
|
||||||
.and_then(|details| details.detail)
|
|
||||||
.unwrap_or("".to_string());
|
|
||||||
|
|
||||||
let description = completion
|
|
||||||
.label_details
|
|
||||||
.clone()
|
|
||||||
.and_then(|details| details.description)
|
|
||||||
.map(|description| format!(" ({description})"))
|
|
||||||
.unwrap_or("".to_string());
|
|
||||||
|
|
||||||
let def = "def ";
|
|
||||||
let code = format!("{def}{name}{detail}{description}");
|
|
||||||
|
|
||||||
let name_start = def.len();
|
|
||||||
let name_end = name_start + name.len();
|
|
||||||
let detail_end = name_end + detail.len();
|
|
||||||
let description_end = detail_end + description.len();
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
code,
|
|
||||||
spans: vec![
|
|
||||||
CodeLabelSpan::code_range(name_start..name_end),
|
|
||||||
CodeLabelSpan::code_range(name_end..detail_end),
|
|
||||||
CodeLabelSpan::code_range(detail_end..description_end),
|
|
||||||
],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
CompletionKind::Operator => {
|
|
||||||
let def_a = "def a ";
|
|
||||||
let code = format!("{def_a}{name} b");
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
code,
|
|
||||||
spans: vec![CodeLabelSpan::code_range(
|
|
||||||
def_a.len()..def_a.len() + name.len(),
|
|
||||||
)],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn label_for_symbol(&self, symbol: Symbol) -> Option<CodeLabel> {
|
|
||||||
let name = &symbol.name;
|
|
||||||
|
|
||||||
let (code, filter_range, display_range) = match symbol.kind {
|
|
||||||
SymbolKind::Module | SymbolKind::Class | SymbolKind::Interface | SymbolKind::Struct => {
|
|
||||||
let defmodule = "defmodule ";
|
|
||||||
let code = format!("{defmodule}{name}");
|
|
||||||
let filter_range = 0..name.len();
|
|
||||||
let display_range = defmodule.len()..defmodule.len() + name.len();
|
|
||||||
(code, filter_range, display_range)
|
|
||||||
}
|
|
||||||
SymbolKind::Function | SymbolKind::Constant => {
|
|
||||||
let def = "def ";
|
|
||||||
let code = format!("{def}{name}");
|
|
||||||
let filter_range = 0..name.len();
|
|
||||||
let display_range = def.len()..def.len() + name.len();
|
|
||||||
(code, filter_range, display_range)
|
|
||||||
}
|
|
||||||
_ => return None,
|
|
||||||
};
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
spans: vec![CodeLabelSpan::code_range(display_range)],
|
|
||||||
filter_range: filter_range.into(),
|
|
||||||
code,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,167 +0,0 @@
|
||||||
use std::fs;
|
|
||||||
|
|
||||||
use zed::lsp::{Completion, CompletionKind, Symbol};
|
|
||||||
use zed::{CodeLabel, CodeLabelSpan, LanguageServerId};
|
|
||||||
use zed_extension_api::settings::LspSettings;
|
|
||||||
use zed_extension_api::{self as zed, Result};
|
|
||||||
|
|
||||||
pub struct LexicalBinary {
|
|
||||||
pub path: String,
|
|
||||||
pub args: Option<Vec<String>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Lexical {
|
|
||||||
cached_binary_path: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Lexical {
|
|
||||||
pub const LANGUAGE_SERVER_ID: &'static str = "lexical";
|
|
||||||
|
|
||||||
pub fn new() -> Self {
|
|
||||||
Self {
|
|
||||||
cached_binary_path: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn language_server_binary(
|
|
||||||
&mut self,
|
|
||||||
language_server_id: &LanguageServerId,
|
|
||||||
worktree: &zed::Worktree,
|
|
||||||
) -> Result<LexicalBinary> {
|
|
||||||
let binary_settings = LspSettings::for_worktree("lexical", worktree)
|
|
||||||
.ok()
|
|
||||||
.and_then(|lsp_settings| lsp_settings.binary);
|
|
||||||
let binary_args = binary_settings
|
|
||||||
.as_ref()
|
|
||||||
.and_then(|binary_settings| binary_settings.arguments.clone());
|
|
||||||
|
|
||||||
if let Some(path) = binary_settings.and_then(|binary_settings| binary_settings.path) {
|
|
||||||
return Ok(LexicalBinary {
|
|
||||||
path,
|
|
||||||
args: binary_args,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(path) = worktree.which("lexical") {
|
|
||||||
return Ok(LexicalBinary {
|
|
||||||
path,
|
|
||||||
args: binary_args,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(path) = &self.cached_binary_path {
|
|
||||||
if fs::metadata(path).map_or(false, |stat| stat.is_file()) {
|
|
||||||
return Ok(LexicalBinary {
|
|
||||||
path: path.clone(),
|
|
||||||
args: binary_args,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
zed::set_language_server_installation_status(
|
|
||||||
language_server_id,
|
|
||||||
&zed::LanguageServerInstallationStatus::CheckingForUpdate,
|
|
||||||
);
|
|
||||||
let release = zed::latest_github_release(
|
|
||||||
"lexical-lsp/lexical",
|
|
||||||
zed::GithubReleaseOptions {
|
|
||||||
require_assets: true,
|
|
||||||
pre_release: false,
|
|
||||||
},
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let asset_name = format!("lexical-{version}.zip", version = release.version);
|
|
||||||
|
|
||||||
let asset = release
|
|
||||||
.assets
|
|
||||||
.iter()
|
|
||||||
.find(|asset| asset.name == asset_name)
|
|
||||||
.ok_or_else(|| format!("no asset found matching {:?}", asset_name))?;
|
|
||||||
|
|
||||||
let version_dir = format!("lexical-{}", release.version);
|
|
||||||
let binary_path = format!("{version_dir}/lexical/bin/start_lexical.sh");
|
|
||||||
|
|
||||||
if !fs::metadata(&binary_path).map_or(false, |stat| stat.is_file()) {
|
|
||||||
zed::set_language_server_installation_status(
|
|
||||||
language_server_id,
|
|
||||||
&zed::LanguageServerInstallationStatus::Downloading,
|
|
||||||
);
|
|
||||||
|
|
||||||
zed::download_file(
|
|
||||||
&asset.download_url,
|
|
||||||
&version_dir,
|
|
||||||
zed::DownloadedFileType::Zip,
|
|
||||||
)
|
|
||||||
.map_err(|e| format!("failed to download file: {e}"))?;
|
|
||||||
|
|
||||||
zed::make_file_executable(&binary_path)?;
|
|
||||||
zed::make_file_executable(&format!("{version_dir}/lexical/bin/debug_shell.sh"))?;
|
|
||||||
zed::make_file_executable(&format!("{version_dir}/lexical/priv/port_wrapper.sh"))?;
|
|
||||||
|
|
||||||
let entries =
|
|
||||||
fs::read_dir(".").map_err(|e| format!("failed to list working directory {e}"))?;
|
|
||||||
for entry in entries {
|
|
||||||
let entry = entry.map_err(|e| format!("failed to load directory entry {e}"))?;
|
|
||||||
if entry.file_name().to_str() != Some(&version_dir) {
|
|
||||||
fs::remove_dir_all(entry.path()).ok();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
self.cached_binary_path = Some(binary_path.clone());
|
|
||||||
Ok(LexicalBinary {
|
|
||||||
path: binary_path,
|
|
||||||
args: binary_args,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn label_for_completion(&self, completion: Completion) -> Option<CodeLabel> {
|
|
||||||
match completion.kind? {
|
|
||||||
CompletionKind::Module
|
|
||||||
| CompletionKind::Class
|
|
||||||
| CompletionKind::Interface
|
|
||||||
| CompletionKind::Struct => {
|
|
||||||
let name = completion.label;
|
|
||||||
let defmodule = "defmodule ";
|
|
||||||
let code = format!("{defmodule}{name}");
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
code,
|
|
||||||
spans: vec![CodeLabelSpan::code_range(
|
|
||||||
defmodule.len()..defmodule.len() + name.len(),
|
|
||||||
)],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
CompletionKind::Function | CompletionKind::Constant => {
|
|
||||||
let name = completion.label;
|
|
||||||
let def = "def ";
|
|
||||||
let code = format!("{def}{name}");
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
code,
|
|
||||||
spans: vec![CodeLabelSpan::code_range(def.len()..def.len() + name.len())],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
CompletionKind::Operator => {
|
|
||||||
let name = completion.label;
|
|
||||||
let def_a = "def a ";
|
|
||||||
let code = format!("{def_a}{name} b");
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
code,
|
|
||||||
spans: vec![CodeLabelSpan::code_range(
|
|
||||||
def_a.len()..def_a.len() + name.len(),
|
|
||||||
)],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn label_for_symbol(&self, _symbol: Symbol) -> Option<CodeLabel> {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,180 +0,0 @@
|
||||||
use std::fs;
|
|
||||||
|
|
||||||
use zed::lsp::{Completion, CompletionKind, Symbol, SymbolKind};
|
|
||||||
use zed::{CodeLabel, CodeLabelSpan, LanguageServerId};
|
|
||||||
use zed_extension_api::{self as zed, Result};
|
|
||||||
|
|
||||||
pub struct NextLs {
|
|
||||||
cached_binary_path: Option<String>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl NextLs {
|
|
||||||
pub const LANGUAGE_SERVER_ID: &'static str = "next-ls";
|
|
||||||
|
|
||||||
pub fn new() -> Self {
|
|
||||||
Self {
|
|
||||||
cached_binary_path: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn language_server_binary_path(
|
|
||||||
&mut self,
|
|
||||||
language_server_id: &LanguageServerId,
|
|
||||||
worktree: &zed::Worktree,
|
|
||||||
) -> Result<String> {
|
|
||||||
if let Some(path) = worktree.which("nextls") {
|
|
||||||
return Ok(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(path) = &self.cached_binary_path {
|
|
||||||
if fs::metadata(path).map_or(false, |stat| stat.is_file()) {
|
|
||||||
return Ok(path.clone());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
zed::set_language_server_installation_status(
|
|
||||||
language_server_id,
|
|
||||||
&zed::LanguageServerInstallationStatus::CheckingForUpdate,
|
|
||||||
);
|
|
||||||
let release = zed::latest_github_release(
|
|
||||||
"elixir-tools/next-ls",
|
|
||||||
zed::GithubReleaseOptions {
|
|
||||||
require_assets: true,
|
|
||||||
pre_release: false,
|
|
||||||
},
|
|
||||||
)?;
|
|
||||||
|
|
||||||
let (platform, arch) = zed::current_platform();
|
|
||||||
let asset_name = format!(
|
|
||||||
"next_ls_{os}_{arch}{extension}",
|
|
||||||
os = match platform {
|
|
||||||
zed::Os::Mac => "darwin",
|
|
||||||
zed::Os::Linux => "linux",
|
|
||||||
zed::Os::Windows => "windows",
|
|
||||||
},
|
|
||||||
arch = match arch {
|
|
||||||
zed::Architecture::Aarch64 => "arm64",
|
|
||||||
zed::Architecture::X8664 => "amd64",
|
|
||||||
zed::Architecture::X86 =>
|
|
||||||
return Err(format!("unsupported architecture: {arch:?}")),
|
|
||||||
},
|
|
||||||
extension = match platform {
|
|
||||||
zed::Os::Mac | zed::Os::Linux => "",
|
|
||||||
zed::Os::Windows => ".exe",
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
let asset = release
|
|
||||||
.assets
|
|
||||||
.iter()
|
|
||||||
.find(|asset| asset.name == asset_name)
|
|
||||||
.ok_or_else(|| format!("no asset found matching {:?}", asset_name))?;
|
|
||||||
|
|
||||||
let version_dir = format!("next-ls-{}", release.version);
|
|
||||||
fs::create_dir_all(&version_dir).map_err(|e| format!("failed to create directory: {e}"))?;
|
|
||||||
|
|
||||||
let binary_path = format!("{version_dir}/next-ls");
|
|
||||||
|
|
||||||
if !fs::metadata(&binary_path).map_or(false, |stat| stat.is_file()) {
|
|
||||||
zed::set_language_server_installation_status(
|
|
||||||
language_server_id,
|
|
||||||
&zed::LanguageServerInstallationStatus::Downloading,
|
|
||||||
);
|
|
||||||
|
|
||||||
zed::download_file(
|
|
||||||
&asset.download_url,
|
|
||||||
&binary_path,
|
|
||||||
zed::DownloadedFileType::Uncompressed,
|
|
||||||
)
|
|
||||||
.map_err(|e| format!("failed to download file: {e}"))?;
|
|
||||||
|
|
||||||
zed::make_file_executable(&binary_path)?;
|
|
||||||
|
|
||||||
let entries =
|
|
||||||
fs::read_dir(".").map_err(|e| format!("failed to list working directory {e}"))?;
|
|
||||||
for entry in entries {
|
|
||||||
let entry = entry.map_err(|e| format!("failed to load directory entry {e}"))?;
|
|
||||||
if entry.file_name().to_str() != Some(&version_dir) {
|
|
||||||
fs::remove_dir_all(entry.path()).ok();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
self.cached_binary_path = Some(binary_path.clone());
|
|
||||||
Ok(binary_path)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn label_for_completion(&self, completion: Completion) -> Option<CodeLabel> {
|
|
||||||
match completion.kind? {
|
|
||||||
CompletionKind::Module
|
|
||||||
| CompletionKind::Class
|
|
||||||
| CompletionKind::Interface
|
|
||||||
| CompletionKind::Struct => {
|
|
||||||
let name = completion.label;
|
|
||||||
let defmodule = "defmodule ";
|
|
||||||
let code = format!("{defmodule}{name}");
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
code,
|
|
||||||
spans: vec![CodeLabelSpan::code_range(
|
|
||||||
defmodule.len()..defmodule.len() + name.len(),
|
|
||||||
)],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
CompletionKind::Function | CompletionKind::Constant => {
|
|
||||||
let name = completion.label;
|
|
||||||
let def = "def ";
|
|
||||||
let code = format!("{def}{name}");
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
code,
|
|
||||||
spans: vec![CodeLabelSpan::code_range(def.len()..def.len() + name.len())],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
CompletionKind::Operator => {
|
|
||||||
let name = completion.label;
|
|
||||||
let def_a = "def a ";
|
|
||||||
let code = format!("{def_a}{name} b");
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
code,
|
|
||||||
spans: vec![CodeLabelSpan::code_range(
|
|
||||||
def_a.len()..def_a.len() + name.len(),
|
|
||||||
)],
|
|
||||||
filter_range: (0..name.len()).into(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn label_for_symbol(&self, symbol: Symbol) -> Option<CodeLabel> {
|
|
||||||
let name = &symbol.name;
|
|
||||||
|
|
||||||
let (code, filter_range, display_range) = match symbol.kind {
|
|
||||||
SymbolKind::Module | SymbolKind::Class | SymbolKind::Interface | SymbolKind::Struct => {
|
|
||||||
let defmodule = "defmodule ";
|
|
||||||
let code = format!("{defmodule}{name}");
|
|
||||||
let filter_range = 0..name.len();
|
|
||||||
let display_range = defmodule.len()..defmodule.len() + name.len();
|
|
||||||
(code, filter_range, display_range)
|
|
||||||
}
|
|
||||||
SymbolKind::Function | SymbolKind::Constant => {
|
|
||||||
let def = "def ";
|
|
||||||
let code = format!("{def}{name}");
|
|
||||||
let filter_range = 0..name.len();
|
|
||||||
let display_range = def.len()..def.len() + name.len();
|
|
||||||
(code, filter_range, display_range)
|
|
||||||
}
|
|
||||||
_ => return None,
|
|
||||||
};
|
|
||||||
|
|
||||||
Some(CodeLabel {
|
|
||||||
spans: vec![CodeLabelSpan::code_range(display_range)],
|
|
||||||
filter_range: filter_range.into(),
|
|
||||||
code,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Add table
Add a link
Reference in a new issue