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,234 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'sleeping_king_studios/docs/data'
4
+
5
+ module SleepingKingStudios::Docs::Data
6
+ # Object representing a Ruby module.
7
+ #
8
+ # Each module can define the following elements:
9
+ #
10
+ # - Defined In files.
11
+ # - extend-ed modules.
12
+ # - include-ed modules.
13
+ # - Definitions (classes and modules).
14
+ # - Constants.
15
+ # - Class attributes.
16
+ # - Class methods.
17
+ # - Instance attributes.
18
+ # - Instance methods.
19
+ #
20
+ # Additionally, a module can have a description and metadata tags.
21
+ #
22
+ # @see SleepingKingStudios::Docs::Data::ClassObject
23
+ # @see SleepingKingStudios::Docs::Data::Metadata
24
+ # @see SleepingKingStudios::Docs::Data::NamespaceObject
25
+ class ModuleObject < NamespaceObject # rubocop:disable Metrics/ClassLength
26
+ JSON_PROPERTIES = %i[
27
+ data_path
28
+ description
29
+ extended_modules
30
+ included_modules
31
+ metadata
32
+ ].freeze
33
+ private_constant :JSON_PROPERTIES
34
+
35
+ PARAGRAPH_BREAK = /\n{2,}/
36
+ private_constant :PARAGRAPH_BREAK
37
+
38
+ # Generates a JSON-compatible representation of the module.
39
+ #
40
+ # Returns a Hash with the following keys:
41
+ #
42
+ # - 'name': The full, qualified name of the module.
43
+ # - 'slug': The name of the module in url-safe format.
44
+ # - 'files': A list of the files where the module is defined.
45
+ # - 'short_description': A short description of the module.
46
+ #
47
+ # Additionally, the returned Hash will conditionally include the following
48
+ # keys, if the module defines at least one of the corresponding code
49
+ # objects.
50
+ #
51
+ # - 'class_attributes': The class attributes, if any.
52
+ # - 'class_methods': The class methods, if any.
53
+ # - 'constants': The constants, if any.
54
+ # - 'defined_classes': The defined Classes, if any.
55
+ # - 'defined_modules': The defined Modules, if any.
56
+ # - 'description': The full description of the module, minus the first
57
+ # clause.
58
+ # - 'extended_modules': A list of the modules that extend the module.
59
+ # - 'included_modules': A list of the modules that are included in the
60
+ # module.
61
+ # - 'instance_attributes': The instance attributes, if any.
62
+ # - 'instance_methods': The instance methods, if any.
63
+ # - 'metadata': Additional metadata tags from the documentation.
64
+ #
65
+ # @return [Hash{String => Object}] the representation of the module.
66
+ def as_json
67
+ JSON_PROPERTIES.reduce(super) do |memo, property_name|
68
+ value = send(property_name)
69
+
70
+ next memo if empty?(value)
71
+
72
+ memo.update(property_name.to_s => value)
73
+ end
74
+ end
75
+
76
+ # The path to the data file.
77
+ #
78
+ # @return [String] the file path.
79
+ def data_path
80
+ @data_path ||= name.split('::').map { |str| slugify(str) }.join('/')
81
+ end
82
+
83
+ # The full description of the module, minus the first clause.
84
+ #
85
+ # The remainder of the module description, if any, after subtracting the
86
+ # short description (separated by the first paragraph break).
87
+ #
88
+ # @return [String] the remaining description.
89
+ #
90
+ # @see #short_description.
91
+ def description
92
+ return @description if @description
93
+
94
+ @short_description, @description = split_docstring
95
+
96
+ @description
97
+ end
98
+
99
+ # A list of the modules that extend the original module.
100
+ #
101
+ # For each extending Module, it returns a Hash with the following keys:
102
+ #
103
+ # - 'name': The name of the extending module.
104
+ # - 'slug': A url-safe, hyphen-separated representation of the name.
105
+ # - 'path': The path to the data file for the module.
106
+ #
107
+ # @return [Array<Hash{String => String}>] the extended modules.
108
+ def extended_modules
109
+ @extended_modules ||=
110
+ native
111
+ .class_mixins
112
+ .map { |obj| format_inclusion(obj) }
113
+ .sort_by { |hsh| hsh['name'] }
114
+ end
115
+
116
+ # A list of the files where the module is defined.
117
+ #
118
+ # @return [Array<String>] the list of files.
119
+ def files
120
+ @files ||= native.files.map(&:first)
121
+ end
122
+
123
+ # A list of the modules that are included in the original module.
124
+ #
125
+ # For each included Module, it returns a Hash with the following keys:
126
+ #
127
+ # - 'name': The name of the included module.
128
+ # - 'slug': A url-safe, hyphen-separated representation of the name.
129
+ # - 'path': The path to the data file for the module.
130
+ #
131
+ # @return [Array<Hash{String => String}>] the included modules.
132
+ def included_modules
133
+ @included_modules ||=
134
+ native
135
+ .instance_mixins
136
+ .map { |obj| format_inclusion(obj) }
137
+ .sort_by { |hsh| hsh['name'] }
138
+ end
139
+
140
+ # Additional metadata tags from the documentation.
141
+ #
142
+ # @see SleepingKingStudios::Docs::Data::Metadata.
143
+ def metadata
144
+ @metadata ||= format_metadata
145
+ end
146
+
147
+ # The path to the defining class or module's data file.
148
+ #
149
+ # @return [String] the file path.
150
+ def parent_path
151
+ return @parent_path if @parent_path
152
+
153
+ return @parent_path = '' if native.parent.root?
154
+
155
+ parent_class =
156
+ if native.parent.is_a?(YARD::CodeObjects::ClassObject)
157
+ SleepingKingStudios::Docs::Data::ClassObject
158
+ else
159
+ SleepingKingStudios::Docs::Data::ModuleObject
160
+ end
161
+ parent_object = parent_class.new(native: native.parent)
162
+
163
+ @parent_path = parent_object.data_path
164
+ end
165
+
166
+ # A short description of the module.
167
+ #
168
+ # The first part of the module description, separated by the first paragraph
169
+ # break. Typically should fit on a single line of text.
170
+ #
171
+ # @return [String] the short description.
172
+ #
173
+ # @see #description.
174
+ def short_description
175
+ return @short_description if @short_description
176
+
177
+ @short_description, @description = split_docstring
178
+
179
+ @short_description
180
+ end
181
+
182
+ # @return [String] the type of the namespace.
183
+ def type
184
+ 'module'
185
+ end
186
+
187
+ private
188
+
189
+ def format_inclusion(obj)
190
+ json = {
191
+ 'name' => obj.path,
192
+ 'slug' => slugify(obj.name)
193
+ }
194
+
195
+ return json unless registry.include?(obj)
196
+
197
+ json.merge(
198
+ 'path' => obj.path.split('::').map { |str| slugify(str) }.join('/')
199
+ )
200
+ end
201
+
202
+ def format_metadata
203
+ SleepingKingStudios::Docs::Data::Metadata
204
+ .new(native:)
205
+ .as_json
206
+ end
207
+
208
+ def inherited_constant?(constant_object)
209
+ constant_object.data_path != "#{data_path}/#{constant_object.slug}"
210
+ end
211
+
212
+ def inherited_method?(method_object)
213
+ !method_object.data_path.match?(
214
+ %r{\A#{data_path}/(c|i)-#{Regexp.escape(method_object.slug)}\z}
215
+ )
216
+ end
217
+
218
+ def required_json
219
+ super.merge(
220
+ 'files' => files,
221
+ 'parent_path' => parent_path,
222
+ 'short_description' => short_description
223
+ )
224
+ end
225
+
226
+ def split_docstring
227
+ match = native.docstring.match(PARAGRAPH_BREAK)
228
+
229
+ return native.docstring.to_s unless match
230
+
231
+ [match.pre_match.to_s, match.post_match.to_s]
232
+ end
233
+ end
234
+ end
@@ -0,0 +1,375 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'sleeping_king_studios/tools/toolbelt'
4
+
5
+ require 'sleeping_king_studios/docs/data'
6
+
7
+ module SleepingKingStudios::Docs::Data
8
+ # Object representing a Ruby namespace.
9
+ #
10
+ # Each namespace can define the following elements:
11
+ #
12
+ # - Definitions (classes and modules).
13
+ # - Constants.
14
+ # - Class attributes.
15
+ # - Class methods.
16
+ # - Instance attributes.
17
+ # - Instance methods.
18
+ #
19
+ # In most cases, there should be one documented namespace, reflecting the
20
+ # top-level namespace, and in most cases should only include defined classes
21
+ # and/or modules. Top-level constants may be required on a case-by-case basis.
22
+ #
23
+ # All other namespace properties (attributes and methods) should be reserved
24
+ # for classes and modules, whose representation inherits from NamespaceObject.
25
+ #
26
+ # @see SleepingKingStudios::Docs::Data::ClassObject
27
+ # @see SleepingKingStudios::Docs::Data::ModuleObject
28
+ # @see SleepingKingStudios::Docs::Data::RootObject
29
+ class NamespaceObject < SleepingKingStudios::Docs::Data::Base # rubocop:disable Metrics/ClassLength
30
+ JSON_PROPERTIES = %i[
31
+ class_attributes
32
+ class_methods
33
+ constants
34
+ defined_classes
35
+ defined_modules
36
+ instance_attributes
37
+ instance_methods
38
+ ].freeze
39
+ private_constant :JSON_PROPERTIES
40
+
41
+ # Generates a JSON-compatible representation of the namespace.
42
+ #
43
+ # Returns a Hash with the following keys:
44
+ #
45
+ # - 'name': The full, qualified name of the namespace.
46
+ # - 'slug': The name of the namespace in url-safe format.
47
+ #
48
+ # Additionally, the returned Hash will conditionally include the following
49
+ # keys, if the namespace defines at least one of the corresponding code
50
+ # objects.
51
+ #
52
+ # - 'class_attributes': The class attributes, if any.
53
+ # - 'class_methods': The class methods, if any.
54
+ # - 'constants': The constants, if any.
55
+ # - 'defined_classes': The defined Classes, if any.
56
+ # - 'defined_modules': The defined Modules, if any.
57
+ # - 'instance_attributes': The instance attributes, if any.
58
+ # - 'instance_methods': The instance methods, if any.
59
+ #
60
+ # @return [Hash{String => Object}] the representation of the namespace.
61
+ def as_json
62
+ JSON_PROPERTIES.reduce(required_json) do |memo, property_name|
63
+ value = send(property_name)
64
+
65
+ next memo if empty?(value)
66
+
67
+ memo.update(property_name.to_s => value)
68
+ end
69
+ end
70
+
71
+ # Finds the class attributes defined for the namespace.
72
+ #
73
+ # For each class attribute, it returns a Hash with the following keys:
74
+ #
75
+ # - 'name': The name of the attribute.
76
+ # - 'read': True if the attribute defines a reader method.
77
+ # - 'write': True if the attribute defines a writer method.
78
+ # - 'path': The path to the reader method data file, or the writer method
79
+ # data file if the attribute does not define a reader method.
80
+ #
81
+ # @return [Array<Hash>] the class attributes.
82
+ def class_attributes
83
+ @class_attributes ||=
84
+ find_class_attributes(native)
85
+ .map { |name, options| format_attribute(name, options) }
86
+ .compact
87
+ .sort_by { |hsh| hsh['name'] }
88
+ end
89
+
90
+ # Finds the class methods defined for the namespace.
91
+ #
92
+ # For each method, it returns a Hash with the following keys:
93
+ #
94
+ # - 'name': The name of the method, including trailing characters such as
95
+ # `=' or `?'.
96
+ # - 'path': The path to the method data file.
97
+ #
98
+ # @return [Array<Hash{String => String}>] the documented class methods.
99
+ def class_methods # rubocop:disable Metrics/CyclomaticComplexity
100
+ @class_methods ||=
101
+ native
102
+ .meths
103
+ .select { |obj| obj.scope == :class && !obj.is_attribute? }
104
+ .reject { |obj| private_method?(obj) }
105
+ .reject(&:is_alias?)
106
+ .map { |obj| format_method(obj) }
107
+ .sort_by { |hsh| hsh['name'] }
108
+ end
109
+
110
+ # Finds the names of the constants defined under this namespace.
111
+ #
112
+ # @return [Array<String>] the names of the constants.
113
+ def constants
114
+ @constants ||=
115
+ native
116
+ .constants
117
+ .reject { |obj| private_constant?(obj) }
118
+ .map { |obj| format_constant(obj) }
119
+ .sort_by { |hsh| hsh['name'] }
120
+ end
121
+
122
+ # Finds the Classes defined under this namespace, if any.
123
+ #
124
+ # For each defined Class, it returns a Hash with the following keys:
125
+ #
126
+ # - 'name': The name of the defined Class.
127
+ # - 'slug': A url-safe, hyphen-separated representation of the name.
128
+ #
129
+ # @example
130
+ # # Given a class LaunchWindow in the namespace Space::Operations:
131
+ # namespace.name
132
+ # #=> 'Space::Operations'
133
+ # namespace.defined_classes
134
+ # #=> [{ 'name' => 'LaunchWindow', 'slug' => 'launch-window' }]
135
+ #
136
+ # @return [Array<Hash>] the defined classes.
137
+ def defined_classes
138
+ @defined_classes ||=
139
+ native
140
+ .children
141
+ .select { |obj| obj.type == :class }
142
+ .reject { |obj| private_definition?(obj) }
143
+ .map { |obj| format_definition(obj) }
144
+ .sort_by { |hsh| hsh['name'] }
145
+ end
146
+
147
+ # Finds the Modules defined under this namespace, if any.
148
+ #
149
+ # For each defined Module, it returns a Hash with the following keys:
150
+ #
151
+ # - 'name': The name of the defined Module.
152
+ # - 'slug': A url-safe, hyphen-separated representation of the name.
153
+ #
154
+ # @example
155
+ # # Given a class FuelConsumer in the namespace Space::Engineering:
156
+ # namespace.name
157
+ # #=> 'Space::Engineering'
158
+ # namespace.defined_classes
159
+ # #=> [{ 'name' => 'FuelConsumer', 'slug' => 'fuel-consumer' }]
160
+ #
161
+ # @return [Array<Hash>] the defined modules.
162
+ def defined_modules
163
+ @defined_modules ||=
164
+ native
165
+ .children
166
+ .select { |obj| obj.type == :module }
167
+ .reject { |obj| private_definition?(obj) }
168
+ .map { |obj| format_definition(obj) }
169
+ .sort_by { |hsh| hsh['name'] }
170
+ end
171
+
172
+ # Finds the instance attributes defined for the namespace.
173
+ #
174
+ # For each instance attribute, it returns a Hash with the following keys:
175
+ #
176
+ # - 'name': The name of the attribute.
177
+ # - 'read': True if the attribute defines a reader method.
178
+ # - 'write': True if the attribute defines a writer method.
179
+ # - 'path': The path to the reader method data file, or the writer method
180
+ # data file if the attribute does not define a reader method.
181
+ #
182
+ # @return [Array<Hash>] the instance attributes.
183
+ def instance_attributes
184
+ @instance_attributes ||=
185
+ find_instance_attributes(native)
186
+ .map { |name, options| format_attribute(name, options) }
187
+ .compact
188
+ .sort_by { |hsh| hsh['name'] }
189
+ end
190
+
191
+ # Finds the instance methods defined for the namespace.
192
+ #
193
+ # For each method, it returns a Hash with the following keys:
194
+ #
195
+ # - 'name': The name of the method, including trailing characters such as
196
+ # `=' or `?'.
197
+ # - 'path': The path to the method data file.
198
+ #
199
+ # @return [Array<Hash{String => String}>] the documented instance methods.
200
+ def instance_methods # rubocop:disable Metrics/CyclomaticComplexity
201
+ @instance_methods ||=
202
+ native
203
+ .meths
204
+ .select { |obj| obj.scope == :instance && !obj.is_attribute? }
205
+ .reject { |obj| private_method?(obj) }
206
+ .reject(&:is_alias?)
207
+ .map { |obj| format_method(obj) }
208
+ .sort_by { |hsh| hsh['name'] }
209
+ end
210
+
211
+ # The full, qualified name of the namespace.
212
+ #
213
+ # For the root namespace, should return an empty string. For a Class or a
214
+ # Module, should return the full name, e.g. "MyGem::MyModule::MyClass".
215
+ #
216
+ # @return [String] the qualified name.
217
+ def name
218
+ @name ||= native.path
219
+ end
220
+
221
+ # The name of the namespace in url-safe format.
222
+ #
223
+ # @return [String] the namespace name.
224
+ def slug
225
+ @slug ||= slugify(name.split('::').last || '')
226
+ end
227
+
228
+ # @return [String] the type of the namespace.
229
+ def type
230
+ 'namespace'
231
+ end
232
+
233
+ private
234
+
235
+ def find_class_attributes(native_object) # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
236
+ attributes = {}
237
+
238
+ if native_object.is_a?(YARD::CodeObjects::ClassObject)
239
+ ancestors = native_object.inheritance_tree - [native_object]
240
+ attributes = ancestors.reverse.reduce(attributes) do |hsh, obj|
241
+ next hsh if obj.is_a?(YARD::CodeObjects::Proxy)
242
+
243
+ hsh.merge(find_class_attributes(obj))
244
+ end
245
+ end
246
+
247
+ attributes = native_object.class_mixins.reverse.reduce(attributes) \
248
+ do |hsh, obj|
249
+ next hsh if obj.is_a?(YARD::CodeObjects::Proxy)
250
+
251
+ hsh.merge(find_instance_attributes(obj))
252
+ end
253
+
254
+ attributes
255
+ .transform_values { |attribute| attribute.merge(inherited: true) }
256
+ .merge(native_object.class_attributes)
257
+ end
258
+
259
+ def find_instance_attributes(native_object) # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
260
+ attributes = {}
261
+
262
+ if native_object.is_a?(YARD::CodeObjects::ClassObject)
263
+ ancestors = native_object.inheritance_tree - [native_object]
264
+ attributes = ancestors.reverse.reduce(attributes) do |hsh, obj|
265
+ next hsh if obj.is_a?(YARD::CodeObjects::Proxy)
266
+
267
+ hsh.merge(find_instance_attributes(obj))
268
+ end
269
+ end
270
+
271
+ attributes = native_object.instance_mixins.reverse.reduce(attributes) \
272
+ do |hsh, obj|
273
+ next hsh if obj.is_a?(YARD::CodeObjects::Proxy)
274
+
275
+ hsh.merge(find_instance_attributes(obj))
276
+ end
277
+
278
+ attributes
279
+ .transform_values { |attribute| attribute.merge(inherited: true) }
280
+ .merge(native_object.instance_attributes)
281
+ end
282
+
283
+ def format_attribute(name, options) # rubocop:disable Metrics/AbcSize, Metrics/MethodLength
284
+ return nil unless public_attribute?(options)
285
+
286
+ method_object =
287
+ SleepingKingStudios::Docs::Data::MethodObject
288
+ .new(native: options.values.find { |method| !method.nil? })
289
+
290
+ {
291
+ 'name' => name.to_s,
292
+ 'read' => !options[:read].nil?,
293
+ 'write' => !options[:write].nil?,
294
+ 'path' => method_object.data_path,
295
+ 'slug' => method_object.slug,
296
+ 'inherited' => !!options[:inherited] # rubocop:disable Style/DoubleNegation
297
+ }
298
+ end
299
+
300
+ def format_constant(native_constant)
301
+ constant_object =
302
+ SleepingKingStudios::Docs::Data::ConstantObject
303
+ .new(native: native_constant)
304
+
305
+ {
306
+ 'name' => native_constant.name.to_s,
307
+ 'path' => constant_object.data_path,
308
+ 'slug' => constant_object.slug,
309
+ 'inherited' => inherited_constant?(constant_object)
310
+ }
311
+ end
312
+
313
+ def format_definition(obj)
314
+ {
315
+ 'name' => obj.name.to_s,
316
+ 'slug' => slugify(obj.name)
317
+ }
318
+ end
319
+
320
+ def format_method(native_method) # rubocop:disable Metrics/MethodLength
321
+ method_object =
322
+ SleepingKingStudios::Docs::Data::MethodObject
323
+ .new(native: native_method)
324
+ hsh = {
325
+ 'name' => native_method.name.to_s,
326
+ 'path' => method_object.data_path,
327
+ 'slug' => method_object.slug,
328
+ 'inherited' => inherited_method?(method_object)
329
+ }
330
+
331
+ return hsh unless method_object.constructor?
332
+
333
+ hsh.merge('constructor' => true)
334
+ end
335
+
336
+ def inherited_constant?(_constant_object)
337
+ false
338
+ end
339
+
340
+ def inherited_method?(_method_object)
341
+ false
342
+ end
343
+
344
+ def private_constant?(constant_object)
345
+ return true unless constant_object.visibility == :public
346
+
347
+ constant_object.tags.any? { |tag| tag.tag_name == 'private' }
348
+ end
349
+
350
+ def private_definition?(definition_object)
351
+ definition_object.tags.any? { |tag| tag.tag_name == 'private' }
352
+ end
353
+
354
+ def private_method?(method_object)
355
+ return true unless method_object.visibility == :public
356
+
357
+ method_object.tags.any? { |tag| tag.tag_name == 'private' }
358
+ end
359
+
360
+ def public_attribute?(options)
361
+ options
362
+ .values
363
+ .select { |value| value.is_a?(YARD::CodeObjects::MethodObject) }
364
+ .any? { |obj| !private_method?(obj) }
365
+ end
366
+
367
+ def required_json
368
+ {
369
+ 'name' => name,
370
+ 'slug' => slug,
371
+ 'type' => type
372
+ }
373
+ end
374
+ end
375
+ end
@@ -0,0 +1,40 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'sleeping_king_studios/docs/data'
4
+
5
+ module SleepingKingStudios::Docs::Data
6
+ # Object representing the top-level Ruby namespace.
7
+ #
8
+ # Each namespace can define the following elements:
9
+ #
10
+ # - Definitions (classes and modules).
11
+ # - Constants.
12
+ # - Class attributes.
13
+ # - Class methods.
14
+ # - Instance attributes.
15
+ # - Instance methods.
16
+ #
17
+ # @see SleepingKingStudios::Docs::Data::NamespaceObject
18
+ class RootObject < SleepingKingStudios::Docs::Data::NamespaceObject
19
+ # The path to the data file.
20
+ #
21
+ # @return [String] the file path.
22
+ def data_path
23
+ 'root'
24
+ end
25
+
26
+ # The full, qualified name of the namespace.
27
+ #
28
+ # @return [String] the qualified name.
29
+ def name
30
+ 'root'
31
+ end
32
+
33
+ # The name of the namespace in url-safe format.
34
+ #
35
+ # @return [String] the namespace name.
36
+ def slug
37
+ 'root'
38
+ end
39
+ end
40
+ end
@@ -0,0 +1,35 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'sleeping_king_studios/docs/data/see_tags'
4
+
5
+ module SleepingKingStudios::Docs::Data::SeeTags
6
+ # Data object representing a plain text @see tag.
7
+ #
8
+ # @abstract
9
+ class Base < SleepingKingStudios::Docs::Data::Base
10
+ # @param native [YARD::Tags::Tag] the YARD object representing the @see tag.
11
+ # @param parent [YARD::Tags::Tag] the YARD object representing the parent
12
+ # object, which contains the @see tag.
13
+ def initialize(native:, parent:)
14
+ super(native:)
15
+
16
+ @parent = parent
17
+ end
18
+
19
+ # @return [YARD::Tags::Tag] the YARD object representing the parent object,
20
+ # which contains the @see tag.
21
+ attr_reader :parent
22
+
23
+ # The text to display when rendering the tag.
24
+ #
25
+ # @return [String] the text to display.
26
+ def text
27
+ @text ||= native.text
28
+ end
29
+
30
+ # @return [Boolean] true if the tag has display text, otherwise false.
31
+ def text?
32
+ !native.text.nil?
33
+ end
34
+ end
35
+ end