sleeping_king_studios-docs 0.1.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 (116) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGELOG.md +5 -0
  3. data/CODE_OF_CONDUCT.md +132 -0
  4. data/LICENSE +22 -0
  5. data/README.md +101 -0
  6. data/lib/sleeping_king_studios/docs/commands/generate.rb +268 -0
  7. data/lib/sleeping_king_studios/docs/commands/generators/base.rb +97 -0
  8. data/lib/sleeping_king_studios/docs/commands/generators/data_generator.rb +68 -0
  9. data/lib/sleeping_king_studios/docs/commands/generators/reference_generator.rb +59 -0
  10. data/lib/sleeping_king_studios/docs/commands/generators.rb +15 -0
  11. data/lib/sleeping_king_studios/docs/commands/installation/install_jekyll.rb +273 -0
  12. data/lib/sleeping_king_studios/docs/commands/installation/install_templates.rb +125 -0
  13. data/lib/sleeping_king_studios/docs/commands/installation/install_workflow.rb +121 -0
  14. data/lib/sleeping_king_studios/docs/commands/installation.rb +15 -0
  15. data/lib/sleeping_king_studios/docs/commands/parse.rb +51 -0
  16. data/lib/sleeping_king_studios/docs/commands/write_file.rb +89 -0
  17. data/lib/sleeping_king_studios/docs/commands.rb +14 -0
  18. data/lib/sleeping_king_studios/docs/data/base.rb +48 -0
  19. data/lib/sleeping_king_studios/docs/data/class_object.rb +119 -0
  20. data/lib/sleeping_king_studios/docs/data/constant_object.rb +161 -0
  21. data/lib/sleeping_king_studios/docs/data/metadata.rb +196 -0
  22. data/lib/sleeping_king_studios/docs/data/method_object.rb +555 -0
  23. data/lib/sleeping_king_studios/docs/data/module_object.rb +234 -0
  24. data/lib/sleeping_king_studios/docs/data/namespace_object.rb +375 -0
  25. data/lib/sleeping_king_studios/docs/data/root_object.rb +40 -0
  26. data/lib/sleeping_king_studios/docs/data/see_tags/base.rb +35 -0
  27. data/lib/sleeping_king_studios/docs/data/see_tags/class_method_tag.rb +99 -0
  28. data/lib/sleeping_king_studios/docs/data/see_tags/constant_tag.rb +60 -0
  29. data/lib/sleeping_king_studios/docs/data/see_tags/definition_tag.rb +52 -0
  30. data/lib/sleeping_king_studios/docs/data/see_tags/instance_method_tag.rb +69 -0
  31. data/lib/sleeping_king_studios/docs/data/see_tags/link_tag.rb +53 -0
  32. data/lib/sleeping_king_studios/docs/data/see_tags/namespace_item_tag.rb +56 -0
  33. data/lib/sleeping_king_studios/docs/data/see_tags/reference_tag.rb +92 -0
  34. data/lib/sleeping_king_studios/docs/data/see_tags/text_tag.rb +30 -0
  35. data/lib/sleeping_king_studios/docs/data/see_tags/unstructured_tag.rb +37 -0
  36. data/lib/sleeping_king_studios/docs/data/see_tags.rb +101 -0
  37. data/lib/sleeping_king_studios/docs/data/types/grammar.treetop +49 -0
  38. data/lib/sleeping_king_studios/docs/data/types/key_value_type.rb +54 -0
  39. data/lib/sleeping_king_studios/docs/data/types/parameterized_type.rb +57 -0
  40. data/lib/sleeping_king_studios/docs/data/types/parser.rb +143 -0
  41. data/lib/sleeping_king_studios/docs/data/types/type.rb +100 -0
  42. data/lib/sleeping_king_studios/docs/data/types.rb +19 -0
  43. data/lib/sleeping_king_studios/docs/data.rb +29 -0
  44. data/lib/sleeping_king_studios/docs/errors/file_already_exists.rb +22 -0
  45. data/lib/sleeping_king_studios/docs/errors/file_error.rb +30 -0
  46. data/lib/sleeping_king_studios/docs/errors/file_not_found.rb +22 -0
  47. data/lib/sleeping_king_studios/docs/errors/invalid_directory.rb +22 -0
  48. data/lib/sleeping_king_studios/docs/errors/invalid_file.rb +22 -0
  49. data/lib/sleeping_king_studios/docs/errors.rb +19 -0
  50. data/lib/sleeping_king_studios/docs/registry.rb +22 -0
  51. data/lib/sleeping_king_studios/docs/registry_query.rb +93 -0
  52. data/lib/sleeping_king_studios/docs/tasks/base.rb +20 -0
  53. data/lib/sleeping_king_studios/docs/tasks/generate.rb +39 -0
  54. data/lib/sleeping_king_studios/docs/tasks/installation/install_jekyll.rb +67 -0
  55. data/lib/sleeping_king_studios/docs/tasks/installation/install_templates.rb +39 -0
  56. data/lib/sleeping_king_studios/docs/tasks/installation/install_workflow.rb +53 -0
  57. data/lib/sleeping_king_studios/docs/tasks/installation.rb +8 -0
  58. data/lib/sleeping_king_studios/docs/tasks/update.rb +35 -0
  59. data/lib/sleeping_king_studios/docs/tasks.rb +14 -0
  60. data/lib/sleeping_king_studios/docs/templates/config.yml.erb +22 -0
  61. data/lib/sleeping_king_studios/docs/templates/deploy-pages.yml.erb +67 -0
  62. data/lib/sleeping_king_studios/docs/templates/includes/breadcrumbs.md +7 -0
  63. data/lib/sleeping_king_studios/docs/templates/includes/reference/attribute.md +23 -0
  64. data/lib/sleeping_king_studios/docs/templates/includes/reference/attributes/heading.md +15 -0
  65. data/lib/sleeping_king_studios/docs/templates/includes/reference/class.md +27 -0
  66. data/lib/sleeping_king_studios/docs/templates/includes/reference/constant.md +8 -0
  67. data/lib/sleeping_king_studios/docs/templates/includes/reference/constants/heading.md +6 -0
  68. data/lib/sleeping_king_studios/docs/templates/includes/reference/constants/inherited.md +4 -0
  69. data/lib/sleeping_king_studios/docs/templates/includes/reference/constants/overview.md +60 -0
  70. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/breadcrumbs.md +21 -0
  71. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/class_attributes.md +9 -0
  72. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/class_methods.md +9 -0
  73. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/constants.md +9 -0
  74. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/constructor.md +12 -0
  75. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/definitions.md +23 -0
  76. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/details.md +34 -0
  77. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/instance_attributes.md +9 -0
  78. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/instance_methods.md +17 -0
  79. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/overview.md +74 -0
  80. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/subclasses.md +10 -0
  81. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/class_attributes.md +10 -0
  82. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/class_methods.md +10 -0
  83. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/constants.md +10 -0
  84. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/instance_attributes.md +10 -0
  85. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/instance_methods.md +10 -0
  86. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/overview.md +41 -0
  87. data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents.md +27 -0
  88. data/lib/sleeping_king_studios/docs/templates/includes/reference/method.md +27 -0
  89. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/aliases.md +4 -0
  90. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/heading.md +14 -0
  91. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/inherited.md +4 -0
  92. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/overload.md +18 -0
  93. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/overloads.md +12 -0
  94. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/overview.md +41 -0
  95. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/parameters.md +31 -0
  96. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/post_overview.md +51 -0
  97. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/raises.md +14 -0
  98. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/return_types.md +9 -0
  99. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/returns.md +14 -0
  100. data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/yields.md +42 -0
  101. data/lib/sleeping_king_studios/docs/templates/includes/reference/module.md +23 -0
  102. data/lib/sleeping_king_studios/docs/templates/includes/reference/namespace.md +33 -0
  103. data/lib/sleeping_king_studios/docs/templates/includes/reference/parent_link.md +6 -0
  104. data/lib/sleeping_king_studios/docs/templates/includes/reference/reference_link.md +7 -0
  105. data/lib/sleeping_king_studios/docs/templates/includes/reference/see_link.md +7 -0
  106. data/lib/sleeping_king_studios/docs/templates/includes/reference/type.md +14 -0
  107. data/lib/sleeping_king_studios/docs/templates/includes/reference/type_list.md +4 -0
  108. data/lib/sleeping_king_studios/docs/templates/includes/reference/types/array_type.md +1 -0
  109. data/lib/sleeping_king_studios/docs/templates/includes/reference/types/hash_type.md +1 -0
  110. data/lib/sleeping_king_studios/docs/templates/includes/reference/types/ordered_type.md +1 -0
  111. data/lib/sleeping_king_studios/docs/templates/pages/index.md.erb +22 -0
  112. data/lib/sleeping_king_studios/docs/templates/pages/reference.md.erb +14 -0
  113. data/lib/sleeping_king_studios/docs/templates/pages/versions.md.erb +13 -0
  114. data/lib/sleeping_king_studios/docs/version.rb +59 -0
  115. data/lib/sleeping_king_studios/docs.rb +27 -0
  116. metadata +243 -0
