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.
- checksums.yaml +7 -0
- data/CHANGELOG.md +5 -0
- data/CODE_OF_CONDUCT.md +132 -0
- data/LICENSE +22 -0
- data/README.md +101 -0
- data/lib/sleeping_king_studios/docs/commands/generate.rb +268 -0
- data/lib/sleeping_king_studios/docs/commands/generators/base.rb +97 -0
- data/lib/sleeping_king_studios/docs/commands/generators/data_generator.rb +68 -0
- data/lib/sleeping_king_studios/docs/commands/generators/reference_generator.rb +59 -0
- data/lib/sleeping_king_studios/docs/commands/generators.rb +15 -0
- data/lib/sleeping_king_studios/docs/commands/installation/install_jekyll.rb +273 -0
- data/lib/sleeping_king_studios/docs/commands/installation/install_templates.rb +125 -0
- data/lib/sleeping_king_studios/docs/commands/installation/install_workflow.rb +121 -0
- data/lib/sleeping_king_studios/docs/commands/installation.rb +15 -0
- data/lib/sleeping_king_studios/docs/commands/parse.rb +51 -0
- data/lib/sleeping_king_studios/docs/commands/write_file.rb +89 -0
- data/lib/sleeping_king_studios/docs/commands.rb +14 -0
- data/lib/sleeping_king_studios/docs/data/base.rb +48 -0
- data/lib/sleeping_king_studios/docs/data/class_object.rb +119 -0
- data/lib/sleeping_king_studios/docs/data/constant_object.rb +161 -0
- data/lib/sleeping_king_studios/docs/data/metadata.rb +196 -0
- data/lib/sleeping_king_studios/docs/data/method_object.rb +555 -0
- data/lib/sleeping_king_studios/docs/data/module_object.rb +234 -0
- data/lib/sleeping_king_studios/docs/data/namespace_object.rb +375 -0
- data/lib/sleeping_king_studios/docs/data/root_object.rb +40 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/base.rb +35 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/class_method_tag.rb +99 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/constant_tag.rb +60 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/definition_tag.rb +52 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/instance_method_tag.rb +69 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/link_tag.rb +53 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/namespace_item_tag.rb +56 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/reference_tag.rb +92 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/text_tag.rb +30 -0
- data/lib/sleeping_king_studios/docs/data/see_tags/unstructured_tag.rb +37 -0
- data/lib/sleeping_king_studios/docs/data/see_tags.rb +101 -0
- data/lib/sleeping_king_studios/docs/data/types/grammar.treetop +49 -0
- data/lib/sleeping_king_studios/docs/data/types/key_value_type.rb +54 -0
- data/lib/sleeping_king_studios/docs/data/types/parameterized_type.rb +57 -0
- data/lib/sleeping_king_studios/docs/data/types/parser.rb +143 -0
- data/lib/sleeping_king_studios/docs/data/types/type.rb +100 -0
- data/lib/sleeping_king_studios/docs/data/types.rb +19 -0
- data/lib/sleeping_king_studios/docs/data.rb +29 -0
- data/lib/sleeping_king_studios/docs/errors/file_already_exists.rb +22 -0
- data/lib/sleeping_king_studios/docs/errors/file_error.rb +30 -0
- data/lib/sleeping_king_studios/docs/errors/file_not_found.rb +22 -0
- data/lib/sleeping_king_studios/docs/errors/invalid_directory.rb +22 -0
- data/lib/sleeping_king_studios/docs/errors/invalid_file.rb +22 -0
- data/lib/sleeping_king_studios/docs/errors.rb +19 -0
- data/lib/sleeping_king_studios/docs/registry.rb +22 -0
- data/lib/sleeping_king_studios/docs/registry_query.rb +93 -0
- data/lib/sleeping_king_studios/docs/tasks/base.rb +20 -0
- data/lib/sleeping_king_studios/docs/tasks/generate.rb +39 -0
- data/lib/sleeping_king_studios/docs/tasks/installation/install_jekyll.rb +67 -0
- data/lib/sleeping_king_studios/docs/tasks/installation/install_templates.rb +39 -0
- data/lib/sleeping_king_studios/docs/tasks/installation/install_workflow.rb +53 -0
- data/lib/sleeping_king_studios/docs/tasks/installation.rb +8 -0
- data/lib/sleeping_king_studios/docs/tasks/update.rb +35 -0
- data/lib/sleeping_king_studios/docs/tasks.rb +14 -0
- data/lib/sleeping_king_studios/docs/templates/config.yml.erb +22 -0
- data/lib/sleeping_king_studios/docs/templates/deploy-pages.yml.erb +67 -0
- data/lib/sleeping_king_studios/docs/templates/includes/breadcrumbs.md +7 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/attribute.md +23 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/attributes/heading.md +15 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/class.md +27 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/constant.md +8 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/constants/heading.md +6 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/constants/inherited.md +4 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/constants/overview.md +60 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/breadcrumbs.md +21 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/class_attributes.md +9 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/class_methods.md +9 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/constants.md +9 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/constructor.md +12 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/definitions.md +23 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/details.md +34 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/instance_attributes.md +9 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/instance_methods.md +17 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/overview.md +74 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/subclasses.md +10 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/class_attributes.md +10 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/class_methods.md +10 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/constants.md +10 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/instance_attributes.md +10 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/instance_methods.md +10 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents/overview.md +41 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/definitions/table-of-contents.md +27 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/method.md +27 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/aliases.md +4 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/heading.md +14 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/inherited.md +4 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/overload.md +18 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/overloads.md +12 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/overview.md +41 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/parameters.md +31 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/post_overview.md +51 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/raises.md +14 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/return_types.md +9 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/returns.md +14 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/methods/yields.md +42 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/module.md +23 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/namespace.md +33 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/parent_link.md +6 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/reference_link.md +7 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/see_link.md +7 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/type.md +14 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/type_list.md +4 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/types/array_type.md +1 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/types/hash_type.md +1 -0
- data/lib/sleeping_king_studios/docs/templates/includes/reference/types/ordered_type.md +1 -0
- data/lib/sleeping_king_studios/docs/templates/pages/index.md.erb +22 -0
- data/lib/sleeping_king_studios/docs/templates/pages/reference.md.erb +14 -0
- data/lib/sleeping_king_studios/docs/templates/pages/versions.md.erb +13 -0
- data/lib/sleeping_king_studios/docs/version.rb +59 -0
- data/lib/sleeping_king_studios/docs.rb +27 -0
- 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
|