prism 1.2.0 → 1.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (68) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +46 -1
  3. data/Makefile +1 -1
  4. data/config.yml +429 -2
  5. data/docs/build_system.md +8 -11
  6. data/docs/releasing.md +1 -1
  7. data/docs/relocation.md +34 -0
  8. data/docs/ruby_api.md +1 -1
  9. data/ext/prism/api_node.c +1824 -1305
  10. data/ext/prism/extconf.rb +13 -36
  11. data/ext/prism/extension.c +298 -109
  12. data/ext/prism/extension.h +4 -4
  13. data/include/prism/ast.h +442 -2
  14. data/include/prism/defines.h +26 -8
  15. data/include/prism/options.h +47 -1
  16. data/include/prism/util/pm_buffer.h +10 -0
  17. data/include/prism/version.h +2 -2
  18. data/include/prism.h +51 -4
  19. data/lib/prism/dot_visitor.rb +26 -0
  20. data/lib/prism/dsl.rb +14 -6
  21. data/lib/prism/ffi.rb +93 -28
  22. data/lib/prism/inspect_visitor.rb +4 -1
  23. data/lib/prism/node.rb +1886 -105
  24. data/lib/prism/parse_result/errors.rb +1 -1
  25. data/lib/prism/parse_result/newlines.rb +1 -1
  26. data/lib/prism/parse_result.rb +54 -2
  27. data/lib/prism/polyfill/append_as_bytes.rb +15 -0
  28. data/lib/prism/reflection.rb +4 -4
  29. data/lib/prism/relocation.rb +504 -0
  30. data/lib/prism/serialize.rb +1252 -765
  31. data/lib/prism/string_query.rb +30 -0
  32. data/lib/prism/translation/parser/builder.rb +61 -0
  33. data/lib/prism/translation/parser/compiler.rb +228 -162
  34. data/lib/prism/translation/parser/lexer.rb +435 -61
  35. data/lib/prism/translation/parser.rb +51 -3
  36. data/lib/prism/translation/parser35.rb +12 -0
  37. data/lib/prism/translation/ripper.rb +13 -3
  38. data/lib/prism/translation/ruby_parser.rb +17 -7
  39. data/lib/prism/translation.rb +1 -0
  40. data/lib/prism.rb +9 -7
  41. data/prism.gemspec +11 -1
  42. data/rbi/prism/dsl.rbi +10 -7
  43. data/rbi/prism/node.rbi +44 -17
  44. data/rbi/prism/parse_result.rbi +17 -0
  45. data/rbi/prism/string_query.rbi +12 -0
  46. data/rbi/prism/translation/parser35.rbi +6 -0
  47. data/rbi/prism.rbi +39 -36
  48. data/sig/prism/dsl.rbs +6 -4
  49. data/sig/prism/node.rbs +29 -15
  50. data/sig/prism/parse_result.rbs +10 -0
  51. data/sig/prism/relocation.rbs +185 -0
  52. data/sig/prism/serialize.rbs +4 -2
  53. data/sig/prism/string_query.rbs +11 -0
  54. data/sig/prism.rbs +22 -1
  55. data/src/diagnostic.c +2 -2
  56. data/src/node.c +39 -0
  57. data/src/options.c +31 -0
  58. data/src/prettyprint.c +62 -0
  59. data/src/prism.c +738 -199
  60. data/src/regexp.c +7 -3
  61. data/src/serialize.c +18 -0
  62. data/src/static_literals.c +1 -1
  63. data/src/util/pm_buffer.c +40 -0
  64. data/src/util/pm_char.c +1 -1
  65. data/src/util/pm_constant_pool.c +6 -2
  66. data/src/util/pm_string.c +1 -0
  67. data/src/util/pm_strncasecmp.c +13 -1
  68. metadata +13 -7
@@ -17,7 +17,7 @@ module Prism
17
17
 
18
18
  # Formats the errors in a human-readable way and return them as a string.
19
19
  def format
