solargraph 0.51.2 → 0.53.0

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 (162) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/plugins.yml +40 -0
  3. data/.github/workflows/rspec.yml +1 -3
  4. data/.github/workflows/typecheck.yml +34 -0
  5. data/.yardopts +2 -2
  6. data/CHANGELOG.md +55 -5
  7. data/README.md +13 -16
  8. data/SPONSORS.md +1 -7
  9. data/lib/solargraph/api_map/cache.rb +60 -20
  10. data/lib/solargraph/api_map/store.rb +47 -11
  11. data/lib/solargraph/api_map.rb +161 -95
  12. data/lib/solargraph/bench.rb +2 -2
  13. data/lib/solargraph/cache.rb +29 -5
  14. data/lib/solargraph/complex_type/type_methods.rb +54 -9
  15. data/lib/solargraph/complex_type/unique_type.rb +155 -58
  16. data/lib/solargraph/complex_type.rb +73 -16
  17. data/lib/solargraph/convention.rb +0 -1
  18. data/lib/solargraph/converters/dd.rb +5 -0
  19. data/lib/solargraph/converters/dl.rb +3 -0
  20. data/lib/solargraph/converters/dt.rb +3 -0
  21. data/lib/solargraph/diagnostics/rubocop.rb +8 -7
  22. data/lib/solargraph/diagnostics/rubocop_helpers.rb +1 -0
  23. data/lib/solargraph/diagnostics/type_check.rb +1 -0
  24. data/lib/solargraph/diagnostics.rb +2 -2
  25. data/lib/solargraph/doc_map.rb +146 -0
  26. data/lib/solargraph/gem_pins.rb +64 -0
  27. data/lib/solargraph/language_server/host/cataloger.rb +1 -0
  28. data/lib/solargraph/language_server/host/diagnoser.rb +2 -2
  29. data/lib/solargraph/language_server/host/dispatch.rb +10 -4
  30. data/lib/solargraph/language_server/host/message_worker.rb +4 -0
  31. data/lib/solargraph/language_server/host/sources.rb +7 -4
  32. data/lib/solargraph/language_server/host.rb +15 -6
  33. data/lib/solargraph/language_server/message/completion_item/resolve.rb +3 -1
  34. data/lib/solargraph/language_server/message/extended/check_gem_version.rb +13 -1
  35. data/lib/solargraph/language_server/message/initialize.rb +5 -2
  36. data/lib/solargraph/language_server/message/text_document/hover.rb +2 -0
  37. data/lib/solargraph/language_server/message/text_document.rb +0 -1
  38. data/lib/solargraph/language_server/message/workspace/did_change_configuration.rb +5 -0
  39. data/lib/solargraph/language_server/transport/adapter.rb +16 -1
  40. data/lib/solargraph/language_server/transport/data_reader.rb +2 -0
  41. data/lib/solargraph/library.rb +70 -16
  42. data/lib/solargraph/location.rb +1 -0
  43. data/lib/solargraph/parser/comment_ripper.rb +4 -0
  44. data/lib/solargraph/parser/node_methods.rb +47 -7
  45. data/lib/solargraph/parser/node_processor/base.rb +9 -0
  46. data/lib/solargraph/parser/{legacy → parser_gem}/class_methods.rb +31 -5
  47. data/lib/solargraph/parser/{legacy → parser_gem}/flawed_builder.rb +3 -1
  48. data/lib/solargraph/parser/{legacy → parser_gem}/node_chainer.rb +49 -36
  49. data/lib/solargraph/parser/parser_gem/node_methods.rb +499 -0
  50. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/alias_node.rb +1 -1
  51. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/args_node.rb +4 -1
  52. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/begin_node.rb +1 -1
  53. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/block_node.rb +3 -2
  54. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/casgn_node.rb +2 -2
  55. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/cvasgn_node.rb +1 -1
  56. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/def_node.rb +1 -1
  57. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/defs_node.rb +2 -2
  58. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/gvasgn_node.rb +1 -1
  59. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/ivasgn_node.rb +2 -2
  60. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/lvasgn_node.rb +2 -2
  61. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/namespace_node.rb +2 -2
  62. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/orasgn_node.rb +1 -1
  63. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/resbody_node.rb +3 -3
  64. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/sclass_node.rb +1 -1
  65. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/send_node.rb +2 -2
  66. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/sym_node.rb +1 -1
  67. data/lib/solargraph/parser/parser_gem/node_processors.rb +54 -0
  68. data/lib/solargraph/parser/parser_gem.rb +12 -0
  69. data/lib/solargraph/parser/region.rb +1 -1
  70. data/lib/solargraph/parser/snippet.rb +2 -0
  71. data/lib/solargraph/parser.rb +8 -9
  72. data/lib/solargraph/pin/base.rb +64 -9
  73. data/lib/solargraph/pin/base_variable.rb +6 -2
  74. data/lib/solargraph/pin/block.rb +13 -8
  75. data/lib/solargraph/pin/closure.rb +17 -2
  76. data/lib/solargraph/pin/common.rb +7 -3
  77. data/lib/solargraph/pin/conversions.rb +33 -3
  78. data/lib/solargraph/pin/delegated_method.rb +1 -1
  79. data/lib/solargraph/pin/documenting.rb +25 -34
  80. data/lib/solargraph/pin/instance_variable.rb +4 -0
  81. data/lib/solargraph/pin/local_variable.rb +13 -1
  82. data/lib/solargraph/pin/method.rb +169 -18
  83. data/lib/solargraph/pin/namespace.rb +18 -5
  84. data/lib/solargraph/pin/parameter.rb +44 -14
  85. data/lib/solargraph/pin/reference/override.rb +2 -2
  86. data/lib/solargraph/pin/reference.rb +8 -0
  87. data/lib/solargraph/pin/search.rb +3 -3
  88. data/lib/solargraph/pin/signature.rb +123 -3
  89. data/lib/solargraph/pin.rb +0 -1
  90. data/lib/solargraph/range.rb +2 -2
  91. data/lib/solargraph/rbs_map/conversions.rb +287 -45
  92. data/lib/solargraph/rbs_map/core_fills.rb +6 -29
  93. data/lib/solargraph/rbs_map/core_map.rb +2 -1
  94. data/lib/solargraph/rbs_map/core_signs.rb +2 -0
  95. data/lib/solargraph/rbs_map/stdlib_map.rb +2 -8
  96. data/lib/solargraph/rbs_map.rb +20 -11
  97. data/lib/solargraph/shell.rb +62 -59
  98. data/lib/solargraph/source/chain/array.rb +32 -0
  99. data/lib/solargraph/source/chain/block_symbol.rb +13 -0
  100. data/lib/solargraph/source/chain/call.rb +99 -46
  101. data/lib/solargraph/source/chain/constant.rb +15 -1
  102. data/lib/solargraph/source/chain/if.rb +23 -0
  103. data/lib/solargraph/source/chain/link.rb +8 -2
  104. data/lib/solargraph/source/chain/or.rb +1 -1
  105. data/lib/solargraph/source/chain/z_super.rb +3 -3
  106. data/lib/solargraph/source/chain.rb +29 -14
  107. data/lib/solargraph/source/change.rb +3 -0
  108. data/lib/solargraph/source/cursor.rb +2 -0
  109. data/lib/solargraph/source/source_chainer.rb +8 -5
  110. data/lib/solargraph/source.rb +18 -19
  111. data/lib/solargraph/source_map/clip.rb +11 -23
  112. data/lib/solargraph/source_map/mapper.rb +12 -1
  113. data/lib/solargraph/source_map.rb +15 -5
  114. data/lib/solargraph/type_checker/checks.rb +10 -2
  115. data/lib/solargraph/type_checker.rb +92 -26
  116. data/lib/solargraph/version.rb +1 -1
  117. data/lib/solargraph/workspace/config.rb +8 -6
  118. data/lib/solargraph/workspace.rb +3 -2
  119. data/lib/solargraph/yard_map/cache.rb +6 -0
  120. data/lib/solargraph/yard_map/helpers.rb +1 -1
  121. data/lib/solargraph/yard_map/mapper/to_method.rb +11 -1
  122. data/lib/solargraph/yard_map/mapper.rb +1 -1
  123. data/lib/solargraph/yard_map/to_method.rb +11 -4
  124. data/lib/solargraph/yard_map.rb +1 -292
  125. data/lib/solargraph/yard_tags.rb +20 -0
  126. data/lib/solargraph/yardoc.rb +52 -0
  127. data/lib/solargraph.rb +6 -4
  128. data/solargraph.gemspec +3 -2
  129. metadata +51 -58
  130. data/lib/solargraph/api_map/bundler_methods.rb +0 -22
  131. data/lib/solargraph/documentor.rb +0 -76
  132. data/lib/solargraph/parser/legacy/node_methods.rb +0 -325
  133. data/lib/solargraph/parser/legacy/node_processors/alias_node.rb +0 -23
  134. data/lib/solargraph/parser/legacy/node_processors/begin_node.rb +0 -15
  135. data/lib/solargraph/parser/legacy/node_processors/sym_node.rb +0 -18
  136. data/lib/solargraph/parser/legacy/node_processors.rb +0 -55
  137. data/lib/solargraph/parser/legacy.rb +0 -12
  138. data/lib/solargraph/parser/rubyvm/class_methods.rb +0 -153
  139. data/lib/solargraph/parser/rubyvm/node_chainer.rb +0 -160
  140. data/lib/solargraph/parser/rubyvm/node_methods.rb +0 -317
  141. data/lib/solargraph/parser/rubyvm/node_processors/args_node.rb +0 -85
  142. data/lib/solargraph/parser/rubyvm/node_processors/block_node.rb +0 -42
  143. data/lib/solargraph/parser/rubyvm/node_processors/casgn_node.rb +0 -33
  144. data/lib/solargraph/parser/rubyvm/node_processors/cvasgn_node.rb +0 -23
  145. data/lib/solargraph/parser/rubyvm/node_processors/def_node.rb +0 -75
  146. data/lib/solargraph/parser/rubyvm/node_processors/defs_node.rb +0 -68
  147. data/lib/solargraph/parser/rubyvm/node_processors/gvasgn_node.rb +0 -23
  148. data/lib/solargraph/parser/rubyvm/node_processors/ivasgn_node.rb +0 -38
  149. data/lib/solargraph/parser/rubyvm/node_processors/kw_arg_node.rb +0 -39
  150. data/lib/solargraph/parser/rubyvm/node_processors/lit_node.rb +0 -20
  151. data/lib/solargraph/parser/rubyvm/node_processors/lvasgn_node.rb +0 -27
  152. data/lib/solargraph/parser/rubyvm/node_processors/namespace_node.rb +0 -39
  153. data/lib/solargraph/parser/rubyvm/node_processors/opt_arg_node.rb +0 -26
  154. data/lib/solargraph/parser/rubyvm/node_processors/orasgn_node.rb +0 -15
  155. data/lib/solargraph/parser/rubyvm/node_processors/resbody_node.rb +0 -51
  156. data/lib/solargraph/parser/rubyvm/node_processors/sclass_node.rb +0 -32
  157. data/lib/solargraph/parser/rubyvm/node_processors/scope_node.rb +0 -15
  158. data/lib/solargraph/parser/rubyvm/node_processors/send_node.rb +0 -279
  159. data/lib/solargraph/parser/rubyvm/node_processors.rb +0 -64
  160. data/lib/solargraph/parser/rubyvm/node_wrapper.rb +0 -47
  161. data/lib/solargraph/parser/rubyvm.rb +0 -40
  162. data/lib/yard-solargraph.rb +0 -33
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Solargraph
4
+ class Source
5
+ class Chain
6
+ class If < Link
7
+ def word
8
+ '<if>'
9
+ end
10
+
11
+ # @param links [::Array<Link>]
12
+ def initialize links
13
+ @links = links
14
+ end
15
+
16
+ def resolve api_map, name_pin, locals
17
+ types = @links.map { |link| link.infer(api_map, name_pin, locals) }
18
+ [Solargraph::Pin::ProxyType.anonymous(Solargraph::ComplexType.try_parse(types.map(&:tag).uniq.join(', ')))]
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
@@ -7,8 +7,10 @@ module Solargraph
7
7
  # @return [String]
