solargraph 0.58.0 → 0.59.0.dev.1

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 (166) hide show
  1. checksums.yaml +4 -4
  2. data/.envrc +3 -0
  3. data/.gitattributes +2 -0
  4. data/.github/workflows/linting.yml +4 -5
  5. data/.github/workflows/plugins.yml +40 -36
  6. data/.github/workflows/rspec.yml +45 -13
  7. data/.github/workflows/typecheck.yml +2 -2
  8. data/.rubocop_todo.yml +27 -49
  9. data/CHANGELOG.md +3 -0
  10. data/README.md +3 -3
  11. data/Rakefile +1 -0
  12. data/bin/solargraph +8 -8
  13. data/lib/solargraph/api_map/cache.rb +110 -110
  14. data/lib/solargraph/api_map/constants.rb +289 -279
  15. data/lib/solargraph/api_map/index.rb +204 -193
  16. data/lib/solargraph/api_map/source_to_yard.rb +109 -97
  17. data/lib/solargraph/api_map/store.rb +387 -384
  18. data/lib/solargraph/api_map.rb +1000 -945
  19. data/lib/solargraph/complex_type/conformance.rb +176 -0
  20. data/lib/solargraph/complex_type/type_methods.rb +242 -228
  21. data/lib/solargraph/complex_type/unique_type.rb +632 -482
  22. data/lib/solargraph/complex_type.rb +549 -444
  23. data/lib/solargraph/convention/data_definition/data_definition_node.rb +93 -91
  24. data/lib/solargraph/convention/data_definition.rb +108 -105
  25. data/lib/solargraph/convention/struct_definition/struct_assignment_node.rb +62 -61
  26. data/lib/solargraph/convention/struct_definition/struct_definition_node.rb +103 -102
  27. data/lib/solargraph/convention/struct_definition.rb +168 -164
  28. data/lib/solargraph/diagnostics/require_not_found.rb +54 -53
  29. data/lib/solargraph/diagnostics/rubocop.rb +119 -118
  30. data/lib/solargraph/diagnostics/rubocop_helpers.rb +70 -68
  31. data/lib/solargraph/diagnostics/type_check.rb +56 -55
  32. data/lib/solargraph/doc_map.rb +200 -439
  33. data/lib/solargraph/equality.rb +34 -34
  34. data/lib/solargraph/gem_pins.rb +97 -98
  35. data/lib/solargraph/language_server/host/dispatch.rb +131 -130
  36. data/lib/solargraph/language_server/host/message_worker.rb +113 -112
  37. data/lib/solargraph/language_server/host/sources.rb +100 -99
  38. data/lib/solargraph/language_server/host.rb +883 -878
  39. data/lib/solargraph/language_server/message/extended/check_gem_version.rb +109 -114
  40. data/lib/solargraph/language_server/message/extended/document.rb +24 -23
  41. data/lib/solargraph/language_server/message/text_document/completion.rb +58 -56
  42. data/lib/solargraph/language_server/message/text_document/definition.rb +42 -40
  43. data/lib/solargraph/language_server/message/text_document/document_symbol.rb +28 -26
  44. data/lib/solargraph/language_server/message/text_document/formatting.rb +150 -148
  45. data/lib/solargraph/language_server/message/text_document/hover.rb +60 -58
  46. data/lib/solargraph/language_server/message/text_document/signature_help.rb +25 -24
  47. data/lib/solargraph/language_server/message/text_document/type_definition.rb +27 -25
  48. data/lib/solargraph/language_server/message/workspace/workspace_symbol.rb +25 -23
  49. data/lib/solargraph/library.rb +729 -683
  50. data/lib/solargraph/location.rb +87 -82
  51. data/lib/solargraph/logging.rb +57 -37
  52. data/lib/solargraph/parser/comment_ripper.rb +76 -69
  53. data/lib/solargraph/parser/flow_sensitive_typing.rb +483 -255
  54. data/lib/solargraph/parser/node_processor/base.rb +122 -92
  55. data/lib/solargraph/parser/node_processor.rb +63 -62
  56. data/lib/solargraph/parser/parser_gem/class_methods.rb +167 -149
  57. data/lib/solargraph/parser/parser_gem/node_chainer.rb +191 -166
  58. data/lib/solargraph/parser/parser_gem/node_methods.rb +506 -486
  59. data/lib/solargraph/parser/parser_gem/node_processors/and_node.rb +22 -22
  60. data/lib/solargraph/parser/parser_gem/node_processors/args_node.rb +61 -59
  61. data/lib/solargraph/parser/parser_gem/node_processors/begin_node.rb +24 -15
  62. data/lib/solargraph/parser/parser_gem/node_processors/block_node.rb +46 -46
  63. data/lib/solargraph/parser/parser_gem/node_processors/def_node.rb +60 -53
  64. data/lib/solargraph/parser/parser_gem/node_processors/if_node.rb +53 -23
  65. data/lib/solargraph/parser/parser_gem/node_processors/ivasgn_node.rb +41 -40
  66. data/lib/solargraph/parser/parser_gem/node_processors/lvasgn_node.rb +30 -29
  67. data/lib/solargraph/parser/parser_gem/node_processors/masgn_node.rb +61 -59
  68. data/lib/solargraph/parser/parser_gem/node_processors/opasgn_node.rb +98 -98
  69. data/lib/solargraph/parser/parser_gem/node_processors/or_node.rb +22 -0
  70. data/lib/solargraph/parser/parser_gem/node_processors/orasgn_node.rb +17 -17
  71. data/lib/solargraph/parser/parser_gem/node_processors/resbody_node.rb +39 -38
  72. data/lib/solargraph/parser/parser_gem/node_processors/sclass_node.rb +53 -52
  73. data/lib/solargraph/parser/parser_gem/node_processors/send_node.rb +296 -291
  74. data/lib/solargraph/parser/parser_gem/node_processors/when_node.rb +23 -0
  75. data/lib/solargraph/parser/parser_gem/node_processors/while_node.rb +33 -29
  76. data/lib/solargraph/parser/parser_gem/node_processors.rb +74 -70
  77. data/lib/solargraph/parser/region.rb +75 -69
  78. data/lib/solargraph/parser/snippet.rb +17 -17
  79. data/lib/solargraph/pin/base.rb +761 -729
  80. data/lib/solargraph/pin/base_variable.rb +418 -126
  81. data/lib/solargraph/pin/block.rb +126 -104
  82. data/lib/solargraph/pin/breakable.rb +13 -9
  83. data/lib/solargraph/pin/callable.rb +278 -231
  84. data/lib/solargraph/pin/closure.rb +68 -72
  85. data/lib/solargraph/pin/common.rb +94 -79
  86. data/lib/solargraph/pin/compound_statement.rb +55 -0
  87. data/lib/solargraph/pin/conversions.rb +124 -123
  88. data/lib/solargraph/pin/delegated_method.rb +131 -120
  89. data/lib/solargraph/pin/documenting.rb +115 -114
  90. data/lib/solargraph/pin/instance_variable.rb +38 -34
  91. data/lib/solargraph/pin/keyword.rb +16 -20
  92. data/lib/solargraph/pin/local_variable.rb +31 -75
  93. data/lib/solargraph/pin/method.rb +720 -672
  94. data/lib/solargraph/pin/method_alias.rb +42 -34
  95. data/lib/solargraph/pin/namespace.rb +121 -115
  96. data/lib/solargraph/pin/parameter.rb +338 -275
  97. data/lib/solargraph/pin/proxy_type.rb +40 -39
  98. data/lib/solargraph/pin/reference/override.rb +47 -47
  99. data/lib/solargraph/pin/reference/superclass.rb +17 -15
  100. data/lib/solargraph/pin/reference.rb +41 -39
  101. data/lib/solargraph/pin/search.rb +62 -61
  102. data/lib/solargraph/pin/signature.rb +69 -61
  103. data/lib/solargraph/pin/symbol.rb +53 -53
  104. data/lib/solargraph/pin/until.rb +18 -18
  105. data/lib/solargraph/pin/while.rb +18 -18
  106. data/lib/solargraph/pin.rb +46 -44
  107. data/lib/solargraph/pin_cache.rb +665 -245
  108. data/lib/solargraph/position.rb +118 -119
  109. data/lib/solargraph/range.rb +112 -112
  110. data/lib/solargraph/rbs_map/conversions.rb +846 -823
  111. data/lib/solargraph/rbs_map/core_map.rb +65 -58
  112. data/lib/solargraph/rbs_map/stdlib_map.rb +72 -43
  113. data/lib/solargraph/rbs_map.rb +217 -163
  114. data/lib/solargraph/shell.rb +397 -352
  115. data/lib/solargraph/source/chain/call.rb +372 -337
  116. data/lib/solargraph/source/chain/constant.rb +28 -26
  117. data/lib/solargraph/source/chain/hash.rb +35 -34
  118. data/lib/solargraph/source/chain/if.rb +29 -28
  119. data/lib/solargraph/source/chain/instance_variable.rb +34 -13
  120. data/lib/solargraph/source/chain/literal.rb +53 -48
  121. data/lib/solargraph/source/chain/or.rb +31 -23
  122. data/lib/solargraph/source/chain.rb +294 -291
  123. data/lib/solargraph/source/change.rb +89 -82
  124. data/lib/solargraph/source/cursor.rb +172 -166
  125. data/lib/solargraph/source/encoding_fixes.rb +23 -23
  126. data/lib/solargraph/source/source_chainer.rb +204 -194
  127. data/lib/solargraph/source/updater.rb +59 -55
  128. data/lib/solargraph/source.rb +524 -498
  129. data/lib/solargraph/source_map/clip.rb +237 -226
  130. data/lib/solargraph/source_map/data.rb +37 -34
  131. data/lib/solargraph/source_map/mapper.rb +282 -259
  132. data/lib/solargraph/source_map.rb +220 -212
  133. data/lib/solargraph/type_checker/problem.rb +34 -32
  134. data/lib/solargraph/type_checker/rules.rb +157 -84
  135. data/lib/solargraph/type_checker.rb +895 -814
  136. data/lib/solargraph/version.rb +5 -5
  137. data/lib/solargraph/workspace/config.rb +257 -255
  138. data/lib/solargraph/workspace/gemspecs.rb +367 -0
  139. data/lib/solargraph/workspace/require_paths.rb +98 -97
  140. data/lib/solargraph/workspace.rb +362 -220
  141. data/lib/solargraph/yard_map/helpers.rb +45 -44
  142. data/lib/solargraph/yard_map/mapper/to_method.rb +134 -130
  143. data/lib/solargraph/yard_map/mapper/to_namespace.rb +32 -31
  144. data/lib/solargraph/yard_map/mapper.rb +84 -79
  145. data/lib/solargraph/yardoc.rb +97 -87
  146. data/lib/solargraph.rb +126 -105
  147. data/rbs/fills/rubygems/0/dependency.rbs +193 -0
  148. data/rbs/fills/tuple/tuple.rbs +28 -0
  149. data/rbs/shims/ast/0/node.rbs +5 -0
  150. data/rbs/shims/diff-lcs/1.5/diff-lcs.rbs +11 -0
  151. data/rbs_collection.yaml +1 -1
  152. data/solargraph.gemspec +2 -1
  153. metadata +23 -17
  154. data/lib/solargraph/type_checker/checks.rb +0 -124
  155. data/lib/solargraph/type_checker/param_def.rb +0 -37
  156. data/lib/solargraph/yard_map/to_method.rb +0 -89
  157. data/sig/shims/ast/0/node.rbs +0 -5
  158. /data/{sig → rbs}/shims/ast/2.4/.rbs_meta.yaml +0 -0
  159. /data/{sig → rbs}/shims/ast/2.4/ast.rbs +0 -0
  160. /data/{sig → rbs}/shims/parser/3.2.0.1/builders/default.rbs +0 -0
  161. /data/{sig → rbs}/shims/parser/3.2.0.1/manifest.yaml +0 -0
  162. /data/{sig → rbs}/shims/parser/3.2.0.1/parser.rbs +0 -0
  163. /data/{sig → rbs}/shims/parser/3.2.0.1/polyfill.rbs +0 -0
  164. /data/{sig → rbs}/shims/thor/1.2.0.1/.rbs_meta.yaml +0 -0
  165. /data/{sig → rbs}/shims/thor/1.2.0.1/manifest.yaml +0 -0
  166. /data/{sig → rbs}/shims/thor/1.2.0.1/thor.rbs +0 -0