20
- error_lines = {}
20
+ error_lines = {} #: Hash[Integer, Array[ParseError]]
21
21
  parse_result.errors.each do |error|
22
22
  location = error.location
23
23
  (location.start_line..location.end_line).each do |line|
@@ -63,7 +63,7 @@ module Prism
63
63
 
64
64
  class Node
65
65
  def newline_flag? # :nodoc:
66
- @newline_flag ? true : false
66
+ !!defined?(@newline_flag)
67
67
  end
68
68
 
69
69
  def newline_flag!(lines) # :nodoc:
@@ -48,6 +48,16 @@ module Prism
48
48
  @offsets = offsets # set after parsing is done
49
49
  end
50
50
 
51
+ # Replace the value of start_line with the given value.
52
+ def replace_start_line(start_line)
53
+ @start_line = start_line
54
+ end
55
+
56
+ # Replace the value of offsets with the given value.
57
+ def replace_offsets(offsets)
58
+ @offsets.replace(offsets)
59
+ end
60
+
51
61
  # Returns the encoding of the source code, which is set by parameters to the
52
62
  # parser or by the encoding magic comment.
53
63
  def encoding
@@ -132,6 +142,13 @@ module Prism
132
142
  code_units_offset(byte_offset, encoding) - code_units_offset(line_start(byte_offset), encoding)
133
143
  end
134
144
 
145
+ # Freeze this object and the objects it contains.
146
+ def deep_freeze
147
+ source.freeze
148
+ offsets.freeze
149
+ freeze
150
+ end
151
+
135
152
  private
136
153
 
137
154
  # Binary search through the offsets to find the line number for the given
@@ -204,8 +221,8 @@ module Prism
204
221
  LengthCounter.new(source, encoding)
205
222
  end
206
223
 
207
- @cache = {}
208
- @offsets = []
224
+ @cache = {} #: Hash[Integer, Integer]
225
+ @offsets = [] #: Array[Integer]
209
226
  end
210
227
 
211
228
  # Retrieve the code units offset from the given byte offset.
@@ -854,5 +871,40 @@ module Prism
854
871
  location
855
872
  super
856
873
  end
874
+
875
+ # Freeze this object and the objects it contains.
876
+ def deep_freeze
877
+ value.freeze
878
+ location.freeze
879
+ freeze
880
+ end
881
+ end
882
+
883
+ # This object is passed to the various Prism.* methods that accept the
884
+ # `scopes` option as an element of the list. It defines both the local
885
+ # variables visible at that scope as well as the forwarding parameters
886
+ # available at that scope.
887
+ class Scope
888
+ # The list of local variables that are defined in this scope. This should be
889
+ # defined as an array of symbols.
890
+ attr_reader :locals
891
+
892
+ # The list of local variables that are forwarded to the next scope. This
893
+ # should by defined as an array of symbols containing the specific values of
894
+ # :*, :**, :&, or :"...".
895
+ attr_reader :forwarding
896
+
897
+ # Create a new scope object with the given locals and forwarding.
898
+ def initialize(locals, forwarding)
899
+ @locals = locals
900
+ @forwarding = forwarding
901
+ end
902
+ end
903
+
904
+ # Create a new scope with the given locals and forwarding options that is
905
+ # suitable for passing into one of the Prism.* methods that accepts the
906
+ # `scopes` option.
907
+ def self.scope(locals: [], forwarding: [])
908
+ Scope.new(locals, forwarding)
857
909
  end
858
910
  end
@@ -0,0 +1,15 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Polyfill for String#append_as_bytes, which didn't exist until Ruby 3.4.
4
+ if !("".respond_to?(:append_as_bytes))
5
+ String.include(
6
+ Module.new {
7
+ def append_as_bytes(*args)
8
+ args.each do |arg|
9
+ arg = Integer === arg ? [arg].pack("C") : arg.b
10
+ self.<<(arg) # steep:ignore
11
+ end
12
+ end
13
+ }
14
+ )
15
+ end
@@ -334,7 +334,7 @@ module Prism
334
334
  when :parameters_node