8
8
  attr_reader :word
9
9
 
10
+ # @return [Pin::Base]
10
11
  attr_accessor :last_context
11
12
 
13
+ # @param word [String]
12
14
  def initialize word = '<undefined>'
13
15
  @word = word
14
16
  end
@@ -23,12 +25,16 @@ module Solargraph
23
25
 
24
26
  # @param api_map [ApiMap]
25
27
  # @param name_pin [Pin::Base]
26
- # @param locals [Array<Pin::Base>]
27
- # @return [Array<Pin::Base>]
28
+ # @param locals [::Enumerable<Pin::Base>]
29
+ # @return [::Array<Pin::Base>]
28
30
  def resolve api_map, name_pin, locals
29
31
  []
30
32
  end
31
33
 
34
+ def inspect
35
+ "#{self.class} #{word}"
36
+ end
37
+
32
38
  def head?
33
39
  @head ||= false
34
40
  end
@@ -8,7 +8,7 @@ module Solargraph
8
8
  '<or>'
9
9
  end
10
10
 
11
- # @param type [String]
11
+ # @param links [::Array<Link>]
12
12
  def initialize links
13
13
  @links = links
14
14
  end
@@ -7,11 +7,11 @@ module Solargraph
7
7
  # @return [String]
8
8
  attr_reader :word
