solargraph 0.58.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 (150) hide show
  1. checksums.yaml +4 -4
  2. data/.gitattributes +2 -0
  3. data/.gitignore +1 -0
  4. data/CHANGELOG.md +9 -0
  5. data/bin/solargraph +8 -8
  6. data/lib/solargraph/api_map/cache.rb +110 -110
  7. data/lib/solargraph/api_map/constants.rb +279 -279
  8. data/lib/solargraph/api_map/index.rb +193 -193
  9. data/lib/solargraph/api_map/source_to_yard.rb +97 -97
  10. data/lib/solargraph/api_map/store.rb +384 -384
  11. data/lib/solargraph/api_map.rb +945 -945
  12. data/lib/solargraph/complex_type/type_methods.rb +228 -228
  13. data/lib/solargraph/complex_type/unique_type.rb +482 -482
  14. data/lib/solargraph/complex_type.rb +444 -444
  15. data/lib/solargraph/convention/data_definition/data_definition_node.rb +91 -91
  16. data/lib/solargraph/convention/data_definition.rb +105 -105
  17. data/lib/solargraph/convention/struct_definition/struct_assignment_node.rb +61 -61
  18. data/lib/solargraph/convention/struct_definition/struct_definition_node.rb +102 -102
  19. data/lib/solargraph/convention/struct_definition.rb +164 -164
  20. data/lib/solargraph/diagnostics/require_not_found.rb +53 -53
  21. data/lib/solargraph/diagnostics/rubocop.rb +118 -118
  22. data/lib/solargraph/diagnostics/rubocop_helpers.rb +68 -68
  23. data/lib/solargraph/diagnostics/type_check.rb +55 -55
  24. data/lib/solargraph/doc_map.rb +439 -439
  25. data/lib/solargraph/equality.rb +34 -34
  26. data/lib/solargraph/gem_pins.rb +98 -98
  27. data/lib/solargraph/language_server/host/diagnoser.rb +89 -89
  28. data/lib/solargraph/language_server/host/dispatch.rb +130 -130
  29. data/lib/solargraph/language_server/host/message_worker.rb +112 -112
  30. data/lib/solargraph/language_server/host/sources.rb +99 -99
  31. data/lib/solargraph/language_server/host.rb +878 -878
  32. data/lib/solargraph/language_server/message/extended/check_gem_version.rb +114 -114
  33. data/lib/solargraph/language_server/message/extended/document.rb +23 -23
  34. data/lib/solargraph/language_server/message/text_document/completion.rb +56 -56
  35. data/lib/solargraph/language_server/message/text_document/definition.rb +40 -40
  36. data/lib/solargraph/language_server/message/text_document/document_symbol.rb +26 -26
  37. data/lib/solargraph/language_server/message/text_document/formatting.rb +148 -148
  38. data/lib/solargraph/language_server/message/text_document/hover.rb +58 -58
  39. data/lib/solargraph/language_server/message/text_document/signature_help.rb +24 -24
  40. data/lib/solargraph/language_server/message/text_document/type_definition.rb +25 -25
  41. data/lib/solargraph/language_server/message/workspace/workspace_symbol.rb +23 -23
  42. data/lib/solargraph/library.rb +683 -683
  43. data/lib/solargraph/location.rb +82 -82
  44. data/lib/solargraph/logging.rb +37 -37
  45. data/lib/solargraph/parser/comment_ripper.rb +69 -69
  46. data/lib/solargraph/parser/flow_sensitive_typing.rb +255 -255
  47. data/lib/solargraph/parser/node_processor/base.rb +92 -92
  48. data/lib/solargraph/parser/node_processor.rb +62 -62
  49. data/lib/solargraph/parser/parser_gem/class_methods.rb +149 -149
  50. data/lib/solargraph/parser/parser_gem/node_chainer.rb +166 -166
  51. data/lib/solargraph/parser/parser_gem/node_methods.rb +486 -486
  52. data/lib/solargraph/parser/parser_gem/node_processors/and_node.rb +22 -22
  53. data/lib/solargraph/parser/parser_gem/node_processors/args_node.rb +59 -59
  54. data/lib/solargraph/parser/parser_gem/node_processors/begin_node.rb +15 -15
  55. data/lib/solargraph/parser/parser_gem/node_processors/block_node.rb +46 -46
  56. data/lib/solargraph/parser/parser_gem/node_processors/def_node.rb +53 -53
  57. data/lib/solargraph/parser/parser_gem/node_processors/if_node.rb +23 -23
  58. data/lib/solargraph/parser/parser_gem/node_processors/ivasgn_node.rb +40 -40
  59. data/lib/solargraph/parser/parser_gem/node_processors/lvasgn_node.rb +29 -29
  60. data/lib/solargraph/parser/parser_gem/node_processors/masgn_node.rb +59 -59
  61. data/lib/solargraph/parser/parser_gem/node_processors/opasgn_node.rb +98 -98
  62. data/lib/solargraph/parser/parser_gem/node_processors/orasgn_node.rb +17 -17
  63. data/lib/solargraph/parser/parser_gem/node_processors/resbody_node.rb +38 -38
  64. data/lib/solargraph/parser/parser_gem/node_processors/sclass_node.rb +52 -52
  65. data/lib/solargraph/parser/parser_gem/node_processors/send_node.rb +291 -291
  66. data/lib/solargraph/parser/parser_gem/node_processors/while_node.rb +29 -29
  67. data/lib/solargraph/parser/parser_gem/node_processors.rb +70 -70
  68. data/lib/solargraph/parser/region.rb +69 -69
  69. data/lib/solargraph/parser/snippet.rb +17 -17
  70. data/lib/solargraph/pin/base.rb +729 -729
  71. data/lib/solargraph/pin/base_variable.rb +126 -126
  72. data/lib/solargraph/pin/block.rb +104 -104
  73. data/lib/solargraph/pin/breakable.rb +9 -9
  74. data/lib/solargraph/pin/callable.rb +231 -231
  75. data/lib/solargraph/pin/closure.rb +72 -72
  76. data/lib/solargraph/pin/common.rb +79 -79
  77. data/lib/solargraph/pin/conversions.rb +123 -123
  78. data/lib/solargraph/pin/delegated_method.rb +120 -120
  79. data/lib/solargraph/pin/documenting.rb +114 -114
  80. data/lib/solargraph/pin/instance_variable.rb +34 -34
  81. data/lib/solargraph/pin/keyword.rb +20 -20
  82. data/lib/solargraph/pin/local_variable.rb +75 -75
  83. data/lib/solargraph/pin/method.rb +672 -672
  84. data/lib/solargraph/pin/method_alias.rb +34 -34
  85. data/lib/solargraph/pin/namespace.rb +115 -115
  86. data/lib/solargraph/pin/parameter.rb +275 -275
  87. data/lib/solargraph/pin/proxy_type.rb +39 -39
  88. data/lib/solargraph/pin/reference/override.rb +47 -47
  89. data/lib/solargraph/pin/reference/superclass.rb +15 -15
  90. data/lib/solargraph/pin/reference.rb +39 -39
  91. data/lib/solargraph/pin/search.rb +61 -61
  92. data/lib/solargraph/pin/signature.rb +61 -61
  93. data/lib/solargraph/pin/symbol.rb +53 -53
  94. data/lib/solargraph/pin/until.rb +18 -18
  95. data/lib/solargraph/pin/while.rb +18 -18
  96. data/lib/solargraph/pin.rb +44 -44
  97. data/lib/solargraph/pin_cache.rb +245 -245
  98. data/lib/solargraph/position.rb +132 -119
  99. data/lib/solargraph/range.rb +112 -112
  100. data/lib/solargraph/rbs_map/conversions.rb +823 -823
  101. data/lib/solargraph/rbs_map/core_map.rb +58 -58
  102. data/lib/solargraph/rbs_map/stdlib_map.rb +43 -43
  103. data/lib/solargraph/rbs_map.rb +163 -163
  104. data/lib/solargraph/shell.rb +352 -352
  105. data/lib/solargraph/source/chain/call.rb +337 -337
  106. data/lib/solargraph/source/chain/constant.rb +26 -26
  107. data/lib/solargraph/source/chain/hash.rb +34 -34
  108. data/lib/solargraph/source/chain/if.rb +28 -28
  109. data/lib/solargraph/source/chain/instance_variable.rb +13 -13
  110. data/lib/solargraph/source/chain/literal.rb +48 -48
  111. data/lib/solargraph/source/chain/or.rb +23 -23
  112. data/lib/solargraph/source/chain.rb +291 -291
  113. data/lib/solargraph/source/change.rb +82 -82
  114. data/lib/solargraph/source/cursor.rb +166 -166
  115. data/lib/solargraph/source/encoding_fixes.rb +23 -23
  116. data/lib/solargraph/source/source_chainer.rb +194 -194
  117. data/lib/solargraph/source/updater.rb +55 -55
  118. data/lib/solargraph/source.rb +498 -498
  119. data/lib/solargraph/source_map/clip.rb +226 -226
  120. data/lib/solargraph/source_map/data.rb +34 -34
  121. data/lib/solargraph/source_map/mapper.rb +259 -259
  122. data/lib/solargraph/source_map.rb +212 -212
  123. data/lib/solargraph/type_checker/checks.rb +124 -124
  124. data/lib/solargraph/type_checker/param_def.rb +37 -37
  125. data/lib/solargraph/type_checker/problem.rb +32 -32
  126. data/lib/solargraph/type_checker/rules.rb +84 -84
  127. data/lib/solargraph/type_checker.rb +814 -814
  128. data/lib/solargraph/version.rb +5 -5
  129. data/lib/solargraph/workspace/config.rb +255 -255
  130. data/lib/solargraph/workspace/require_paths.rb +97 -97
  131. data/lib/solargraph/workspace.rb +220 -220
  132. data/lib/solargraph/yard_map/helpers.rb +44 -44
  133. data/lib/solargraph/yard_map/mapper/to_method.rb +130 -130
  134. data/lib/solargraph/yard_map/mapper/to_namespace.rb +31 -31
  135. data/lib/solargraph/yard_map/mapper.rb +79 -79
  136. data/lib/solargraph/yard_map/to_method.rb +89 -89
  137. data/lib/solargraph/yardoc.rb +87 -87
  138. data/lib/solargraph.rb +105 -105
  139. data/rbs_collection.yaml +1 -1
  140. metadata +13 -12
  141. /data/{sig → rbs}/shims/ast/0/node.rbs +0 -0
  142. /data/{sig → rbs}/shims/ast/2.4/.rbs_meta.yaml +0 -0
  143. /data/{sig → rbs}/shims/ast/2.4/ast.rbs +0 -0
  144. /data/{sig → rbs}/shims/parser/3.2.0.1/builders/default.rbs +0 -0
  145. /data/{sig → rbs}/shims/parser/3.2.0.1/manifest.yaml +0 -0
  146. /data/{sig → rbs}/shims/parser/3.2.0.1/parser.rbs +0 -0
  147. /data/{sig → rbs}/shims/parser/3.2.0.1/polyfill.rbs +0 -0
  148. /data/{sig → rbs}/shims/thor/1.2.0.1/.rbs_meta.yaml +0 -0
  149. /data/{sig → rbs}/shims/thor/1.2.0.1/manifest.yaml +0 -0
  150. /data/{sig → rbs}/shims/thor/1.2.0.1/thor.rbs +0 -0