335
335
  [NodeListField.new(:requireds), NodeListField.new(:optionals), OptionalNodeField.new(:rest), NodeListField.new(:posts), NodeListField.new(:keywords), OptionalNodeField.new(:keyword_rest), OptionalNodeField.new(:block)]
336
336
  when :parentheses_node
337
- [OptionalNodeField.new(:body), LocationField.new(:opening_loc), LocationField.new(:closing_loc)]
337
+ [FlagsField.new(:flags, [:multiple_statements?]), OptionalNodeField.new(:body), LocationField.new(:opening_loc), LocationField.new(:closing_loc)]
338
338
  when :pinned_expression_node
339
339
  [NodeField.new(:expression), LocationField.new(:operator_loc), LocationField.new(:lparen_loc), LocationField.new(:rparen_loc)]
340
340
  when :pinned_variable_node
@@ -360,7 +360,7 @@ module Prism
360
360
  when :rescue_modifier_node
361
361
  [NodeField.new(:expression), LocationField.new(:keyword_loc), NodeField.new(:rescue_expression)]
362
362
  when :rescue_node
363
- [LocationField.new(:keyword_loc), NodeListField.new(:exceptions), OptionalLocationField.new(:operator_loc), OptionalNodeField.new(:reference), OptionalNodeField.new(:statements), OptionalNodeField.new(:subsequent)]
363
+ [LocationField.new(:keyword_loc), NodeListField.new(:exceptions), OptionalLocationField.new(:operator_loc), OptionalNodeField.new(:reference), OptionalLocationField.new(:then_keyword_loc), OptionalNodeField.new(:statements), OptionalNodeField.new(:subsequent)]
364
364
  when :rest_parameter_node
365
365
  [FlagsField.new(:flags, [:repeated_parameter?]), OptionalConstantField.new(:name), OptionalLocationField.new(:name_loc), LocationField.new(:operator_loc)]
366
366
  when :retry_node
@@ -396,11 +396,11 @@ module Prism
396
396
  when :unless_node
397
397
  [LocationField.new(:keyword_loc), NodeField.new(:predicate), OptionalLocationField.new(:then_keyword_loc), OptionalNodeField.new(:statements), OptionalNodeField.new(:else_clause), OptionalLocationField.new(:end_keyword_loc)]
398
398
  when :until_node
399
- [FlagsField.new(:flags, [:begin_modifier?]), LocationField.new(:keyword_loc), OptionalLocationField.new(:closing_loc), NodeField.new(:predicate), OptionalNodeField.new(:statements)]
399
+ [FlagsField.new(:flags, [:begin_modifier?]), LocationField.new(:keyword_loc), OptionalLocationField.new(:do_keyword_loc), OptionalLocationField.new(:closing_loc), NodeField.new(:predicate), OptionalNodeField.new(:statements)]
400
400
  when :when_node
401
401
  [LocationField.new(:keyword_loc), NodeListField.new(:conditions), OptionalLocationField.new(:then_keyword_loc), OptionalNodeField.new(:statements)]
402
402
  when :while_node
403
- [FlagsField.new(:flags, [:begin_modifier?]), LocationField.new(:keyword_loc), OptionalLocationField.new(:closing_loc), NodeField.new(:predicate), OptionalNodeField.new(:statements)]
403
+ [FlagsField.new(:flags, [:begin_modifier?]), LocationField.new(:keyword_loc), OptionalLocationField.new(:do_keyword_loc), OptionalLocationField.new(:closing_loc), NodeField.new(:predicate), OptionalNodeField.new(:statements)]
404
404
  when :x_string_node
405
405
  [FlagsField.new(:flags, [:forced_utf8_encoding?, :forced_binary_encoding?]), LocationField.new(:opening_loc), LocationField.new(:content_loc), LocationField.new(:closing_loc), StringField.new(:unescaped)]
406
406
  when :yield_node