9
9
 
10
- # @return [Array<Chain>]
10
+ # @return [::Array<Chain>]
11
11
  attr_reader :arguments
12
12
 
13
13
  # @param word [String]
14
- # @param arguments [Array<Chain>]
14
+ # @param arguments [::Array<Chain>]
15
15
  # @param with_block [Boolean] True if the chain is inside a block
16
16
  # @param head [Boolean] True if the call is the start of its chain
17
17
  def initialize word, with_block = false
@@ -20,7 +20,7 @@ module Solargraph
20
20
 
21
21
  # @param api_map [ApiMap]
22
22
  # @param name_pin [Pin::Base]
23
- # @param locals [Array<Pin::Base>]
23
+ # @param locals [::Array<Pin::Base>]
24
24
  def resolve api_map, name_pin, locals
25
25
  return super_pins(api_map, name_pin)
26
26
  end
@@ -19,10 +19,13 @@ module Solargraph
19
19
  autoload :GlobalVariable, 'solargraph/source/chain/global_variable'
20
20
  autoload :Literal, 'solargraph/source/chain/literal'
21
21
  autoload :Head, 'solargraph/source/chain/head'
22
+ autoload :If, 'solargraph/source/chain/if'
22
23
  autoload :Or, 'solargraph/source/chain/or'
23
24
  autoload :BlockVariable, 'solargraph/source/chain/block_variable'
25
+ autoload :BlockSymbol, 'solargraph/source/chain/block_symbol'
24
26
  autoload :ZSuper, 'solargraph/source/chain/z_super'
25
27
  autoload :Hash, 'solargraph/source/chain/hash'
28
+ autoload :Array, 'solargraph/source/chain/array'
26
29
 
27
30
  @@inference_stack = []
28
31
  @@inference_depth = 0
@@ -30,12 +33,14 @@ module Solargraph
30
33
  UNDEFINED_CALL = Chain::Call.new('<undefined>')
