solargraph 0.47.2 → 0.53.3

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 (185) hide show
  1. checksums.yaml +4 -4
  2. data/.github/FUNDING.yml +1 -0
  3. data/.github/workflows/plugins.yml +40 -0
  4. data/.github/workflows/rspec.yml +4 -8
  5. data/.github/workflows/typecheck.yml +34 -0
  6. data/.yardopts +2 -2
  7. data/CHANGELOG.md +137 -3
  8. data/LICENSE +1 -1
  9. data/README.md +19 -16
  10. data/SPONSORS.md +2 -9
  11. data/lib/solargraph/api_map/cache.rb +60 -20
  12. data/lib/solargraph/api_map/source_to_yard.rb +17 -10
  13. data/lib/solargraph/api_map/store.rb +60 -12
  14. data/lib/solargraph/api_map.rb +171 -99
  15. data/lib/solargraph/bench.rb +3 -2
  16. data/lib/solargraph/cache.rb +77 -0
  17. data/lib/solargraph/complex_type/type_methods.rb +61 -12
  18. data/lib/solargraph/complex_type/unique_type.rb +193 -16
  19. data/lib/solargraph/complex_type.rb +113 -10
  20. data/lib/solargraph/convention/rakefile.rb +17 -0
  21. data/lib/solargraph/convention.rb +2 -3
  22. data/lib/solargraph/converters/dd.rb +5 -0
  23. data/lib/solargraph/converters/dl.rb +3 -0
  24. data/lib/solargraph/converters/dt.rb +3 -0
  25. data/lib/solargraph/diagnostics/rubocop.rb +23 -8
  26. data/lib/solargraph/diagnostics/rubocop_helpers.rb +4 -1
  27. data/lib/solargraph/diagnostics/type_check.rb +1 -0
  28. data/lib/solargraph/diagnostics.rb +2 -2
  29. data/lib/solargraph/doc_map.rb +171 -0
  30. data/lib/solargraph/gem_pins.rb +64 -0
  31. data/lib/solargraph/language_server/host/cataloger.rb +2 -1
  32. data/lib/solargraph/language_server/host/diagnoser.rb +2 -2
  33. data/lib/solargraph/language_server/host/dispatch.rb +15 -5
  34. data/lib/solargraph/language_server/host/message_worker.rb +4 -0
  35. data/lib/solargraph/language_server/host/sources.rb +7 -4
  36. data/lib/solargraph/language_server/host.rb +50 -26
  37. data/lib/solargraph/language_server/message/completion_item/resolve.rb +3 -1
  38. data/lib/solargraph/language_server/message/extended/check_gem_version.rb +13 -1
  39. data/lib/solargraph/language_server/message/extended/download_core.rb +1 -5
  40. data/lib/solargraph/language_server/message/initialize.rb +13 -0
  41. data/lib/solargraph/language_server/message/initialized.rb +1 -0
  42. data/lib/solargraph/language_server/message/text_document/document_symbol.rb +4 -1
  43. data/lib/solargraph/language_server/message/text_document/formatting.rb +4 -4
  44. data/lib/solargraph/language_server/message/text_document/hover.rb +2 -0
  45. data/lib/solargraph/language_server/message/text_document/signature_help.rb +1 -6
  46. data/lib/solargraph/language_server/message/text_document/type_definition.rb +24 -0
  47. data/lib/solargraph/language_server/message/text_document.rb +1 -1
  48. data/lib/solargraph/language_server/message/workspace/did_change_configuration.rb +5 -0
  49. data/lib/solargraph/language_server/message/workspace/did_change_watched_files.rb +10 -3
  50. data/lib/solargraph/language_server/message.rb +1 -0
  51. data/lib/solargraph/language_server/transport/adapter.rb +16 -1
  52. data/lib/solargraph/language_server/transport/data_reader.rb +2 -0
  53. data/lib/solargraph/library.rb +124 -37
  54. data/lib/solargraph/location.rb +1 -0
  55. data/lib/solargraph/page.rb +6 -0
  56. data/lib/solargraph/parser/comment_ripper.rb +4 -0
  57. data/lib/solargraph/parser/node_methods.rb +47 -7
  58. data/lib/solargraph/parser/node_processor/base.rb +9 -0
  59. data/lib/solargraph/parser/{legacy → parser_gem}/class_methods.rb +31 -5
  60. data/lib/solargraph/parser/{legacy → parser_gem}/flawed_builder.rb +3 -1
  61. data/lib/solargraph/parser/{legacy → parser_gem}/node_chainer.rb +57 -41
  62. data/lib/solargraph/parser/parser_gem/node_methods.rb +499 -0
  63. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/alias_node.rb +1 -1
  64. data/lib/solargraph/parser/parser_gem/node_processors/args_node.rb +53 -0
  65. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/begin_node.rb +1 -1
  66. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/block_node.rb +3 -2
  67. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/casgn_node.rb +14 -4
  68. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/cvasgn_node.rb +1 -1
  69. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/def_node.rb +7 -20
  70. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/defs_node.rb +2 -2
  71. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/gvasgn_node.rb +1 -1
  72. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/ivasgn_node.rb +2 -2
  73. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/lvasgn_node.rb +2 -2
  74. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/namespace_node.rb +2 -2
  75. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/orasgn_node.rb +1 -1
  76. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/resbody_node.rb +3 -3
  77. data/lib/solargraph/parser/parser_gem/node_processors/sclass_node.rb +42 -0
  78. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/send_node.rb +2 -2
  79. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/sym_node.rb +1 -1
  80. data/lib/solargraph/parser/parser_gem/node_processors.rb +54 -0
  81. data/lib/solargraph/parser/parser_gem.rb +12 -0
  82. data/lib/solargraph/parser/region.rb +1 -1
  83. data/lib/solargraph/parser/snippet.rb +2 -0
  84. data/lib/solargraph/parser.rb +9 -10
  85. data/lib/solargraph/pin/base.rb +69 -11
  86. data/lib/solargraph/pin/base_variable.rb +8 -4
  87. data/lib/solargraph/pin/block.rb +21 -28
  88. data/lib/solargraph/pin/closure.rb +17 -2
  89. data/lib/solargraph/pin/common.rb +7 -3
  90. data/lib/solargraph/pin/conversions.rb +34 -8
  91. data/lib/solargraph/pin/delegated_method.rb +97 -0
  92. data/lib/solargraph/pin/documenting.rb +25 -34
  93. data/lib/solargraph/pin/instance_variable.rb +4 -0
  94. data/lib/solargraph/pin/local_variable.rb +13 -1
  95. data/lib/solargraph/pin/method.rb +270 -16
  96. data/lib/solargraph/pin/namespace.rb +17 -1
  97. data/lib/solargraph/pin/parameter.rb +52 -17
  98. data/lib/solargraph/pin/reference/override.rb +2 -2
  99. data/lib/solargraph/pin/reference.rb +8 -0
  100. data/lib/solargraph/pin/search.rb +4 -4
  101. data/lib/solargraph/pin/signature.rb +143 -0
  102. data/lib/solargraph/pin.rb +2 -1
  103. data/lib/solargraph/range.rb +4 -6
  104. data/lib/solargraph/rbs_map/conversions.rb +601 -0
  105. data/lib/solargraph/rbs_map/core_fills.rb +47 -0
  106. data/lib/solargraph/rbs_map/core_map.rb +28 -0
  107. data/lib/solargraph/rbs_map/stdlib_map.rb +33 -0
  108. data/lib/solargraph/rbs_map.rb +84 -0
  109. data/lib/solargraph/shell.rb +69 -48
  110. data/lib/solargraph/source/chain/array.rb +32 -0
  111. data/lib/solargraph/source/chain/block_symbol.rb +13 -0
  112. data/lib/solargraph/source/chain/call.rb +125 -61
  113. data/lib/solargraph/source/chain/constant.rb +15 -1
  114. data/lib/solargraph/source/chain/if.rb +23 -0
  115. data/lib/solargraph/source/chain/link.rb +8 -2
  116. data/lib/solargraph/source/chain/or.rb +1 -1
  117. data/lib/solargraph/source/chain/z_super.rb +3 -3
  118. data/lib/solargraph/source/chain.rb +44 -14
  119. data/lib/solargraph/source/change.rb +3 -0
  120. data/lib/solargraph/source/cursor.rb +2 -0
  121. data/lib/solargraph/source/source_chainer.rb +8 -5
  122. data/lib/solargraph/source.rb +18 -19
  123. data/lib/solargraph/source_map/clip.rb +30 -23
  124. data/lib/solargraph/source_map/mapper.rb +20 -5
  125. data/lib/solargraph/source_map.rb +28 -13
  126. data/lib/solargraph/type_checker/checks.rb +10 -2
  127. data/lib/solargraph/type_checker.rb +201 -98
  128. data/lib/solargraph/version.rb +1 -1
  129. data/lib/solargraph/views/environment.erb +2 -2
  130. data/lib/solargraph/workspace/config.rb +14 -11
  131. data/lib/solargraph/workspace.rb +28 -17
  132. data/lib/solargraph/yard_map/cache.rb +6 -0
  133. data/lib/solargraph/yard_map/helpers.rb +1 -1
  134. data/lib/solargraph/yard_map/mapper/to_method.rb +18 -5
  135. data/lib/solargraph/yard_map/mapper.rb +1 -1
  136. data/lib/solargraph/yard_map/to_method.rb +11 -4
  137. data/lib/solargraph/yard_map.rb +1 -443
  138. data/lib/solargraph/yard_tags.rb +20 -0
  139. data/lib/solargraph/yardoc.rb +52 -0
  140. data/lib/solargraph.rb +8 -6
  141. data/solargraph.gemspec +19 -8
  142. metadata +162 -98
  143. data/.travis.yml +0 -19
  144. data/lib/solargraph/api_map/bundler_methods.rb +0 -22
  145. data/lib/solargraph/compat.rb +0 -37
  146. data/lib/solargraph/convention/rspec.rb +0 -30
  147. data/lib/solargraph/documentor.rb +0 -76
  148. data/lib/solargraph/parser/legacy/node_methods.rb +0 -325
  149. data/lib/solargraph/parser/legacy/node_processors/alias_node.rb +0 -23
  150. data/lib/solargraph/parser/legacy/node_processors/args_node.rb +0 -35
  151. data/lib/solargraph/parser/legacy/node_processors/begin_node.rb +0 -15
  152. data/lib/solargraph/parser/legacy/node_processors/def_node.rb +0 -63
  153. data/lib/solargraph/parser/legacy/node_processors/sclass_node.rb +0 -21
  154. data/lib/solargraph/parser/legacy/node_processors.rb +0 -54
  155. data/lib/solargraph/parser/legacy.rb +0 -12
  156. data/lib/solargraph/parser/rubyvm/class_methods.rb +0 -144
  157. data/lib/solargraph/parser/rubyvm/node_chainer.rb +0 -160
  158. data/lib/solargraph/parser/rubyvm/node_methods.rb +0 -315
  159. data/lib/solargraph/parser/rubyvm/node_processors/args_node.rb +0 -85
  160. data/lib/solargraph/parser/rubyvm/node_processors/block_node.rb +0 -42
  161. data/lib/solargraph/parser/rubyvm/node_processors/casgn_node.rb +0 -22
  162. data/lib/solargraph/parser/rubyvm/node_processors/cvasgn_node.rb +0 -23
  163. data/lib/solargraph/parser/rubyvm/node_processors/defs_node.rb +0 -57
  164. data/lib/solargraph/parser/rubyvm/node_processors/gvasgn_node.rb +0 -23
  165. data/lib/solargraph/parser/rubyvm/node_processors/ivasgn_node.rb +0 -38
  166. data/lib/solargraph/parser/rubyvm/node_processors/kw_arg_node.rb +0 -39
  167. data/lib/solargraph/parser/rubyvm/node_processors/lit_node.rb +0 -20
  168. data/lib/solargraph/parser/rubyvm/node_processors/lvasgn_node.rb +0 -27
  169. data/lib/solargraph/parser/rubyvm/node_processors/namespace_node.rb +0 -39
  170. data/lib/solargraph/parser/rubyvm/node_processors/opt_arg_node.rb +0 -26
  171. data/lib/solargraph/parser/rubyvm/node_processors/orasgn_node.rb +0 -15
  172. data/lib/solargraph/parser/rubyvm/node_processors/resbody_node.rb +0 -45
  173. data/lib/solargraph/parser/rubyvm/node_processors/sclass_node.rb +0 -21
  174. data/lib/solargraph/parser/rubyvm/node_processors/scope_node.rb +0 -15
  175. data/lib/solargraph/parser/rubyvm/node_processors/send_node.rb +0 -277
  176. data/lib/solargraph/parser/rubyvm/node_processors/sym_node.rb +0 -18
  177. data/lib/solargraph/parser/rubyvm/node_processors.rb +0 -63
  178. data/lib/solargraph/parser/rubyvm.rb +0 -40
  179. data/lib/solargraph/yard_map/core_docs.rb +0 -170
  180. data/lib/solargraph/yard_map/core_fills.rb +0 -208
  181. data/lib/solargraph/yard_map/core_gen.rb +0 -76
  182. data/lib/solargraph/yard_map/rdoc_to_yard.rb +0 -140
  183. data/lib/solargraph/yard_map/stdlib_fills.rb +0 -43
  184. data/lib/yard-solargraph.rb +0 -33
  185. data/yardoc/2.2.2.tar.gz +0 -0
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Solargraph
4
4
  module Parser