@@ -0,0 +1,555 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'sleeping_king_studios/docs/data'
4
+
5
+ module SleepingKingStudios::Docs::Data
6
+ # Object representing a Ruby method.
7
+ #
8
+ # Each method has a name and a signature, and can define the following
9
+ # elements:
10
+ #
11
+ # - Parameters.
12
+ # - A return type.
13
+ # - Raised exceptions.
14
+ # - A yielded block.
15
+ #
16
+ # In addition, a method may have a short description, a full description, and
17
+ # metadata.
18
+ class MethodObject < SleepingKingStudios::Docs::Data::Base # rubocop:disable Metrics/ClassLength
19
+ JSON_PROPERTIES = %i[
20
+ aliases
21
+ constructor
22
+ data_path
23
+ description
24
+ metadata
25
+ options
26
+ overloads
27
+ params
28
+ raises
29
+ returns
30
+ short_description
31
+ yield_params
32
+ yield_returns
33
+ yields
34
+ ].freeze
35
+ private_constant :JSON_PROPERTIES
36
+
37
+ NAME_SEPARATOR = /::|#|\./
38
+ private_constant :NAME_SEPARATOR
39
+
40
+ PARAGRAPH_BREAK = /\n{2,}/
41
+ private_constant :PARAGRAPH_BREAK
42
+
43
+ # @return [true, false] true if the method object is an alias of another
44
+ # method; otherwise false.
45
+ def alias?
46
+ native.is_alias?
47
+ end
48
+
49
+ # @return [Array<String>] the names of the method aliases, if any.
50
+ def aliases
51
+ native.aliases.map { |obj| obj.name.to_s }.sort
52
+ end
53
+
54
+ # Generates a JSON-compatible representation of the method.
55
+ #
56
+ # Returns a Hash with the following keys:
57
+ #
58
+ # - 'name': The full, qualified name of the method.
59
+ # - 'slug': The name of the method in url-safe format.
60
+ # - 'signature': A String representation of the method and its parameters.
61
+ # - 'short_description': A short description of the method.
62
+ #
63
+ # Additionally, the returned Hash will conditionally include the following
64
+ # keys, if the method defines at least one of the corresponding code
65
+ # objects.
66
+ #
67
+ # - 'description': The full description of the method, minus the first
68
+ # clause.
69
+ # - 'metadata': Additional metadata tags from the documentation.
70
+ # - 'params': The method's parameters, as documented.
71
+ # - 'options': The method's option parameters, if any.
72
+ # - 'raises': The exceptions raised by the method.
73
+ # - 'returns': The method's return types and description.
74
+ # - 'yields': The block yielded by the method.
75
+ # - 'yieldparams': The yielded blocks' parameters.
76
+ # - 'yieldreturn': The value returned by the block.
77
+ #
78
+ # Finally, the method may have one or more overloads, which replace or
79
+ # supplement the method with alternative signatures or documentation.
80
+ #
81
+ # - 'overloads': The method overloads, if any. Each overload has the same
82
+ # keys as a full method object.
83
+ #
84
+ # @return [Hash{String => Object}] the representation of the method.
85
+ def as_json
86
+ JSON_PROPERTIES
87
+ .reduce(required_json) do |memo, property_name|
88
+ value = send(property_name)
89
+
90
+ next memo if empty?(value)
91
+
92
+ memo.update(property_name.to_s => value)
93
+ end
94
+ .then { |json| merge_pure_overload(json) }
95
+ end
96
+
97
+ # @return [Boolean] true if the method is a class method; otherwise false.
98
+ def class_method?
99
+ !instance_method?
100
+ end
101
+
102
+ # @return [Boolean] true if the method is a constructor; otherwise false.
103
+ def constructor?
104
+ native.constructor?
105
+ end
106
+ alias constructor constructor?
107
+
108
+ # The path to the data file.
109
+ #
110
+ # @return [String] the file path.
111
+ def data_path
112
+ return @data_path if @data_path
113
+
114
+ *scope_names, method_name =
115
+ name.split(NAME_SEPARATOR).reject(&:empty?)
116
+
117
+ scope_names = scope_names.map { |str| slugify(str) }
118
+ scope_names << "#{instance_method? ? 'i' : 'c'}-#{slugify(method_name)}"
119
+
120
+ @data_path = scope_names.join('/')
121
+ end
122
+
123
+ # The full description of the method, minus the first clause.
124
+ #
125
+ # The remainder of the method description, if any, after subtracting the
126
+ # short description (separated by the first paragraph break).
127
+ #
128
+ # @return [String] the remaining description.
129
+ #
130
+ # @see #short_description.
131
+ def description
132
+ return @description if @description
133
+
134
+ @short_description, @description = split_docstring
135
+
136
+ @description
137
+ end
138
+
139
+ # @return [Boolean] true if the method is an instance method; otherwise
140
+ # false.
141
+ def instance_method?
142
+ native.path[-(1 + native.name.length)] == '#'
143
+ end
144
+
145
+ # Additional metadata tags from the documentation.
146
+ #
147
+ # @see SleepingKingStudios::Docs::Data::Metadata.
148
+ def metadata
149
+ @metadata ||= format_metadata
150
+ end
151
+
152
+ # The full, qualified name of the method.
153
+ #
154
+ # @return [String] the qualified name.
155
+ def name
156
+ @name ||= native.path
157
+ end
158
+
159
+ # The documented options of the method.
160
+ #
161
+ # Each option is a Hash with the following keys:
162
+ #
163
+ # - 'name': The name of the parameter.
164
+ # - 'type': The JSON representation of the parsed Type object.
165
+ # - 'description': The description of the parameter.
166
+ #
167
+ # @return [Array<Hash>] the method options.
168
+ def options
169
+ @options ||=
170
+ native
171
+ .tags
172
+ .select { |tag| tag.tag_name == 'option' }
173
+ .group_by(&:name)
174
+ .map { |name, tags| format_options(name, tags) }
175
+ end
176
+
177
+ # Checks if the method has been completely overloaded.
178
+ #
179
+ # A completely overloaded method has no description and no tags other than
180
+ # exactly one @overload tag. This case is common when documenting a method
181
+ # with a different signature.
182
+ #
183
+ # @return [true, false] true if the method is completely overloaded;
184
+ # otherwise false.
185
+ def overloaded?
186
+ return false unless native.docstring.empty?
187
+
188
+ return false unless native.tags.all? { |tag| tag.tag_name == 'overload' }
189
+
190
+ native.tags.size == 1
191
+ end
192
+
193
+ # The documented overloads for the method.
194
+ #
195
+ # Each overload is a JSON representation of a method, and includes the same
196
+ # tags and properties (with the exception that an overload cannot itself
197
+ # have overloads).
198
+ #
199
+ # @see #as_json
200
+ def overloads
201
+ native
202
+ .tags
203
+ .select { |tag| tag.tag_name == 'overload' }
204
+ .map { |tag| format_overload(tag) }
205
+ end
206
+
207
+ # The documented parameters of the method.
208
+ #
209
+ # Each parameter is a Hash with the following keys:
210
+ #
211
+ # - 'name': The name of the parameter.
212
+ # - 'type': The JSON representation of the parsed Type object.
213
+ # - 'description': The description of the parameter.
214
+ #
215
+ # @return [Array<Hash>] the method parameters.
216
+ def params
217
+ @params ||=
218
+ native
219
+ .tags
220
+ .select { |tag| tag.tag_name == 'param' }
221
+ .map { |tag| format_param(tag) }
222
+ end
223
+
224
+ # The path to the defining class or module's data file.
225
+ #
226
+ # @return [String] the file path.
227
+ def parent_path
228
+ return @parent_path if @parent_path
229
+
230
+ return @parent_path = '' if native.parent.root?
231
+
232
+ parent_class =
233
+ if native.parent.is_a?(YARD::CodeObjects::ClassObject)
234
+ SleepingKingStudios::Docs::Data::ClassObject
235
+ else
236
+ SleepingKingStudios::Docs::Data::ModuleObject
237
+ end
238
+ parent_object = parent_class.new(native: native.parent)
239
+
240
+ @parent_path = parent_object.data_path
241
+ end
242
+
243
+ # The documented raised exceptions of the method.
244
+ #
245
+ # Each raised exception is a Hash with the following keys:
246
+ #
247
+ # - 'type': The type of exception raised.
248
+ # - 'description': The description of the exception, or details on when the
249
+ # exception is raised.
250
+ #
251
+ # @return [Array<Hash>] the exceptions raised.by the method.
252
+ def raises
253
+ @raises ||=
254
+ native
255
+ .tags
256
+ .select { |tag| tag.tag_name == 'raise' }
257
+ .map { |tag| format_return(tag) }
258
+ end
259
+
260
+ # The return type or types of the method.
261
+ #
262
+ # Each return type or types is a Hash with the following keys:
263
+ #
264
+ # - 'type': The JSON representation of the parsed Type object.
265
+ # - 'description': The description of the returned type, or details on when
266
+ # that type will be returned.
267
+ #
268
+ # @return [Array<Hash>] the return types of the method.
269
+ #
270
+ # @see SleepingKingStudios::Docs::Data::Types::Type.
271
+ def returns
272
+ @returns ||=
273
+ native
274
+ .tags
275
+ .select { |tag| tag.tag_name == 'return' }
276
+ .map { |tag| format_return(tag) }
277
+ end
278
+
279
+ # A short description of the method.
280
+ #
281
+ # The first part of the method description, separated by the first
282
+ # paragraph break. Typically should fit on a single line of text.
283
+ #
284
+ # @return [String] the short description.
285
+ #
286
+ # @see #description.
287
+ def short_description
288
+ return @short_description if @short_description
289
+
290
+ @short_description, @description = split_docstring
291
+
292
+ @short_description
293
+ end
294
+
295
+ # The name and parameters of the method.
296
+ #
297
+ # @return [String] the method signature.
298
+ def signature
299
+ return @signature if @signature
300
+
301
+ unless native.signature.include?('(')
302
+ return @signature = native.signature.sub(/\Adef /, '')
303
+ end
304
+
305
+ @signature = generate_signature
306
+ end
307
+
308
+ # The name of the method in url-safe format.
309
+ #
310
+ # @return [String] the method name.
311
+ def slug
312
+ @slug ||= slugify(name.sub(/\A::/, '').split(/#|\./).last)
313
+ end
314
+
315
+ # The parameters of the yielded to the block.
316
+ #
317
+ # Each parameter is a Hash with the following keys:
318
+ #
319
+ # - 'name': The name of the parameter.
320
+ # - 'type': The JSON representation of the parsed Type object.
321
+ # - 'description': The description of the parameter.
322
+ #
323
+ # @return [Array<Hash>] the yielded parameters.
324
+ def yield_params
325
+ @yield_params ||=
326
+ native
327
+ .tags
328
+ .select { |tag| tag.tag_name == 'yieldparam' }
329
+ .map { |tag| format_yield_param(tag) }
330
+ end
331
+
332
+ # The return type or types of the yielded block.
333
+ #
334
+ # Each return type or types is a Hash with the following keys:
335
+ #
336
+ # - 'type': The JSON representation of the parsed Type object.
337
+ # - 'description': The description of the returned type, or details on when
338
+ # that type will be returned.
339
+ #
340
+ # @return [Array<Hash>] the return types of the yielded block.
341
+ #
342
+ # @see SleepingKingStudios::Docs::Data::Types::Type.
343
+ def yield_returns
344
+ @yield_returns ||=
345
+ native
346
+ .tags
347
+ .select { |tag| tag.tag_name == 'yieldreturn' }
348
+ .map { |tag| format_return(tag) }
349
+ end
350
+
351
+ # The yielded block or blocks of method.
352
+ #
353
+ # Each yielded block is a Hash with the following keys:
354
+ #
355
+ # - 'description': The description of the yielded block or details on when
356
+ # the block will be yielded.
357
+ # - 'parameters': (Optional) The parameters yielded to the block.
358
+ #
359
+ # @return [Array<Hash>] the yielded blocks.
360
+ #
361
+ # @see #yieldparams
362
+ # @see #yieldreturns
363
+ def yields
364
+ native
365
+ .tags
366
+ .select { |tag| tag.tag_name == 'yield' }
367
+ .map { |tag| format_yield(tag) }
368
+ end
369
+
370
+ private
371
+
372
+ def extract_parameters
373
+ native
374
+ .signature
375
+ .sub(/\A(def )?#{Regexp.escape(native.name)}\(/, '')
376
+ .split(')')
377
+ .first
378
+ end
379
+
380
+ def extract_yield_parameters
381
+ return '' unless native.signature.match?(/\) *{/)
382
+
383
+ native
384
+ .signature
385
+ .split(')')
386
+ .last
387
+ .sub(/\A ?\{ ?\|/, '')
388
+ .sub(/\| ?\}\z/, '')
389
+ end
390
+
391
+ def format_metadata
392
+ SleepingKingStudios::Docs::Data::Metadata
393
+ .new(native:)
394
+ .as_json
395
+ end
396
+
397
+ def format_option(tag)
398
+ {
399
+ 'description' => tag.pair.text,
400
+ 'name' => tag.pair.name,
401
+ 'type' => parse_types(tag.pair)
402
+ }
403
+ end
404
+
405
+ def format_options(name, tags)
406
+ {
407
+ 'name' => name,
408
+ 'opts' => tags.map { |tag| format_option(tag) }
409
+ }
410
+ end
411
+
412
+ def format_overload(tag)
413
+ self
414
+ .class
415
+ .new(native: tag)
416
+ .as_json
417
+ .tap { |hsh| hsh.delete('data_path') }
418
+ end
419
+
420
+ def format_param(tag)
421
+ json = {
422
+ 'description' => tag.text,
423
+ 'name' => tag.name,
424
+ 'type' => parse_types(tag)
425
+ }
426
+
427
+ return json unless parameter_defaults.key?(tag.name)
428
+
429
+ json.merge('default' => parameter_defaults[tag.name])
430
+ end
431
+
432
+ def format_return(tag)
433
+ {
434
+ 'description' => tag.text,
435
+ 'type' => parse_types(tag)
436
+ }
437
+ end
438
+
439
+ def format_yield(tag)
440
+ json = { 'description' => tag.text }
441
+
442
+ return json unless tag.types
443
+
444
+ json.merge('parameters' => tag.types)
445
+ end
446
+
447
+ def format_yield_param(tag)
448
+ json = {
449
+ 'description' => tag.text,
450
+ 'name' => tag.name,
451
+ 'type' => parse_types(tag)
452
+ }
453
+
454
+ return json unless yield_defaults.key?(tag.name)
455
+
456
+ json.merge('default' => yield_defaults[tag.name])
457
+ end
458
+
459
+ def generate_signature
460
+ params = extract_parameters
461
+ yields = extract_yield_parameters
462
+ buffer = "#{native.name}(#{strip_rubocop_directives(params)})"
463
+
464
+ return buffer if yields.empty?
465
+
466
+ "#{buffer} { |#{strip_rubocop_directives(yields)}| }"
467
+ end
468
+
469
+ def merge_pure_overload(json)
470
+ return json unless overloaded?
471
+
472
+ overload = json.delete('overloads').first
473
+
474
+ json.merge(overload)
475
+ end
476
+
477
+ def parse_types(tag)
478
+ (tag.types || []).reduce([]) do |memo, type|
479
+ memo + type_parser.parse(type).map(&:as_json)
480
+ end
481
+ end
482
+
483
+ def parameter_defaults
484
+ return @parameter_defaults if @parameter_defaults
485
+
486
+ return @parameter_defaults = {} unless signature.include?('(')
487
+
488
+ @parameter_defaults ||=
489
+ signature[(1 + native.name.size)...-1]
490
+ .split(', ')
491
+ .map { |param| param.split(/ = |: ?/) }
492
+ .select { |tuple| tuple.size == 2 }
493
+ .to_h
494
+ end
495
+
496
+ def required_json
497
+ {
498
+ 'name' => name,
499
+ 'parent_path' => parent_path,
500
+ 'signature' => signature,
501
+ 'slug' => slug
502
+ }
503
+ end
504
+
505
+ def rubocop_directive?(str)
506
+ return true if str == 'rubocop:disable'
507
+
508
+ str.include? '/'
509
+ end
510
+
511
+ def split_docstring
512
+ match = native.docstring.match(PARAGRAPH_BREAK)
513
+
514
+ return native.docstring.to_s unless match
515
+
516
+ [match.pre_match.to_s, match.post_match.to_s]
517
+ end
518
+
519
+ def strip_rubocop_directives(parameters)
520
+ return '' if parameters.nil?
521
+
522
+ parameters
523
+ .split(/, +/)
524
+ .map { |segment| strip_rubocop_directives_from_segment(segment) }
525
+ .reject(&:empty?)
526
+ .join(', ')
527
+ end
528
+
529
+ def strip_rubocop_directives_from_segment(segment)
530
+ segment
531
+ .strip
532
+ .split(/ +/)
533
+ .reject { |str| str == '#' || rubocop_directive?(str) }
534
+ .join(' ')
535
+ end
536
+
537
+ def type_parser
538
+ @type_parser ||=
539
+ SleepingKingStudios::Docs::Data::Types::Parser.new
540
+ end
541
+
542
+ def yield_defaults # rubocop:disable Metrics/CyclomaticComplexity
543
+ yield_tag = native.tags.find { |tag| tag.tag_name == 'yield' }
544
+
545
+ return {} unless yield_tag
546
+
547
+ @yield_defaults ||=
548
+ yield_tag
549
+ .types
550
+ &.map { |param| param.split(/ = |: ?/) }
551
+ &.select { |tuple| tuple.size == 2 }
552
+ .to_h
553
+ end
554
+ end
555
+ end