dox 1.3.0 → 2.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 +5 -5
- data/.rubocop.yml +6 -3
- data/.ruby-version +1 -1
- data/.travis.yml +7 -11
- data/CHANGES.md +19 -1
- data/Gemfile +1 -1
- data/README.md +109 -124
- data/Rakefile +3 -3
- data/bin/console +3 -3
- data/dox.gemspec +14 -13
- data/lib/dox.rb +9 -1
- data/lib/dox/config.rb +39 -5
- data/lib/dox/dsl/action.rb +10 -4
- data/lib/dox/dsl/documentation.rb +4 -0
- data/lib/dox/dsl/resource.rb +0 -2
- data/lib/dox/dsl/syntax.rb +1 -0
- data/lib/dox/entities/action.rb +33 -13
- data/lib/dox/entities/example.rb +11 -1
- data/lib/dox/entities/resource.rb +4 -6
- data/lib/dox/entities/resource_group.rb +2 -3
- data/lib/dox/formatter.rb +10 -11
- data/lib/dox/formatters/multipart.rb +2 -0
- data/lib/dox/formatters/plain.rb +5 -1
- data/lib/dox/printers/action_printer.rb +18 -27
- data/lib/dox/printers/base_printer.rb +36 -17
- data/lib/dox/printers/document_printer.rb +37 -7
- data/lib/dox/printers/example_request_printer.rb +69 -0
- data/lib/dox/printers/example_response_printer.rb +86 -0
- data/lib/dox/printers/resource_group_printer.rb +7 -7
- data/lib/dox/printers/resource_printer.rb +12 -7
- data/lib/dox/util/http.rb +64 -0
- data/lib/dox/version.rb +1 -1
- metadata +78 -21
- data/lib/dox/printers/example_printer.rb +0 -100
data/lib/dox/formatters/plain.rb
CHANGED
@@ -2,7 +2,11 @@ module Dox
|
|
2
2
|
module Formatters
|
3
3
|
class Plain < Dox::Formatters::Base
|
4
4
|
def format
|
5
|
-
body
|
5
|
+
return body if body.encoding == Encoding::UTF_8
|
6
|
+
|
7
|
+
body.encode(Encoding::UTF_8)
|
8
|
+
rescue Encoding::UndefinedConversionError
|
9
|
+
"#{body.encoding} stream"
|
6
10
|
end
|
7
11
|
end
|
8
12
|
end
|
@@ -3,44 +3,35 @@ module Dox
|
|
3
3
|
class ActionPrinter < BasePrinter
|
4
4
|
def print(action)
|
5
5
|
self.action = action
|
6
|
-
@
|
7
|
-
@output.puts action_uri_params if action.uri_params.present?
|
6
|
+
@action_hash = find_or_add(find_or_add(spec, action.path.to_s), action.verb.downcase.to_sym)
|
8
7
|
|
9
|
-
|
10
|
-
|
11
|
-
|
8
|
+
add_action
|
9
|
+
add_action_params
|
10
|
+
|
11
|
+
print_examples
|
12
12
|
end
|
13
13
|
|
14
14
|
private
|
15
15
|
|
16
|
-
attr_accessor :action
|
17
|
-
|
18
|
-
def action_title
|
19
|
-
<<-HEREDOC
|
16
|
+
attr_accessor :action, :action_hash
|
20
17
|
|
21
|
-
|
22
|
-
|
23
|
-
|
18
|
+
def add_action
|
19
|
+
action_hash['summary'] = action.name
|
20
|
+
action_hash['tags'] = [action.resource]
|
21
|
+
action_hash['description'] = format_desc(action.desc)
|
24
22
|
end
|
25
23
|
|
26
|
-
def
|
27
|
-
|
28
|
-
+ Parameters
|
29
|
-
#{formatted_params(action.uri_params)}
|
30
|
-
HEREDOC
|
31
|
-
end
|
24
|
+
def add_action_params
|
25
|
+
return unless action.params.present?
|
32
26
|
|
33
|
-
|
34
|
-
@example_printer ||= ExamplePrinter.new(@output)
|
27
|
+
action_hash['parameters'] = action.params
|
35
28
|
end
|
36
29
|
|
37
|
-
def
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
desc
|
43
|
-
end.flatten.join("\n")
|
30
|
+
def print_examples
|
31
|
+
action.examples.each do |example|
|
32
|
+
ExampleRequestPrinter.new(action_hash).print(example)
|
33
|
+
ExampleResponsePrinter.new(action_hash).print(example)
|
34
|
+
end
|
44
35
|
end
|
45
36
|
end
|
46
37
|
end
|
@@ -1,37 +1,56 @@
|
|
1
|
+
require 'rexml/document'
|
2
|
+
|
1
3
|
module Dox
|
2
4
|
module Printers
|
3
5
|
class BasePrinter
|
4
|
-
|
5
|
-
|
6
|
+
attr_reader :spec
|
7
|
+
|
8
|
+
def initialize(spec)
|
9
|
+
@spec = spec || {}
|
6
10
|
end
|
7
11
|
|
8
12
|
def print
|
9
13
|
raise NotImplementedError
|
10
14
|
end
|
11
15
|
|
12
|
-
|
16
|
+
def find_or_add(hash, key, default = {})
|
17
|
+
return hash[key] if hash.key?(key)
|
13
18
|
|
14
|
-
|
15
|
-
Dox.config.desc_folder_path
|
19
|
+
hash[key] = default
|
16
20
|
end
|
17
21
|
|
18
|
-
def
|
19
|
-
return
|
22
|
+
def read_file(path, root_path: Dox.config.descriptions_location)
|
23
|
+
return '' unless root_path
|
24
|
+
|
25
|
+
File.read(File.join(root_path, path))
|
26
|
+
end
|
20
27
|
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
content(path)
|
28
|
+
def formatted_body(body_str, content_type)
|
29
|
+
case content_type
|
30
|
+
when %r{application\/.*json}
|
31
|
+
JSON.parse(body_str)
|
32
|
+
when /xml/
|
33
|
+
pretty_xml(body_str)
|
28
34
|
else
|
29
|
-
|
35
|
+
body_str
|
30
36
|
end
|
31
37
|
end
|
32
38
|
|
33
|
-
def
|
34
|
-
|
39
|
+
def pretty_xml(xml_string)
|
40
|
+
doc = REXML::Document.new(xml_string)
|
41
|
+
formatter = REXML::Formatters::Pretty.new
|
42
|
+
formatter.compact = true
|
43
|
+
result = ''
|
44
|
+
formatter.write(doc, result)
|
45
|
+
result
|
46
|
+
end
|
47
|
+
|
48
|
+
def format_desc(description)
|
49
|
+
desc = description
|
50
|
+
desc = '' if desc.nil?
|
51
|
+
desc = read_file(desc) if desc.end_with?('.md')
|
52
|
+
|
53
|
+
desc
|
35
54
|
end
|
36
55
|
end
|
37
56
|
end
|
@@ -1,26 +1,56 @@
|
|
1
1
|
module Dox
|
2
2
|
module Printers
|
3
3
|
class DocumentPrinter < BasePrinter
|
4
|
+
def initialize(output)
|
5
|
+
super(body)
|
6
|
+
@output = output
|
7
|
+
end
|
8
|
+
|
4
9
|
def print(passed_examples)
|
5
|
-
|
10
|
+
spec['paths'] = {}
|
11
|
+
spec['tags'] = []
|
12
|
+
spec['x-tagGroups'] = []
|
6
13
|
|
7
14
|
passed_examples.sort.each do |_, resource_group|
|
8
15
|
group_printer.print(resource_group)
|
9
16
|
end
|
17
|
+
|
18
|
+
order_groups
|
19
|
+
|
20
|
+
@output.puts(JSON.pretty_generate(spec))
|
10
21
|
end
|
11
22
|
|
12
23
|
private
|
13
24
|
|
14
|
-
def
|
15
|
-
|
25
|
+
def body
|
26
|
+
{
|
27
|
+
openapi: Dox.config.openapi_version || '3.0.0',
|
28
|
+
info: {
|
29
|
+
title: Dox.config.title || 'API Documentation',
|
30
|
+
description: adjust_description(Dox.config.header_description || ''),
|
31
|
+
version: Dox.config.api_version || '1.0'
|
32
|
+
}
|
33
|
+
}
|
34
|
+
end
|
35
|
+
|
36
|
+
def adjust_description(description)
|
37
|
+
description.end_with?('.md') ? acquire_desc(description) : description
|
38
|
+
end
|
39
|
+
|
40
|
+
def acquire_desc(path)
|
41
|
+
read_file(path)
|
16
42
|
end
|
17
43
|
|
18
|
-
def
|
19
|
-
@
|
44
|
+
def group_printer
|
45
|
+
@group_printer ||= ResourceGroupPrinter.new(spec)
|
20
46
|
end
|
21
47
|
|
22
|
-
def
|
23
|
-
Dox.config.
|
48
|
+
def order_groups
|
49
|
+
return if (Dox.config.groups_order || []).empty?
|
50
|
+
|
51
|
+
spec['x-tagGroups'] = spec['x-tagGroups'].sort_by do |tag|
|
52
|
+
Dox.config.groups_order.index(tag[:name]) || 100
|
53
|
+
end
|
24
54
|
end
|
25
55
|
end
|
26
56
|
end
|
@@ -0,0 +1,69 @@
|
|
1
|
+
module Dox
|
2
|
+
module Printers
|
3
|
+
class ExampleRequestPrinter < BasePrinter
|
4
|
+
def print(example)
|
5
|
+
self.example = example
|
6
|
+
add_example_request
|
7
|
+
end
|
8
|
+
|
9
|
+
private
|
10
|
+
|
11
|
+
attr_accessor :example
|
12
|
+
|
13
|
+
def add_example_request
|
14
|
+
spec['parameters'] = add_new_header_params(find_or_add(spec, 'parameters', []))
|
15
|
+
return if example.request_body.empty?
|
16
|
+
|
17
|
+
add_content(find_or_add(spec, 'requestBody'))
|
18
|
+
end
|
19
|
+
|
20
|
+
def add_content(body)
|
21
|
+
add_content_name(body['content'] = find_or_add(body, 'content'))
|
22
|
+
end
|
23
|
+
|
24
|
+
def add_content_name(body)
|
25
|
+
req_header = find_headers(example.request_headers)
|
26
|
+
add_example(body[req_header] = find_or_add(body, req_header))
|
27
|
+
add_schema(body[req_header], Dox.config.schema_request_folder_path)
|
28
|
+
end
|
29
|
+
|
30
|
+
def add_example(body)
|
31
|
+
add_desc(body['examples'] = find_or_add(body, 'examples'))
|
32
|
+
end
|
33
|
+
|
34
|
+
def add_desc(body)
|
35
|
+
body[example.desc] = { 'summary' => example.desc,
|
36
|
+
'value' => formatted_body(example.request_body,
|
37
|
+
example.request_content_type) }
|
38
|
+
end
|
39
|
+
|
40
|
+
def add_schema(body, path)
|
41
|
+
return if example.request_schema.nil?
|
42
|
+
return unless path
|
43
|
+
|
44
|
+
file_path = File.join(path, "#{example.request_schema}.json")
|
45
|
+
|
46
|
+
body['schema'] = File.file?(file_path) ? { '$ref' => file_path } : JSON.parse(example.request_schema)
|
47
|
+
end
|
48
|
+
|
49
|
+
def find_headers(headers)
|
50
|
+
headers.find { |key, _| key == 'Accept' }&.last || 'any'
|
51
|
+
end
|
52
|
+
|
53
|
+
def acquire_header_params
|
54
|
+
example.request_headers.map do |key, value|
|
55
|
+
{ name: key, in: :header, example: value }
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
def add_new_header_params(header_params)
|
60
|
+
example.request_headers.each do |key, value|
|
61
|
+
header_params.push(name: key, in: :header, example: value) unless
|
62
|
+
header_params.detect { |hash| hash[:name] == key }
|
63
|
+
end
|
64
|
+
|
65
|
+
header_params
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
69
|
+
end
|
@@ -0,0 +1,86 @@
|
|
1
|
+
module Dox
|
2
|
+
module Printers
|
3
|
+
class ExampleResponsePrinter < BasePrinter
|
4
|
+
def print(example)
|
5
|
+
self.example = example
|
6
|
+
add_example_response
|
7
|
+
end
|
8
|
+
|
9
|
+
private
|
10
|
+
|
11
|
+
attr_accessor :example
|
12
|
+
|
13
|
+
def add_example_response
|
14
|
+
add_statuses(find_or_add(find_or_add(spec, 'responses'), example.response_status.to_s))
|
15
|
+
end
|
16
|
+
|
17
|
+
def add_statuses(body)
|
18
|
+
add_status_desc(body)
|
19
|
+
add_content(body)
|
20
|
+
add_headers(body)
|
21
|
+
end
|
22
|
+
|
23
|
+
def add_status_desc(body)
|
24
|
+
body['description'] = Util::Http::HTTP_STATUS_CODES[example.response_status]
|
25
|
+
end
|
26
|
+
|
27
|
+
def add_content(body)
|
28
|
+
add_content_name(body['content'] = find_or_add(body, 'content'))
|
29
|
+
end
|
30
|
+
|
31
|
+
def add_content_name(body)
|
32
|
+
resp_header = find_headers(example.response_headers)
|
33
|
+
|
34
|
+
add_example(body[resp_header] = find_or_add(body, resp_header))
|
35
|
+
add_schema(body[resp_header], Dox.config.schema_response_folder_path)
|
36
|
+
end
|
37
|
+
|
38
|
+
def add_example(body)
|
39
|
+
return if example.response_body.empty?
|
40
|
+
|
41
|
+
add_desc(body['examples'] = find_or_add(body, 'examples'))
|
42
|
+
end
|
43
|
+
|
44
|
+
def add_desc(body)
|
45
|
+
body[example.desc] = { 'summary' => example.desc,
|
46
|
+
'value' => formatted_body(example.response_body,
|
47
|
+
find_headers(example.response_headers)) }
|
48
|
+
end
|
49
|
+
|
50
|
+
def add_schema(body, path)
|
51
|
+
return unless path
|
52
|
+
|
53
|
+
schema = find_schema
|
54
|
+
|
55
|
+
return unless schema
|
56
|
+
|
57
|
+
add_schema_to_hash(body, path, schema)
|
58
|
+
end
|
59
|
+
|
60
|
+
def find_schema
|
61
|
+
if example.response_success?
|
62
|
+
example.response_schema_success
|
63
|
+
else
|
64
|
+
example.response_schema_fail || Dox.config.schema_response_fail_file_path
|
65
|
+
end
|
66
|
+
end
|
67
|
+
|
68
|
+
def add_schema_to_hash(body, path, schema)
|
69
|
+
body['schema'] =
|
70
|
+
if schema.is_a?(Pathname)
|
71
|
+
{ '$ref' => schema }
|
72
|
+
else
|
73
|
+
{ '$ref' => File.join(path, "#{schema}.json") }
|
74
|
+
end
|
75
|
+
end
|
76
|
+
|
77
|
+
def find_headers(headers)
|
78
|
+
headers.find { |key, _| key == 'Content-Type' }&.last || 'any'
|
79
|
+
end
|
80
|
+
|
81
|
+
def add_headers(body)
|
82
|
+
body['headers'] = Hash[example.response_headers.map { |key, value| [key, { description: value }] }]
|
83
|
+
end
|
84
|
+
end
|
85
|
+
end
|
86
|
+
end
|
@@ -3,7 +3,7 @@ module Dox
|
|
3
3
|
class ResourceGroupPrinter < BasePrinter
|
4
4
|
def print(resource_group)
|
5
5
|
self.resource_group = resource_group
|
6
|
-
|
6
|
+
add_resource_group
|
7
7
|
|
8
8
|
resource_group.resources.each do |_, resource|
|
9
9
|
resource_printer.print(resource)
|
@@ -14,16 +14,16 @@ module Dox
|
|
14
14
|
|
15
15
|
attr_accessor :resource_group
|
16
16
|
|
17
|
-
def
|
18
|
-
|
17
|
+
def add_resource_group
|
18
|
+
spec['x-tagGroups'].push(name: resource_group.name, 'tags' => []) unless group_included?
|
19
|
+
end
|
19
20
|
|
20
|
-
|
21
|
-
|
22
|
-
HEREDOC
|
21
|
+
def group_included?
|
22
|
+
spec['x-tagGroups'].find { |group| group[:name] == resource_group.name }
|
23
23
|
end
|
24
24
|
|
25
25
|
def resource_printer
|
26
|
-
@resource_printer ||= ResourcePrinter.new(
|
26
|
+
@resource_printer ||= ResourcePrinter.new(spec)
|
27
27
|
end
|
28
28
|
end
|
29
29
|
end
|
@@ -3,7 +3,7 @@ module Dox
|
|
3
3
|
class ResourcePrinter < BasePrinter
|
4
4
|
def print(resource)
|
5
5
|
self.resource = resource
|
6
|
-
|
6
|
+
add_resources
|
7
7
|
|
8
8
|
resource.actions.each do |_, action|
|
9
9
|
action_printer.print(action)
|
@@ -14,16 +14,21 @@ module Dox
|
|
14
14
|
|
15
15
|
attr_accessor :resource
|
16
16
|
|
17
|
-
def
|
18
|
-
|
17
|
+
def add_resources
|
18
|
+
add_to_tags
|
19
|
+
add_to_groups
|
20
|
+
end
|
21
|
+
|
22
|
+
def add_to_tags
|
23
|
+
spec['tags'] = spec['tags'].push(name: resource.name, description: format_desc(resource.desc)).uniq
|
24
|
+
end
|
19
25
|
|
20
|
-
|
21
|
-
|
22
|
-
HEREDOC
|
26
|
+
def add_to_groups
|
27
|
+
spec['x-tagGroups'].find { |group| group[:name] == resource.group }['tags'].push(resource.name)
|
23
28
|
end
|
24
29
|
|
25
30
|
def action_printer
|
26
|
-
@action_printer ||= ActionPrinter.new(
|
31
|
+
@action_printer ||= ActionPrinter.new(spec['paths'])
|
27
32
|
end
|
28
33
|
end
|
29
34
|
end
|
data/lib/dox/util/http.rb
CHANGED
@@ -2,6 +2,70 @@ module Dox
|
|
2
2
|
module Util
|
3
3
|
module Http
|
4
4
|
VERB = ['POST', 'GET', 'PUT', 'PATCH', 'DELETE', 'HEAD'].freeze
|
5
|
+
HTTP_STATUS_CODES = {
|
6
|
+
100 => 'Continue',
|
7
|
+
101 => 'Switching Protocols',
|
8
|
+
102 => 'Processing',
|
9
|
+
103 => 'Early Hints',
|
10
|
+
200 => 'OK',
|
11
|
+
201 => 'Created',
|
12
|
+
202 => 'Accepted',
|
13
|
+
203 => 'Non-Authoritative Information',
|
14
|
+
204 => 'No Content',
|
15
|
+
205 => 'Reset Content',
|
16
|
+
206 => 'Partial Content',
|
17
|
+
207 => 'Multi-Status',
|
18
|
+
208 => 'Already Reported',
|
19
|
+
226 => 'IM Used',
|
20
|
+
300 => 'Multiple Choices',
|
21
|
+
301 => 'Moved Permanently',
|
22
|
+
302 => 'Found',
|
23
|
+
303 => 'See Other',
|
24
|
+
304 => 'Not Modified',
|
25
|
+
305 => 'Use Proxy',
|
26
|
+
307 => 'Temporary Redirect',
|
27
|
+
308 => 'Permanent Redirect',
|
28
|
+
400 => 'Bad Request',
|
29
|
+
401 => 'Unauthorized',
|
30
|
+
402 => 'Payment Required',
|
31
|
+
403 => 'Forbidden',
|
32
|
+
404 => 'Not Found',
|
33
|
+
405 => 'Method Not Allowed',
|
34
|
+
406 => 'Not Acceptable',
|
35
|
+
407 => 'Proxy Authentication Required',
|
36
|
+
408 => 'Request Timeout',
|
37
|
+
409 => 'Conflict',
|
38
|
+
410 => 'Gone',
|
39
|
+
411 => 'Length Required',
|
40
|
+
412 => 'Precondition Failed',
|
41
|
+
413 => 'Payload Too Large',
|
42
|
+
414 => 'URI Too Long',
|
43
|
+
415 => 'Unsupported Media Type',
|
44
|
+
416 => 'Range Not Satisfiable',
|
45
|
+
417 => 'Expectation Failed',
|
46
|
+
421 => 'Misdirected Request',
|
47
|
+
422 => 'Unprocessable Entity',
|
48
|
+
423 => 'Locked',
|
49
|
+
424 => 'Failed Dependency',
|
50
|
+
425 => 'Too Early',
|
51
|
+
426 => 'Upgrade Required',
|
52
|
+
428 => 'Precondition Required',
|
53
|
+
429 => 'Too Many Requests',
|
54
|
+
431 => 'Request Header Fields Too Large',
|
55
|
+
451 => 'Unavailable for Legal Reasons',
|
56
|
+
500 => 'Internal Server Error',
|
57
|
+
501 => 'Not Implemented',
|
58
|
+
502 => 'Bad Gateway',
|
59
|
+
503 => 'Service Unavailable',
|
60
|
+
504 => 'Gateway Timeout',
|
61
|
+
505 => 'HTTP Version Not Supported',
|
62
|
+
506 => 'Variant Also Negotiates',
|
63
|
+
507 => 'Insufficient Storage',
|
64
|
+
508 => 'Loop Detected',
|
65
|
+
509 => 'Bandwidth Limit Exceeded',
|
66
|
+
510 => 'Not Extended',
|
67
|
+
511 => 'Network Authentication Required'
|
68
|
+
}.freeze
|
5
69
|
|
6
70
|
def self.verb?(value)
|
7
71
|
VERB.include?(value.upcase)
|