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,551 +1,683 @@
1
- # frozen_string_literal: true
2
-
3
- require 'pathname'
4
-
5
- module Solargraph
6
- # A Library handles coordination between a Workspace and an ApiMap.
7
- #
8
- class Library
9
- include Logging
10
-
11
- # @return [Solargraph::Workspace]
12
- attr_reader :workspace
13
-
14
- # @return [String, nil]
15
- attr_reader :name
16
-
17
- # @return [Source, nil]
18
- attr_reader :current
19
-
20
- # @param workspace [Solargraph::Workspace]
21
- # @param name [String, nil]
22
- def initialize workspace = Solargraph::Workspace.new, name = nil
23
- @workspace = workspace
24
- @name = name
25
- @synchronized = false
26
- end
27
-
28
- def inspect
29
- # Let's not deal with insane data dumps in spec failures
30
- to_s
31
- end
32
-
33
- # True if the ApiMap is up to date with the library's workspace and open
34
- # files.
35
- #
36
- # @return [Boolean]
37
- def synchronized?
38
- @synchronized
39
- end
40
-
41
- # Attach a source to the library.
42
- #
43
- # The attached source does not need to be a part of the workspace. The
44
- # library will include it in the ApiMap while it's attached. Only one
45
- # source can be attached to the library at a time.
46
- #
47
- # @param source [Source, nil]
48
- # @return [void]
49
- def attach source
50
- mutex.synchronize do
51
- if @current && (!source || @current.filename != source.filename) && source_map_hash.key?(@current.filename) && !workspace.has_file?(@current.filename)
52
- source_map_hash.delete @current.filename
53
- source_map_external_require_hash.delete @current.filename
54
- @external_requires = nil
55
- @synchronized = false
56
- end
57
- @current = source
58
- maybe_map @current
59
- catalog_inlock
60
- end
61
- end
62
-
63
- # True if the specified file is currently attached.
64
- #
65
- # @param filename [String]
66
- # @return [Boolean]
67
- def attached? filename
68
- !@current.nil? && @current.filename == filename
69
- end
70
- alias open? attached?
71
-
72
- # Detach the specified file if it is currently attached to the library.
73
- #
74
- # @param filename [String]
75
- # @return [Boolean] True if the specified file was detached
76
- def detach filename
77
- return false if @current.nil? || @current.filename != filename
78
- attach nil
79
- true
80
- end
81
-
82
- # True if the specified file is included in the workspace (but not
83
- # necessarily open).
84
- #
85
- # @param filename [String]
86
- # @return [Boolean]
87
- def contain? filename
88
- workspace.has_file?(filename)
89
- end
90
-
91
- # Create a source to be added to the workspace. The file is ignored if it is
92
- # neither open in the library nor included in the workspace.
93
- #
94
- # @param filename [String]
95
- # @param text [String] The contents of the file
96
- # @return [Boolean] True if the file was added to the workspace.
97
- def create filename, text
98
- result = false
99
- mutex.synchronize do
100
- next unless contain?(filename) || open?(filename) || workspace.would_merge?(filename)
101
- @synchronized = false
102
- source = Solargraph::Source.load_string(text, filename)
103
- workspace.merge(source)
104
- result = true
105
- end
106
- result
107
- end
108
-
109
- # Create file sources from files on disk. A file is ignored if it is
110
- # neither open in the library nor included in the workspace.
111
- #
112
- # @param filenames [Array<String>]
113
- # @return [Boolean] True if at least one file was added to the workspace.
114
- def create_from_disk *filenames
115
- result = false
116
- mutex.synchronize do
117
- sources = filenames
118
- .reject { |filename| File.directory?(filename) || !File.exist?(filename) }
119
- .map { |filename| Solargraph::Source.load_string(File.read(filename), filename) }
120
- result = workspace.merge(*sources)
121
- sources.each { |source| maybe_map source }
122
- end
123
- result
124
- end
125
-
126
- # Delete files from the library. Deleting a file will make it unavailable
127
- # for checkout and optionally remove it from the workspace unless the
128
- # workspace configuration determines that it should still exist.
129
- #
130
- # @param filenames [Array<String>]
131
- # @return [Boolean] True if any file was deleted
132
- def delete *filenames
133
- result = false
134
- filenames.each do |filename|
135
- detach filename
136
- mutex.synchronize do
137
- result ||= workspace.remove(filename)
138
- @synchronized = !result if synchronized?
139
- end
140
- end
141
- result
142
- end
143
-
144
- # Close a file in the library. Closing a file will make it unavailable for
145
- # checkout although it may still exist in the workspace.
146
- #
147
- # @param filename [String]
148
- # @return [void]
149
- def close filename
150
- mutex.synchronize do
151
- @synchronized = false
152
- @current = nil if @current && @current.filename == filename
153
- catalog
154
- end
155
- end
156
-
157
- # Get completion suggestions at the specified file and location.
158
- #
159
- # @param filename [String] The file to analyze
160
- # @param line [Integer] The zero-based line number
161
- # @param column [Integer] The zero-based column number
162
- # @return [SourceMap::Completion]
163
- # @todo Take a Location instead of filename/line/column
164
- def completions_at filename, line, column
165
- position = Position.new(line, column)
166
- cursor = Source::Cursor.new(read(filename), position)
167
- api_map.clip(cursor).complete
168
- rescue FileNotFoundError => e
169
- handle_file_not_found filename, e
170
- end
171
-
172
- # Get definition suggestions for the expression at the specified file and
173
- # location.
174
- #
175
- # @param filename [String] The file to analyze
176
- # @param line [Integer] The zero-based line number
177
- # @param column [Integer] The zero-based column number
178
- # @return [Array<Solargraph::Pin::Base>]
179
- # @todo Take filename/position instead of filename/line/column
180
- def definitions_at filename, line, column
181
- position = Position.new(line, column)
182
- cursor = Source::Cursor.new(read(filename), position)
183
- if cursor.comment?
184
- source = read(filename)
185
- offset = Solargraph::Position.to_offset(source.code, Solargraph::Position.new(line, column))
186
- lft = source.code[0..offset-1].match(/\[[a-z0-9_:<, ]*?([a-z0-9_:]*)\z/i)
187
- rgt = source.code[offset..-1].match(/^([a-z0-9_]*)(:[a-z0-9_:]*)?[\]>, ]/i)
188
- if lft && rgt
189
- tag = (lft[1] + rgt[1]).sub(/:+$/, '')
190
- clip = api_map.clip(cursor)
191
- clip.translate tag
192
- else
193
- []
194
- end
195
- else
196
- api_map.clip(cursor).define.map { |pin| pin.realize(api_map) }
197
- end
198
- rescue FileNotFoundError => e
199
- handle_file_not_found(filename, e)
200
- end
201
-
202
- # Get signature suggestions for the method at the specified file and
203
- # location.
204
- #
205
- # @param filename [String] The file to analyze
206
- # @param line [Integer] The zero-based line number
207
- # @param column [Integer] The zero-based column number
208
- # @return [Array<Solargraph::Pin::Base>]
209
- # @todo Take filename/position instead of filename/line/column
210
- def signatures_at filename, line, column
211
- position = Position.new(line, column)
212
- cursor = Source::Cursor.new(read(filename), position)
213
- api_map.clip(cursor).signify
214
- end
215
-
216
- # @param filename [String]
217
- # @param line [Integer]
218
- # @param column [Integer]
219
- # @param strip [Boolean] Strip special characters from variable names
220
- # @param only [Boolean] Search for references in the current file only
221
- # @return [Array<Solargraph::Range>]
222
- # @todo Take a Location instead of filename/line/column
223
- def references_from filename, line, column, strip: false, only: false
224
- cursor = api_map.cursor_at(filename, Position.new(line, column))
225
- clip = api_map.clip(cursor)
226
- pin = clip.define.first
227
- return [] unless pin
228
- result = []
229
- files = if only
230
- [api_map.source_map(filename)]
231
- else
232
- (workspace.sources + (@current ? [@current] : []))
233
- end
234
- files.uniq(&:filename).each do |source|
235
- found = source.references(pin.name)
236
- found.select! do |loc|
237
- referenced = definitions_at(loc.filename, loc.range.ending.line, loc.range.ending.character).first
238
- referenced && referenced.path == pin.path
239
- end
240
- # HACK: for language clients that exclude special characters from the start of variable names
241
- if strip && match = cursor.word.match(/^[^a-z0-9_]+/i)
242
- found.map! do |loc|
243
- Solargraph::Location.new(loc.filename, Solargraph::Range.from_to(loc.range.start.line, loc.range.start.column + match[0].length, loc.range.ending.line, loc.range.ending.column))
244
- end
245
- end
246
- result.concat(found.sort do |a, b|
247
- a.range.start.line <=> b.range.start.line
248
- end)
249
- end
250
- result.uniq
251
- end
252
-
253
- # Get the pins at the specified location or nil if the pin does not exist.
254
- #
255
- # @param location [Location]
256
- # @return [Array<Solargraph::Pin::Base>]
257
- def locate_pins location
258
- api_map.locate_pins(location).map { |pin| pin.realize(api_map) }
259
- end
260
-
261
- # Match a require reference to a file.
262
- #
263
- # @param location [Location]
264
- # @return [Location, nil]
265
- def locate_ref location
266
- map = source_map_hash[location.filename]
267
- return if map.nil?
268
- pin = map.requires.select { |p| p.location.range.contain?(location.range.start) }.first
269
- return nil if pin.nil?
270
- workspace.require_paths.each do |path|
271
- full = Pathname.new(path).join("#{pin.name}.rb").to_s
272
- next unless source_map_hash.key?(full)
273
- return Location.new(full, Solargraph::Range.from_to(0, 0, 0, 0))
274
- end
275
- nil
276
- rescue FileNotFoundError
277
- nil
278
- end
279
-
280
- # Get an array of pins that match a path.
281
- #
282
- # @param path [String]
283
- # @return [Array<Solargraph::Pin::Base>]
284
- def get_path_pins path
285
- api_map.get_path_suggestions(path)
286
- end
287
-
288
- # @param query [String]
289
- # @return [Array<YARD::CodeObjects::Base>]
290
- def document query
291
- api_map.document query
292
- end
293
-
294
- # @param query [String]
295
- # @return [Array<String>]
296
- def search query
297
- api_map.search query
298
- end
299
-
300
- # Get an array of all symbols in the workspace that match the query.
301
- #
302
- # @param query [String]
303
- # @return [Array<Pin::Base>]
304
- def query_symbols query
305
- api_map.query_symbols query
306
- end
307
-
308
- # Get an array of document symbols.
309
- #
310
- # Document symbols are composed of namespace, method, and constant pins.
311
- # The results of this query are appropriate for building the response to a
312
- # textDocument/documentSymbol message in the language server protocol.
313
- #
314
- # @param filename [String]
315
- # @return [Array<Solargraph::Pin::Base>]
316
- def document_symbols filename
317
- api_map.document_symbols(filename)
318
- end
319
-
320
- # @param path [String]
321
- # @return [Array<Solargraph::Pin::Base>]
322
- def path_pins path
323
- api_map.get_path_suggestions(path)
324
- end
325
-
326
- def source_maps
327
- source_map_hash.values
328
- end
329
-
330
- # Get the current text of a file in the library.
331
- #
332
- # @param filename [String]
333
- # @return [String]
334
- def read_text filename
335
- source = read(filename)
336
- source.code
337
- end
338
-
339
- # Get diagnostics about a file.
340
- #
341
- # @param filename [String]
342
- # @return [Array<Hash>]
343
- def diagnose filename
344
- # @todo Only open files get diagnosed. Determine whether anything or
345
- # everything in the workspace should get diagnosed, or if there should
346
- # be an option to do so.
347
- #
348
- return [] unless open?(filename)
349
- result = []
350
- source = read(filename)
351
- catalog
352
- repargs = {}
353
- workspace.config.reporters.each do |line|
354
- if line == 'all!'
355
- Diagnostics.reporters.each do |reporter|
356
- repargs[reporter] ||= []
357
- end
358
- else
359
- args = line.split(':').map(&:strip)
360
- name = args.shift
361
- reporter = Diagnostics.reporter(name)
362
- raise DiagnosticsError, "Diagnostics reporter #{name} does not exist" if reporter.nil?
363
- repargs[reporter] ||= []
364
- repargs[reporter].concat args
365
- end
366
- end
367
- repargs.each_pair do |reporter, args|
368
- result.concat reporter.new(*args.uniq).diagnose(source, api_map)
369
- end
370
- result
371
- end
372
-
373
- # Update the ApiMap from the library's workspace and open files.
374
- #
375
- # @return [void]
376
- def catalog
377
- mutex.synchronize do
378
- catalog_inlock
379
- end
380
- end
381
-
382
- private def catalog_inlock
383
- return if synchronized?
384
- logger.info "Cataloging #{workspace.directory.empty? ? 'generic workspace' : workspace.directory}"
385
- api_map.catalog bench
386
- @synchronized = true
387
- logger.info "Catalog complete (#{api_map.source_maps.length} files, #{api_map.pins.length} pins)" if logger.info?
388
- end
389
-
390
- def bench
391
- Bench.new(
392
- source_maps: source_map_hash.values,
393
- workspace: workspace,
394
- external_requires: external_requires
395
- )
396
- end
397
-
398
- # Get an array of foldable ranges for the specified file.
399
- #
400
- # @deprecated The library should not need to handle folding ranges. The
401
- # source itself has all the information it needs.
402
- #
403
- # @param filename [String]
404
- # @return [Array<Range>]
405
- def folding_ranges filename
406
- read(filename).folding_ranges
407
- end
408
-
409
- # Create a library from a directory.
410
- #
411
- # @param directory [String] The path to be used for the workspace
412
- # @param name [String, nil]
413
- # @return [Solargraph::Library]
414
- def self.load directory = '', name = nil
415
- Solargraph::Library.new(Solargraph::Workspace.new(directory), name)
416
- end
417
-
418
- # Try to merge a source into the library's workspace. If the workspace is
419
- # not configured to include the source, it gets ignored.
420
- #
421
- # @param source [Source]
422
- # @return [Boolean] True if the source was merged into the workspace.
423
- def merge source
424
- Logging.logger.debug "Merging source: #{source.filename}"
425
- result = false
426
- mutex.synchronize do
427
- result = workspace.merge(source)
428
- maybe_map source
429
- end
430
- # catalog
431
- result
432
- end
433
-
434
- def source_map_hash
435
- @source_map_hash ||= {}
436
- end
437
-
438
- def mapped?
439
- (workspace.filenames - source_map_hash.keys).empty?
440
- end
441
-
442
- def next_map
443
- return false if mapped?
444
- mutex.synchronize do
445
- @synchronized = false
446
- src = workspace.sources.find { |s| !source_map_hash.key?(s.filename) }
447
- if src
448
- Logging.logger.debug "Mapping #{src.filename}"
449
- source_map_hash[src.filename] = Solargraph::SourceMap.map(src)
450
- find_external_requires(source_map_hash[src.filename])
451
- source_map_hash[src.filename]
452
- else
453
- false
454
- end
455
- end
456
- end
457
-
458
- def map!
459
- workspace.sources.each do |src|
460
- source_map_hash[src.filename] = Solargraph::SourceMap.map(src)
461
- find_external_requires(source_map_hash[src.filename])
462
- end
463
- self
464
- end
465
-
466
- def pins
467
- @pins ||= []
468
- end
469
-
470
- def external_requires
471
- @external_requires ||= source_map_external_require_hash.values.flatten.to_set
472
- end
473
-
474
- private
475
-
476
- def source_map_external_require_hash
477
- @source_map_external_require_hash ||= {}
478
- end
479
-
480
- # @param source_map [SourceMap]
481
- def find_external_requires source_map
482
- new_set = source_map.requires.map(&:name).to_set
483
- # return if new_set == source_map_external_require_hash[source_map.filename]
484
- source_map_external_require_hash[source_map.filename] = new_set.reject do |path|
485
- workspace.require_paths.any? do |base|
486
- full = Pathname.new(base).join("#{path}.rb").to_s
487
- workspace.filenames.include?(full)
488
- end
489
- end
490
- @external_requires = nil
491
- end
492
-
493
- # @return [Mutex]
494
- def mutex
495
- @mutex ||= Mutex.new
496
- end
497
-
498
- # @return [ApiMap]
499
- def api_map
500
- @api_map ||= Solargraph::ApiMap.new
501
- end
502
-
503
- # Get the source for an open file or create a new source if the file
504
- # exists on disk. Sources created from disk are not added to the open
505
- # workspace files, i.e., the version on disk remains the authoritative
506
- # version.
507
- #
508
- # @raise [FileNotFoundError] if the file does not exist
509
- # @param filename [String]
510
- # @return [Solargraph::Source]
511
- def read filename
512
- return @current if @current && @current.filename == filename
513
- raise FileNotFoundError, "File not found: #{filename}" unless workspace.has_file?(filename)
514
- workspace.source(filename)
515
- end
516
-
517
- def handle_file_not_found filename, error
518
- if workspace.source(filename)
519
- Solargraph.logger.debug "#{filename} is not cataloged in the ApiMap"
520
- nil
521
- else
522
- raise error
523
- end
524
- end
525
-
526
- def maybe_map source
527
- return unless source
528
- return unless @current == source || workspace.has_file?(source.filename)
529
- if source_map_hash.key?(source.filename)
530
- return if source_map_hash[source.filename].code == source.code &&
531
- source_map_hash[source.filename].source.synchronized? &&
532
- source.synchronized?
533
- if source.synchronized?
534
- new_map = Solargraph::SourceMap.map(source)
535
- unless source_map_hash[source.filename].try_merge!(new_map)
536
- source_map_hash[source.filename] = new_map
537
- find_external_requires(source_map_hash[source.filename])
538
- @synchronized = false
539
- end
540
- else
541
- # @todo Smelly instance variable access
542
- source_map_hash[source.filename].instance_variable_set(:@source, source)
543
- end
544
- else
545
- source_map_hash[source.filename] = Solargraph::SourceMap.map(source)
546
- find_external_requires(source_map_hash[source.filename])
547
- @synchronized = false
548
- end
549
- end
550
- end
551
- end
1
+ # frozen_string_literal: true
2
+
3
+ require 'pathname'
4
+ require 'observer'
5
+ require 'open3'
6
+
7
+ module Solargraph
8
+ # A Library handles coordination between a Workspace and an ApiMap.
9
+ #
10
+ class Library
11
+ include Logging
12
+ include Observable
13
+
14
+ # @return [Solargraph::Workspace]
15
+ attr_reader :workspace
16
+
17
+ # @return [String, nil]
18
+ attr_reader :name
19
+
20
+ # @return [Source, nil]
21
+ attr_reader :current
22
+
23
+ # @return [LanguageServer::Progress, nil]
24
+ attr_reader :cache_progress
25
+
26
+ # @param workspace [Solargraph::Workspace]
27
+ # @param name [String, nil]
28
+ def initialize workspace = Solargraph::Workspace.new, name = nil
29
+ @workspace = workspace
30
+ @name = name
31
+ # @type [Integer, nil]
32
+ @total = nil
33
+ # @type [Source, nil]
34
+ @current = nil
35
+ @sync_count = 0
36
+ end
37
+
38
+ def inspect
39
+ # Let's not deal with insane data dumps in spec failures
40
+ to_s
41
+ end
42
+
43
+ # True if the ApiMap is up to date with the library's workspace and open
44
+ # files.
45
+ #
46
+ # @return [Boolean]
47
+ def synchronized?
48
+ @sync_count < 2
49
+ end
50
+
51
+ # Attach a source to the library.
52
+ #
53
+ # The attached source does not need to be a part of the workspace. The
54
+ # library will include it in the ApiMap while it's attached. Only one
55
+ # source can be attached to the library at a time.
56
+ #
57
+ # @param source [Source, nil]
58
+ # @return [void]
59
+ def attach source
60
+ if @current && (!source || @current.filename != source.filename) && source_map_hash.key?(@current.filename) && !workspace.has_file?(@current.filename)
61
+ source_map_hash.delete @current.filename
62
+ source_map_external_require_hash.delete @current.filename
63
+ @external_requires = nil
64
+ end
65
+ changed = source && @current != source
66
+ @current = source
67
+ maybe_map @current
68
+ catalog if changed
69
+ end
70
+
71
+ # True if the specified file is currently attached.
72
+ #
73
+ # @param filename [String]
74
+ # @return [Boolean]
75
+ def attached? filename
76
+ !@current.nil? && @current.filename == filename
77
+ end
78
+ alias open? attached?
79
+
80
+ # Detach the specified file if it is currently attached to the library.
81
+ #
82
+ # @param filename [String]
83
+ # @return [Boolean] True if the specified file was detached
84
+ def detach filename
85
+ return false if @current.nil? || @current.filename != filename
86
+ attach nil
87
+ true
88
+ end
89
+
90
+ # True if the specified file is included in the workspace (but not
91
+ # necessarily open).
92
+ #
93
+ # @param filename [String]
94
+ # @return [Boolean]
95
+ def contain? filename
96
+ workspace.has_file?(filename)
97
+ end
98
+
99
+ # Create a source to be added to the workspace. The file is ignored if it is
100
+ # neither open in the library nor included in the workspace.
101
+ #
102
+ # @param filename [String]
103
+ # @param text [String] The contents of the file
104
+ # @return [Boolean] True if the file was added to the workspace.
105
+ def create filename, text
106
+ return false unless contain?(filename) || open?(filename)
107
+ source = Solargraph::Source.load_string(text, filename)
108
+ workspace.merge(source)
109
+ true
110
+ end
111
+
112
+ # Create file sources from files on disk. A file is ignored if it is
113
+ # neither open in the library nor included in the workspace.
114
+ #
115
+ # @param filenames [Array<String>]
116
+ # @return [Boolean] True if at least one file was added to the workspace.
117
+ def create_from_disk *filenames
118
+ sources = filenames
119
+ .reject { |filename| File.directory?(filename) || !File.exist?(filename) }
120
+ .map { |filename| Solargraph::Source.load_string(File.read(filename), filename) }
121
+ result = workspace.merge(*sources)
122
+ sources.each { |source| maybe_map source }
123
+ result
124
+ end
125
+
126
+ # Delete files from the library. Deleting a file will make it unavailable
127
+ # for checkout and optionally remove it from the workspace unless the
128
+ # workspace configuration determines that it should still exist.
129
+ #
130
+ # @param filenames [Array<String>]
131
+ # @return [Boolean] True if any file was deleted
132
+ def delete *filenames
133
+ result = false
134
+ filenames.each do |filename|
135
+ detach filename
136
+ source_map_hash.delete(filename)
137
+ result ||= workspace.remove(filename)
138
+ end
139
+ result
140
+ end
141
+
142
+ # Close a file in the library. Closing a file will make it unavailable for
143
+ # checkout although it may still exist in the workspace.
144
+ #
145
+ # @param filename [String]
146
+ # @return [void]
147
+ def close filename
148
+ return unless @current&.filename == filename
149
+
150
+ @current = nil
151
+ catalog unless workspace.has_file?(filename)
152
+ end
153
+
154
+ # Get completion suggestions at the specified file and location.
155
+ #
156
+ # @param filename [String] The file to analyze
157
+ # @param line [Integer] The zero-based line number
158
+ # @param column [Integer] The zero-based column number
159
+ # @return [SourceMap::Completion, nil]
160
+ # @todo Take a Location instead of filename/line/column
161
+ def completions_at filename, line, column
162
+ sync_catalog
163
+ position = Position.new(line, column)
164
+ cursor = Source::Cursor.new(read(filename), position)
165
+ mutex.synchronize { api_map.clip(cursor).complete }
166
+ rescue FileNotFoundError => e
167
+ handle_file_not_found filename, e
168
+ end
169
+
170
+ # Get definition suggestions for the expression at the specified file and
171
+ # location.
172
+ #
173
+ # @param filename [String] The file to analyze
174
+ # @param line [Integer] The zero-based line number
175
+ # @param column [Integer] The zero-based column number
176
+ # @return [Array<Solargraph::Pin::Base>, nil]
177
+ # @todo Take filename/position instead of filename/line/column
178
+ def definitions_at filename, line, column
179
+ sync_catalog
180
+ position = Position.new(line, column)
181
+ cursor = Source::Cursor.new(read(filename), position)
182
+ if cursor.comment?
183
+ source = read(filename)
184
+ offset = Solargraph::Position.to_offset(source.code, Solargraph::Position.new(line, column))
185
+ lft = source.code[0..offset-1].match(/\[[a-z0-9_:<, ]*?([a-z0-9_:]*)\z/i)
186
+ rgt = source.code[offset..-1].match(/^([a-z0-9_]*)(:[a-z0-9_:]*)?[\]>, ]/i)
187
+ if lft && rgt
188
+ tag = (lft[1] + rgt[1]).sub(/:+$/, '')
189
+ clip = mutex.synchronize { api_map.clip(cursor) }
190
+ clip.translate tag
191
+ else
192
+ []
193
+ end
194
+ else
195
+ mutex.synchronize do
196
+ clip = api_map.clip(cursor)
197
+ clip.define.map { |pin| pin.realize(api_map) }
198
+ end
199
+ end
200
+ rescue FileNotFoundError => e
201
+ handle_file_not_found(filename, e)
202
+ end
203
+
204
+ # Get type definition suggestions for the expression at the specified file and
205
+ # location.
206
+ #
207
+ # @param filename [String] The file to analyze
208
+ # @param line [Integer] The zero-based line number
209
+ # @param column [Integer] The zero-based column number
210
+ # @return [Array<Solargraph::Pin::Base>, nil]
211
+ # @todo Take filename/position instead of filename/line/column
212
+ def type_definitions_at filename, line, column
213
+ sync_catalog
214
+ position = Position.new(line, column)
215
+ cursor = Source::Cursor.new(read(filename), position)
216
+ mutex.synchronize { api_map.clip(cursor).types }
217
+ rescue FileNotFoundError => e
218
+ handle_file_not_found filename, e
219
+ end
220
+
221
+ # Get signature suggestions for the method at the specified file and
222
+ # location.
223
+ #
224
+ # @param filename [String] The file to analyze
225
+ # @param line [Integer] The zero-based line number
226
+ # @param column [Integer] The zero-based column number
227
+ # @return [Array<Solargraph::Pin::Base>]
228
+ # @todo Take filename/position instead of filename/line/column
229
+ def signatures_at filename, line, column
230
+ sync_catalog
231
+ position = Position.new(line, column)
232
+ cursor = Source::Cursor.new(read(filename), position)
233
+ mutex.synchronize { api_map.clip(cursor).signify }
234
+ end
235
+
236
+ # @param filename [String]
237
+ # @param line [Integer]
238
+ # @param column [Integer]
239
+ # @param strip [Boolean] Strip special characters from variable names
240
+ # @param only [Boolean] Search for references in the current file only
241
+ # @return [Array<Solargraph::Location>]
242
+ # @todo Take a Location instead of filename/line/column
243
+ def references_from filename, line, column, strip: false, only: false
244
+ sync_catalog
245
+ cursor = Source::Cursor.new(read(filename), [line, column])
246
+ clip = mutex.synchronize { api_map.clip(cursor) }
247
+ pin = clip.define.first
248
+ return [] unless pin
249
+ result = []
250
+ files = if only
251
+ [api_map.source_map(filename)]
252
+ else
253
+ (workspace.sources + (@current ? [@current] : []))
254
+ end
255
+ files.uniq(&:filename).each do |source|
256
+ found = source.references(pin.name)
257
+ found.select! do |loc|
258
+ referenced = definitions_at(loc.filename, loc.range.ending.line, loc.range.ending.character).first
259
+ referenced&.path == pin.path
260
+ end
261
+ if pin.path == 'Class#new'
262
+ caller = cursor.chain.base.infer(api_map, clip.send(:closure), clip.locals).first
263
+ if caller.defined?
264
+ found.select! do |loc|
265
+ clip = api_map.clip_at(loc.filename, loc.range.start)
266
+ other = clip.send(:cursor).chain.base.infer(api_map, clip.send(:closure), clip.locals).first
267
+ caller == other
268
+ end
269
+ else
270
+ found.clear
271
+ end
272
+ end
273
+ # HACK: for language clients that exclude special characters from the start of variable names
274
+ if strip && match = cursor.word.match(/^[^a-z0-9_]+/i)
275
+ found.map! do |loc|
276
+ Solargraph::Location.new(loc.filename, Solargraph::Range.from_to(loc.range.start.line, loc.range.start.column + match[0].length, loc.range.ending.line, loc.range.ending.column))
277
+ end
278
+ end
279
+ result.concat(found.sort do |a, b|
280
+ a.range.start.line <=> b.range.start.line
281
+ end)
282
+ end
283
+ result.uniq
284
+ end
285
+
286
+ # Get the pins at the specified location or nil if the pin does not exist.
287
+ #
288
+ # @param location [Location]
289
+ # @return [Array<Solargraph::Pin::Base>]
290
+ def locate_pins location
291
+ sync_catalog
292
+ mutex.synchronize { api_map.locate_pins(location).map { |pin| pin.realize(api_map) } }
293
+ end
294
+
295
+ # Match a require reference to a file.
296
+ #
297
+ # @param location [Location]
298
+ # @return [Location, nil]
299
+ def locate_ref location
300
+ map = source_map_hash[location.filename]
301
+ return if map.nil?
302
+ pin = map.requires.select { |p| p.location.range.contain?(location.range.start) }.first
303
+ return nil if pin.nil?
304
+ # @param full [String]
305
+ return_if_match = proc do |full|
306
+ if source_map_hash.key?(full)
307
+ return Location.new(full, Solargraph::Range.from_to(0, 0, 0, 0))
308
+ end
309
+ end
310
+ workspace.require_paths.each do |path|
311
+ full = File.join path, pin.name
312
+ return_if_match.(full)
313
+ return_if_match.(full << ".rb")
314
+ end
315
+ nil
316
+ rescue FileNotFoundError
317
+ nil
318
+ end
319
+
320
+ # Get an array of pins that match a path.
321
+ #
322
+ # @param path [String]
323
+ # @return [Enumerable<Solargraph::Pin::Base>]
324
+ def get_path_pins path
325
+ sync_catalog
326
+ mutex.synchronize { api_map.get_path_suggestions(path) }
327
+ end
328
+
329
+ # @param query [String]
330
+ # @return [Enumerable<YARD::CodeObjects::Base>]
331
+ # @return [Array(ApiMap, Enumerable<Pin::Base>)]
332
+ def document query
333
+ sync_catalog
334
+ mutex.synchronize { [api_map, api_map.get_path_pins(query)] }
335
+ end
336
+
337
+ # @param query [String]
338
+ # @return [Array<String>]
339
+ def search query
340
+ sync_catalog
341
+ mutex.synchronize { api_map.search query }
342
+ end
343
+
344
+ # Get an array of all symbols in the workspace that match the query.
345
+ #
346
+ # @param query [String]
347
+ # @return [Array<Pin::Base>]
348
+ def query_symbols query
349
+ sync_catalog
350
+ mutex.synchronize { api_map.query_symbols query }
351
+ end
352
+
353
+ # Get an array of document symbols.
354
+ #
355
+ # Document symbols are composed of namespace, method, and constant pins.
356
+ # The results of this query are appropriate for building the response to a
357
+ # textDocument/documentSymbol message in the language server protocol.
358
+ #
359
+ # @param filename [String]
360
+ # @return [Array<Solargraph::Pin::Base>]
361
+ def document_symbols filename
362
+ sync_catalog
363
+ mutex.synchronize { api_map.document_symbols(filename) }
364
+ end
365
+
366
+ # @param path [String]
367
+ # @return [Enumerable<Solargraph::Pin::Base>]
368
+ def path_pins path
369
+ sync_catalog
370
+ mutex.synchronize { api_map.get_path_suggestions(path) }
371
+ end
372
+
373
+ # @return [Array<SourceMap>]
374
+ def source_maps
375
+ source_map_hash.values
376
+ end
377
+
378
+ # Get the current text of a file in the library.
379
+ #
380
+ # @param filename [String]
381
+ # @return [String]
382
+ def read_text filename
383
+ source = read(filename)
384
+ source.code
385
+ end
386
+
387
+ # Get diagnostics about a file.
388
+ #
389
+ # @param filename [String]
390
+ # @return [Array<Hash>]
391
+ def diagnose filename
392
+ # @todo Only open files get diagnosed. Determine whether anything or
393
+ # everything in the workspace should get diagnosed, or if there should
394
+ # be an option to do so.
395
+ #
396
+ sync_catalog
397
+ return [] unless open?(filename)
398
+ result = []
399
+ source = read(filename)
400
+
401
+ # @type [Hash{Class<Solargraph::Diagnostics::Base> => Array<String>}]
402
+ repargs = {}
403
+ workspace.config.reporters.each do |line|
404
+ if line == 'all!'
405
+ Diagnostics.reporters.each do |reporter_name|
406
+ repargs[Diagnostics.reporter(reporter_name)] ||= []
407
+ end
408
+ else
409
+ args = line.split(':').map(&:strip)
410
+ name = args.shift
411
+ reporter = Diagnostics.reporter(name)
412
+ raise DiagnosticsError, "Diagnostics reporter #{name} does not exist" if reporter.nil?
413
+ repargs[reporter] ||= []
414
+ repargs[reporter].concat args
415
+ end
416
+ end
417
+ repargs.each_pair do |reporter, args|
418
+ result.concat reporter.new(*args.uniq).diagnose(source, api_map)
419
+ end
420
+ result
421
+ end
422
+
423
+ # Update the ApiMap from the library's workspace and open files.
424
+ #
425
+ # @return [void]
426
+ def catalog
427
+ @sync_count += 1
428
+ end
429
+
430
+ # @return [Bench]
431
+ def bench
432
+ Bench.new(
433
+ source_maps: source_map_hash.values,
434
+ workspace: workspace,
435
+ external_requires: external_requires,
436
+ live_map: @current ? source_map_hash[@current.filename] : nil
437
+ )
438
+ end
439
+
440
+ # Create a library from a directory.
441
+ #
442
+ # @param directory [String] The path to be used for the workspace
443
+ # @param name [String, nil]
444
+ # @return [Solargraph::Library]
445
+ def self.load directory = '', name = nil
446
+ Solargraph::Library.new(Solargraph::Workspace.new(directory), name)
447
+ end
448
+
449
+ # Try to merge a source into the library's workspace. If the workspace is
450
+ # not configured to include the source, it gets ignored.
451
+ #
452
+ # @param source [Source]
453
+ # @return [Boolean] True if the source was merged into the workspace.
454
+ def merge source
455
+ result = workspace.merge(source)
456
+ maybe_map source
457
+ result
458
+ end
459
+
460
+ # @return [Hash{String => SourceMap}]
461
+ def source_map_hash
462
+ @source_map_hash ||= {}
463
+ end
464
+
465
+ def mapped?
466
+ (workspace.filenames - source_map_hash.keys).empty?
467
+ end
468
+
469
+ # @return [SourceMap, Boolean]
470
+ def next_map
471
+ return false if mapped?
472
+ src = workspace.sources.find { |s| !source_map_hash.key?(s.filename) }
473
+ if src
474
+ Logging.logger.debug "Mapping #{src.filename}"
475
+ source_map_hash[src.filename] = Solargraph::SourceMap.map(src)
476
+ source_map_hash[src.filename]
477
+ else
478
+ false
479
+ end
480
+ end
481
+
482
+ # @return [self]
483
+ def map!
484
+ workspace.sources.each do |src|
485
+ source_map_hash[src.filename] = Solargraph::SourceMap.map(src)
486
+ find_external_requires source_map_hash[src.filename]
487
+ end
488
+ self
489
+ end
490
+
491
+ # @return [Array<Solargraph::Pin::Base>]
492
+ def pins
493
+ @pins ||= []
494
+ end
495
+
496
+ # @return [Set<String>]
497
+ def external_requires
498
+ @external_requires ||= source_map_external_require_hash.values.flatten.to_set
499
+ end
500
+
501
+ private
502
+
503
+ # @return [Hash{String => Array<String>}]
504
+ def source_map_external_require_hash
505
+ @source_map_external_require_hash ||= {}
506
+ end
507
+
508
+ # @param source_map [SourceMap]
509
+ # @return [void]
510
+ def find_external_requires source_map
511
+ # @type [Set<String>]
512
+ new_set = source_map.requires.map(&:name).to_set
513
+ # return if new_set == source_map_external_require_hash[source_map.filename]
514
+ _filenames = nil
515
+ filenames = ->{ _filenames ||= workspace.filenames.to_set }
516
+ source_map_external_require_hash[source_map.filename] = new_set.reject do |path|
517
+ workspace.require_paths.any? do |base|
518
+ full = File.join(base, path)
519
+ filenames[].include?(full) or filenames[].include?(full << ".rb")
520
+ end
521
+ end
522
+ @external_requires = nil
523
+ end
524
+
525
+ # @return [Thread::Mutex]
526
+ def mutex
527
+ @mutex ||= Mutex.new
528
+ end
529
+
530
+ # @return [ApiMap]
531
+ def api_map
532
+ @api_map ||= Solargraph::ApiMap.new
533
+ end
534
+
535
+ # Get the source for an open file or create a new source if the file
536
+ # exists on disk. Sources created from disk are not added to the open
537
+ # workspace files, i.e., the version on disk remains the authoritative
538
+ # version.
539
+ #
540
+ # @raise [FileNotFoundError] if the file does not exist
541
+ # @param filename [String]
542
+ # @return [Solargraph::Source]
543
+ def read filename
544
+ return @current if @current && @current.filename == filename
545
+ raise FileNotFoundError, "File not found: #{filename}" unless workspace.has_file?(filename)
546
+ workspace.source(filename)
547
+ end
548
+
549
+ # @param filename [String]
550
+ # @param error [FileNotFoundError]
551
+ # @return [nil]
552
+ def handle_file_not_found filename, error
553
+ if workspace.source(filename)
554
+ Solargraph.logger.debug "#{filename} is not cataloged in the ApiMap"
555
+ nil
556
+ else
557
+ raise error
558
+ end
559
+ end
560
+
561
+ # @param source [Source, nil]
562
+ # @return [void]
563
+ def maybe_map source
564
+ return unless source
565
+ return unless @current == source || workspace.has_file?(source.filename)
566
+ if source_map_hash.key?(source.filename)
567
+ new_map = Solargraph::SourceMap.map(source)
568
+ source_map_hash[source.filename] = new_map
569
+ else
570
+ source_map_hash[source.filename] = Solargraph::SourceMap.map(source)
571
+ end
572
+ end
573
+
574
+ # @return [Set<Gem::Specification>]
575
+ def cache_errors
576
+ @cache_errors ||= Set.new
577
+ end
578
+
579
+ # @return [void]
580
+ def cache_next_gemspec
581
+ return if @cache_progress
582
+
583
+ spec = cacheable_specs.first
584
+ return end_cache_progress unless spec
585
+
586
+ pending = api_map.uncached_gemspecs.length - cache_errors.length - 1
587
+
588
+ if Yardoc.processing?(spec)
589
+ logger.info "Enqueuing cache of #{spec.name} #{spec.version} (already being processed)"
590
+ queued_gemspec_cache.push(spec)
591
+ return if pending - queued_gemspec_cache.length < 1
592
+
593
+ catalog
594
+ sync_catalog
595
+ else
596
+ logger.info "Caching #{spec.name} #{spec.version}"
597
+ Thread.new do
598
+ report_cache_progress spec.name, pending
599
+ _o, e, s = Open3.capture3(workspace.command_path, 'cache', spec.name, spec.version.to_s)
600
+ if s.success?
601
+ logger.info "Cached #{spec.name} #{spec.version}"
602
+ else
603
+ cache_errors.add spec
604
+ logger.warn "Error caching gemspec #{spec.name} #{spec.version}"
605
+ logger.warn e
606
+ end
607
+ end_cache_progress
608
+ catalog
609
+ sync_catalog
610
+ end
611
+ end
612
+ end
613
+
614
+ # @return [Array<Gem::Specification>]
615
+ def cacheable_specs
616
+ cacheable = api_map.uncached_yard_gemspecs +
617
+ api_map.uncached_rbs_collection_gemspecs -
618
+ queued_gemspec_cache -
619
+ cache_errors.to_a
620
+ return cacheable unless cacheable.empty?
621
+
622
+ queued_gemspec_cache
623
+ end
624
+
625
+ # @return [Array<Gem::Specification>]
626
+ def queued_gemspec_cache
627
+ @queued_gemspec_cache ||= []
628
+ end
629
+
630
+ # @param gem_name [String]
631
+ # @param pending [Integer]
632
+ # @return [void]
633
+ def report_cache_progress gem_name, pending
634
+ @total ||= pending
635
+ @total = pending if pending > @total
636
+ finished = @total - pending
637
+ pct = if @total.zero?
638
+ 0
639
+ else
640
+ ((finished.to_f / @total.to_f) * 100).to_i
641
+ end
642
+ message = "#{gem_name}#{pending > 0 ? " (+#{pending})" : ''}"
643
+ # "
644
+ if @cache_progress
645
+ @cache_progress.report(message, pct)
646
+ else
647
+ @cache_progress = LanguageServer::Progress.new('Caching gem')
648
+ # If we don't send both a begin and a report, the progress notification
649
+ # might get stuck in the status bar forever
650
+ @cache_progress.begin(message, pct)
651
+ changed
652
+ notify_observers @cache_progress
653
+ @cache_progress.report(message, pct)
654
+ end
655
+ changed
656
+ notify_observers @cache_progress
657
+ end
658
+
659
+ # @return [void]
660
+ def end_cache_progress
661
+ changed if @cache_progress&.finish('done')
662
+ notify_observers @cache_progress
663
+ @cache_progress = nil
664
+ @total = nil
665
+ end
666
+
667
+ # @return [void]
668
+ def sync_catalog
669
+ return if @sync_count == 0
670
+
671
+ mutex.synchronize do
672
+ logger.info "Cataloging #{workspace.directory.empty? ? 'generic workspace' : workspace.directory}"
673
+ source_map_hash.values.each { |map| find_external_requires(map) }
674
+ api_map.catalog bench
675
+ logger.info "Catalog complete (#{api_map.source_maps.length} files, #{api_map.pins.length} pins)"
676
+ logger.info "#{api_map.uncached_yard_gemspecs.length} uncached YARD gemspecs"
677
+ logger.info "#{api_map.uncached_rbs_collection_gemspecs.length} uncached RBS collection gemspecs"
678
+ cache_next_gemspec
679
+ @sync_count = 0
680
+ end
681
+ end
682
+ end
683
+ end