solargraph 0.50.0 → 0.58.2

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 (264) hide show
  1. checksums.yaml +4 -4
  2. data/.gitattributes +2 -0
  3. data/.github/workflows/linting.yml +127 -0
  4. data/.github/workflows/plugins.yml +218 -0
  5. data/.github/workflows/rspec.yml +58 -12
  6. data/.github/workflows/typecheck.yml +39 -0
  7. data/.gitignore +8 -0
  8. data/.overcommit.yml +72 -0
  9. data/.rspec +1 -0
  10. data/.rubocop.yml +66 -0
  11. data/.rubocop_todo.yml +1279 -0
  12. data/.yardopts +3 -2
  13. data/CHANGELOG.md +306 -3
  14. data/README.md +29 -18
  15. data/Rakefile +125 -13
  16. data/SPONSORS.md +2 -9
  17. data/bin/solargraph +3 -0
  18. data/lib/solargraph/api_map/cache.rb +110 -70
  19. data/lib/solargraph/api_map/constants.rb +279 -0
  20. data/lib/solargraph/api_map/index.rb +193 -0
  21. data/lib/solargraph/api_map/source_to_yard.rb +97 -81
  22. data/lib/solargraph/api_map/store.rb +384 -268
  23. data/lib/solargraph/api_map.rb +945 -704
  24. data/lib/solargraph/bench.rb +21 -3
  25. data/lib/solargraph/complex_type/type_methods.rb +228 -134
  26. data/lib/solargraph/complex_type/unique_type.rb +482 -132
  27. data/lib/solargraph/complex_type.rb +444 -254
  28. data/lib/solargraph/convention/active_support_concern.rb +111 -0
  29. data/lib/solargraph/convention/base.rb +20 -3
  30. data/lib/solargraph/convention/data_definition/data_assignment_node.rb +61 -0
  31. data/lib/solargraph/convention/data_definition/data_definition_node.rb +91 -0
  32. data/lib/solargraph/convention/data_definition.rb +105 -0
  33. data/lib/solargraph/convention/gemspec.rb +3 -2
  34. data/lib/solargraph/convention/struct_definition/struct_assignment_node.rb +61 -0
  35. data/lib/solargraph/convention/struct_definition/struct_definition_node.rb +102 -0
  36. data/lib/solargraph/convention/struct_definition.rb +164 -0
  37. data/lib/solargraph/convention.rb +36 -7
  38. data/lib/solargraph/converters/dd.rb +5 -0
  39. data/lib/solargraph/converters/dl.rb +3 -0
  40. data/lib/solargraph/converters/dt.rb +3 -0
  41. data/lib/solargraph/diagnostics/require_not_found.rb +53 -53
  42. data/lib/solargraph/diagnostics/rubocop.rb +118 -112
  43. data/lib/solargraph/diagnostics/rubocop_helpers.rb +68 -65
  44. data/lib/solargraph/diagnostics/type_check.rb +55 -54
  45. data/lib/solargraph/diagnostics.rb +2 -2
  46. data/lib/solargraph/doc_map.rb +439 -0
  47. data/lib/solargraph/environ.rb +9 -2
  48. data/lib/solargraph/equality.rb +34 -0
  49. data/lib/solargraph/gem_pins.rb +98 -0
  50. data/lib/solargraph/language_server/host/diagnoser.rb +89 -89
  51. data/lib/solargraph/language_server/host/dispatch.rb +130 -111
  52. data/lib/solargraph/language_server/host/message_worker.rb +112 -59
  53. data/lib/solargraph/language_server/host/sources.rb +99 -156
  54. data/lib/solargraph/language_server/host.rb +878 -869
  55. data/lib/solargraph/language_server/message/base.rb +20 -12
  56. data/lib/solargraph/language_server/message/completion_item/resolve.rb +3 -1
  57. data/lib/solargraph/language_server/message/extended/check_gem_version.rb +114 -100
  58. data/lib/solargraph/language_server/message/extended/document.rb +23 -20
  59. data/lib/solargraph/language_server/message/extended/document_gems.rb +3 -3
  60. data/lib/solargraph/language_server/message/initialize.rb +28 -1
  61. data/lib/solargraph/language_server/message/initialized.rb +1 -0
  62. data/lib/solargraph/language_server/message/text_document/completion.rb +56 -59
  63. data/lib/solargraph/language_server/message/text_document/definition.rb +40 -38
  64. data/lib/solargraph/language_server/message/text_document/document_symbol.rb +26 -23
  65. data/lib/solargraph/language_server/message/text_document/formatting.rb +148 -126
  66. data/lib/solargraph/language_server/message/text_document/hover.rb +58 -56
  67. data/lib/solargraph/language_server/message/text_document/signature_help.rb +24 -24
  68. data/lib/solargraph/language_server/message/text_document/type_definition.rb +25 -0
  69. data/lib/solargraph/language_server/message/text_document.rb +1 -1
  70. data/lib/solargraph/language_server/message/workspace/did_change_configuration.rb +5 -0
  71. data/lib/solargraph/language_server/message/workspace/did_change_workspace_folders.rb +2 -0
  72. data/lib/solargraph/language_server/message/workspace/workspace_symbol.rb +23 -23
  73. data/lib/solargraph/language_server/message.rb +1 -0
  74. data/lib/solargraph/language_server/progress.rb +143 -0
  75. data/lib/solargraph/language_server/request.rb +4 -1
  76. data/lib/solargraph/language_server/transport/adapter.rb +16 -1
  77. data/lib/solargraph/language_server/transport/data_reader.rb +2 -0
  78. data/lib/solargraph/language_server.rb +1 -0
  79. data/lib/solargraph/library.rb +683 -551
  80. data/lib/solargraph/location.rb +82 -37
  81. data/lib/solargraph/logging.rb +37 -27
  82. data/lib/solargraph/page.rb +9 -0
  83. data/lib/solargraph/parser/comment_ripper.rb +69 -52
  84. data/lib/solargraph/parser/flow_sensitive_typing.rb +255 -0
  85. data/lib/solargraph/parser/node_processor/base.rb +92 -77
  86. data/lib/solargraph/parser/node_processor.rb +62 -43
  87. data/lib/solargraph/parser/{legacy → parser_gem}/class_methods.rb +149 -135
  88. data/lib/solargraph/parser/{legacy → parser_gem}/flawed_builder.rb +4 -1
  89. data/lib/solargraph/parser/{legacy → parser_gem}/node_chainer.rb +166 -148
  90. data/lib/solargraph/parser/{legacy → parser_gem}/node_methods.rb +486 -325
  91. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/alias_node.rb +3 -2
  92. data/lib/solargraph/parser/parser_gem/node_processors/and_node.rb +22 -0
  93. data/lib/solargraph/parser/parser_gem/node_processors/args_node.rb +59 -0
  94. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/begin_node.rb +15 -15
  95. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/block_node.rb +46 -42
  96. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/casgn_node.rb +4 -3
  97. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/cvasgn_node.rb +3 -2
  98. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/def_node.rb +53 -63
  99. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/defs_node.rb +4 -3
  100. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/gvasgn_node.rb +3 -2
  101. data/lib/solargraph/parser/parser_gem/node_processors/if_node.rb +23 -0
  102. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/ivasgn_node.rb +40 -38
  103. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/lvasgn_node.rb +29 -28
  104. data/lib/solargraph/parser/parser_gem/node_processors/masgn_node.rb +59 -0
  105. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/namespace_node.rb +10 -9
  106. data/lib/solargraph/parser/parser_gem/node_processors/opasgn_node.rb +98 -0
  107. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/orasgn_node.rb +17 -16
  108. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/resbody_node.rb +38 -36
  109. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/sclass_node.rb +52 -42
  110. data/lib/solargraph/parser/{legacy → parser_gem}/node_processors/send_node.rb +291 -257
  111. data/lib/solargraph/parser/{rubyvm → parser_gem}/node_processors/sym_node.rb +4 -2
  112. data/lib/solargraph/parser/parser_gem/node_processors/until_node.rb +29 -0
  113. data/lib/solargraph/parser/parser_gem/node_processors/while_node.rb +29 -0
  114. data/lib/solargraph/parser/parser_gem/node_processors.rb +70 -0
  115. data/lib/solargraph/parser/parser_gem.rb +12 -0
  116. data/lib/solargraph/parser/region.rb +69 -66
  117. data/lib/solargraph/parser/snippet.rb +17 -13
  118. data/lib/solargraph/parser.rb +9 -12
  119. data/lib/solargraph/pin/base.rb +729 -299
  120. data/lib/solargraph/pin/base_variable.rb +126 -84
  121. data/lib/solargraph/pin/block.rb +104 -73
  122. data/lib/solargraph/pin/breakable.rb +9 -0
  123. data/lib/solargraph/pin/callable.rb +231 -0
  124. data/lib/solargraph/pin/closure.rb +72 -37
  125. data/lib/solargraph/pin/common.rb +79 -70
  126. data/lib/solargraph/pin/constant.rb +2 -0
  127. data/lib/solargraph/pin/conversions.rb +123 -92
  128. data/lib/solargraph/pin/delegated_method.rb +120 -0
  129. data/lib/solargraph/pin/documenting.rb +114 -105
  130. data/lib/solargraph/pin/instance_variable.rb +34 -30
  131. data/lib/solargraph/pin/keyword.rb +20 -15
  132. data/lib/solargraph/pin/local_variable.rb +75 -55
  133. data/lib/solargraph/pin/method.rb +672 -335
  134. data/lib/solargraph/pin/method_alias.rb +34 -31
  135. data/lib/solargraph/pin/namespace.rb +115 -94
  136. data/lib/solargraph/pin/parameter.rb +275 -206
  137. data/lib/solargraph/pin/proxy_type.rb +39 -29
  138. data/lib/solargraph/pin/reference/override.rb +47 -29
  139. data/lib/solargraph/pin/reference/require.rb +2 -2
  140. data/lib/solargraph/pin/reference/superclass.rb +15 -10
  141. data/lib/solargraph/pin/reference.rb +39 -14
  142. data/lib/solargraph/pin/search.rb +61 -56
  143. data/lib/solargraph/pin/signature.rb +61 -23
  144. data/lib/solargraph/pin/singleton.rb +1 -1
  145. data/lib/solargraph/pin/symbol.rb +53 -47
  146. data/lib/solargraph/pin/until.rb +18 -0
  147. data/lib/solargraph/pin/while.rb +18 -0
  148. data/lib/solargraph/pin.rb +44 -38
  149. data/lib/solargraph/pin_cache.rb +245 -0
  150. data/lib/solargraph/position.rb +132 -100
  151. data/lib/solargraph/range.rb +112 -95
  152. data/lib/solargraph/rbs_map/conversions.rb +823 -394
  153. data/lib/solargraph/rbs_map/core_fills.rb +53 -30
  154. data/lib/solargraph/rbs_map/core_map.rb +58 -38
  155. data/lib/solargraph/rbs_map/stdlib_map.rb +43 -36
  156. data/lib/solargraph/rbs_map.rb +163 -73
  157. data/lib/solargraph/shell.rb +352 -244
  158. data/lib/solargraph/source/chain/array.rb +37 -0
  159. data/lib/solargraph/source/chain/block_symbol.rb +13 -0
  160. data/lib/solargraph/source/chain/block_variable.rb +1 -1
  161. data/lib/solargraph/source/chain/call.rb +337 -215
  162. data/lib/solargraph/source/chain/constant.rb +26 -75
  163. data/lib/solargraph/source/chain/hash.rb +34 -28
  164. data/lib/solargraph/source/chain/head.rb +1 -1
  165. data/lib/solargraph/source/chain/if.rb +28 -0
  166. data/lib/solargraph/source/chain/instance_variable.rb +13 -13
  167. data/lib/solargraph/source/chain/link.rb +44 -6
  168. data/lib/solargraph/source/chain/literal.rb +48 -23
  169. data/lib/solargraph/source/chain/or.rb +23 -23
  170. data/lib/solargraph/source/chain/z_super.rb +4 -4
  171. data/lib/solargraph/source/chain.rb +291 -179
  172. data/lib/solargraph/source/change.rb +82 -79
  173. data/lib/solargraph/source/cursor.rb +166 -164
  174. data/lib/solargraph/source/encoding_fixes.rb +23 -23
  175. data/lib/solargraph/source/source_chainer.rb +194 -191
  176. data/lib/solargraph/source/updater.rb +55 -54
  177. data/lib/solargraph/source.rb +498 -522
  178. data/lib/solargraph/source_map/clip.rb +226 -229
  179. data/lib/solargraph/source_map/data.rb +34 -0
  180. data/lib/solargraph/source_map/mapper.rb +259 -243
  181. data/lib/solargraph/source_map.rb +212 -180
  182. data/lib/solargraph/type_checker/checks.rb +124 -112
  183. data/lib/solargraph/type_checker/param_def.rb +37 -35
  184. data/lib/solargraph/type_checker/problem.rb +32 -32
  185. data/lib/solargraph/type_checker/rules.rb +84 -57
  186. data/lib/solargraph/type_checker.rb +814 -549
  187. data/lib/solargraph/version.rb +5 -5
  188. data/lib/solargraph/views/_method.erb +10 -10
  189. data/lib/solargraph/views/_namespace.erb +3 -3
  190. data/lib/solargraph/views/document.erb +10 -10
  191. data/lib/solargraph/views/environment.erb +3 -5
  192. data/lib/solargraph/workspace/config.rb +255 -231
  193. data/lib/solargraph/workspace/require_paths.rb +97 -0
  194. data/lib/solargraph/workspace.rb +220 -212
  195. data/lib/solargraph/yard_map/cache.rb +6 -0
  196. data/lib/solargraph/yard_map/helpers.rb +44 -16
  197. data/lib/solargraph/yard_map/mapper/to_constant.rb +8 -5
  198. data/lib/solargraph/yard_map/mapper/to_method.rb +130 -81
  199. data/lib/solargraph/yard_map/mapper/to_namespace.rb +31 -27
  200. data/lib/solargraph/yard_map/mapper.rb +79 -77
  201. data/lib/solargraph/yard_map/to_method.rb +89 -79
  202. data/lib/solargraph/yard_map.rb +1 -284
  203. data/lib/solargraph/yard_tags.rb +20 -0
  204. data/lib/solargraph/yardoc.rb +87 -0
  205. data/lib/solargraph.rb +105 -69
  206. data/rbs/fills/bundler/0/bundler.rbs +4271 -0
  207. data/rbs/fills/open3/0/open3.rbs +172 -0
  208. data/rbs/fills/rubygems/0/basic_specification.rbs +326 -0
  209. data/rbs/fills/rubygems/0/errors.rbs +364 -0
  210. data/rbs/fills/rubygems/0/spec_fetcher.rbs +107 -0
  211. data/rbs/fills/rubygems/0/specification.rbs +1753 -0
  212. data/rbs/fills/tuple/tuple.rbs +149 -0
  213. data/rbs/shims/ast/0/node.rbs +5 -0
  214. data/rbs/shims/ast/2.4/.rbs_meta.yaml +9 -0
  215. data/rbs/shims/ast/2.4/ast.rbs +73 -0
  216. data/rbs/shims/parser/3.2.0.1/builders/default.rbs +195 -0
  217. data/rbs/shims/parser/3.2.0.1/manifest.yaml +7 -0
  218. data/rbs/shims/parser/3.2.0.1/parser.rbs +201 -0
  219. data/rbs/shims/parser/3.2.0.1/polyfill.rbs +4 -0
  220. data/rbs/shims/thor/1.2.0.1/.rbs_meta.yaml +9 -0
  221. data/rbs/shims/thor/1.2.0.1/manifest.yaml +7 -0
  222. data/rbs/shims/thor/1.2.0.1/thor.rbs +17 -0
  223. data/rbs_collection.yaml +19 -0
  224. data/solargraph.gemspec +39 -11
  225. metadata +354 -97
  226. data/lib/.rubocop.yml +0 -22
  227. data/lib/solargraph/api_map/bundler_methods.rb +0 -22
  228. data/lib/solargraph/cache.rb +0 -53
  229. data/lib/solargraph/convention/rspec.rb +0 -30
  230. data/lib/solargraph/documentor.rb +0 -76
  231. data/lib/solargraph/language_server/host/cataloger.rb +0 -56
  232. data/lib/solargraph/parser/legacy/node_processors/alias_node.rb +0 -23
  233. data/lib/solargraph/parser/legacy/node_processors/args_node.rb +0 -35
  234. data/lib/solargraph/parser/legacy/node_processors/begin_node.rb +0 -15
  235. data/lib/solargraph/parser/legacy/node_processors/sym_node.rb +0 -18
  236. data/lib/solargraph/parser/legacy/node_processors.rb +0 -54
  237. data/lib/solargraph/parser/legacy.rb +0 -12
  238. data/lib/solargraph/parser/node_methods.rb +0 -43
  239. data/lib/solargraph/parser/rubyvm/class_methods.rb +0 -149
  240. data/lib/solargraph/parser/rubyvm/node_chainer.rb +0 -160
  241. data/lib/solargraph/parser/rubyvm/node_methods.rb +0 -315
  242. data/lib/solargraph/parser/rubyvm/node_processors/args_node.rb +0 -85
  243. data/lib/solargraph/parser/rubyvm/node_processors/block_node.rb +0 -42
  244. data/lib/solargraph/parser/rubyvm/node_processors/casgn_node.rb +0 -33
  245. data/lib/solargraph/parser/rubyvm/node_processors/cvasgn_node.rb +0 -23
  246. data/lib/solargraph/parser/rubyvm/node_processors/def_node.rb +0 -75
  247. data/lib/solargraph/parser/rubyvm/node_processors/defs_node.rb +0 -68
  248. data/lib/solargraph/parser/rubyvm/node_processors/gvasgn_node.rb +0 -23
  249. data/lib/solargraph/parser/rubyvm/node_processors/ivasgn_node.rb +0 -38
  250. data/lib/solargraph/parser/rubyvm/node_processors/kw_arg_node.rb +0 -39
  251. data/lib/solargraph/parser/rubyvm/node_processors/lit_node.rb +0 -20
  252. data/lib/solargraph/parser/rubyvm/node_processors/lvasgn_node.rb +0 -27
  253. data/lib/solargraph/parser/rubyvm/node_processors/namespace_node.rb +0 -39
  254. data/lib/solargraph/parser/rubyvm/node_processors/opt_arg_node.rb +0 -26
  255. data/lib/solargraph/parser/rubyvm/node_processors/orasgn_node.rb +0 -15
  256. data/lib/solargraph/parser/rubyvm/node_processors/resbody_node.rb +0 -45
  257. data/lib/solargraph/parser/rubyvm/node_processors/sclass_node.rb +0 -32
  258. data/lib/solargraph/parser/rubyvm/node_processors/scope_node.rb +0 -15
  259. data/lib/solargraph/parser/rubyvm/node_processors/send_node.rb +0 -279
  260. data/lib/solargraph/parser/rubyvm/node_processors.rb +0 -63
  261. data/lib/solargraph/parser/rubyvm/node_wrapper.rb +0 -47
  262. data/lib/solargraph/parser/rubyvm.rb +0 -40
  263. data/lib/solargraph/rbs_map/core_signs.rb +0 -33
  264. data/lib/yard-solargraph.rb +0 -33