31
34
  UNDEFINED_CONSTANT = Chain::Constant.new('<undefined>')
32
35
 
33
- # @return [Array<Source::Chain::Link>]
36
+ # @return [::Array<Source::Chain::Link>]
34
37
  attr_reader :links
35
38
 
36
39
  attr_reader :node
37
40
 
38
- # @param links [Array<Chain::Link>]
41
+ # @param node [Parser::AST::Node, nil]
42
+ # @param links [::Array<Chain::Link>]
43
+ # @param splat [Boolean]
39
44
  def initialize links, node = nil, splat = false
40
45
  @links = links.clone
41
46
  @links.push UNDEFINED_CALL if @links.empty?
@@ -56,8 +61,9 @@ module Solargraph
56
61
 
57
62
  # @param api_map [ApiMap]
58
63
  # @param name_pin [Pin::Base]
59
- # @param locals [Array<Pin::Base>]
60
- # @return [Array<Pin::Base>]
64
+ # @param locals [::Enumerable<Pin::LocalVariable>]
65
+ #
66
+ # @return [::Array<Pin::Base>]
61
67
  def define api_map, name_pin, locals
62
68
  return [] if undefined?
63
69
  working_pin = name_pin
@@ -73,7 +79,7 @@ module Solargraph
73
79
 
74
80
  # @param api_map [ApiMap]
75
81
  # @param name_pin [Pin::Base]
76
- # @param locals [Array<Pin::Base>]
82
+ # @param locals [::Enumerable<Pin::LocalVariable>]
77
83
  # @return [ComplexType]
78
84
  def infer api_map, name_pin, locals
79
85
  from_here = base.infer(api_map, name_pin, locals) unless links.length == 1
@@ -113,26 +119,29 @@ module Solargraph
113
119
 
114
120
  private
115
121
 
116
- # @param pins [Array<Pin::Base>]
122
+ # @param pins [::Array<Pin::Base>]
117
123
  # @param context [Pin::Base]
118
124
  # @param api_map [ApiMap]
125
+ # @param locals [::Enumerable<Pin::LocalVariable>]
119
126
  # @return [ComplexType]
120
127
  def infer_first_defined pins, context, api_map, locals
121
128
  possibles = []
129
+ # @todo this param tag shouldn't be needed to probe the type
130
+ # @todo ...but given it is needed, typecheck should complain that it is needed
131
+ # @param pin [Pin::Base]
122
132
  pins.each do |pin|
123
133
  # Avoid infinite recursion
124
134
  next if @@inference_stack.include?(pin.identity)
135
+
125
136
  @@inference_stack.push pin.identity
126
137
  type = pin.typify(api_map)
127
138
  @@inference_stack.pop
128
139
  if type.defined?
129
- if type.parameterized?
130
- type = type.resolve_parameters(pin.closure, context)
131
- # idx = pin.closure.parameters.index(type.subtypes.first.name)
132
- # next if idx.nil?
133
- # param_type = context.return_type.all_params[idx]
134
- # next unless param_type
135
- # type = ComplexType.try_parse(param_type.to_s)
140
+ if type.generic?
141
+ # @todo even at strong, no typechecking complaint
142
+ # happens when a [Pin::Base,nil] is passed into a method
143
+ # that accepts only [Pin::Namespace] as an argument
144
+ type = type.resolve_generics(pin.closure, context.return_type)
136
145
  end
137
146
  if type.defined?
138
147
  possibles.push type
@@ -143,10 +152,13 @@ module Solargraph
143
152
  if possibles.empty?
144
153
  # Limit method inference recursion
145
154
  return ComplexType::UNDEFINED if @@inference_depth >= 10 && pins.first.is_a?(Pin::Method)
155
+
146
156
  @@inference_depth += 1
157
+ # @param pin [Pin::Base]
147
158
  pins.each do |pin|
148
159
  # Avoid infinite recursion
149
160
  next if @@inference_stack.include?(pin.identity)
161
+
150
162
  @@inference_stack.push pin.identity
151
163
  type = pin.probe(api_map)
152
164
  @@inference_stack.pop
@@ -158,6 +170,7 @@ module Solargraph
158
170
  @@inference_depth -= 1
159
171
  end
160
172
  return ComplexType::UNDEFINED if possibles.empty?
173
+
161
174
  type = if possibles.length > 1
162
175
  sorted = possibles.map { |t| t.rooted? ? "::#{t}" : t.to_s }.sort { |a, _| a == 'nil' ? 1 : 0 }
163
176
  ComplexType.parse(*sorted)
@@ -165,10 +178,12 @@ module Solargraph
165
178
  ComplexType.parse(possibles.map(&:to_s).join(', '))
166
179
  end
167
180
  return type if context.nil? || context.return_type.undefined?
168
- type.self_to(context.return_type.namespace)
181
+
182
+ type.self_to(context.return_type.tag)
169
183
  end
170
184
 
171
185
  # @param type [ComplexType]
186
+ # @return [ComplexType]
172
187
  def maybe_nil type
173
188
  return type if type.undefined? || type.void? || type.nullable?
174
189
  return type unless nullable?
@@ -69,6 +69,9 @@ module Solargraph
69
69
 
70
70
  private
71
71
 
