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,651 +1,729 @@
1
- # frozen_string_literal: true
2
-
3
- module Solargraph
4
- module Pin
5
- # The base class for map pins.
6
- #
7
- class Base
8
- include Common
9
- include Conversions
10
- include Documenting
11
- include Logging
12
-
13
- # @return [YARD::CodeObjects::Base]
14
- attr_reader :code_object
15
-
16
- # @return [Solargraph::Location]
17
- attr_reader :location
18
-
19
- # @return [Solargraph::Location]
20
- attr_reader :type_location
21
-
22
- # @return [String]
23
- attr_reader :name
24
-
25
- # @return [String]
26
- attr_reader :path
27
-
28
- # @return [::Symbol]
29
- attr_accessor :source
30
-
31
- def presence_certain?
32
- true
33
- end
34
-
35
- # @param location [Solargraph::Location, nil]
36
- # @param type_location [Solargraph::Location, nil]
37
- # @param closure [Solargraph::Pin::Closure, nil]
38
- # @param name [String]
39
- # @param comments [String]
40
- # @param source [Symbol, nil]
41
- # @param docstring [YARD::Docstring, nil]
42
- # @param directives [::Array<YARD::Tags::Directive>, nil]
43
- def initialize location: nil, type_location: nil, closure: nil, source: nil, name: '', comments: '', docstring: nil, directives: nil
44
- @location = location
45
- @type_location = type_location
46
- @closure = closure
47
- @name = name
48
- @comments = comments
49
- @source = source
50
- @identity = nil
51
- @docstring = docstring
52
- @directives = directives
53
- assert_source_provided
54
- end
55
-
56
- # @param other [self]
57
- # @param attrs [Hash{Symbol => Object}]
58
- #
59
- # @return [self]
60
- def combine_with(other, attrs={})
61
- raise "tried to combine #{other.class} with #{self.class}" unless other.class == self.class
62
- type_location = choose(other, :type_location)
63
- location = choose(other, :location)
64
- combined_name = combine_name(other)
65
- new_attrs = {
66
- location: location,
67
- type_location: type_location,
68
- name: combined_name,
69
- closure: choose_pin_attr_with_same_name(other, :closure),
70
- comments: choose_longer(other, :comments),
71
- source: :combined,
72
- docstring: choose(other, :docstring),
73
- directives: combine_directives(other),
74
- }.merge(attrs)
75
- assert_same_macros(other)
76
- logger.debug { "Base#combine_with(path=#{path}) - other.comments=#{other.comments.inspect}, self.comments = #{self.comments}" }
77
- out = self.class.new(**new_attrs)
78
- out.reset_generated!
79
- out
80
- end
81
-
82
- # @param other [self]
83
- # @param attr [::Symbol]
84
- # @sg-ignore
85
- # @return [undefined]
86
- def choose_longer(other, attr)
87
- # @type [undefined]
88
- val1 = send(attr)
89
- # @type [undefined]
90
- val2 = other.send(attr)
91
- return val1 if val1 == val2
92
- return val2 if val1.nil?
93
- # @sg-ignore
94
- val1.length > val2.length ? val1 : val2
95
- end
96
-
97
- # @param other [self]
98
- # @return [::Array<YARD::Tags::Directive>, nil]
99
- def combine_directives(other)
100
- return self.directives if other.directives.empty?
101
- return other.directives if directives.empty?
102
- [directives + other.directives].uniq
103
- end
104
-
105
- # @param other [self]
106
- # @return [String]
107
- def combine_name(other)
108
- if needs_consistent_name? || other.needs_consistent_name?
109
- assert_same(other, :name)
110
- else
111
- choose(other, :name)
112
- end
113
- end
114
-
115
- # @return [void]
116
- def reset_generated!
117
- # @return_type doesn't go here as subclasses tend to assign it
118
- # themselves in constructors, and they will deal with setting
119
- # it in any methods that call this
120
- #
121
- # @docstring also doesn't go here, as there is code which
122
- # directly manipulates docstring without editing comments
123
- # (e.g., Api::Map::Store#index processes overrides that way
124
- #
125
- # Same with @directives, @macros, @maybe_directives, which
126
- # regenerate docstring
127
- @deprecated = nil
128
- reset_conversions
129
- end
130
-
131
- def needs_consistent_name?
132
- true
133
- end
134
-
135
- # @sg-ignore def should infer as symbol - "Not enough arguments to Module#protected"
136
- protected def equality_fields
137
- [name, location, type_location, closure, source]
138
- end
139
-
140
- # @param other [self]
141
- # @return [ComplexType]
142
- def combine_return_type(other)
143
- if return_type.undefined?
144
- other.return_type
145
- elsif other.return_type.undefined?
146
- return_type
147
- elsif dodgy_return_type_source? && !other.dodgy_return_type_source?
148
- other.return_type
149
- elsif other.dodgy_return_type_source? && !dodgy_return_type_source?
150
- return_type
151
- else
152
- all_items = return_type.items + other.return_type.items
153
- if all_items.any? { |item| item.selfy? } && all_items.any? { |item| item.rooted_tag == context.rooted_tag }
154
- # assume this was a declaration that should have said 'self'
155
- all_items.delete_if { |item| item.rooted_tag == context.rooted_tag }
156
- end
157
- ComplexType.new(all_items)
158
- end
159
- end
160
-
161
- def dodgy_return_type_source?
162
- # uses a lot of 'Object' instead of 'self'
163
- location&.filename&.include?('core_ext/object/')
164
- end
165
-
166
- # when choices are arbitrary, make sure the choice is consistent
167
- #
168
- # @param other [Pin::Base]
169
- # @param attr [::Symbol]
170
- #
171
- # @return [Object, nil]
172
- def choose(other, attr)
173
- results = [self, other].map(&attr).compact
174
- # true and false are different classes and can't be sorted
175
- return true if results.any? { |r| r == true || r == false }
176
- results.min
177
- rescue
178
- STDERR.puts("Problem handling #{attr} for \n#{self.inspect}\n and \n#{other.inspect}\n\n#{self.send(attr).inspect} vs #{other.send(attr).inspect}")
179
- raise
180
- end
181
-
182
- # @param other [self]
183
- # @param attr [Symbol]
184
- # @sg-ignore
185
- # @return [undefined]
186
- def choose_node(other, attr)
187
- if other.object_id < attr.object_id
188
- other.send(attr)
189
- else
190
- send(attr)
191
- end
192
- end
193
-
194
- # @param other [self]
195
- # @param attr [::Symbol]
196
- # @sg-ignore
197
- # @return [undefined]
198
- def prefer_rbs_location(other, attr)
199
- if rbs_location? && !other.rbs_location?
200
- self.send(attr)
201
- elsif !rbs_location? && other.rbs_location?
202
- other.send(attr)
203
- else
204
- choose(other, attr)
205
- end
206
- end
207
-
208
- def rbs_location?
209
- type_location&.rbs?
210
- end
211
-
212
- # @param other [self]
213
- # @return [void]
214
- def assert_same_macros(other)
215
- return unless self.source == :yardoc && other.source == :yardoc
216
- assert_same_count(other, :macros)
217
- assert_same_array_content(other, :macros) { |macro| macro.tag.name }
218
- end
219
-
220
- # @param other [self]
221
- # @param attr [::Symbol]
222
- # @return [void]
223
- # @todo strong typechecking should complain when there are no block-related tags
224
- def assert_same_array_content(other, attr, &block)
225
- arr1 = send(attr)
226
- raise "Expected #{attr} on #{self} to be an Enumerable, got #{arr1.class}" unless arr1.is_a?(::Enumerable)
227
- # @type arr1 [::Enumerable]
228
- arr2 = other.send(attr)
229
- raise "Expected #{attr} on #{other} to be an Enumerable, got #{arr2.class}" unless arr2.is_a?(::Enumerable)
230
- # @type arr2 [::Enumerable]
231
-
232
- # @sg-ignore
233
- # @type [undefined]
234
- values1 = arr1.map(&block)
235
- # @type [undefined]
236
- values2 = arr2.map(&block)
237
- # @sg-ignore
238
- return arr1 if values1 == values2
239
- Solargraph.assert_or_log("combine_with_#{attr}".to_sym,
240
- "Inconsistent #{attr.inspect} values between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self values = #{values1}\nother values =#{attr} = #{values2}")
241
- arr1
242
- end
243
-
244
- # @param other [self]
245
- # @param attr [::Symbol]
246
- #
247
- # @return [::Enumerable]
248
- def assert_same_count(other, attr)
249
- # @type [::Enumerable]
250
- arr1 = self.send(attr)
251
- raise "Expected #{attr} on #{self} to be an Enumerable, got #{arr1.class}" unless arr1.is_a?(::Enumerable)
252
- # @type [::Enumerable]
253
- arr2 = other.send(attr)
254
- raise "Expected #{attr} on #{other} to be an Enumerable, got #{arr2.class}" unless arr2.is_a?(::Enumerable)
255
- return arr1 if arr1.count == arr2.count
256
- Solargraph.assert_or_log("combine_with_#{attr}".to_sym,
257
- "Inconsistent #{attr.inspect} count value between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self.#{attr} = #{arr1.inspect}\nother.#{attr} = #{arr2.inspect}")
258
- arr1
259
- end
260
-
261
- # @param other [self]
262
- # @param attr [::Symbol]
263
- #
264
- # @return [Object, nil]
265
- def assert_same(other, attr)
266
- return false if other.nil?
267
- val1 = send(attr)
268
- val2 = other.send(attr)
269
- return val1 if val1 == val2
270
- Solargraph.assert_or_log("combine_with_#{attr}".to_sym,
271
- "Inconsistent #{attr.inspect} values between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self.#{attr} = #{val1.inspect}\nother.#{attr} = #{val2.inspect}")
272
- val1
273
- end
274
-
275
- # @param other [self]
276
- # @param attr [::Symbol]
277
- # @sg-ignore
278
- # @return [undefined]
279
- def choose_pin_attr_with_same_name(other, attr)
280
- # @type [Pin::Base, nil]
281
- val1 = send(attr)
282
- # @type [Pin::Base, nil]
283
- val2 = other.send(attr)
284
- raise "Expected pin for #{attr} on\n#{self.inspect},\ngot #{val1.inspect}" unless val1.nil? || val1.is_a?(Pin::Base)
285
- raise "Expected pin for #{attr} on\n#{other.inspect},\ngot #{val2.inspect}" unless val2.nil? || val2.is_a?(Pin::Base)
286
- if val1&.name != val2&.name
287
- Solargraph.assert_or_log("combine_with_#{attr}_name".to_sym,
288
- "Inconsistent #{attr.inspect} name values between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self.#{attr} = #{val1.inspect}\nother.#{attr} = #{val2.inspect}")
289
- end
290
- choose_pin_attr(other, attr)
291
- end
292
-
293
- def choose_pin_attr(other, attr)
294
- # @type [Pin::Base, nil]
295
- val1 = send(attr)
296
- # @type [Pin::Base, nil]
297
- val2 = other.send(attr)
298
- if val1.class != val2.class
299
- Solargraph.assert_or_log("combine_with_#{attr}_class".to_sym,
300
- "Inconsistent #{attr.inspect} class values between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self.#{attr} = #{val1.inspect}\nother.#{attr} = #{val2.inspect}")
301
- return val1
302
- end
303
- # arbitrary way of choosing a pin
304
- [val1, val2].compact.min_by { _1.best_location.to_s }
305
- end
306
-
307
- def assert_source_provided
308
- Solargraph.assert_or_log(:source, "source not provided - #{@path} #{@source} #{self.class}") if source.nil?
309
- end
310
-
311
- # @return [String]
312
- def comments
313
- @comments ||= ''
314
- end
315
-
316
- # @param generics_to_resolve [Enumerable<String>]
317
- # @param return_type_context [ComplexType, nil]
318
- # @param context [ComplexType]
319
- # @param resolved_generic_values [Hash{String => ComplexType}]
320
- # @return [self]
321
- def resolve_generics_from_context(generics_to_resolve, return_type_context = nil, resolved_generic_values: {})
322
- proxy return_type.resolve_generics_from_context(generics_to_resolve,
323
- return_type_context,
324
- resolved_generic_values: resolved_generic_values)
325
- end
326
-
327
- # @yieldparam [ComplexType]
328
- # @yieldreturn [ComplexType]
329
- # @return [self]
330
- def transform_types(&transform)
331
- proxy return_type.transform(&transform)
332
- end
333
-
334
- # Determine the concrete type for each of the generic type
335
- # parameters used in this method based on the parameters passed
336
- # into the its class and return a new method pin.
337
- #
338
- # @param definitions [Pin::Namespace] The module/class which uses generic types
339
- # @param context_type [ComplexType] The receiver type
340
- # @return [self]
341
- def resolve_generics definitions, context_type
342
- transform_types { |t| t.resolve_generics(definitions, context_type) if t }
343
- end
344
-
345
- def all_rooted?
346
- !return_type || return_type.all_rooted?
347
- end
348
-
349
- # @param generics_to_erase [::Array<String>]
350
- # @return [self]
351
- def erase_generics(generics_to_erase)
352
- return self if generics_to_erase.empty?
353
- transform_types { |t| t.erase_generics(generics_to_erase) }
354
- end
355
-
356
- # @return [String, nil]
357
- def filename
358
- return nil if location.nil?
359
- location.filename
360
- end
361
-
362
- # @return [Integer]
363
- def completion_item_kind
364
- LanguageServer::CompletionItemKinds::KEYWORD
365
- end
366
-
367
- # @return [Integer, nil]
368
- def symbol_kind
369
- nil
370
- end
371
-
372
- def to_s
373
- desc
374
- end
375
-
376
- # @return [Boolean]
377
- def variable?
378
- false
379
- end
380
-
381
- # @return [Location, nil]
382
- def best_location
383
- location || type_location
384
- end
385
-
386
- # True if the specified pin is a near match to this one. A near match
387
- # indicates that the pins contain mostly the same data. Any differences
388
- # between them should not have an impact on the API surface.
389
- #
390
- # @param other [Solargraph::Pin::Base, Object]
391
- # @return [Boolean]
392
- def nearly? other
393
- self.class == other.class &&
394
- name == other.name &&
395
- (closure == other.closure || (closure && closure.nearly?(other.closure))) &&
396
- (comments == other.comments ||
397
- (((maybe_directives? == false && other.maybe_directives? == false) || compare_directives(directives, other.directives)) &&
398
- compare_docstring_tags(docstring, other.docstring))
399
- )
400
- end
401
-
402
- # Pin equality is determined using the #nearly? method and also
403
- # requiring both pins to have the same location.
404
- #
405
- # @param other [self]
406
- def == other
407
- return false unless nearly? other
408
- comments == other.comments && location == other.location
409
- end
410
-
411
- # The pin's return type.
412
- #
413
- # @return [ComplexType]
414
- def return_type
415
- @return_type ||= ComplexType::UNDEFINED
416
- end
417
-
418
- # @return [YARD::Docstring]
419
- def docstring
420
- parse_comments unless @docstring
421
- @docstring ||= Solargraph::Source.parse_docstring('').to_docstring
422
- end
423
-
424
- # @return [::Array<YARD::Tags::Directive>]
425
- def directives
426
- parse_comments unless @directives
427
- @directives
428
- end
429
-
430
- # @return [::Array<YARD::Tags::MacroDirective>]
431
- def macros
432
- @macros ||= collect_macros
433
- end
434
-
435
- # Perform a quick check to see if this pin possibly includes YARD
436
- # directives. This method does not require parsing the comments.
437
- #
438
- # After the comments have been parsed, this method will return false if
439
- # no directives were found, regardless of whether it previously appeared
440
- # possible.
441
- #
442
- # @return [Boolean]
443
- def maybe_directives?
444
- return !@directives.empty? if defined?(@directives) && @directives
445
- @maybe_directives ||= comments.include?('@!')
446
- end
447
-
448
- # @return [Boolean]
449
- def deprecated?
450
- @deprecated ||= docstring.has_tag?('deprecated')
451
- end
452
-
453
- # Get a fully qualified type from the pin's return type.
454
- #
455
- # The relative type is determined from YARD documentation (@return,
456
- # @param, @type, etc.) and its namespaces are fully qualified using the
457
- # provided ApiMap.
458
- #
459
- # @param api_map [ApiMap]
460
- # @return [ComplexType]
461
- def typify api_map
462
- return_type.qualify(api_map, namespace)
463
- end
464
-
465
- # Infer the pin's return type via static code analysis.
466
- #
467
- # @param api_map [ApiMap]
468
- # @return [ComplexType]
469
- def probe api_map
470
- typify api_map
471
- end
472
-
473
- # @deprecated Use #typify and/or #probe instead
474
- # @param api_map [ApiMap]
475
- # @return [ComplexType]
476
- def infer api_map
477
- Solargraph::Logging.logger.warn "WARNING: Pin #infer methods are deprecated. Use #typify or #probe instead."
478
- type = typify(api_map)
479
- return type unless type.undefined?
480
- probe api_map
481
- end
482
-
483
- def proxied?
484
- @proxied ||= false
485
- end
486
-
487
- def probed?
488
- @probed ||= false
489
- end
490
-
491
- # @param api_map [ApiMap]
492
- # @return [self]
493
- def realize api_map
494
- return self if return_type.defined?
495
- type = typify(api_map)
496
- return proxy(type) if type.defined?
497
- type = probe(api_map)
498
- return self if type.undefined?
499
- result = proxy(type)
500
- result.probed = true
501
- result
502
- end
503
-
504
- # Return a proxy for this pin with the specified return type. Other than
505
- # the return type and the #proxied? setting, the proxy should be a clone
506
- # of the original.
507
- #
508
- # @param return_type [ComplexType]
509
- # @return [self]
510
- def proxy return_type
511
- result = dup
512
- result.return_type = return_type
513
- result.proxied = true
514
- result
515
- end
516
-
517
- # @deprecated
518
- # @return [String]
519
- def identity
520
- @identity ||= "#{closure&.path}|#{name}|#{location}"
521
- end
522
-
523
- # @return [String, nil]
524
- def to_rbs
525
- return_type.to_rbs
526
- end
527
-
528
- # @return [String]
529
- def type_desc
530
- rbs = to_rbs
531
- # RBS doesn't have a way to represent a Class<x> type
532
- rbs = return_type.rooted_tags if return_type.name == 'Class'
533
- if path
534
- if rbs
535
- path + ' ' + rbs
536
- else
537
- path
538
- end
539
- else
540
- rbs
541
- end
542
- end
543
-
544
- # @return [String]
545
- def inner_desc
546
- closure_info = closure&.desc
547
- binder_info = binder&.desc
548
- "name=#{name.inspect} return_type=#{type_desc}, context=#{context.rooted_tags}, closure=#{closure_info}, binder=#{binder_info}"
549
- end
550
-
551
- def desc
552
- "[#{inner_desc}]"
553
- end
554
-
555
- # @return [String]
556
- def inspect
557
- "#<#{self.class} `#{self.inner_desc}`#{all_location_text} via #{source.inspect}>"
558
- end
559
-
560
- def all_location_text
561
- if location.nil? && type_location.nil?
562
- ''
563
- elsif !location.nil? && type_location.nil?
564
- " at #{location.inspect})"
565
- elsif !type_location.nil? && location.nil?
566
- " at #{type_location.inspect})"
567
- else
568
- " at (#{location.inspect} and #{type_location.inspect})"
569
- end
570
- end
571
-
572
- def reset_generated!
573
- end
574
-
575
- protected
576
-
577
- # @return [Boolean]
578
- attr_writer :probed
579
-
580
- # @return [Boolean]
581
- attr_writer :proxied
582
-
583
- # @return [ComplexType]
584
- attr_writer :return_type
585
-
586
- attr_writer :docstring
587
-
588
- attr_writer :directives
589
-
590
- private
591
-
592
- # @return [void]
593
- def parse_comments
594
- # HACK: Avoid a NoMethodError on nil with empty overload tags
595
- if comments.nil? || comments.empty? || comments.strip.end_with?('@overload')
596
- @docstring = nil
597
- @directives = []
598
- else
599
- # HACK: Pass a dummy code object to the parser for plugins that
600
- # expect it not to be nil
601
- parse = Solargraph::Source.parse_docstring(comments)
602
- @docstring = parse.to_docstring
603
- @directives = parse.directives
604
- end
605
- end
606
-
607
- # True if two docstrings have the same tags, regardless of any other
608
- # differences.
609
- #
610
- # @param d1 [YARD::Docstring]
611
- # @param d2 [YARD::Docstring]
612
- # @return [Boolean]
613
- def compare_docstring_tags d1, d2
614
- return false if d1.tags.length != d2.tags.length
615
- d1.tags.each_index do |i|
616
- return false unless compare_tags(d1.tags[i], d2.tags[i])
617
- end
618
- true
619
- end
620
-
621
- # @param dir1 [::Array<YARD::Tags::Directive>]
622
- # @param dir2 [::Array<YARD::Tags::Directive>]
623
- # @return [Boolean]
624
- def compare_directives dir1, dir2
625
- return false if dir1.length != dir2.length
626
- dir1.each_index do |i|
627
- return false unless compare_tags(dir1[i].tag, dir2[i].tag)
628
- end
629
- true
630
- end
631
-
632
- # @param tag1 [YARD::Tags::Tag]
633
- # @param tag2 [YARD::Tags::Tag]
634
- # @return [Boolean]
635
- def compare_tags tag1, tag2
636
- tag1.class == tag2.class &&
637
- tag1.tag_name == tag2.tag_name &&
638
- tag1.text == tag2.text &&
639
- tag1.name == tag2.name &&
640
- tag1.types == tag2.types
641
- end
642
-
643
- # @return [::Array<YARD::Tags::Handlers::Directive>]
644
- def collect_macros
645
- return [] unless maybe_directives?
646
- parse = Solargraph::Source.parse_docstring(comments)
647
- parse.directives.select{ |d| d.tag.tag_name == 'macro' }
648
- end
649
- end
650
- end
651
- end
1
+ # frozen_string_literal: true
2
+
3
+ module Solargraph
4
+ module Pin
5
+ # The base class for map pins.
6
+ #
7
+ class Base
8
+ include Common
9
+ include Conversions
10
+ include Documenting
11
+ include Logging
12
+
13
+ # @return [YARD::CodeObjects::Base]
14
+ attr_reader :code_object
15
+
16
+ # @return [Solargraph::Location, nil]
17
+ attr_reader :location
18
+
19
+ # @return [Solargraph::Location, nil]
20
+ attr_reader :type_location
21
+
22
+ # @return [String]
23
+ attr_reader :name
24
+
25
+ # @return [String]
26
+ attr_reader :path
27
+
28
+ # @return [::Symbol]
29
+ attr_accessor :source
30
+
31
+ # @type [::Numeric, nil] A priority for determining if pins should be combined or not
32
+ # A nil priority is considered the be the lowest. All code, yard & rbs pins have nil priority
33
+ # Between 2 pins, the one with the higher priority gets chosen. If the priorities are equal, they are combined.
34
+ attr_reader :combine_priority
35
+
36
+ def presence_certain?
37
+ true
38
+ end
39
+
40
+ # @param location [Solargraph::Location, nil]
41
+ # @param type_location [Solargraph::Location, nil]
42
+ # @param closure [Solargraph::Pin::Closure, nil]
43
+ # @param name [String]
44
+ # @param comments [String]
45
+ # @param source [Symbol, nil]
46
+ # @param docstring [YARD::Docstring, nil]
47
+ # @param directives [::Array<YARD::Tags::Directive>, nil]
48
+ # @param combine_priority [::Numeric, nil] See attr_reader for combine_priority
49
+ def initialize location: nil, type_location: nil, closure: nil, source: nil, name: '', comments: '', docstring: nil, directives: nil, combine_priority: nil
50
+ @location = location
51
+ @type_location = type_location
52
+ @closure = closure
53
+ @name = name
54
+ @comments = comments
55
+ @source = source
56
+ @identity = nil
57
+ @docstring = docstring
58
+ @directives = directives
59
+ @combine_priority = combine_priority
60
+
61
+ assert_source_provided
62
+ assert_location_provided
63
+ end
64
+
65
+ # @return [void]
66
+ def assert_location_provided
67
+ return unless best_location.nil? && %i[yardoc source rbs].include?(source)
68
+
69
+ Solargraph.assert_or_log(:best_location, "Neither location nor type_location provided - #{path} #{source} #{self.class}")
70
+ end
71
+
72
+ # @return [Pin::Closure, nil]
73
+ def closure
74
+ Solargraph.assert_or_log(:closure, "Closure not set on #{self.class} #{name.inspect} from #{source.inspect}") unless @closure
75
+ # @type [Pin::Closure, nil]
76
+ @closure
77
+ end
78
+
79
+ # @param other [self]
80
+ # @param attrs [Hash{::Symbol => Object}]
81
+ #
82
+ # @return [self]
83
+ def combine_with(other, attrs={})
84
+ raise "tried to combine #{other.class} with #{self.class}" unless other.class == self.class
85
+ priority_choice = choose_priority(other)
86
+ return priority_choice unless priority_choice.nil?
87
+
88
+ type_location = choose(other, :type_location)
89
+ location = choose(other, :location)
90
+ combined_name = combine_name(other)
91
+ new_attrs = {
92
+ location: location,
93
+ type_location: type_location,
94
+ name: combined_name,
95
+ closure: choose_pin_attr_with_same_name(other, :closure),
96
+ comments: choose_longer(other, :comments),
97
+ source: :combined,
98
+ docstring: choose(other, :docstring),
99
+ directives: combine_directives(other),
100
+ combine_priority: combine_priority
101
+ }.merge(attrs)
102
+ assert_same_macros(other)
103
+ logger.debug { "Base#combine_with(path=#{path}) - other.comments=#{other.comments.inspect}, self.comments = #{self.comments}" }
104
+ out = self.class.new(**new_attrs)
105
+ out.reset_generated!
106
+ out
107
+ end
108
+
109
+ # @param other [self]
110
+ # @return [self, nil] Returns either the pin chosen based on priority or nil
111
+ # A nil return means that the combination process must proceed
112
+ def choose_priority(other)
113
+ if combine_priority.nil? && !other.combine_priority.nil?
114
+ return other
115
+ elsif other.combine_priority.nil? && !combine_priority.nil?
116
+ return self
117
+ elsif !combine_priority.nil? && !other.combine_priority.nil?
118
+ if combine_priority > other.combine_priority
119
+ return self
120
+ elsif combine_priority < other.combine_priority
121
+ return other
122
+ end
123
+ end
124
+
125
+ nil
126
+ end
127
+
128
+ # @param other [self]
129
+ # @param attr [::Symbol]
130
+ # @sg-ignore
131
+ # @return [undefined]
132
+ def choose_longer(other, attr)
133
+ # @type [undefined]
134
+ val1 = send(attr)
135
+ # @type [undefined]
136
+ val2 = other.send(attr)
137
+ return val1 if val1 == val2
138
+ return val2 if val1.nil?
139
+ val1.length > val2.length ? val1 : val2
140
+ end
141
+
142
+ # @param other [self]
143
+ # @return [::Array<YARD::Tags::Directive>, nil]
144
+ def combine_directives(other)
145
+ return self.directives if other.directives.empty?
146
+ return other.directives if directives.empty?
147
+ [directives + other.directives].uniq
148
+ end
149
+
150
+ # @param other [self]
151
+ # @return [String]
152
+ def combine_name(other)
153
+ if needs_consistent_name? || other.needs_consistent_name?
154
+ assert_same(other, :name)
155
+ else
156
+ choose(other, :name)
157
+ end
158
+ end
159
+
160
+ # @return [void]
161
+ def reset_generated!
162
+ # @return_type doesn't go here as subclasses tend to assign it
163
+ # themselves in constructors, and they will deal with setting
164
+ # it in any methods that call this
165
+ #
166
+ # @docstring also doesn't go here, as there is code which
167
+ # directly manipulates docstring without editing comments
168
+ # (e.g., Api::Map::Store#index processes overrides that way
169
+ #
170
+ # Same with @directives, @macros, @maybe_directives, which
171
+ # regenerate docstring
172
+ @deprecated = nil
173
+ reset_conversions
174
+ end
175
+
176
+ def needs_consistent_name?
177
+ true
178
+ end
179
+
180
+ # @sg-ignore def should infer as symbol - "Not enough arguments to Module#protected"
181
+ protected def equality_fields
182
+ [name, location, type_location, closure, source]
183
+ end
184
+
185
+ # @param other [self]
186
+ # @return [ComplexType]
187
+ def combine_return_type(other)
188
+ if return_type.undefined?
189
+ other.return_type
190
+ elsif other.return_type.undefined?
191
+ return_type
192
+ elsif dodgy_return_type_source? && !other.dodgy_return_type_source?
193
+ other.return_type
194
+ elsif other.dodgy_return_type_source? && !dodgy_return_type_source?
195
+ return_type
196
+ else
197
+ all_items = return_type.items + other.return_type.items
198
+ if all_items.any? { |item| item.selfy? } && all_items.any? { |item| item.rooted_tag == context.reduce_class_type.rooted_tag }
199
+ # assume this was a declaration that should have said 'self'
200
+ all_items.delete_if { |item| item.rooted_tag == context.reduce_class_type.rooted_tag }
201
+ end
202
+ ComplexType.new(all_items)
203
+ end
204
+ end
205
+
206
+ def dodgy_return_type_source?
207
+ # uses a lot of 'Object' instead of 'self'
208
+ location&.filename&.include?('core_ext/object/')
209
+ end
210
+
211
+ # when choices are arbitrary, make sure the choice is consistent
212
+ #
213
+ # @param other [Pin::Base]
214
+ # @param attr [::Symbol]
215
+ #
216
+ # @return [Object, nil]
217
+ def choose(other, attr)
218
+ results = [self, other].map(&attr).compact
219
+ # true and false are different classes and can't be sorted
220
+ return true if results.any? { |r| r == true || r == false }
221
+ results.min
222
+ rescue
223
+ STDERR.puts("Problem handling #{attr} for \n#{self.inspect}\n and \n#{other.inspect}\n\n#{self.send(attr).inspect} vs #{other.send(attr).inspect}")
224
+ raise
225
+ end
226
+
227
+ # @param other [self]
228
+ # @param attr [::Symbol]
229
+ # @sg-ignore
230
+ # @return [undefined]
231
+ def choose_node(other, attr)
232
+ if other.object_id < attr.object_id
233
+ other.send(attr)
234
+ else
235
+ send(attr)
236
+ end
237
+ end
238
+
239
+ # @param other [self]
240
+ # @param attr [::Symbol]
241
+ # @sg-ignore
242
+ # @return [undefined]
243
+ def prefer_rbs_location(other, attr)
244
+ if rbs_location? && !other.rbs_location?
245
+ self.send(attr)
246
+ elsif !rbs_location? && other.rbs_location?
247
+ other.send(attr)
248
+ else
249
+ choose(other, attr)
250
+ end
251
+ end
252
+
253
+ def rbs_location?
254
+ type_location&.rbs?
255
+ end
256
+
257
+ # @param other [self]
258
+ # @return [void]
259
+ def assert_same_macros(other)
260
+ return unless self.source == :yardoc && other.source == :yardoc
261
+ assert_same_count(other, :macros)
262
+ # @param [YARD::Tags::MacroDirective]
263
+ assert_same_array_content(other, :macros) { |macro| macro.tag.name }
264
+ end
265
+
266
+ # @param other [self]
267
+ # @param attr [::Symbol]
268
+ # @return [void]
269
+ # @todo strong typechecking should complain when there are no block-related tags
270
+ def assert_same_array_content(other, attr, &block)
271
+ arr1 = send(attr)
272
+ raise "Expected #{attr} on #{self} to be an Enumerable, got #{arr1.class}" unless arr1.is_a?(::Enumerable)
273
+ # @type arr1 [::Enumerable]
274
+ arr2 = other.send(attr)
275
+ raise "Expected #{attr} on #{other} to be an Enumerable, got #{arr2.class}" unless arr2.is_a?(::Enumerable)
276
+ # @type arr2 [::Enumerable]
277
+
278
+ # @type [undefined]
279
+ values1 = arr1.map(&block)
280
+ # @type [undefined]
281
+ values2 = arr2.map(&block)
282
+ # @sg-ignore
283
+ return arr1 if values1 == values2
284
+ Solargraph.assert_or_log("combine_with_#{attr}".to_sym,
285
+ "Inconsistent #{attr.inspect} values between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self values = #{values1}\nother values =#{attr} = #{values2}")
286
+ arr1
287
+ end
288
+
289
+ # @param other [self]
290
+ # @param attr [::Symbol]
291
+ #
292
+ # @return [::Enumerable]
293
+ def assert_same_count(other, attr)
294
+ # @type [::Enumerable]
295
+ arr1 = self.send(attr)
296
+ raise "Expected #{attr} on #{self} to be an Enumerable, got #{arr1.class}" unless arr1.is_a?(::Enumerable)
297
+ # @type [::Enumerable]
298
+ arr2 = other.send(attr)
299
+ raise "Expected #{attr} on #{other} to be an Enumerable, got #{arr2.class}" unless arr2.is_a?(::Enumerable)
300
+ return arr1 if arr1.count == arr2.count
301
+ Solargraph.assert_or_log("combine_with_#{attr}".to_sym,
302
+ "Inconsistent #{attr.inspect} count value between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self.#{attr} = #{arr1.inspect}\nother.#{attr} = #{arr2.inspect}")
303
+ arr1
304
+ end
305
+
306
+ # @param other [self]
307
+ # @param attr [::Symbol]
308
+ #
309
+ # @sg-ignore
310
+ # @return [undefined]
311
+ def assert_same(other, attr)
312
+ return false if other.nil?
313
+ val1 = send(attr)
314
+ val2 = other.send(attr)
315
+ return val1 if val1 == val2
316
+ Solargraph.assert_or_log("combine_with_#{attr}".to_sym,
317
+ "Inconsistent #{attr.inspect} values between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self.#{attr} = #{val1.inspect}\nother.#{attr} = #{val2.inspect}")
318
+ val1
319
+ end
320
+
321
+ # @param other [self]
322
+ # @param attr [::Symbol]
323
+ # @sg-ignore
324
+ # @return [undefined]
325
+ def choose_pin_attr_with_same_name(other, attr)
326
+ # @type [Pin::Base, nil]
327
+ val1 = send(attr)
328
+ # @type [Pin::Base, nil]
329
+ val2 = other.send(attr)
330
+ raise "Expected pin for #{attr} on\n#{self.inspect},\ngot #{val1.inspect}" unless val1.nil? || val1.is_a?(Pin::Base)
331
+ raise "Expected pin for #{attr} on\n#{other.inspect},\ngot #{val2.inspect}" unless val2.nil? || val2.is_a?(Pin::Base)
332
+ if val1&.name != val2&.name
333
+ Solargraph.assert_or_log("combine_with_#{attr}_name".to_sym,
334
+ "Inconsistent #{attr.inspect} name values between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self.#{attr} = #{val1.inspect}\nother.#{attr} = #{val2.inspect}")
335
+ end
336
+ choose_pin_attr(other, attr)
337
+ end
338
+
339
+ # @param other [self]
340
+ # @param attr [::Symbol]
341
+ #
342
+ # @sg-ignore Missing @return tag for Solargraph::Pin::Base#choose_pin_attr
343
+ # @return [undefined]
344
+ def choose_pin_attr(other, attr)
345
+ # @type [Pin::Base, nil]
346
+ val1 = send(attr)
347
+ # @type [Pin::Base, nil]
348
+ val2 = other.send(attr)
349
+ if val1.class != val2.class
350
+ # :nocov:
351
+ Solargraph.assert_or_log("combine_with_#{attr}_class".to_sym,
352
+ "Inconsistent #{attr.inspect} class values between \nself =#{inspect} and \nother=#{other.inspect}:\n\n self.#{attr} = #{val1.inspect}\nother.#{attr} = #{val2.inspect}")
353
+ return val1
354
+ # :nocov:
355
+ end
356
+ # arbitrary way of choosing a pin
357
+ [val1, val2].compact.max_by do |closure|
358
+ [
359
+ # maximize number of gates, as types in other combined pins may
360
+ # depend on those gates
361
+ closure.gates.length,
362
+ # use basename so that results don't vary system to system
363
+ File.basename(closure.best_location.to_s)
364
+ ]
365
+ end
366
+ end
367
+
368
+ # @return [void]
369
+ def assert_source_provided
370
+ Solargraph.assert_or_log(:source, "source not provided - #{@path} #{@source} #{self.class}") if source.nil?
371
+ end
372
+
373
+ # @return [String]
374
+ def comments
375
+ @comments ||= ''
376
+ end
377
+
378
+ # @param generics_to_resolve [Enumerable<String>]
379
+ # @param return_type_context [ComplexType, nil]
380
+ # @param context [ComplexType]
381
+ # @param resolved_generic_values [Hash{String => ComplexType}]
382
+ # @return [self]
383
+ def resolve_generics_from_context(generics_to_resolve, return_type_context = nil, resolved_generic_values: {})
384
+ proxy return_type.resolve_generics_from_context(generics_to_resolve,
385
+ return_type_context,
386
+ resolved_generic_values: resolved_generic_values)
387
+ end
388
+
389
+ # @yieldparam [ComplexType]
390
+ # @yieldreturn [ComplexType]
391
+ # @return [self]
392
+ def transform_types(&transform)
393
+ proxy return_type.transform(&transform)
394
+ end
395
+
396
+ # Determine the concrete type for each of the generic type
397
+ # parameters used in this method based on the parameters passed
398
+ # into the its class and return a new method pin.
399
+ #
400
+ # @param definitions [Pin::Namespace] The module/class which uses generic types
401
+ # @param context_type [ComplexType] The receiver type
402
+ # @return [self]
403
+ def resolve_generics definitions, context_type
404
+ transform_types { |t| t.resolve_generics(definitions, context_type) if t }
405
+ end
406
+
407
+ def all_rooted?
408
+ !return_type || return_type.all_rooted?
409
+ end
410
+
411
+ # @param generics_to_erase [::Array<String>]
412
+ # @return [self]
413
+ def erase_generics(generics_to_erase)
414
+ return self if generics_to_erase.empty?
415
+ transform_types { |t| t.erase_generics(generics_to_erase) }
416
+ end
417
+
418
+ # @return [String, nil]
419
+ def filename
420
+ return nil if location.nil?
421
+ location.filename
422
+ end
423
+
424
+ # @return [Integer]
425
+ def completion_item_kind
426
+ LanguageServer::CompletionItemKinds::KEYWORD
427
+ end
428
+
429
+ # @return [Integer, nil]
430
+ def symbol_kind
431
+ nil
432
+ end
433
+
434
+ def to_s
435
+ desc
436
+ end
437
+
438
+ # @return [Boolean]
439
+ def variable?
440
+ false
441
+ end
442
+
443
+ # @return [Location, nil]
444
+ def best_location
445
+ location || type_location
446
+ end
447
+
448
+ # True if the specified pin is a near match to this one. A near match
449
+ # indicates that the pins contain mostly the same data. Any differences
450
+ # between them should not have an impact on the API surface.
451
+ #
452
+ # @param other [Solargraph::Pin::Base, Object]
453
+ # @return [Boolean]
454
+ def nearly? other
455
+ self.class == other.class &&
456
+ name == other.name &&
457
+ (closure == other.closure || (closure && closure.nearly?(other.closure))) &&
458
+ (comments == other.comments ||
459
+ (((maybe_directives? == false && other.maybe_directives? == false) || compare_directives(directives, other.directives)) &&
460
+ compare_docstring_tags(docstring, other.docstring))
461
+ )
462
+ end
463
+
464
+ # Pin equality is determined using the #nearly? method and also
465
+ # requiring both pins to have the same location.
466
+ #
467
+ # @param other [Object]
468
+ def == other
469
+ return false unless nearly? other
470
+ # @sg-ignore Should add more explicit type check on other
471
+ comments == other.comments && location == other.location
472
+ end
473
+
474
+ # The pin's return type.
475
+ #
476
+ # @return [ComplexType]
477
+ def return_type
478
+ @return_type ||= ComplexType::UNDEFINED
479
+ end
480
+
481
+ # @return [YARD::Docstring]
482
+ def docstring
483
+ parse_comments unless @docstring
484
+ @docstring ||= Solargraph::Source.parse_docstring('').to_docstring
485
+ end
486
+
487
+ # @return [::Array<YARD::Tags::Directive>]
488
+ def directives
489
+ parse_comments unless @directives
490
+ @directives
491
+ end
492
+
493
+ # @return [::Array<YARD::Tags::MacroDirective>]
494
+ def macros
495
+ @macros ||= collect_macros
496
+ end
497
+
498
+ # Perform a quick check to see if this pin possibly includes YARD
499
+ # directives. This method does not require parsing the comments.
500
+ #
501
+ # After the comments have been parsed, this method will return false if
502
+ # no directives were found, regardless of whether it previously appeared
503
+ # possible.
504
+ #
505
+ # @return [Boolean]
506
+ def maybe_directives?
507
+ return !@directives.empty? if defined?(@directives) && @directives
508
+ @maybe_directives ||= comments.include?('@!')
509
+ end
510
+
511
+ # @return [Boolean]
512
+ def deprecated?
513
+ @deprecated ||= docstring.has_tag?('deprecated')
514
+ end
515
+
516
+ # Get a fully qualified type from the pin's return type.
517
+ #
518
+ # The relative type is determined from YARD documentation (@return,
519
+ # @param, @type, etc.) and its namespaces are fully qualified using the
520
+ # provided ApiMap.
521
+ #
522
+ # @param api_map [ApiMap]
523
+ # @return [ComplexType]
524
+ def typify api_map
525
+ return_type.qualify(api_map, *(closure&.gates || ['']))
526
+ end
527
+
528
+ # Infer the pin's return type via static code analysis.
529
+ #
530
+ # @param api_map [ApiMap]
531
+ # @return [ComplexType]
532
+ def probe api_map
533
+ typify api_map
534
+ end
535
+
536
+ # @deprecated Use #typify and/or #probe instead
537
+ # @param api_map [ApiMap]
538
+ # @return [ComplexType]
539
+ def infer api_map
540
+ Solargraph::Logging.logger.warn "WARNING: Pin #infer methods are deprecated. Use #typify or #probe instead."
541
+ type = typify(api_map)
542
+ return type unless type.undefined?
543
+ probe api_map
544
+ end
545
+
546
+ def proxied?
547
+ @proxied ||= false
548
+ end
549
+
550
+ def probed?
551
+ @probed ||= false
552
+ end
553
+
554
+ # @param api_map [ApiMap]
555
+ # @return [self]
556
+ def realize api_map
557
+ return self if return_type.defined?
558
+ type = typify(api_map)
559
+ return proxy(type) if type.defined?
560
+ type = probe(api_map)
561
+ return self if type.undefined?
562
+ result = proxy(type)
563
+ result.probed = true
564
+ result
565
+ end
566
+
567
+ # Return a proxy for this pin with the specified return type. Other than
568
+ # the return type and the #proxied? setting, the proxy should be a clone
569
+ # of the original.
570
+ #
571
+ # @param return_type [ComplexType]
572
+ # @return [self]
573
+ def proxy return_type
574
+ result = dup
575
+ result.return_type = return_type
576
+ result.proxied = true
577
+ result
578
+ end
579
+
580
+ # @deprecated
581
+ # @return [String]
582
+ def identity
583
+ @identity ||= "#{closure&.path}|#{name}|#{location}"
584
+ end
585
+
586
+ # The namespaces available for resolving the current namespace. Each gate
587
+ # should be a fully qualified namespace or the root namespace (i.e., an
588
+ # empty string.)
589
+ #
590
+ # Example: Given the name 'Bar' and the gates ['Foo', ''],
591
+ # the fully qualified namespace should be 'Foo::Bar' or 'Bar'.
592
+ #
593
+ # @return [Array<String>]
594
+ def gates
595
+ @gates ||= closure&.gates || ['']
596
+ end
597
+
598
+ # @return [String, nil]
599
+ def to_rbs
600
+ return_type.to_rbs
601
+ end
602
+
603
+ # @return [String, nil]
604
+ def type_desc
605
+ rbs = to_rbs
606
+ # RBS doesn't have a way to represent a Class<x> type
607
+ rbs = return_type.rooted_tags if return_type.name == 'Class'
608
+ if path
609
+ if rbs
610
+ path + ' ' + rbs
611
+ else
612
+ path
613
+ end
614
+ else
615
+ rbs
616
+ end
617
+ end
618
+
619
+ # @return [String]
620
+ def inner_desc
621
+ closure_info = closure&.desc
622
+ binder_info = binder&.desc
623
+ "name=#{name.inspect} return_type=#{type_desc}, context=#{context.rooted_tags}, closure=#{closure_info}, binder=#{binder_info}"
624
+ end
625
+
626
+ # @return [String]
627
+ def desc
628
+ "[#{inner_desc}]"
629
+ end
630
+
631
+ # @return [String]
632
+ def inspect
633
+ "#<#{self.class} `#{self.inner_desc}`#{all_location_text} via #{source.inspect}>"
634
+ end
635
+
636
+ # @return [String]
637
+ def all_location_text
638
+ if location.nil? && type_location.nil?
639
+ ''
640
+ elsif !location.nil? && type_location.nil?
641
+ " at #{location.inspect})"
642
+ elsif !type_location.nil? && location.nil?
643
+ " at #{type_location.inspect})"
644
+ else
645
+ " at (#{location.inspect} and #{type_location.inspect})"
646
+ end
647
+ end
648
+
649
+ # @return [void]
650
+ def reset_generated!
651
+ end
652
+
653
+ protected
654
+
655
+ # @return [Boolean]
656
+ attr_writer :probed
657
+
658
+ # @return [Boolean]
659
+ attr_writer :proxied
660
+
661
+ # @return [ComplexType]
662
+ attr_writer :return_type
663
+
664
+ attr_writer :docstring
665
+
666
+ attr_writer :directives
667
+
668
+ private
669
+
670
+ # @return [void]
671
+ def parse_comments
672
+ # HACK: Avoid a NoMethodError on nil with empty overload tags
673
+ if comments.nil? || comments.empty? || comments.strip.end_with?('@overload')
674
+ @docstring = nil
675
+ @directives = []
676
+ else
677
+ # HACK: Pass a dummy code object to the parser for plugins that
678
+ # expect it not to be nil
679
+ parse = Solargraph::Source.parse_docstring(comments)
680
+ @docstring = parse.to_docstring
681
+ @directives = parse.directives
682
+ end
683
+ end
684
+
685
+ # True if two docstrings have the same tags, regardless of any other
686
+ # differences.
687
+ #
688
+ # @param d1 [YARD::Docstring]
689
+ # @param d2 [YARD::Docstring]
690
+ # @return [Boolean]
691
+ def compare_docstring_tags d1, d2
692
+ return false if d1.tags.length != d2.tags.length
693
+ d1.tags.each_index do |i|
694
+ return false unless compare_tags(d1.tags[i], d2.tags[i])
695
+ end
696
+ true
697
+ end
698
+
699
+ # @param dir1 [::Array<YARD::Tags::Directive>]
700
+ # @param dir2 [::Array<YARD::Tags::Directive>]
701
+ # @return [Boolean]
702
+ def compare_directives dir1, dir2
703
+ return false if dir1.length != dir2.length
704
+ dir1.each_index do |i|
705
+ return false unless compare_tags(dir1[i].tag, dir2[i].tag)
706
+ end
707
+ true
708
+ end
709
+
710
+ # @param tag1 [YARD::Tags::Tag]
711
+ # @param tag2 [YARD::Tags::Tag]
712
+ # @return [Boolean]
713
+ def compare_tags tag1, tag2
714
+ tag1.class == tag2.class &&
715
+ tag1.tag_name == tag2.tag_name &&
716
+ tag1.text == tag2.text &&
717
+ tag1.name == tag2.name &&
718
+ tag1.types == tag2.types
719
+ end
720
+
721
+ # @return [::Array<YARD::Tags::Handlers::Directive>]
722
+ def collect_macros
723
+ return [] unless maybe_directives?
724
+ parse = Solargraph::Source.parse_docstring(comments)
725
+ parse.directives.select{ |d| d.tag.tag_name == 'macro' }
726
+ end
727
+ end
728
+ end
729
+ end