solargraph 0.58.1 → 0.58.2

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.
Files changed (147) hide show
  1. checksums.yaml +4 -4
  2. data/.gitignore +1 -0
  3. data/CHANGELOG.md +7 -1
  4. data/lib/solargraph/api_map/cache.rb +110 -110
  5. data/lib/solargraph/api_map/constants.rb +279 -279
  6. data/lib/solargraph/api_map/index.rb +193 -193
  7. data/lib/solargraph/api_map/source_to_yard.rb +97 -97
  8. data/lib/solargraph/api_map/store.rb +384 -384
  9. data/lib/solargraph/api_map.rb +945 -945
  10. data/lib/solargraph/complex_type/type_methods.rb +228 -228
  11. data/lib/solargraph/complex_type/unique_type.rb +482 -482
  12. data/lib/solargraph/complex_type.rb +444 -444
  13. data/lib/solargraph/convention/data_definition/data_definition_node.rb +91 -91
  14. data/lib/solargraph/convention/data_definition.rb +105 -105
  15. data/lib/solargraph/convention/struct_definition/struct_assignment_node.rb +61 -61
  16. data/lib/solargraph/convention/struct_definition/struct_definition_node.rb +102 -102
  17. data/lib/solargraph/convention/struct_definition.rb +164 -164
  18. data/lib/solargraph/diagnostics/require_not_found.rb +53 -53
  19. data/lib/solargraph/diagnostics/rubocop.rb +118 -118
  20. data/lib/solargraph/diagnostics/rubocop_helpers.rb +68 -68
  21. data/lib/solargraph/diagnostics/type_check.rb +55 -55
  22. data/lib/solargraph/doc_map.rb +439 -439
  23. data/lib/solargraph/equality.rb +34 -34
  24. data/lib/solargraph/gem_pins.rb +98 -98
  25. data/lib/solargraph/language_server/host/diagnoser.rb +89 -89
  26. data/lib/solargraph/language_server/host/dispatch.rb +130 -130
  27. data/lib/solargraph/language_server/host/message_worker.rb +112 -112
  28. data/lib/solargraph/language_server/host/sources.rb +99 -99
  29. data/lib/solargraph/language_server/host.rb +878 -878
  30. data/lib/solargraph/language_server/message/extended/check_gem_version.rb +114 -114
  31. data/lib/solargraph/language_server/message/extended/document.rb +23 -23
  32. data/lib/solargraph/language_server/message/text_document/completion.rb +56 -56
  33. data/lib/solargraph/language_server/message/text_document/definition.rb +40 -40
  34. data/lib/solargraph/language_server/message/text_document/document_symbol.rb +26 -26
  35. data/lib/solargraph/language_server/message/text_document/formatting.rb +148 -148
  36. data/lib/solargraph/language_server/message/text_document/hover.rb +58 -58
  37. data/lib/solargraph/language_server/message/text_document/signature_help.rb +24 -24
  38. data/lib/solargraph/language_server/message/text_document/type_definition.rb +25 -25
  39. data/lib/solargraph/language_server/message/workspace/workspace_symbol.rb +23 -23
  40. data/lib/solargraph/library.rb +683 -683
  41. data/lib/solargraph/location.rb +82 -82
  42. data/lib/solargraph/logging.rb +37 -37
  43. data/lib/solargraph/parser/comment_ripper.rb +69 -69
  44. data/lib/solargraph/parser/flow_sensitive_typing.rb +255 -255
  45. data/lib/solargraph/parser/node_processor/base.rb +92 -92
  46. data/lib/solargraph/parser/node_processor.rb +62 -62
  47. data/lib/solargraph/parser/parser_gem/class_methods.rb +149 -149
  48. data/lib/solargraph/parser/parser_gem/node_chainer.rb +166 -166
  49. data/lib/solargraph/parser/parser_gem/node_methods.rb +486 -486
  50. data/lib/solargraph/parser/parser_gem/node_processors/and_node.rb +22 -22
  51. data/lib/solargraph/parser/parser_gem/node_processors/args_node.rb +59 -59
  52. data/lib/solargraph/parser/parser_gem/node_processors/begin_node.rb +15 -15
  53. data/lib/solargraph/parser/parser_gem/node_processors/block_node.rb +46 -46
  54. data/lib/solargraph/parser/parser_gem/node_processors/def_node.rb +53 -53
  55. data/lib/solargraph/parser/parser_gem/node_processors/if_node.rb +23 -23
  56. data/lib/solargraph/parser/parser_gem/node_processors/ivasgn_node.rb +40 -40
  57. data/lib/solargraph/parser/parser_gem/node_processors/lvasgn_node.rb +29 -29
  58. data/lib/solargraph/parser/parser_gem/node_processors/masgn_node.rb +59 -59
  59. data/lib/solargraph/parser/parser_gem/node_processors/opasgn_node.rb +98 -98
  60. data/lib/solargraph/parser/parser_gem/node_processors/orasgn_node.rb +17 -17
  61. data/lib/solargraph/parser/parser_gem/node_processors/resbody_node.rb +38 -38
  62. data/lib/solargraph/parser/parser_gem/node_processors/sclass_node.rb +52 -52
  63. data/lib/solargraph/parser/parser_gem/node_processors/send_node.rb +291 -291
  64. data/lib/solargraph/parser/parser_gem/node_processors/while_node.rb +29 -29
  65. data/lib/solargraph/parser/parser_gem/node_processors.rb +70 -70
  66. data/lib/solargraph/parser/region.rb +69 -69
  67. data/lib/solargraph/parser/snippet.rb +17 -17
  68. data/lib/solargraph/pin/base.rb +729 -729
  69. data/lib/solargraph/pin/base_variable.rb +126 -126
  70. data/lib/solargraph/pin/block.rb +104 -104
  71. data/lib/solargraph/pin/breakable.rb +9 -9
  72. data/lib/solargraph/pin/callable.rb +231 -231
  73. data/lib/solargraph/pin/closure.rb +72 -72
  74. data/lib/solargraph/pin/common.rb +79 -79
  75. data/lib/solargraph/pin/conversions.rb +123 -123
  76. data/lib/solargraph/pin/delegated_method.rb +120 -120
  77. data/lib/solargraph/pin/documenting.rb +114 -114
  78. data/lib/solargraph/pin/instance_variable.rb +34 -34
  79. data/lib/solargraph/pin/keyword.rb +20 -20
  80. data/lib/solargraph/pin/local_variable.rb +75 -75
  81. data/lib/solargraph/pin/method.rb +672 -672
  82. data/lib/solargraph/pin/method_alias.rb +34 -34
  83. data/lib/solargraph/pin/namespace.rb +115 -115
  84. data/lib/solargraph/pin/parameter.rb +275 -275
  85. data/lib/solargraph/pin/proxy_type.rb +39 -39
  86. data/lib/solargraph/pin/reference/override.rb +47 -47
  87. data/lib/solargraph/pin/reference/superclass.rb +15 -15
  88. data/lib/solargraph/pin/reference.rb +39 -39
  89. data/lib/solargraph/pin/search.rb +61 -61
  90. data/lib/solargraph/pin/signature.rb +61 -61
  91. data/lib/solargraph/pin/symbol.rb +53 -53
  92. data/lib/solargraph/pin/until.rb +18 -18
  93. data/lib/solargraph/pin/while.rb +18 -18
  94. data/lib/solargraph/pin.rb +44 -44
  95. data/lib/solargraph/pin_cache.rb +245 -245
  96. data/lib/solargraph/position.rb +132 -119
  97. data/lib/solargraph/range.rb +112 -112
  98. data/lib/solargraph/rbs_map/conversions.rb +823 -823
  99. data/lib/solargraph/rbs_map/core_map.rb +58 -58
  100. data/lib/solargraph/rbs_map/stdlib_map.rb +43 -43
  101. data/lib/solargraph/rbs_map.rb +163 -163
  102. data/lib/solargraph/shell.rb +352 -352
  103. data/lib/solargraph/source/chain/call.rb +337 -337
  104. data/lib/solargraph/source/chain/constant.rb +26 -26
  105. data/lib/solargraph/source/chain/hash.rb +34 -34
  106. data/lib/solargraph/source/chain/if.rb +28 -28
  107. data/lib/solargraph/source/chain/instance_variable.rb +13 -13
  108. data/lib/solargraph/source/chain/literal.rb +48 -48
  109. data/lib/solargraph/source/chain/or.rb +23 -23
  110. data/lib/solargraph/source/chain.rb +291 -291
  111. data/lib/solargraph/source/change.rb +82 -82
  112. data/lib/solargraph/source/cursor.rb +166 -166
  113. data/lib/solargraph/source/source_chainer.rb +194 -194
  114. data/lib/solargraph/source/updater.rb +55 -55
  115. data/lib/solargraph/source.rb +498 -498
  116. data/lib/solargraph/source_map/clip.rb +226 -226
  117. data/lib/solargraph/source_map/data.rb +34 -34
  118. data/lib/solargraph/source_map/mapper.rb +259 -259
  119. data/lib/solargraph/source_map.rb +212 -212
  120. data/lib/solargraph/type_checker/checks.rb +124 -124
  121. data/lib/solargraph/type_checker/param_def.rb +37 -37
  122. data/lib/solargraph/type_checker/problem.rb +32 -32
  123. data/lib/solargraph/type_checker/rules.rb +84 -84
  124. data/lib/solargraph/type_checker.rb +814 -814
  125. data/lib/solargraph/version.rb +1 -1
  126. data/lib/solargraph/workspace/config.rb +255 -255
  127. data/lib/solargraph/workspace/require_paths.rb +97 -97
  128. data/lib/solargraph/workspace.rb +220 -220
  129. data/lib/solargraph/yard_map/helpers.rb +44 -44
  130. data/lib/solargraph/yard_map/mapper/to_method.rb +130 -130
  131. data/lib/solargraph/yard_map/mapper/to_namespace.rb +31 -31
  132. data/lib/solargraph/yard_map/mapper.rb +79 -79
  133. data/lib/solargraph/yard_map/to_method.rb +89 -89
  134. data/lib/solargraph/yardoc.rb +87 -87
  135. data/lib/solargraph.rb +105 -105
  136. data/rbs_collection.yaml +1 -1
  137. metadata +12 -12
  138. /data/{sig → rbs}/shims/ast/0/node.rbs +0 -0
  139. /data/{sig → rbs}/shims/ast/2.4/.rbs_meta.yaml +0 -0
  140. /data/{sig → rbs}/shims/ast/2.4/ast.rbs +0 -0
  141. /data/{sig → rbs}/shims/parser/3.2.0.1/builders/default.rbs +0 -0
  142. /data/{sig → rbs}/shims/parser/3.2.0.1/manifest.yaml +0 -0
  143. /data/{sig → rbs}/shims/parser/3.2.0.1/parser.rbs +0 -0
  144. /data/{sig → rbs}/shims/parser/3.2.0.1/polyfill.rbs +0 -0
  145. /data/{sig → rbs}/shims/thor/1.2.0.1/.rbs_meta.yaml +0 -0
  146. /data/{sig → rbs}/shims/thor/1.2.0.1/manifest.yaml +0 -0
  147. /data/{sig → rbs}/shims/thor/1.2.0.1/thor.rbs +0 -0