72
+ # @param text [String]
73
+ # @param insert [String]
74
+ # @return [String]
72
75
  def commit text, insert
73
76
  start_offset = Position.to_offset(text, range.start)
74
77
  end_offset = Position.to_offset(text, range.ending)
@@ -115,6 +115,7 @@ module Solargraph
115
115
  end
116
116
  alias receiver recipient
117
117
 
118
+ # @return [AST::Node]
118
119
  def node
119
120
  @node ||= source.node_at(position.line, position.column)
120
121
  end
@@ -135,6 +136,7 @@ module Solargraph
135
136
  end
136
137
  end
137
138
 
139
+ # @return [Parser::AST::Node, nil]
138
140
  def recipient_node
139
141
  @recipient_node ||= Solargraph::Parser::NodeMethods.find_recipient_node(self)
140
142
  end
@@ -14,10 +14,10 @@ module Solargraph
14
14
 
15
15
  class << self
16
16
  # @param source [Source]
17
- # @param position [Position]
17
+ # @param position [Position, Array(Integer, Integer)]
18
18
  # @return [Source::Chain]
19
19
  def chain source, position
20
- new(source, position).chain
20
+ new(source, Solargraph::Position.normalize(position)).chain
21
21
  end
22
22
  end
23
23
 
@@ -41,12 +41,12 @@ module Solargraph
41
41
  parent = nil
42
42
  if !source.repaired? && source.parsed? && source.synchronized?
43
43
  tree = source.tree_at(position.line, position.column)
44
- # node, parent = source.tree_at(position.line, position.column)[0..2]
45
- tree.shift while tree.length > 1 && tree.first.type == :SCOPE
46
44
  node, parent = tree[0..2]
47
45
  elsif source.parsed? && source.repaired? && end_of_phrase == '.'
48
46
  node, parent = source.tree_at(fixed_position.line, fixed_position.column)[0..2]
49
47
  node = Parser.parse(fixed_phrase) if node.nil?
48
+ elsif source.repaired?
49
+ node = Parser.parse(fixed_phrase)
50
50
  else
51
51
  node, parent = source.tree_at(fixed_position.line, fixed_position.column)[0..2] unless source.error_ranges.any?{|r| r.nil? || r.include?(fixed_position)}
52
52
  # Exception for positions that chain literal nodes in unsynchronized sources
@@ -58,7 +58,7 @@ module Solargraph
58
58
  end
59
59
  return Chain.new([Chain::UNDEFINED_CALL]) if node.nil? || (node.type == :sym && !phrase.start_with?(':'))
60
60
  # chain = NodeChainer.chain(node, source.filename, parent && parent.type == :block)
61
- chain = Parser.chain(node, source.filename, parent && [:ITER, :block].include?(parent.type))
61
+ chain = Parser.chain(node, source.filename, parent)
62
62
  if source.repaired? || !source.parsed? || !source.synchronized?
63
63
  if end_of_phrase.strip == '.'
64
64
  chain.links.push Chain::UNDEFINED_CALL
@@ -126,10 +126,13 @@ module Solargraph
126
126
  Position.line_char_to_offset(@source.code, line, column)
127
127
  end
128
128
 
129
+ # @return [Integer]
129
130
  def signature_data
130
131
  @signature_data ||= get_signature_data_at(offset)
131
132
  end
132
133
 
134
+ # @param index [Integer]
135
+ # @return [Integer]
133
136
  def get_signature_data_at index
134
137
  brackets = 0
135
138
  squares = 0
@@ -8,7 +8,6 @@ module Solargraph
8
8
  class Source
9
9
  autoload :Updater, 'solargraph/source/updater'
10
10
  autoload :Change, 'solargraph/source/change'
11
- autoload :Mapper, 'solargraph/source/mapper'
12
11
  autoload :EncodingFixes, 'solargraph/source/encoding_fixes'
13
12
  autoload :Cursor, 'solargraph/source/cursor'
14
13
  autoload :Chain, 'solargraph/source/chain'
@@ -33,7 +32,7 @@ module Solargraph
33
32
  attr_reader :version
34
33
 
35
34
  # @param code [String]
36
- # @param filename [String]
35
+ # @param filename [String, nil]
37
36
  # @param version [Integer]
38
37
  def initialize code, filename = nil, version = 0
39
38
  @code = normalize(code)
@@ -213,6 +212,7 @@ module Solargraph
213
212
  end
214
213
  end
215
214
 
215
+ # @return [::Array<Range>]
216
216
  def string_ranges
217
217
  @string_ranges ||= Parser.string_ranges(node)
218
218
  end
@@ -250,7 +250,7 @@ module Solargraph
250
250
  end
251
251
 
252
252
  # @param node [Parser::AST::Node]
253
- # @return [String]
253
+ # @return [String, nil]
254
254
  def comments_for node
255
255
  rng = Range.from_node(node)
256
256
  stringified_comments[rng.start.line] ||= begin
@@ -301,7 +301,7 @@ module Solargraph
301
301
 
302
302
  # Get a hash of comments grouped by the line numbers of the associated code.
303
303
  #
304
- # @return [Hash{Integer => Array<Parser::Source::Comment>}]
304
+ # @return [Hash{Integer => String}]
305
305
  def associated_comments
