solargraph 0.51.2 → 0.53.0

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 (162) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/plugins.yml +40 -0
  3. data/.github/workflows/rspec.yml +1 -3
  4. data/.github/workflows/typecheck.yml +34 -0
  5. data/.yardopts +2 -2
  6. data/CHANGELOG.md +55 -5
  7. data/README.md +13 -16
  8. data/SPONSORS.md +1 -7
  9. data/lib/solargraph/api_map/cache.rb +60 -20
  10. data/lib/solargraph/api_map/store.rb +47 -11
  11. data/lib/solargraph/api_map.rb +161 -95
  12. data/lib/solargraph/bench.rb +2 -2
  13. data/lib/solargraph/cache.rb +29 -5
  14. data/lib/solargraph/complex_type/type_methods.rb +54 -9
  15. data/lib/solargraph/complex_type/unique_type.rb +155 -58
  16. data/lib/solargraph/complex_type.rb +73 -16
  17. data/lib/solargraph/convention.rb +0 -1
  18. data/lib/solargraph/converters/dd.rb +5 -0
  19. data/lib/solargraph/converters/dl.rb +3 -0
  20. data/lib/solargraph/converters/dt.rb +3 -0
  21. data/lib/solargraph/diagnostics/rubocop.rb +8 -7
  22. data/lib/solargraph/diagnostics/rubocop_helpers.rb +1 -0
  23. data/lib/solargraph/diagnostics/type_check.rb +1 -0
  24. data/lib/solargraph/diagnostics.rb +2 -2
  25. data/lib/solargraph/doc_map.rb +146 -0
  26. data/lib/solargraph/gem_pins.rb +64 -0
  27. data/lib/solargraph/language_server/host/cataloger.rb +1 -0
  28. data/lib/solargraph/language_server/host/diagnoser.rb +2 -2
  29. data/lib/solargraph/language_server/host/dispatch.rb +10 -4
  30. data/lib/solargraph/language_server/host/message_worker.rb +4 -0
  31. data/lib/solargraph/language_server/host/sources.rb +7 -4
  32. data/lib/solargraph/language_server/host.rb +15 -6
  33. data/lib/solargraph/language_server/message/completion_item/resolve.rb +3 -1
  34. data/lib/solargraph/language_server/message/extended/check_gem_version.rb +13 -1
  35. data/lib/solargraph/language_server/message/initialize.rb +5 -2
  36. data/lib/solargraph/language_server/message/text_document/hover.rb +2 -0
  37. data/lib/solargraph/language_server/message/text_document.rb +0 -1
  38. data/lib/solargraph/language_server/message/workspace/did_change_configuration.rb +5 -0
  39. data/lib/solargraph/language_server/transport/adapter.rb +16 -1
  40. data/lib/solargraph/language_server/transport/data_reader.rb +2 -0
  41. data/lib/solargraph/library.rb +70 -16
  42. data/lib/solargraph/location.rb +1 -0
  43. data/lib/solargraph/parser/comment_ripper.rb +4 -0
  44. data/lib/solargraph/parser/node_methods.rb +47 -7
  45. data/lib/solargraph/parser/node_processor/base.rb +9 -0
  46. data/lib/solargraph/parser/{legacy → parser_gem}/class_methods.rb +31 -5
  47. data/lib/solargraph/parser/{legacy → parser_gem}/flawed_builder.rb +3 -1
  48. data/lib/solargraph/parser/{legacy → parser_gem}/node_chainer.rb +49 -36
  49. data/lib/solargraph/parser/parser_gem/node_methods.rb +499 -0
  50. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/alias_node.rb +1 -1
  51. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/args_node.rb +4 -1
  52. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/begin_node.rb +1 -1
  53. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/block_node.rb +3 -2
  54. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/casgn_node.rb +2 -2
  55. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/cvasgn_node.rb +1 -1
  56. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/def_node.rb +1 -1
  57. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/defs_node.rb +2 -2
  58. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/gvasgn_node.rb +1 -1
  59. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/ivasgn_node.rb +2 -2
  60. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/lvasgn_node.rb +2 -2
  61. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/namespace_node.rb +2 -2
  62. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/orasgn_node.rb +1 -1
  63. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/resbody_node.rb +3 -3
  64. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/sclass_node.rb +1 -1
  65. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/send_node.rb +2 -2
  66. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/sym_node.rb +1 -1
  67. data/lib/solargraph/parser/parser_gem/node_processors.rb +54 -0
  68. data/lib/solargraph/parser/parser_gem.rb +12 -0
  69. data/lib/solargraph/parser/region.rb +1 -1
  70. data/lib/solargraph/parser/snippet.rb +2 -0
  71. data/lib/solargraph/parser.rb +8 -9
  72. data/lib/solargraph/pin/base.rb +64 -9
  73. data/lib/solargraph/pin/base_variable.rb +6 -2
  74. data/lib/solargraph/pin/block.rb +13 -8
  75. data/lib/solargraph/pin/closure.rb +17 -2
  76. data/lib/solargraph/pin/common.rb +7 -3
  77. data/lib/solargraph/pin/conversions.rb +33 -3
  78. data/lib/solargraph/pin/delegated_method.rb +1 -1
  79. data/lib/solargraph/pin/documenting.rb +25 -34
  80. data/lib/solargraph/pin/instance_variable.rb +4 -0
  81. data/lib/solargraph/pin/local_variable.rb +13 -1
  82. data/lib/solargraph/pin/method.rb +169 -18
  83. data/lib/solargraph/pin/namespace.rb +18 -5
  84. data/lib/solargraph/pin/parameter.rb +44 -14
  85. data/lib/solargraph/pin/reference/override.rb +2 -2
  86. data/lib/solargraph/pin/reference.rb +8 -0
  87. data/lib/solargraph/pin/search.rb +3 -3
  88. data/lib/solargraph/pin/signature.rb +123 -3
  89. data/lib/solargraph/pin.rb +0 -1
  90. data/lib/solargraph/range.rb +2 -2
  91. data/lib/solargraph/rbs_map/conversions.rb +287 -45
  92. data/lib/solargraph/rbs_map/core_fills.rb +6 -29
  93. data/lib/solargraph/rbs_map/core_map.rb +2 -1
  94. data/lib/solargraph/rbs_map/core_signs.rb +2 -0
  95. data/lib/solargraph/rbs_map/stdlib_map.rb +2 -8
  96. data/lib/solargraph/rbs_map.rb +20 -11
  97. data/lib/solargraph/shell.rb +62 -59
  98. data/lib/solargraph/source/chain/array.rb +32 -0
  99. data/lib/solargraph/source/chain/block_symbol.rb +13 -0
  100. data/lib/solargraph/source/chain/call.rb +99 -46
  101. data/lib/solargraph/source/chain/constant.rb +15 -1
  102. data/lib/solargraph/source/chain/if.rb +23 -0
  103. data/lib/solargraph/source/chain/link.rb +8 -2
  104. data/lib/solargraph/source/chain/or.rb +1 -1
  105. data/lib/solargraph/source/chain/z_super.rb +3 -3
  106. data/lib/solargraph/source/chain.rb +29 -14
  107. data/lib/solargraph/source/change.rb +3 -0
  108. data/lib/solargraph/source/cursor.rb +2 -0
  109. data/lib/solargraph/source/source_chainer.rb +8 -5
  110. data/lib/solargraph/source.rb +18 -19
  111. data/lib/solargraph/source_map/clip.rb +11 -23
  112. data/lib/solargraph/source_map/mapper.rb +12 -1
  113. data/lib/solargraph/source_map.rb +15 -5
  114. data/lib/solargraph/type_checker/checks.rb +10 -2
  115. data/lib/solargraph/type_checker.rb +92 -26
  116. data/lib/solargraph/version.rb +1 -1
  117. data/lib/solargraph/workspace/config.rb +8 -6
  118. data/lib/solargraph/workspace.rb +3 -2
  119. data/lib/solargraph/yard_map/cache.rb +6 -0
  120. data/lib/solargraph/yard_map/helpers.rb +1 -1
  121. data/lib/solargraph/yard_map/mapper/to_method.rb +11 -1
  122. data/lib/solargraph/yard_map/mapper.rb +1 -1
  123. data/lib/solargraph/yard_map/to_method.rb +11 -4
  124. data/lib/solargraph/yard_map.rb +1 -292
  125. data/lib/solargraph/yard_tags.rb +20 -0
  126. data/lib/solargraph/yardoc.rb +52 -0
  127. data/lib/solargraph.rb +6 -4
  128. data/solargraph.gemspec +3 -2
  129. metadata +51 -58
  130. data/lib/solargraph/api_map/bundler_methods.rb +0 -22
  131. data/lib/solargraph/documentor.rb +0 -76
  132. data/lib/solargraph/parser/legacy/node_methods.rb +0 -325
  133. data/lib/solargraph/parser/legacy/node_processors/alias_node.rb +0 -23
  134. data/lib/solargraph/parser/legacy/node_processors/begin_node.rb +0 -15
  135. data/lib/solargraph/parser/legacy/node_processors/sym_node.rb +0 -18
  136. data/lib/solargraph/parser/legacy/node_processors.rb +0 -55
  137. data/lib/solargraph/parser/legacy.rb +0 -12
  138. data/lib/solargraph/parser/rubyvm/class_methods.rb +0 -153
  139. data/lib/solargraph/parser/rubyvm/node_chainer.rb +0 -160
  140. data/lib/solargraph/parser/rubyvm/node_methods.rb +0 -317
  141. data/lib/solargraph/parser/rubyvm/node_processors/args_node.rb +0 -85
  142. data/lib/solargraph/parser/rubyvm/node_processors/block_node.rb +0 -42
  143. data/lib/solargraph/parser/rubyvm/node_processors/casgn_node.rb +0 -33
  144. data/lib/solargraph/parser/rubyvm/node_processors/cvasgn_node.rb +0 -23
  145. data/lib/solargraph/parser/rubyvm/node_processors/def_node.rb +0 -75
  146. data/lib/solargraph/parser/rubyvm/node_processors/defs_node.rb +0 -68
  147. data/lib/solargraph/parser/rubyvm/node_processors/gvasgn_node.rb +0 -23
  148. data/lib/solargraph/parser/rubyvm/node_processors/ivasgn_node.rb +0 -38
  149. data/lib/solargraph/parser/rubyvm/node_processors/kw_arg_node.rb +0 -39
  150. data/lib/solargraph/parser/rubyvm/node_processors/lit_node.rb +0 -20
  151. data/lib/solargraph/parser/rubyvm/node_processors/lvasgn_node.rb +0 -27
  152. data/lib/solargraph/parser/rubyvm/node_processors/namespace_node.rb +0 -39
  153. data/lib/solargraph/parser/rubyvm/node_processors/opt_arg_node.rb +0 -26
  154. data/lib/solargraph/parser/rubyvm/node_processors/orasgn_node.rb +0 -15
  155. data/lib/solargraph/parser/rubyvm/node_processors/resbody_node.rb +0 -51
  156. data/lib/solargraph/parser/rubyvm/node_processors/sclass_node.rb +0 -32
  157. data/lib/solargraph/parser/rubyvm/node_processors/scope_node.rb +0 -15
  158. data/lib/solargraph/parser/rubyvm/node_processors/send_node.rb +0 -279
  159. data/lib/solargraph/parser/rubyvm/node_processors.rb +0 -64
  160. data/lib/solargraph/parser/rubyvm/node_wrapper.rb +0 -47
  161. data/lib/solargraph/parser/rubyvm.rb +0 -40
  162. data/lib/yard-solargraph.rb +0 -33
