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