@@ -0,0 +1,504 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Prism
4
+ # Prism parses deterministically for the same input. This provides a nice
5
+ # property that is exposed through the #node_id API on nodes. Effectively this
6
+ # means that for the same input, these values will remain consistent every
7
+ # time the source is parsed. This means we can reparse the source same with a
8
+ # #node_id value and find the exact same node again.
9
+ #
10
+ # The Relocation module provides an API around this property. It allows you to
11
+ # "save" nodes and locations using a minimal amount of memory (just the
12
+ # node_id and a field identifier) and then reify them later.
13
+ module Relocation
14
+ # An entry in a repository that will lazily reify its values when they are
15
+ # first accessed.
16
+ class Entry
17
+ # Raised if a value that could potentially be on an entry is missing
18
+ # because it was either not configured on the repository or it has not yet
19
+ # been fetched.
20
+ class MissingValueError < StandardError
21
+ end
22
+
23
+ # Initialize a new entry with the given repository.
24
+ def initialize(repository)
25
+ @repository = repository
26
+ @values = nil
27
+ end
28
+
29
+ # Fetch the filepath of the value.
30
+ def filepath
31
+ fetch_value(:filepath)
32
+ end
33
+
34
+ # Fetch the start line of the value.
35
+ def start_line
36
+ fetch_value(:start_line)
37
+ end
38
+
39
+ # Fetch the end line of the value.
40
+ def end_line
41
+ fetch_value(:end_line)
42
+ end
43
+
44
+ # Fetch the start byte offset of the value.
45
+ def start_offset
46
+ fetch_value(:start_offset)
47
+ end
48
+
49
+ # Fetch the end byte offset of the value.
50
+ def end_offset
51
+ fetch_value(:end_offset)
52
+ end
53
+
54
+ # Fetch the start character offset of the value.
55
+ def start_character_offset
56
+ fetch_value(:start_character_offset)
57
+ end
58
+
59
+ # Fetch the end character offset of the value.
60
+ def end_character_offset
61
+ fetch_value(:end_character_offset)
62
+ end
63
+
64
+ # Fetch the start code units offset of the value, for the encoding that
65
+ # was configured on the repository.
66
+ def start_code_units_offset
67
+ fetch_value(:start_code_units_offset)
68
+ end
69
+
70
+ # Fetch the end code units offset of the value, for the encoding that was
71
+ # configured on the repository.
72
+ def end_code_units_offset
73
+ fetch_value(:end_code_units_offset)
74
+ end
75
+
76
+ # Fetch the start byte column of the value.
77
+ def start_column
78
+ fetch_value(:start_column)
79
+ end
80
+
81
+ # Fetch the end byte column of the value.
82
+ def end_column
83
+ fetch_value(:end_column)
84
+ end
85
+
86
+ # Fetch the start character column of the value.
87
+ def start_character_column
88
+ fetch_value(:start_character_column)
89
+ end
90
+
91
+ # Fetch the end character column of the value.
92
+ def end_character_column
93
+ fetch_value(:end_character_column)
94
+ end
95
+
96
+ # Fetch the start code units column of the value, for the encoding that
97
+ # was configured on the repository.
98
+ def start_code_units_column
99
+ fetch_value(:start_code_units_column)
100
+ end
101
+
102
+ # Fetch the end code units column of the value, for the encoding that was
103
+ # configured on the repository.
104
+ def end_code_units_column
105
+ fetch_value(:end_code_units_column)
106
+ end
107
+
108
+ # Fetch the leading comments of the value.
109
+ def leading_comments
110
+ fetch_value(:leading_comments)
111
+ end
112
+
113
+ # Fetch the trailing comments of the value.
114
+ def trailing_comments
115
+ fetch_value(:trailing_comments)
116
+ end
117
+
118
+ # Fetch the leading and trailing comments of the value.
119
+ def comments
120
+ leading_comments.concat(trailing_comments)
121
+ end
122
+
123
+ # Reify the values on this entry with the given values. This is an
124
+ # internal-only API that is called from the repository when it is time to
125
+ # reify the values.
126
+ def reify!(values) # :nodoc:
127
+ @repository = nil
128
+ @values = values
129
+ end
130
+
131
+ private
132
+
133
+ # Fetch a value from the entry, raising an error if it is missing.
134
+ def fetch_value(name)
135
+ values.fetch(name) do
136
+ raise MissingValueError, "No value for #{name}, make sure the " \
137
+ "repository has been properly configured"
138
+ end
139
+ end
140
+
141
+ # Return the values from the repository, reifying them if necessary.
142
+ def values
143
+ @values || (@repository.reify!; @values)
144
+ end
145
+ end
146
+
147
+ # Represents the source of a repository that will be reparsed.
148
+ class Source
149
+ # The value that will need to be reparsed.
150
+ attr_reader :value
151
+
152
+ # Initialize the source with the given value.
153
+ def initialize(value)
154
+ @value = value
155
+ end
156
+
157
+ # Reparse the value and return the parse result.
158
+ def result
159
+ raise NotImplementedError, "Subclasses must implement #result"
160
+ end
161
+
162
+ # Create a code units cache for the given encoding.
163
+ def code_units_cache(encoding)
164
+ result.code_units_cache(encoding)
165
+ end
166
+ end
167
+
168
+ # A source that is represented by a file path.
169
+ class SourceFilepath < Source
170
+ # Reparse the file and return the parse result.
171
+ def result
172
+ Prism.parse_file(value)
173
+ end
174
+ end
175
+
176
+ # A source that is represented by a string.
177
+ class SourceString < Source
178
+ # Reparse the string and return the parse result.
179
+ def result
180
+ Prism.parse(value)
181
+ end
182
+ end
183
+
184
+ # A field that represents the file path.
185
+ class FilepathField
186
+ # The file path that this field represents.
187
+ attr_reader :value
188
+
189
+ # Initialize a new field with the given file path.
190
+ def initialize(value)
191
+ @value = value
192
+ end
193
+
194
+ # Fetch the file path.
195
+ def fields(_value)
196
+ { filepath: value }
197
+ end
198
+ end
199
+
200
+ # A field representing the start and end lines.
201
+ class LinesField
202
+ # Fetches the start and end line of a value.
203
+ def fields(value)
204
+ { start_line: value.start_line, end_line: value.end_line }
205
+ end
206
+ end
207
+
208
+ # A field representing the start and end byte offsets.
209
+ class OffsetsField
210
+ # Fetches the start and end byte offset of a value.
211
+ def fields(value)
212
+ { start_offset: value.start_offset, end_offset: value.end_offset }
213
+ end
214
+ end
215
+
216
+ # A field representing the start and end character offsets.
217
+ class CharacterOffsetsField
218
+ # Fetches the start and end character offset of a value.
219
+ def fields(value)
220
+ {
221
+ start_character_offset: value.start_character_offset,
222
+ end_character_offset: value.end_character_offset
223
+ }
224
+ end
225
+ end
226
+
227
+ # A field representing the start and end code unit offsets.
228
+ class CodeUnitOffsetsField
229
+ # A pointer to the repository object that is used for lazily creating a
230
+ # code units cache.
231
+ attr_reader :repository
232
+
233
+ # The associated encoding for the code units.
234
+ attr_reader :encoding
235
+
236
+ # Initialize a new field with the associated repository and encoding.
237
+ def initialize(repository, encoding)
238
+ @repository = repository
239
+ @encoding = encoding
240
+ @cache = nil
241
+ end
242
+
243
+ # Fetches the start and end code units offset of a value for a particular
244
+ # encoding.
245
+ def fields(value)
246
+ {
247
+ start_code_units_offset: value.cached_start_code_units_offset(cache),
248
+ end_code_units_offset: value.cached_end_code_units_offset(cache)
249
+ }
250
+ end
251
+
252
+ private
253
+
254
+ # Lazily create a code units cache for the associated encoding.
255
+ def cache
256
+ @cache ||= repository.code_units_cache(encoding)
257
+ end
258
+ end
259
+
260
+ # A field representing the start and end byte columns.
261
+ class ColumnsField
262
+ # Fetches the start and end byte column of a value.
263
+ def fields(value)
264
+ { start_column: value.start_column, end_column: value.end_column }
265
+ end
266
+ end
267
+
268
+ # A field representing the start and end character columns.
269
+ class CharacterColumnsField
270
+ # Fetches the start and end character column of a value.
271
+ def fields(value)
272
+ {
273
+ start_character_column: value.start_character_column,
274
+ end_character_column: value.end_character_column
275
+ }
276
+ end
277
+ end
278
+
279
+ # A field representing the start and end code unit columns for a specific
280
+ # encoding.
281
+ class CodeUnitColumnsField
282
+ # The repository object that is used for lazily creating a code units
283
+ # cache.
284
+ attr_reader :repository
285
+
286
+ # The associated encoding for the code units.
287
+ attr_reader :encoding
288
+
289
+ # Initialize a new field with the associated repository and encoding.
290
+ def initialize(repository, encoding)
291
+ @repository = repository
292
+ @encoding = encoding
293
+ @cache = nil
294
+ end
295
+
296
+ # Fetches the start and end code units column of a value for a particular
297
+ # encoding.
298
+ def fields(value)
299
+ {
300
+ start_code_units_column: value.cached_start_code_units_column(cache),
301
+ end_code_units_column: value.cached_end_code_units_column(cache)
302
+ }
303
+ end
304
+
305
+ private
306
+
307
+ # Lazily create a code units cache for the associated encoding.
308
+ def cache
309
+ @cache ||= repository.code_units_cache(encoding)
310
+ end
311
+ end
312
+
313
+ # An abstract field used as the parent class of the two comments fields.
314
+ class CommentsField
315
+ # An object that represents a slice of a comment.
316
+ class Comment
317
+ # The slice of the comment.
318
+ attr_reader :slice
319
+
320
+ # Initialize a new comment with the given slice.
321
+ def initialize(slice)
322
+ @slice = slice
323
+ end
324
+ end
325
+
326
+ private
327
+
328
+ # Create comment objects from the given values.
329
+ def comments(values)
330
+ values.map { |value| Comment.new(value.slice) }
331
+ end
332
+ end
333
+
334
+ # A field representing the leading comments.
335
+ class LeadingCommentsField < CommentsField
336
+ # Fetches the leading comments of a value.
337
+ def fields(value)
338
+ { leading_comments: comments(value.leading_comments) }
339
+ end
340
+ end
341
+
342
+ # A field representing the trailing comments.
343
+ class TrailingCommentsField < CommentsField
344
+ # Fetches the trailing comments of a value.
345
+ def fields(value)
346
+ { trailing_comments: comments(value.trailing_comments) }
347
+ end
348
+ end
349
+
350
+ # A repository is a configured collection of fields and a set of entries
351
+ # that knows how to reparse a source and reify the values.
352
+ class Repository
353
+ # Raised when multiple fields of the same type are configured on the same
354
+ # repository.
355
+ class ConfigurationError < StandardError
356
+ end
357
+
358
+ # The source associated with this repository. This will be either a
359
+ # SourceFilepath (the most common use case) or a SourceString.
360
+ attr_reader :source
361
+
362
+ # The fields that have been configured on this repository.
363
+ attr_reader :fields
364
+
365
+ # The entries that have been saved on this repository.
366
+ attr_reader :entries
367
+
368
+ # Initialize a new repository with the given source.
369
+ def initialize(source)
370
+ @source = source
371
+ @fields = {}
372
+ @entries = Hash.new { |hash, node_id| hash[node_id] = {} }
373
+ end
374
+
375
+ # Create a code units cache for the given encoding from the source.
376
+ def code_units_cache(encoding)
377
+ source.code_units_cache(encoding)
378
+ end
379
+
380
+ # Configure the filepath field for this repository and return self.
381
+ def filepath
382
+ raise ConfigurationError, "Can only specify filepath for a filepath source" unless source.is_a?(SourceFilepath)
383
+ field(:filepath, FilepathField.new(source.value))
384
+ end
385
+
386
+ # Configure the lines field for this repository and return self.
387
+ def lines
388
+ field(:lines, LinesField.new)
389
+ end
390
+
391
+ # Configure the offsets field for this repository and return self.
392
+ def offsets
393
+ field(:offsets, OffsetsField.new)
394
+ end
395
+
396
+ # Configure the character offsets field for this repository and return
397
+ # self.
398
+ def character_offsets
399
+ field(:character_offsets, CharacterOffsetsField.new)
400
+ end
401
+
402
+ # Configure the code unit offsets field for this repository for a specific
403
+ # encoding and return self.
404
+ def code_unit_offsets(encoding)
405
+ field(:code_unit_offsets, CodeUnitOffsetsField.new(self, encoding))
406
+ end
407
+
408
+ # Configure the columns field for this repository and return self.
409
+ def columns
410
+ field(:columns, ColumnsField.new)
411
+ end
412
+
413
+ # Configure the character columns field for this repository and return
414
+ # self.
415
+ def character_columns
416
+ field(:character_columns, CharacterColumnsField.new)
417
+ end
418
+
419
+ # Configure the code unit columns field for this repository for a specific
420
+ # encoding and return self.
421
+ def code_unit_columns(encoding)
422
+ field(:code_unit_columns, CodeUnitColumnsField.new(self, encoding))
423
+ end
424
+
425
+ # Configure the leading comments field for this repository and return
426
+ # self.
427
+ def leading_comments
428
+ field(:leading_comments, LeadingCommentsField.new)
429
+ end
430
+
431
+ # Configure the trailing comments field for this repository and return
432
+ # self.
433
+ def trailing_comments
434
+ field(:trailing_comments, TrailingCommentsField.new)
435
+ end
436
+
437
+ # Configure both the leading and trailing comment fields for this
438
+ # repository and return self.
439
+ def comments
440
+ leading_comments.trailing_comments
441
+ end
442
+
443
+ # This method is called from nodes and locations when they want to enter
444
+ # themselves into the repository. It it internal-only and meant to be
445
+ # called from the #save* APIs.
446
+ def enter(node_id, field_name) # :nodoc:
447
+ entry = Entry.new(self)
448
+ @entries[node_id][field_name] = entry
449
+ entry
450
+ end
451
+
452
+ # This method is called from the entries in the repository when they need
453
+ # to reify their values. It is internal-only and meant to be called from
454
+ # the various value APIs.
455
+ def reify! # :nodoc:
456
+ result = source.result
457
+
458
+ # Attach the comments if they have been requested as part of the
459
+ # configuration of this repository.
460
+ if fields.key?(:leading_comments) || fields.key?(:trailing_comments)
461
+ result.attach_comments!
462
+ end
463
+
464
+ queue = [result.value] #: Array[Prism::node]
465
+ while (node = queue.shift)
466
+ @entries[node.node_id].each do |field_name, entry|
467
+ value = node.public_send(field_name)
468
+ values = {} #: Hash[Symbol, untyped]
469
+
470
+ fields.each_value do |field|
471
+ values.merge!(field.fields(value))
472
+ end
473
+
474
+ entry.reify!(values)
475
+ end
476
+
477
+ queue.concat(node.compact_child_nodes)
478
+ end
479
+
480
+ @entries.clear
481
+ end
482
+
483
+ private
484
+
485
+ # Append the given field to the repository and return the repository so
486
+ # that these calls can be chained.
487
+ def field(name, value)
488
+ raise ConfigurationError, "Cannot specify multiple #{name} fields" if @fields.key?(name)
489
+ @fields[name] = value
490
+ self
491
+ end
492
+ end
493
+
494
+ # Create a new repository for the given filepath.
495
+ def self.filepath(value)
496
+ Repository.new(SourceFilepath.new(value))
497
+ end
498
+
499
+ # Create a new repository for the given string.
500
+ def self.string(value)
501
+ Repository.new(SourceString.new(value))
502
+ end
503
+ end
504
+ end