5
- module Legacy
5
+ module ParserGem
6
6
  # A factory for generating chains from nodes.
7
7
  #
8
8
  class NodeChainer
@@ -10,11 +10,12 @@ module Solargraph
10
10
  Chain = Source::Chain
11
11
 
12
12
  # @param node [Parser::AST::Node]
13
- # @param filename [String]
14
- def initialize node, filename = nil, in_block = false
13
+ # @param filename [String, nil]
14
+ # @param parent [Parser::AST::Node, nil]
15
+ def initialize node, filename = nil, parent = nil
15
16
  @node = node
16
17
  @filename = filename
17
- @in_block = in_block ? 1 : 0
18
+ @parent = parent
18
19
  end
19
20
 
20
21
  # @return [Source::Chain]
@@ -25,10 +26,11 @@ module Solargraph
25
26
 
26
27
  class << self
27
28
  # @param node [Parser::AST::Node]
28
- # @param filename [String]
29
+ # @param filename [String, nil]
30
+ # @param parent [Parser::AST::Node, nil]
29
31
  # @return [Source::Chain]
30
- def chain node, filename = nil, in_block = false
31
- NodeChainer.new(node, filename, in_block).chain
32
+ def chain node, filename = nil, parent = nil
33
+ NodeChainer.new(node, filename, parent).chain
32
34
  end
