solargraph 0.58.1 → 0.59.0.dev.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.envrc +3 -0
- data/.github/workflows/linting.yml +4 -5
- data/.github/workflows/plugins.yml +40 -36
- data/.github/workflows/rspec.yml +45 -13
- data/.github/workflows/typecheck.yml +2 -2
- data/.rubocop_todo.yml +27 -49
- data/README.md +3 -3
- data/Rakefile +1 -0
- data/lib/solargraph/api_map/cache.rb +110 -110
- data/lib/solargraph/api_map/constants.rb +289 -279
- data/lib/solargraph/api_map/index.rb +204 -193
- data/lib/solargraph/api_map/source_to_yard.rb +109 -97
- data/lib/solargraph/api_map/store.rb +387 -384
- data/lib/solargraph/api_map.rb +1000 -945
- data/lib/solargraph/complex_type/conformance.rb +176 -0
- data/lib/solargraph/complex_type/type_methods.rb +242 -228
- data/lib/solargraph/complex_type/unique_type.rb +632 -482
- data/lib/solargraph/complex_type.rb +549 -444
- data/lib/solargraph/convention/data_definition/data_definition_node.rb +93 -91
- data/lib/solargraph/convention/data_definition.rb +108 -105
- data/lib/solargraph/convention/struct_definition/struct_assignment_node.rb +62 -61
- data/lib/solargraph/convention/struct_definition/struct_definition_node.rb +103 -102
- data/lib/solargraph/convention/struct_definition.rb +168 -164
- data/lib/solargraph/diagnostics/require_not_found.rb +54 -53
- data/lib/solargraph/diagnostics/rubocop.rb +119 -118
- data/lib/solargraph/diagnostics/rubocop_helpers.rb +70 -68
- data/lib/solargraph/diagnostics/type_check.rb +56 -55
- data/lib/solargraph/doc_map.rb +200 -439
- data/lib/solargraph/equality.rb +34 -34
- data/lib/solargraph/gem_pins.rb +97 -98
- data/lib/solargraph/language_server/host/dispatch.rb +131 -130
- data/lib/solargraph/language_server/host/message_worker.rb +113 -112
- data/lib/solargraph/language_server/host/sources.rb +100 -99
- data/lib/solargraph/language_server/host.rb +883 -878
- data/lib/solargraph/language_server/message/extended/check_gem_version.rb +109 -114
- data/lib/solargraph/language_server/message/extended/document.rb +24 -23
- data/lib/solargraph/language_server/message/text_document/completion.rb +58 -56
- data/lib/solargraph/language_server/message/text_document/definition.rb +42 -40
- data/lib/solargraph/language_server/message/text_document/document_symbol.rb +28 -26
- data/lib/solargraph/language_server/message/text_document/formatting.rb +150 -148
- data/lib/solargraph/language_server/message/text_document/hover.rb +60 -58
- data/lib/solargraph/language_server/message/text_document/signature_help.rb +25 -24
- data/lib/solargraph/language_server/message/text_document/type_definition.rb +27 -25
- data/lib/solargraph/language_server/message/workspace/workspace_symbol.rb +25 -23
- data/lib/solargraph/library.rb +729 -683
- data/lib/solargraph/location.rb +87 -82
- data/lib/solargraph/logging.rb +57 -37
- data/lib/solargraph/parser/comment_ripper.rb +76 -69
- data/lib/solargraph/parser/flow_sensitive_typing.rb +483 -255
- data/lib/solargraph/parser/node_processor/base.rb +122 -92
- data/lib/solargraph/parser/node_processor.rb +63 -62
- data/lib/solargraph/parser/parser_gem/class_methods.rb +167 -149
- data/lib/solargraph/parser/parser_gem/node_chainer.rb +191 -166
- data/lib/solargraph/parser/parser_gem/node_methods.rb +506 -486
- data/lib/solargraph/parser/parser_gem/node_processors/and_node.rb +22 -22
- data/lib/solargraph/parser/parser_gem/node_processors/args_node.rb +61 -59
- data/lib/solargraph/parser/parser_gem/node_processors/begin_node.rb +24 -15
- data/lib/solargraph/parser/parser_gem/node_processors/block_node.rb +46 -46
- data/lib/solargraph/parser/parser_gem/node_processors/def_node.rb +60 -53
- data/lib/solargraph/parser/parser_gem/node_processors/if_node.rb +53 -23
- data/lib/solargraph/parser/parser_gem/node_processors/ivasgn_node.rb +41 -40
- data/lib/solargraph/parser/parser_gem/node_processors/lvasgn_node.rb +30 -29
- data/lib/solargraph/parser/parser_gem/node_processors/masgn_node.rb +61 -59
- data/lib/solargraph/parser/parser_gem/node_processors/opasgn_node.rb +98 -98
- data/lib/solargraph/parser/parser_gem/node_processors/or_node.rb +22 -0
- data/lib/solargraph/parser/parser_gem/node_processors/orasgn_node.rb +17 -17
- data/lib/solargraph/parser/parser_gem/node_processors/resbody_node.rb +39 -38
- data/lib/solargraph/parser/parser_gem/node_processors/sclass_node.rb +53 -52
- data/lib/solargraph/parser/parser_gem/node_processors/send_node.rb +296 -291
- data/lib/solargraph/parser/parser_gem/node_processors/when_node.rb +23 -0
- data/lib/solargraph/parser/parser_gem/node_processors/while_node.rb +33 -29
- data/lib/solargraph/parser/parser_gem/node_processors.rb +74 -70
- data/lib/solargraph/parser/region.rb +75 -69
- data/lib/solargraph/parser/snippet.rb +17 -17
- data/lib/solargraph/pin/base.rb +761 -729
- data/lib/solargraph/pin/base_variable.rb +418 -126
- data/lib/solargraph/pin/block.rb +126 -104
- data/lib/solargraph/pin/breakable.rb +13 -9
- data/lib/solargraph/pin/callable.rb +278 -231
- data/lib/solargraph/pin/closure.rb +68 -72
- data/lib/solargraph/pin/common.rb +94 -79
- data/lib/solargraph/pin/compound_statement.rb +55 -0
- data/lib/solargraph/pin/conversions.rb +124 -123
- data/lib/solargraph/pin/delegated_method.rb +131 -120
- data/lib/solargraph/pin/documenting.rb +115 -114
- data/lib/solargraph/pin/instance_variable.rb +38 -34
- data/lib/solargraph/pin/keyword.rb +16 -20
- data/lib/solargraph/pin/local_variable.rb +31 -75
- data/lib/solargraph/pin/method.rb +720 -672
- data/lib/solargraph/pin/method_alias.rb +42 -34
- data/lib/solargraph/pin/namespace.rb +121 -115
- data/lib/solargraph/pin/parameter.rb +338 -275
- data/lib/solargraph/pin/proxy_type.rb +40 -39
- data/lib/solargraph/pin/reference/override.rb +47 -47
- data/lib/solargraph/pin/reference/superclass.rb +17 -15
- data/lib/solargraph/pin/reference.rb +41 -39
- data/lib/solargraph/pin/search.rb +62 -61
- data/lib/solargraph/pin/signature.rb +69 -61
- data/lib/solargraph/pin/symbol.rb +53 -53
- data/lib/solargraph/pin/until.rb +18 -18
- data/lib/solargraph/pin/while.rb +18 -18
- data/lib/solargraph/pin.rb +46 -44
- data/lib/solargraph/pin_cache.rb +665 -245
- data/lib/solargraph/position.rb +118 -119
- data/lib/solargraph/range.rb +112 -112
- data/lib/solargraph/rbs_map/conversions.rb +846 -823
- data/lib/solargraph/rbs_map/core_map.rb +65 -58
- data/lib/solargraph/rbs_map/stdlib_map.rb +72 -43
- data/lib/solargraph/rbs_map.rb +217 -163
- data/lib/solargraph/shell.rb +397 -352
- data/lib/solargraph/source/chain/call.rb +372 -337
- data/lib/solargraph/source/chain/constant.rb +28 -26
- data/lib/solargraph/source/chain/hash.rb +35 -34
- data/lib/solargraph/source/chain/if.rb +29 -28
- data/lib/solargraph/source/chain/instance_variable.rb +34 -13
- data/lib/solargraph/source/chain/literal.rb +53 -48
- data/lib/solargraph/source/chain/or.rb +31 -23
- data/lib/solargraph/source/chain.rb +294 -291
- data/lib/solargraph/source/change.rb +89 -82
- data/lib/solargraph/source/cursor.rb +172 -166
- data/lib/solargraph/source/source_chainer.rb +204 -194
- data/lib/solargraph/source/updater.rb +59 -55
- data/lib/solargraph/source.rb +524 -498
- data/lib/solargraph/source_map/clip.rb +237 -226
- data/lib/solargraph/source_map/data.rb +37 -34
- data/lib/solargraph/source_map/mapper.rb +282 -259
- data/lib/solargraph/source_map.rb +220 -212
- data/lib/solargraph/type_checker/problem.rb +34 -32
- data/lib/solargraph/type_checker/rules.rb +157 -84
- data/lib/solargraph/type_checker.rb +895 -814
- data/lib/solargraph/version.rb +1 -1
- data/lib/solargraph/workspace/config.rb +257 -255
- data/lib/solargraph/workspace/gemspecs.rb +367 -0
- data/lib/solargraph/workspace/require_paths.rb +98 -97
- data/lib/solargraph/workspace.rb +362 -220
- data/lib/solargraph/yard_map/helpers.rb +45 -44
- data/lib/solargraph/yard_map/mapper/to_method.rb +134 -130
- data/lib/solargraph/yard_map/mapper/to_namespace.rb +32 -31
- data/lib/solargraph/yard_map/mapper.rb +84 -79
- data/lib/solargraph/yardoc.rb +97 -87
- data/lib/solargraph.rb +126 -105
- data/rbs/fills/rubygems/0/dependency.rbs +193 -0
- data/rbs/fills/tuple/tuple.rbs +28 -0
- data/rbs/shims/ast/0/node.rbs +5 -0
- data/rbs/shims/diff-lcs/1.5/diff-lcs.rbs +11 -0
- data/rbs_collection.yaml +1 -1
- data/solargraph.gemspec +2 -1
- metadata +22 -17
- data/lib/solargraph/type_checker/checks.rb +0 -124
- data/lib/solargraph/type_checker/param_def.rb +0 -37
- data/lib/solargraph/yard_map/to_method.rb +0 -89
- data/sig/shims/ast/0/node.rbs +0 -5
- /data/{sig → rbs}/shims/ast/2.4/.rbs_meta.yaml +0 -0
- /data/{sig → rbs}/shims/ast/2.4/ast.rbs +0 -0
- /data/{sig → rbs}/shims/parser/3.2.0.1/builders/default.rbs +0 -0
- /data/{sig → rbs}/shims/parser/3.2.0.1/manifest.yaml +0 -0
- /data/{sig → rbs}/shims/parser/3.2.0.1/parser.rbs +0 -0
- /data/{sig → rbs}/shims/parser/3.2.0.1/polyfill.rbs +0 -0
- /data/{sig → rbs}/shims/thor/1.2.0.1/.rbs_meta.yaml +0 -0
- /data/{sig → rbs}/shims/thor/1.2.0.1/manifest.yaml +0 -0
- /data/{sig → rbs}/shims/thor/1.2.0.1/thor.rbs +0 -0
data/lib/solargraph/doc_map.rb
CHANGED
|
@@ -1,439 +1,200 @@
|
|
|
1
|
-
# frozen_string_literal: true
|
|
2
|
-
|
|
3
|
-
require 'pathname'
|
|
4
|
-
require 'benchmark'
|
|
5
|
-
require 'open3'
|
|
6
|
-
|
|
7
|
-
module Solargraph
|
|
8
|
-
# A collection of pins generated from
|
|
9
|
-
#
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
# @
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
@
|
|
55
|
-
@
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
end
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
@
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
nil
|
|
202
|
-
end
|
|
203
|
-
|
|
204
|
-
# @return [Hash{String => Array<Gem::Specification>}]
|
|
205
|
-
def required_gems_map
|
|
206
|
-
@required_gems_map ||= requires.to_h { |path| [path, resolve_path_to_gemspecs(path)] }
|
|
207
|
-
end
|
|
208
|
-
|
|
209
|
-
# @return [Hash{String => Gem::Specification}]
|
|
210
|
-
def preference_map
|
|
211
|
-
@preference_map ||= preferences.to_h { |gemspec| [gemspec.name, gemspec] }
|
|
212
|
-
end
|
|
213
|
-
|
|
214
|
-
# @param gemspec [Gem::Specification]
|
|
215
|
-
# @return [Array<Pin::Base>, nil]
|
|
216
|
-
def deserialize_yard_pin_cache gemspec
|
|
217
|
-
if yard_pins_in_memory.key?([gemspec.name, gemspec.version])
|
|
218
|
-
return yard_pins_in_memory[[gemspec.name, gemspec.version]]
|
|
219
|
-
end
|
|
220
|
-
|
|
221
|
-
cached = PinCache.deserialize_yard_gem(gemspec)
|
|
222
|
-
if cached
|
|
223
|
-
logger.info { "Loaded #{cached.length} cached YARD pins from #{gemspec.name}:#{gemspec.version}" }
|
|
224
|
-
yard_pins_in_memory[[gemspec.name, gemspec.version]] = cached
|
|
225
|
-
cached
|
|
226
|
-
else
|
|
227
|
-
logger.debug "No YARD pin cache for #{gemspec.name}:#{gemspec.version}"
|
|
228
|
-
@uncached_yard_gemspecs.push gemspec
|
|
229
|
-
nil
|
|
230
|
-
end
|
|
231
|
-
end
|
|
232
|
-
|
|
233
|
-
# @param gemspec [Gem::Specification]
|
|
234
|
-
# @return [void]
|
|
235
|
-
def deserialize_combined_pin_cache(gemspec)
|
|
236
|
-
unless combined_pins_in_memory[[gemspec.name, gemspec.version]].nil?
|
|
237
|
-
return combined_pins_in_memory[[gemspec.name, gemspec.version]]
|
|
238
|
-
end
|
|
239
|
-
|
|
240
|
-
rbs_map = RbsMap.from_gemspec(gemspec, rbs_collection_path, rbs_collection_config_path)
|
|
241
|
-
rbs_version_cache_key = rbs_map.cache_key
|
|
242
|
-
|
|
243
|
-
cached = PinCache.deserialize_combined_gem(gemspec, rbs_version_cache_key)
|
|
244
|
-
if cached
|
|
245
|
-
logger.info { "Loaded #{cached.length} cached YARD pins from #{gemspec.name}:#{gemspec.version}" }
|
|
246
|
-
combined_pins_in_memory[[gemspec.name, gemspec.version]] = cached
|
|
247
|
-
return combined_pins_in_memory[[gemspec.name, gemspec.version]]
|
|
248
|
-
end
|
|
249
|
-
|
|
250
|
-
rbs_collection_pins = deserialize_rbs_collection_cache gemspec, rbs_version_cache_key
|
|
251
|
-
|
|
252
|
-
yard_pins = deserialize_yard_pin_cache gemspec
|
|
253
|
-
|
|
254
|
-
if !rbs_collection_pins.nil? && !yard_pins.nil?
|
|
255
|
-
logger.debug { "Combining pins for #{gemspec.name}:#{gemspec.version}" }
|
|
256
|
-
combined_pins = GemPins.combine(yard_pins, rbs_collection_pins)
|
|
257
|
-
PinCache.serialize_combined_gem(gemspec, rbs_version_cache_key, combined_pins)
|
|
258
|
-
combined_pins_in_memory[[gemspec.name, gemspec.version]] = combined_pins
|
|
259
|
-
logger.info { "Generated #{combined_pins_in_memory[[gemspec.name, gemspec.version]].length} combined pins for #{gemspec.name} #{gemspec.version}" }
|
|
260
|
-
return combined_pins
|
|
261
|
-
end
|
|
262
|
-
|
|
263
|
-
if !yard_pins.nil?
|
|
264
|
-
logger.debug { "Using only YARD pins for #{gemspec.name}:#{gemspec.version}" }
|
|
265
|
-
combined_pins_in_memory[[gemspec.name, gemspec.version]] = yard_pins
|
|
266
|
-
return combined_pins_in_memory[[gemspec.name, gemspec.version]]
|
|
267
|
-
elsif !rbs_collection_pins.nil?
|
|
268
|
-
logger.debug { "Using only RBS collection pins for #{gemspec.name}:#{gemspec.version}" }
|
|
269
|
-
combined_pins_in_memory[[gemspec.name, gemspec.version]] = rbs_collection_pins
|
|
270
|
-
return combined_pins_in_memory[[gemspec.name, gemspec.version]]
|
|
271
|
-
else
|
|
272
|
-
logger.debug { "Pins not yet cached for #{gemspec.name}:#{gemspec.version}" }
|
|
273
|
-
return nil
|
|
274
|
-
end
|
|
275
|
-
end
|
|
276
|
-
|
|
277
|
-
# @param path [String] require path that might be in the RBS stdlib collection
|
|
278
|
-
# @return [void]
|
|
279
|
-
def deserialize_stdlib_rbs_map path
|
|
280
|
-
map = RbsMap::StdlibMap.load(path)
|
|
281
|
-
if map.resolved?
|
|
282
|
-
logger.debug { "Loading stdlib pins for #{path}" }
|
|
283
|
-
@pins.concat map.pins
|
|
284
|
-
logger.debug { "Loaded #{map.pins.length} stdlib pins for #{path}" }
|
|
285
|
-
map.pins
|
|
286
|
-
else
|
|
287
|
-
# @todo Temporarily ignoring unresolved `require 'set'`
|
|
288
|
-
logger.debug { "Require path #{path} could not be resolved in RBS" } unless path == 'set'
|
|
289
|
-
nil
|
|
290
|
-
end
|
|
291
|
-
end
|
|
292
|
-
|
|
293
|
-
# @param gemspec [Gem::Specification]
|
|
294
|
-
# @param rbs_version_cache_key [String]
|
|
295
|
-
# @return [Array<Pin::Base>, nil]
|
|
296
|
-
def deserialize_rbs_collection_cache gemspec, rbs_version_cache_key
|
|
297
|
-
return if rbs_collection_pins_in_memory.key?([gemspec, rbs_version_cache_key])
|
|
298
|
-
cached = PinCache.deserialize_rbs_collection_gem(gemspec, rbs_version_cache_key)
|
|
299
|
-
if cached
|
|
300
|
-
logger.info { "Loaded #{cached.length} pins from RBS collection cache for #{gemspec.name}:#{gemspec.version}" } unless cached.empty?
|
|
301
|
-
rbs_collection_pins_in_memory[[gemspec, rbs_version_cache_key]] = cached
|
|
302
|
-
cached
|
|
303
|
-
else
|
|
304
|
-
logger.debug "No RBS collection pin cache for #{gemspec.name} #{gemspec.version}"
|
|
305
|
-
@uncached_rbs_collection_gemspecs.push gemspec
|
|
306
|
-
nil
|
|
307
|
-
end
|
|
308
|
-
end
|
|
309
|
-
|
|
310
|
-
# @param path [String]
|
|
311
|
-
# @return [::Array<Gem::Specification>, nil]
|
|
312
|
-
def resolve_path_to_gemspecs path
|
|
313
|
-
return nil if path.empty?
|
|
314
|
-
return gemspecs_required_from_bundler if path == 'bundler/require'
|
|
315
|
-
|
|
316
|
-
# @type [Gem::Specification, nil]
|
|
317
|
-
gemspec = Gem::Specification.find_by_path(path)
|
|
318
|
-
if gemspec.nil?
|
|
319
|
-
gem_name_guess = path.split('/').first
|
|
320
|
-
begin
|
|
321
|
-
# this can happen when the gem is included via a local path in
|
|
322
|
-
# a Gemfile; Gem doesn't try to index the paths in that case.
|
|
323
|
-
#
|
|
324
|
-
# See if we can make a good guess:
|
|
325
|
-
potential_gemspec = Gem::Specification.find_by_name(gem_name_guess)
|
|
326
|
-
file = "lib/#{path}.rb"
|
|
327
|
-
gemspec = potential_gemspec if potential_gemspec.files.any? { |gemspec_file| file == gemspec_file }
|
|
328
|
-
rescue Gem::MissingSpecError
|
|
329
|
-
logger.debug { "Require path #{path} could not be resolved to a gem via find_by_path or guess of #{gem_name_guess}" }
|
|
330
|
-
[]
|
|
331
|
-
end
|
|
332
|
-
end
|
|
333
|
-
return nil if gemspec.nil?
|
|
334
|
-
[gemspec_or_preference(gemspec)]
|
|
335
|
-
end
|
|
336
|
-
|
|
337
|
-
# @param gemspec [Gem::Specification]
|
|
338
|
-
# @return [Gem::Specification]
|
|
339
|
-
def gemspec_or_preference gemspec
|
|
340
|
-
# :nocov: dormant feature
|
|
341
|
-
return gemspec unless preference_map.key?(gemspec.name)
|
|
342
|
-
return gemspec if gemspec.version == preference_map[gemspec.name].version
|
|
343
|
-
|
|
344
|
-
change_gemspec_version gemspec, preference_map[gemspec.name].version
|
|
345
|
-
# :nocov:
|
|
346
|
-
end
|
|
347
|
-
|
|
348
|
-
# @param gemspec [Gem::Specification]
|
|
349
|
-
# @param version [Gem::Version]
|
|
350
|
-
# @return [Gem::Specification]
|
|
351
|
-
def change_gemspec_version gemspec, version
|
|
352
|
-
Gem::Specification.find_by_name(gemspec.name, "= #{version}")
|
|
353
|
-
rescue Gem::MissingSpecError
|
|
354
|
-
Solargraph.logger.info "Gem #{gemspec.name} version #{version} not found. Using #{gemspec.version} instead"
|
|
355
|
-
gemspec
|
|
356
|
-
end
|
|
357
|
-
|
|
358
|
-
# @param gemspec [Gem::Specification]
|
|
359
|
-
# @return [Array<Gem::Specification>]
|
|
360
|
-
def fetch_dependencies gemspec
|
|
361
|
-
# @param spec [Gem::Dependency]
|
|
362
|
-
# @param deps [Set<Gem::Specification>]
|
|
363
|
-
only_runtime_dependencies(gemspec).each_with_object(Set.new) do |spec, deps|
|
|
364
|
-
Solargraph.logger.info "Adding #{spec.name} dependency for #{gemspec.name}"
|
|
365
|
-
dep = Gem.loaded_specs[spec.name]
|
|
366
|
-
# @todo is next line necessary?
|
|
367
|
-
# @sg-ignore Unresolved call to requirement on Gem::Dependency
|
|
368
|
-
dep ||= Gem::Specification.find_by_name(spec.name, spec.requirement)
|
|
369
|
-
deps.merge fetch_dependencies(dep) if deps.add?(dep)
|
|
370
|
-
rescue Gem::MissingSpecError
|
|
371
|
-
# @sg-ignore Unresolved call to requirement on Gem::Dependency
|
|
372
|
-
Solargraph.logger.warn "Gem dependency #{spec.name} #{spec.requirement} for #{gemspec.name} not found in RubyGems."
|
|
373
|
-
end.to_a
|
|
374
|
-
end
|
|
375
|
-
|
|
376
|
-
# @param gemspec [Gem::Specification]
|
|
377
|
-
# @return [Array<Gem::Dependency>]
|
|
378
|
-
def only_runtime_dependencies gemspec
|
|
379
|
-
gemspec.dependencies - gemspec.development_dependencies
|
|
380
|
-
end
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
def inspect
|
|
384
|
-
self.class.inspect
|
|
385
|
-
end
|
|
386
|
-
|
|
387
|
-
# @return [Array<Gem::Specification>, nil]
|
|
388
|
-
def gemspecs_required_from_bundler
|
|
389
|
-
# @todo Handle projects with custom Bundler/Gemfile setups
|
|
390
|
-
return unless workspace.gemfile?
|
|
391
|
-
|
|
392
|
-
if workspace.gemfile? && Bundler.definition&.lockfile&.to_s&.start_with?(workspace.directory)
|
|
393
|
-
# Find only the gems bundler is now using
|
|
394
|
-
Bundler.definition.locked_gems.specs.flat_map do |lazy_spec|
|
|
395
|
-
logger.info "Handling #{lazy_spec.name}:#{lazy_spec.version}"
|
|
396
|
-
[Gem::Specification.find_by_name(lazy_spec.name, lazy_spec.version)]
|
|
397
|
-
rescue Gem::MissingSpecError => e
|
|
398
|
-
logger.info("Could not find #{lazy_spec.name}:#{lazy_spec.version} with find_by_name, falling back to guess")
|
|
399
|
-
# can happen in local filesystem references
|
|
400
|
-
specs = resolve_path_to_gemspecs lazy_spec.name
|
|
401
|
-
logger.warn "Gem #{lazy_spec.name} #{lazy_spec.version} from bundle not found: #{e}" if specs.nil?
|
|
402
|
-
next specs
|
|
403
|
-
end.compact
|
|
404
|
-
else
|
|
405
|
-
logger.info 'Fetching gemspecs required from Bundler (bundler/require)'
|
|
406
|
-
gemspecs_required_from_external_bundle
|
|
407
|
-
end
|
|
408
|
-
end
|
|
409
|
-
|
|
410
|
-
# @return [Array<Gem::Specification>, nil]
|
|
411
|
-
def gemspecs_required_from_external_bundle
|
|
412
|
-
logger.info 'Fetching gemspecs required from external bundle'
|
|
413
|
-
return [] unless workspace&.directory
|
|
414
|
-
|
|
415
|
-
Solargraph.with_clean_env do
|
|
416
|
-
cmd = [
|
|
417
|
-
'ruby', '-e',
|
|
418
|
-
"require 'bundler'; require 'json'; Dir.chdir('#{workspace&.directory}') { puts Bundler.definition.locked_gems.specs.map { |spec| [spec.name, spec.version] }.to_h.to_json }"
|
|
419
|
-
]
|
|
420
|
-
o, e, s = Open3.capture3(*cmd)
|
|
421
|
-
if s.success?
|
|
422
|
-
Solargraph.logger.debug "External bundle: #{o}"
|
|
423
|
-
hash = o && !o.empty? ? JSON.parse(o.split("\n").last) : {}
|
|
424
|
-
hash.flat_map do |name, version|
|
|
425
|
-
Gem::Specification.find_by_name(name, version)
|
|
426
|
-
rescue Gem::MissingSpecError => e
|
|
427
|
-
logger.info("Could not find #{name}:#{version} with find_by_name, falling back to guess")
|
|
428
|
-
# can happen in local filesystem references
|
|
429
|
-
specs = resolve_path_to_gemspecs name
|
|
430
|
-
logger.warn "Gem #{name} #{version} from bundle not found: #{e}" if specs.nil?
|
|
431
|
-
next specs
|
|
432
|
-
end.compact
|
|
433
|
-
else
|
|
434
|
-
Solargraph.logger.warn "Failed to load gems from bundle at #{workspace&.directory}: #{e}"
|
|
435
|
-
end
|
|
436
|
-
end
|
|
437
|
-
end
|
|
438
|
-
end
|
|
439
|
-
end
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
require 'pathname'
|
|
4
|
+
require 'benchmark'
|
|
5
|
+
require 'open3'
|
|
6
|
+
|
|
7
|
+
module Solargraph
|
|
8
|
+
# A collection of pins generated from specific 'require' statements
|
|
9
|
+
# in code. Multiple can be created per workspace, to represent the
|
|
10
|
+
# pins available in different files based on their particular
|
|
11
|
+
# 'require' lines.
|
|
12
|
+
#
|
|
13
|
+
class DocMap
|
|
14
|
+
include Logging
|
|
15
|
+
|
|
16
|
+
# @return [Workspace]
|
|
17
|
+
attr_reader :workspace
|
|
18
|
+
|
|
19
|
+
# @return [Environ]
|
|
20
|
+
attr_reader :environ
|
|
21
|
+
|
|
22
|
+
# @param requires [Array<String>]
|
|
23
|
+
# @param workspace [Workspace, nil]
|
|
24
|
+
# @param out [IO, nil] output stream for logging
|
|
25
|
+
def initialize requires, workspace, out: $stderr
|
|
26
|
+
@provided_requires = requires.compact
|
|
27
|
+
@workspace = workspace
|
|
28
|
+
@out = out
|
|
29
|
+
end
|
|
30
|
+
|
|
31
|
+
# @return [Array<String>]
|
|
32
|
+
def requires
|
|
33
|
+
@requires ||= @provided_requires + (workspace.global_environ&.requires || [])
|
|
34
|
+
end
|
|
35
|
+
alias required requires
|
|
36
|
+
|
|
37
|
+
# @sg-ignore flow sensitive typing needs to understand reassignment
|
|
38
|
+
# @return [Array<Gem::Specification>]
|
|
39
|
+
def uncached_gemspecs
|
|
40
|
+
if @uncached_gemspecs.nil?
|
|
41
|
+
@uncached_gemspecs = []
|
|
42
|
+
pins # force lazy-loaded pin lookup
|
|
43
|
+
end
|
|
44
|
+
@uncached_gemspecs
|
|
45
|
+
end
|
|
46
|
+
|
|
47
|
+
# @return [Array<Pin::Base>]
|
|
48
|
+
def pins
|
|
49
|
+
@pins ||= load_serialized_gem_pins + (workspace.global_environ&.pins || [])
|
|
50
|
+
end
|
|
51
|
+
|
|
52
|
+
# @return [void]
|
|
53
|
+
def reset_pins!
|
|
54
|
+
@uncached_gemspecs = nil
|
|
55
|
+
@pins = nil
|
|
56
|
+
end
|
|
57
|
+
|
|
58
|
+
# @return [Solargraph::PinCache]
|
|
59
|
+
def pin_cache
|
|
60
|
+
@pin_cache ||= workspace.fresh_pincache
|
|
61
|
+
end
|
|
62
|
+
|
|
63
|
+
def any_uncached?
|
|
64
|
+
uncached_gemspecs.any?
|
|
65
|
+
end
|
|
66
|
+
|
|
67
|
+
# Cache all pins needed for the sources in this doc_map
|
|
68
|
+
# @param out [StringIO, IO, nil] output stream for logging
|
|
69
|
+
# @param rebuild [Boolean] whether to rebuild the pins even if they are cached
|
|
70
|
+
# @return [void]
|
|
71
|
+
def cache_doc_map_gems! out, rebuild: false
|
|
72
|
+
unless uncached_gemspecs.empty?
|
|
73
|
+
logger.info do
|
|
74
|
+
gem_desc = uncached_gemspecs.map { |gemspec| "#{gemspec.name}:#{gemspec.version}" }.join(', ')
|
|
75
|
+
"Caching pins for gems: #{gem_desc}"
|
|
76
|
+
end
|
|
77
|
+
end
|
|
78
|
+
time = Benchmark.measure do
|
|
79
|
+
uncached_gemspecs.each do |gemspec|
|
|
80
|
+
cache(gemspec, rebuild: rebuild, out: out)
|
|
81
|
+
end
|
|
82
|
+
end
|
|
83
|
+
milliseconds = (time.real * 1000).round
|
|
84
|
+
if (milliseconds > 500) && uncached_gemspecs.any? && out && uncached_gemspecs.any?
|
|
85
|
+
out.puts "Built #{uncached_gemspecs.length} gems in #{milliseconds} ms"
|
|
86
|
+
end
|
|
87
|
+
reset_pins!
|
|
88
|
+
end
|
|
89
|
+
|
|
90
|
+
# @return [Array<String>]
|
|
91
|
+
def unresolved_requires
|
|
92
|
+
@unresolved_requires ||= required_gems_map.select { |_, gemspecs| gemspecs.nil? }.keys
|
|
93
|
+
end
|
|
94
|
+
|
|
95
|
+
# @return [Array<Gem::Specification>]
|
|
96
|
+
# @param out [IO, nil]
|
|
97
|
+
def dependencies out: $stderr
|
|
98
|
+
@dependencies ||=
|
|
99
|
+
begin
|
|
100
|
+
gem_deps = gemspecs
|
|
101
|
+
.flat_map { |spec| workspace.fetch_dependencies(spec, out: out) }
|
|
102
|
+
.uniq(&:name)
|
|
103
|
+
stdlib_deps = gemspecs
|
|
104
|
+
.flat_map { |spec| workspace.stdlib_dependencies(spec.name) }
|
|
105
|
+
.flat_map { |dep_name| workspace.resolve_require(dep_name) }
|
|
106
|
+
.compact
|
|
107
|
+
existing_gems = gemspecs.map(&:name)
|
|
108
|
+
(gem_deps + stdlib_deps).reject { |gemspec| existing_gems.include? gemspec.name }
|
|
109
|
+
end
|
|
110
|
+
end
|
|
111
|
+
|
|
112
|
+
# Cache gem documentation if needed for this doc_map
|
|
113
|
+
#
|
|
114
|
+
# @param gemspec [Gem::Specification]
|
|
115
|
+
# @param rebuild [Boolean] whether to rebuild the pins even if they are cached
|
|
116
|
+
# @param out [StringIO, IO, nil] output stream for logging
|
|
117
|
+
#
|
|
118
|
+
# @return [void]
|
|
119
|
+
def cache gemspec, rebuild: false, out: nil
|
|
120
|
+
pin_cache.cache_gem(gemspec: gemspec,
|
|
121
|
+
rebuild: rebuild,
|
|
122
|
+
out: out)
|
|
123
|
+
end
|
|
124
|
+
|
|
125
|
+
private
|
|
126
|
+
|
|
127
|
+
# @return [Array<Gem::Specification>]
|
|
128
|
+
def gemspecs
|
|
129
|
+
@gemspecs ||= required_gems_map.values.compact.flatten
|
|
130
|
+
end
|
|
131
|
+
|
|
132
|
+
# @param out [IO, nil]
|
|
133
|
+
# @return [Array<Pin::Base>]
|
|
134
|
+
def load_serialized_gem_pins out: @out
|
|
135
|
+
serialized_pins = []
|
|
136
|
+
with_gemspecs, without_gemspecs = required_gems_map.partition { |_, v| v }
|
|
137
|
+
# @sg-ignore Need better typing for Hash[]
|
|
138
|
+
# @type [Array<String>]
|
|
139
|
+
missing_paths = Hash[without_gemspecs].keys
|
|
140
|
+
# @type [Array<Gem::Specification>]
|
|
141
|
+
gemspecs = Hash[with_gemspecs].values.flatten.compact + dependencies(out: out).to_a
|
|
142
|
+
|
|
143
|
+
# if we are type checking a gem project, we should not include
|
|
144
|
+
# pins from rbs or yard from that gem here - we use our own
|
|
145
|
+
# parser for those pins
|
|
146
|
+
|
|
147
|
+
# @param gemspec [Gem::Specification, Bundler::LazySpecification, Bundler::StubSpecification]
|
|
148
|
+
gemspecs.reject! do |gemspec|
|
|
149
|
+
gemspec.respond_to?(:source) &&
|
|
150
|
+
gemspec.source.instance_of?(Bundler::Source::Gemspec) &&
|
|
151
|
+
gemspec.source.respond_to?(:path) &&
|
|
152
|
+
gemspec.source.path == Pathname.new('.')
|
|
153
|
+
end
|
|
154
|
+
|
|
155
|
+
missing_paths.each do |path|
|
|
156
|
+
# this will load from disk if needed; no need to manage
|
|
157
|
+
# uncached_gemspecs to trigger that later
|
|
158
|
+
stdlib_name_guess = path.split('/').first
|
|
159
|
+
|
|
160
|
+
# try to resolve the stdlib name
|
|
161
|
+
# @type [Array<String>]
|
|
162
|
+
deps = workspace.stdlib_dependencies(stdlib_name_guess) || []
|
|
163
|
+
[stdlib_name_guess, *deps].compact.each do |potential_stdlib_name|
|
|
164
|
+
# @sg-ignore Need to support splatting in literal array
|
|
165
|
+
rbs_pins = pin_cache.cache_stdlib_rbs_map potential_stdlib_name
|
|
166
|
+
serialized_pins.concat rbs_pins if rbs_pins
|
|
167
|
+
end
|
|
168
|
+
end
|
|
169
|
+
|
|
170
|
+
existing_pin_count = serialized_pins.length
|
|
171
|
+
time = Benchmark.measure do
|
|
172
|
+
gemspecs.each do |gemspec|
|
|
173
|
+
# only deserializes already-cached gems
|
|
174
|
+
gemspec_pins = pin_cache.deserialize_combined_pin_cache gemspec
|
|
175
|
+
if gemspec_pins
|
|
176
|
+
serialized_pins.concat gemspec_pins
|
|
177
|
+
else
|
|
178
|
+
uncached_gemspecs << gemspec
|
|
179
|
+
end
|
|
180
|
+
end
|
|
181
|
+
end
|
|
182
|
+
pins_processed = serialized_pins.length - existing_pin_count
|
|
183
|
+
milliseconds = (time.real * 1000).round
|
|
184
|
+
if (milliseconds > 500) && out && gemspecs.any?
|
|
185
|
+
out.puts "Deserialized #{serialized_pins.length} gem pins from #{PinCache.base_dir} in #{milliseconds} ms"
|
|
186
|
+
end
|
|
187
|
+
uncached_gemspecs.uniq! { |gemspec| "#{gemspec.name}:#{gemspec.version}" }
|
|
188
|
+
serialized_pins
|
|
189
|
+
end
|
|
190
|
+
|
|
191
|
+
# @return [Hash{String => Array<Gem::Specification>}]
|
|
192
|
+
def required_gems_map
|
|
193
|
+
@required_gems_map ||= requires.to_h { |path| [path, workspace.resolve_require(path)] }
|
|
194
|
+
end
|
|
195
|
+
|
|
196
|
+
def inspect
|
|
197
|
+
self.class.inspect
|
|
198
|
+
end
|
|
199
|
+
end
|
|
200
|
+
end
|