306
306
  @associated_comments ||= begin
307
307
  result = {}
@@ -323,6 +323,8 @@ module Solargraph
323
323
 
324
324
  private
325
325
 
326
+ # @param line [Integer]
327
+ # @return [Integer]
326
328
  def first_not_empty_from line
327
329
  cursor = line
328
330
  cursor += 1 while cursor < code_lines.length && code_lines[cursor].strip.empty?
@@ -332,7 +334,7 @@ module Solargraph
332
334
 
333
335
  # @param top [Parser::AST::Node]
334
336
  # @param result [Array<Range>]
335
- # @param parent [Symbol]
337
+ # @param parent [Symbol, nil]
336
338
  # @return [void]
337
339
  def inner_folding_ranges top, result = [], parent = nil
338
340
  return unless Parser.is_ast_node?(top)
@@ -411,7 +413,7 @@ module Solargraph
411
413
  result
412
414
  end
413
415
 
414
- # @param n [Parser::AST::Node]
416
+ # @param n [Parser::AST::Node, nil]
415
417
  # @return [Array<Parser::AST::Node>]
416
418
  def string_nodes_in n
417
419
  result = []
@@ -425,30 +427,23 @@ module Solargraph
425
427
  result
426
428
  end
427
429
 
428
- # @param node [Parser::AST::Node]
430
+ # @param node [Parser::AST::Node, nil]
429
431
  # @param position [Position]
430
432
  # @param stack [Array<Parser::AST::Node>]
431
433
  # @return [void]
432
434
  def inner_tree_at node, position, stack
433
435
  return if node.nil?
434
436
  here = Range.from_node(node)
435
- if here.contain?(position) || colonized(here, position, node)
437
+ if here.contain?(position)
436
438
  stack.unshift node
437
439
  node.children.each do |c|
438
440
  next unless Parser.is_ast_node?(c)
439
- next if !Parser.rubyvm? && c.loc.expression.nil?
441
+ next if c.loc.expression.nil?
440
442
  inner_tree_at(c, position, stack)
441
443
  end
442
444
  end
443
445
  end
444
446
 
445
- def colonized range, position, node
446
- node.type == :COLON2 &&
447
- range.ending.line == position.line &&
448
- range.ending.character == position.character - 2 &&
449
- code[Position.to_offset(code, Position.new(position.line, position.character - 2)), 2] == '::'
450
- end
451
-
452
447
  protected
453
448
 
454
449
  # @return [String]
@@ -464,7 +459,7 @@ module Solargraph
464
459
  @code = val
465
460
  end
466
461
 
467
- # @return [Parser::AST::Node]
462
+ # @return [Parser::AST::Node, nil]
468
463
  attr_writer :node
469
464
 
470
465
  # @return [Array<Range>]
@@ -476,7 +471,7 @@ module Solargraph
476
471
  # @return [Boolean]
477
472
  attr_writer :parsed
478
473
 
479
- # @return [Array<Parser::Source::Comment>]
474
+ # @return [Hash{Integer => String}
480
475
  attr_writer :comments
481
476
 
482
477
  # @return [Boolean]
@@ -503,7 +498,7 @@ module Solargraph
503
498
  end
504
499
 
505
500
  # @param code [String]
506
- # @param filename [String]
501
+ # @param filename [String, nil]
507
502
  # @param version [Integer]
508
503
  # @return [Solargraph::Source]
509
504
  def load_string code, filename = nil, version = 0
@@ -516,6 +511,10 @@ module Solargraph
516
511
  # HACK: Pass a dummy code object to the parser for plugins that
517
512
  # expect it not to be nil
518
513
  YARD::Docstring.parser.parse(comments, YARD::CodeObjects::Base.new(:root, 'stub'))
514
+ rescue StandardError => e
515
+ Solargraph.logger.info "YARD failed to parse docstring: [#{e.class}] #{e.message}"
516
+ Solargraph.logger.debug "Unparsed comment: #{comments}"
517
+ YARD::Docstring.parser
519
518
  end
520
519
  end
521
520
  end
@@ -13,7 +13,7 @@ module Solargraph
13
13
  @cursor = cursor
14
14
  end
15
15
 
16
- # @return [Array<Pin::Base>]
16
+ # @return [Array<Pin::Base>] Relevant pins for infering the type of the Cursor's position
17
17
  def define
18
18
  return [] if cursor.comment? || cursor.chain.literal?
19
19
  result = cursor.chain.define(api_map, block, locals)
@@ -55,18 +55,19 @@ module Solargraph
55
55
  return ComplexType.try_parse('Object') if dfn && dfn.path == 'Class#new'
56
56
  end
57
57
  return result unless result.tag == 'self'
58
- ComplexType.try_parse(cursor.chain.base.infer(api_map, block, locals).namespace)
58
+ ComplexType.try_parse(cursor.chain.base.infer(api_map, block, locals).tag)
59
59
  end
60
60
 
61
61
  # Get an array of all the locals that are visible from the cursors's
62
62
  # position. Locals can be local variables, method parameters, or block
63
63
  # parameters. The array starts with the nearest local pin.
64
64
  #
65
- # @return [Array<Solargraph::Pin::Base>]
65
+ # @return [::Array<Solargraph::Pin::Base>]
66
66
  def locals