@@ -1,683 +1,683 @@
1
- # frozen_string_literal: true
2
-
3
- require 'pathname'
4
- require 'observer'
5
- require 'open3'
6
-
7
- module Solargraph
8
- # A Library handles coordination between a Workspace and an ApiMap.
9
- #
10
- class Library
11
- include Logging
12
- include Observable
13
-
14
- # @return [Solargraph::Workspace]
15
- attr_reader :workspace
16
-
17
- # @return [String, nil]
18
- attr_reader :name
19
-
20
- # @return [Source, nil]
21
- attr_reader :current
22
-
23
- # @return [LanguageServer::Progress, nil]
24
- attr_reader :cache_progress
25
-
26
- # @param workspace [Solargraph::Workspace]
27
- # @param name [String, nil]
28
- def initialize workspace = Solargraph::Workspace.new, name = nil
29
- @workspace = workspace
30
- @name = name
31
- # @type [Integer, nil]
32
- @total = nil
33
- # @type [Source, nil]
34
- @current = nil
35
- @sync_count = 0
36
- end
37
-
38
- def inspect
39
- # Let's not deal with insane data dumps in spec failures
40
- to_s
41
- end
42
-
43
- # True if the ApiMap is up to date with the library's workspace and open
44
- # files.
45
- #
46
- # @return [Boolean]
47
- def synchronized?
48
- @sync_count < 2
49
- end
50
-
51
- # Attach a source to the library.
52
- #
53
- # The attached source does not need to be a part of the workspace. The
54
- # library will include it in the ApiMap while it's attached. Only one
55
- # source can be attached to the library at a time.
56
- #
57
- # @param source [Source, nil]
58
- # @return [void]
59
- def attach source
60
- if @current && (!source || @current.filename != source.filename) && source_map_hash.key?(@current.filename) && !workspace.has_file?(@current.filename)
61
- source_map_hash.delete @current.filename
62
- source_map_external_require_hash.delete @current.filename
63
- @external_requires = nil
64
- end
65
- changed = source && @current != source
66
- @current = source
67
- maybe_map @current
68
- catalog if changed
69
- end
70
-
71
- # True if the specified file is currently attached.
72
- #
73
- # @param filename [String]
74
- # @return [Boolean]
75
- def attached? filename
76
- !@current.nil? && @current.filename == filename
77
- end
78
- alias open? attached?
79
-
80
- # Detach the specified file if it is currently attached to the library.
81
- #
82
- # @param filename [String]
83
- # @return [Boolean] True if the specified file was detached
84
- def detach filename
85
- return false if @current.nil? || @current.filename != filename
86
- attach nil
87
- true
88
- end
89
-
90
- # True if the specified file is included in the workspace (but not
91
- # necessarily open).
92
- #
93
- # @param filename [String]
94
- # @return [Boolean]
95
- def contain? filename
96
- workspace.has_file?(filename)
97
- end
98
-
99
- # Create a source to be added to the workspace. The file is ignored if it is
100
- # neither open in the library nor included in the workspace.
101
- #
102
- # @param filename [String]
103
- # @param text [String] The contents of the file
104
- # @return [Boolean] True if the file was added to the workspace.
105
- def create filename, text
106
- return false unless contain?(filename) || open?(filename)
107
- source = Solargraph::Source.load_string(text, filename)
108
- workspace.merge(source)
109
- true
110
- end
111
-
112
- # Create file sources from files on disk. A file is ignored if it is
113
- # neither open in the library nor included in the workspace.
114
- #
115
- # @param filenames [Array<String>]
116
- # @return [Boolean] True if at least one file was added to the workspace.
117
- def create_from_disk *filenames
118
- sources = filenames
119
- .reject { |filename| File.directory?(filename) || !File.exist?(filename) }
120
- .map { |filename| Solargraph::Source.load_string(File.read(filename), filename) }
121
- result = workspace.merge(*sources)
122
- sources.each { |source| maybe_map source }
123
- result
124
- end
125
-
126
- # Delete files from the library. Deleting a file will make it unavailable
127
- # for checkout and optionally remove it from the workspace unless the
128
- # workspace configuration determines that it should still exist.
129
- #
130
- # @param filenames [Array<String>]
131
- # @return [Boolean] True if any file was deleted
132
- def delete *filenames
133
- result = false
134
- filenames.each do |filename|
135
- detach filename
136
- source_map_hash.delete(filename)
137
- result ||= workspace.remove(filename)
138
- end
139
- result
140
- end
141
-
142
- # Close a file in the library. Closing a file will make it unavailable for
143
- # checkout although it may still exist in the workspace.
144
- #
145
- # @param filename [String]
146
- # @return [void]
147
- def close filename
148
- return unless @current&.filename == filename
149
-
150
- @current = nil
151
- catalog unless workspace.has_file?(filename)
152
- end
153
-
154
- # Get completion suggestions at the specified file and location.
155
- #
156
- # @param filename [String] The file to analyze
157
- # @param line [Integer] The zero-based line number
158
- # @param column [Integer] The zero-based column number
159
- # @return [SourceMap::Completion, nil]
160
- # @todo Take a Location instead of filename/line/column
161
- def completions_at filename, line, column
162
- sync_catalog
163
- position = Position.new(line, column)
164
- cursor = Source::Cursor.new(read(filename), position)
165
- mutex.synchronize { api_map.clip(cursor).complete }
166
- rescue FileNotFoundError => e
167
- handle_file_not_found filename, e
168
- end
169
-
170
- # Get definition suggestions for the expression at the specified file and
171
- # location.
172
- #
173
- # @param filename [String] The file to analyze
174
- # @param line [Integer] The zero-based line number
175
- # @param column [Integer] The zero-based column number
176
- # @return [Array<Solargraph::Pin::Base>, nil]
177
- # @todo Take filename/position instead of filename/line/column
178
- def definitions_at filename, line, column
179
- sync_catalog
180
- position = Position.new(line, column)
181
- cursor = Source::Cursor.new(read(filename), position)
182
- if cursor.comment?
183
- source = read(filename)
184
- offset = Solargraph::Position.to_offset(source.code, Solargraph::Position.new(line, column))
185
- lft = source.code[0..offset-1].match(/\[[a-z0-9_:<, ]*?([a-z0-9_:]*)\z/i)
186
- rgt = source.code[offset..-1].match(/^([a-z0-9_]*)(:[a-z0-9_:]*)?[\]>, ]/i)
187
- if lft && rgt
188
- tag = (lft[1] + rgt[1]).sub(/:+$/, '')
189
- clip = mutex.synchronize { api_map.clip(cursor) }
190
- clip.translate tag
191
- else
192
- []
193
- end
194
- else
195
- mutex.synchronize do
196
- clip = api_map.clip(cursor)
197
- clip.define.map { |pin| pin.realize(api_map) }
198
- end
199
- end
200
- rescue FileNotFoundError => e
201
- handle_file_not_found(filename, e)
202
- end
203
-
204
- # Get type definition suggestions for the expression at the specified file and
205
- # location.
206
- #
207
- # @param filename [String] The file to analyze
208
- # @param line [Integer] The zero-based line number
209
- # @param column [Integer] The zero-based column number
210
- # @return [Array<Solargraph::Pin::Base>, nil]
211
- # @todo Take filename/position instead of filename/line/column
212
- def type_definitions_at filename, line, column
213
- sync_catalog
214
- position = Position.new(line, column)
215
- cursor = Source::Cursor.new(read(filename), position)
216
- mutex.synchronize { api_map.clip(cursor).types }
217
- rescue FileNotFoundError => e
218
- handle_file_not_found filename, e
219
- end
220
-
221
- # Get signature suggestions for the method at the specified file and
222
- # location.
223
- #
224
- # @param filename [String] The file to analyze
225
- # @param line [Integer] The zero-based line number
226
- # @param column [Integer] The zero-based column number
227
- # @return [Array<Solargraph::Pin::Base>]
228
- # @todo Take filename/position instead of filename/line/column
229
- def signatures_at filename, line, column
230
- sync_catalog
231
- position = Position.new(line, column)
232
- cursor = Source::Cursor.new(read(filename), position)
233
- mutex.synchronize { api_map.clip(cursor).signify }
234
- end
235
-
236
- # @param filename [String]
237
- # @param line [Integer]
238
- # @param column [Integer]
239
- # @param strip [Boolean] Strip special characters from variable names
240
- # @param only [Boolean] Search for references in the current file only
241
- # @return [Array<Solargraph::Location>]
242
- # @todo Take a Location instead of filename/line/column
243
- def references_from filename, line, column, strip: false, only: false
244
- sync_catalog
245
- cursor = Source::Cursor.new(read(filename), [line, column])
246
- clip = mutex.synchronize { api_map.clip(cursor) }
247
- pin = clip.define.first
248
- return [] unless pin
249
- result = []
250
- files = if only
251
- [api_map.source_map(filename)]
252
- else
253
- (workspace.sources + (@current ? [@current] : []))
254
- end
255
- files.uniq(&:filename).each do |source|
256
- found = source.references(pin.name)
257
- found.select! do |loc|
258
- referenced = definitions_at(loc.filename, loc.range.ending.line, loc.range.ending.character).first
259
- referenced&.path == pin.path
260
- end
261
- if pin.path == 'Class#new'
262
- caller = cursor.chain.base.infer(api_map, clip.send(:closure), clip.locals).first
263
- if caller.defined?
264
- found.select! do |loc|
265
- clip = api_map.clip_at(loc.filename, loc.range.start)
266
- other = clip.send(:cursor).chain.base.infer(api_map, clip.send(:closure), clip.locals).first
267
- caller == other
268
- end
269
- else
270
- found.clear
271
- end
272
- end
273
- # HACK: for language clients that exclude special characters from the start of variable names
274
- if strip && match = cursor.word.match(/^[^a-z0-9_]+/i)
275
- found.map! do |loc|
276
- Solargraph::Location.new(loc.filename, Solargraph::Range.from_to(loc.range.start.line, loc.range.start.column + match[0].length, loc.range.ending.line, loc.range.ending.column))
277
- end
278
- end
279
- result.concat(found.sort do |a, b|
280
- a.range.start.line <=> b.range.start.line
281
- end)
282
- end
283
- result.uniq
284
- end
285
-
286
- # Get the pins at the specified location or nil if the pin does not exist.
287
- #
288
- # @param location [Location]
289
- # @return [Array<Solargraph::Pin::Base>]
290
- def locate_pins location
291
- sync_catalog
292
- mutex.synchronize { api_map.locate_pins(location).map { |pin| pin.realize(api_map) } }
293
- end
294
-
295
- # Match a require reference to a file.
296
- #
297
- # @param location [Location]
298
- # @return [Location, nil]
299
- def locate_ref location
300
- map = source_map_hash[location.filename]
301
- return if map.nil?
302
- pin = map.requires.select { |p| p.location.range.contain?(location.range.start) }.first
303
- return nil if pin.nil?
304
- # @param full [String]
305
- return_if_match = proc do |full|
306
- if source_map_hash.key?(full)
307
- return Location.new(full, Solargraph::Range.from_to(0, 0, 0, 0))
308
- end
309
- end
310
- workspace.require_paths.each do |path|
311
- full = File.join path, pin.name
312
- return_if_match.(full)
313
- return_if_match.(full << ".rb")
314
- end
315
- nil
316
- rescue FileNotFoundError
317
- nil
318
- end
319
-
320
- # Get an array of pins that match a path.
321
- #
322
- # @param path [String]
323
- # @return [Enumerable<Solargraph::Pin::Base>]
324
- def get_path_pins path
325
- sync_catalog
326
- mutex.synchronize { api_map.get_path_suggestions(path) }
327
- end
328
-
329
- # @param query [String]
330
- # @return [Enumerable<YARD::CodeObjects::Base>]
331
- # @return [Array(ApiMap, Enumerable<Pin::Base>)]
332
- def document query
333
- sync_catalog
334
- mutex.synchronize { [api_map, api_map.get_path_pins(query)] }
335
- end
336
-
337
- # @param query [String]
338
- # @return [Array<String>]
339
- def search query
340
- sync_catalog
341
- mutex.synchronize { api_map.search query }
342
- end
343
-
344
- # Get an array of all symbols in the workspace that match the query.
345
- #
346
- # @param query [String]
347
- # @return [Array<Pin::Base>]
348
- def query_symbols query
349
- sync_catalog
350
- mutex.synchronize { api_map.query_symbols query }
351
- end
352
-
353
- # Get an array of document symbols.
354
- #
355
- # Document symbols are composed of namespace, method, and constant pins.
356
- # The results of this query are appropriate for building the response to a
357
- # textDocument/documentSymbol message in the language server protocol.
358
- #
359
- # @param filename [String]
360
- # @return [Array<Solargraph::Pin::Base>]
361
- def document_symbols filename
362
- sync_catalog
363
- mutex.synchronize { api_map.document_symbols(filename) }
364
- end
365
-
366
- # @param path [String]
367
- # @return [Enumerable<Solargraph::Pin::Base>]
368
- def path_pins path
369
- sync_catalog
370
- mutex.synchronize { api_map.get_path_suggestions(path) }
371
- end
372
-
373
- # @return [Array<SourceMap>]
374
- def source_maps
375
- source_map_hash.values
376
- end
377
-
378
- # Get the current text of a file in the library.
379
- #
380
- # @param filename [String]
381
- # @return [String]
382
- def read_text filename
383
- source = read(filename)
384
- source.code
385
- end
386
-
387
- # Get diagnostics about a file.
388
- #
389
- # @param filename [String]
390
- # @return [Array<Hash>]
391
- def diagnose filename
392
- # @todo Only open files get diagnosed. Determine whether anything or
393
- # everything in the workspace should get diagnosed, or if there should
394
- # be an option to do so.
395
- #
396
- sync_catalog
397
- return [] unless open?(filename)
398
- result = []
399
- source = read(filename)
400
-
401
- # @type [Hash{Class<Solargraph::Diagnostics::Base> => Array<String>}]
402
- repargs = {}
403
- workspace.config.reporters.each do |line|
404
- if line == 'all!'
405
- Diagnostics.reporters.each do |reporter_name|
406
- repargs[Diagnostics.reporter(reporter_name)] ||= []
407
- end
408
- else
409
- args = line.split(':').map(&:strip)
410
- name = args.shift
411
- reporter = Diagnostics.reporter(name)
412
- raise DiagnosticsError, "Diagnostics reporter #{name} does not exist" if reporter.nil?
413
- repargs[reporter] ||= []
414
- repargs[reporter].concat args
415
- end
416
- end
417
- repargs.each_pair do |reporter, args|
418
- result.concat reporter.new(*args.uniq).diagnose(source, api_map)
419
- end
420
- result
421
- end
422
-
423
- # Update the ApiMap from the library's workspace and open files.
424
- #
425
- # @return [void]
426
- def catalog
427
- @sync_count += 1
428
- end
429
-
430
- # @return [Bench]
431
- def bench
432
- Bench.new(
433
- source_maps: source_map_hash.values,
434
- workspace: workspace,
435
- external_requires: external_requires,
436
- live_map: @current ? source_map_hash[@current.filename] : nil
437
- )
438
- end
439
-
440
- # Create a library from a directory.
441
- #
442
- # @param directory [String] The path to be used for the workspace
443
- # @param name [String, nil]
444
- # @return [Solargraph::Library]
445
- def self.load directory = '', name = nil
446
- Solargraph::Library.new(Solargraph::Workspace.new(directory), name)
447
- end
448
-
449
- # Try to merge a source into the library's workspace. If the workspace is
450
- # not configured to include the source, it gets ignored.
451
- #
452
- # @param source [Source]
453
- # @return [Boolean] True if the source was merged into the workspace.
454
- def merge source
455
- result = workspace.merge(source)
456
- maybe_map source
457
- result
458
- end
459
-
460
- # @return [Hash{String => SourceMap}]
461
- def source_map_hash
462
- @source_map_hash ||= {}
463
- end
464
-
465
- def mapped?
466
- (workspace.filenames - source_map_hash.keys).empty?
467
- end
468
-
469
- # @return [SourceMap, Boolean]
470
- def next_map
471
- return false if mapped?
472
- src = workspace.sources.find { |s| !source_map_hash.key?(s.filename) }
473
- if src
474
- Logging.logger.debug "Mapping #{src.filename}"
475
- source_map_hash[src.filename] = Solargraph::SourceMap.map(src)
476
- source_map_hash[src.filename]
477
- else
478
- false
479
- end
480
- end
481
-
482
- # @return [self]
483
- def map!
484
- workspace.sources.each do |src|
485
- source_map_hash[src.filename] = Solargraph::SourceMap.map(src)
486
- find_external_requires source_map_hash[src.filename]
487
- end
488
- self
489
- end
490
-
491
- # @return [Array<Solargraph::Pin::Base>]
492
- def pins
493
- @pins ||= []
494
- end
495
-
496
- # @return [Set<String>]
497
- def external_requires
498
- @external_requires ||= source_map_external_require_hash.values.flatten.to_set
499
- end
500
-
501
- private
502
-
503
- # @return [Hash{String => Array<String>}]
504
- def source_map_external_require_hash
505
- @source_map_external_require_hash ||= {}
506
- end
507
-
508
- # @param source_map [SourceMap]
509
- # @return [void]
510
- def find_external_requires source_map
511
- # @type [Set<String>]
512
- new_set = source_map.requires.map(&:name).to_set
513
- # return if new_set == source_map_external_require_hash[source_map.filename]
514
- _filenames = nil
515
- filenames = ->{ _filenames ||= workspace.filenames.to_set }
516
- source_map_external_require_hash[source_map.filename] = new_set.reject do |path|
517
- workspace.require_paths.any? do |base|
518
- full = File.join(base, path)
519
- filenames[].include?(full) or filenames[].include?(full << ".rb")
520
- end
521
- end
522
- @external_requires = nil
523
- end
524
-
525
- # @return [Thread::Mutex]
526
- def mutex
527
- @mutex ||= Mutex.new
528
- end
529
-
530
- # @return [ApiMap]
531
- def api_map
532
- @api_map ||= Solargraph::ApiMap.new
533
- end
534
-
535
- # Get the source for an open file or create a new source if the file
536
- # exists on disk. Sources created from disk are not added to the open
537
- # workspace files, i.e., the version on disk remains the authoritative
538
- # version.
539
- #
540
- # @raise [FileNotFoundError] if the file does not exist
541
- # @param filename [String]
542
- # @return [Solargraph::Source]
543
- def read filename
544
- return @current if @current && @current.filename == filename
545
- raise FileNotFoundError, "File not found: #{filename}" unless workspace.has_file?(filename)
546
- workspace.source(filename)
547
- end
548
-
549
- # @param filename [String]
550
- # @param error [FileNotFoundError]
551
- # @return [nil]
552
- def handle_file_not_found filename, error
553
- if workspace.source(filename)
554
- Solargraph.logger.debug "#{filename} is not cataloged in the ApiMap"
555
- nil
556
- else
557
- raise error
558
- end
559
- end
560
-
561
- # @param source [Source, nil]
562
- # @return [void]
563
- def maybe_map source
564
- return unless source
565
- return unless @current == source || workspace.has_file?(source.filename)
566
- if source_map_hash.key?(source.filename)
567
- new_map = Solargraph::SourceMap.map(source)
568
- source_map_hash[source.filename] = new_map
569
- else
570
- source_map_hash[source.filename] = Solargraph::SourceMap.map(source)
571
- end
572
- end
573
-
574
- # @return [Set<Gem::Specification>]
575
- def cache_errors
576
- @cache_errors ||= Set.new
577
- end
578
-
579
- # @return [void]
580
- def cache_next_gemspec
581
- return if @cache_progress
582
-
583
- spec = cacheable_specs.first
584
- return end_cache_progress unless spec
585
-
586
- pending = api_map.uncached_gemspecs.length - cache_errors.length - 1
587
-
588
- if Yardoc.processing?(spec)
589
- logger.info "Enqueuing cache of #{spec.name} #{spec.version} (already being processed)"
590
- queued_gemspec_cache.push(spec)
591
- return if pending - queued_gemspec_cache.length < 1
592
-
593
- catalog
594
- sync_catalog
595
- else
596
- logger.info "Caching #{spec.name} #{spec.version}"
597
- Thread.new do
598
- report_cache_progress spec.name, pending
599
- _o, e, s = Open3.capture3(workspace.command_path, 'cache', spec.name, spec.version.to_s)
600
- if s.success?
601
- logger.info "Cached #{spec.name} #{spec.version}"
602
- else
603
- cache_errors.add spec
604
- logger.warn "Error caching gemspec #{spec.name} #{spec.version}"
605
- logger.warn e
606
- end
607
- end_cache_progress
608
- catalog
609
- sync_catalog
610
- end
611
- end
612
- end
613
-
614
- # @return [Array<Gem::Specification>]
615
- def cacheable_specs
616
- cacheable = api_map.uncached_yard_gemspecs +
617
- api_map.uncached_rbs_collection_gemspecs -
618
- queued_gemspec_cache -
619
- cache_errors.to_a
620
- return cacheable unless cacheable.empty?
621
-
622
- queued_gemspec_cache
623
- end
624
-
625
- # @return [Array<Gem::Specification>]
626
- def queued_gemspec_cache
627
- @queued_gemspec_cache ||= []
628
- end
629
-
630
- # @param gem_name [String]
631
- # @param pending [Integer]
632
- # @return [void]
633
- def report_cache_progress gem_name, pending
634
- @total ||= pending
635
- @total = pending if pending > @total
636
- finished = @total - pending
637
- pct = if @total.zero?
638
- 0
639
- else
640
- ((finished.to_f / @total.to_f) * 100).to_i
641
- end
642
- message = "#{gem_name}#{pending > 0 ? " (+#{pending})" : ''}"
643
- # "
644
- if @cache_progress
645
- @cache_progress.report(message, pct)
646
- else
647
- @cache_progress = LanguageServer::Progress.new('Caching gem')
648
- # If we don't send both a begin and a report, the progress notification
649
- # might get stuck in the status bar forever
650
- @cache_progress.begin(message, pct)
651
- changed
652
- notify_observers @cache_progress
653
- @cache_progress.report(message, pct)
654
- end
655
- changed
656
- notify_observers @cache_progress
657
- end
658
-
659
- # @return [void]
660
- def end_cache_progress
661
- changed if @cache_progress&.finish('done')
662
- notify_observers @cache_progress
663
- @cache_progress = nil
664
- @total = nil
665
- end
666
-
667
- # @return [void]
668
- def sync_catalog
669
- return if @sync_count == 0
670
-
671
- mutex.synchronize do
672
- logger.info "Cataloging #{workspace.directory.empty? ? 'generic workspace' : workspace.directory}"
673
- source_map_hash.values.each { |map| find_external_requires(map) }
674
- api_map.catalog bench
675
- logger.info "Catalog complete (#{api_map.source_maps.length} files, #{api_map.pins.length} pins)"
676
- logger.info "#{api_map.uncached_yard_gemspecs.length} uncached YARD gemspecs"
677
- logger.info "#{api_map.uncached_rbs_collection_gemspecs.length} uncached RBS collection gemspecs"
678
- cache_next_gemspec
679
- @sync_count = 0
680
- end
681
- end
682
- end
683
- end
1
+ # frozen_string_literal: true
2
+
3
+ require 'pathname'
4
+ require 'observer'
5
+ require 'open3'
6
+
7
+ module Solargraph
8
+ # A Library handles coordination between a Workspace and an ApiMap.
9
+ #
10
+ class Library
11
+ include Logging
12
+ include Observable
13
+
14
+ # @return [Solargraph::Workspace]
15
+ attr_reader :workspace
16
+
17
+ # @return [String, nil]
18
+ attr_reader :name
19
+
20
+ # @return [Source, nil]
21
+ attr_reader :current
22
+
23
+ # @return [LanguageServer::Progress, nil]
24
+ attr_reader :cache_progress
25
+
26
+ # @param workspace [Solargraph::Workspace]
27
+ # @param name [String, nil]
28
+ def initialize workspace = Solargraph::Workspace.new, name = nil
29
+ @workspace = workspace
30
+ @name = name
31
+ # @type [Integer, nil]
32
+ @total = nil
33
+ # @type [Source, nil]
34
+ @current = nil
35
+ @sync_count = 0
36
+ end
37
+
38
+ def inspect
39
+ # Let's not deal with insane data dumps in spec failures
40
+ to_s
41
+ end
42
+
43
+ # True if the ApiMap is up to date with the library's workspace and open
44
+ # files.
45
+ #
46
+ # @return [Boolean]
47
+ def synchronized?
48
+ @sync_count < 2
49
+ end
50
+
51
+ # Attach a source to the library.
52
+ #
53
+ # The attached source does not need to be a part of the workspace. The
54
+ # library will include it in the ApiMap while it's attached. Only one
55
+ # source can be attached to the library at a time.
56
+ #
57
+ # @param source [Source, nil]
58
+ # @return [void]
59
+ def attach source
60
+ if @current && (!source || @current.filename != source.filename) && source_map_hash.key?(@current.filename) && !workspace.has_file?(@current.filename)
61
+ source_map_hash.delete @current.filename
62
+ source_map_external_require_hash.delete @current.filename
63
+ @external_requires = nil
64
+ end
65
+ changed = source && @current != source
66
+ @current = source
67
+ maybe_map @current
68
+ catalog if changed
69
+ end
70
+
71
+ # True if the specified file is currently attached.
72
+ #
73
+ # @param filename [String]
74
+ # @return [Boolean]
75
+ def attached? filename
76
+ !@current.nil? && @current.filename == filename
77
+ end
78
+ alias open? attached?
79
+
80
+ # Detach the specified file if it is currently attached to the library.
81
+ #
82
+ # @param filename [String]
83
+ # @return [Boolean] True if the specified file was detached
84
+ def detach filename
85
+ return false if @current.nil? || @current.filename != filename
86
+ attach nil
87
+ true
88
+ end
89
+
90
+ # True if the specified file is included in the workspace (but not
91
+ # necessarily open).
92
+ #
93
+ # @param filename [String]
94
+ # @return [Boolean]
95
+ def contain? filename
96
+ workspace.has_file?(filename)
97
+ end
98
+
99
+ # Create a source to be added to the workspace. The file is ignored if it is
100
+ # neither open in the library nor included in the workspace.
101
+ #
102
+ # @param filename [String]
103
+ # @param text [String] The contents of the file
104
+ # @return [Boolean] True if the file was added to the workspace.
105
+ def create filename, text
106
+ return false unless contain?(filename) || open?(filename)
107
+ source = Solargraph::Source.load_string(text, filename)
108
+ workspace.merge(source)
109
+ true
110
+ end
111
+
112
+ # Create file sources from files on disk. A file is ignored if it is
113
+ # neither open in the library nor included in the workspace.
114
+ #
115
+ # @param filenames [Array<String>]
116
+ # @return [Boolean] True if at least one file was added to the workspace.
117
+ def create_from_disk *filenames
118
+ sources = filenames
119
+ .reject { |filename| File.directory?(filename) || !File.exist?(filename) }
120
+ .map { |filename| Solargraph::Source.load_string(File.read(filename), filename) }
121
+ result = workspace.merge(*sources)
122
+ sources.each { |source| maybe_map source }
123
+ result
124
+ end
125
+
126
+ # Delete files from the library. Deleting a file will make it unavailable
127
+ # for checkout and optionally remove it from the workspace unless the
128
+ # workspace configuration determines that it should still exist.
129
+ #
130
+ # @param filenames [Array<String>]
131
+ # @return [Boolean] True if any file was deleted
132
+ def delete *filenames
133
+ result = false
134
+ filenames.each do |filename|
135
+ detach filename
136
+ source_map_hash.delete(filename)
137
+ result ||= workspace.remove(filename)
138
+ end
139
+ result
140
+ end
141
+
142
+ # Close a file in the library. Closing a file will make it unavailable for
143
+ # checkout although it may still exist in the workspace.
144
+ #
145
+ # @param filename [String]
146
+ # @return [void]
147
+ def close filename
148
+ return unless @current&.filename == filename
149
+
150
+ @current = nil
151
+ catalog unless workspace.has_file?(filename)
152
+ end
153
+
154
+ # Get completion suggestions at the specified file and location.
155
+ #
156
+ # @param filename [String] The file to analyze
157
+ # @param line [Integer] The zero-based line number
158
+ # @param column [Integer] The zero-based column number
159
+ # @return [SourceMap::Completion, nil]
160
+ # @todo Take a Location instead of filename/line/column
161
+ def completions_at filename, line, column
162
+ sync_catalog
163
+ position = Position.new(line, column)
164
+ cursor = Source::Cursor.new(read(filename), position)
165
+ mutex.synchronize { api_map.clip(cursor).complete }
166
+ rescue FileNotFoundError => e
167
+ handle_file_not_found filename, e
168
+ end
169
+
170
+ # Get definition suggestions for the expression at the specified file and
171
+ # location.
172
+ #
173
+ # @param filename [String] The file to analyze
174
+ # @param line [Integer] The zero-based line number
175
+ # @param column [Integer] The zero-based column number
176
+ # @return [Array<Solargraph::Pin::Base>, nil]
177
+ # @todo Take filename/position instead of filename/line/column
178
+ def definitions_at filename, line, column
179
+ sync_catalog
180
+ position = Position.new(line, column)
181
+ cursor = Source::Cursor.new(read(filename), position)
182
+ if cursor.comment?
183
+ source = read(filename)
184
+ offset = Solargraph::Position.to_offset(source.code, Solargraph::Position.new(line, column))
185
+ lft = source.code[0..offset-1].match(/\[[a-z0-9_:<, ]*?([a-z0-9_:]*)\z/i)
186
+ rgt = source.code[offset..-1].match(/^([a-z0-9_]*)(:[a-z0-9_:]*)?[\]>, ]/i)
187
+ if lft && rgt
188
+ tag = (lft[1] + rgt[1]).sub(/:+$/, '')
189
+ clip = mutex.synchronize { api_map.clip(cursor) }
190
+ clip.translate tag
191
+ else
192
+ []
193
+ end
194
+ else
195
+ mutex.synchronize do
196
+ clip = api_map.clip(cursor)
197
+ clip.define.map { |pin| pin.realize(api_map) }
198
+ end
199
+ end
200
+ rescue FileNotFoundError => e
201
+ handle_file_not_found(filename, e)
202
+ end
203
+
204
+ # Get type definition suggestions for the expression at the specified file and
205
+ # location.
206
+ #
207
+ # @param filename [String] The file to analyze
208
+ # @param line [Integer] The zero-based line number
209
+ # @param column [Integer] The zero-based column number
210
+ # @return [Array<Solargraph::Pin::Base>, nil]
211
+ # @todo Take filename/position instead of filename/line/column
212
+ def type_definitions_at filename, line, column
213
+ sync_catalog
214
+ position = Position.new(line, column)
215
+ cursor = Source::Cursor.new(read(filename), position)
216
+ mutex.synchronize { api_map.clip(cursor).types }
217
+ rescue FileNotFoundError => e
218
+ handle_file_not_found filename, e
219
+ end
220
+
221
+ # Get signature suggestions for the method at the specified file and
222
+ # location.
223
+ #
224
+ # @param filename [String] The file to analyze
225
+ # @param line [Integer] The zero-based line number
226
+ # @param column [Integer] The zero-based column number
227
+ # @return [Array<Solargraph::Pin::Base>]
228
+ # @todo Take filename/position instead of filename/line/column
229
+ def signatures_at filename, line, column
230
+ sync_catalog
231
+ position = Position.new(line, column)
232
+ cursor = Source::Cursor.new(read(filename), position)
233
+ mutex.synchronize { api_map.clip(cursor).signify }
234
+ end
235
+
236
+ # @param filename [String]
237
+ # @param line [Integer]
238
+ # @param column [Integer]
239
+ # @param strip [Boolean] Strip special characters from variable names
240
+ # @param only [Boolean] Search for references in the current file only
241
+ # @return [Array<Solargraph::Location>]
242
+ # @todo Take a Location instead of filename/line/column
243
+ def references_from filename, line, column, strip: false, only: false
244
+ sync_catalog
245
+ cursor = Source::Cursor.new(read(filename), [line, column])
246
+ clip = mutex.synchronize { api_map.clip(cursor) }
247
+ pin = clip.define.first
248
+ return [] unless pin
249
+ result = []
250
+ files = if only
251
+ [api_map.source_map(filename)]
252
+ else
253
+ (workspace.sources + (@current ? [@current] : []))
254
+ end
255
+ files.uniq(&:filename).each do |source|
256
+ found = source.references(pin.name)
257
+ found.select! do |loc|
258
+ referenced = definitions_at(loc.filename, loc.range.ending.line, loc.range.ending.character).first
259
+ referenced&.path == pin.path
260
+ end
261
+ if pin.path == 'Class#new'
262
+ caller = cursor.chain.base.infer(api_map, clip.send(:closure), clip.locals).first
263
+ if caller.defined?
264
+ found.select! do |loc|
265
+ clip = api_map.clip_at(loc.filename, loc.range.start)
266
+ other = clip.send(:cursor).chain.base.infer(api_map, clip.send(:closure), clip.locals).first
267
+ caller == other
268
+ end
269
+ else
270
+ found.clear
271
+ end
272
+ end
273
+ # HACK: for language clients that exclude special characters from the start of variable names
274
+ if strip && match = cursor.word.match(/^[^a-z0-9_]+/i)
275
+ found.map! do |loc|
276
+ Solargraph::Location.new(loc.filename, Solargraph::Range.from_to(loc.range.start.line, loc.range.start.column + match[0].length, loc.range.ending.line, loc.range.ending.column))
277
+ end
278
+ end
279
+ result.concat(found.sort do |a, b|
280
+ a.range.start.line <=> b.range.start.line
281
+ end)
282
+ end
283
+ result.uniq
284
+ end
285
+
286
+ # Get the pins at the specified location or nil if the pin does not exist.
287
+ #
288
+ # @param location [Location]
289
+ # @return [Array<Solargraph::Pin::Base>]
290
+ def locate_pins location
291
+ sync_catalog
292
+ mutex.synchronize { api_map.locate_pins(location).map { |pin| pin.realize(api_map) } }
293
+ end
294
+
295
+ # Match a require reference to a file.
296
+ #
297
+ # @param location [Location]
298
+ # @return [Location, nil]
299
+ def locate_ref location
300
+ map = source_map_hash[location.filename]
301
+ return if map.nil?
302
+ pin = map.requires.select { |p| p.location.range.contain?(location.range.start) }.first
303
+ return nil if pin.nil?
304
+ # @param full [String]
305
+ return_if_match = proc do |full|
306
+ if source_map_hash.key?(full)
307
+ return Location.new(full, Solargraph::Range.from_to(0, 0, 0, 0))
308
+ end
309
+ end
310
+ workspace.require_paths.each do |path|
311
+ full = File.join path, pin.name
312
+ return_if_match.(full)
313
+ return_if_match.(full << ".rb")
314
+ end
315
+ nil
316
+ rescue FileNotFoundError
317
+ nil
318
+ end
319
+
320
+ # Get an array of pins that match a path.
321
+ #
322
+ # @param path [String]
323
+ # @return [Enumerable<Solargraph::Pin::Base>]
324
+ def get_path_pins path
325
+ sync_catalog
326
+ mutex.synchronize { api_map.get_path_suggestions(path) }
327
+ end
328
+
329
+ # @param query [String]
330
+ # @return [Enumerable<YARD::CodeObjects::Base>]
331
+ # @return [Array(ApiMap, Enumerable<Pin::Base>)]
332
+ def document query
333
+ sync_catalog
334
+ mutex.synchronize { [api_map, api_map.get_path_pins(query)] }
335
+ end
336
+
337
+ # @param query [String]
338
+ # @return [Array<String>]
339
+ def search query
340
+ sync_catalog
341
+ mutex.synchronize { api_map.search query }
342
+ end
343
+
344
+ # Get an array of all symbols in the workspace that match the query.
345
+ #
346
+ # @param query [String]
347
+ # @return [Array<Pin::Base>]
348
+ def query_symbols query
349
+ sync_catalog
350
+ mutex.synchronize { api_map.query_symbols query }
351
+ end
352
+
353
+ # Get an array of document symbols.
354
+ #
355
+ # Document symbols are composed of namespace, method, and constant pins.
356
+ # The results of this query are appropriate for building the response to a
357
+ # textDocument/documentSymbol message in the language server protocol.
358
+ #
359
+ # @param filename [String]
360
+ # @return [Array<Solargraph::Pin::Base>]
361
+ def document_symbols filename
362
+ sync_catalog
363
+ mutex.synchronize { api_map.document_symbols(filename) }
364
+ end
365
+
366
+ # @param path [String]
367
+ # @return [Enumerable<Solargraph::Pin::Base>]
368
+ def path_pins path
369
+ sync_catalog
370
+ mutex.synchronize { api_map.get_path_suggestions(path) }
371
+ end
372
+
373
+ # @return [Array<SourceMap>]
374
+ def source_maps
375
+ source_map_hash.values
376
+ end
377
+
378
+ # Get the current text of a file in the library.
379
+ #
380
+ # @param filename [String]
381
+ # @return [String]
382
+ def read_text filename
383
+ source = read(filename)
384
+ source.code
385
+ end
386
+
387
+ # Get diagnostics about a file.
388
+ #
389
+ # @param filename [String]
390
+ # @return [Array<Hash>]
391
+ def diagnose filename
392
+ # @todo Only open files get diagnosed. Determine whether anything or
393
+ # everything in the workspace should get diagnosed, or if there should
394
+ # be an option to do so.
395
+ #
396
+ sync_catalog
397
+ return [] unless open?(filename)
398
+ result = []
399
+ source = read(filename)
400
+
401
+ # @type [Hash{Class<Solargraph::Diagnostics::Base> => Array<String>}]
402
+ repargs = {}
403
+ workspace.config.reporters.each do |line|
404
+ if line == 'all!'
405
+ Diagnostics.reporters.each do |reporter_name|
406
+ repargs[Diagnostics.reporter(reporter_name)] ||= []
407
+ end
408
+ else
409
+ args = line.split(':').map(&:strip)
410
+ name = args.shift
411
+ reporter = Diagnostics.reporter(name)
412
+ raise DiagnosticsError, "Diagnostics reporter #{name} does not exist" if reporter.nil?
413
+ repargs[reporter] ||= []
414
+ repargs[reporter].concat args
415
+ end
416
+ end
417
+ repargs.each_pair do |reporter, args|
418
+ result.concat reporter.new(*args.uniq).diagnose(source, api_map)
419
+ end
420
+ result
421
+ end
422
+
423
+ # Update the ApiMap from the library's workspace and open files.
424
+ #
425
+ # @return [void]
426
+ def catalog
427
+ @sync_count += 1
428
+ end
429
+
430
+ # @return [Bench]
431
+ def bench
432
+ Bench.new(
433
+ source_maps: source_map_hash.values,
434
+ workspace: workspace,
435
+ external_requires: external_requires,
436
+ live_map: @current ? source_map_hash[@current.filename] : nil
437
+ )
438
+ end
439
+
440
+ # Create a library from a directory.
441
+ #
442
+ # @param directory [String] The path to be used for the workspace
443
+ # @param name [String, nil]
444
+ # @return [Solargraph::Library]
445
+ def self.load directory = '', name = nil
446
+ Solargraph::Library.new(Solargraph::Workspace.new(directory), name)
447
+ end
448
+
449
+ # Try to merge a source into the library's workspace. If the workspace is
450
+ # not configured to include the source, it gets ignored.
451
+ #
452
+ # @param source [Source]
453
+ # @return [Boolean] True if the source was merged into the workspace.
454
+ def merge source
455
+ result = workspace.merge(source)
456
+ maybe_map source
457
+ result
458
+ end
459
+
460
+ # @return [Hash{String => SourceMap}]
461
+ def source_map_hash
462
+ @source_map_hash ||= {}
463
+ end
464
+
465
+ def mapped?
466
+ (workspace.filenames - source_map_hash.keys).empty?
467
+ end
468
+
469
+ # @return [SourceMap, Boolean]
470
+ def next_map
471
+ return false if mapped?
472
+ src = workspace.sources.find { |s| !source_map_hash.key?(s.filename) }
473
+ if src
474
+ Logging.logger.debug "Mapping #{src.filename}"
475
+ source_map_hash[src.filename] = Solargraph::SourceMap.map(src)
476
+ source_map_hash[src.filename]
477
+ else
478
+ false
479
+ end
480
+ end
481
+
482
+ # @return [self]
483
+ def map!
484
+ workspace.sources.each do |src|
485
+ source_map_hash[src.filename] = Solargraph::SourceMap.map(src)
486
+ find_external_requires source_map_hash[src.filename]
487
+ end
488
+ self
489
+ end
490
+
491
+ # @return [Array<Solargraph::Pin::Base>]
492
+ def pins
493
+ @pins ||= []
494
+ end
495
+
496
+ # @return [Set<String>]
497
+ def external_requires
498
+ @external_requires ||= source_map_external_require_hash.values.flatten.to_set
499
+ end
500
+
501
+ private
502
+
503
+ # @return [Hash{String => Array<String>}]
504
+ def source_map_external_require_hash
505
+ @source_map_external_require_hash ||= {}
506
+ end
507
+
508
+ # @param source_map [SourceMap]
509
+ # @return [void]
510
+ def find_external_requires source_map
511
+ # @type [Set<String>]
512
+ new_set = source_map.requires.map(&:name).to_set
513
+ # return if new_set == source_map_external_require_hash[source_map.filename]
514
+ _filenames = nil
515
+ filenames = ->{ _filenames ||= workspace.filenames.to_set }
516
+ source_map_external_require_hash[source_map.filename] = new_set.reject do |path|
517
+ workspace.require_paths.any? do |base|
518
+ full = File.join(base, path)
519
+ filenames[].include?(full) or filenames[].include?(full << ".rb")
520
+ end
521
+ end
522
+ @external_requires = nil
523
+ end
524
+
525
+ # @return [Thread::Mutex]
526
+ def mutex
527
+ @mutex ||= Mutex.new
528
+ end
529
+
530
+ # @return [ApiMap]
531
+ def api_map
532
+ @api_map ||= Solargraph::ApiMap.new
533
+ end
534
+
535
+ # Get the source for an open file or create a new source if the file
536
+ # exists on disk. Sources created from disk are not added to the open
537
+ # workspace files, i.e., the version on disk remains the authoritative
538
+ # version.
539
+ #
540
+ # @raise [FileNotFoundError] if the file does not exist
541
+ # @param filename [String]
542
+ # @return [Solargraph::Source]
543
+ def read filename
544
+ return @current if @current && @current.filename == filename
545
+ raise FileNotFoundError, "File not found: #{filename}" unless workspace.has_file?(filename)
546
+ workspace.source(filename)
547
+ end
548
+
549
+ # @param filename [String]
550
+ # @param error [FileNotFoundError]
551
+ # @return [nil]
552
+ def handle_file_not_found filename, error
553
+ if workspace.source(filename)
554
+ Solargraph.logger.debug "#{filename} is not cataloged in the ApiMap"
555
+ nil
556
+ else
557
+ raise error
558
+ end
559
+ end
560
+
561
+ # @param source [Source, nil]
562
+ # @return [void]
563
+ def maybe_map source
564
+ return unless source
565
+ return unless @current == source || workspace.has_file?(source.filename)
566
+ if source_map_hash.key?(source.filename)
567
+ new_map = Solargraph::SourceMap.map(source)
568
+ source_map_hash[source.filename] = new_map
569
+ else
570
+ source_map_hash[source.filename] = Solargraph::SourceMap.map(source)
571
+ end
572
+ end
573
+
574
+ # @return [Set<Gem::Specification>]
575
+ def cache_errors
576
+ @cache_errors ||= Set.new
577
+ end
578
+
579
+ # @return [void]
580
+ def cache_next_gemspec
581
+ return if @cache_progress
582
+
583
+ spec = cacheable_specs.first
584
+ return end_cache_progress unless spec
585
+
586
+ pending = api_map.uncached_gemspecs.length - cache_errors.length - 1
587
+
588
+ if Yardoc.processing?(spec)
589
+ logger.info "Enqueuing cache of #{spec.name} #{spec.version} (already being processed)"
590
+ queued_gemspec_cache.push(spec)
591
+ return if pending - queued_gemspec_cache.length < 1
592
+
593
+ catalog
594
+ sync_catalog
595
+ else
596
+ logger.info "Caching #{spec.name} #{spec.version}"
597
+ Thread.new do
598
+ report_cache_progress spec.name, pending
599
+ _o, e, s = Open3.capture3(workspace.command_path, 'cache', spec.name, spec.version.to_s)
600
+ if s.success?
601
+ logger.info "Cached #{spec.name} #{spec.version}"
602
+ else
603
+ cache_errors.add spec
604
+ logger.warn "Error caching gemspec #{spec.name} #{spec.version}"
605
+ logger.warn e
606
+ end
607
+ end_cache_progress
608
+ catalog
609
+ sync_catalog
610
+ end
611
+ end
612
+ end
613
+
614
+ # @return [Array<Gem::Specification>]
615
+ def cacheable_specs
616
+ cacheable = api_map.uncached_yard_gemspecs +
617
+ api_map.uncached_rbs_collection_gemspecs -
618
+ queued_gemspec_cache -
619
+ cache_errors.to_a
620
+ return cacheable unless cacheable.empty?
621
+
622
+ queued_gemspec_cache
623
+ end
624
+
625
+ # @return [Array<Gem::Specification>]
626
+ def queued_gemspec_cache
627
+ @queued_gemspec_cache ||= []
628
+ end
629
+
630
+ # @param gem_name [String]
631
+ # @param pending [Integer]
632
+ # @return [void]
633
+ def report_cache_progress gem_name, pending
634
+ @total ||= pending
635
+ @total = pending if pending > @total
636
+ finished = @total - pending
637
+ pct = if @total.zero?
638
+ 0
639
+ else
640
+ ((finished.to_f / @total.to_f) * 100).to_i
641
+ end
642
+ message = "#{gem_name}#{pending > 0 ? " (+#{pending})" : ''}"
643
+ # "
644
+ if @cache_progress
645
+ @cache_progress.report(message, pct)
646
+ else
647
+ @cache_progress = LanguageServer::Progress.new('Caching gem')
648
+ # If we don't send both a begin and a report, the progress notification
649
+ # might get stuck in the status bar forever
650
+ @cache_progress.begin(message, pct)
651
+ changed
652
+ notify_observers @cache_progress
653
+ @cache_progress.report(message, pct)
654
+ end
655
+ changed
656
+ notify_observers @cache_progress
657
+ end
658
+
659
+ # @return [void]
660
+ def end_cache_progress
661
+ changed if @cache_progress&.finish('done')
662
+ notify_observers @cache_progress
663
+ @cache_progress = nil
664
+ @total = nil
665
+ end
666
+
667
+ # @return [void]
668
+ def sync_catalog
669
+ return if @sync_count == 0
670
+
671
+ mutex.synchronize do
672
+ logger.info "Cataloging #{workspace.directory.empty? ? 'generic workspace' : workspace.directory}"
673
+ source_map_hash.values.each { |map| find_external_requires(map) }
674
+ api_map.catalog bench
675
+ logger.info "Catalog complete (#{api_map.source_maps.length} files, #{api_map.pins.length} pins)"
676
+ logger.info "#{api_map.uncached_yard_gemspecs.length} uncached YARD gemspecs"
677
+ logger.info "#{api_map.uncached_rbs_collection_gemspecs.length} uncached RBS collection gemspecs"
678
+ cache_next_gemspec
679
+ @sync_count = 0
680
+ end
681
+ end
682
+ end
683
+ end