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,89 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'fileutils'
4
+
5
+ require 'cuprum/command'
6
+ require 'cuprum/exception_handling'
7
+
8
+ require 'sleeping_king_studios/docs/commands'
9
+
10
+ module SleepingKingStudios::Docs::Commands
11
+ # Writes the given contents to a file.
12
+ class WriteFile < Cuprum::Command
13
+ include Cuprum::ExceptionHandling
14
+
15
+ # @!method call(contents:, file_path:)
16
+ # Writes the given contents to the specified file.
17
+ #
18
+ # Recursively generates the required directories, if needed.
19
+ #
20
+ # @param contents [String] the contents to write.
21
+ # @param file_path [String] the path of the file.
22
+ #
23
+ # @return [Cuprum::Result] a passing result if the directories are
24
+ # successfully created and the file is successfully written.
25
+ # @return [Cuprum::Result<Cuprum::Error>] a failing result if the command
26
+ # was unable to create the directories or write the file.
27
+
28
+ # @param force [Boolean] if true, overwrites an existing file.
29
+ def initialize(force: false)
30
+ super()
31
+
32
+ @force = force
33
+ end
34
+
35
+ # @return [Boolean] if true, overwrites an existing file.
36
+ def force?
37
+ @force
38
+ end
39
+
40
+ private
41
+
42
+ def check_if_file_exists(file_path:)
43
+ return if force?
44
+ return unless File.exist?(file_path)
45
+
46
+ error =
47
+ SleepingKingStudios::Docs::Errors::FileAlreadyExists
48
+ .new(path: file_path)
49
+
50
+ failure(error)
51
+ end
52
+
53
+ def create_directory(file_path:)
54
+ dir_path = File.dirname(file_path)
55
+
56
+ FileUtils.mkdir_p(dir_path)
57
+ rescue Errno::EEXIST => exception
58
+ failure(invalid_directory_error(exception))
59
+ end
60
+
61
+ def create_file(contents:, file_path:)
62
+ File.write(file_path, contents)
63
+ rescue Errno::EISDIR => exception
64
+ failure(invalid_file_error(exception))
65
+ end
66
+
67
+ def invalid_directory_error(exception)
68
+ dir_path = exception.message.split(' - ').last
69
+
70
+ SleepingKingStudios::Docs::Errors::InvalidDirectory.new(path: dir_path)
71
+ end
72
+
73
+ def invalid_file_error(exception)
74
+ file_path = exception.message.split(' - ').last
75
+
76
+ SleepingKingStudios::Docs::Errors::InvalidFile.new(path: file_path)
77
+ end
78
+
79
+ def process(contents:, file_path:)
80
+ file_path = File.expand_path(file_path)
81
+
82
+ step { check_if_file_exists(file_path:) }
83
+
84
+ step { create_directory(file_path:) }
85
+
86
+ step { create_file(contents:, file_path:) }
87
+ end
88
+ end
89
+ end
@@ -0,0 +1,14 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'sleeping_king_studios/docs'
4
+
5
+ module SleepingKingStudios::Docs
6
+ # Namespace for commands, which parse and generate documentation files.
7
+ module Commands
8
+ autoload :Generate, 'sleeping_king_studios/docs/commands/generate'
9
+ autoload :Generators, 'sleeping_king_studios/docs/commands/generators'
10
+ autoload :Installation, 'sleeping_king_studios/docs/commands/installation'
11
+ autoload :Parse, 'sleeping_king_studios/docs/commands/parse'
12
+ autoload :WriteFile, 'sleeping_king_studios/docs/commands/write_file'
13
+ end
14
+ end
@@ -0,0 +1,48 @@
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
+ # Base object representing a Ruby object documented in YARD.
9
+ #
10
+ # @abstract
11
+ class Base
12
+ # @param native [YARD::Tags::Tag] the YARD object representing the
13
+ # documented object.
14
+ def initialize(native:)
15
+ @native = native
16
+ @registry = SleepingKingStudios::Docs::Registry.instance
17
+ end
18
+
19
+ # Generates a JSON-compatible representation of the object.
20
+ #
21
+ # @return [Hash] the JSON representation.
22
+ def as_json
23
+ {}
24
+ end
25
+
26
+ private
27
+
28
+ attr_reader :native
29
+
30
+ attr_reader :registry
31
+
32
+ def empty?(value)
33
+ return true if value.nil?
34
+
35
+ return false unless value.respond_to?(:empty?)
36
+
37
+ value.empty?
38
+ end
39
+
40
+ def slugify(str)
41
+ tools.string_tools.underscore(str).tr('_', '-')
42
+ end
43
+
44
+ def tools
45
+ SleepingKingStudios::Tools::Toolbelt.instance
46
+ end
47
+ end
48
+ end
@@ -0,0 +1,119 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'sleeping_king_studios/docs/data'
4
+
5
+ module SleepingKingStudios::Docs::Data
6
+ # Object representing a Ruby class.
7
+ #
8
+ # Each class can define the following elements:
9
+ #
10
+ # - Defined In files.
11
+ # - inherited classes.
12
+ # - extend-ed modules.
13
+ # - include-ed modules.
14
+ # - Definitions (classes and modules).
15
+ # - Constants.
16
+ # - Class attributes.
17
+ # - Class methods.
18
+ # - A constructor method.
19
+ # - Instance attributes.
20
+ # - Instance methods.
21
+ # - Known subclasses.
22
+ #
23
+ # Additionally, a class can have a description and metadata tags.
24
+ #
25
+ # @see SleepingKingStudios::Docs::Data::Metadata
26
+ # @see SleepingKingStudios::Docs::Data::ModuleObject
27
+ class ClassObject < ModuleObject
28
+ JSON_PROPERTIES = %i[
29
+ direct_subclasses
30
+ inherited_classes
31
+ ].freeze
32
+ private_constant :JSON_PROPERTIES
33
+
34
+ # Generates a JSON-compatible representation of the module.
35
+ #
36
+ # Returns a Hash with the following keys:
37
+ #
38
+ # - 'name': The full, qualified name of the module.
39
+ # - 'slug': The name of the module in url-safe format.
40
+ # - 'files': A list of the files where the module is defined.
41
+ # - 'short_description': A short description of the module.
42
+ # - 'constructor': True if the class defines a constructor, otherwise false.
43
+ #
44
+ # Additionally, the returned Hash will conditionally include the following
45
+ # keys, if the module defines at least one of the corresponding code
46
+ # objects.
47
+ #
48
+ # - 'class_attributes': The class attributes, if any.
49
+ # - 'class_methods': The class methods, if any.
50
+ # - 'constants': The constants, if any.
51
+ # - 'defined_classes': The defined Classes, if any.
52
+ # - 'defined_modules': The defined Modules, if any.
53
+ # - 'description': The full description of the module, minus the first
54
+ # clause.
55
+ # - 'extended_modules': A list of the modules that extend the module.
56
+ # - 'included_modules': A list of the modules that are included in the
57
+ # module.
58
+ # - 'instance_attributes': The instance attributes, if any.
59
+ # - 'instance_methods': The instance methods, if any.
60
+ # - 'metadata': Additional metadata tags from the documentation.
61
+ #
62
+ # @return [Hash{String => Object}] the representation of the module.
63
+ def as_json
64
+ JSON_PROPERTIES.reduce(super.merge('constructor' => constructor?)) \
65
+ do |memo, property_name|
66
+ value = send(property_name)
67
+
68
+ next memo if empty?(value)
69
+
70
+ memo.update(property_name.to_s => value)
71
+ end
72
+ end
73
+
74
+ # @return [Boolean] true if the class defines a constructor, otherwise
75
+ # false.
76
+ def constructor?
77
+ native.meths.any? { |meth| meth.name == :initialize }
78
+ end
79
+
80
+ # A list of the direct subclasses of the class.
81
+ #
82
+ # For each subclass, it returns a Hash with the following keys:
83
+ #
84
+ # - 'name': The name of the inherited class.
85
+ # - 'slug': A url-safe, hyphen-separated representation of the name.
86
+ # - 'path': The path to the data file for the class.
87
+ #
88
+ # @return [Array<Hash{String => String}>] the direct subclasses.
89
+ def direct_subclasses
90
+ registry
91
+ .select do |obj|
92
+ obj.type == :class && obj.inheritance_tree[1] == native
93
+ end
94
+ .map { |obj| format_inclusion(obj) }
95
+ .sort_by { |hsh| hsh['name'] }
96
+ end
97
+
98
+ # A list of the classes that are inherited by the class.
99
+ #
100
+ # For each inherited Class, it returns a Hash with the following keys:
101
+ #
102
+ # - 'name': The name of the inherited class.
103
+ # - 'slug': A url-safe, hyphen-separated representation of the name.
104
+ # - 'path': The path to the data file for the class.
105
+ #
106
+ # @return [Array<Hash{String => String}>] the inherited classes.
107
+ def inherited_classes
108
+ @inherited_classes ||=
109
+ native
110
+ .inheritance_tree[1..]
111
+ .map { |obj| format_inclusion(obj) }
112
+ end
113
+
114
+ # @return [String] the type of the namespace.
115
+ def type
116
+ 'class'
117
+ end
118
+ end
119
+ end
@@ -0,0 +1,161 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'sleeping_king_studios/docs/data'
4
+
5
+ module SleepingKingStudios::Docs::Data
6
+ # Object representing a Ruby constant.
7
+ #
8
+ # Each constant has a name and a value. In addition, a constant may have a
9
+ # short description, a full description, and metadata.
10
+ class ConstantObject < SleepingKingStudios::Docs::Data::Base
11
+ JSON_PROPERTIES = %i[
12
+ data_path
13
+ description
14
+ metadata
15
+ ].freeze
16
+ private_constant :JSON_PROPERTIES
17
+
18
+ PARAGRAPH_BREAK = /\n{2,}/
19
+ private_constant :PARAGRAPH_BREAK
20
+
21
+ # Generates a JSON-compatible representation of the constant.
22
+ #
23
+ # Returns a Hash with the following keys:
24
+ #
25
+ # - 'name': The full, qualified name of the constant.
26
+ # - 'slug': The name of the constant in url-safe format.
27
+ # - 'value': A String representation of the value of the constant.
28
+ # - 'short_description': A short description of the constant.
29
+ #
30
+ # Additionally, the returned Hash will conditionally include the following
31
+ # keys, if the constant defines at least one of the corresponding code
32
+ # objects.
33
+ #
34
+ # - 'description': The full description of the constant, minus the first
35
+ # clause.
36
+ # - 'metadata': Additional metadata tags from the documentation.
37
+ #
38
+ # @return [Hash{String => Object}] the representation of the constant.
39
+ def as_json
40
+ JSON_PROPERTIES.reduce(required_json) do |memo, property_name|
41
+ value = send(property_name)
42
+
43
+ next memo if empty?(value)
44
+
45
+ memo.update(property_name.to_s => value)
46
+ end
47
+ end
48
+
49
+ # The path to the data file.
50
+ #
51
+ # @return [String] the file path.
52
+ def data_path
53
+ @data_path ||= name.split('::').map { |str| slugify(str) }.join('/')
54
+ end
55
+
56
+ # The full description of the constant, minus the first clause.
57
+ #
58
+ # The remainder of the constant description, if any, after subtracting the
59
+ # short description (separated by the first paragraph break).
60
+ #
61
+ # @return [String] the remaining description.
62
+ #
63
+ # @see #short_description.
64
+ def description
65
+ return @description if @description
66
+
67
+ @short_description, @description = split_docstring
68
+
69
+ @description
70
+ end
71
+
72
+ # Additional metadata tags from the documentation.
73
+ #
74
+ # @see SleepingKingStudios::Docs::Data::Metadata.
75
+ def metadata
76
+ @metadata ||= format_metadata
77
+ end
78
+
79
+ # The full, qualified name of the constant.
80
+ #
81
+ # @return [String] the qualified name.
82
+ def name
83
+ @name ||= native.path
84
+ end
85
+
86
+ # The path to the defining class or module's data file.
87
+ #
88
+ # @return [String] the file path.
89
+ def parent_path
90
+ return @parent_path if @parent_path
91
+
92
+ return @parent_path = '' if native.parent.root?
93
+
94
+ parent_class =
95
+ if native.parent.is_a?(YARD::CodeObjects::ClassObject)
96
+ SleepingKingStudios::Docs::Data::ClassObject
97
+ else
98
+ SleepingKingStudios::Docs::Data::ModuleObject
99
+ end
100
+ parent_object = parent_class.new(native: native.parent)
101
+
102
+ @parent_path = parent_object.data_path
103
+ end
104
+
105
+ # A short description of the constant.
106
+ #
107
+ # The first part of the constant description, separated by the first
108
+ # paragraph break. Typically should fit on a single line of text.
109
+ #
110
+ # @return [String] the short description.
111
+ #
112
+ # @see #description.
113
+ def short_description
114
+ return @short_description if @short_description
115
+
116
+ @short_description, @description = split_docstring
117
+
118
+ @short_description
119
+ end
120
+
121
+ # The name of the constant in url-safe format.
122
+ #
123
+ # @return [String] the constant name.
124
+ def slug
125
+ @slug ||= slugify(name.split('::').last)
126
+ end
127
+
128
+ # A String representation of the value of the constant.
129
+ #
130
+ # @return [String] the constant value.
131
+ def value
132
+ native.value
133
+ end
134
+
135
+ private
136
+
137
+ def required_json
138
+ {
139
+ 'name' => name,
140
+ 'parent_path' => parent_path,
141
+ 'slug' => slug,
142
+ 'short_description' => short_description,
143
+ 'value' => value
144
+ }
145
+ end
146
+
147
+ def format_metadata
148
+ SleepingKingStudios::Docs::Data::Metadata
149
+ .new(native:)
150
+ .as_json
151
+ end
152
+
153
+ def split_docstring
154
+ match = native.docstring.match(PARAGRAPH_BREAK)
155
+
156
+ return native.docstring.to_s unless match
157
+
158
+ [match.pre_match.to_s, match.post_match.to_s]
159
+ end
160
+ end
161
+ end
@@ -0,0 +1,196 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'set'
4
+
5
+ require 'sleeping_king_studios/docs/data'
6
+
7
+ module SleepingKingStudios::Docs::Data
8
+ # Object representing the metadata tags for a Ruby module.
9
+ #
10
+ # Each module can have the following metadata tags:
11
+ #
12
+ # - @example
13
+ # - @note
14
+ # - @see
15
+ # - @todo
16
+ #
17
+ # Other tags are not currently supported.
18
+ #
19
+ # @see SleepingKingStudios::Docs::Data::ModuleObject.
20
+ # @see SleepingKingStudios::Docs::Data::SeeTags.
21
+ class Metadata < SleepingKingStudios::Docs::Data::Base
22
+ EMPTYABLE_PROPERTIES = Set.new(
23
+ %i[
24
+ abstract
25
+ deprecated
26
+ ]
27
+ ).freeze
28
+ private_constant :EMPTYABLE_PROPERTIES
29
+
30
+ METADATA_PROPERTIES = %i[
31
+ abstract
32
+ api
33
+ authors
34
+ deprecated
35
+ examples
36
+ notes
37
+ see
38
+ since
39
+ todos
40
+ versions
41
+ ].freeze
42
+ private_constant :METADATA_PROPERTIES
43
+
44
+ # Returns the @abstract tag, if any, defined for the module.
45
+ #
46
+ # @return [String] the contents of the @abstract tag, or an empty string if
47
+ # the tag does not have contents.
48
+ # @return [nil] if there is no @abstract tag defined.
49
+ def abstract
50
+ @abstract ||=
51
+ native
52
+ .tags
53
+ .find { |tag| tag.tag_name == 'abstract' }
54
+ &.text
55
+ end
56
+
57
+ # Returns the @api tag, if any, defined for the module.
58
+ #
59
+ # @return [String] the contents of the @api tag.
60
+ # @return [nil] if there is no @api tag defined.
61
+ def api
62
+ @api ||=
63
+ native
64
+ .tags
65
+ .find { |tag| tag.tag_name == 'api' }
66
+ &.text
67
+ end
68
+
69
+ # Generates a JSON-compatible representation of the metadata.
70
+ #
71
+ # Returns a Hash with zero or more of the following keys:
72
+ #
73
+ # - 'examples': An Array of Hashes with keys 'name' and 'text' and String
74
+ # values.
75
+ # - 'notes': An Array of Strings.
76
+ # - 'see': An Array of Hashes with String values.
77
+ # - 'todo': An Array of Strings.
78
+ #
79
+ # @return [Hash{String => Object}] the representation of the metadata.
80
+ def as_json
81
+ METADATA_PROPERTIES
82
+ .reduce({}) do |memo, property_name|
83
+ value = send(property_name)
84
+
85
+ next memo if emptyable?(property_name) ? value.nil? : empty?(value)
86
+
87
+ memo.update(property_name.to_s => value)
88
+ end
89
+ end
90
+
91
+ # Collects the @author tags defined for the module.
92
+ #
93
+ # @return [Array<String>] the collected authors.
94
+ def authors
95
+ @authors ||=
96
+ select_tags('author') { |tag| !tag.text.empty? }
97
+ .map(&:text)
98
+ end
99
+
100
+ # Returns the @deprecated tag, if any, defined for the module.
101
+ #
102
+ # @return [String] the contents of the @deprecated tag, or an empty string
103
+ # if the tag does not have contents.
104
+ # @return [nil] if there is no @deprecated tag defined.
105
+ def deprecated
106
+ @deprecated ||=
107
+ native
108
+ .tags
109
+ .find { |tag| tag.tag_name == 'deprecated' }
110
+ &.text
111
+ end
112
+
113
+ # Collects the @example tags defined for the module.
114
+ #
115
+ # Each @example tag is represented as a Hash with the following keys:
116
+ #
117
+ # - 'name': The displayed name of the example. May be an empty String.
118
+ # - 'text': The text for the example.
119
+ #
120
+ # @return [Array<Hash{String => String}>] the collected examples.
121
+ def examples
122
+ @examples ||=
123
+ select_tags('example')
124
+ .map { |tag| { 'name' => tag.name, 'text' => tag.text } }
125
+ end
126
+
127
+ # Collects the @note tags defined for the module.
128
+ #
129
+ # @return [Array<String>] the collected notes.
130
+ def notes
131
+ @notes ||= select_tags('note').map(&:text)
132
+ end
133
+
134
+ # Collects the @see tags defined the for the module.
135
+ #
136
+ # Each @see tag is represented as a Hash with String keys and values, and
137
+ # has at a minimum the 'text' key. See the SeeTag#as_json method for
138
+ # details.
139
+ #
140
+ # @return [Array<Hash{String => String}>]
141
+ def see
142
+ @see ||= select_tags('see').map { |tag| format_see_tag(tag) }
143
+ end
144
+
145
+ # Collects the @since tags defined for the module.
146
+ #
147
+ # @return [Array<String>] the collected @since tags.
148
+ def since
149
+ @since ||=
150
+ select_tags('since') { |tag| !tag.text.empty? }
151
+ .map(&:text)
152
+ end
153
+
154
+ # Collects the @todo tags defined for the module.
155
+ #
156
+ # @return [Array<String>] the collected todos.
157
+ def todos
158
+ @todos ||= select_tags('todo').map(&:text)
159
+ end
160
+
161
+ # Collects the @version tags defined for the module.
162
+ #
163
+ # @return [Array<String>] the collected @version tags.
164
+ def versions
165
+ @versions ||=
166
+ select_tags('version') { |tag| !tag.text.empty? }
167
+ .map(&:text)
168
+ end
169
+
170
+ private
171
+
172
+ def definition_tag?(tag)
173
+ tag.type == :class || tag.type == :module # rubocop:disable Style/MultipleComparison
174
+ end
175
+
176
+ def emptyable?(property_name)
177
+ EMPTYABLE_PROPERTIES.include?(property_name)
178
+ end
179
+
180
+ def format_see_tag(tag)
181
+ SleepingKingStudios::Docs::Data::SeeTags
182
+ .build(
183
+ native: tag,
184
+ parent: definition_tag?(native) ? native : native.parent
185
+ )
186
+ .as_json
187
+ end
188
+
189
+ def select_tags(tag_name)
190
+ native.tags.select do |tag|
191
+ (tag.tag_name == tag_name) &&
192
+ (!block_given? || yield(tag))
193
+ end
194
+ end
195
+ end
196
+ end