@@ -1,10 +1,8 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'rubygems'
4
- require 'set'
5
3
  require 'pathname'
6
4
  require 'yard'
7
- require 'yard-solargraph'
5
+ require 'solargraph/yard_tags'
8
6
 
9
7
  module Solargraph
10
8
  # An aggregate provider for information about workspaces, sources, gems, and
@@ -14,9 +12,6 @@ module Solargraph
14
12
  autoload :Cache, 'solargraph/api_map/cache'
15
13
  autoload :SourceToYard, 'solargraph/api_map/source_to_yard'
16
14
  autoload :Store, 'solargraph/api_map/store'
17
- autoload :BundlerMethods, 'solargraph/api_map/bundler_methods'
18
-
19
- include SourceToYard
20
15
 
21
16
  # @return [Array<String>]
22
17
  attr_reader :unresolved_requires
@@ -59,34 +54,33 @@ module Solargraph
59
54
  # Catalog a bench.
60
55
  #
61
56
  # @param bench [Bench]
57
+ # @return [self]
62
58
  def catalog bench
63
59
  implicit.clear
64
60
  @cache.clear
65
61
  @source_map_hash = bench.source_maps.map { |s| [s.filename, s] }.to_h
66
62
  pins = bench.source_maps.map(&:pins).flatten
