ruby-lsp 0.16.6 → 0.17.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/README.md +2 -2
- data/VERSION +1 -1
- data/exe/ruby-lsp +21 -4
- data/exe/ruby-lsp-check +1 -3
- data/exe/ruby-lsp-doctor +1 -4
- data/lib/core_ext/uri.rb +3 -0
- data/lib/ruby_indexer/lib/ruby_indexer/{collector.rb → declaration_listener.rb} +258 -140
- data/lib/ruby_indexer/lib/ruby_indexer/entry.rb +101 -12
- data/lib/ruby_indexer/lib/ruby_indexer/index.rb +187 -12
- data/lib/ruby_indexer/ruby_indexer.rb +1 -1
- data/lib/ruby_indexer/test/classes_and_modules_test.rb +106 -10
- data/lib/ruby_indexer/test/configuration_test.rb +4 -5
- data/lib/ruby_indexer/test/constant_test.rb +11 -8
- data/lib/ruby_indexer/test/index_test.rb +528 -0
- data/lib/ruby_indexer/test/instance_variables_test.rb +131 -0
- data/lib/ruby_indexer/test/method_test.rb +93 -0
- data/lib/ruby_indexer/test/test_case.rb +3 -1
- data/lib/ruby_lsp/addon.rb +8 -8
- data/lib/ruby_lsp/document.rb +3 -3
- data/lib/ruby_lsp/internal.rb +1 -0
- data/lib/ruby_lsp/listeners/code_lens.rb +11 -0
- data/lib/ruby_lsp/listeners/completion.rb +144 -51
- data/lib/ruby_lsp/listeners/definition.rb +77 -12
- data/lib/ruby_lsp/listeners/document_highlight.rb +1 -1
- data/lib/ruby_lsp/listeners/document_link.rb +1 -1
- data/lib/ruby_lsp/listeners/hover.rb +60 -6
- data/lib/ruby_lsp/listeners/semantic_highlighting.rb +59 -3
- data/lib/ruby_lsp/listeners/signature_help.rb +4 -4
- data/lib/ruby_lsp/node_context.rb +28 -0
- data/lib/ruby_lsp/requests/code_action_resolve.rb +73 -2
- data/lib/ruby_lsp/requests/code_actions.rb +16 -15
- data/lib/ruby_lsp/requests/completion.rb +22 -13
- data/lib/ruby_lsp/requests/completion_resolve.rb +26 -10
- data/lib/ruby_lsp/requests/definition.rb +21 -5
- data/lib/ruby_lsp/requests/document_highlight.rb +2 -2
- data/lib/ruby_lsp/requests/hover.rb +5 -6
- data/lib/ruby_lsp/requests/on_type_formatting.rb +8 -4
- data/lib/ruby_lsp/requests/signature_help.rb +3 -3
- data/lib/ruby_lsp/requests/support/common.rb +20 -1
- data/lib/ruby_lsp/requests/workspace_symbol.rb +3 -1
- data/lib/ruby_lsp/server.rb +10 -4
- metadata +10 -8
@@ -36,20 +36,36 @@ module RubyLsp
|
|
36
36
|
@item = item
|
37
37
|
end
|
38
38
|
|
39
|
-
sig { override.returns(
|
39
|
+
sig { override.returns(T::Hash[Symbol, T.untyped]) }
|
40
40
|
def perform
|
41
|
+
# Based on the spec https://microsoft.github.io/language-server-protocol/specification#textDocument_completion,
|
42
|
+
# a completion resolve request must always return the original completion item without modifying ANY fields
|
43
|
+
# other than label details and documentation. If we modify anything, the completion behaviour might be broken.
|
44
|
+
#
|
45
|
+
# For example, forgetting to return the `insertText` included in the original item will make the editor use the
|
46
|
+
# `label` for the text edit instead
|
41
47
|
label = @item[:label]
|
42
48
|
entries = @index[label] || []
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
49
|
+
|
50
|
+
owner_name = @item.dig(:data, :owner_name)
|
51
|
+
|
52
|
+
if owner_name
|
53
|
+
entries = entries.select do |entry|
|
54
|
+
(entry.is_a?(RubyIndexer::Entry::Member) || entry.is_a?(RubyIndexer::Entry::InstanceVariable)) &&
|
55
|
+
entry.owner&.name == owner_name
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
@item[:labelDetails] = Interface::CompletionItemLabelDetails.new(
|
60
|
+
description: entries.take(MAX_DOCUMENTATION_ENTRIES).map(&:file_name).join(","),
|
52
61
|
)
|
62
|
+
|
63
|
+
@item[:documentation] = Interface::MarkupContent.new(
|
64
|
+
kind: "markdown",
|
65
|
+
value: markdown_from_index_entries(label, entries, MAX_DOCUMENTATION_ENTRIES),
|
66
|
+
)
|
67
|
+
|
68
|
+
@item
|
53
69
|
end
|
54
70
|
end
|
55
71
|
end
|
@@ -12,11 +12,13 @@ module RubyLsp
|
|
12
12
|
# definition of the symbol under the cursor.
|
13
13
|
#
|
14
14
|
# Currently supported targets:
|
15
|
+
#
|
15
16
|
# - Classes
|
16
17
|
# - Modules
|
17
18
|
# - Constants
|
18
19
|
# - Require paths
|
19
|
-
# - Methods invoked on self only
|
20
|
+
# - Methods invoked on self only and on receivers where the type is unknown
|
21
|
+
# - Instance variables
|
20
22
|
#
|
21
23
|
# # Example
|
22
24
|
#
|
@@ -45,11 +47,25 @@ module RubyLsp
|
|
45
47
|
)
|
46
48
|
@dispatcher = dispatcher
|
47
49
|
|
48
|
-
|
50
|
+
node_context = document.locate_node(
|
49
51
|
position,
|
50
|
-
node_types: [
|
52
|
+
node_types: [
|
53
|
+
Prism::CallNode,
|
54
|
+
Prism::ConstantReadNode,
|
55
|
+
Prism::ConstantPathNode,
|
56
|
+
Prism::BlockArgumentNode,
|
57
|
+
Prism::InstanceVariableReadNode,
|
58
|
+
Prism::InstanceVariableAndWriteNode,
|
59
|
+
Prism::InstanceVariableOperatorWriteNode,
|
60
|
+
Prism::InstanceVariableOrWriteNode,
|
61
|
+
Prism::InstanceVariableTargetNode,
|
62
|
+
Prism::InstanceVariableWriteNode,
|
63
|
+
],
|
51
64
|
)
|
52
65
|
|
66
|
+
target = node_context.node
|
67
|
+
parent = node_context.parent
|
68
|
+
|
53
69
|
if target.is_a?(Prism::ConstantReadNode) && parent.is_a?(Prism::ConstantPathNode)
|
54
70
|
# If the target is part of a constant path node, we need to find the exact portion of the constant that the
|
55
71
|
# user is requesting to go to definition for
|
@@ -70,13 +86,13 @@ module RubyLsp
|
|
70
86
|
@response_builder,
|
71
87
|
global_state,
|
72
88
|
document.uri,
|
73
|
-
|
89
|
+
node_context,
|
74
90
|
dispatcher,
|
75
91
|
typechecker_enabled,
|
76
92
|
)
|
77
93
|
|
78
94
|
Addon.addons.each do |addon|
|
79
|
-
addon.create_definition_listener(@response_builder, document.uri,
|
95
|
+
addon.create_definition_listener(@response_builder, document.uri, node_context, dispatcher)
|
80
96
|
end
|
81
97
|
end
|
82
98
|
|
@@ -36,12 +36,12 @@ module RubyLsp
|
|
36
36
|
end
|
37
37
|
def initialize(document, position, dispatcher)
|
38
38
|
super()
|
39
|
-
|
39
|
+
node_context = document.locate_node(position)
|
40
40
|
@response_builder = T.let(
|
41
41
|
ResponseBuilders::CollectionResponseBuilder[Interface::DocumentHighlight].new,
|
42
42
|
ResponseBuilders::CollectionResponseBuilder[Interface::DocumentHighlight],
|
43
43
|
)
|
44
|
-
Listeners::DocumentHighlight.new(@response_builder,
|
44
|
+
Listeners::DocumentHighlight.new(@response_builder, node_context.node, node_context.parent, dispatcher)
|
45
45
|
end
|
46
46
|
|
47
47
|
sig { override.returns(T::Array[Interface::DocumentHighlight]) }
|
@@ -41,10 +41,9 @@ module RubyLsp
|
|
41
41
|
end
|
42
42
|
def initialize(document, global_state, position, dispatcher, typechecker_enabled)
|
43
43
|
super()
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
)
|
44
|
+
node_context = document.locate_node(position, node_types: Listeners::Hover::ALLOWED_TARGETS)
|
45
|
+
target = node_context.node
|
46
|
+
parent = node_context.parent
|
48
47
|
|
49
48
|
if (Listeners::Hover::ALLOWED_TARGETS.include?(parent.class) &&
|
50
49
|
!Listeners::Hover::ALLOWED_TARGETS.include?(target.class)) ||
|
@@ -66,9 +65,9 @@ module RubyLsp
|
|
66
65
|
@target = T.let(target, T.nilable(Prism::Node))
|
67
66
|
uri = document.uri
|
68
67
|
@response_builder = T.let(ResponseBuilders::Hover.new, ResponseBuilders::Hover)
|
69
|
-
Listeners::Hover.new(@response_builder, global_state, uri,
|
68
|
+
Listeners::Hover.new(@response_builder, global_state, uri, node_context, dispatcher, typechecker_enabled)
|
70
69
|
Addon.addons.each do |addon|
|
71
|
-
addon.create_hover_listener(@response_builder,
|
70
|
+
addon.create_hover_listener(@response_builder, node_context, dispatcher)
|
72
71
|
end
|
73
72
|
|
74
73
|
@dispatcher = dispatcher
|
@@ -213,12 +213,13 @@ module RubyLsp
|
|
213
213
|
sig { void }
|
214
214
|
def auto_indent_after_end_keyword
|
215
215
|
current_line = @lines[@position[:line]]
|
216
|
-
return unless current_line
|
216
|
+
return unless current_line && current_line.strip == "end"
|
217
217
|
|
218
|
-
|
218
|
+
node_context = @document.locate_node({
|
219
219
|
line: @position[:line],
|
220
220
|
character: @position[:character] - 1,
|
221
221
|
})
|
222
|
+
target = node_context.node
|
222
223
|
|
223
224
|
statements = case target
|
224
225
|
when Prism::IfNode, Prism::UnlessNode, Prism::ForNode, Prism::WhileNode, Prism::UntilNode
|
@@ -226,11 +227,14 @@ module RubyLsp
|
|
226
227
|
end
|
227
228
|
return unless statements
|
228
229
|
|
230
|
+
current_indentation = find_indentation(current_line)
|
229
231
|
statements.body.each do |node|
|
230
232
|
loc = node.location
|
231
|
-
next unless loc.start_column ==
|
233
|
+
next unless loc.start_column == current_indentation
|
232
234
|
|
233
|
-
|
235
|
+
(loc.start_line..loc.end_line).each do |line|
|
236
|
+
add_edit_with_text(" ", { line: line - 1, character: 0 })
|
237
|
+
end
|
234
238
|
end
|
235
239
|
|
236
240
|
move_cursor_to(@position[:line], @position[:character])
|
@@ -51,17 +51,17 @@ module RubyLsp
|
|
51
51
|
end
|
52
52
|
def initialize(document, global_state, position, context, dispatcher, typechecker_enabled) # rubocop:disable Metrics/ParameterLists
|
53
53
|
super()
|
54
|
-
|
54
|
+
node_context = document.locate_node(
|
55
55
|
{ line: position[:line], character: position[:character] },
|
56
56
|
node_types: [Prism::CallNode],
|
57
57
|
)
|
58
58
|
|
59
|
-
target = adjust_for_nested_target(
|
59
|
+
target = adjust_for_nested_target(node_context.node, node_context.parent, position)
|
60
60
|
|
61
61
|
@target = T.let(target, T.nilable(Prism::Node))
|
62
62
|
@dispatcher = dispatcher
|
63
63
|
@response_builder = T.let(ResponseBuilders::SignatureHelp.new, ResponseBuilders::SignatureHelp)
|
64
|
-
Listeners::SignatureHelp.new(@response_builder, global_state,
|
64
|
+
Listeners::SignatureHelp.new(@response_builder, global_state, node_context, dispatcher, typechecker_enabled)
|
65
65
|
end
|
66
66
|
|
67
67
|
sig { override.returns(T.nilable(Interface::SignatureHelp)) }
|
@@ -155,7 +155,8 @@ module RubyLsp
|
|
155
155
|
end
|
156
156
|
def constant_name(node)
|
157
157
|
node.full_name
|
158
|
-
rescue Prism::ConstantPathNode::DynamicPartsInConstantPathError
|
158
|
+
rescue Prism::ConstantPathNode::DynamicPartsInConstantPathError,
|
159
|
+
Prism::ConstantPathNode::MissingNodesInConstantPathError
|
159
160
|
nil
|
160
161
|
end
|
161
162
|
|
@@ -167,6 +168,24 @@ module RubyLsp
|
|
167
168
|
constant_name(path)
|
168
169
|
end
|
169
170
|
end
|
171
|
+
|
172
|
+
# Iterates over each part of a constant path, so that we can easily push response items for each section of the
|
173
|
+
# name. For example, for `Foo::Bar::Baz`, this method will invoke the block with `Foo`, then `Bar` and finally
|
174
|
+
# `Baz`.
|
175
|
+
sig do
|
176
|
+
params(
|
177
|
+
node: Prism::Node,
|
178
|
+
block: T.proc.params(part: Prism::Node).void,
|
179
|
+
).void
|
180
|
+
end
|
181
|
+
def each_constant_path_part(node, &block)
|
182
|
+
current = T.let(node, T.nilable(Prism::Node))
|
183
|
+
|
184
|
+
while current.is_a?(Prism::ConstantPathNode)
|
185
|
+
block.call(current)
|
186
|
+
current = current.parent
|
187
|
+
end
|
188
|
+
end
|
170
189
|
end
|
171
190
|
end
|
172
191
|
end
|
@@ -39,7 +39,7 @@ module RubyLsp
|
|
39
39
|
next if @global_state.typechecker && not_in_dependencies?(file_path)
|
40
40
|
|
41
41
|
# We should never show private symbols when searching the entire workspace
|
42
|
-
next if entry.
|
42
|
+
next if entry.private?
|
43
43
|
|
44
44
|
kind = kind_for_entry(entry)
|
45
45
|
loc = entry.location
|
@@ -79,6 +79,8 @@ module RubyLsp
|
|
79
79
|
entry.name == "initialize" ? Constant::SymbolKind::CONSTRUCTOR : Constant::SymbolKind::METHOD
|
80
80
|
when RubyIndexer::Entry::Accessor
|
81
81
|
Constant::SymbolKind::PROPERTY
|
82
|
+
when RubyIndexer::Entry::InstanceVariable
|
83
|
+
Constant::SymbolKind::FIELD
|
82
84
|
end
|
83
85
|
end
|
84
86
|
end
|
data/lib/ruby_lsp/server.rb
CHANGED
@@ -104,7 +104,7 @@ module RubyLsp
|
|
104
104
|
),
|
105
105
|
)
|
106
106
|
|
107
|
-
$stderr.puts(errored_addons.map(&:
|
107
|
+
$stderr.puts(errored_addons.map(&:errors_details).join("\n\n"))
|
108
108
|
end
|
109
109
|
end
|
110
110
|
|
@@ -134,7 +134,7 @@ module RubyLsp
|
|
134
134
|
when Hash
|
135
135
|
# If the configuration is already a hash, merge it with a default value of `true`. That way clients don't have
|
136
136
|
# to opt-in to every single feature
|
137
|
-
Hash.new(true).merge!(configured_features)
|
137
|
+
Hash.new(true).merge!(configured_features.transform_keys(&:to_s))
|
138
138
|
else
|
139
139
|
# If no configuration was passed by the client, just enable every feature
|
140
140
|
Hash.new(true)
|
@@ -498,6 +498,13 @@ module RubyLsp
|
|
498
498
|
),
|
499
499
|
)
|
500
500
|
raise Requests::CodeActionResolve::CodeActionError
|
501
|
+
when Requests::CodeActionResolve::Error::UnknownCodeAction
|
502
|
+
send_message(
|
503
|
+
Notification.window_show_error(
|
504
|
+
"Unknown code action",
|
505
|
+
),
|
506
|
+
)
|
507
|
+
raise Requests::CodeActionResolve::CodeActionError
|
501
508
|
else
|
502
509
|
send_message(Result.new(id: message[:id], response: result))
|
503
510
|
end
|
@@ -619,8 +626,7 @@ module RubyLsp
|
|
619
626
|
when Constant::FileChangeType::CREATED
|
620
627
|
index.index_single(indexable)
|
621
628
|
when Constant::FileChangeType::CHANGED
|
622
|
-
index.
|
623
|
-
index.index_single(indexable)
|
629
|
+
index.handle_change(indexable)
|
624
630
|
when Constant::FileChangeType::DELETED
|
625
631
|
index.delete(indexable)
|
626
632
|
end
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: ruby-lsp
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.17.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Shopify
|
8
8
|
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date: 2024-
|
11
|
+
date: 2024-05-31 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: language_server-protocol
|
@@ -30,20 +30,20 @@ dependencies:
|
|
30
30
|
requirements:
|
31
31
|
- - ">="
|
32
32
|
- !ruby/object:Gem::Version
|
33
|
-
version: 0.
|
33
|
+
version: 0.29.0
|
34
34
|
- - "<"
|
35
35
|
- !ruby/object:Gem::Version
|
36
|
-
version: '0.
|
36
|
+
version: '0.30'
|
37
37
|
type: :runtime
|
38
38
|
prerelease: false
|
39
39
|
version_requirements: !ruby/object:Gem::Requirement
|
40
40
|
requirements:
|
41
41
|
- - ">="
|
42
42
|
- !ruby/object:Gem::Version
|
43
|
-
version: 0.
|
43
|
+
version: 0.29.0
|
44
44
|
- - "<"
|
45
45
|
- !ruby/object:Gem::Version
|
46
|
-
version: '0.
|
46
|
+
version: '0.30'
|
47
47
|
- !ruby/object:Gem::Dependency
|
48
48
|
name: sorbet-runtime
|
49
49
|
requirement: !ruby/object:Gem::Requirement
|
@@ -78,8 +78,8 @@ files:
|
|
78
78
|
- lib/rubocop/cop/ruby_lsp/use_language_server_aliases.rb
|
79
79
|
- lib/rubocop/cop/ruby_lsp/use_register_with_handler_method.rb
|
80
80
|
- lib/ruby-lsp.rb
|
81
|
-
- lib/ruby_indexer/lib/ruby_indexer/collector.rb
|
82
81
|
- lib/ruby_indexer/lib/ruby_indexer/configuration.rb
|
82
|
+
- lib/ruby_indexer/lib/ruby_indexer/declaration_listener.rb
|
83
83
|
- lib/ruby_indexer/lib/ruby_indexer/entry.rb
|
84
84
|
- lib/ruby_indexer/lib/ruby_indexer/index.rb
|
85
85
|
- lib/ruby_indexer/lib/ruby_indexer/indexable_path.rb
|
@@ -90,6 +90,7 @@ files:
|
|
90
90
|
- lib/ruby_indexer/test/configuration_test.rb
|
91
91
|
- lib/ruby_indexer/test/constant_test.rb
|
92
92
|
- lib/ruby_indexer/test/index_test.rb
|
93
|
+
- lib/ruby_indexer/test/instance_variables_test.rb
|
93
94
|
- lib/ruby_indexer/test/method_test.rb
|
94
95
|
- lib/ruby_indexer/test/prefix_tree_test.rb
|
95
96
|
- lib/ruby_indexer/test/test_case.rb
|
@@ -111,6 +112,7 @@ files:
|
|
111
112
|
- lib/ruby_lsp/listeners/semantic_highlighting.rb
|
112
113
|
- lib/ruby_lsp/listeners/signature_help.rb
|
113
114
|
- lib/ruby_lsp/load_sorbet.rb
|
115
|
+
- lib/ruby_lsp/node_context.rb
|
114
116
|
- lib/ruby_lsp/parameter_scope.rb
|
115
117
|
- lib/ruby_lsp/requests.rb
|
116
118
|
- lib/ruby_lsp/requests/code_action_resolve.rb
|
@@ -178,7 +180,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
178
180
|
- !ruby/object:Gem::Version
|
179
181
|
version: '0'
|
180
182
|
requirements: []
|
181
|
-
rubygems_version: 3.5.
|
183
|
+
rubygems_version: 3.5.10
|
182
184
|
signing_key:
|
183
185
|
specification_version: 4
|
184
186
|
summary: An opinionated language server for Ruby
|