@@ -1,259 +1,282 @@
1
- # frozen_string_literal: true
2
-
3
- module Solargraph
4
- class SourceMap
5
- # The Mapper generates pins and other data for SourceMaps.
6
- #
7
- # This class is used internally by the SourceMap class. Users should not
8
- # normally need to call it directly.
9
- #
10
- class Mapper
11
- # include Source::NodeMethods
12
-
13
- private_class_method :new
14
-
15
- DIRECTIVE_REGEXP = /(@\!method|@\!attribute|@\!visibility|@\!domain|@\!macro|@\!parse|@\!override)/.freeze
16
-
17
- # Generate the data.
18
- #
19
- # @param source [Source]
20
- # @return [Array]
21
- def map source
22
- @source = source
23
- @filename = source.filename
24
- @code = source.code
25
- @comments = source.comments
26
- @pins, @locals = Parser.map(source)
27
- @pins.each { |p| p.source = :code }
28
- @locals.each { |l| l.source = :code }
29
- process_comment_directives
30
- [@pins, @locals]
31
- # rescue Exception => e
32
- # Solargraph.logger.warn "Error mapping #{source.filename}: [#{e.class}] #{e.message}"
33
- # Solargraph.logger.warn e.backtrace.join("\n")
34
- # [[], []]
35
- end
36
-
37
- # @param filename [String]
38
- # @param code [String]
39
- # @return [Array]
40
- def unmap filename, code
41
- s = Position.new(0, 0)
42
- e = Position.from_offset(code, code.length)
43
- location = Location.new(filename, Range.new(s, e))
44
- [[Pin::Namespace.new(location: location, name: '', source: :source_map)], []]
45
- end
46
-
47
- class << self
48
- # @param source [Source]
49
- # @return [Array]
50
- def map source
51
- return new.unmap(source.filename, source.code) unless source.parsed?
52
- new.map source
53
- end
54
- end
55
-
56
- # @return [Array<Solargraph::Pin::Base>]
57
- def pins
58
- # @type [Array<Solargraph::Pin::Base>]
59
- @pins ||= []
60
- end
61
-
62
- # @param position [Solargraph::Position]
63
- # @return [Solargraph::Pin::Closure]
64
- def closure_at(position)
65
- pins.select{|pin| pin.is_a?(Pin::Closure) and pin.location.range.contain?(position)}.last
66
- end
67
-
68
- # @param source_position [Position]
69
- # @param comment_position [Position]
70
- # @param comment [String]
71
- # @return [void]
72
- def process_comment source_position, comment_position, comment
73
- return unless comment.encode('UTF-8', invalid: :replace, replace: '?') =~ DIRECTIVE_REGEXP
74
- cmnt = remove_inline_comment_hashes(comment)
75
- parse = Solargraph::Source.parse_docstring(cmnt)
76
- last_line = 0
77
- # @param d [YARD::Tags::Directive]
78
- parse.directives.each do |d|
79
- line_num = find_directive_line_number(cmnt, d.tag.tag_name, last_line)
80
- pos = Solargraph::Position.new(comment_position.line + line_num - 1, comment_position.column)
81
- process_directive(source_position, pos, d)
82
- last_line = line_num + 1
83
- end
84
- end
85
-
86
- # @param comment [String]
87
- # @param tag [String]
88
- # @param start [Integer]
89
- # @return [Integer]
90
- def find_directive_line_number comment, tag, start
91
- # Avoid overruning the index
92
- return start unless start < comment.lines.length
93
- num = comment.lines[start..-1].find_index do |line|
94
- # Legacy method directives might be `@method` instead of `@!method`
95
- # @todo Legacy syntax should probably emit a warning
96
- line.include?("@!#{tag}") || (tag == 'method' && line.include?("@#{tag}"))
97
- end
98
- num.to_i + start
99
- end
100
-
101
- # @param source_position [Position]
102
- # @param comment_position [Position]
103
- # @param directive [YARD::Tags::Directive]
104
- # @return [void]
105
- def process_directive source_position, comment_position, directive
106
- docstring = Solargraph::Source.parse_docstring(directive.tag.text).to_docstring
107
- location = Location.new(@filename, Range.new(comment_position, comment_position))
108
- case directive.tag.tag_name
109
- when 'method'
110
- namespace = closure_at(source_position) || @pins.first
111
- if namespace.location.range.start.line < comment_position.line
112
- namespace = closure_at(comment_position)
113
- end
114
- begin
115
- src = Solargraph::Source.load_string("def #{directive.tag.name};end", @source.filename)
116
- region = Parser::Region.new(source: src, closure: namespace)
117
- method_gen_pins = Parser.process_node(src.node, region).first.select { |pin| pin.is_a?(Pin::Method) }
118
- gen_pin = method_gen_pins.last
119
- return if gen_pin.nil?
120
- # Move the location to the end of the line so it gets recognized
121
- # as originating from a comment
122
- shifted = Solargraph::Position.new(comment_position.line, @code.lines[comment_position.line].to_s.chomp.length)
123
- # @todo: Smelly instance variable access
124
- gen_pin.instance_variable_set(:@comments, docstring.all.to_s)
125
- gen_pin.instance_variable_set(:@location, Solargraph::Location.new(@filename, Range.new(shifted, shifted)))
126
- gen_pin.instance_variable_set(:@explicit, false)
127
- @pins.push gen_pin
128
- rescue Parser::SyntaxError => e
129
- # @todo Handle error in directive
130
- end
131
- when 'attribute'
132
- return if directive.tag.name.nil?
133
- namespace = closure_at(source_position)
134
- t = (directive.tag.types.nil? || directive.tag.types.empty?) ? nil : directive.tag.types.flatten.join('')
135
- if t.nil? || t.include?('r')
136
- pins.push Solargraph::Pin::Method.new(
137
- location: location,
138
- closure: namespace,
139
- name: directive.tag.name,
140
- comments: docstring.all.to_s,
141
- scope: namespace.is_a?(Pin::Singleton) ? :class : :instance,
142
- visibility: :public,
143
- explicit: false,
144
- attribute: true,
145
- source: :source_map
146
- )
147
- end
148
- if t.nil? || t.include?('w')
149
- method_pin = Solargraph::Pin::Method.new(
150
- location: location,
151
- closure: namespace,
152
- name: "#{directive.tag.name}=",
153
- comments: docstring.all.to_s,
154
- scope: namespace.is_a?(Pin::Singleton) ? :class : :instance,
155
- visibility: :public,
156
- attribute: true,
157
- source: :source_map
158
- )
159
- pins.push method_pin
160
- method_pin.parameters.push Pin::Parameter.new(name: 'value', decl: :arg, closure: pins.last, source: :source_map)
161
- if pins.last.return_type.defined?
162
- pins.last.docstring.add_tag YARD::Tags::Tag.new(:param, '', pins.last.return_type.to_s.split(', '), 'value')
163
- end
164
- end
165
- when 'visibility'
166
-
167
- kind = directive.tag.text&.to_sym
168
- return unless [:private, :protected, :public].include?(kind)
169
-
170
- name = directive.tag.name
171
- closure = closure_at(source_position) || @pins.first
172
- if closure.location.range.start.line < comment_position.line
173
- closure = closure_at(comment_position)
174
- end
175
- if closure.is_a?(Pin::Method) && no_empty_lines?(comment_position.line, source_position.line)
176
- # @todo Smelly instance variable access
177
- closure.instance_variable_set(:@visibility, kind)
178
- else
179
- matches = pins.select{ |pin| pin.is_a?(Pin::Method) && pin.name == name && pin.namespace == namespace && pin.context.scope == namespace.is_a?(Pin::Singleton) ? :class : :instance }
180
- matches.each do |pin|
181
- # @todo Smelly instance variable access
182
- pin.instance_variable_set(:@visibility, kind)
183
- end
184
- end
185
-
186
- when 'parse'
187
- begin
188
- ns = closure_at(source_position)
189
- src = Solargraph::Source.load_string(directive.tag.text, @source.filename)
190
- region = Parser::Region.new(source: src, closure: ns)
191
- # @todo These pins may need to be marked not explicit
192
- index = @pins.length
193
- loff = if @code.lines[comment_position.line].strip.end_with?('@!parse')
194
- comment_position.line + 1
195
- else
196
- comment_position.line
197
- end
198
- Parser.process_node(src.node, region, @pins)
199
- @pins[index..-1].each do |p|
200
- # @todo Smelly instance variable access
201
- p.location.range.start.instance_variable_set(:@line, p.location.range.start.line + loff)
202
- p.location.range.ending.instance_variable_set(:@line, p.location.range.ending.line + loff)
203
- end
204
- rescue Parser::SyntaxError => e
205
- # @todo Handle parser errors in !parse directives
206
- end
207
- when 'domain'
208
- namespace = closure_at(source_position) || Pin::ROOT_PIN
209
- namespace.domains.concat directive.tag.types unless directive.tag.types.nil?
210
- when 'override'
211
- pins.push Pin::Reference::Override.new(location, directive.tag.name, docstring.tags,
212
- source: :source_map)
213
- when 'macro'
214
- # @todo Handle macros
215
- end
216
- end
217
-
218
- # @param line1 [Integer]
219
- # @param line2 [Integer]
220
- def no_empty_lines?(line1, line2)
221
- @code.lines[line1..line2].none? { |line| line.strip.empty? }
222
- end
223
-
224
- # @param comment [String]
225
- # @return [String]
226
- def remove_inline_comment_hashes comment
227
- ctxt = ''
228
- num = nil
229
- started = false
230
- comment.lines.each { |l|
231
- # Trim the comment and minimum leading whitespace
232
- p = l.encode('UTF-8', invalid: :replace, replace: '?').gsub(/^#+/, '')
233
- if num.nil? && !p.strip.empty?
234
- num = p.index(/[^ ]/)
235
- started = true
236
- elsif started && !p.strip.empty?
237
- cur = p.index(/[^ ]/)
238
- num = cur if cur < num
239
- end
240
- ctxt += "#{p[num..-1]}" if started
241
- }
242
- ctxt
243
- end
244
-
245
- # @return [void]
246
- def process_comment_directives
247
- return unless @code.encode('UTF-8', invalid: :replace, replace: '?') =~ DIRECTIVE_REGEXP
248
- code_lines = @code.lines
249
- @source.associated_comments.each do |line, comments|
250
- src_pos = line ? Position.new(line, code_lines[line].to_s.chomp.index(/[^\s]/) || 0) : Position.new(code_lines.length, 0)
251
- com_pos = Position.new(line + 1 - comments.lines.length, 0)
252
- process_comment(src_pos, com_pos, comments)
253
- end
254
- rescue StandardError => e
255
- raise e.class, "Error processing comment directives in #{@filename}: #{e.message}"
256
- end
257
- end
258
- end
259
- end
1
+ # frozen_string_literal: true
2
+
3
+ module Solargraph
4
+ class SourceMap
5
+ # The Mapper generates pins and other data for SourceMaps.
6
+ #
7
+ # This class is used internally by the SourceMap class. Users should not
8
+ # normally need to call it directly.
9
+ #
10
+ class Mapper
11
+ # include Source::NodeMethods
12
+
13
+ private_class_method :new
14
+
15
+ DIRECTIVE_REGEXP = /(@\!method|@\!attribute|@\!visibility|@\!domain|@\!macro|@\!parse|@\!override)/.freeze
16
+
17
+ # Generate the data.
18
+ #
19
+ # @param source [Source]
20
+ # @return [Array]
21
+ def map source
22
+ @source = source
23
+ @filename = source.filename
24
+ @code = source.code
25
+ @comments = source.comments
26
+ @pins, @locals = Parser.map(source)
27
+ # @param p [Solargraph::Pin::Base]
28
+ @pins.each { |p| p.source = :code }
29
+ @locals.each { |l| l.source = :code }
30
+ process_comment_directives
31
+ [@pins, @locals]
32
+ # rescue Exception => e
33
+ # Solargraph.logger.warn "Error mapping #{source.filename}: [#{e.class}] #{e.message}"
34
+ # Solargraph.logger.warn e.backtrace.join("\n")
35
+ # [[], []]
36
+ end
37
+
38
+ # @param filename [String]
39
+ # @param code [String]
40
+ # @return [Array]
41
+ def unmap filename, code
42
+ s = Position.new(0, 0)
43
+ e = Position.from_offset(code, code.length)
44
+ location = Location.new(filename, Range.new(s, e))
45
+ [[Pin::Namespace.new(location: location, name: '', source: :source_map)], []]
46
+ end
47
+
48
+ class << self
49
+ # @param source [Source]
50
+ # @return [Array]
51
+ def map source
52
+ # @sg-ignore Need to add nil check here
53
+ return new.unmap(source.filename, source.code) unless source.parsed?
54
+ new.map source
55
+ end
56
+ end
57
+
58
+ # @return [Array<Solargraph::Pin::Base>]
59
+ def pins
60
+ # @type [Array<Solargraph::Pin::Base>]
61
+ @pins ||= []
62
+ end
63
+
64
+ # @param position [Solargraph::Position]
65
+ # @return [Solargraph::Pin::Closure]
66
+ def closure_at(position)
67
+ # @sg-ignore Need to add nil check here
68
+ pins.select{|pin| pin.is_a?(Pin::Closure) and pin.location.range.contain?(position)}.last
69
+ end
70
+
71
+ # @param source_position [Position]
72
+ # @param comment_position [Position]
73
+ # @param comment [String]
74
+ # @return [void]
75
+ def process_comment source_position, comment_position, comment
76
+ return unless comment.encode('UTF-8', invalid: :replace, replace: '?') =~ DIRECTIVE_REGEXP
77
+ cmnt = remove_inline_comment_hashes(comment)
78
+ parse = Solargraph::Source.parse_docstring(cmnt)
79
+ last_line = 0
80
+ # @param d [YARD::Tags::Directive]
81
+ parse.directives.each do |d|
82
+ line_num = find_directive_line_number(cmnt, d.tag.tag_name, last_line)
83
+ pos = Solargraph::Position.new(comment_position.line + line_num - 1, comment_position.column)
84
+ process_directive(source_position, pos, d)
85
+ last_line = line_num + 1
86
+ end
87
+ end
88
+
89
+ # @param comment [String]
90
+ # @param tag [String]
91
+ # @param start [Integer]
92
+ # @return [Integer]
93
+ def find_directive_line_number comment, tag, start
94
+ # Avoid overruning the index
95
+ return start unless start < comment.lines.length
96
+ # @sg-ignore Need to add nil check here
97
+ num = comment.lines[start..-1].find_index do |line|
98
+ # Legacy method directives might be `@method` instead of `@!method`
99
+ # @todo Legacy syntax should probably emit a warning
100
+ line.include?("@!#{tag}") || (tag == 'method' && line.include?("@#{tag}"))
101
+ end
102
+ # @sg-ignore Need to add nil check here
103
+ num.to_i + start
104
+ end
105
+
106
+ # @param source_position [Position]
107
+ # @param comment_position [Position]
108
+ # @param directive [YARD::Tags::Directive]
109
+ # @return [void]
110
+ def process_directive source_position, comment_position, directive
111
+ # @sg-ignore Need to add nil check here
112
+ docstring = Solargraph::Source.parse_docstring(directive.tag.text).to_docstring
113
+ location = Location.new(@filename, Range.new(comment_position, comment_position))
114
+ case directive.tag.tag_name
115
+ when 'method'
116
+ namespace = closure_at(source_position) || @pins.first
117
+ # @todo Missed nil violation
118
+ # @todo Need to add nil check here
119
+ if namespace.location.range.start.line < comment_position.line
120
+ namespace = closure_at(comment_position)
121
+ end
122
+ begin
123
+ src = Solargraph::Source.load_string("def #{directive.tag.name};end", @source.filename)
124
+ region = Parser::Region.new(source: src, closure: namespace)
125
+ # @type [Array<Pin::Method>]
126
+ method_gen_pins = Parser.process_node(src.node, region).first.select { |pin| pin.is_a?(Pin::Method) }
127
+ gen_pin = method_gen_pins.last
128
+ return if gen_pin.nil?
129
+ # Move the location to the end of the line so it gets recognized
130
+ # as originating from a comment
131
+ shifted = Solargraph::Position.new(comment_position.line, @code.lines[comment_position.line].to_s.chomp.length)
132
+ # @todo: Smelly instance variable access
133
+ gen_pin.instance_variable_set(:@comments, docstring.all.to_s)
134
+ gen_pin.instance_variable_set(:@location, Solargraph::Location.new(@filename, Range.new(shifted, shifted)))
135
+ gen_pin.instance_variable_set(:@explicit, false)
136
+ @pins.push gen_pin
137
+ rescue Parser::SyntaxError => e
138
+ # @todo Handle error in directive
139
+ end
140
+ when 'attribute'
141
+ return if directive.tag.name.nil?
142
+ namespace = closure_at(source_position)
143
+ t = (directive.tag.types.nil? || directive.tag.types.empty?) ? nil : directive.tag.types.flatten.join('')
144
+ if t.nil? || t.include?('r')
145
+ pins.push Solargraph::Pin::Method.new(
146
+ location: location,
147
+ closure: namespace,
148
+ name: directive.tag.name,
149
+ comments: docstring.all.to_s,
150
+ scope: namespace.is_a?(Pin::Singleton) ? :class : :instance,
151
+ visibility: :public,
152
+ explicit: false,
153
+ attribute: true,
154
+ source: :source_map
155
+ )
156
+ end
157
+ if t.nil? || t.include?('w')
158
+ method_pin = Solargraph::Pin::Method.new(
159
+ location: location,
160
+ closure: namespace,
161
+ name: "#{directive.tag.name}=",
162
+ comments: docstring.all.to_s,
163
+ scope: namespace.is_a?(Pin::Singleton) ? :class : :instance,
164
+ visibility: :public,
165
+ attribute: true,
166
+ source: :source_map
167
+ )
168
+ pins.push method_pin
169
+ method_pin.parameters.push Pin::Parameter.new(name: 'value', decl: :arg, closure: pins.last, source: :source_map)
170
+ if pins.last.return_type.defined?
171
+ pins.last.docstring.add_tag YARD::Tags::Tag.new(:param, '', pins.last.return_type.to_s.split(', '), 'value')
172
+ end
173
+ end
174
+ when 'visibility'
175
+
176
+ kind = directive.tag.text&.to_sym
177
+ # @sg-ignore Need to look at Tuple#include? handling
178
+ return unless [:private, :protected, :public].include?(kind)
179
+
180
+ name = directive.tag.name
181
+ closure = closure_at(source_position) || @pins.first
182
+ # @todo Missed nil violation
183
+ # @todo Need to add nil check here
184
+ if closure.location.range.start.line < comment_position.line
185
+ closure = closure_at(comment_position)
186
+ end
187
+ if closure.is_a?(Pin::Method) && no_empty_lines?(comment_position.line, source_position.line)
188
+ # @todo Smelly instance variable access
189
+ closure.instance_variable_set(:@visibility, kind)
190
+ else
191
+ matches = pins.select{ |pin| pin.is_a?(Pin::Method) && pin.name == name && pin.namespace == namespace && pin.context.scope == namespace.is_a?(Pin::Singleton) ? :class : :instance }
192
+ matches.each do |pin|
193
+ # @todo Smelly instance variable access
194
+ pin.instance_variable_set(:@visibility, kind)
195
+ end
196
+ end
197
+
198
+ when 'parse'
199
+ begin
200
+ ns = closure_at(source_position)
201
+ # @sg-ignore Need to add nil check here
202
+ src = Solargraph::Source.load_string(directive.tag.text, @source.filename)
203
+ region = Parser::Region.new(source: src, closure: ns)
204
+ # @todo These pins may need to be marked not explicit
205
+ index = @pins.length
206
+ loff = if @code.lines[comment_position.line].strip.end_with?('@!parse')
207
+ comment_position.line + 1
208
+ else
209
+ comment_position.line
210
+ end
211
+ locals = []
212
+ ivars = []
213
+ Parser.process_node(src.node, region, @pins, locals, ivars)
214
+ @pins.concat ivars
215
+ # @sg-ignore Need to add nil check here
216
+ @pins[index..-1].each do |p|
217
+ # @todo Smelly instance variable access
218
+ p.location.range.start.instance_variable_set(:@line, p.location.range.start.line + loff)
219
+ p.location.range.ending.instance_variable_set(:@line, p.location.range.ending.line + loff)
220
+ end
221
+ rescue Parser::SyntaxError => e
222
+ # @todo Handle parser errors in !parse directives
223
+ end
224
+ when 'domain'
225
+ namespace = closure_at(source_position) || Pin::ROOT_PIN
226
+ # @sg-ignore flow sensitive typing should be able to handle redefinition
227
+ namespace.domains.concat directive.tag.types unless directive.tag.types.nil?
228
+ when 'override'
229
+ # @sg-ignore Need to add nil check here
230
+ pins.push Pin::Reference::Override.new(location, directive.tag.name, docstring.tags,
231
+ source: :source_map)
232
+ when 'macro'
233
+ # @todo Handle macros
234
+ end
235
+ end
236
+
237
+ # @param line1 [Integer]
238
+ # @param line2 [Integer]
239
+ # @sg-ignore Need to add nil check here
240
+ def no_empty_lines?(line1, line2)
241
+ # @sg-ignore Need to add nil check here
242
+ @code.lines[line1..line2].none? { |line| line.strip.empty? }
243
+ end
244
+
245
+ # @param comment [String]
246
+ # @return [String]
247
+ def remove_inline_comment_hashes comment
248
+ ctxt = ''
249
+ num = nil
250
+ started = false
251
+ comment.lines.each { |l|
252
+ # Trim the comment and minimum leading whitespace
253
+ p = l.encode('UTF-8', invalid: :replace, replace: '?').gsub(/^#+/, '')
254
+ if num.nil? && !p.strip.empty?
255
+ num = p.index(/[^ ]/)
256
+ started = true
257
+ elsif started && !p.strip.empty?
258
+ cur = p.index(/[^ ]/)
259
+ # @sg-ignore Need to add nil check here
260
+ num = cur if cur < num
261
+ end
262
+ ctxt += "#{p[num..-1]}" if started
263
+ }
264
+ ctxt
265
+ end
266
+
267
+ # @return [void]
268
+ def process_comment_directives
269
+ return unless @code.encode('UTF-8', invalid: :replace, replace: '?') =~ DIRECTIVE_REGEXP
270
+ code_lines = @code.lines
271
+ @source.associated_comments.each do |line, comments|
272
+ src_pos = line ? Position.new(line, code_lines[line].to_s.chomp.index(/[^\s]/) || 0) : Position.new(code_lines.length, 0)
273
+ # @sg-ignore Need to add nil check here
274
+ com_pos = Position.new(line + 1 - comments.lines.length, 0)
275
+ process_comment(src_pos, com_pos, comments)
276
+ end
277
+ rescue StandardError => e
278
+ raise e.class, "Error processing comment directives in #{@filename}: #{e.message}"
279
+ end
280
+ end
281
+ end
282
+ end