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.
- checksums.yaml +4 -4
- data/.github/FUNDING.yml +1 -0
- data/.github/workflows/plugins.yml +40 -0
- data/.github/workflows/rspec.yml +4 -8
- data/.github/workflows/typecheck.yml +34 -0
- data/.yardopts +2 -2
- data/CHANGELOG.md +137 -3
- data/LICENSE +1 -1
- data/README.md +19 -16
- data/SPONSORS.md +2 -9
- data/lib/solargraph/api_map/cache.rb +60 -20
- data/lib/solargraph/api_map/source_to_yard.rb +17 -10
- data/lib/solargraph/api_map/store.rb +60 -12
- data/lib/solargraph/api_map.rb +171 -99
- data/lib/solargraph/bench.rb +3 -2
- data/lib/solargraph/cache.rb +77 -0
- data/lib/solargraph/complex_type/type_methods.rb +61 -12
- data/lib/solargraph/complex_type/unique_type.rb +193 -16
- data/lib/solargraph/complex_type.rb +113 -10
- data/lib/solargraph/convention/rakefile.rb +17 -0
- data/lib/solargraph/convention.rb +2 -3
- data/lib/solargraph/converters/dd.rb +5 -0
- data/lib/solargraph/converters/dl.rb +3 -0
- data/lib/solargraph/converters/dt.rb +3 -0
- data/lib/solargraph/diagnostics/rubocop.rb +23 -8
- data/lib/solargraph/diagnostics/rubocop_helpers.rb +4 -1
- data/lib/solargraph/diagnostics/type_check.rb +1 -0
- data/lib/solargraph/diagnostics.rb +2 -2
- data/lib/solargraph/doc_map.rb +171 -0
- data/lib/solargraph/gem_pins.rb +64 -0
- data/lib/solargraph/language_server/host/cataloger.rb +2 -1
- data/lib/solargraph/language_server/host/diagnoser.rb +2 -2
- data/lib/solargraph/language_server/host/dispatch.rb +15 -5
- data/lib/solargraph/language_server/host/message_worker.rb +4 -0
- data/lib/solargraph/language_server/host/sources.rb +7 -4
- data/lib/solargraph/language_server/host.rb +50 -26
- data/lib/solargraph/language_server/message/completion_item/resolve.rb +3 -1
- data/lib/solargraph/language_server/message/extended/check_gem_version.rb +13 -1
- data/lib/solargraph/language_server/message/extended/download_core.rb +1 -5
- data/lib/solargraph/language_server/message/initialize.rb +13 -0
- data/lib/solargraph/language_server/message/initialized.rb +1 -0
- data/lib/solargraph/language_server/message/text_document/document_symbol.rb +4 -1
- data/lib/solargraph/language_server/message/text_document/formatting.rb +4 -4
- data/lib/solargraph/language_server/message/text_document/hover.rb +2 -0
- data/lib/solargraph/language_server/message/text_document/signature_help.rb +1 -6
- data/lib/solargraph/language_server/message/text_document/type_definition.rb +24 -0
- data/lib/solargraph/language_server/message/text_document.rb +1 -1
- data/lib/solargraph/language_server/message/workspace/did_change_configuration.rb +5 -0
- data/lib/solargraph/language_server/message/workspace/did_change_watched_files.rb +10 -3
- data/lib/solargraph/language_server/message.rb +1 -0
- data/lib/solargraph/language_server/transport/adapter.rb +16 -1
- data/lib/solargraph/language_server/transport/data_reader.rb +2 -0
- data/lib/solargraph/library.rb +124 -37
- data/lib/solargraph/location.rb +1 -0
- data/lib/solargraph/page.rb +6 -0
- data/lib/solargraph/parser/comment_ripper.rb +4 -0
- data/lib/solargraph/parser/node_methods.rb +47 -7
- data/lib/solargraph/parser/node_processor/base.rb +9 -0
- data/lib/solargraph/parser/{legacy → parser_gem}/class_methods.rb +31 -5
- data/lib/solargraph/parser/{legacy → parser_gem}/flawed_builder.rb +3 -1
- data/lib/solargraph/parser/{legacy → parser_gem}/node_chainer.rb +57 -41
- data/lib/solargraph/parser/parser_gem/node_methods.rb +499 -0
- data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/alias_node.rb +1 -1
- data/lib/solargraph/parser/parser_gem/node_processors/args_node.rb +53 -0
- data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/begin_node.rb +1 -1
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/block_node.rb +3 -2
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/casgn_node.rb +14 -4
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/cvasgn_node.rb +1 -1
- data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/def_node.rb +7 -20
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/defs_node.rb +2 -2
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/gvasgn_node.rb +1 -1
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/ivasgn_node.rb +2 -2
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/lvasgn_node.rb +2 -2
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/namespace_node.rb +2 -2
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/orasgn_node.rb +1 -1
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/resbody_node.rb +3 -3
- data/lib/solargraph/parser/parser_gem/node_processors/sclass_node.rb +42 -0
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/send_node.rb +2 -2
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/sym_node.rb +1 -1
- data/lib/solargraph/parser/parser_gem/node_processors.rb +54 -0
- data/lib/solargraph/parser/parser_gem.rb +12 -0
- data/lib/solargraph/parser/region.rb +1 -1
- data/lib/solargraph/parser/snippet.rb +2 -0
- data/lib/solargraph/parser.rb +9 -10
- data/lib/solargraph/pin/base.rb +69 -11
- data/lib/solargraph/pin/base_variable.rb +8 -4
- data/lib/solargraph/pin/block.rb +21 -28
- data/lib/solargraph/pin/closure.rb +17 -2
- data/lib/solargraph/pin/common.rb +7 -3
- data/lib/solargraph/pin/conversions.rb +34 -8
- data/lib/solargraph/pin/delegated_method.rb +97 -0
- data/lib/solargraph/pin/documenting.rb +25 -34
- data/lib/solargraph/pin/instance_variable.rb +4 -0
- data/lib/solargraph/pin/local_variable.rb +13 -1
- data/lib/solargraph/pin/method.rb +270 -16
- data/lib/solargraph/pin/namespace.rb +17 -1
- data/lib/solargraph/pin/parameter.rb +52 -17
- data/lib/solargraph/pin/reference/override.rb +2 -2
- data/lib/solargraph/pin/reference.rb +8 -0
- data/lib/solargraph/pin/search.rb +4 -4
- data/lib/solargraph/pin/signature.rb +143 -0
- data/lib/solargraph/pin.rb +2 -1
- data/lib/solargraph/range.rb +4 -6
- data/lib/solargraph/rbs_map/conversions.rb +601 -0
- data/lib/solargraph/rbs_map/core_fills.rb +47 -0
- data/lib/solargraph/rbs_map/core_map.rb +28 -0
- data/lib/solargraph/rbs_map/stdlib_map.rb +33 -0
- data/lib/solargraph/rbs_map.rb +84 -0
- data/lib/solargraph/shell.rb +69 -48
- data/lib/solargraph/source/chain/array.rb +32 -0
- data/lib/solargraph/source/chain/block_symbol.rb +13 -0
- data/lib/solargraph/source/chain/call.rb +125 -61
- data/lib/solargraph/source/chain/constant.rb +15 -1
- data/lib/solargraph/source/chain/if.rb +23 -0
- data/lib/solargraph/source/chain/link.rb +8 -2
- data/lib/solargraph/source/chain/or.rb +1 -1
- data/lib/solargraph/source/chain/z_super.rb +3 -3
- data/lib/solargraph/source/chain.rb +44 -14
- data/lib/solargraph/source/change.rb +3 -0
- data/lib/solargraph/source/cursor.rb +2 -0
- data/lib/solargraph/source/source_chainer.rb +8 -5
- data/lib/solargraph/source.rb +18 -19
- data/lib/solargraph/source_map/clip.rb +30 -23
- data/lib/solargraph/source_map/mapper.rb +20 -5
- data/lib/solargraph/source_map.rb +28 -13
- data/lib/solargraph/type_checker/checks.rb +10 -2
- data/lib/solargraph/type_checker.rb +201 -98
- data/lib/solargraph/version.rb +1 -1
- data/lib/solargraph/views/environment.erb +2 -2
- data/lib/solargraph/workspace/config.rb +14 -11
- data/lib/solargraph/workspace.rb +28 -17
- data/lib/solargraph/yard_map/cache.rb +6 -0
- data/lib/solargraph/yard_map/helpers.rb +1 -1
- data/lib/solargraph/yard_map/mapper/to_method.rb +18 -5
- data/lib/solargraph/yard_map/mapper.rb +1 -1
- data/lib/solargraph/yard_map/to_method.rb +11 -4
- data/lib/solargraph/yard_map.rb +1 -443
- data/lib/solargraph/yard_tags.rb +20 -0
- data/lib/solargraph/yardoc.rb +52 -0
- data/lib/solargraph.rb +8 -6
- data/solargraph.gemspec +19 -8
- metadata +162 -98
- data/.travis.yml +0 -19
- data/lib/solargraph/api_map/bundler_methods.rb +0 -22
- data/lib/solargraph/compat.rb +0 -37
- data/lib/solargraph/convention/rspec.rb +0 -30
- data/lib/solargraph/documentor.rb +0 -76
- data/lib/solargraph/parser/legacy/node_methods.rb +0 -325
- data/lib/solargraph/parser/legacy/node_processors/alias_node.rb +0 -23
- data/lib/solargraph/parser/legacy/node_processors/args_node.rb +0 -35
- data/lib/solargraph/parser/legacy/node_processors/begin_node.rb +0 -15
- data/lib/solargraph/parser/legacy/node_processors/def_node.rb +0 -63
- data/lib/solargraph/parser/legacy/node_processors/sclass_node.rb +0 -21
- data/lib/solargraph/parser/legacy/node_processors.rb +0 -54
- data/lib/solargraph/parser/legacy.rb +0 -12
- data/lib/solargraph/parser/rubyvm/class_methods.rb +0 -144
- data/lib/solargraph/parser/rubyvm/node_chainer.rb +0 -160
- data/lib/solargraph/parser/rubyvm/node_methods.rb +0 -315
- data/lib/solargraph/parser/rubyvm/node_processors/args_node.rb +0 -85
- data/lib/solargraph/parser/rubyvm/node_processors/block_node.rb +0 -42
- data/lib/solargraph/parser/rubyvm/node_processors/casgn_node.rb +0 -22
- data/lib/solargraph/parser/rubyvm/node_processors/cvasgn_node.rb +0 -23
- data/lib/solargraph/parser/rubyvm/node_processors/defs_node.rb +0 -57
- data/lib/solargraph/parser/rubyvm/node_processors/gvasgn_node.rb +0 -23
- data/lib/solargraph/parser/rubyvm/node_processors/ivasgn_node.rb +0 -38
- data/lib/solargraph/parser/rubyvm/node_processors/kw_arg_node.rb +0 -39
- data/lib/solargraph/parser/rubyvm/node_processors/lit_node.rb +0 -20
- data/lib/solargraph/parser/rubyvm/node_processors/lvasgn_node.rb +0 -27
- data/lib/solargraph/parser/rubyvm/node_processors/namespace_node.rb +0 -39
- data/lib/solargraph/parser/rubyvm/node_processors/opt_arg_node.rb +0 -26
- data/lib/solargraph/parser/rubyvm/node_processors/orasgn_node.rb +0 -15
- data/lib/solargraph/parser/rubyvm/node_processors/resbody_node.rb +0 -45
- data/lib/solargraph/parser/rubyvm/node_processors/sclass_node.rb +0 -21
- data/lib/solargraph/parser/rubyvm/node_processors/scope_node.rb +0 -15
- data/lib/solargraph/parser/rubyvm/node_processors/send_node.rb +0 -277
- data/lib/solargraph/parser/rubyvm/node_processors/sym_node.rb +0 -18
- data/lib/solargraph/parser/rubyvm/node_processors.rb +0 -63
- data/lib/solargraph/parser/rubyvm.rb +0 -40
- data/lib/solargraph/yard_map/core_docs.rb +0 -170
- data/lib/solargraph/yard_map/core_fills.rb +0 -208
- data/lib/solargraph/yard_map/core_gen.rb +0 -76
- data/lib/solargraph/yard_map/rdoc_to_yard.rb +0 -140
- data/lib/solargraph/yard_map/stdlib_fills.rb +0 -43
- data/lib/yard-solargraph.rb +0 -33
- 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
|
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
|
-
|
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
|
-
@
|
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,
|
31
|
-
NodeChainer.new(node, filename,
|
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
|
-
@
|
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
|
-
|
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,
|
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
|
-
|
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
|
-
|
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'
|
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
|
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
|
112
|
+
result.concat generate_links(n.children.last)
|
120
113
|
elsif n.type == :block_pass
|
121
|
-
|
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
|
-
|
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
|
-
|
143
|
-
|
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
|