67
- external_requires = bench.external_requires
68
63
  source_map_hash.each_value do |map|
69
64
  implicit.merge map.environ
70
65
  end
71
- external_requires.merge implicit.requires
72
- external_requires.merge bench.workspace.config.required
73
- @rbs_maps = external_requires.map { |r| load_rbs_map(r) }
74
- unresolved_requires = @rbs_maps.reject(&:resolved?).map(&:library)
75
- yard_map.change(unresolved_requires, bench.workspace.directory, bench.workspace.source_gems)
76
- @store = Store.new(@@core_map.pins + @rbs_maps.flat_map(&:pins) + yard_map.pins + implicit.pins + pins)
77
- @unresolved_requires = yard_map.unresolved_requires
78
- @missing_docs = yard_map.missing_docs
66
+ unresolved_requires = (bench.external_requires + implicit.requires + bench.workspace.config.required).uniq
67
+ @doc_map = DocMap.new(unresolved_requires, []) # @todo Implement gem dependencies
68
+ @store = Store.new(@@core_map.pins + @doc_map.pins + implicit.pins + pins)
69
+ @unresolved_requires = @doc_map.unresolved_requires
70
+ @missing_docs = [] # @todo Implement missing docs
79
71
  @rebindable_method_names = nil
80
72
  store.block_pins.each { |blk| blk.rebind(self) }
81
73
  self
82
74
  end
83
75
 
84
- def core_pins
85
- @@core_map.pins
76
+ # @return [::Array<Gem::Specification>]
77
+ def uncached_gemspecs
78
+ @doc_map&.uncached_gemspecs || []
86
79
  end