@@ -1,291 +1,291 @@
1
- # frozen_string_literal: true
2
-
3
- # HACK: Fix autoload issue
4
- require 'solargraph/source/chain/link'
5
-
6
- module Solargraph
7
- class Source
8
- #
9
- # Represents an expression as a single call chain at the parse
10
- # tree level, made up of constants, variables, and method calls,
11
- # each represented as a Link object.
12
- #
13
- # Computes Pins and/or ComplexTypes representing the interpreted
14
- # expression.
15
- #
16
- class Chain
17
- include Equality
18
- include Logging
19
-
20
- autoload :Link, 'solargraph/source/chain/link'
21
- autoload :Call, 'solargraph/source/chain/call'
22
- autoload :QCall, 'solargraph/source/chain/q_call'
23
- autoload :Variable, 'solargraph/source/chain/variable'
24
- autoload :ClassVariable, 'solargraph/source/chain/class_variable'
25
- autoload :Constant, 'solargraph/source/chain/constant'
26
- autoload :InstanceVariable, 'solargraph/source/chain/instance_variable'
27
- autoload :GlobalVariable, 'solargraph/source/chain/global_variable'
28
- autoload :Literal, 'solargraph/source/chain/literal'
29
- autoload :Head, 'solargraph/source/chain/head'
30
- autoload :If, 'solargraph/source/chain/if'
31
- autoload :Or, 'solargraph/source/chain/or'
32
- autoload :BlockVariable, 'solargraph/source/chain/block_variable'
33
- autoload :BlockSymbol, 'solargraph/source/chain/block_symbol'
34
- autoload :ZSuper, 'solargraph/source/chain/z_super'
35
- autoload :Hash, 'solargraph/source/chain/hash'
36
- autoload :Array, 'solargraph/source/chain/array'
37
-
38
- @@inference_stack = []
39
- @@inference_depth = 0
40
- @@inference_invalidation_key = nil
41
- @@inference_cache = {}
42
-
43
- UNDEFINED_CALL = Chain::Call.new('<undefined>', nil)
44
- UNDEFINED_CONSTANT = Chain::Constant.new('<undefined>')
45
-
46
- # @return [::Array<Source::Chain::Link>]
47
- attr_reader :links
48
-
49
- attr_reader :node
50
-
51
- # @sg-ignore Fix "Not enough arguments to Module#protected"
52
- protected def equality_fields
53
- [links, node]
54
- end
55
-
56
- # @param node [Parser::AST::Node, nil]
57
- # @param links [::Array<Chain::Link>]
58
- # @param splat [Boolean]
59
- def initialize links, node = nil, splat = false
60
- @links = links.clone
61
- @links.push UNDEFINED_CALL if @links.empty?
62
- head = true
63
- @links.map! do |link|
64
- result = (head ? link.clone_head : link.clone_body)
65
- head = false
66
- result
67
- end
68
- @node = node
69
- @splat = splat
70
- end
71
-
72
- # @return [Chain]
73
- def base
74
- @base ||= Chain.new(links[0..-2])
75
- end
76
-
77
- # Determine potential Pins returned by this chain of words
78
- #
79
- # @param api_map [ApiMap]
80
- #
81
- # @param name_pin [Pin::Base] A pin
82
- # representing the place in which expression is evaluated (e.g.,
83
- # a Method pin, or a Module or Class pin if not run within a
84
- # method - both in terms of the closure around the chain, as well
85
- # as the self type used for any method calls in head position.
86
- #
87
- # Requirements for name_pin:
88
- #
89
- # * name_pin.context: This should be a type representing the
90
- # namespace where we can look up non-local variables and
91
- # method names. If it is a Class<X>, we will look up
92
- # :class scoped methods/variables.
93
- #
94
- # * name_pin.binder: Used for method call lookups only
95
- # (Chain::Call links). For method calls as the first
96
- # element in the chain, 'name_pin.binder' should be the
97
- # same as name_pin.context above. For method calls later
98
- # in the chain (e.g., 'b' in a.b.c), it should represent
99
- # 'a'.
100
- #
101
- # @param locals [::Array<Pin::LocalVariable>] Any local
102
- # variables / method parameters etc visible by the statement
103
- #
104
- # @return [::Array<Pin::Base>] Pins representing possible return
105
- # types of this method.
106
- def define api_map, name_pin, locals
107
- return [] if undefined?
108
-
109
- # working_pin is the surrounding closure pin for the link
110
- # being processed, whose #binder method will provide the LHS /
111
- # 'self type' of the next link (same as the #return_type method
112
- # --the type of the result so far).
113
- #
114
- # @todo ProxyType uses 'type' for the binder, but '
115
- working_pin = name_pin
116
- links[0..-2].each do |link|
117
- pins = link.resolve(api_map, working_pin, locals)
118
- type = infer_from_definitions(pins, working_pin, api_map, locals)
119
- if type.undefined?
120
- logger.debug { "Chain#define(links=#{links.map(&:desc)}, name_pin=#{name_pin.inspect}, locals=#{locals}) => [] - undefined type from #{link.desc}" }
121
- return []
122
- end
123
- # We continue to use the context from the head pin, in case
124
- # we need it to, for instance, provide context for a block
125
- # evaluation. However, we use the last link's return type
126
- # for the binder, as this is chaining off of it, and the
127
- # binder is now the lhs of the rhs we are evaluating.
128
- working_pin = Pin::ProxyType.anonymous(name_pin.context, binder: type, closure: name_pin, source: :chain)
129
- logger.debug { "Chain#define(links=#{links.map(&:desc)}, name_pin=#{name_pin.inspect}, locals=#{locals}) - after processing #{link.desc}, new working_pin=#{working_pin} with binder #{working_pin.binder}" }
130
- end
131
- links.last.last_context = working_pin
132
- links.last.resolve(api_map, working_pin, locals)
133
- end
134
-
135
- # @param api_map [ApiMap]
136
- # @param name_pin [Pin::Base] The pin for the closure in which this code runs
137
- # @param locals [::Array<Pin::LocalVariable>]
138
- # @return [ComplexType]
139
- # @sg-ignore
140
- def infer api_map, name_pin, locals
141
- cache_key = [node, node&.location, links, name_pin&.return_type, locals]
142
- if @@inference_invalidation_key == api_map.hash
143
- cached = @@inference_cache[cache_key]
144
- return cached if cached
145
- else
146
- @@inference_invalidation_key = api_map.hash
147
- @@inference_cache = {}
148
- end
149
- out = infer_uncached(api_map, name_pin, locals).downcast_to_literal_if_possible
150
- logger.debug { "Chain#infer() - caching result - cache_key_hash=#{cache_key.hash}, links.map(&:hash)=#{links.map(&:hash)}, links=#{links}, cache_key.map(&:hash) = #{cache_key.map(&:hash)}, cache_key=#{cache_key}" }
151
- @@inference_cache[cache_key] = out
152
- end
153
-
154
- # @param api_map [ApiMap]
155
- # @param name_pin [Pin::Base]
156
- # @param locals [::Array<Pin::LocalVariable>]
157
- # @return [ComplexType]
158
- def infer_uncached api_map, name_pin, locals
159
- pins = define(api_map, name_pin, locals)
160
- if pins.empty?
161
- logger.debug { "Chain#infer_uncached(links=#{links.map(&:desc)}, locals=#{locals.map(&:desc)}) => undefined - no pins" }
162
- return ComplexType::UNDEFINED
163
- end
164
- type = infer_from_definitions(pins, links.last.last_context, api_map, locals)
165
- out = maybe_nil(type)
166
- logger.debug { "Chain#infer_uncached(links=#{self.links.map(&:desc)}, locals=#{locals.map(&:desc)}, name_pin=#{name_pin}, name_pin.closure=#{name_pin.closure.inspect}, name_pin.binder=#{name_pin.binder}) => #{out.rooted_tags.inspect}" }
167
- out
168
- end
169
-
170
- # @return [Boolean]
171
- def literal?
172
- links.last.is_a?(Chain::Literal)
173
- end
174
-
175
- def undefined?
176
- links.any?(&:undefined?)
177
- end
178
-
179
- def defined?
180
- !undefined?
181
- end
182
-
183
- # @return [Boolean]
184
- def constant?
185
- links.last.is_a?(Chain::Constant)
186
- end
187
-
188
- def splat?
189
- @splat
190
- end
191
-
192
- def nullable?
193
- links.any?(&:nullable?)
194
- end
195
-
196
- include Logging
197
-
198
- # @return [String]
199
- def desc
200
- links.map(&:desc).to_s
201
- end
202
-
203
- def to_s
204
- desc
205
- end
206
-
207
- include Logging
208
-
209
- private
210
-
211
- # @param pins [::Array<Pin::Base>]
212
- # @param context [Pin::Base]
213
- # @param api_map [ApiMap]
214
- # @param locals [::Enumerable<Pin::LocalVariable>]
215
- # @return [ComplexType]
216
- def infer_from_definitions pins, context, api_map, locals
217
- # @type [::Array<ComplexType>]
218
- types = []
219
- unresolved_pins = []
220
- # @todo this param tag shouldn't be needed to probe the type
221
- # @todo ...but given it is needed, typecheck should complain that it is needed
222
- # @param pin [Pin::Base]
223
- pins.each do |pin|
224
- # Avoid infinite recursion
225
- next if @@inference_stack.include?(pin)
226
-
227
- @@inference_stack.push pin
228
- type = pin.typify(api_map)
229
- @@inference_stack.pop
230
- if type.defined?
231
- if type.generic?
232
- # @todo even at strong, no typechecking complaint
233
- # happens when a [Pin::Base,nil] is passed into a method
234
- # that accepts only [Pin::Namespace] as an argument
235
- type = type.resolve_generics(pin.closure, context.binder)
236
- end
237
- types << type
238
- else
239
- unresolved_pins << pin
240
- end
241
- end
242
-
243
- # Limit method inference recursion
244
- if @@inference_depth >= 10 && pins.first.is_a?(Pin::Method)
245
- return ComplexType::UNDEFINED
246
- end
247
-
248
- @@inference_depth += 1
249
- # @param pin [Pin::Base]
250
- unresolved_pins.each do |pin|
251
- # Avoid infinite recursion
252
- if @@inference_stack.include?(pin.identity)
253
- next
254
- end
255
-
256
- @@inference_stack.push(pin.identity)
257
- type = pin.probe(api_map)
258
- @@inference_stack.pop
259
- types.push type if type
260
- end
261
- @@inference_depth -= 1
262
-
263
- type = if types.empty?
264
- ComplexType::UNDEFINED
265
- elsif types.length > 1
266
- # Move nil to the end by convention
267
-
268
- # @param a [ComplexType::UniqueType]
269
- sorted = types.flat_map(&:items).sort { |a, _| a.tag == 'nil' ? 1 : 0 }
270
- ComplexType.new(sorted.uniq)
271
- else
272
- ComplexType.new(types)
273
- end
274
- if context.nil? || context.return_type.undefined?
275
- # up to downstream to resolve self type
276
- return type
277
- end
278
-
279
- type.self_to_type(context.return_type)
280
- end
281
-
282
- # @param type [ComplexType]
283
- # @return [ComplexType]
284
- def maybe_nil type
285
- return type if type.undefined? || type.void? || type.nullable?
286
- return type unless nullable?
287
- ComplexType.new(type.items + [ComplexType::NIL])
288
- end
289
- end
290
- end
291
- end
1
+ # frozen_string_literal: true
2
+
3
+ # HACK: Fix autoload issue
4
+ require 'solargraph/source/chain/link'
5
+
6
+ module Solargraph
7
+ class Source
8
+ #
9
+ # Represents an expression as a single call chain at the parse
10
+ # tree level, made up of constants, variables, and method calls,
11
+ # each represented as a Link object.
12
+ #
13
+ # Computes Pins and/or ComplexTypes representing the interpreted
14
+ # expression.
15
+ #
16
+ class Chain
17
+ include Equality
18
+ include Logging
19
+
20
+ autoload :Link, 'solargraph/source/chain/link'
21
+ autoload :Call, 'solargraph/source/chain/call'
22
+ autoload :QCall, 'solargraph/source/chain/q_call'
23
+ autoload :Variable, 'solargraph/source/chain/variable'
24
+ autoload :ClassVariable, 'solargraph/source/chain/class_variable'
25
+ autoload :Constant, 'solargraph/source/chain/constant'
26
+ autoload :InstanceVariable, 'solargraph/source/chain/instance_variable'
27
+ autoload :GlobalVariable, 'solargraph/source/chain/global_variable'
28
+ autoload :Literal, 'solargraph/source/chain/literal'
29
+ autoload :Head, 'solargraph/source/chain/head'
30
+ autoload :If, 'solargraph/source/chain/if'
31
+ autoload :Or, 'solargraph/source/chain/or'
32
+ autoload :BlockVariable, 'solargraph/source/chain/block_variable'
33
+ autoload :BlockSymbol, 'solargraph/source/chain/block_symbol'
34
+ autoload :ZSuper, 'solargraph/source/chain/z_super'
35
+ autoload :Hash, 'solargraph/source/chain/hash'
36
+ autoload :Array, 'solargraph/source/chain/array'
37
+
38
+ @@inference_stack = []
39
+ @@inference_depth = 0
40
+ @@inference_invalidation_key = nil
41
+ @@inference_cache = {}
42
+
43
+ UNDEFINED_CALL = Chain::Call.new('<undefined>', nil)
44
+ UNDEFINED_CONSTANT = Chain::Constant.new('<undefined>')
45
+
46
+ # @return [::Array<Source::Chain::Link>]
47
+ attr_reader :links
48
+
49
+ attr_reader :node
50
+
51
+ # @sg-ignore Fix "Not enough arguments to Module#protected"
52
+ protected def equality_fields
53
+ [links, node]
54
+ end
55
+
56
+ # @param node [Parser::AST::Node, nil]
57
+ # @param links [::Array<Chain::Link>]
58
+ # @param splat [Boolean]
59
+ def initialize links, node = nil, splat = false
60
+ @links = links.clone
61
+ @links.push UNDEFINED_CALL if @links.empty?
62
+ head = true
63
+ @links.map! do |link|
64
+ result = (head ? link.clone_head : link.clone_body)
65
+ head = false
66
+ result
67
+ end
68
+ @node = node
69
+ @splat = splat
70
+ end
71
+
72
+ # @return [Chain]
73
+ def base
74
+ @base ||= Chain.new(links[0..-2])
75
+ end
76
+
77
+ # Determine potential Pins returned by this chain of words
78
+ #
79
+ # @param api_map [ApiMap]
80
+ #
81
+ # @param name_pin [Pin::Base] A pin
82
+ # representing the place in which expression is evaluated (e.g.,
83
+ # a Method pin, or a Module or Class pin if not run within a
84
+ # method - both in terms of the closure around the chain, as well
85
+ # as the self type used for any method calls in head position.
86
+ #
87
+ # Requirements for name_pin:
88
+ #
89
+ # * name_pin.context: This should be a type representing the
90
+ # namespace where we can look up non-local variables and
91
+ # method names. If it is a Class<X>, we will look up
92
+ # :class scoped methods/variables.
93
+ #
94
+ # * name_pin.binder: Used for method call lookups only
95
+ # (Chain::Call links). For method calls as the first
96
+ # element in the chain, 'name_pin.binder' should be the
97
+ # same as name_pin.context above. For method calls later
98
+ # in the chain (e.g., 'b' in a.b.c), it should represent
99
+ # 'a'.
100
+ #
101
+ # @param locals [::Array<Pin::LocalVariable>] Any local
102
+ # variables / method parameters etc visible by the statement
103
+ #
104
+ # @return [::Array<Pin::Base>] Pins representing possible return
105
+ # types of this method.
106
+ def define api_map, name_pin, locals
107
+ return [] if undefined?
108
+
109
+ # working_pin is the surrounding closure pin for the link
110
+ # being processed, whose #binder method will provide the LHS /
111
+ # 'self type' of the next link (same as the #return_type method
112
+ # --the type of the result so far).
113
+ #
114
+ # @todo ProxyType uses 'type' for the binder, but '
115
+ working_pin = name_pin
116
+ links[0..-2].each do |link|
117
+ pins = link.resolve(api_map, working_pin, locals)
118
+ type = infer_from_definitions(pins, working_pin, api_map, locals)
119
+ if type.undefined?
120
+ logger.debug { "Chain#define(links=#{links.map(&:desc)}, name_pin=#{name_pin.inspect}, locals=#{locals}) => [] - undefined type from #{link.desc}" }
121
+ return []
122
+ end
123
+ # We continue to use the context from the head pin, in case
124
+ # we need it to, for instance, provide context for a block
125
+ # evaluation. However, we use the last link's return type
126
+ # for the binder, as this is chaining off of it, and the
127
+ # binder is now the lhs of the rhs we are evaluating.
128
+ working_pin = Pin::ProxyType.anonymous(name_pin.context, binder: type, closure: name_pin, source: :chain)
129
+ logger.debug { "Chain#define(links=#{links.map(&:desc)}, name_pin=#{name_pin.inspect}, locals=#{locals}) - after processing #{link.desc}, new working_pin=#{working_pin} with binder #{working_pin.binder}" }
130
+ end
131
+ links.last.last_context = working_pin
132
+ links.last.resolve(api_map, working_pin, locals)
133
+ end
134
+
135
+ # @param api_map [ApiMap]
136
+ # @param name_pin [Pin::Base] The pin for the closure in which this code runs
137
+ # @param locals [::Array<Pin::LocalVariable>]
138
+ # @return [ComplexType]
139
+ # @sg-ignore
140
+ def infer api_map, name_pin, locals
141
+ cache_key = [node, node&.location, links, name_pin&.return_type, locals]
142
+ if @@inference_invalidation_key == api_map.hash
143
+ cached = @@inference_cache[cache_key]
144
+ return cached if cached
145
+ else
146
+ @@inference_invalidation_key = api_map.hash
147
+ @@inference_cache = {}
148
+ end
149
+ out = infer_uncached(api_map, name_pin, locals).downcast_to_literal_if_possible
150
+ logger.debug { "Chain#infer() - caching result - cache_key_hash=#{cache_key.hash}, links.map(&:hash)=#{links.map(&:hash)}, links=#{links}, cache_key.map(&:hash) = #{cache_key.map(&:hash)}, cache_key=#{cache_key}" }
151
+ @@inference_cache[cache_key] = out
152
+ end
153
+
154
+ # @param api_map [ApiMap]
155
+ # @param name_pin [Pin::Base]
156
+ # @param locals [::Array<Pin::LocalVariable>]
157
+ # @return [ComplexType]
158
+ def infer_uncached api_map, name_pin, locals
159
+ pins = define(api_map, name_pin, locals)
160
+ if pins.empty?
161
+ logger.debug { "Chain#infer_uncached(links=#{links.map(&:desc)}, locals=#{locals.map(&:desc)}) => undefined - no pins" }
162
+ return ComplexType::UNDEFINED
163
+ end
164
+ type = infer_from_definitions(pins, links.last.last_context, api_map, locals)
165
+ out = maybe_nil(type)
166
+ logger.debug { "Chain#infer_uncached(links=#{self.links.map(&:desc)}, locals=#{locals.map(&:desc)}, name_pin=#{name_pin}, name_pin.closure=#{name_pin.closure.inspect}, name_pin.binder=#{name_pin.binder}) => #{out.rooted_tags.inspect}" }
167
+ out
168
+ end
169
+
170
+ # @return [Boolean]
171
+ def literal?
172
+ links.last.is_a?(Chain::Literal)
173
+ end
174
+
175
+ def undefined?
176
+ links.any?(&:undefined?)
177
+ end
178
+
179
+ def defined?
180
+ !undefined?
181
+ end
182
+
183
+ # @return [Boolean]
184
+ def constant?
185
+ links.last.is_a?(Chain::Constant)
186
+ end
187
+
188
+ def splat?
189
+ @splat
190
+ end
191
+
192
+ def nullable?
193
+ links.any?(&:nullable?)
194
+ end
195
+
196
+ include Logging
197
+
198
+ # @return [String]
199
+ def desc
200
+ links.map(&:desc).to_s
201
+ end
202
+
203
+ def to_s
204
+ desc
205
+ end
206
+
207
+ include Logging
208
+
209
+ private
210
+
211
+ # @param pins [::Array<Pin::Base>]
212
+ # @param context [Pin::Base]
213
+ # @param api_map [ApiMap]
214
+ # @param locals [::Enumerable<Pin::LocalVariable>]
215
+ # @return [ComplexType]
216
+ def infer_from_definitions pins, context, api_map, locals
217
+ # @type [::Array<ComplexType>]
218
+ types = []
219
+ unresolved_pins = []
220
+ # @todo this param tag shouldn't be needed to probe the type
221
+ # @todo ...but given it is needed, typecheck should complain that it is needed
222
+ # @param pin [Pin::Base]
223
+ pins.each do |pin|
224
+ # Avoid infinite recursion
225
+ next if @@inference_stack.include?(pin)
226
+
227
+ @@inference_stack.push pin
228
+ type = pin.typify(api_map)
229
+ @@inference_stack.pop
230
+ if type.defined?
231
+ if type.generic?
232
+ # @todo even at strong, no typechecking complaint
233
+ # happens when a [Pin::Base,nil] is passed into a method
234
+ # that accepts only [Pin::Namespace] as an argument
235
+ type = type.resolve_generics(pin.closure, context.binder)
236
+ end
237
+ types << type
238
+ else
239
+ unresolved_pins << pin
240
+ end
241
+ end
242
+
243
+ # Limit method inference recursion
244
+ if @@inference_depth >= 10 && pins.first.is_a?(Pin::Method)
245
+ return ComplexType::UNDEFINED
246
+ end
247
+
248
+ @@inference_depth += 1
249
+ # @param pin [Pin::Base]
250
+ unresolved_pins.each do |pin|
251
+ # Avoid infinite recursion
252
+ if @@inference_stack.include?(pin.identity)
253
+ next
254
+ end
255
+
256
+ @@inference_stack.push(pin.identity)
257
+ type = pin.probe(api_map)
258
+ @@inference_stack.pop
259
+ types.push type if type
260
+ end
261
+ @@inference_depth -= 1
262
+
263
+ type = if types.empty?
264
+ ComplexType::UNDEFINED
265
+ elsif types.length > 1
266
+ # Move nil to the end by convention
267
+
268
+ # @param a [ComplexType::UniqueType]
269
+ sorted = types.flat_map(&:items).sort { |a, _| a.tag == 'nil' ? 1 : 0 }
270
+ ComplexType.new(sorted.uniq)
271
+ else
272
+ ComplexType.new(types)
273
+ end
274
+ if context.nil? || context.return_type.undefined?
275
+ # up to downstream to resolve self type
276
+ return type
277
+ end
278
+
279
+ type.self_to_type(context.return_type)
280
+ end
281
+
282
+ # @param type [ComplexType]
283
+ # @return [ComplexType]
284
+ def maybe_nil type
285
+ return type if type.undefined? || type.void? || type.nullable?
286
+ return type unless nullable?
287
+ ComplexType.new(type.items + [ComplexType::NIL])
288
+ end
289
+ end
290
+ end
291
+ end