67
67
  @locals ||= source_map.locals_at(location)
68
68
  end
69
69
 
70
+ # @return [::Array<String>]
70
71
  def gates
71
72
  block.gates
72
73
  end
@@ -99,11 +100,12 @@ module Solargraph
99
100
  @source_map ||= api_map.source_map(cursor.filename)
100
101
  end
101
102
 
103
+ # @return [Location]
102
104
  def location
103
105
  Location.new(source_map.filename, Solargraph::Range.new(cursor.position, cursor.position))
104
106
  end
105
107
 
106
- # @return [Solargraph::Pin::Base]
108
+ # @return [Solargraph::Pin::Closure]
107
109
  def block
108
110
  @block ||= source_map.locate_block_pin(cursor.node_position.line, cursor.node_position.character)
109
111
  end
@@ -115,22 +117,7 @@ module Solargraph
115
117
  @context_pin ||= source_map.locate_named_path_pin(cursor.node_position.line, cursor.node_position.character)
116
118
  end
117
119
 
118
- # @return [Array<Pin::Base>]
119
- def yielded_self_pins
120
- return [] unless block.is_a?(Pin::Block) && block.receiver
121
- chain = Parser.chain(block.receiver, source_map.source.filename)
122
- receiver_pin = chain.define(api_map, context_pin, locals).first
123
- return [] if receiver_pin.nil?
124
- result = []
125
- ys = receiver_pin.docstring.tag(:yieldpublic)
126
- unless ys.nil? || ys.types.empty?
127
- ysct = ComplexType.try_parse(*ys.types).qualify(api_map, receiver_pin.context.namespace)
128
- result.concat api_map.get_complex_type_methods(ysct, '', false)
129
- end
130
- result
131
- end
132
-
133
- # @return [Array<Pin::KeywordParam]
120
+ # @return [Array<Pin::KeywordParam>]
134
121
  def complete_keyword_parameters
135
122
  return [] unless cursor.argument? && cursor.chain.links.one? && cursor.word =~ /^[a-z0-9_]*:?$/
136
123
  pins = signify
@@ -154,7 +141,7 @@ module Solargraph
154
141
  result
155
142
  end
156
143
 
157
- # @param result [Array<Pin::Base>]
144
+ # @param result [Enumerable<Pin::Base>]
158
145
  # @return [Completion]
159
146
  def package_completions result
160
147
  frag_start = cursor.start_of_word.to_s.downcase
@@ -165,6 +152,7 @@ module Solargraph
165
152
  Completion.new(filtered, cursor.range)
166
153
  end
167
154
 
155
+ # @return [Completion]
168
156
  def tag_complete
169
157
  result = []