33
35
 
34
36
  # @param code [String]
@@ -47,54 +49,43 @@ module Solargraph
47
49
  # @return [Array<Chain::Link>]
48
50
  def generate_links n
49
51
  return [] unless n.is_a?(::Parser::AST::Node)
50
- return generate_links(n.children[0]) if n.type == :begin
51
52
  return generate_links(n.children[0]) if n.type == :splat
53
+ # @type [Array<Chain::Link>]
52
54
  result = []
53
55
  if n.type == :block
54
- @in_block += 1
55
- result.concat generate_links(n.children[0])
56
- @in_block -= 1
56
+ result.concat NodeChainer.chain(n.children[0], @filename, n).links
57
57
  elsif n.type == :send
58
58
  if n.children[0].is_a?(::Parser::AST::Node)
59
59
  result.concat generate_links(n.children[0])
60
- args = []
61
- n.children[2..-1].each do |c|
62
- args.push NodeChainer.chain(c)
63
- end
64
- result.push Chain::Call.new(n.children[1].to_s, args, @in_block > 0 || block_passed?(n))
60
+ result.push Chain::Call.new(n.children[1].to_s, node_args(n), passed_block(n))
65
61
  elsif n.children[0].nil?
66
62
  args = []
67
63
  n.children[2..-1].each do |c|