87
80
 
88
- def yard_map
89
- @yard_map ||= YardMap.new
81
+ # @return [Array<Pin::Base>]
82
+ def core_pins
83
+ @@core_map.pins
90
84
  end
91
85
 
92
86
  # @param name [String]
@@ -95,6 +89,7 @@ module Solargraph
95
89
  store.named_macros[name]
96
90
  end
97
91
 
92
+ # @return [Set<String>]
98
93
  def required
99
94
  @required ||= Set.new
100
95
  end
@@ -120,7 +115,7 @@ module Solargraph
120
115
  # @return [SourceMap::Clip]
121
116
  def clip_at filename, position
122
117
  position = Position.normalize(position)
123
- SourceMap::Clip.new(self, cursor_at(filename, position))
118
+ clip(cursor_at(filename, position))
124
119
  end
125
120
 
126
121
  # Create an ApiMap with a workspace in the specified directory.
@@ -137,14 +132,31 @@ module Solargraph
137
132
  api_map
138
133
  end
139
134
 
135
+ # Create an ApiMap with a workspace in the specified directory and cache
136
+ # any missing gems.
137
+ #
138
+ # @param directory [String]
139
+ # @return [ApiMap]
140
+ def self.load_with_cache directory
141
+ api_map = load(directory)
142
+ return api_map if api_map.uncached_gemspecs.empty?
143
+
144
+ api_map.uncached_gemspecs.each do |gemspec|
145
+ Solargraph.logger.info "Caching #{gemspec.name} #{gemspec.version}..."
146
+ pins = GemPins.build(gemspec)
147
+ Solargraph::Cache.save('gems', "#{gemspec.name}-#{gemspec.version}.ser", pins)
148
+ end
149
+ load(directory)
150
+ end
151
+
140
152
  # @return [Array<Solargraph::Pin::Base>]
141
153
  def pins
142
154
  store.pins
143
155
  end
144
156
 
157
+ # @return [Set<String>]
145
158
  def rebindable_method_names
146
159
  @rebindable_method_names ||= begin
147
- # result = yard_map.rebindable_method_names
148
160
  result = ['instance_eval', 'instance_exec', 'class_eval', 'class_exec', 'module_eval', 'module_exec', 'define_method'].to_set
149
161
  source_maps.each do |map|
150
162
  result.merge map.rebindable_method_names
@@ -199,22 +211,55 @@ module Solargraph
199
211
  result
200
212
  end
201
213
 
202
- # Get a fully qualified namespace name. This method will start the search
203
- # in the specified context until it finds a match for the name.
214
+ # @param namespace [String]
215
+ # @param context [String]
216
+ # @return [Array<Pin::Namespace>]
217
+ def get_namespace_pins namespace, context
218
+ store.fqns_pins(qualify(namespace, context))
219
+ end
220
+
221
+ # Determine fully qualified tag for a given tag used inside the
222
+ # definition of another tag ("context"). This method will start
223
+ # the search in the specified context until it finds a match for
224
+ # the tag.
225
+ #
226
+ # Does not recurse into qualifying the type parameters, but
227
+ # returns any which were passed in unchanged.
228
+ #
229
+ # @param tag [String, nil] The namespace to
230
+ # match, complete with generic parameters set to appropriate
231
+ # values if available
232
+ # @param context_tag [String] The context in which the tag was
233
+ # referenced; start from here to resolve the name
234
+ # @return [String, nil] fully qualified tag
235
+ def qualify tag, context_tag = ''
236
+ return tag if ['self', nil].include?(tag)
237
+ context_type = ComplexType.parse(context_tag)
238
+ type = ComplexType.parse(tag)
239
+ fqns = qualify_namespace(type.rooted_namespace, context_type.rooted_namespace)
240
+ return nil if fqns.nil?
241
+ fqns + type.substring
242
+ end
243
+
244
+ # Determine fully qualified namespace for a given namespace used
245
+ # inside the definition of another tag ("context"). This method
246
+ # will start the search in the specified context until it finds a
247
+ # match for the namespace.
204
248
  #
205
- # @param namespace [String, nil] The namespace to match
206
- # @param context [String] The context to search
207
- # @return [String, nil]
208
- def qualify namespace, context = ''
209
- return namespace if ['self', nil].include?(namespace)
210
- cached = cache.get_qualified_namespace(namespace, context)
249
+ # @param namespace [String, nil] The namespace to
250
+ # match
251
+ # @param context_namespace [String] The context namespace in which the
252
+ # tag was referenced; start from here to resolve the name
253
+ # @return [String, nil] fully qualified namespace
254
+ def qualify_namespace(namespace, context_namespace = '')
255
+ cached = cache.get_qualified_namespace(namespace, context_namespace)
211
256
  return cached.clone unless cached.nil?