170
158
  match = source_map.code[0..cursor.offset-1].match(/[\[<, ]([a-z0-9_:]*)\z/i)
@@ -183,6 +171,7 @@ module Solargraph
183
171
  package_completions(result)
184
172
  end
185
173
 
174
+ # @return [Completion]
186
175
  def code_complete
187
176
  result = []
188
177
  result.concat complete_keyword_parameters
@@ -224,14 +213,13 @@ module Solargraph
224
213
  result.concat api_map.get_constants(context_pin.context.namespace, *gates)
225
214
  result.concat api_map.get_methods(block.binder.namespace, scope: block.binder.scope, visibility: [:public, :private, :protected])
226
215
  result.concat api_map.get_methods('Kernel')
227
- # result.concat ApiMap.keywords
228
216
  result.concat api_map.keyword_pins.to_a
229
- result.concat yielded_self_pins
230
217
  end
231
218
  end
232
219
  package_completions(result)
233
220
  end
234
221
 
222
+ # @return [Array<Pin::Base>]
235
223
  def file_global_methods
236
224
  return [] if cursor.word.empty?
237
225
  source_map.pins.select do |pin|
@@ -64,6 +64,10 @@ module Solargraph
64
64
  pins.select{|pin| pin.is_a?(Pin::Closure) and pin.location.range.contain?(position)}.last
65
65
  end
66
66
 
67
+ # @param source_position [Position]
68
+ # @param comment_position [Position]
69
+ # @param comment [String]
70
+ # @return [void]
67
71
  def process_comment source_position, comment_position, comment
68
72
  return unless comment.encode('UTF-8', invalid: :replace, replace: '?') =~ DIRECTIVE_REGEXP
69
73
  cmnt = remove_inline_comment_hashes(comment)
@@ -79,6 +83,8 @@ module Solargraph
79
83
  end
80
84
 
81
85
  # @param comment [String]
86
+ # @param tag [String]
87
+ # @param start [Integer]
82
88
  # @return [Integer]
83
89
  def find_directive_line_number comment, tag, start
84
90
  # Avoid overruning the index
@@ -107,7 +113,8 @@ module Solargraph
107
113
  begin
108
114
  src = Solargraph::Source.load_string("def #{directive.tag.name};end", @source.filename)
109
115
  region = Parser::Region.new(source: src, closure: namespace)
110
- gen_pin = Parser.process_node(src.node, region).first.last
116
+ method_gen_pins = Parser.process_node(src.node, region).first.select { |pin| pin.is_a?(Pin::Method) }
117
+ gen_pin = method_gen_pins.last
111
118
  return if gen_pin.nil?
112
119
  # Move the location to the end of the line so it gets recognized
113
120
  # as originating from a comment
@@ -203,10 +210,14 @@ module Solargraph
203
210
  end
204
211
  end
205
212
 
213
+ # @param line1 [Integer]
214
+ # @param line2 [Integer]
206
215
  def no_empty_lines?(line1, line2)
207
216
  @code.lines[line1..line2].none? { |line| line.strip.empty? }
208
217
  end
209
218
 
219
+ # @param comment [String]
220
+ # @return [String]
210
221
  def remove_inline_comment_hashes comment
211
222
  ctxt = ''
212
223
  num = nil
@@ -1,8 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  require 'yard'
4
- require 'yard-solargraph'
5
- require 'set'
4
+ require 'solargraph/yard_tags'
6
5
 
7
6
  module Solargraph
8
7
  # An index of pins and other ApiMap-related data for a Source.
@@ -18,7 +17,7 @@ module Solargraph
18
17
  # @return [Array<Pin::Base>]
19
18
  attr_reader :pins
20
19
 
21
- # @return [Array<Pin::Base>]
20
+ # @return [Array<Pin::LocalVariable>]
22
21
  attr_reader :locals
23
22
 
24
23
  # @param source [Source]
@@ -35,13 +34,16 @@ module Solargraph
35
34
  @pin_select_cache = {}
36
35
  end
37
36
 
37
+ # @param klass [Class]
38
+ # @return [Array<Pin::Base>]
38
39
  def pins_by_class klass
39
40
  @pin_select_cache[klass] ||= @pin_class_hash.select { |key, _| key <= klass }.values.flatten
40
41
  end
41
42
 
43
+ # @return [Set<String>]
42
44
  def rebindable_method_names
43
45
  @rebindable_method_names ||= pins_by_class(Pin::Method)
44
- .select { |pin| pin.comments && pin.comments.include?('@yieldself') }
46
+ .select { |pin| pin.comments && pin.comments.include?('@yieldreceiver') }
45
47
  .map(&:name)
46
48
  .to_set
47
49
  end
@@ -99,10 +101,16 @@ module Solargraph
99
101
  (pins + locals).select { |pin| pin.location == location }
100
102
  end
101
103
 
104
+ # @param line [Integer]
105
+ # @param character [Integer]
106
+ # @return [Pin::Method,Pin::Namespace]
102
107
  def locate_named_path_pin line, character
103
108
  _locate_pin line, character, Pin::Namespace, Pin::Method
104
109
  end
105
110
 
111
+ # @param line [Integer]
112
+ # @param character [Integer]
113
+ # @return [Pin::Namespace,Pin::Method,Pin::Block]
106
114
  def locate_block_pin line, character
107
115
  _locate_pin line, character, Pin::Namespace, Pin::Method, Pin::Block
108
116
  end
@@ -166,6 +174,8 @@ module Solargraph
166
174
  @convention_pins || []
167
175
  end
168
176
 
177
+ # @param pins [Array<Pin::Base>]
178
+ # @return [Array<Pin::Base>]
169
179
  def convention_pins=(pins)
170
180
  # unmemoizing the document_symbols in case it was called from any of convnetions
171
181
  @document_symbols = nil
@@ -175,7 +185,7 @@ module Solargraph
175
185
  # @param line [Integer]
176
186
  # @param character [Integer]
177
187
  # @param klasses [Array<Class>]
178
- # @return [Pin::Base]
188
+ # @return [Pin::Base, nil]
179
189
  def _locate_pin line, character, *klasses
180
190
  position = Position.new(line, character)
181
191
  found = nil
@@ -51,8 +51,13 @@ module Solargraph
51
51
  # @return [Boolean]
52
52
  def any_types_match? api_map, expected, inferred
53
53
  return duck_types_match?(api_map, expected, inferred) if expected.duck_type?
54
+ # walk through the union expected type and see if any members
55
+ # of the union match the inferred type
54
56
  expected.each do |exp|
55
57
  next if exp.duck_type?
58
+ # @todo: there should be a level of typechecking where all
59
+ # unique types in the inferred must match one of the
60
+ # expected unique types
56
61
  inferred.each do |inf|
57
62
  # return true if exp == inf || api_map.super_and_sub?(fuzz(inf), fuzz(exp))
58
63
  return true if exp == inf || either_way?(api_map, inf, exp)
@@ -103,9 +108,12 @@ module Solargraph
103
108
  # @param cls2 [ComplexType::UniqueType]
104
109
  # @return [Boolean]
105
110
  def either_way?(api_map, cls1, cls2)
106
- f1 = fuzz(cls1)
107
- f2 = fuzz(cls2)
111
+ # @todo there should be a level of typechecking which uses the
112
+ # full tag with parameters to determine compatibility
113
+ f1 = cls1.name
114
+ f2 = cls2.name
108
115
  api_map.type_include?(f1, f2) || api_map.super_and_sub?(f1, f2) || api_map.super_and_sub?(f2, f1)
116
+ # api_map.type_include?(f1, f2) || api_map.super_and_sub?(f1, f2) || api_map.super_and_sub?(f2, f1)
109
117
  end
110
118
  end
111
119
  end