68
- args.push NodeChainer.chain(c)
64
+ args.push NodeChainer.chain(c, @filename, n)
69
65
  end
70
- result.push Chain::Call.new(n.children[1].to_s, args, @in_block > 0 || block_passed?(n))
66
+ result.push Chain::Call.new(n.children[1].to_s, node_args(n), passed_block(n))
71
67
  else
72
68
  raise "No idea what to do with #{n}"
73
69
  end
74
70
  elsif n.type == :csend
75
71
  if n.children[0].is_a?(::Parser::AST::Node)
76
72
  result.concat generate_links(n.children[0])
77
- args = []
78
- n.children[2..-1].each do |c|
79
- args.push NodeChainer.chain(c)
80
- end
81
- result.push Chain::QCall.new(n.children[1].to_s, args, @in_block > 0 || block_passed?(n))
73
+ result.push Chain::QCall.new(n.children[1].to_s, node_args(n))
82
74
  elsif n.children[0].nil?
83
- args = []
84
- n.children[2..-1].each do |c|
85
- args.push NodeChainer.chain(c)
86
- end
87
- result.push Chain::QCall.new(n.children[1].to_s, args, @in_block > 0 || block_passed?(n))
75
+ result.push Chain::QCall.new(n.children[1].to_s, node_args(n))
88
76
  else
89
77
  raise "No idea what to do with #{n}"
90
78
  end
91
79
  elsif n.type == :self
92
80
  result.push Chain::Head.new('self')
93
81
  elsif n.type == :zsuper
94
- result.push Chain::ZSuper.new('super', @in_block > 0 || block_passed?(n))
82
+ result.push Chain::ZSuper.new('super')
95
83
  elsif n.type == :super
96
- args = n.children.map { |c| NodeChainer.chain(c) }
97
- result.push Chain::Call.new('super', args, @in_block > 0 || block_passed?(n))
84
+ args = n.children.map { |c| NodeChainer.chain(c, @filename, n) }
85
+ result.push Chain::Call.new('super', args)
86
+ elsif n.type == :yield
87
+ args = n.children.map { |c| NodeChainer.chain(c, @filename, n) }
88
+ result.push Chain::Call.new('yield', args)
98
89
  elsif n.type == :const
99
90
  const = unpack_name(n)
100
91
  result.push Chain::Constant.new(const)
@@ -114,24 +105,37 @@ module Solargraph
114
105
  elsif n.type == :and
115
106
  result.concat generate_links(n.children.last)
116
107
  elsif n.type == :or
117
- result.push Chain::Or.new([NodeChainer.chain(n.children[0], @filename), NodeChainer.chain(n.children[1], @filename)])
108
+ result.push Chain::Or.new([NodeChainer.chain(n.children[0], @filename), NodeChainer.chain(n.children[1], @filename, n)])
109
+ elsif n.type == :if
110
+ result.push Chain::If.new([NodeChainer.chain(n.children[1], @filename), NodeChainer.chain(n.children[2], @filename, n)])
118
111
  elsif [:begin, :kwbegin].include?(n.type)
119
- result.concat generate_links(n.children[0])
112
+ result.concat generate_links(n.children.last)
120
113
  elsif n.type == :block_pass
121
- result.push Chain::BlockVariable.new("&#{n.children[0].children[0].to_s}")
114
+ block_variable_name_node = n.children[0]
115
+ if block_variable_name_node.nil?
116
+ # anonymous block forwarding (e.g., "&")
117
+ # added in Ruby 3.1 - https://bugs.ruby-lang.org/issues/11256
118
+ result.push Chain::BlockVariable.new(nil)
119
+ else
120
+ if block_variable_name_node.type == :sym
121
+ result.push Chain::BlockSymbol.new("#{block_variable_name_node.children[0].to_s}")
122
+ else
123
+ result.push Chain::BlockVariable.new("&#{block_variable_name_node.children[0].to_s}")
124
+ end
125
+ end
122
126
  elsif n.type == :hash
123
127
  result.push Chain::Hash.new('::Hash', hash_is_splatted?(n))
128
+ elsif n.type == :array
129
+ chained_children = n.children.map { |c| NodeChainer.chain(c) }
130
+ result.push Source::Chain::Array.new(chained_children)
124
131
  else
125
132
  lit = infer_literal_node_type(n)
126
- # if lit == '::Hash'
127
- # result.push Chain::Hash.new(lit, hash_is_splatted?(n))
128
- # else
129
- result.push (lit ? Chain::Literal.new(lit) : Chain::Link.new)
130
- # end
133
+ result.push (lit ? Chain::Literal.new(lit) : Chain::Link.new)
131
134
  end
132
135
  result
133
136
  end
134
137
 
138
+ # @param node [Parser::AST::Node]
135
139
  def hash_is_splatted? node
136
140
  return false unless Parser.is_ast_node?(node) && node.type == :hash
137
141
  return false unless Parser.is_ast_node?(node.children.last) && node.children.last.type == :kwsplat