212
257
  result = if namespace.start_with?('::')
213
258
  inner_qualify(namespace[2..-1], '', Set.new)
214
259
  else
215
- inner_qualify(namespace, context, Set.new)
260
+ inner_qualify(namespace, context_namespace, Set.new)
216
261
  end
217
- cache.set_qualified_namespace(namespace, context, result)
262
+ cache.set_qualified_namespace(namespace, context_namespace, result)
218
263
  result
219
264
  end
220
265
 
@@ -240,49 +285,50 @@ module Solargraph
240
285
  # Get an array of class variable pins for a namespace.
241
286
  #
242
287
  # @param namespace [String] A fully qualified namespace
243
- # @return [Array<Solargraph::Pin::ClassVariable>]
288
+ # @return [Enumerable<Solargraph::Pin::ClassVariable>]
244
289
  def get_class_variable_pins(namespace)
245
290
  prefer_non_nil_variables(store.get_class_variables(namespace))
246
291
  end
247
292
 
248
- # @return [Array<Solargraph::Pin::Base>]
293
+ # @return [Enumerable<Solargraph::Pin::Base>]
249
294
  def get_symbols
250
295
  store.get_symbols
251
296
  end
252
297
 
253
- # @return [Array<Solargraph::Pin::GlobalVariable>]
298
+ # @return [Enumerable<Solargraph::Pin::GlobalVariable>]
254
299
  def get_global_variable_pins
255
300
  store.pins_by_class(Pin::GlobalVariable)
256
301
  end
257
302
 
258
303
  # Get an array of methods available in a particular context.
259
304
  #
260
- # @param fqns [String] The fully qualified namespace to search for methods
305
+ # @param rooted_tag [String] The fully qualified namespace to search for methods
261
306
  # @param scope [Symbol] :class or :instance
262
307
  # @param visibility [Array<Symbol>] :public, :protected, and/or :private
263
308
  # @param deep [Boolean] True to include superclasses, mixins, etc.
264
309
  # @return [Array<Solargraph::Pin::Method>]
265
- def get_methods fqns, scope: :instance, visibility: [:public], deep: true
266
- cached = cache.get_methods(fqns, scope, visibility, deep)
310
+ def get_methods rooted_tag, scope: :instance, visibility: [:public], deep: true
311
+ cached = cache.get_methods(rooted_tag, scope, visibility, deep)
267
312
  return cached.clone unless cached.nil?
268
313
  result = []
269
314
  skip = Set.new
270
- if fqns == ''
315
+ if rooted_tag == ''
271
316
  # @todo Implement domains
272
317
  implicit.domains.each do |domain|
273
318
  type = ComplexType.try_parse(domain)
274
319
  next if type.undefined?
275
320
  result.concat inner_get_methods(type.name, type.scope, visibility, deep, skip)
276
321
  end
277
- result.concat inner_get_methods(fqns, :class, visibility, deep, skip)
278
- result.concat inner_get_methods(fqns, :instance, visibility, deep, skip)
322
+ result.concat inner_get_methods(rooted_tag, :class, visibility, deep, skip)
323
+ result.concat inner_get_methods(rooted_tag, :instance, visibility, deep, skip)
279
324
  result.concat inner_get_methods('Kernel', :instance, visibility, deep, skip)
280
325
  else
281
- result.concat inner_get_methods(fqns, scope, visibility, deep, skip)
326
+ result.concat inner_get_methods(rooted_tag, scope, visibility, deep, skip)
282
327
  result.concat inner_get_methods('Kernel', :instance, [:public], deep, skip) if visibility.include?(:private)
328
+ result.concat inner_get_methods('Module', scope, visibility, deep, skip)
283
329
  end
284
330
  resolved = resolve_method_aliases(result, visibility)
285
- cache.set_methods(fqns, scope, visibility, deep, resolved)
331
+ cache.set_methods(rooted_tag, scope, visibility, deep, resolved)
286
332
  resolved
287
333
  end
288
334
 
@@ -320,27 +366,27 @@ module Solargraph
320
366
  visibility.push :protected
321
367
  visibility.push :private if internal
322
368
  end
323
- result.merge get_methods(type.namespace, scope: type.scope, visibility: visibility)
369
+ result.merge get_methods(type.tag, scope: type.scope, visibility: visibility)
324
370
  end
325
371
  end
326
372
  end
327
373
  result.to_a
328
374
  end
329
375
 
330
- # Get a stack of method pins for a method name in a namespace. The order
331
- # of the pins corresponds to the ancestry chain, with highest precedence
332
- # first.
376
+ # Get a stack of method pins for a method name in a potentially
377
+ # parameterized namespace. The order of the pins corresponds to
378
+ # the ancestry chain, with highest precedence first.
333
379
  #
334
380
  # @example
335
381
  # api_map.get_method_stack('Subclass', 'method_name')
