solargraph 0.50.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 (264) 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 +218 -0
  5. data/.github/workflows/rspec.yml +58 -12
  6. data/.github/workflows/typecheck.yml +39 -0
  7. data/.gitignore +8 -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 +3 -2
  13. data/CHANGELOG.md +306 -3
  14. data/README.md +29 -18
  15. data/Rakefile +125 -13
  16. data/SPONSORS.md +2 -9
  17. data/bin/solargraph +3 -0
  18. data/lib/solargraph/api_map/cache.rb +110 -70
  19. data/lib/solargraph/api_map/constants.rb +279 -0
  20. data/lib/solargraph/api_map/index.rb +193 -0
  21. data/lib/solargraph/api_map/source_to_yard.rb +97 -81
  22. data/lib/solargraph/api_map/store.rb +384 -268
  23. data/lib/solargraph/api_map.rb +945 -704
  24. data/lib/solargraph/bench.rb +21 -3
  25. data/lib/solargraph/complex_type/type_methods.rb +228 -134
  26. data/lib/solargraph/complex_type/unique_type.rb +482 -132
  27. data/lib/solargraph/complex_type.rb +444 -254
  28. data/lib/solargraph/convention/active_support_concern.rb +111 -0
  29. data/lib/solargraph/convention/base.rb +20 -3
  30. data/lib/solargraph/convention/data_definition/data_assignment_node.rb +61 -0
  31. data/lib/solargraph/convention/data_definition/data_definition_node.rb +91 -0
  32. data/lib/solargraph/convention/data_definition.rb +105 -0
  33. data/lib/solargraph/convention/gemspec.rb +3 -2
  34. data/lib/solargraph/convention/struct_definition/struct_assignment_node.rb +61 -0
  35. data/lib/solargraph/convention/struct_definition/struct_definition_node.rb +102 -0
  36. data/lib/solargraph/convention/struct_definition.rb +164 -0
  37. data/lib/solargraph/convention.rb +36 -7
  38. data/lib/solargraph/converters/dd.rb +5 -0
  39. data/lib/solargraph/converters/dl.rb +3 -0
  40. data/lib/solargraph/converters/dt.rb +3 -0
  41. data/lib/solargraph/diagnostics/require_not_found.rb +53 -53
  42. data/lib/solargraph/diagnostics/rubocop.rb +118 -112
  43. data/lib/solargraph/diagnostics/rubocop_helpers.rb +68 -65
  44. data/lib/solargraph/diagnostics/type_check.rb +55 -54
  45. data/lib/solargraph/diagnostics.rb +2 -2
  46. data/lib/solargraph/doc_map.rb +439 -0
  47. data/lib/solargraph/environ.rb +9 -2
  48. data/lib/solargraph/equality.rb +34 -0
  49. data/lib/solargraph/gem_pins.rb +98 -0
  50. data/lib/solargraph/language_server/host/diagnoser.rb +89 -89
  51. data/lib/solargraph/language_server/host/dispatch.rb +130 -111
  52. data/lib/solargraph/language_server/host/message_worker.rb +112 -59
  53. data/lib/solargraph/language_server/host/sources.rb +99 -156
  54. data/lib/solargraph/language_server/host.rb +878 -869
  55. data/lib/solargraph/language_server/message/base.rb +20 -12
  56. data/lib/solargraph/language_server/message/completion_item/resolve.rb +3 -1
  57. data/lib/solargraph/language_server/message/extended/check_gem_version.rb +114 -100
  58. data/lib/solargraph/language_server/message/extended/document.rb +23 -20
  59. data/lib/solargraph/language_server/message/extended/document_gems.rb +3 -3
  60. data/lib/solargraph/language_server/message/initialize.rb +28 -1
  61. data/lib/solargraph/language_server/message/initialized.rb +1 -0
  62. data/lib/solargraph/language_server/message/text_document/completion.rb +56 -59
  63. data/lib/solargraph/language_server/message/text_document/definition.rb +40 -38
  64. data/lib/solargraph/language_server/message/text_document/document_symbol.rb +26 -23
  65. data/lib/solargraph/language_server/message/text_document/formatting.rb +148 -126
  66. data/lib/solargraph/language_server/message/text_document/hover.rb +58 -56
  67. data/lib/solargraph/language_server/message/text_document/signature_help.rb +24 -24
  68. data/lib/solargraph/language_server/message/text_document/type_definition.rb +25 -0
  69. data/lib/solargraph/language_server/message/text_document.rb +1 -1
  70. data/lib/solargraph/language_server/message/workspace/did_change_configuration.rb +5 -0
  71. data/lib/solargraph/language_server/message/workspace/did_change_workspace_folders.rb +2 -0
  72. data/lib/solargraph/language_server/message/workspace/workspace_symbol.rb +23 -23
  73. data/lib/solargraph/language_server/message.rb +1 -0
  74. data/lib/solargraph/language_server/progress.rb +143 -0
  75. data/lib/solargraph/language_server/request.rb +4 -1
  76. data/lib/solargraph/language_server/transport/adapter.rb +16 -1
  77. data/lib/solargraph/language_server/transport/data_reader.rb +2 -0
  78. data/lib/solargraph/language_server.rb +1 -0
  79. data/lib/solargraph/library.rb +683 -551
  80. data/lib/solargraph/location.rb +82 -37
  81. data/lib/solargraph/logging.rb +37 -27
  82. data/lib/solargraph/page.rb +9 -0
  83. data/lib/solargraph/parser/comment_ripper.rb +69 -52
  84. data/lib/solargraph/parser/flow_sensitive_typing.rb +255 -0
  85. data/lib/solargraph/parser/node_processor/base.rb +92 -77
  86. data/lib/solargraph/parser/node_processor.rb +62 -43
  87. data/lib/solargraph/parser/{legacy → parser_gem}/class_methods.rb +149 -135
  88. data/lib/solargraph/parser/{legacy → parser_gem}/flawed_builder.rb +4 -1
  89. data/lib/solargraph/parser/{legacy → parser_gem}/node_chainer.rb +166 -148
  90. data/lib/solargraph/parser/{legacy → parser_gem}/node_methods.rb +486 -325
  91. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/alias_node.rb +3 -2
  92. data/lib/solargraph/parser/parser_gem/node_processors/and_node.rb +22 -0
  93. data/lib/solargraph/parser/parser_gem/node_processors/args_node.rb +59 -0
  94. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/begin_node.rb +15 -15
  95. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/block_node.rb +46 -42
  96. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/casgn_node.rb +4 -3
  97. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/cvasgn_node.rb +3 -2
  98. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/def_node.rb +53 -63
  99. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/defs_node.rb +4 -3
  100. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/gvasgn_node.rb +3 -2
  101. data/lib/solargraph/parser/parser_gem/node_processors/if_node.rb +23 -0
  102. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/ivasgn_node.rb +40 -38
  103. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/lvasgn_node.rb +29 -28
  104. data/lib/solargraph/parser/parser_gem/node_processors/masgn_node.rb +59 -0
  105. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/namespace_node.rb +10 -9
  106. data/lib/solargraph/parser/parser_gem/node_processors/opasgn_node.rb +98 -0
  107. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/orasgn_node.rb +17 -16
  108. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/resbody_node.rb +38 -36
  109. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/sclass_node.rb +52 -42
  110. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/send_node.rb +291 -257
  111. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/sym_node.rb +4 -2
  112. data/lib/solargraph/parser/parser_gem/node_processors/until_node.rb +29 -0
  113. data/lib/solargraph/parser/parser_gem/node_processors/while_node.rb +29 -0
  114. data/lib/solargraph/parser/parser_gem/node_processors.rb +70 -0
  115. data/lib/solargraph/parser/parser_gem.rb +12 -0
  116. data/lib/solargraph/parser/region.rb +69 -66
  117. data/lib/solargraph/parser/snippet.rb +17 -13
  118. data/lib/solargraph/parser.rb +9 -12
  119. data/lib/solargraph/pin/base.rb +729 -299
  120. data/lib/solargraph/pin/base_variable.rb +126 -84
  121. data/lib/solargraph/pin/block.rb +104 -73
  122. data/lib/solargraph/pin/breakable.rb +9 -0
  123. data/lib/solargraph/pin/callable.rb +231 -0
  124. data/lib/solargraph/pin/closure.rb +72 -37
  125. data/lib/solargraph/pin/common.rb +79 -70
  126. data/lib/solargraph/pin/constant.rb +2 -0
  127. data/lib/solargraph/pin/conversions.rb +123 -92
  128. data/lib/solargraph/pin/delegated_method.rb +120 -0
  129. data/lib/solargraph/pin/documenting.rb +114 -105
  130. data/lib/solargraph/pin/instance_variable.rb +34 -30
  131. data/lib/solargraph/pin/keyword.rb +20 -15
  132. data/lib/solargraph/pin/local_variable.rb +75 -55
  133. data/lib/solargraph/pin/method.rb +672 -335
  134. data/lib/solargraph/pin/method_alias.rb +34 -31
  135. data/lib/solargraph/pin/namespace.rb +115 -94
  136. data/lib/solargraph/pin/parameter.rb +275 -206
  137. data/lib/solargraph/pin/proxy_type.rb +39 -29
  138. data/lib/solargraph/pin/reference/override.rb +47 -29
  139. data/lib/solargraph/pin/reference/require.rb +2 -2
  140. data/lib/solargraph/pin/reference/superclass.rb +15 -10
  141. data/lib/solargraph/pin/reference.rb +39 -14
  142. data/lib/solargraph/pin/search.rb +61 -56
  143. data/lib/solargraph/pin/signature.rb +61 -23
  144. data/lib/solargraph/pin/singleton.rb +1 -1
  145. data/lib/solargraph/pin/symbol.rb +53 -47
  146. data/lib/solargraph/pin/until.rb +18 -0
  147. data/lib/solargraph/pin/while.rb +18 -0
  148. data/lib/solargraph/pin.rb +44 -38
  149. data/lib/solargraph/pin_cache.rb +245 -0
  150. data/lib/solargraph/position.rb +132 -100
  151. data/lib/solargraph/range.rb +112 -95
  152. data/lib/solargraph/rbs_map/conversions.rb +823 -394
  153. data/lib/solargraph/rbs_map/core_fills.rb +53 -30
  154. data/lib/solargraph/rbs_map/core_map.rb +58 -38
  155. data/lib/solargraph/rbs_map/stdlib_map.rb +43 -36
  156. data/lib/solargraph/rbs_map.rb +163 -73
  157. data/lib/solargraph/shell.rb +352 -244
  158. data/lib/solargraph/source/chain/array.rb +37 -0
  159. data/lib/solargraph/source/chain/block_symbol.rb +13 -0
  160. data/lib/solargraph/source/chain/block_variable.rb +1 -1
  161. data/lib/solargraph/source/chain/call.rb +337 -215
  162. data/lib/solargraph/source/chain/constant.rb +26 -75
  163. data/lib/solargraph/source/chain/hash.rb +34 -28
  164. data/lib/solargraph/source/chain/head.rb +1 -1
  165. data/lib/solargraph/source/chain/if.rb +28 -0
  166. data/lib/solargraph/source/chain/instance_variable.rb +13 -13
  167. data/lib/solargraph/source/chain/link.rb +44 -6
  168. data/lib/solargraph/source/chain/literal.rb +48 -23
  169. data/lib/solargraph/source/chain/or.rb +23 -23
  170. data/lib/solargraph/source/chain/z_super.rb +4 -4
  171. data/lib/solargraph/source/chain.rb +291 -179
  172. data/lib/solargraph/source/change.rb +82 -79
  173. data/lib/solargraph/source/cursor.rb +166 -164
  174. data/lib/solargraph/source/encoding_fixes.rb +23 -23
  175. data/lib/solargraph/source/source_chainer.rb +194 -191
  176. data/lib/solargraph/source/updater.rb +55 -54
  177. data/lib/solargraph/source.rb +498 -522
  178. data/lib/solargraph/source_map/clip.rb +226 -229
  179. data/lib/solargraph/source_map/data.rb +34 -0
  180. data/lib/solargraph/source_map/mapper.rb +259 -243
  181. data/lib/solargraph/source_map.rb +212 -180
  182. data/lib/solargraph/type_checker/checks.rb +124 -112
  183. data/lib/solargraph/type_checker/param_def.rb +37 -35
  184. data/lib/solargraph/type_checker/problem.rb +32 -32
  185. data/lib/solargraph/type_checker/rules.rb +84 -57
  186. data/lib/solargraph/type_checker.rb +814 -549
  187. data/lib/solargraph/version.rb +5 -5
  188. data/lib/solargraph/views/_method.erb +10 -10
  189. data/lib/solargraph/views/_namespace.erb +3 -3
  190. data/lib/solargraph/views/document.erb +10 -10
  191. data/lib/solargraph/views/environment.erb +3 -5
  192. data/lib/solargraph/workspace/config.rb +255 -231
  193. data/lib/solargraph/workspace/require_paths.rb +97 -0
  194. data/lib/solargraph/workspace.rb +220 -212
  195. data/lib/solargraph/yard_map/cache.rb +6 -0
  196. data/lib/solargraph/yard_map/helpers.rb +44 -16
  197. data/lib/solargraph/yard_map/mapper/to_constant.rb +8 -5
  198. data/lib/solargraph/yard_map/mapper/to_method.rb +130 -81
  199. data/lib/solargraph/yard_map/mapper/to_namespace.rb +31 -27
  200. data/lib/solargraph/yard_map/mapper.rb +79 -77
  201. data/lib/solargraph/yard_map/to_method.rb +89 -79
  202. data/lib/solargraph/yard_map.rb +1 -284
  203. data/lib/solargraph/yard_tags.rb +20 -0
  204. data/lib/solargraph/yardoc.rb +87 -0
  205. data/lib/solargraph.rb +105 -69
  206. data/rbs/fills/bundler/0/bundler.rbs +4271 -0
  207. data/rbs/fills/open3/0/open3.rbs +172 -0
  208. data/rbs/fills/rubygems/0/basic_specification.rbs +326 -0
  209. data/rbs/fills/rubygems/0/errors.rbs +364 -0
  210. data/rbs/fills/rubygems/0/spec_fetcher.rbs +107 -0
  211. data/rbs/fills/rubygems/0/specification.rbs +1753 -0
  212. data/rbs/fills/tuple/tuple.rbs +149 -0
  213. data/rbs/shims/ast/0/node.rbs +5 -0
  214. data/rbs/shims/ast/2.4/.rbs_meta.yaml +9 -0
  215. data/rbs/shims/ast/2.4/ast.rbs +73 -0
  216. data/rbs/shims/parser/3.2.0.1/builders/default.rbs +195 -0
  217. data/rbs/shims/parser/3.2.0.1/manifest.yaml +7 -0
  218. data/rbs/shims/parser/3.2.0.1/parser.rbs +201 -0
  219. data/rbs/shims/parser/3.2.0.1/polyfill.rbs +4 -0
  220. data/rbs/shims/thor/1.2.0.1/.rbs_meta.yaml +9 -0
  221. data/rbs/shims/thor/1.2.0.1/manifest.yaml +7 -0
  222. data/rbs/shims/thor/1.2.0.1/thor.rbs +17 -0
  223. data/rbs_collection.yaml +19 -0
  224. data/solargraph.gemspec +39 -11
  225. metadata +354 -97
  226. data/lib/.rubocop.yml +0 -22
  227. data/lib/solargraph/api_map/bundler_methods.rb +0 -22
  228. data/lib/solargraph/cache.rb +0 -53
  229. data/lib/solargraph/convention/rspec.rb +0 -30
  230. data/lib/solargraph/documentor.rb +0 -76
  231. data/lib/solargraph/language_server/host/cataloger.rb +0 -56
  232. data/lib/solargraph/parser/legacy/node_processors/alias_node.rb +0 -23
  233. data/lib/solargraph/parser/legacy/node_processors/args_node.rb +0 -35
  234. data/lib/solargraph/parser/legacy/node_processors/begin_node.rb +0 -15
  235. data/lib/solargraph/parser/legacy/node_processors/sym_node.rb +0 -18
  236. data/lib/solargraph/parser/legacy/node_processors.rb +0 -54
  237. data/lib/solargraph/parser/legacy.rb +0 -12
  238. data/lib/solargraph/parser/node_methods.rb +0 -43
  239. data/lib/solargraph/parser/rubyvm/class_methods.rb +0 -149
  240. data/lib/solargraph/parser/rubyvm/node_chainer.rb +0 -160
  241. data/lib/solargraph/parser/rubyvm/node_methods.rb +0 -315
  242. data/lib/solargraph/parser/rubyvm/node_processors/args_node.rb +0 -85
  243. data/lib/solargraph/parser/rubyvm/node_processors/block_node.rb +0 -42
  244. data/lib/solargraph/parser/rubyvm/node_processors/casgn_node.rb +0 -33
  245. data/lib/solargraph/parser/rubyvm/node_processors/cvasgn_node.rb +0 -23
  246. data/lib/solargraph/parser/rubyvm/node_processors/def_node.rb +0 -75
  247. data/lib/solargraph/parser/rubyvm/node_processors/defs_node.rb +0 -68
  248. data/lib/solargraph/parser/rubyvm/node_processors/gvasgn_node.rb +0 -23
  249. data/lib/solargraph/parser/rubyvm/node_processors/ivasgn_node.rb +0 -38
  250. data/lib/solargraph/parser/rubyvm/node_processors/kw_arg_node.rb +0 -39
  251. data/lib/solargraph/parser/rubyvm/node_processors/lit_node.rb +0 -20
  252. data/lib/solargraph/parser/rubyvm/node_processors/lvasgn_node.rb +0 -27
  253. data/lib/solargraph/parser/rubyvm/node_processors/namespace_node.rb +0 -39
  254. data/lib/solargraph/parser/rubyvm/node_processors/opt_arg_node.rb +0 -26
  255. data/lib/solargraph/parser/rubyvm/node_processors/orasgn_node.rb +0 -15
  256. data/lib/solargraph/parser/rubyvm/node_processors/resbody_node.rb +0 -45
  257. data/lib/solargraph/parser/rubyvm/node_processors/sclass_node.rb +0 -32
  258. data/lib/solargraph/parser/rubyvm/node_processors/scope_node.rb +0 -15
  259. data/lib/solargraph/parser/rubyvm/node_processors/send_node.rb +0 -279
  260. data/lib/solargraph/parser/rubyvm/node_processors.rb +0 -63
  261. data/lib/solargraph/parser/rubyvm/node_wrapper.rb +0 -47
  262. data/lib/solargraph/parser/rubyvm.rb +0 -40
  263. data/lib/solargraph/rbs_map/core_signs.rb +0 -33
  264. data/lib/yard-solargraph.rb +0 -33