@@ -139,8 +143,20 @@ module Solargraph
139
143
  true
140
144
  end
141
145
 
142
- def block_passed? node
143
- node.children.last.is_a?(::Parser::AST::Node) && node.children.last.type == :block_pass
146
+ # @param node [Parser::AST::Node]
147
+ # @return [Source::Chain, nil]
148
+ def passed_block node
149
+ return unless node == @node && @parent&.type == :block
150
+
151
+ NodeChainer.chain(@parent.children[2], @filename)
152
+ end
153
+
154
+ # @param node [Parser::AST::Node]
155
+ # @return [Array<Source::Chain>]
156
+ def node_args node
157
+ node.children[2..-1].map do |child|
158
+ NodeChainer.chain(child, @filename, node)
159
+ end
144
160
  end
145
161
  end
146
162
  end
@@ -0,0 +1,499 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'parser'
4
+ require 'ast'
5
+
6
+ # Teach AST::Node#children about its generic type
7
+ #
8
+ # @todo contribute back to https://github.com/ruby/gem_rbs_collection/blob/main/gems/ast/2.4/ast.rbs
9
+ #
10
+ # @!parse
11
+ # module ::AST
12
+ # class Node
13
+ # # New children
14
+ #
15
+ # # @return [Array<AST::Node>]
16
+ # attr_reader :children
17
+ # end
18
+ # end
19
+
20
+ # https://github.com/whitequark/parser
21
+ module Solargraph
22
+ module Parser
23
+ module ParserGem
24
+ module NodeMethods
25
+ module_function
26
+
27
+ # @param node [Parser::AST::Node]
28
+ # @return [String]
29
+ def unpack_name(node)
30
+ pack_name(node).join("::")
31
+ end
32
+
33
+ # @param node [Parser::AST::Node]
34
+ # @return [Array<String>]
35
+ def pack_name(node)
36
+ # @type [Array<String>]
37
+ parts = []
38
+ if node.is_a?(AST::Node)
39
+ node.children.each { |n|
40
+ if n.is_a?(AST::Node)
41
+ if n.type == :cbase
42
+ parts = [''] + pack_name(n)
43
+ else
44
+ parts += pack_name(n)
45
+ end
46
+ else
47
+ parts.push n unless n.nil?
48
+ end
49
+ }
50
+ end
51
+ parts
52
+ end
53
+
54
+ # @param node [Parser::AST::Node]
55
+ # @return [String, nil]
56
+ def infer_literal_node_type node
57
+ return nil unless node.is_a?(AST::Node)
58
+ if node.type == :str || node.type == :dstr
59
+ return '::String'
60
+ elsif node.type == :array
61
+ return '::Array'
62
+ elsif node.type == :hash
63
+ return '::Hash'
64
+ elsif node.type == :int
65
+ return '::Integer'
66
+ elsif node.type == :float
67
+ return '::Float'
68
+ elsif node.type == :sym || node.type == :dsym
69
+ return '::Symbol'
70
+ elsif node.type == :regexp
71
+ return '::Regexp'
72
+ elsif node.type == :irange
73
+ return '::Range'
74
+ elsif node.type == :true || node.type == :false
75
+ return '::Boolean'
76
+ # @todo Support `nil` keyword in types
77
+ # elsif node.type == :nil
78
+ # return 'NilClass'
79
+ end
80
+ nil
81
+ end
82
+
83
+ # @param node [Parser::AST::Node]
84
+ # @return [Position]
85
+ def get_node_start_position(node)
86
+ Position.new(node.loc.line, node.loc.column)
87
+ end
88
+
89
+ # @param node [Parser::AST::Node]
90
+ # @return [Position]
91
+ def get_node_end_position(node)
92
+ Position.new(node.loc.last_line, node.loc.last_column)
93
+ end
94
+
95
+ # @param node [Parser::AST::Node]
96
+ # @param signature [String]
97
+ #
98
+ # @return [String]
99
+ def drill_signature node, signature
100
+ return signature unless node.is_a?(AST::Node)
101
+ if node.type == :const or node.type == :cbase
102
+ unless node.children[0].nil?
103
+ signature += drill_signature(node.children[0], signature)
104
+ end
105
+ signature += '::' unless signature.empty?
106
+ signature += node.children[1].to_s
107
+ elsif node.type == :lvar or node.type == :ivar or node.type == :cvar
108
+ signature += '.' unless signature.empty?
109
+ signature += node.children[0].to_s
110
+ elsif node.type == :send
111
+ unless node.children[0].nil?
112
+ signature += drill_signature(node.children[0], signature)
113
+ end
114
+ signature += '.' unless signature.empty?
115
+ signature += node.children[1].to_s
116
+ end
117
+ signature
118
+ end
119
+
120
+ # @param node [Parser::AST::Node]
121
+ # @return [Hash{Parser::AST::Node => Chain}]
122
+ def convert_hash node
123
+ return {} unless Parser.is_ast_node?(node)
124
+ return convert_hash(node.children[0]) if node.type == :kwsplat
125
+ return convert_hash(node.children[0]) if Parser.is_ast_node?(node.children[0]) && node.children[0].type == :kwsplat
126
+ return {} unless node.type == :hash
127
+ result = {}
128
+ node.children.each do |pair|
129
+ result[pair.children[0].children[0]] = Solargraph::Parser.chain(pair.children[1])
130
+ end
131
+ result
132
+ end
133
+
134
+ NIL_NODE = ::Parser::AST::Node.new(:nil)
135
+
136
+ # @param node [Parser::AST::Node]
137
+ #
138
+ # @return [Array<Parser::AST::Node>]
139
+ def const_nodes_from node
140
+ return [] unless Parser.is_ast_node?(node)
141
+ result = []
142
+ if node.type == :const
143
+ result.push node
144
+ else
145
+ node.children.each { |child| result.concat const_nodes_from(child) }
146
+ end
147
+ result
148
+ end
149
+
150
+ # @param node [Parser::AST::Node]
151
+ def splatted_hash? node
152
+ Parser.is_ast_node?(node.children[0]) && node.children[0].type == :kwsplat
153
+ end
154
+
155
+ # @param node [Parser::AST::Node]
156
+ def splatted_call? node
157
+ return false unless Parser.is_ast_node?(node)
158
+ Parser.is_ast_node?(node.children[0]) && node.children[0].type == :kwsplat && node.children[0].children[0].type != :hash
159
+ end
160
+
161
+ # @param nodes [Enumerable<Parser::AST::Node>]
162
+ def any_splatted_call?(nodes)
163
+ nodes.any? { |n| splatted_call?(n) }
164
+ end
165
+
166
+ # @todo Temporarily here for testing. Move to Solargraph::Parser.
167
+ # @param node [Parser::AST::Node]
168
+ # @return [Array<Parser::AST::Node>]
169
+ def call_nodes_from node
170
+ return [] unless node.is_a?(::Parser::AST::Node)
171
+ result = []
172
+ if node.type == :block
173
+ result.push node
174
+ if Parser.is_ast_node?(node.children[0]) && node.children[0].children.length > 2
175
+ node.children[0].children[2..-1].each { |child| result.concat call_nodes_from(child) }
176
+ end
177
+ node.children[1..-1].each { |child| result.concat call_nodes_from(child) }
178
+ elsif node.type == :send
179
+ result.push node
180
+ node.children[2..-1].each { |child| result.concat call_nodes_from(child) }
181
+ elsif [:super, :zsuper].include?(node.type)
182
+ result.push node
183
+ node.children.each { |child| result.concat call_nodes_from(child) }
184
+ elsif node.type == :masgn
185
+ # @todo We're treating a mass assignment as a call node, but the
186
+ # type checker still needs the logic to handle it.
187
+ result.push node
188
+ else
189
+ node.children.each { |child| result.concat call_nodes_from(child) }
190
+ end
191
+ result
192
+ end
193
+
194
+ # Find all the nodes within the provided node that potentially return a
195
+ # value.
196
+ #
197
+ # The node parameter typically represents a method's logic, e.g., the
198
+ # second child (after the :args node) of a :def node. A simple one-line
199
+ # method would typically return itself, while a node with conditions
200
+ # would return the resulting node from each conditional branch. Nodes
201
+ # that follow a :return node are assumed to be unreachable. Nil values
202
+ # are converted to nil node types.
203
+ #
204
+ # @param node [Parser::AST::Node]
205
+ # @return [Array<Parser::AST::Node>]
206
+ def returns_from_method_body node
207
+ # @todo is the || NIL_NODE necessary?
208
+ # STDERR.puts("Evaluating expression: #{node.inspect}")
209
+ DeepInference.from_method_body(node).map { |n| n || NIL_NODE }
210
+ end
211
+
212
+ # @param node [Parser::AST::Node]
213
+ # @return [Array<AST::Node>] low-level value nodes in
214
+ # value position. Does not include explicit return
215
+ # statements
216
+ def value_position_nodes_only(node)
217
+ DeepInference.value_position_nodes_only(node).map { |n| n || NIL_NODE }
218
+ end
219
+
220
+ # @param cursor [Solargraph::Source::Cursor]
221
+ # @return [Parser::AST::Node, nil]
222
+ def find_recipient_node cursor
223
+ return repaired_find_recipient_node(cursor) if cursor.source.repaired? && cursor.source.code[cursor.offset - 1] == '('
224
+ source = cursor.source
225
+ position = cursor.position
226
+ offset = cursor.offset
227
+ tree = if source.synchronized?
228
+ match = source.code[0..offset-1].match(/,\s*\z/)
229
+ if match
230
+ source.tree_at(position.line, position.column - match[0].length)
231
+ else
232
+ source.tree_at(position.line, position.column)
233
+ end
234
+ else
235
+ source.tree_at(position.line, position.column - 1)
236
+ end
237
+ prev = nil
238
+ tree.each do |node|
239
+ if node.type == :send
240
+ args = node.children[2..-1]
241
+ if !args.empty?
242
+ return node if prev && args.include?(prev)
243
+ else
244
+ if source.synchronized?
245
+ return node if source.code[0..offset-1] =~ /\(\s*\z/ && source.code[offset..-1] =~ /^\s*\)/
246
+ else
247
+ return node if source.code[0..offset-1] =~ /\([^\(]*\z/
248
+ end
249
+ end
250
+ end
251
+ prev = node
252
+ end
253
+ nil
254
+ end
255
+
256
+ # @param cursor [Solargraph::Source::Cursor]
257
+ # @return [Parser::AST::Node, nil]
258
+ def repaired_find_recipient_node cursor
259
+ cursor = cursor.source.cursor_at([cursor.position.line, cursor.position.column - 1])
260
+ node = cursor.source.tree_at(cursor.position.line, cursor.position.column).first
261
+ return node if node && node.type == :send
262
+ end
263
+
264
+ #
265
+ # Concepts:
266
+ #
267
+ # - statement - one single node in the AST. Generally used
268
+ # synonymously with how the Parser gem uses the term
269
+ # 'expression'. This may have side effects (e.g.,
270
+ # registering a method in the namespace, modifying
271
+ # variables or doing I/O). It may encapsulate multiple
272
+ # other statements (see compound statement).
273
+ #
274
+ # - value - something that can be assigned to a variable by
275
+ # evaluating a statement
276
+ #
277
+ # - value node - the 'lowest level' AST node whose return
278
+ # type is a subset of the value type of the overall
279
+ # statement. Might be a literal, a method call, etc - the
280
+ # goal is to find the lowest level node, which we can use
281
+ # Chains and Pins later on to determine the type of.
282
+ #
283
+ # e.g., if the node 'b ? 123 : 456' were a return value, we
284
+ # know the actual return values possible are 123 and 456,
285
+ # and can disregard the rest.
286
+ #
287
+ # - value type - the type representing the multiple possible
288
+ # values that can result from evaluation of the statement.
289
+ #
290
+ # - return type - the type describing the values a statement
291
+ # might evaluate to. When used with a method, the term
292
+ # describes the values that may result from the method
293
+ # being called, and includes explicit return statements
294
+ # within the method body's closure.
295
+ #
296
+ # - method body - a compound statement with parameters whose
297
+ # return value type must account both for the explicit
298
+ # 'return' statemnts as well as the final statements
299
+ # executed in any given control flow through the method.
300
+ #
301
+ # - explicit return statement - a statement which, when part of a
302
+ # method body, is a possible value of a call to that method -
303
+ # e.g., "return 123"
304
+ #
305
+ # - compound statement - a statement which can be expanded to
306
+ # be multiple statements in a row, executed in the context
307
+ # of a method which can be explicitly returned from.
308
+ #
309
+ # - value position - the positions in the AST where the
310
+ # return type of the statement would be one of the return
311
+ # types of any compound statements it is a part of. For a
312
+ # compound statement, the last of the child statements
313
+ # would be in return position. This concept can be applied
314
+ # recursively through e.g. conditionals to find a list of
315
+ # statements in value positions.
316
+ module DeepInference
317
+ class << self
318
+ CONDITIONAL_ALL_BUT_FIRST = [:if, :unless, :or_asgn]
319
+ CONDITIONAL_ALL = [:or]
320
+ ONLY_ONE_CHILD = [:return]
321
+ FIRST_TWO_CHILDREN = [:rescue]
322
+ COMPOUND_STATEMENTS = [:begin, :kwbegin]
323
+ SKIPPABLE = [:def, :defs, :class, :sclass, :module]
324
+ FUNCTION_VALUE = [:block]
325
+ CASE_STATEMENT = [:case]
326
+
327
+ # @param node [AST::Node] a method body compound statement
328
+ # @param include_explicit_returns [Boolean] If true,
329
+ # include the value nodes of the parameter of the
330
+ # 'return' statements in the type returned.
331
+ # @return [Array<AST::Node>] low-level value nodes from
332
+ # both nodes in value position as well as explicit
333
+ # return statements in the method's closure.
334
+ def from_method_body node
335
+ from_value_position_statement(node, include_explicit_returns: true)
336
+ end
337
+
338
+ # @param node [AST::Node] an individual statement, to be
339
+ # evaluated outside the context of a containing method
340
+ # @return [Array<AST::Node>] low-level value nodes in
341
+ # value position. Does not include explicit return
342
+ # statements
343
+ def value_position_nodes_only(node)
344
+ from_value_position_statement(node, include_explicit_returns: false)
345
+ end
346
+
347
+ # Look at known control statements and use them to find
348
+ # more specific return nodes.
349
+ #
350
+ # @param node [Parser::AST::Node] Statement which is in
351
+ # value position for a method body
352
+ # @param include_explicit_returns [Boolean] If true,
353
+ # include the value nodes of the parameter of the
354
+ # 'return' statements in the type returned.
355
+ # @return [Array<Parser::AST::Node>]
356
+ def from_value_position_statement node, include_explicit_returns: true
357
+ # STDERR.puts("from_expression called on #{node.inspect}")
358
+ return [] unless node.is_a?(::Parser::AST::Node)
359
+ # @type [Array<Parser::AST::Node>]
360
+ result = []
361
+ if COMPOUND_STATEMENTS.include?(node.type)
362
+ result.concat from_value_position_compound_statement node
363
+ elsif CONDITIONAL_ALL_BUT_FIRST.include?(node.type)
364
+ result.concat reduce_to_value_nodes(node.children[1..-1])
365
+ # result.push NIL_NODE unless node.children[2]
366
+ elsif CONDITIONAL_ALL.include?(node.type)
367
+ result.concat reduce_to_value_nodes(node.children)
368
+ elsif ONLY_ONE_CHILD.include?(node.type)
369
+ result.concat reduce_to_value_nodes([node.children[0]])
370
+ elsif FIRST_TWO_CHILDREN.include?(node.type)
371
+ result.concat reduce_to_value_nodes([node.children[0], node.children[1]])
372
+ elsif FUNCTION_VALUE.include?(node.type)
373
+ # the block itself is a first class value that could be returned
374
+ result.push node
375
+ # @todo any explicit returns actually return from
376
+ # scope in which the proc is run. This asssumes
377
+ # that the function is executed here.
378
+ result.concat explicit_return_values_from_compound_statement(node.children[2]) if include_explicit_returns
379
+ elsif CASE_STATEMENT.include?(node.type)
380
+ node.children[1..-1].each do |cc|
381
+ if cc.nil?
382
+ result.push NIL_NODE
383
+ elsif cc.type == :when
384
+ result.concat reduce_to_value_nodes([cc.children.last])
385
+ else
386
+ # else clause in case
387
+ result.concat reduce_to_value_nodes([cc])
388
+ end
389
+ end
390
+ elsif node.type == :resbody
391
+ result.concat reduce_to_value_nodes([node.children[2]])
392
+ else
393
+ result.push node
394
+ end
395
+ result
396
+ end
397
+
398
+ # Treat parent as as a begin block and use the last node's
399
+ # return node plus any explicit return nodes' return nodes. e.g.,
400
+ #
401
+ # 123
402
+ # 456
403
+ # return 'a' if foo == bar
404
+ # 789
405
+ #
406
+ # would return 'a' and 789.
407
+ #
408
+ # @param parent [Parser::AST::Node]
409
+ #
410
+ # @return [Array<Parser::AST::Node>]
411
+ def from_value_position_compound_statement parent
412
+ result = []
413
+ nodes = parent.children.select{|n| n.is_a?(AST::Node)}
414
+ nodes.each_with_index do |node, idx|
415
+ if node.type == :block
416
+ result.concat explicit_return_values_from_compound_statement(node.children[2])
417
+ elsif node.type == :rescue
418
+ # body statements
419
+ result.concat from_value_position_statement(node.children[0])
420
+ # rescue statements
421
+ result.concat from_value_position_statement(node.children[1])
422
+ elsif SKIPPABLE.include?(node.type)
423
+ next
424
+ elsif node.type == :resbody
425
+ result.concat reduce_to_value_nodes([node.children[2]])
426
+ elsif node.type == :return
427
+ result.concat reduce_to_value_nodes([node.children[0]])
428
+ # Return here because the rest of the code is
429
+ # unreachable and shouldn't be looked at
430
+ return result
431
+ else
432
+ result.concat explicit_return_values_from_compound_statement(node)
433
+ end
434
+ # handle last line of compound statements, which is in
435
+ # value position. we already have the explicit values
436
+ # from above; now we need to also gather the value
437
+ # position nodes
438
+ result.concat from_value_position_statement(nodes.last, include_explicit_returns: false) if idx == nodes.length - 1
439
+ end
440
+ result
441
+ end
442
+
443
+ private
444
+
445
+ # Useful when this statement isn't in value position, but
446
+ # we care explicit return statements nonetheless.
447
+ #
448
+ # @param parent [Parser::AST::Node]
449
+ #
450
+ # @return [Array<Parser::AST::Node>]
451
+ def explicit_return_values_from_compound_statement parent
452
+ return [] unless parent.is_a?(::Parser::AST::Node)
453
+ result = []
454
+ nodes = parent.children.select{|n| n.is_a?(::Parser::AST::Node)}
455
+ nodes.each do |node|
456
+ next if SKIPPABLE.include?(node.type)
457
+ if node.type == :return
458
+ result.concat reduce_to_value_nodes([node.children[0]])
459
+ # Return the result here because the rest of the code is
460
+ # unreachable
461
+ return result
462
+ else
463
+ result.concat explicit_return_values_from_compound_statement(node)
464
+ end
465
+ end
466
+ result
467
+ end
468
+
469
+ # @param nodes [Enumerable<Parser::AST::Node, BaseObject>]
470
+ # @return [Array<Parser::AST::Node, nil>]
471
+ def reduce_to_value_nodes nodes
472
+ result = []
473
+ nodes.each do |node|
474
+ if !node.is_a?(::Parser::AST::Node)
475
+ result.push nil
476
+ elsif COMPOUND_STATEMENTS.include?(node.type)
477
+ result.concat from_value_position_compound_statement(node)
478
+ elsif CONDITIONAL_ALL_BUT_FIRST.include?(node.type)
479
+ result.concat reduce_to_value_nodes(node.children[1..-1])
480
+ elsif node.type == :return
481
+ result.concat reduce_to_value_nodes([node.children[0]])
482
+ elsif node.type == :or
483
+ result.concat reduce_to_value_nodes(node.children)
484
+ elsif node.type == :block
485
+ result.concat explicit_return_values_from_compound_statement(node.children[2])
486
+ elsif node.type == :resbody
487
+ result.concat reduce_to_value_nodes([node.children[2]])
488
+ else
489
+ result.push node
490
+ end
491
+ end
492
+ result
493
+ end
494
+ end
495
+ end
496
+ end
497
+ end
498
+ end
499
+ end
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Solargraph
4
4
  module Parser
5
- module Rubyvm
5
+ module ParserGem
6
6
  module NodeProcessors
7
7
  class AliasNode < Parser::NodeProcessor::Base
8
8
  def process