336
382
  # #=> [ <Subclass#method_name pin>, <Superclass#method_name pin> ]
337
383
  #
338
- # @param fqns [String]
339
- # @param name [String]
384
+ # @param rooted_tag [String] Parameterized namespace, fully qualified
385
+ # @param name [String] Method name to look up
340
386
  # @param scope [Symbol] :instance or :class
341
387
  # @return [Array<Solargraph::Pin::Method>]
342
- def get_method_stack fqns, name, scope: :instance
343
- get_methods(fqns, scope: scope, visibility: [:private, :protected, :public]).select { |p| p.name == name }
388
+ def get_method_stack rooted_tag, name, scope: :instance
389
+ get_methods(rooted_tag, scope: scope, visibility: [:private, :protected, :public]).select { |p| p.name == name }
344
390
  end
345
391
 
346
392
  # Get an array of all suggestions that match the specified path.
@@ -348,7 +394,7 @@ module Solargraph
348
394
  # @deprecated Use #get_path_pins instead.
349
395
  #
350
396
  # @param path [String] The path to find
351
- # @return [Array<Solargraph::Pin::Base>]
397
+ # @return [Enumerable<Solargraph::Pin::Base>]
352
398
  def get_path_suggestions path
353
399
  return [] if path.nil?
354
400
  resolve_method_aliases store.get_path_pins(path)
@@ -357,7 +403,7 @@ module Solargraph
357
403
  # Get an array of pins that match the specified path.
358
404
  #
359
405
  # @param path [String]
360
- # @return [Array<Pin::Base>]
406
+ # @return [Enumerable<Pin::Base>]
361
407
  def get_path_pins path
362
408
  get_path_suggestions(path)
363
409
  end
@@ -370,15 +416,9 @@ module Solargraph
370
416
  # @param query [String] The text to match
371
417
  # @return [Array<String>]
372
418
  def search query
373
- rake_yard(store)
374
- found = []
375
- code_object_paths.each do |k|
376
- if (found.empty? || (query.include?('.') || query.include?('#')) || !(k.include?('.') || k.include?('#'))) &&
377
- k.downcase.include?(query.downcase)
378
- found.push k
379
- end
380
- end
381
- found
419
+ pins.map(&:path)
420
+ .compact
421
+ .select { |path| path.downcase.include?(query.downcase) }
382
422
  end
383
423
 
384
424
  # Get YARD documentation for the specified path.
@@ -386,13 +426,13 @@ module Solargraph
386
426
  # @example
387
427
  # api_map.document('String#split')
388
428
  #
429
+ # @todo This method is likely superfluous. Calling get_path_pins directly
430
+ # should be sufficient.
431
+ #
389
432
  # @param path [String] The path to find
390
- # @return [Array<YARD::CodeObjects::Base>]
433
+ # @return [Enumerable<Pin::Base>]
391
434
  def document path
392
- rake_yard(store)
393
- docs = []
394
- docs.push code_object_at(path) unless code_object_at(path).nil?
395
- docs
435
+ get_path_pins(path)
396
436
  end
397
437
 
398
438
  # Get an array of all symbols in the workspace that match the query.
@@ -418,6 +458,10 @@ module Solargraph
418
458
  # @return [SourceMap::Clip]
419
459
  def clip cursor
420
460
  raise FileNotFoundError, "ApiMap did not catalog #{cursor.filename}" unless source_map_hash.key?(cursor.filename)
461
+
462
+ # @todo Clip caches are disabled pending resolution of a stale cache bug
463
+ # cache.get_clip(cursor) ||
464
+ # SourceMap::Clip.new(self, cursor).tap { |clip| cache.set_clip(cursor, clip) }
421
465
  SourceMap::Clip.new(self, cursor)
422
466
  end
423
467
 
@@ -469,13 +513,15 @@ module Solargraph
469
513
  false
470
514
  end
471
515
 
472
- # Check if the host class includes the specified module.
516
+ # Check if the host class includes the specified module, ignoring
517
+ # type parameters used.
518
+ #
519
+ # @param host_ns [String] The class namesapce (no type parameters)
520
+ # @param module_ns [String] The module namespace (no type parameters)
473
521
  #
474
- # @param host [String] The class
475
- # @param mod [String] The module
476
522
  # @return [Boolean]
477
- def type_include?(host, mod)
478
- store.get_includes(host).include?(mod)
523
+ def type_include?(host_ns, module_ns)
524
+ store.get_includes(host_ns).map { |inc_tag| ComplexType.parse(inc_tag).name }.include?(module_ns)
479
525
  end
480
526
 
481
527
  private
@@ -485,14 +531,6 @@ module Solargraph
485
531
  # @return [Hash{String => SourceMap}]
486
532
  attr_reader :source_map_hash
487
533
 