@@ -1,243 +1,259 @@
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: '')], []]
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
- @pins ||= []
59
- end
60
-
61
- # @param position [Solargraph::Position]
62
- # @return [Solargraph::Pin::Closure]
63
- def closure_at(position)
64
- pins.select{|pin| pin.is_a?(Pin::Closure) and pin.location.range.contain?(position)}.last
65
- end
66
-
67
- def process_comment source_position, comment_position, comment
68
- return unless comment.encode('UTF-8', invalid: :replace, replace: '?') =~ DIRECTIVE_REGEXP
69
- cmnt = remove_inline_comment_hashes(comment)
70
- parse = Solargraph::Source.parse_docstring(cmnt)
71
- last_line = 0
72
- # @param d [YARD::Tags::Directive]
73
- parse.directives.each do |d|
74
- line_num = find_directive_line_number(cmnt, d.tag.tag_name, last_line)
75
- pos = Solargraph::Position.new(comment_position.line + line_num - 1, comment_position.column)
76
- process_directive(source_position, pos, d)
77
- last_line = line_num + 1
78
- end
79
- end
80
-
81
- # @param comment [String]
82
- # @return [Integer]
83
- def find_directive_line_number comment, tag, start
84
- # Avoid overruning the index
85
- return start unless start < comment.lines.length
86
- num = comment.lines[start..-1].find_index do |line|
87
- # Legacy method directives might be `@method` instead of `@!method`
88
- # @todo Legacy syntax should probably emit a warning
89
- line.include?("@!#{tag}") || (tag == 'method' && line.include?("@#{tag}"))
90
- end
91
- num.to_i + start
92
- end
93
-
94
- # @param source_position [Position]
95
- # @param comment_position [Position]
96
- # @param directive [YARD::Tags::Directive]
97
- # @return [void]
98
- def process_directive source_position, comment_position, directive
99
- docstring = Solargraph::Source.parse_docstring(directive.tag.text).to_docstring
100
- location = Location.new(@filename, Range.new(comment_position, comment_position))
101
- case directive.tag.tag_name
102
- when 'method'
103
- namespace = closure_at(source_position) || @pins.first
104
- if namespace.location.range.start.line < comment_position.line
105
- namespace = closure_at(comment_position)
106
- end
107
- begin
108
- src = Solargraph::Source.load_string("def #{directive.tag.name};end", @source.filename)
109
- region = Parser::Region.new(source: src, closure: namespace)
110
- gen_pin = Parser.process_node(src.node, region).first.last
111
- return if gen_pin.nil?
112
- # Move the location to the end of the line so it gets recognized
113
- # as originating from a comment
114
- shifted = Solargraph::Position.new(comment_position.line, @code.lines[comment_position.line].to_s.chomp.length)
115
- # @todo: Smelly instance variable access
116
- gen_pin.instance_variable_set(:@comments, docstring.all.to_s)
117
- gen_pin.instance_variable_set(:@location, Solargraph::Location.new(@filename, Range.new(shifted, shifted)))
118
- gen_pin.instance_variable_set(:@explicit, false)
119
- @pins.push gen_pin
120
- rescue Parser::SyntaxError => e
121
- # @todo Handle error in directive
122
- end
123
- when 'attribute'
124
- return if directive.tag.name.nil?
125
- namespace = closure_at(source_position)
126
- t = (directive.tag.types.nil? || directive.tag.types.empty?) ? nil : directive.tag.types.flatten.join('')
127
- if t.nil? || t.include?('r')
128
- pins.push Solargraph::Pin::Method.new(
129
- location: location,
130
- closure: namespace,
131
- name: directive.tag.name,
132
- comments: docstring.all.to_s,
133
- scope: namespace.is_a?(Pin::Singleton) ? :class : :instance,
134
- visibility: :public,
135
- explicit: false,
136
- attribute: true
137
- )
138
- end
139
- if t.nil? || t.include?('w')
140
- pins.push Solargraph::Pin::Method.new(
141
- location: location,
142
- closure: namespace,
143
- name: "#{directive.tag.name}=",
144
- comments: docstring.all.to_s,
145
- scope: namespace.is_a?(Pin::Singleton) ? :class : :instance,
146
- visibility: :public,
147
- attribute: true
148
- )
149
- pins.last.parameters.push Pin::Parameter.new(name: 'value', decl: :arg, closure: pins.last)
150
- if pins.last.return_type.defined?
151
- pins.last.docstring.add_tag YARD::Tags::Tag.new(:param, '', pins.last.return_type.to_s.split(', '), 'value')
152
- end
153
- end
154
- when 'visibility'
155
- begin
156
- kind = directive.tag.text&.to_sym
157
- return unless [:private, :protected, :public].include?(kind)
158
-
159
- name = directive.tag.name
160
- closure = closure_at(source_position) || @pins.first
161
- if closure.location.range.start.line < comment_position.line
162
- closure = closure_at(comment_position)
163
- end
164
- if closure.is_a?(Pin::Method) && no_empty_lines?(comment_position.line, source_position.line)
165
- # @todo Smelly instance variable access
166
- closure.instance_variable_set(:@visibility, kind)
167
- else
168
- 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 }
169
- matches.each do |pin|
170
- # @todo Smelly instance variable access
171
- pin.instance_variable_set(:@visibility, kind)
172
- end
173
- end
174
- end
175
- when 'parse'
176
- begin
177
- ns = closure_at(source_position)
178
- src = Solargraph::Source.load_string(directive.tag.text, @source.filename)
179
- region = Parser::Region.new(source: src, closure: ns)
180
- # @todo These pins may need to be marked not explicit
181
- index = @pins.length
182
- loff = if @code.lines[comment_position.line].strip.end_with?('@!parse')
183
- comment_position.line + 1
184
- else
185
- comment_position.line
186
- end
187
- Parser.process_node(src.node, region, @pins)
188
- @pins[index..-1].each do |p|
189
- # @todo Smelly instance variable access
190
- p.location.range.start.instance_variable_set(:@line, p.location.range.start.line + loff)
191
- p.location.range.ending.instance_variable_set(:@line, p.location.range.ending.line + loff)
192
- end
193
- rescue Parser::SyntaxError => e
194
- # @todo Handle parser errors in !parse directives
195
- end
196
- when 'domain'
197
- namespace = closure_at(source_position) || Pin::ROOT_PIN
198
- namespace.domains.concat directive.tag.types unless directive.tag.types.nil?
199
- when 'override'
200
- pins.push Pin::Reference::Override.new(location, directive.tag.name, docstring.tags)
201
- when 'macro'
202
- # @todo Handle macros
203
- end
204
- end
205
-
206
- def no_empty_lines?(line1, line2)
207
- @code.lines[line1..line2].none? { |line| line.strip.empty? }
208
- end
209
-
210
- def remove_inline_comment_hashes comment
211
- ctxt = ''
212
- num = nil
213
- started = false
214
- comment.lines.each { |l|
215
- # Trim the comment and minimum leading whitespace
216
- p = l.encode('UTF-8', invalid: :replace, replace: '?').gsub(/^#+/, '')
217
- if num.nil? && !p.strip.empty?
218
- num = p.index(/[^ ]/)
219
- started = true
220
- elsif started && !p.strip.empty?
221
- cur = p.index(/[^ ]/)
222
- num = cur if cur < num
223
- end
224
- ctxt += "#{p[num..-1]}" if started
225
- }
226
- ctxt
227
- end
228
-
229
- # @return [void]
230
- def process_comment_directives
231
- return unless @code.encode('UTF-8', invalid: :replace, replace: '?') =~ DIRECTIVE_REGEXP
232
- code_lines = @code.lines
233
- @source.associated_comments.each do |line, comments|
234
- src_pos = line ? Position.new(line, code_lines[line].to_s.chomp.index(/[^\s]/) || 0) : Position.new(code_lines.length, 0)
235
- com_pos = Position.new(line + 1 - comments.lines.length, 0)
236
- process_comment(src_pos, com_pos, comments)
237
- end
238
- rescue StandardError => e
239
- raise e.class, "Error processing comment directives in #{@filename}: #{e.message}"
240
- end
241
- end
242
- end
243
- 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
+ @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