dox 1.2.0 → 2.0.0.beta1

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.
@@ -1,26 +1,46 @@
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
- print_meta_info
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
+ @output.puts(JSON.pretty_generate(spec))
10
19
  end
11
20
 
12
21
  private
13
22
 
14
- def group_printer
15
- @group_printer ||= ResourceGroupPrinter.new(@output)
23
+ def body
24
+ {
25
+ openapi: Dox.config.openapi_version || '3.0.0',
26
+ info: {
27
+ title: Dox.config.title || 'API Documentation',
28
+ description: adjust_description(Dox.config.description || ''),
29
+ version: Dox.config.api_version || '1.0'
30
+ }
31
+ }
16
32
  end
17
33
 
18
- def print_meta_info
19
- @output.puts(print_desc(api_desc_path))
34
+ def adjust_description(description)
35
+ description.end_with?('.md') ? acquire_desc(description) : description
20
36
  end
21
37
 
22
- def api_desc_path
23
- Dox.config.header_file_path
38
+ def acquire_desc(path)
39
+ read_file(path)
40
+ end
41
+
42
+ def group_printer
43
+ @group_printer ||= ResourceGroupPrinter.new(spec)
24
44
  end
25
45
  end
26
46
  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
- @output.puts resource_group_title
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 resource_group_title
18
- <<-HEREDOC
17
+ def add_resource_group
18
+ spec['x-tagGroups'].push(name: resource_group.name, 'tags' => []) unless group_included?
19
+ end
19
20
 
20
- # Group #{resource_group.name}
21
- #{print_desc(resource_group.desc)}
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(@output)
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
- @output.puts resource_title
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 resource_title
18
- <<-HEREDOC
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
- ## #{resource.name} [#{resource.endpoint}]
21
- #{print_desc(resource.desc)}
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(@output)
31
+ @action_printer ||= ActionPrinter.new(spec['paths'])
27
32
  end
28
33
  end
29
34
  end
@@ -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)
@@ -1,3 +1,3 @@
1
1
  module Dox
2
- VERSION = '1.2.0'
2
+ VERSION = '2.0.0.beta1'.freeze
3
3
  end
metadata CHANGED
@@ -1,60 +1,60 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: dox
3
3
  version: !ruby/object:Gem::Version
4
- version: 1.2.0
4
+ version: 2.0.0.beta1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Melita Kokot
8
8
  - Vedran Hrnčić
9
- autorequire:
9
+ autorequire:
10
10
  bindir: exe
11
11
  cert_chain: []
12
- date: 2019-11-27 00:00:00.000000000 Z
12
+ date: 2020-08-10 00:00:00.000000000 Z
13
13
  dependencies:
14
14
  - !ruby/object:Gem::Dependency
15
15
  name: bundler
16
16
  requirement: !ruby/object:Gem::Requirement
17
17
  requirements:
18
- - - "~>"
18
+ - - ">="
19
19
  - !ruby/object:Gem::Version
20
- version: '1.11'
20
+ version: '0'
21
21
  type: :development
22
22
  prerelease: false
23
23
  version_requirements: !ruby/object:Gem::Requirement
24
24
  requirements:
25
- - - "~>"
25
+ - - ">="
26
26
  - !ruby/object:Gem::Version
27
- version: '1.11'
27
+ version: '0'
28
28
  - !ruby/object:Gem::Dependency
29
- name: rake
29
+ name: codeclimate-test-reporter
30
30
  requirement: !ruby/object:Gem::Requirement
31
31
  requirements:
32
- - - "~>"
32
+ - - ">="
33
33
  - !ruby/object:Gem::Version
34
- version: '10.0'
34
+ version: '0'
35
35
  type: :development
36
36
  prerelease: false
37
37
  version_requirements: !ruby/object:Gem::Requirement
38
38
  requirements:
39
- - - "~>"
39
+ - - ">="
40
40
  - !ruby/object:Gem::Version
41
- version: '10.0'
41
+ version: '0'
42
42
  - !ruby/object:Gem::Dependency
43
- name: rspec
43
+ name: json
44
44
  requirement: !ruby/object:Gem::Requirement
45
45
  requirements:
46
- - - "~>"
46
+ - - ">="
47
47
  - !ruby/object:Gem::Version
48
- version: '3.0'
48
+ version: '0'
49
49
  type: :development
50
50
  prerelease: false
51
51
  version_requirements: !ruby/object:Gem::Requirement
52
52
  requirements:
53
- - - "~>"
53
+ - - ">="
54
54
  - !ruby/object:Gem::Version
55
- version: '3.0'
55
+ version: '0'
56
56
  - !ruby/object:Gem::Dependency
57
- name: pry
57
+ name: pry-nav
58
58
  requirement: !ruby/object:Gem::Requirement
59
59
  requirements:
60
60
  - - ">="
@@ -68,7 +68,7 @@ dependencies:
68
68
  - !ruby/object:Gem::Version
69
69
  version: '0'
70
70
  - !ruby/object:Gem::Dependency
71
- name: simplecov
71
+ name: pry-rails
72
72
  requirement: !ruby/object:Gem::Requirement
73
73
  requirements:
74
74
  - - ">="
@@ -82,7 +82,7 @@ dependencies:
82
82
  - !ruby/object:Gem::Version
83
83
  version: '0'
84
84
  - !ruby/object:Gem::Dependency