488
- # @param library [String]
489
- # @return [RbsMap]
490
- def load_rbs_map library
491
- # map = RbsMap.load(library)
492
- # return map if map.resolved?
493
- RbsMap::StdlibMap.load(library)
494
- end
495
-
496
534
  # @return [ApiMap::Store]
497
535
  def store
498
536
  @store ||= Store.new
@@ -501,15 +539,19 @@ module Solargraph
501
539
  # @return [Solargraph::ApiMap::Cache]
502
540
  attr_reader :cache
503
541
 
504
- # @param fqns [String] A fully qualified namespace
542
+ # @param rooted_tag [String] A fully qualified namespace, with
543
+ # generic parameter values if applicable
505
544
  # @param scope [Symbol] :class or :instance
506
545
  # @param visibility [Array<Symbol>] :public, :protected, and/or :private
507
546
  # @param deep [Boolean]
508
547
  # @param skip [Set<String>]
509
548
  # @param no_core [Boolean] Skip core classes if true
510
549
  # @return [Array<Pin::Base>]
511
- def inner_get_methods fqns, scope, visibility, deep, skip, no_core = false
512
- return [] if no_core && fqns =~ /^(Object|BasicObject|Class|Module|Kernel)$/
550
+ def inner_get_methods rooted_tag, scope, visibility, deep, skip, no_core = false
551
+ rooted_type = ComplexType.parse(rooted_tag)
552
+ fqns = rooted_type.namespace
553
+ fqns_generic_params = rooted_type.all_params
554
+ return [] if no_core && fqns =~ /^(Object|BasicObject|Class|Module)$/
513
555
  reqstr = "#{fqns}|#{scope}|#{visibility.sort}|#{deep}"
514
556
  return [] if skip.include?(reqstr)
515
557
  skip.add reqstr
@@ -520,12 +562,33 @@ module Solargraph
520
562
  result.concat inner_get_methods(fqim, scope, visibility, deep, skip, true) unless fqim.nil?
521
563
  end
522
564
  end
523
- result.concat store.get_methods(fqns, scope: scope, visibility: visibility).sort{ |a, b| a.name <=> b.name }
565
+ # Store#get_methods doesn't know about full tags, just
566
+ # namespaces; resolving the generics in the method pins is this
567
+ # class' responsibility
568
+ raw_methods = store.get_methods(fqns, scope: scope, visibility: visibility).sort{ |a, b| a.name <=> b.name }
569
+ namespace_pin = store.get_path_pins(fqns).select{|p| p.is_a?(Pin::Namespace)}.first
570
+ methods = if rooted_tag != fqns
571
+ methods = raw_methods.map do |method_pin|
572
+ method_pin.resolve_generics(namespace_pin, rooted_type)
573
+ end
574
+ else
575
+ raw_methods
576
+ end
577
+ result.concat methods
524
578
  if deep
525
579
  if scope == :instance
526
- store.get_includes(fqns).reverse.each do |im|
527
- fqim = qualify(im, fqns)
528
- result.concat inner_get_methods(fqim, scope, visibility, deep, skip, true) unless fqim.nil?
580
+ store.get_includes(fqns).reverse.each do |include_tag|
581
+ rooted_include_tag = qualify(include_tag, rooted_tag)
582
+ # Ensure the types returned by the included methods are
583
+ # relative to the generics passed to the include. e.g.,
584
+ # Foo<String> might include Enumerable<String>
585
+ #
586
+ # @todo perform the same translation in the other areas
587
+ # here after adding a spec and handling things correctly
588
+ # in ApiMap::Store and RbsMap::Conversions
589
+ resolved_include_type = ComplexType.parse(rooted_include_tag).resolve_generics(namespace_pin, rooted_type)
590
+ methods = inner_get_methods(resolved_include_type.tag, scope, visibility, deep, skip, true)
591
+ result.concat methods
529
592
  end
530
593
  fqsc = qualify_superclass(fqns)
531
594
  unless fqsc.nil?
@@ -589,6 +652,8 @@ module Solargraph
589
652
  qualify namespace, context.split('::')[0..-2].join('::')
590
653
  end
591
654
 
655
+ # @param fqsub [String]
656
+ # @return [String, nil]
592
657
  def qualify_superclass fqsub
593
658
  sup = store.get_superclass(fqsub)
594
659
  return nil if sup.nil?
@@ -598,11 +663,12 @@ module Solargraph
598
663
  qualify(sup, parts.join('::'))
599
664
  end
600
665
 
601
- # @param name [String]
602
- # @param root [String]
603
- # @param skip [Set<String>]
604
- # @return [String, nil]
666
+ # @param name [String] Namespace to fully qualify
667
+ # @param root [String] The context to search
668
+ # @param skip [Set<String>] Contexts already searched
669
+ # @return [String, nil] Fully qualified ("rooted") namespace
605
670
  def inner_qualify name, root, skip
671
+ return name if name == ComplexType::GENERIC_TAG_NAME
606
672
  return nil if name.nil?