@@ -1,549 +1,814 @@
1
- # frozen_string_literal: true
2
-
3
- module Solargraph
4
- # A static analysis tool for validating data types.
5
- #
6
- class TypeChecker
7
- autoload :Problem, 'solargraph/type_checker/problem'
8
- autoload :ParamDef, 'solargraph/type_checker/param_def'
9
- autoload :Rules, 'solargraph/type_checker/rules'
10
- autoload :Checks, 'solargraph/type_checker/checks'
11
-
12
- include Checks
13
- include Parser::NodeMethods
14
-
15
- # @return [String]
16
- attr_reader :filename
17
-
18
- # @return [Rules]
19
- attr_reader :rules
20
-
21
- # @return [ApiMap]
22
- attr_reader :api_map
23
-
24
- # @param filename [String]
25
- # @param api_map [ApiMap]
26
- # @param level [Symbol]
27
- def initialize filename, api_map: nil, level: :normal
28
- @filename = filename
29
- # @todo Smarter directory resolution
30
- @api_map = api_map || Solargraph::ApiMap.load(File.dirname(filename))
31
- @rules = Rules.new(level)
32
- @marked_ranges = []
33
- end
34
-
35
- # @return [SourceMap]
36
- def source_map
37
- @source_map ||= api_map.source_map(filename)
38
- end
39
-
40
- # @return [Array<Problem>]
41
- def problems
42
- @problems ||= begin
43
- without_ignored(
44
- method_tag_problems
45
- .concat variable_type_tag_problems
46
- .concat const_problems
47
- .concat call_problems
48
- )
49
- end
50
- end
51
-
52
- class << self
53
- # @param filename [String]
54
- # @return [self]
55
- def load filename, level = :normal
56
- source = Solargraph::Source.load(filename)
57
- api_map = Solargraph::ApiMap.new
58
- api_map.map(source)
59
- new(filename, api_map: api_map, level: level)
60
- end
61
-
62
- # @param code [String]
63
- # @param filename [String, nil]
64
- # @return [self]
65
- def load_string code, filename = nil, level = :normal
66
- source = Solargraph::Source.load_string(code, filename)
67
- api_map = Solargraph::ApiMap.new
68
- api_map.map(source)
69
- new(filename, api_map: api_map, level: level)
70
- end
71
- end
72
-
73
- private
74
-
75
- # @return [Array<Problem>]
76
- def method_tag_problems
77
- result = []
78
- # @param pin [Pin::Method]
79
- source_map.pins_by_class(Pin::Method).each do |pin|
80
- result.concat method_return_type_problems_for(pin)
81
- result.concat method_param_type_problems_for(pin)
82
- end
83
- result
84
- end
85
-
86
- # @param pin [Pin::Method]
87
- # @return [Array<Problem>]
88
- def method_return_type_problems_for pin
89
- return [] if pin.is_a?(Pin::MethodAlias)
90
- result = []
91
- declared = pin.typify(api_map).self_to(pin.full_context.namespace)
92
- if declared.undefined?
93
- if pin.return_type.undefined? && rules.require_type_tags?
94
- result.push Problem.new(pin.location, "Missing @return tag for #{pin.path}", pin: pin)
95
- elsif pin.return_type.defined? && !resolved_constant?(pin)
96
- result.push Problem.new(pin.location, "Unresolved return type #{pin.return_type} for #{pin.path}", pin: pin)
97
- elsif rules.must_tag_or_infer? && pin.probe(api_map).undefined?
98
- result.push Problem.new(pin.location, "Untyped method #{pin.path} could not be inferred")
99
- end
100
- elsif rules.validate_tags?
101
- unless pin.node.nil? || declared.void? || virtual_pin?(pin) || abstract?(pin)
102
- inferred = pin.probe(api_map).self_to(pin.full_context.namespace)
103
- if inferred.undefined?
104
- unless rules.ignore_all_undefined? || external?(pin)
105
- result.push Problem.new(pin.location, "#{pin.path} return type could not be inferred", pin: pin)
106
- end
107
- else
108
- unless (rules.rank > 1 ? all_types_match?(api_map, inferred, declared) : any_types_match?(api_map, declared, inferred))
109
- result.push Problem.new(pin.location, "Declared return type #{declared} does not match inferred type #{inferred} for #{pin.path}", pin: pin)
110
- end
111
- end
112
- end
113
- end
114
- result
115
- end
116
-
117
- # @todo This is not optimal. A better solution would probably be to mix
118
- # namespace alias into types at the ApiMap level.
119
- #
120
- # @param pin [Pin::Base]
121
- # @return [Boolean]
122
- def resolved_constant? pin
123
- return true if pin.typify(api_map).defined?
124
- api_map.get_constants('', *pin.closure.gates)
125
- .select { |p| p.name == pin.return_type.namespace }
126
- .any? { |p| p.infer(api_map).defined? }
127
- end
128
-
129
- def virtual_pin? pin
130
- pin.location && source_map.source.comment_at?(pin.location.range.ending)
131
- end
132
-
133
- # @param pin [Pin::Method]
134
- # @return [Array<Problem>]
135
- def method_param_type_problems_for pin
136
- stack = api_map.get_method_stack(pin.namespace, pin.name, scope: pin.scope)
137
- params = first_param_hash(stack)
138
- result = []
139
- if rules.require_type_tags?
140
- pin.signatures.each do |sig|
141
- sig.parameters.each do |par|
142
- break if par.decl == :restarg || par.decl == :kwrestarg || par.decl == :blockarg
143
- unless params[par.name]
144
- result.push Problem.new(pin.location, "Missing @param tag for #{par.name} on #{pin.path}", pin: pin)
145
- end
146
- end
147
- end
148
- end
149
- params.each_pair do |name, data|
150
- type = data[:qualified]
151
- if type.undefined?
152
- result.push Problem.new(pin.location, "Unresolved type #{data[:tagged]} for #{name} param on #{pin.path}", pin: pin)
153
- end
154
- end
155
- result
156
- end
157
-
158
- def ignored_pins
159
- @ignored_pins ||= []
160
- end
161
-
162
- # @return [Array<Problem>]
163
- def variable_type_tag_problems
164
- result = []
165
- all_variables.each do |pin|
166
- if pin.return_type.defined?
167
- declared = pin.typify(api_map)
168
- next if declared.duck_type?
169
- if declared.defined?
170
- if rules.validate_tags?
171
- inferred = pin.probe(api_map)
172
- if inferred.undefined?
173
- next if rules.ignore_all_undefined?
174
- if declared_externally?(pin)
175
- ignored_pins.push pin
176
- else
177
- result.push Problem.new(pin.location, "Variable type could not be inferred for #{pin.name}", pin: pin)
178
- end
179
- else
180
- unless any_types_match?(api_map, declared, inferred)
181
- result.push Problem.new(pin.location, "Declared type #{declared} does not match inferred type #{inferred} for variable #{pin.name}", pin: pin)
182
- end
183
- end
184
- elsif declared_externally?(pin)
185
- ignored_pins.push pin
186
- end
187
- elsif !pin.is_a?(Pin::Parameter) && !resolved_constant?(pin)
188
- result.push Problem.new(pin.location, "Unresolved type #{pin.return_type} for variable #{pin.name}", pin: pin)
189
- end
190
- else
191
- inferred = pin.probe(api_map)
192
- if inferred.undefined? && declared_externally?(pin)
193
- ignored_pins.push pin
194
- end
195
- end
196
- end
197
- result
198
- end
199
-
200
- # @return [Array<Pin::BaseVariable>]
201
- def all_variables
202
- source_map.pins_by_class(Pin::BaseVariable) + source_map.locals.select { |pin| pin.is_a?(Pin::LocalVariable) }
203
- end
204
-
205
- def const_problems
206
- return [] unless rules.validate_consts?
207
- result = []
208
- Solargraph::Parser::NodeMethods.const_nodes_from(source_map.source.node).each do |const|
209
- rng = Solargraph::Range.from_node(const)
210
- chain = Solargraph::Parser.chain(const, filename)
211
- block_pin = source_map.locate_block_pin(rng.start.line, rng.start.column)
212
- location = Location.new(filename, rng)
213
- locals = source_map.locals_at(location)
214
- pins = chain.define(api_map, block_pin, locals)
215
- if pins.empty?
216
- result.push Problem.new(location, "Unresolved constant #{Solargraph::Parser::NodeMethods.unpack_name(const)}")
217
- @marked_ranges.push location.range
218
- end
219
- end
220
- result
221
- end
222
-
223
- def call_problems
224
- result = []
225
- Solargraph::Parser::NodeMethods.call_nodes_from(source_map.source.node).each do |call|
226
- rng = Solargraph::Range.from_node(call)
227
- next if @marked_ranges.any? { |d| d.contain?(rng.start) }
228
- chain = Solargraph::Parser.chain(call, filename)
229
- block_pin = source_map.locate_block_pin(rng.start.line, rng.start.column)
230
- location = Location.new(filename, rng)
231
- locals = source_map.locals_at(location)
232
- type = chain.infer(api_map, block_pin, locals)
233
- if type.undefined? && !rules.ignore_all_undefined?
234
- base = chain
235
- missing = chain
236
- found = nil
237
- closest = ComplexType::UNDEFINED
238
- until base.links.first.undefined?
239
- found = base.define(api_map, block_pin, locals).first
240
- break if found
241
- missing = base
242
- base = base.base
243
- end
244
- closest = found.typify(api_map) if found
245
- if !found || found.is_a?(Pin::BaseVariable) || (closest.defined? && internal_or_core?(found))
246
- unless closest.parameterized? || ignored_pins.include?(found)
247
- result.push Problem.new(location, "Unresolved call to #{missing.links.last.word}")
248
- @marked_ranges.push rng
249
- end
250
- end
251
- end
252
- result.concat argument_problems_for(chain, api_map, block_pin, locals, location)
253
- end
254
- result
255
- end
256
-
257
- def argument_problems_for chain, api_map, block_pin, locals, location
258
- result = []
259
- base = chain
260
- until base.links.length == 1 && base.undefined?
261
- pins = base.define(api_map, block_pin, locals)
262
- if pins.first.is_a?(Pin::Method)
263
- # @type [Pin::Method]
264
- pin = pins.first
265
- ap = if base.links.last.is_a?(Solargraph::Source::Chain::ZSuper)
266
- arity_problems_for(pin, fake_args_for(block_pin), location)
267
- else
268
- arity_problems_for(pin, base.links.last.arguments, location)
269
- end
270
- unless ap.empty?
271
- result.concat ap
272
- break
273
- end
274
- break unless rules.validate_calls?
275
- params = first_param_hash(pins)
276
-
277
- all_errors = []
278
- pin.signatures.sort { |sig| sig.parameters.length }.each do |sig|
279
- errors = []
280
- sig.parameters.each_with_index do |par, idx|
281
- argchain = base.links.last.arguments[idx]
282
- if argchain.nil? && par.decl == :arg
283
- errors.push Problem.new(location, "Not enough arguments to #{pin.path}")
284
- next
285
- end
286
- if argchain
287
- if par.decl != :arg
288
- errors.concat kwarg_problems_for argchain, api_map, block_pin, locals, location, pin, params, idx
289
- next
290
- else
291
- ptype = params.key?(par.name) ? params[par.name][:qualified] : ComplexType::UNDEFINED
292
- if ptype.nil?
293
- # @todo Some level (strong, I guess) should require the param here
294
- else
295
- argtype = argchain.infer(api_map, block_pin, locals)
296
- if argtype.defined? && ptype.defined? && !any_types_match?(api_map, ptype, argtype)
297
- errors.push Problem.new(location, "Wrong argument type for #{pin.path}: #{par.name} expected #{ptype}, received #{argtype}")
298
- next
299
- end
300
- end
301
- end
302
- elsif par.decl == :kwarg
303
- errors.push Problem.new(location, "Call to #{pin.path} is missing keyword argument #{par.name}")
304
- next
305
- end
306
- end
307
- if errors.empty?
308
- all_errors.clear
309
- break
310
- end
311
- all_errors.concat errors
312
- end
313
- result.concat all_errors
314
- end
315
- base = base.base
316
- end
317
- result
318
- end
319
-
320
- def kwarg_problems_for argchain, api_map, block_pin, locals, location, pin, params, first
321
- result = []
322
- kwargs = convert_hash(argchain.node)
323
- pin.signatures.first.parameters[first..-1].each_with_index do |par, cur|
324
- idx = first + cur
325
- argchain = kwargs[par.name.to_sym]
326
- if par.decl == :kwrestarg || (par.decl == :optarg && idx == pin.parameters.length - 1 && par.asgn_code == '{}')
327
- result.concat kwrestarg_problems_for(api_map, block_pin, locals, location, pin, params, kwargs)
328
- else
329
- if argchain
330
- data = params[par.name]
331
- if data.nil?
332
- # @todo Some level (strong, I guess) should require the param here
333
- else
334
- ptype = data[:qualified]
335
- next if ptype.undefined?
336
- argtype = argchain.infer(api_map, block_pin, locals)
337
- if argtype.defined? && ptype && !any_types_match?(api_map, ptype, argtype)
338
- result.push Problem.new(location, "Wrong argument type for #{pin.path}: #{par.name} expected #{ptype}, received #{argtype}")
339
- end
340
- end
341
- elsif par.decl == :kwarg
342
- result.push Problem.new(location, "Call to #{pin.path} is missing keyword argument #{par.name}")
343
- end
344
- end
345
- end
346
- result
347
- end
348
-
349
- def kwrestarg_problems_for(api_map, block_pin, locals, location, pin, params, kwargs)
350
- result = []
351
- kwargs.each_pair do |pname, argchain|
352
- next unless params.key?(pname.to_s)
353
- ptype = params[pname.to_s][:qualified]
354
- argtype = argchain.infer(api_map, block_pin, locals)
355
- if argtype.defined? && ptype && !any_types_match?(api_map, ptype, argtype)
356
- result.push Problem.new(location, "Wrong argument type for #{pin.path}: #{pname} expected #{ptype}, received #{argtype}")
357
- end
358
- end
359
- result
360
- end
361
-
362
- # @param [Pin::Method]
363
- # @return [Hash]
364
- def param_hash(pin)
365
- tags = pin.docstring.tags(:param)
366
- return {} if tags.empty?
367
- result = {}
368
- tags.each do |tag|
369
- next if tag.types.nil? || tag.types.empty?
370
- result[tag.name.to_s] = {
371
- tagged: tag.types.join(', '),
372
- qualified: Solargraph::ComplexType.try_parse(*tag.types).qualify(api_map, pin.full_context.namespace)
373
- }
374
- end
375
- result
376
- end
377
-
378
- # @param [Array<Pin::Method>]
379
- # @return [Hash]
380
- def first_param_hash(pins)
381
- pins.each do |pin|
382
- result = param_hash(pin)
383
- return result unless result.empty?
384
- end
385
- {}
386
- end
387
-
388
- # @param pin [Pin::Base]
389
- def internal? pin
390
- return false if pin.nil?
391
- pin.location && api_map.bundled?(pin.location.filename)
392
- end
393
-
394
- # True if the pin is either internal (part of the workspace) or from the core/stdlib
395
- def internal_or_core? pin
396
- # @todo RBS pins are not necessarily core/stdlib pins
397
- internal?(pin) || pin.source == :rbs
398
- end
399
-
400
- # @param pin [Pin::Base]
401
- def external? pin
402
- !internal? pin
403
- end
404
-
405
- def declared_externally? pin
406
- return true if pin.assignment.nil?
407
- chain = Solargraph::Parser.chain(pin.assignment, filename)
408
- rng = Solargraph::Range.from_node(pin.assignment)
409
- block_pin = source_map.locate_block_pin(rng.start.line, rng.start.column)
410
- location = Location.new(filename, Range.from_node(pin.assignment))
411
- locals = source_map.locals_at(location)
412
- type = chain.infer(api_map, block_pin, locals)
413
- if type.undefined? && !rules.ignore_all_undefined?
414
- base = chain
415
- missing = chain
416
- found = nil
417
- closest = ComplexType::UNDEFINED
418
- until base.links.first.undefined?
419
- found = base.define(api_map, block_pin, locals).first
420
- break if found
421
- missing = base
422
- base = base.base
423
- end
424
- closest = found.typify(api_map) if found
425
- if !found || closest.defined? || internal?(found)
426
- return false
427
- end
428
- end
429
- true
430
- end
431
-
432
- def arity_problems_for pin, arguments, location
433
- results = pin.signatures.map do |sig|
434
- r = parameterized_arity_problems_for(pin, sig.parameters, arguments, location)
435
- return [] if r.empty?
436
- r
437
- end
438
- results.first
439
- end
440
-
441
- def parameterized_arity_problems_for(pin, parameters, arguments, location)
442
- return [] unless pin.explicit?
443
- return [] if parameters.empty? && arguments.empty?
444
- return [] if pin.anon_splat?
445
- if parameters.empty?
446
- # Functions tagged param_tuple accepts two arguments (e.g., Hash#[]=)
447
- return [] if pin.docstring.tag(:param_tuple) && arguments.length == 2
448
- return [] if arguments.length == 1 && arguments.last.links.last.is_a?(Source::Chain::BlockVariable)
449
- return [Problem.new(location, "Too many arguments to #{pin.path}")]
450
- end
451
- unchecked = arguments.clone
452
- add_params = 0
453
- if unchecked.empty? && parameters.any? { |param| param.decl == :kwarg }
454
- return [Problem.new(location, "Missing keyword arguments to #{pin.path}")]
455
- end
456
- settled_kwargs = 0
457
- unless unchecked.empty?
458
- if any_splatted_call?(unchecked.map(&:node))
459
- settled_kwargs = parameters.count(&:keyword?)
460
- else
461
- kwargs = convert_hash(unchecked.last.node)
462
- if parameters.any? { |param| [:kwarg, :kwoptarg].include?(param.decl) || param.kwrestarg? }
463
- if kwargs.empty?
464
- add_params += 1
465
- else
466
- unchecked.pop
467
- parameters.each do |param|
468
- next unless param.keyword?
469
- if kwargs.key?(param.name.to_sym)
470
- kwargs.delete param.name.to_sym
471
- settled_kwargs += 1
472
- elsif param.decl == :kwarg
473
- return [] if arguments.last.links.last.is_a?(Solargraph::Source::Chain::Hash) && arguments.last.links.last.splatted?
474
- return [Problem.new(location, "Missing keyword argument #{param.name} to #{pin.path}")]
475
- end
476
- end
477
- kwargs.clear if parameters.any?(&:kwrestarg?)
478
- unless kwargs.empty?
479
- return [Problem.new(location, "Unrecognized keyword argument #{kwargs.keys.first} to #{pin.path}")]
480
- end
481
- end
482
- end
483
- end
484
- end
485
- req = required_param_count(parameters)
486
- if req + add_params < unchecked.length
487
- return [] if parameters.any?(&:rest?)
488
- opt = optional_param_count(parameters)
489
- return [] if unchecked.length <= req + opt
490
- if unchecked.length == req + opt + 1 && unchecked.last.links.last.is_a?(Source::Chain::BlockVariable)
491
- return []
492
- end
493
- if req + add_params + 1 == unchecked.length && any_splatted_call?(unchecked.map(&:node)) && (parameters.map(&:decl) & [:kwarg, :kwoptarg, :kwrestarg]).any?
494
- return []
495
- end
496
- return [] if arguments.length - req == parameters.select { |p| [:optarg, :kwoptarg].include?(p.decl) }.length
497
- return [Problem.new(location, "Too many arguments to #{pin.path}")]
498
- elsif unchecked.length < req - settled_kwargs && (arguments.empty? || (!arguments.last.splat? && !arguments.last.links.last.is_a?(Solargraph::Source::Chain::Hash)))
499
- # HACK: Kernel#raise signature is incorrect in Ruby 2.7 core docs.
500
- # See https://github.com/castwide/solargraph/issues/418
501
- unless arguments.empty? && pin.path == 'Kernel#raise'
502
- return [Problem.new(location, "Not enough arguments to #{pin.path}")]
503
- end
504
- end
505
- []
506
- end
507
-
508
- def required_param_count(parameters)
509
- parameters.sum { |param| %i[arg kwarg].include?(param.decl) ? 1 : 0 }
510
- end
511
-
512
- # @param pin [Pin::Method]
513
- def optional_param_count(parameters)
514
- parameters.select { |p| p.decl == :optarg }.length
515
- end
516
-
517
- def abstract? pin
518
- pin.docstring.has_tag?(:abstract) ||
519
- (pin.closure && pin.closure.docstring.has_tag?(:abstract))
520
- end
521
-
522
- def fake_args_for(pin)
523
- args = []
524
- with_opts = false
525
- with_block = false
526
- pin.parameters.each do |pin|
527
- if [:kwarg, :kwoptarg, :kwrestarg].include?(pin.decl)
528
- with_opts = true
529
- elsif pin.decl == :block
530
- with_block = true
531
- elsif pin.decl == :restarg
532
- args.push Solargraph::Source::Chain.new([Solargraph::Source::Chain::Variable.new(pin.name)], nil, true)
533
- else
534
- args.push Solargraph::Source::Chain.new([Solargraph::Source::Chain::Variable.new(pin.name)])
535
- end
536
- end
537
- args.push Solargraph::Parser.chain_string('{}') if with_opts
538
- args.push Solargraph::Parser.chain_string('&') if with_block
539
- args
540
- end
541
-
542
- def without_ignored problems
543
- problems.reject do |problem|
544
- node = source_map.source.node_at(problem.location.range.start.line, problem.location.range.start.column)
545
- source_map.source.comments_for(node)&.include?('@sg-ignore')
546
- end
547
- end
548
- end
549
- end
1
+ # frozen_string_literal: true
2
+
3
+ module Solargraph
4
+ # A static analysis tool for validating data types.
5
+ #
6
+ class TypeChecker
7
+ autoload :Problem, 'solargraph/type_checker/problem'
8
+ autoload :ParamDef, 'solargraph/type_checker/param_def'
9
+ autoload :Rules, 'solargraph/type_checker/rules'
10
+ autoload :Checks, 'solargraph/type_checker/checks'
11
+
12
+ include Checks
13
+ include Parser::NodeMethods
14
+
15
+ # @return [String]
16
+ attr_reader :filename
17
+
18
+ # @return [Rules]
19
+ attr_reader :rules
20
+
21
+ # @return [ApiMap]
22
+ attr_reader :api_map
23
+
24
+ # @param filename [String, nil]
25
+ # @param api_map [ApiMap, nil]
26
+ # @param level [Symbol] Don't complain about anything above this level
27
+ # @param workspace [Workspace, nil] Workspace to use for loading
28
+ # type checker rules modified by user config
29
+ # @param type_checker_rules [Hash{Symbol => Symbol}] Overrides for
30
+ # type checker rules - e.g., :report_undefined => :strong
31
+ # @param rules [Rules] Type checker rules object
32
+ def initialize filename,
33
+ api_map: nil,
34
+ level: :normal,
35
+ workspace: filename ? Workspace.new(File.dirname(filename)) : nil,
36
+ rules: workspace ? workspace.rules(level) : Rules.new(level, {})
37
+ @filename = filename
38
+ # @todo Smarter directory resolution
39
+ @api_map = api_map || Solargraph::ApiMap.load(File.dirname(filename))
40
+ @rules = rules
41
+ # @type [Array<Range>]
42
+ @marked_ranges = []
43
+ end
44
+
45
+ # @return [SourceMap]
46
+ def source_map
47
+ @source_map ||= api_map.source_map(filename)
48
+ end
49
+
50
+ # @return [Source]
51
+ def source
52
+ @source_map.source
53
+ end
54
+
55
+ # @return [Array<Problem>]
56
+ def problems
57
+ @problems ||= begin
58
+ all = method_tag_problems
59
+ .concat(variable_type_tag_problems)
60
+ .concat(const_problems)
61
+ .concat(call_problems)
62
+ unignored = without_ignored(all)
63
+ unignored.concat(unneeded_sgignore_problems)
64
+ end
65
+ end
66
+
67
+ class << self
68
+ # @param filename [String]
69
+ # @param level [Symbol]
70
+ # @return [self]
71
+ def load filename, level = :normal
72
+ source = Solargraph::Source.load(filename)
73
+ api_map = Solargraph::ApiMap.new
74
+ api_map.map(source)
75
+ new(filename, api_map: api_map, level: level)
76
+ end
77
+
78
+ # @param code [String]
79
+ # @param filename [String, nil]
80
+ # @param level [Symbol]
81
+ # @return [self]
82
+ def load_string code, filename = nil, level = :normal
83
+ source = Solargraph::Source.load_string(code, filename)
84
+ api_map = Solargraph::ApiMap.new
85
+ api_map.map(source)
86
+ new(filename, api_map: api_map, level: level)
87
+ end
88
+ end
89
+
90
+ private
91
+
92
+ # @return [Array<Problem>]
93
+ def method_tag_problems
94
+ result = []
95
+ # @param pin [Pin::Method]
96
+ source_map.pins_by_class(Pin::Method).each do |pin|
97
+ result.concat method_return_type_problems_for(pin)
98
+ result.concat method_param_type_problems_for(pin)
99
+ end
100
+ result
101
+ end
102
+
103
+ # @param pin [Pin::Method]
104
+ # @return [Array<Problem>]
105
+ def method_return_type_problems_for pin
106
+ return [] if pin.is_a?(Pin::MethodAlias)
107
+ result = []
108
+ declared = pin.typify(api_map).self_to_type(pin.full_context).qualify(api_map, *pin.gates)
109
+ if declared.undefined?
110
+ if pin.return_type.undefined? && rules.require_type_tags?
111
+ if pin.attribute?
112
+ inferred = pin.probe(api_map).self_to_type(pin.full_context)
113
+ result.push Problem.new(pin.location, "Missing @return tag for #{pin.path}", pin: pin) unless inferred.defined?
114
+ else
115
+ result.push Problem.new(pin.location, "Missing @return tag for #{pin.path}", pin: pin)
116
+ end
117
+ elsif pin.return_type.defined? && !resolved_constant?(pin)
118
+ result.push Problem.new(pin.location, "Unresolved return type #{pin.return_type} for #{pin.path}", pin: pin)
119
+ elsif rules.must_tag_or_infer? && pin.probe(api_map).undefined?
120
+ result.push Problem.new(pin.location, "Untyped method #{pin.path} could not be inferred")
121
+ end
122
+ elsif rules.validate_tags?
123
+ unless pin.node.nil? || declared.void? || virtual_pin?(pin) || abstract?(pin)
124
+ inferred = pin.probe(api_map).self_to_type(pin.full_context)
125
+ if inferred.undefined?
126
+ unless rules.ignore_all_undefined? || external?(pin)
127
+ result.push Problem.new(pin.location, "#{pin.path} return type could not be inferred", pin: pin)
128
+ end
129
+ else
130
+ unless (rules.require_all_return_types_match_inferred? ? all_types_match?(api_map, inferred, declared) : any_types_match?(api_map, declared, inferred))
131
+ result.push Problem.new(pin.location, "Declared return type #{declared.rooted_tags} does not match inferred type #{inferred.rooted_tags} for #{pin.path}", pin: pin)
132
+ end
133
+ end
134
+ end
135
+ end
136
+ result
137
+ end
138
+
139
+ # @todo This is not optimal. A better solution would probably be to mix
140
+ # namespace alias into types at the ApiMap level.
141
+ #
142
+ # @param pin [Pin::Base]
143
+ # @return [Boolean]
144
+ def resolved_constant? pin
145
+ return true if pin.typify(api_map).defined?
146
+ constant_pins = api_map.get_constants('', *pin.closure.gates)
147
+ .select { |p| p.name == pin.return_type.namespace }
148
+ return true if constant_pins.find { |p| p.typify(api_map).defined? }
149
+ # will need to probe when a constant name is assigned to a
150
+ # class/module (alias)
151
+ return true if constant_pins.find { |p| p.probe(api_map).defined? }
152
+ false
153
+ end
154
+
155
+ # @param pin [Pin::Base]
156
+ def virtual_pin? pin
157
+ pin.location && source.comment_at?(pin.location.range.ending)
158
+ end
159
+
160
+ # @param pin [Pin::Method]
161
+ # @return [Array<Problem>]
162
+ def method_param_type_problems_for pin
163
+ stack = api_map.get_method_stack(pin.namespace, pin.name, scope: pin.scope)
164
+ result = []
165
+ pin.signatures.each do |sig|
166
+ params = param_details_from_stack(sig, stack)
167
+ if rules.require_type_tags?
168
+ sig.parameters.each do |par|
169
+ break if par.decl == :restarg || par.decl == :kwrestarg || par.decl == :blockarg
170
+ unless params[par.name]
171
+ if pin.attribute?
172
+ inferred = pin.probe(api_map).self_to_type(pin.full_context)
173
+ if inferred.undefined?
174
+ result.push Problem.new(pin.location, "Missing @param tag for #{par.name} on #{pin.path}", pin: pin)
175
+ end
176
+ else
177
+ result.push Problem.new(pin.location, "Missing @param tag for #{par.name} on #{pin.path}", pin: pin)
178
+ end
179
+ end
180
+ end
181
+ end
182
+ # @param name [String]
183
+ # @param data [Hash{Symbol => BasicObject}]
184
+ params.each_pair do |name, data|
185
+ # @type [ComplexType]
186
+ type = data[:qualified]
187
+ if type.undefined?
188
+ result.push Problem.new(pin.location, "Unresolved type #{data[:tagged]} for #{name} param on #{pin.path}", pin: pin)
189
+ end
190
+ end
191
+ end
192
+ result
193
+ end
194
+
195
+ # @return [Array<Pin::Base>]
196
+ def ignored_pins
197
+ @ignored_pins ||= []
198
+ end
199
+
200
+ # @return [Array<Problem>]
201
+ def variable_type_tag_problems
202
+ result = []
203
+ all_variables.each do |pin|
204
+ if pin.return_type.defined?
205
+ declared = pin.typify(api_map)
206
+ next if declared.duck_type?
207
+ if declared.defined? && pin.assignment
208
+ if rules.validate_tags?
209
+ inferred = pin.probe(api_map)
210
+ if inferred.undefined?
211
+ next if rules.ignore_all_undefined?
212
+ if declared_externally?(pin)
213
+ ignored_pins.push pin
214
+ else
215
+ result.push Problem.new(pin.location, "Variable type could not be inferred for #{pin.name}", pin: pin)
216
+ end
217
+ else
218
+ unless any_types_match?(api_map, declared, inferred)
219
+ result.push Problem.new(pin.location, "Declared type #{declared} does not match inferred type #{inferred} for variable #{pin.name}", pin: pin)
220
+ end
221
+ end
222
+ elsif declared_externally?(pin)
223
+ ignored_pins.push pin
224
+ end
225
+ elsif !pin.is_a?(Pin::Parameter) && !resolved_constant?(pin)
226
+ result.push Problem.new(pin.location, "Unresolved type #{pin.return_type} for variable #{pin.name}", pin: pin)
227
+ end
228
+ elsif pin.assignment
229
+ inferred = pin.probe(api_map)
230
+ if inferred.undefined? && declared_externally?(pin)
231
+ ignored_pins.push pin
232
+ end
233
+ end
234
+ end
235
+ result
236
+ end
237
+
238
+ # @return [Array<Pin::BaseVariable>]
239
+ def all_variables
240
+ source_map.pins_by_class(Pin::BaseVariable) + source_map.locals.select { |pin| pin.is_a?(Pin::LocalVariable) }
241
+ end
242
+
243
+ # @return [Array<Problem>]
244
+ def const_problems
245
+ return [] unless rules.validate_consts?
246
+ result = []
247
+ Solargraph::Parser::NodeMethods.const_nodes_from(source.node).each do |const|
248
+ rng = Solargraph::Range.from_node(const)
249
+ chain = Solargraph::Parser.chain(const, filename)
250
+ closure_pin = source_map.locate_closure_pin(rng.start.line, rng.start.column)
251
+ closure_pin.rebind(api_map)
252
+ location = Location.new(filename, rng)
253
+ locals = source_map.locals_at(location)
254
+ pins = chain.define(api_map, closure_pin, locals)
255
+ if pins.empty?
256
+ result.push Problem.new(location, "Unresolved constant #{Solargraph::Parser::NodeMethods.unpack_name(const)}")
257
+ @marked_ranges.push location.range
258
+ end
259
+ end
260
+ result
261
+ end
262
+
263
+ # @return [Array<Problem>]
264
+ def call_problems
265
+ result = []
266
+ Solargraph::Parser::NodeMethods.call_nodes_from(source.node).each do |call|
267
+ rng = Solargraph::Range.from_node(call)
268
+ next if @marked_ranges.any? { |d| d.contain?(rng.start) }
269
+ chain = Solargraph::Parser.chain(call, filename)
270
+ closure_pin = source_map.locate_closure_pin(rng.start.line, rng.start.column)
271
+ namespace_pin = closure_pin
272
+ if call.type == :block
273
+ # blocks in the AST include the method call as well, so the
274
+ # node returned by #call_nodes_from needs to be backed out
275
+ # one closure
276
+ closure_pin = closure_pin.closure
277
+ end
278
+ closure_pin.rebind(api_map)
279
+ location = Location.new(filename, rng)
280
+ locals = source_map.locals_at(location)
281
+ type = chain.infer(api_map, closure_pin, locals)
282
+ if type.undefined? && !rules.ignore_all_undefined?
283
+ base = chain
284
+ missing = chain
285
+ found = nil
286
+ closest = ComplexType::UNDEFINED
287
+ until base.links.first.undefined?
288
+ found = base.define(api_map, closure_pin, locals).first
289
+ break if found
290
+ missing = base
291
+ base = base.base
292
+ end
293
+ closest = found.typify(api_map) if found
294
+ # @todo remove the internal_or_core? check at a higher-than-strict level
295
+ if !found || found.is_a?(Pin::BaseVariable) || (closest.defined? && internal_or_core?(found))
296
+ unless closest.generic? || ignored_pins.include?(found)
297
+ if closest.defined?
298
+ result.push Problem.new(location, "Unresolved call to #{missing.links.last.word} on #{closest}")
299
+ else
300
+ result.push Problem.new(location, "Unresolved call to #{missing.links.last.word}")
301
+ end
302
+ @marked_ranges.push rng
303
+ end
304
+ end
305
+ end
306
+ result.concat argument_problems_for(chain, api_map, closure_pin, locals, location)
307
+ end
308
+ result
309
+ end
310
+
311
+ # @param chain [Solargraph::Source::Chain]
312
+ # @param api_map [Solargraph::ApiMap]
313
+ # @param closure_pin [Solargraph::Pin::Closure]
314
+ # @param locals [Array<Solargraph::Pin::Base>]
315
+ # @param location [Solargraph::Location]
316
+ # @return [Array<Problem>]
317
+ def argument_problems_for chain, api_map, closure_pin, locals, location
318
+ result = []
319
+ base = chain
320
+ # @type last_base_link [Solargraph::Source::Chain::Call]
321
+ last_base_link = base.links.last
322
+ return [] unless last_base_link.is_a?(Solargraph::Source::Chain::Call)
323
+
324
+ arguments = last_base_link.arguments
325
+
326
+ pins = base.define(api_map, closure_pin, locals)
327
+
328
+ first_pin = pins.first
329
+ if first_pin.is_a?(Pin::DelegatedMethod) && !first_pin.resolvable?(api_map)
330
+ # Do nothing, as we can't find the actual method implementation
331
+ elsif first_pin.is_a?(Pin::Method)
332
+ # @type [Pin::Method]
333
+ pin = first_pin
334
+ ap = if base.links.last.is_a?(Solargraph::Source::Chain::ZSuper)
335
+ arity_problems_for(pin, fake_args_for(closure_pin), location)
336
+ elsif pin.path == 'Class#new'
337
+ fqns = if base.links.one?
338
+ closure_pin.namespace
339
+ else
340
+ base.base.infer(api_map, closure_pin, locals).namespace
341
+ end
342
+ init = api_map.get_method_stack(fqns, 'initialize').first
343
+ init ? arity_problems_for(init, arguments, location) : []
344
+ else
345
+ arity_problems_for(pin, arguments, location)
346
+ end
347
+ return ap unless ap.empty?
348
+ return [] if !rules.validate_calls? || base.links.first.is_a?(Solargraph::Source::Chain::ZSuper)
349
+
350
+ all_errors = []
351
+ pin.signatures.sort { |sig| sig.parameters.length }.each do |sig|
352
+ params = param_details_from_stack(sig, pins)
353
+
354
+ signature_errors = signature_argument_problems_for location, locals, closure_pin, params, arguments, sig, pin
355
+
356
+ if signature_errors.empty?
357
+ # we found a signature that works - meaning errors from
358
+ # other signatures don't matter.
359
+ return []
360
+ end
361
+ all_errors.concat signature_errors
362
+ end
363
+ result.concat all_errors
364
+ end
365
+ result
366
+ end
367
+
368
+ # @param location [Location]
369
+ # @param locals [Array<Pin::LocalVariable>]
370
+ # @param closure_pin [Pin::Closure]
371
+ # @param params [Hash{String => Hash{Symbol => String, Solargraph::ComplexType}}]
372
+ # @param arguments [Array<Source::Chain>]
373
+ # @param sig [Pin::Signature]
374
+ # @param pin [Pin::Method]
375
+ # @param pins [Array<Pin::Method>]
376
+ #
377
+ # @return [Array<Problem>]
378
+ def signature_argument_problems_for location, locals, closure_pin, params, arguments, sig, pin
379
+ errors = []
380
+ # @todo add logic mapping up restarg parameters with
381
+ # arguments (including restarg arguments). Use tuples
382
+ # when possible, and when not, ensure provably
383
+ # incorrect situations are detected.
384
+ sig.parameters.each_with_index do |par, idx|
385
+ return errors if par.decl == :restarg # bail out and assume the rest is valid pending better arg processing
386
+ argchain = arguments[idx]
387
+ if argchain.nil?
388
+ if par.decl == :arg
389
+ final_arg = arguments.last
390
+ if final_arg && final_arg.node.type == :splat
391
+ argchain = final_arg
392
+ return errors
393
+ else
394
+ errors.push Problem.new(location, "Not enough arguments to #{pin.path}")
395
+ end
396
+ else
397
+ final_arg = arguments.last
398
+ argchain = final_arg if final_arg && [:kwsplat, :hash].include?(final_arg.node.type)
399
+ end
400
+ end
401
+ if argchain
402
+ if par.decl != :arg
403
+ errors.concat kwarg_problems_for sig, argchain, api_map, closure_pin, locals, location, pin, params, idx
404
+ next
405
+ else
406
+ if argchain.node.type == :splat && argchain == arguments.last
407
+ final_arg = argchain
408
+ end
409
+ if (final_arg && final_arg.node.type == :splat)
410
+ # The final argument given has been seen and was a
411
+ # splat, which doesn't give us useful types or
412
+ # arities against positional parameters, so let's
413
+ # continue on in case there are any required
414
+ # kwargs we should warn about
415
+ next
416
+ end
417
+ if argchain.node.type == :splat && par != sig.parameters.last
418
+ # we have been given a splat and there are more
419
+ # arguments to come.
420
+
421
+ # @todo Improve this so that we can skip past the
422
+ # rest of the positional parameters here but still
423
+ # process the kwargs
424
+ return errors
425
+ end
426
+ ptype = params.key?(par.name) ? params[par.name][:qualified] : ComplexType::UNDEFINED
427
+ ptype = ptype.self_to_type(par.context)
428
+ if ptype.nil?
429
+ # @todo Some level (strong, I guess) should require the param here
430
+ else
431
+ argtype = argchain.infer(api_map, closure_pin, locals)
432
+ argtype = argtype.self_to_type(closure_pin.context)
433
+ if argtype.defined? && ptype.defined? && !any_types_match?(api_map, ptype, argtype)
434
+ errors.push Problem.new(location, "Wrong argument type for #{pin.path}: #{par.name} expected #{ptype}, received #{argtype}")
435
+ return errors
436
+ end
437
+ end
438
+ end
439
+ elsif par.decl == :kwarg
440
+ errors.push Problem.new(location, "Call to #{pin.path} is missing keyword argument #{par.name}")
441
+ next
442
+ end
443
+ end
444
+ errors
445
+ end
446
+
447
+ # @param sig [Pin::Signature]
448
+ # @param argchain [Source::Chain]
449
+ # @param api_map [ApiMap]
450
+ # @param closure_pin [Pin::Closure]
451
+ # @param locals [Array<Pin::LocalVariable>]
452
+ # @param location [Location]
453
+ # @param pin [Pin::Method]
454
+ # @param params [Hash{String => Hash{Symbol => String, Solargraph::ComplexType}}]
455
+ # @param idx [Integer]
456
+ #
457
+ # @return [Array<Problem>]
458
+ def kwarg_problems_for sig, argchain, api_map, closure_pin, locals, location, pin, params, idx
459
+ result = []
460
+ kwargs = convert_hash(argchain.node)
461
+ par = sig.parameters[idx]
462
+ argchain = kwargs[par.name.to_sym]
463
+ if par.decl == :kwrestarg || (par.decl == :optarg && idx == pin.parameters.length - 1 && par.asgn_code == '{}')
464
+ result.concat kwrestarg_problems_for(api_map, closure_pin, locals, location, pin, params, kwargs)
465
+ else
466
+ if argchain
467
+ data = params[par.name]
468
+ if data.nil?
469
+ # @todo Some level (strong, I guess) should require the param here
470
+ else
471
+ ptype = data[:qualified]
472
+ ptype = ptype.self_to_type(pin.context)
473
+ unless ptype.undefined?
474
+ # @sg-ignore https://github.com/castwide/solargraph/pull/1127
475
+ argtype = argchain.infer(api_map, closure_pin, locals).self_to_type(closure_pin.context)
476
+ # @sg-ignore Unresolved call to defined?
477
+ if argtype.defined? && ptype && !any_types_match?(api_map, ptype, argtype)
478
+ result.push Problem.new(location, "Wrong argument type for #{pin.path}: #{par.name} expected #{ptype}, received #{argtype}")
479
+ end
480
+ end
481
+ end
482
+ elsif par.decl == :kwarg
483
+ result.push Problem.new(location, "Call to #{pin.path} is missing keyword argument #{par.name}")
484
+ end
485
+ end
486
+ result
487
+ end
488
+
489
+ # @param api_map [ApiMap]
490
+ # @param closure_pin [Pin::Closure]
491
+ # @param locals [Array<Pin::LocalVariable>]
492
+ # @param location [Location]
493
+ # @param pin [Pin::Method]
494
+ # @param params [Hash{String => [nil, Hash]}]
495
+ # @param kwargs [Hash{Symbol => Source::Chain}]
496
+ # @return [Array<Problem>]
497
+ def kwrestarg_problems_for(api_map, closure_pin, locals, location, pin, params, kwargs)
498
+ result = []
499
+ kwargs.each_pair do |pname, argchain|
500
+ next unless params.key?(pname.to_s)
501
+ ptype = params[pname.to_s][:qualified]
502
+ ptype = ptype.self_to_type(pin.context)
503
+ argtype = argchain.infer(api_map, closure_pin, locals)
504
+ argtype = argtype.self_to_type(closure_pin.context)
505
+ if argtype.defined? && ptype && !any_types_match?(api_map, ptype, argtype)
506
+ result.push Problem.new(location, "Wrong argument type for #{pin.path}: #{pname} expected #{ptype}, received #{argtype}")
507
+ end
508
+ end
509
+ result
510
+ end
511
+
512
+ # @param param_details [Hash{String => Hash{Symbol => String, ComplexType}}]
513
+ # @param pin [Pin::Method, Pin::Signature]
514
+ # @param relevant_pin [Pin::Method, Pin::Signature] the pin which is under inspection
515
+ # @return [void]
516
+ def add_restkwarg_param_tag_details(param_details, pin, relevant_pin)
517
+ # see if we have additional tags to pay attention to from YARD -
518
+ # e.g., kwargs in a **restkwargs splat
519
+ tags = pin.docstring.tags(:param)
520
+ tags.each do |tag|
521
+ next if param_details.key? tag.name.to_s
522
+ next if tag.types.nil?
523
+ details = {
524
+ tagged: tag.types.join(', '),
525
+ qualified: Solargraph::ComplexType.try_parse(*tag.types).qualify(api_map, pin.full_context.namespace)
526
+ }
527
+ # don't complain about a param that didn't come from the pin we're looking at anyway
528
+ if details[:qualified].defined? ||
529
+ relevant_pin.parameter_names.include?(tag.name.to_s)
530
+ param_details[tag.name.to_s] = details
531
+ end
532
+ end
533
+ end
534
+
535
+ # @param pin [Pin::Signature]
536
+ # @return [Hash{String => Hash{Symbol => String, ComplexType}}]
537
+ def signature_param_details(pin)
538
+ # @type [Hash{String => Hash{Symbol => String, ComplexType}}]
539
+ result = {}
540
+ pin.parameters.each do |param|
541
+ type = param.typify(api_map)
542
+ next if type.nil? || type.undefined?
543
+ result[param.name.to_s] = {
544
+ tagged: type.tags,
545
+ qualified: type
546
+ }
547
+ end
548
+ # see if we have additional tags to pay attention to from YARD -
549
+ # e.g., kwargs in a **restkwargs splat
550
+ tags = pin.docstring.tags(:param)
551
+ tags.each do |tag|
552
+ next if result.key? tag.name.to_s
553
+ next if tag.types.nil?
554
+ result[tag.name.to_s] = {
555
+ tagged: tag.types.join(', '),
556
+ qualified: Solargraph::ComplexType.try_parse(*tag.types).qualify(api_map, *pin.closure.gates)
557
+ }
558
+ end
559
+ result
560
+ end
561
+
562
+ # The original signature defines the parameters, but other
563
+ # signatures and method pins can help by adding type information
564
+ #
565
+ # @param param_details [Hash{String => Hash{Symbol => String, ComplexType}}]
566
+ # @param param_names [Array<String>]
567
+ # @param new_param_details [Hash{String => Hash{Symbol => String, ComplexType}}]
568
+ #
569
+ # @return [void]
570
+ def add_to_param_details(param_details, param_names, new_param_details)
571
+ new_param_details.each do |param_name, details|
572
+ next unless param_names.include?(param_name)
573
+
574
+ param_details[param_name] ||= {}
575
+ param_details[param_name][:tagged] ||= details[:tagged]
576
+ param_details[param_name][:qualified] ||= details[:qualified]
577
+ end
578
+ end
579
+
580
+ # @param signature [Pin::Signature]
581
+ # @param method_pin_stack [Array<Pin::Method>]
582
+ # @return [Hash{String => Hash{Symbol => String, ComplexType}}]
583
+ def param_details_from_stack(signature, method_pin_stack)
584
+ signature_type = signature.typify(api_map)
585
+ signature = signature.proxy signature_type
586
+ param_details = signature_param_details(signature)
587
+ param_names = signature.parameter_names
588
+
589
+ method_pin_stack.each do |method_pin|
590
+ add_restkwarg_param_tag_details(param_details, method_pin, signature)
591
+
592
+ # documentation of types in superclasses should fail back to
593
+ # subclasses if the subclass hasn't documented something
594
+ method_pin.signatures.each do |sig|
595
+ add_restkwarg_param_tag_details(param_details, sig, signature)
596
+ add_to_param_details param_details, param_names, signature_param_details(sig)
597
+ end
598
+ end
599
+ param_details
600
+ end
601
+
602
+ # @param pin [Pin::Base]
603
+ def internal? pin
604
+ return false if pin.nil?
605
+ pin.location && api_map.bundled?(pin.location.filename)
606
+ end
607
+
608
+ # True if the pin is either internal (part of the workspace) or from the core/stdlib
609
+ # @param pin [Pin::Base]
610
+ def internal_or_core? pin
611
+ # @todo RBS pins are not necessarily core/stdlib pins
612
+ internal?(pin) || pin.source == :rbs
613
+ end
614
+
615
+ # @param pin [Pin::Base]
616
+ def external? pin
617
+ !internal? pin
618
+ end
619
+
620
+ # @param pin [Pin::BaseVariable]
621
+ def declared_externally? pin
622
+ raise "No assignment found" if pin.assignment.nil?
623
+
624
+ chain = Solargraph::Parser.chain(pin.assignment, filename)
625
+ rng = Solargraph::Range.from_node(pin.assignment)
626
+ closure_pin = source_map.locate_closure_pin(rng.start.line, rng.start.column)
627
+ location = Location.new(filename, Range.from_node(pin.assignment))
628
+ locals = source_map.locals_at(location)
629
+ type = chain.infer(api_map, closure_pin, locals)
630
+ if type.undefined? && !rules.ignore_all_undefined?
631
+ base = chain
632
+ missing = chain
633
+ found = nil
634
+ closest = ComplexType::UNDEFINED
635
+ until base.links.first.undefined?
636
+ found = base.define(api_map, closure_pin, locals).first
637
+ break if found
638
+ missing = base
639
+ base = base.base
640
+ end
641
+ closest = found.typify(api_map) if found
642
+ if !found || closest.defined? || internal?(found)
643
+ return false
644
+ end
645
+ end
646
+ true
647
+ end
648
+
649
+ # @param pin [Pin::Method]
650
+ # @param arguments [Array<Source::Chain>]
651
+ # @param location [Location]
652
+ # @return [Array<Problem>]
653
+ def arity_problems_for pin, arguments, location
654
+ results = pin.signatures.map do |sig|
655
+ r = parameterized_arity_problems_for(pin, sig.parameters, arguments, location)
656
+ return [] if r.empty?
657
+ r
658
+ end
659
+ results.first
660
+ end
661
+
662
+ # @param pin [Pin::Method]
663
+ # @param parameters [Array<Pin::Parameter>]
664
+ # @param arguments [Array<Source::Chain>]
665
+ # @param location [Location]
666
+ # @return [Array<Problem>]
667
+ def parameterized_arity_problems_for(pin, parameters, arguments, location)
668
+ return [] unless pin.explicit?
669
+ return [] if parameters.empty? && arguments.empty?
670
+ return [] if pin.anon_splat?
671
+ unchecked = arguments.dup # creates copy of and unthaws array
672
+ add_params = 0
673
+ if unchecked.empty? && parameters.any? { |param| param.decl == :kwarg }
674
+ return [Problem.new(location, "Missing keyword arguments to #{pin.path}")]
675
+ end
676
+ settled_kwargs = 0
677
+ unless unchecked.empty?
678
+ if any_splatted_call?(unchecked.map(&:node))
679
+ settled_kwargs = parameters.count(&:keyword?)
680
+ else
681
+ kwargs = convert_hash(unchecked.last.node)
682
+ if parameters.any? { |param| [:kwarg, :kwoptarg].include?(param.decl) || param.kwrestarg? }
683
+ if kwargs.empty?
684
+ add_params += 1
685
+ else
686
+ unchecked.pop
687
+ parameters.each do |param|
688
+ next unless param.keyword?
689
+ if kwargs.key?(param.name.to_sym)
690
+ kwargs.delete param.name.to_sym
691
+ settled_kwargs += 1
692
+ elsif param.decl == :kwarg
693
+ last_arg_last_link = arguments.last.links.last
694
+ return [] if last_arg_last_link.is_a?(Solargraph::Source::Chain::Hash) && last_arg_last_link.splatted?
695
+ return [Problem.new(location, "Missing keyword argument #{param.name} to #{pin.path}")]
696
+ end
697
+ end
698
+ kwargs.clear if parameters.any?(&:kwrestarg?)
699
+ unless kwargs.empty?
700
+ return [Problem.new(location, "Unrecognized keyword argument #{kwargs.keys.first} to #{pin.path}")]
701
+ end
702
+ end
703
+ end
704
+ end
705
+ end
706
+ req = required_param_count(parameters)
707
+ if req + add_params < unchecked.length
708
+ return [] if parameters.any?(&:rest?)
709
+ opt = optional_param_count(parameters)
710
+ return [] if unchecked.length <= req + opt
711
+ if req + add_params + 1 == unchecked.length && any_splatted_call?(unchecked.map(&:node)) && (parameters.map(&:decl) & [:kwarg, :kwoptarg, :kwrestarg]).any?
712
+ return []
713
+ end
714
+ return [] if arguments.length - req == parameters.select { |p| [:optarg, :kwoptarg].include?(p.decl) }.length
715
+ return [Problem.new(location, "Too many arguments to #{pin.path}")]
716
+ elsif unchecked.length < req - settled_kwargs && (arguments.empty? || (!arguments.last.splat? && !arguments.last.links.last.is_a?(Solargraph::Source::Chain::Hash)))
717
+ # HACK: Kernel#raise signature is incorrect in Ruby 2.7 core docs.
718
+ # See https://github.com/castwide/solargraph/issues/418
719
+ unless arguments.empty? && pin.path == 'Kernel#raise'
720
+ return [Problem.new(location, "Not enough arguments to #{pin.path}")]
721
+ end
722
+ end
723
+ []
724
+ end
725
+
726
+ # @param parameters [Enumerable<Pin::Parameter>]
727
+ # @todo need to use generic types in method to choose correct
728
+ # signature and generate Integer as return type
729
+ # @return [Integer]
730
+ def required_param_count(parameters)
731
+ parameters.sum { |param| %i[arg kwarg].include?(param.decl) ? 1 : 0 }
732
+ end
733
+
734
+ # @param parameters [Enumerable<Pin::Parameter>]
735
+ # @param pin [Pin::Method]
736
+ # @return [Integer]
737
+ def optional_param_count(parameters)
738
+ parameters.select { |p| p.decl == :optarg }.length
739
+ end
740
+
741
+ # @param pin [Pin::Method]
742
+ def abstract? pin
743
+ pin.docstring.has_tag?('abstract') ||
744
+ (pin.closure && pin.closure.docstring.has_tag?('abstract'))
745
+ end
746
+
747
+ # @param pin [Pin::Method]
748
+ # @return [Array<Source::Chain>]
749
+ def fake_args_for(pin)
750
+ args = []
751
+ with_opts = false
752
+ with_block = false
753
+ pin.parameters.each do |pin|
754
+ if [:kwarg, :kwoptarg, :kwrestarg].include?(pin.decl)
755
+ with_opts = true
756
+ elsif pin.decl == :block
757
+ with_block = true
758
+ elsif pin.decl == :restarg
759
+ args.push Solargraph::Source::Chain.new([Solargraph::Source::Chain::Variable.new(pin.name)], nil, true)
760
+ else
761
+ args.push Solargraph::Source::Chain.new([Solargraph::Source::Chain::Variable.new(pin.name)])
762
+ end
763
+ end
764
+ args.push Solargraph::Parser.chain_string('{}') if with_opts
765
+ args.push Solargraph::Parser.chain_string('&') if with_block
766
+ args
767
+ end
768
+
769
+ # @return [Set<Integer>]
770
+ def sg_ignore_lines_processed
771
+ @sg_ignore_lines_processed ||= Set.new
772
+ end
773
+
774
+ # @return [Set<Integer>]
775
+ def all_sg_ignore_lines
776
+ source.associated_comments.select do |_line, text|
777
+ text.include?('@sg-ignore')
778
+ end.keys.to_set
779
+ end
780
+
781
+ # @return [Array<Integer>]
782
+ def unprocessed_sg_ignore_lines
783
+ (all_sg_ignore_lines - sg_ignore_lines_processed).to_a.sort
784
+ end
785
+
786
+ # @return [Array<Problem>]
787
+ def unneeded_sgignore_problems
788
+ return [] unless rules.validate_sg_ignores?
789
+
790
+ unprocessed_sg_ignore_lines.map do |line|
791
+ Problem.new(
792
+ Location.new(filename, Range.from_to(line, 0, line, 0)),
793
+ 'Unneeded @sg-ignore comment'
794
+ )
795
+ end
796
+ end
797
+
798
+ # @param problems [Array<Problem>]
799
+ # @return [Array<Problem>]
800
+ def without_ignored problems
801
+ problems.reject do |problem|
802
+ node = source.node_at(problem.location.range.start.line, problem.location.range.start.column)
803
+ ignored = node && source.comments_for(node)&.include?('@sg-ignore')
804
+ unless !ignored || all_sg_ignore_lines.include?(problem.location.range.start.line)
805
+ # :nocov:
806
+ Solargraph.assert_or_log(:sg_ignore) { "@sg-ignore accounting issue - node is #{node}" }
807
+ # :nocov:
808
+ end
809
+ sg_ignore_lines_processed.add problem.location.range.start.line if ignored
810
+ ignored
811
+ end
812
+ end
813
+ end
814
+ end