85
- name: codeclimate-test-reporter
85
+ name: pry-stack_explorer
86
86
  requirement: !ruby/object:Gem::Requirement
87
87
  requirements:
88
88
  - - ">="
@@ -96,34 +96,90 @@ dependencies:
96
96
  - !ruby/object:Gem::Version
97
97
  version: '0'
98
98
  - !ruby/object:Gem::Dependency
99
- name: rspec-core
99
+ name: rails
100
100
  requirement: !ruby/object:Gem::Requirement
101
101
  requirements:
102
102
  - - ">="
103
103
  - !ruby/object:Gem::Version
104
- version: '0'
104
+ version: '4.0'
105
105
  type: :runtime
106
106
  prerelease: false
107
+ version_requirements: !ruby/object:Gem::Requirement
108
+ requirements:
109
+ - - ">="
110
+ - !ruby/object:Gem::Version
111
+ version: '4.0'
112
+ - !ruby/object:Gem::Dependency
113
+ name: rake
114
+ requirement: !ruby/object:Gem::Requirement
115
+ requirements:
116
+ - - "~>"
117
+ - !ruby/object:Gem::Version
118
+ version: '10.0'
119
+ type: :development
120
+ prerelease: false
121
+ version_requirements: !ruby/object:Gem::Requirement
122
+ requirements:
123
+ - - "~>"
124
+ - !ruby/object:Gem::Version
125
+ version: '10.0'
126
+ - !ruby/object:Gem::Dependency
127
+ name: rspec
128
+ requirement: !ruby/object:Gem::Requirement
129
+ requirements:
130
+ - - "~>"
131
+ - !ruby/object:Gem::Version
132
+ version: '3.0'
133
+ type: :development
134
+ prerelease: false
135
+ version_requirements: !ruby/object:Gem::Requirement
136
+ requirements:
137
+ - - "~>"
138
+ - !ruby/object:Gem::Version
139
+ version: '3.0'
140
+ - !ruby/object:Gem::Dependency
141
+ name: rubocop
142
+ requirement: !ruby/object:Gem::Requirement
143
+ requirements:
144
+ - - ">="
145
+ - !ruby/object:Gem::Version
146
+ version: '0'
147
+ type: :development
148
+ prerelease: false
107
149
  version_requirements: !ruby/object:Gem::Requirement
108
150
  requirements:
109
151
  - - ">="
110
152
  - !ruby/object:Gem::Version
111
153
  version: '0'
112
154
  - !ruby/object:Gem::Dependency
113
- name: rails
155
+ name: simplecov
114
156
  requirement: !ruby/object:Gem::Requirement
115
157
  requirements:
116
158
  - - ">="
117
159
  - !ruby/object:Gem::Version
118
- version: '4.0'
160
+ version: '0'
161
+ type: :development
162
+ prerelease: false
163
+ version_requirements: !ruby/object:Gem::Requirement
164
+ requirements:
165
+ - - ">="
166
+ - !ruby/object:Gem::Version
167
+ version: '0'
168
+ - !ruby/object:Gem::Dependency
169
+ name: rspec-core
170
+ requirement: !ruby/object:Gem::Requirement
171
+ requirements:
172
+ - - ">="
173
+ - !ruby/object:Gem::Version
174
+ version: '0'
119
175
  type: :runtime
120
176
  prerelease: false
121
177
  version_requirements: !ruby/object:Gem::Requirement
122
178
  requirements:
123
179
  - - ">="
124
180
  - !ruby/object:Gem::Version
125
- version: '4.0'
126
- description:
181
+ version: '0'
182
+ description:
127
183
  email:
128
184
  - melita.kokot@gmail.com
129
185
  - vrabac266@gmail.com
@@ -174,7 +230,8 @@ files:
174
230
  - lib/dox/printers/action_printer.rb
175
231
  - lib/dox/printers/base_printer.rb
176
232
  - lib/dox/printers/document_printer.rb
177
- - lib/dox/printers/example_printer.rb
233
+ - lib/dox/printers/example_request_printer.rb
234
+ - lib/dox/printers/example_response_printer.rb
178
235
  - lib/dox/printers/resource_group_printer.rb
179
236
  - lib/dox/printers/resource_printer.rb
180
237
  - lib/dox/util/http.rb
@@ -184,7 +241,7 @@ licenses:
184
241
  - MIT
185
242
  metadata:
186
243
  allowed_push_host: https://rubygems.org
187
- post_install_message:
244
+ post_install_message:
188
245
  rdoc_options: []
189
246
  require_paths:
190
247
  - lib
@@ -195,13 +252,12 @@ required_ruby_version: !ruby/object:Gem::Requirement
195
252
  version: 2.0.0
196
253
  required_rubygems_version: !ruby/object:Gem::Requirement
197
254
  requirements:
198
- - - ">="
255
+ - - ">"
199
256
  - !ruby/object:Gem::Version
200
- version: '0'
257
+ version: 1.3.1
201
258
  requirements: []
202
- rubyforge_project:
203
- rubygems_version: 2.5.1
204
- signing_key:
259
+ rubygems_version: 3.0.3
260
+ signing_key:
205
261
  specification_version: 4
206
- summary: Generates API documentation for rspec in api blueprint format.
262
+ summary: Generates API documentation for rspec in OpenAPI format.
207
263
  test_files: []