607
673
  return nil if skip.include?(root)
608
674
  skip.add root
@@ -652,8 +718,8 @@ module Solargraph
652
718
 
653
719
  # Sort an array of pins to put nil or undefined variables last.
654
720
  #
655
- # @param pins [Array<Solargraph::Pin::Base>]
656
- # @return [Array<Solargraph::Pin::Base>]
721
+ # @param pins [Enumerable<Solargraph::Pin::Base>]
722
+ # @return [Enumerable<Solargraph::Pin::Base>]
657
723
  def prefer_non_nil_variables pins
658
724
  result = []
659
725
  nil_pins = []
@@ -667,8 +733,8 @@ module Solargraph
667
733
  result + nil_pins
668
734
  end
669
735
 
670
- # @param pins [Array<Pin::Base>]
671
- # @param visibility [Array<Symbol>]
736
+ # @param pins [Enumerable<Pin::Base>]
737
+ # @param visibility [Enumerable<Symbol>]
672
738
  # @return [Array<Pin::Base>]
673
739
  def resolve_method_aliases pins, visibility = [:public, :private, :protected]
674
740
  result = []
@@ -685,7 +751,7 @@ module Solargraph
685
751
  def resolve_method_alias pin
686
752
  return pin if !pin.is_a?(Pin::MethodAlias) || @method_alias_stack.include?(pin.path)
687
753
  @method_alias_stack.push pin.path
688
- origin = get_method_stack(pin.full_context.namespace, pin.original, scope: pin.scope).first
754
+ origin = get_method_stack(pin.full_context.tag, pin.original, scope: pin.scope).first
689
755
  @method_alias_stack.pop
690
756
  return pin if origin.nil?
691
757
  args = {
@@ -1,6 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'set'
4
3
 
5
4
  module Solargraph
6
5
  # A container of source maps and workspace data to be cataloged in an ApiMap.
@@ -21,7 +20,8 @@ module Solargraph
21
20
  def initialize source_maps: [], workspace: Workspace.new, external_requires: []
22
21
  @source_maps = source_maps.to_set
23
22
  @workspace = workspace
24
- @external_requires = external_requires.to_set
23
+ @external_requires = external_requires.reject { |path| workspace.would_require?(path) }
24
+ .to_set
25
25
  end
26
26
  end
27
27
  end
@@ -1,4 +1,5 @@
1
1
  require 'fileutils'
2
+ require 'rbs'
2
3
 
3
4
  module Solargraph
4
5
  module Cache
@@ -14,7 +15,7 @@ module Solargraph
14
15
  File.join(Dir.home, '.cache', 'solargraph')
15
16
  end
16
17
 
17
- # The working directory for the current Ruby and Solargraph versions.
18
+ # The working directory for the current Ruby, RBS, and Solargraph versions.
18
19
  #
19
20
  # @return [String]
20
21
  def work_dir
@@ -23,9 +24,21 @@ module Solargraph
23
24
  File.join(base_dir, "ruby-#{RUBY_VERSION}", "rbs-#{RBS::VERSION}", "solargraph-#{Solargraph::VERSION}")
24
25
  end
25
26
 
27
+ # Append the given path to the current cache directory (`work_dir`).
28
+ #
29
+ # @example
30
+ # Cache.join('date-3.4.1.ser')
31
+ #
32
+ # @param path [Array<String>]
33
+ # @return [String]
34
+ def join *path
35
+ File.join(work_dir, *path)
36
+ end
37
+
38
+ # @param path [Array<String>]
26
39
  # @return [Array<Solargraph::Pin::Base>, nil]
27
40
  def load *path
28
- file = File.join(work_dir, *path)
41
+ file = join(*path)
29
42
  return nil unless File.file?(file)
30
43
  Marshal.load(File.read(file, mode: 'rb'))
31
44
  rescue StandardError => e
@@ -34,17 +47,28 @@ module Solargraph
34
47
  nil
35
48
  end
36
49
 
37
- # @return [Boolean]
50
+ def exist? *path
51
+ File.file? join(*path)
52
+ end
53
+
54
+ # @param path [Array<String>]
55
+ # @param pins [Array<Pin::Base>]
56
+ # @return [void]
38
57
  def save *path, pins
39
- return false if pins.empty?
40
58
  file = File.join(work_dir, *path)
41
59
  base = File.dirname(file)
42
60
  FileUtils.mkdir_p base unless File.directory?(base)
43
61
  ser = Marshal.dump(pins)
44
62
  File.write file, ser, mode: 'wb'
45
- true
46
63
  end
47
64
 
65
+ # @return [void]
66
+ # @param path [Array<String>]
67
+ def uncache *path
68
+ FileUtils.rm_rf File.join(work_dir, *path), secure: true
69
+ end
70
+
71
+ # @return [void]
48
72
  def clear
49
73
  FileUtils.rm_rf base_dir, secure: true
50
74
  end