solargraph 0.52.0 → 0.53.4
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/workflows/plugins.yml +40 -0
- data/.github/workflows/rspec.yml +1 -3
- data/.github/workflows/typecheck.yml +34 -0
- data/CHANGELOG.md +53 -0
- data/README.md +13 -16
- data/SPONSORS.md +1 -7
- data/lib/solargraph/api_map/cache.rb +59 -21
- data/lib/solargraph/api_map/source_to_yard.rb +17 -10
- data/lib/solargraph/api_map/store.rb +45 -9
- data/lib/solargraph/api_map.rb +178 -113
- data/lib/solargraph/bench.rb +3 -2
- data/lib/solargraph/cache.rb +29 -5
- data/lib/solargraph/complex_type/type_methods.rb +53 -8
- data/lib/solargraph/complex_type/unique_type.rb +171 -58
- data/lib/solargraph/complex_type.rb +62 -9
- data/lib/solargraph/convention.rb +0 -1
- 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 +8 -7
- data/lib/solargraph/diagnostics/rubocop_helpers.rb +1 -0
- 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 +1 -0
- 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 +35 -7
- 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/initialize.rb +5 -2
- data/lib/solargraph/language_server/message/text_document/hover.rb +2 -0
- data/lib/solargraph/language_server/message/text_document.rb +0 -1
- data/lib/solargraph/language_server/message/workspace/did_change_configuration.rb +5 -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 +71 -12
- data/lib/solargraph/location.rb +1 -0
- data/lib/solargraph/page.rb +6 -0
- data/lib/solargraph/parser/comment_ripper.rb +3 -0
- data/lib/solargraph/parser/node_methods.rb +47 -8
- data/lib/solargraph/parser/node_processor/base.rb +9 -0
- data/lib/solargraph/parser/{legacy → parser_gem}/class_methods.rb +29 -3
- data/lib/solargraph/parser/{legacy → parser_gem}/flawed_builder.rb +3 -1
- data/lib/solargraph/parser/{legacy → parser_gem}/node_chainer.rb +42 -34
- data/lib/solargraph/parser/{legacy → parser_gem}/node_methods.rb +201 -29
- data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/alias_node.rb +1 -1
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/args_node.rb +4 -1
- 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 +2 -2
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/cvasgn_node.rb +1 -1
- data/lib/solargraph/parser/{legacy → 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/{legacy → parser_gem}/node_processors/sclass_node.rb +1 -1
- data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/send_node.rb +2 -2
- data/lib/solargraph/parser/{rubyvm → 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/snippet.rb +2 -0
- data/lib/solargraph/parser.rb +8 -11
- data/lib/solargraph/pin/base.rb +63 -8
- data/lib/solargraph/pin/base_variable.rb +7 -3
- data/lib/solargraph/pin/block.rb +26 -38
- data/lib/solargraph/pin/closure.rb +17 -2
- data/lib/solargraph/pin/common.rb +7 -3
- data/lib/solargraph/pin/conversions.rb +33 -3
- 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 +110 -16
- data/lib/solargraph/pin/namespace.rb +16 -10
- data/lib/solargraph/pin/parameter.rb +41 -10
- data/lib/solargraph/pin/reference/override.rb +2 -2
- data/lib/solargraph/pin/reference.rb +8 -0
- data/lib/solargraph/pin/search.rb +3 -3
- data/lib/solargraph/pin/signature.rb +114 -2
- data/lib/solargraph/pin.rb +0 -1
- data/lib/solargraph/range.rb +2 -2
- data/lib/solargraph/rbs_map/conversions.rb +213 -61
- data/lib/solargraph/rbs_map/core_fills.rb +12 -28
- data/lib/solargraph/rbs_map/core_map.rb +2 -12
- data/lib/solargraph/rbs_map/stdlib_map.rb +2 -8
- data/lib/solargraph/rbs_map.rb +24 -12
- data/lib/solargraph/shell.rb +62 -59
- data/lib/solargraph/source/chain/array.rb +4 -1
- data/lib/solargraph/source/chain/block_symbol.rb +13 -0
- data/lib/solargraph/source/chain/call.rb +95 -26
- 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 +7 -1
- data/lib/solargraph/source/chain/or.rb +1 -1
- data/lib/solargraph/source/chain/z_super.rb +2 -2
- data/lib/solargraph/source/chain.rb +20 -4
- data/lib/solargraph/source/change.rb +3 -0
- data/lib/solargraph/source/cursor.rb +2 -0
- data/lib/solargraph/source/source_chainer.rb +6 -5
- data/lib/solargraph/source.rb +15 -16
- data/lib/solargraph/source_map/clip.rb +14 -9
- data/lib/solargraph/source_map/mapper.rb +10 -0
- data/lib/solargraph/source_map.rb +12 -10
- data/lib/solargraph/type_checker/checks.rb +10 -2
- data/lib/solargraph/type_checker.rb +96 -21
- data/lib/solargraph/version.rb +1 -1
- data/lib/solargraph/workspace/config.rb +8 -6
- data/lib/solargraph/workspace.rb +15 -2
- 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 +16 -3
- data/lib/solargraph/yard_map/to_method.rb +11 -4
- data/lib/solargraph/yard_map.rb +0 -292
- data/lib/solargraph/yardoc.rb +52 -0
- data/lib/solargraph.rb +4 -1
- data/solargraph.gemspec +2 -2
- metadata +35 -58
- data/lib/solargraph/api_map/bundler_methods.rb +0 -22
- data/lib/solargraph/documentor.rb +0 -76
- data/lib/solargraph/parser/legacy/node_processors/alias_node.rb +0 -23
- data/lib/solargraph/parser/legacy/node_processors/begin_node.rb +0 -15
- data/lib/solargraph/parser/legacy/node_processors/sym_node.rb +0 -18
- data/lib/solargraph/parser/legacy/node_processors.rb +0 -55
- data/lib/solargraph/parser/legacy.rb +0 -12
- data/lib/solargraph/parser/rubyvm/class_methods.rb +0 -151
- data/lib/solargraph/parser/rubyvm/node_chainer.rb +0 -163
- data/lib/solargraph/parser/rubyvm/node_methods.rb +0 -317
- 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 -33
- data/lib/solargraph/parser/rubyvm/node_processors/cvasgn_node.rb +0 -23
- data/lib/solargraph/parser/rubyvm/node_processors/def_node.rb +0 -75
- data/lib/solargraph/parser/rubyvm/node_processors/defs_node.rb +0 -68
- 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 -51
- data/lib/solargraph/parser/rubyvm/node_processors/sclass_node.rb +0 -32
- data/lib/solargraph/parser/rubyvm/node_processors/scope_node.rb +0 -15
- data/lib/solargraph/parser/rubyvm/node_processors/send_node.rb +0 -279
- data/lib/solargraph/parser/rubyvm/node_processors.rb +0 -64
- data/lib/solargraph/parser/rubyvm/node_wrapper.rb +0 -47
- data/lib/solargraph/parser/rubyvm.rb +0 -40
- data/lib/solargraph/rbs_map/core_signs.rb +0 -33
@@ -2,9 +2,23 @@
|
|
2
2
|
|
3
3
|
module Solargraph
|
4
4
|
class ComplexType
|
5
|
-
# Methods for accessing type data
|
5
|
+
# Methods for accessing type data available from
|
6
|
+
# both ComplexType and UniqueType.
|
6
7
|
#
|
8
|
+
# @abstract This mixin relies on these -
|
9
|
+
# instance variables:
|
10
|
+
# @name: String
|
11
|
+
# @subtypes: Array<ComplexType>
|
12
|
+
# @rooted: boolish
|
13
|
+
# methods:
|
14
|
+
# transform()
|
7
15
|
module TypeMethods
|
16
|
+
# @!method transform(new_name = nil, &transform_type)
|
17
|
+
# @param new_name [String, nil]
|
18
|
+
# @yieldparam t [UniqueType]
|
19
|
+
# @yieldreturn [UniqueType]
|
20
|
+
# @return [UniqueType, nil]
|
21
|
+
|
8
22
|
# @return [String]
|
9
23
|
attr_reader :name
|
10
24
|
|
@@ -24,8 +38,7 @@ module Solargraph
|
|
24
38
|
|
25
39
|
# @return [Boolean]
|
26
40
|
def nil_type?
|
27
|
-
@nil_type
|
28
|
-
@nil_type
|
41
|
+
@nil_type ||= (name.casecmp('nil') == 0)
|
29
42
|
end
|
30
43
|
|
31
44
|
# @return [Boolean]
|
@@ -45,6 +58,22 @@ module Solargraph
|
|
45
58
|
name == 'undefined'
|
46
59
|
end
|
47
60
|
|
61
|
+
# @param generics_to_erase [Enumerable<String>]
|
62
|
+
# @return [self]
|
63
|
+
def erase_generics(generics_to_erase)
|
64
|
+
transform do |type|
|
65
|
+
if type.name == ComplexType::GENERIC_TAG_NAME
|
66
|
+
if type.all_params.length == 1 && generics_to_erase.include?(type.all_params.first.to_s)
|
67
|
+
ComplexType::UNDEFINED
|
68
|
+
else
|
69
|
+
type
|
70
|
+
end
|
71
|
+
else
|
72
|
+
type
|
73
|
+
end
|
74
|
+
end
|
75
|
+
end
|
76
|
+
|
48
77
|
# @return [Boolean]
|
49
78
|
def list_parameters?
|
50
79
|
substring.start_with?('<')
|
@@ -80,12 +109,25 @@ module Solargraph
|
|
80
109
|
end.call
|
81
110
|
end
|
82
111
|
|
83
|
-
# @return [
|
112
|
+
# @return [String]
|
113
|
+
def rooted_namespace
|
114
|
+
return namespace unless rooted?
|
115
|
+
"::#{namespace}"
|
116
|
+
end
|
117
|
+
|
118
|
+
# @return [String]
|
119
|
+
def rooted_name
|
120
|
+
return name unless rooted?
|
121
|
+
"::#{name}"
|
122
|
+
end
|
123
|
+
|
124
|
+
# @return [::Symbol] :class or :instance
|
84
125
|
def scope
|
85
126
|
@scope ||= :instance if duck_type? || nil_type?
|
86
127
|
@scope ||= (name == 'Class' || name == 'Module') && !subtypes.empty? ? :class : :instance
|
87
128
|
end
|
88
129
|
|
130
|
+
# @param other [Object]
|
89
131
|
def == other
|
90
132
|
return false unless self.class == other.class
|
91
133
|
tag == other.tag
|
@@ -99,7 +141,7 @@ module Solargraph
|
|
99
141
|
#
|
100
142
|
# @param api_map [ApiMap] The ApiMap that performs qualification
|
101
143
|
# @param context [String] The namespace from which to resolve names
|
102
|
-
# @return [ComplexType] The generated ComplexType
|
144
|
+
# @return [self, ComplexType, UniqueType] The generated ComplexType
|
103
145
|
def qualify api_map, context = ''
|
104
146
|
return self if name == GENERIC_TAG_NAME
|
105
147
|
return ComplexType.new([self]) if duck_type? || void? || undefined?
|
@@ -110,13 +152,16 @@ module Solargraph
|
|
110
152
|
return UniqueType::UNDEFINED
|
111
153
|
end
|
112
154
|
fqns = "::#{fqns}" # Ensure the resulting complex type is rooted
|
113
|
-
|
114
|
-
|
155
|
+
all_ltypes = key_types.map { |t| t.qualify api_map, context }.uniq
|
156
|
+
all_rtypes = value_types.map { |t| t.qualify api_map, context }
|
115
157
|
if list_parameters?
|
158
|
+
rtypes = all_rtypes.uniq
|
116
159
|
Solargraph::ComplexType.parse("#{fqns}<#{rtypes.map(&:tag).join(', ')}>")
|
117
160
|
elsif fixed_parameters?
|
118
|
-
Solargraph::ComplexType.parse("#{fqns}(#{
|
161
|
+
Solargraph::ComplexType.parse("#{fqns}(#{all_rtypes.map(&:tag).join(', ')})")
|
119
162
|
elsif hash_parameters?
|
163
|
+
ltypes = all_ltypes.uniq
|
164
|
+
rtypes = all_rtypes.uniq
|
120
165
|
Solargraph::ComplexType.parse("#{fqns}{#{ltypes.map(&:tag).join(', ')} => #{rtypes.map(&:tag).join(', ')}}")
|
121
166
|
else
|
122
167
|
Solargraph::ComplexType.parse(fqns)
|
@@ -8,7 +8,7 @@ module Solargraph
|
|
8
8
|
class UniqueType
|
9
9
|
include TypeMethods
|
10
10
|
|
11
|
-
attr_reader :all_params
|
11
|
+
attr_reader :all_params, :subtypes, :key_types
|
12
12
|
|
13
13
|
# Create a UniqueType with the specified name and an optional substring.
|
14
14
|
# The substring is the parameter section of a parametrized type, e.g.,
|
@@ -27,18 +27,27 @@ module Solargraph
|
|
27
27
|
end
|
28
28
|
@substring = substring
|
29
29
|
@tag = @name + substring
|
30
|
+
# @type [Array<ComplexType>]
|
30
31
|
@key_types = []
|
32
|
+
# @type [Array<ComplexType>]
|
31
33
|
@subtypes = []
|
34
|
+
# @type [Array<ComplexType>]
|
32
35
|
@all_params = []
|
33
36
|
return unless parameters?
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
37
|
+
# @todo we should be able to probe the type of 'subs' without
|
38
|
+
# hoisting the definition outside of the if statement
|
39
|
+
subs = if @substring.start_with?('<(') && @substring.end_with?(')>')
|
40
|
+
ComplexType.parse(substring[2..-3], partial: true)
|
41
|
+
else
|
42
|
+
ComplexType.parse(substring[1..-2], partial: true)
|
43
|
+
end
|
39
44
|
if hash_parameters?
|
40
45
|
raise ComplexTypeError, "Bad hash type" unless !subs.is_a?(ComplexType) and subs.length == 2 and !subs[0].is_a?(UniqueType) and !subs[1].is_a?(UniqueType)
|
46
|
+
# @todo should be able to resolve map; both types have it
|
47
|
+
# with same return type
|
48
|
+
# @sg-ignore
|
41
49
|
@key_types.concat subs[0].map { |u| ComplexType.new([u]) }
|
50
|
+
# @sg-ignore
|
42
51
|
@subtypes.concat subs[1].map { |u| ComplexType.new([u]) }
|
43
52
|
else
|
44
53
|
@subtypes.concat subs
|
@@ -51,80 +60,184 @@ module Solargraph
|
|
51
60
|
tag
|
52
61
|
end
|
53
62
|
|
63
|
+
# @return [Array<UniqueType>]
|
54
64
|
def items
|
55
65
|
[self]
|
56
66
|
end
|
57
67
|
|
68
|
+
# @return [String]
|
69
|
+
def rbs_name
|
70
|
+
if name == 'undefined'
|
71
|
+
'untyped'
|
72
|
+
else
|
73
|
+
rooted_name
|
74
|
+
end
|
75
|
+
end
|
76
|
+
|
77
|
+
# @return [String]
|
58
78
|
def to_rbs
|
59
|
-
|
79
|
+
if ['Tuple', 'Array'].include?(name) && fixed_parameters?
|
80
|
+
# tuples don't have a name; they're just [foo, bar, baz].
|
81
|
+
if substring == '()'
|
82
|
+
# but there are no zero element tuples, so we go with an array
|
83
|
+
'Array[]'
|
84
|
+
else
|
85
|
+
# already generated surrounded by []
|
86
|
+
parameters_as_rbs
|
87
|
+
end
|
88
|
+
else
|
89
|
+
"#{rbs_name}#{parameters_as_rbs}"
|
90
|
+
end
|
91
|
+
end
|
92
|
+
|
93
|
+
# @return [String]
|
94
|
+
def parameters_as_rbs
|
95
|
+
parameters? ? "[#{all_params.map { |s| s.to_rbs }.join(', ')}]" : ''
|
60
96
|
end
|
61
97
|
|
62
98
|
def generic?
|
63
99
|
name == GENERIC_TAG_NAME || all_params.any?(&:generic?)
|
64
100
|
end
|
65
101
|
|
66
|
-
|
67
|
-
# @param
|
68
|
-
# @
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
return
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
@subtypes.map { |t| t.resolve_generics(definitions, context_type) }.select(&:defined?)
|
86
|
-
else
|
87
|
-
[]
|
102
|
+
|
103
|
+
# @param generics_to_resolve [Enumerable<String>]
|
104
|
+
# @param context_type [UniqueType, nil]
|
105
|
+
# @param resolved_generic_values [Hash{String => ComplexType}] Added to as types are encountered or resolved
|
106
|
+
# @return [UniqueType, ComplexType]
|
107
|
+
def resolve_generics_from_context generics_to_resolve, context_type, resolved_generic_values: {}
|
108
|
+
if name == ComplexType::GENERIC_TAG_NAME
|
109
|
+
type_param = subtypes.first&.name
|
110
|
+
return self unless generics_to_resolve.include? type_param
|
111
|
+
unless context_type.nil? || !resolved_generic_values[type_param].nil?
|
112
|
+
new_binding = true
|
113
|
+
resolved_generic_values[type_param] = context_type
|
114
|
+
end
|
115
|
+
if new_binding
|
116
|
+
resolved_generic_values.transform_values! do |complex_type|
|
117
|
+
complex_type.resolve_generics_from_context(generics_to_resolve, nil, resolved_generic_values: resolved_generic_values)
|
118
|
+
end
|
119
|
+
end
|
120
|
+
return resolved_generic_values[type_param] || self
|
88
121
|
end
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
122
|
+
|
123
|
+
# @todo typechecking should complain when the method being called has no @yieldparam tag
|
124
|
+
new_key_types = resolve_param_generics_from_context(generics_to_resolve, context_type, resolved_generic_values, &:key_types)
|
125
|
+
new_subtypes = resolve_param_generics_from_context(generics_to_resolve, context_type, resolved_generic_values, &:subtypes)
|
126
|
+
recreate(new_key_types: new_key_types, new_subtypes: new_subtypes)
|
127
|
+
end
|
128
|
+
|
129
|
+
# @param generics_to_resolve [Enumerable<String>]
|
130
|
+
# @param context_type [UniqueType]
|
131
|
+
# @param resolved_generic_values [Hash{String => ComplexType}]
|
132
|
+
# @yieldreturn [Array<ComplexType>]
|
133
|
+
# @return [Array<ComplexType>]
|
134
|
+
def resolve_param_generics_from_context(generics_to_resolve, context_type, resolved_generic_values)
|
135
|
+
types = yield self
|
136
|
+
types.each_with_index.flat_map do |ct, i|
|
137
|
+
ct.items.flat_map do |ut|
|
138
|
+
context_params = yield context_type if context_type
|
139
|
+
if context_params && context_params[i]
|
140
|
+
type_arg = context_params[i]
|
141
|
+
type_arg.map do |new_unique_context_type|
|
142
|
+
ut.resolve_generics_from_context generics_to_resolve, new_unique_context_type, resolved_generic_values: resolved_generic_values
|
143
|
+
end
|
95
144
|
else
|
96
|
-
|
145
|
+
ut.resolve_generics_from_context generics_to_resolve, nil, resolved_generic_values: resolved_generic_values
|
97
146
|
end
|
147
|
+
end
|
148
|
+
end
|
149
|
+
end
|
150
|
+
|
151
|
+
# Probe the concrete type for each of the generic type
|
152
|
+
# parameters used in this type, and return a new type if
|
153
|
+
# possible.
|
154
|
+
#
|
155
|
+
# @param definitions [Pin::Namespace, Pin::Method] The module/class/method which uses generic types
|
156
|
+
# @param context_type [ComplexType] The receiver type
|
157
|
+
# @return [UniqueType, ComplexType]
|
158
|
+
def resolve_generics definitions, context_type
|
159
|
+
return self if definitions.nil? || definitions.generics.empty?
|
160
|
+
|
161
|
+
transform(name) do |t|
|
162
|
+
if t.name == GENERIC_TAG_NAME
|
163
|
+
idx = definitions.generics.index(t.subtypes.first&.name)
|
164
|
+
next t if idx.nil?
|
165
|
+
context_type.all_params[idx] || ComplexType::UNDEFINED
|
98
166
|
else
|
99
|
-
|
167
|
+
t
|
100
168
|
end
|
101
|
-
else
|
102
|
-
UniqueType.new(new_name)
|
103
169
|
end
|
170
|
+
end
|
104
171
|
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
# ComplexType.try_parse(param_type.to_s)
|
172
|
+
# @yieldparam t [self]
|
173
|
+
# @yieldreturn [self]
|
174
|
+
# @return [Array<self>]
|
175
|
+
def map &block
|
176
|
+
[block.yield(self)]
|
111
177
|
end
|
112
178
|
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
179
|
+
# @return [Array<UniqueType>]
|
180
|
+
def to_a
|
181
|
+
[self]
|
182
|
+
end
|
183
|
+
|
184
|
+
# @param new_name [String, nil]
|
185
|
+
# @param new_key_types [Array<UniqueType>, nil]
|
186
|
+
# @param new_subtypes [Array<UniqueType>, nil]
|
187
|
+
# @return [self]
|
188
|
+
def recreate(new_name: nil, new_key_types: nil, new_subtypes: nil)
|
189
|
+
new_name ||= name
|
190
|
+
new_key_types ||= @key_types
|
191
|
+
new_subtypes ||= @subtypes
|
192
|
+
if new_key_types.none?(&:defined?) && new_subtypes.none?(&:defined?)
|
193
|
+
# if all subtypes are undefined, erase down to the non-parametric type
|
194
|
+
UniqueType.new(new_name)
|
195
|
+
elsif new_key_types.empty? && new_subtypes.empty?
|
196
|
+
UniqueType.new(new_name)
|
197
|
+
elsif hash_parameters?
|
119
198
|
UniqueType.new(new_name, "{#{new_key_types.join(', ')} => #{new_subtypes.join(', ')}}")
|
120
|
-
elsif
|
121
|
-
if
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
199
|
+
elsif @substring.start_with?('<(')
|
200
|
+
# @todo This clause is probably wrong, and if so, fixing it
|
201
|
+
# will be some level of breaking change. Probably best
|
202
|
+
# handled before real tuple support is rolled out and
|
203
|
+
# folks start relying on it more.
|
204
|
+
#
|
205
|
+
# (String) is a one element tuple in https://yardoc.org/types
|
206
|
+
# <String> is an array of zero or more Strings in https://yardoc.org/types
|
207
|
+
# Array<(String)> could be an Array of one-element tuples or a
|
208
|
+
# one element tuple. https://yardoc.org/types treats it
|
209
|
+
# as the former.
|
210
|
+
# Array<(String), Integer> is not ambiguous if we accept
|
211
|
+
# (String) as a tuple type, but not currently understood
|
212
|
+
# by Solargraph.
|
213
|
+
UniqueType.new(new_name, "<(#{new_subtypes.join(', ')})>")
|
214
|
+
elsif fixed_parameters?
|
215
|
+
UniqueType.new(new_name, "(#{new_subtypes.join(', ')})")
|
126
216
|
else
|
127
|
-
UniqueType.new(new_name)
|
217
|
+
UniqueType.new(new_name, "<#{new_subtypes.join(', ')}>")
|
218
|
+
end
|
219
|
+
end
|
220
|
+
|
221
|
+
# Apply the given transformation to each subtype and then finally to this type
|
222
|
+
#
|
223
|
+
# @param new_name [String, nil]
|
224
|
+
# @yieldparam t [UniqueType]
|
225
|
+
# @yieldreturn [self]
|
226
|
+
# @return [self]
|
227
|
+
def transform(new_name = nil, &transform_type)
|
228
|
+
new_key_types = @key_types.flat_map { |ct| ct.map { |ut| ut.transform(&transform_type) } }.compact
|
229
|
+
new_subtypes = @subtypes.flat_map { |ct| ct.map { |ut| ut.transform(&transform_type) } }.compact
|
230
|
+
new_type = recreate(new_name: new_name || rooted_name, new_key_types: new_key_types, new_subtypes: new_subtypes)
|
231
|
+
yield new_type
|
232
|
+
end
|
233
|
+
|
234
|
+
# Transform references to the 'self' type to the specified concrete namespace
|
235
|
+
# @param dst [String]
|
236
|
+
# @return [UniqueType]
|
237
|
+
def self_to dst
|
238
|
+
transform do |t|
|
239
|
+
next t if t.name != 'self'
|
240
|
+
t.recreate(new_name: dst, new_key_types: [], new_subtypes: [])
|
128
241
|
end
|
129
242
|
end
|
130
243
|
|
@@ -13,6 +13,8 @@ module Solargraph
|
|
13
13
|
|
14
14
|
# @param types [Array<[UniqueType, ComplexType]>]
|
15
15
|
def initialize types = [UniqueType::UNDEFINED]
|
16
|
+
# @todo @items here should not need an annotation
|
17
|
+
# @type [Array<UniqueType>]
|
16
18
|
@items = types.flat_map(&:items).uniq(&:to_s)
|
17
19
|
end
|
18
20
|
|
@@ -28,28 +30,44 @@ module Solargraph
|
|
28
30
|
ComplexType.new(types).reduce_object
|
29
31
|
end
|
30
32
|
|
33
|
+
# @param generics_to_resolve [Enumerable<String>]]
|
34
|
+
# @param context_type [UniqueType, nil]
|
35
|
+
# @param resolved_generic_values [Hash{String => ComplexType}] Added to as types are encountered or resolved
|
36
|
+
# @return [self]
|
37
|
+
def resolve_generics_from_context generics_to_resolve, context_type, resolved_generic_values: {}
|
38
|
+
return self unless generic?
|
39
|
+
|
40
|
+
ComplexType.new(@items.map { |i| i.resolve_generics_from_context(generics_to_resolve, context_type, resolved_generic_values: resolved_generic_values) })
|
41
|
+
end
|
42
|
+
|
43
|
+
# @return [UniqueType]
|
31
44
|
def first
|
32
45
|
@items.first
|
33
46
|
end
|
34
47
|
|
48
|
+
# @return [String]
|
35
49
|
def to_rbs
|
36
|
-
((@items.length > 1 ? '(' : '') +
|
37
|
-
|
38
|
-
|
50
|
+
((@items.length > 1 ? '(' : '') +
|
51
|
+
@items.map(&:to_rbs).join(' | ') +
|
52
|
+
(@items.length > 1 ? ')' : ''))
|
39
53
|
end
|
40
54
|
|
55
|
+
# @yieldparam [UniqueType]
|
56
|
+
# @return [Array]
|
41
57
|
def map &block
|
42
58
|
@items.map &block
|
43
59
|
end
|
44
60
|
|
45
61
|
# @yieldparam [UniqueType]
|
46
|
-
# @return [
|
62
|
+
# @return [Enumerable<UniqueType>]
|
47
63
|
def each &block
|
48
64
|
@items.each &block
|
49
65
|
end
|
50
66
|
|
51
67
|
# @yieldparam [UniqueType]
|
52
|
-
# @return [
|
68
|
+
# @return [void]
|
69
|
+
# @overload each_unique_type()
|
70
|
+
# @return [Enumerator<UniqueType>]
|
53
71
|
def each_unique_type &block
|
54
72
|
return enum_for(__method__) unless block_given?
|
55
73
|
|
@@ -58,14 +76,23 @@ module Solargraph
|
|
58
76
|
end
|
59
77
|
end
|
60
78
|
|
79
|
+
# @return [Integer]
|
61
80
|
def length
|
62
81
|
@items.length
|
63
82
|
end
|
64
83
|
|
84
|
+
# @return [Array<UniqueType>]
|
85
|
+
def to_a
|
86
|
+
@items
|
87
|
+
end
|
88
|
+
|
89
|
+
# @param index [Integer]
|
90
|
+
# @return [UniqueType]
|
65
91
|
def [](index)
|
66
92
|
@items[index]
|
67
93
|
end
|
68
94
|
|
95
|
+
# @return [Array<UniqueType>]
|
69
96
|
def select &block
|
70
97
|
@items.select &block
|
71
98
|
end
|
@@ -81,12 +108,16 @@ module Solargraph
|
|
81
108
|
@items.map(&:namespace)
|
82
109
|
end
|
83
110
|
|
111
|
+
# @param name [Symbol]
|
112
|
+
# @return [Object, nil]
|
84
113
|
def method_missing name, *args, &block
|
85
114
|
return if @items.first.nil?
|
86
115
|
return @items.first.send(name, *args, &block) if respond_to_missing?(name)
|
87
116
|
super
|
88
117
|
end
|
89
118
|
|
119
|
+
# @param name [Symbol]
|
120
|
+
# @param include_private [Boolean]
|
90
121
|
def respond_to_missing?(name, include_private = false)
|
91
122
|
TypeMethods.public_instance_methods.include?(name) || super
|
92
123
|
end
|
@@ -111,7 +142,15 @@ module Solargraph
|
|
111
142
|
any?(&:generic?)
|
112
143
|
end
|
113
144
|
|
114
|
-
# @param
|
145
|
+
# @param new_name [String, nil]
|
146
|
+
# @yieldparam t [UniqueType]
|
147
|
+
# @yieldreturn [UniqueType]
|
148
|
+
# @return [ComplexType]
|
149
|
+
def transform(new_name = nil, &transform_type)
|
150
|
+
ComplexType.new(map { |ut| ut.transform(new_name, &transform_type) })
|
151
|
+
end
|
152
|
+
|
153
|
+
# @param definitions [Pin::Namespace, Pin::Method]
|
115
154
|
# @param context_type [ComplexType]
|
116
155
|
# @return [ComplexType]
|
117
156
|
def resolve_generics definitions, context_type
|
@@ -132,6 +171,7 @@ module Solargraph
|
|
132
171
|
@items.any?(&:nil_type?)
|
133
172
|
end
|
134
173
|
|
174
|
+
# @return [Array<ComplexType>]
|
135
175
|
def all_params
|
136
176
|
@items.first.all_params || []
|
137
177
|
end
|
@@ -140,11 +180,16 @@ module Solargraph
|
|
140
180
|
|
141
181
|
protected
|
142
182
|
|
183
|
+
# @return [ComplexType]
|
143
184
|
def reduce_object
|
144
185
|
return self if name != 'Object' || subtypes.empty?
|
145
186
|
ComplexType.try_parse(reduce_class(subtypes.join(', ')))
|
146
187
|
end
|
147
188
|
|
189
|
+
def bottom?
|
190
|
+
@items.all?(&:bot?)
|
191
|
+
end
|
192
|
+
|
148
193
|
private
|
149
194
|
|
150
195
|
# @todo This is a quick and dirty hack that forces `self` keywords
|
@@ -174,9 +219,13 @@ module Solargraph
|
|
174
219
|
# used internally.
|
175
220
|
#
|
176
221
|
# @param *strings [Array<String>] The type definitions to parse
|
177
|
-
# @
|
178
|
-
# @
|
222
|
+
# @return [ComplexType]
|
223
|
+
# @overload parse(*strings, partial: false)
|
224
|
+
# @todo Need ability to use a literal true as a type below
|
225
|
+
# @param partial [Boolean] True if the string is part of a another type
|
226
|
+
# @return [Array<UniqueType>]
|
179
227
|
def parse *strings, partial: false
|
228
|
+
# @type [Hash{Array<String> => ComplexType}]
|
180
229
|
@cache ||= {}
|
181
230
|
unless partial
|
182
231
|
cached = @cache[strings]
|
@@ -202,6 +251,9 @@ module Solargraph
|
|
202
251
|
raise ComplexTypeError, "Invalid hash thing" unless key_types.nil?
|
203
252
|
# types.push ComplexType.new([UniqueType.new(base[0..-2].strip)])
|
204
253
|
types.push UniqueType.new(base[0..-2].strip)
|
254
|
+
# @todo this should either expand key_type's type
|
255
|
+
# automatically or complain about not being
|
256
|
+
# compatible with key_type's type in type checking
|
205
257
|
key_types = types
|
206
258
|
types = []
|
207
259
|
base.clear
|
@@ -224,7 +276,7 @@ module Solargraph
|
|
224
276
|
paren_stack += 1
|
225
277
|
elsif char == ')'
|
226
278
|
paren_stack -= 1
|
227
|
-
subtype_string += char
|
279
|
+
subtype_string += char
|
228
280
|
raise ComplexTypeError, "Invalid close in type #{type_string}" if paren_stack < 0
|
229
281
|
next
|
230
282
|
elsif char == ',' && point_stack == 0 && curly_stack == 0 && paren_stack == 0
|
@@ -271,5 +323,6 @@ module Solargraph
|
|
271
323
|
NIL = ComplexType.parse('nil')
|
272
324
|
SELF = ComplexType.parse('self')
|
273
325
|
BOOLEAN = ComplexType.parse('Boolean')
|
326
|
+
BOT = ComplexType.parse('bot')
|
274
327
|
end
|
275
328
|
end
|
@@ -1,6 +1,11 @@
|
|
1
|
+
require 'nokogiri'
|
2
|
+
|
1
3
|
module ReverseMarkdown
|
2
4
|
module Converters
|
3
5
|
class Dd < Base
|
6
|
+
# @return [String]
|
7
|
+
# @param node [Nokogiri::XML::Element]
|
8
|
+
# @param state [Hash]
|
4
9
|
def convert node, state = {}
|
5
10
|
content = treat_children(node, state)
|
6
11
|
": #{content.strip}\n"
|
@@ -29,9 +29,9 @@ module Solargraph
|
|
29
29
|
store = RuboCop::ConfigStore.new
|
30
30
|
runner = RuboCop::Runner.new(options, store)
|
31
31
|
result = redirect_stdout{ runner.run(paths) }
|
32
|
-
|
32
|
+
|
33
33
|
return [] if result.empty?
|
34
|
-
|
34
|
+
|
35
35
|
make_array JSON.parse(result)
|
36
36
|
rescue RuboCop::ValidationError, RuboCop::ConfigNotFoundError => e
|
37
37
|
raise DiagnosticsError, "Error in RuboCop configuration: #{e.message}"
|
@@ -48,7 +48,7 @@ module Solargraph
|
|
48
48
|
args.find { |a| a =~ /version=/ }.to_s.split('=').last
|
49
49
|
end
|
50
50
|
|
51
|
-
# @param resp [Hash]
|
51
|
+
# @param resp [Hash{String => Array<Hash{String => Array<Hash{String => undefined}>}>}]
|
52
52
|
# @return [Array<Hash>]
|
53
53
|
def make_array resp
|
54
54
|
diagnostics = []
|
@@ -62,8 +62,8 @@ module Solargraph
|
|
62
62
|
|
63
63
|
# Convert a RuboCop offense to an LSP diagnostic
|
64
64
|
#
|
65
|
-
# @param off [Hash] Offense received from Rubocop
|
66
|
-
# @return [Hash] LSP diagnostic
|
65
|
+
# @param off [Hash{String => unknown}] Offense received from Rubocop
|
66
|
+
# @return [Hash{Symbol => Hash, String, Integer}] LSP diagnostic
|
67
67
|
def offense_to_diagnostic off
|
68
68
|
{
|
69
69
|
range: offense_range(off).to_hash,
|
@@ -81,19 +81,20 @@ module Solargraph
|
|
81
81
|
Range.new(offense_start_position(off), offense_ending_position(off))
|
82
82
|
end
|
83
83
|
|
84
|
-
# @param off [Hash]
|
84
|
+
# @param off [Hash{String => Hash{String => Integer}}]
|
85
85
|
# @return [Position]
|
86
86
|
def offense_start_position off
|
87
87
|
Position.new(off['location']['start_line'] - 1, off['location']['start_column'] - 1)
|
88
88
|
end
|
89
89
|
|
90
|
-
# @param off [Hash]
|
90
|
+
# @param off [Hash{String => Hash{String => Integer}}]
|
91
91
|
# @return [Position]
|
92
92
|
def offense_ending_position off
|
93
93
|
if off['location']['start_line'] != off['location']['last_line']
|
94
94
|
Position.new(off['location']['start_line'], 0)
|
95
95
|
else
|
96
96
|
start_line = off['location']['start_line'] - 1
|
97
|
+
# @type [Integer]
|
97
98
|
last_column = off['location']['last_column']
|
98
99
|
line = @source.code.lines[start_line]
|
99
100
|
col_off = if line.nil? || line.empty?
|
@@ -12,6 +12,7 @@ module Solargraph
|
|
12
12
|
#
|
13
13
|
# @param version [String, nil]
|
14
14
|
# @raise [InvalidRubocopVersionError] if _version_ is not installed
|
15
|
+
# @return [void]
|
15
16
|
def require_rubocop(version = nil)
|
16
17
|
begin
|
17
18
|
gem_path = Gem::Specification.find_by_name('rubocop', version).full_gem_path
|