openapi_first 2.3.0 → 2.5.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 +4 -4
- data/CHANGELOG.md +29 -0
- data/README.md +7 -8
- data/lib/openapi_first/response.rb +6 -2
- data/lib/openapi_first/test/coverage/plan.rb +16 -13
- data/lib/openapi_first/test/coverage/request_task.rb +3 -1
- data/lib/openapi_first/test/coverage/response_task.rb +3 -1
- data/lib/openapi_first/test/coverage/terminal_formatter.rb +10 -6
- data/lib/openapi_first/test/coverage.rb +19 -12
- data/lib/openapi_first/test.rb +51 -11
- data/lib/openapi_first/version.rb +1 -1
- metadata +5 -4
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: be641d2241f6650f7eb0c9bc4a5d78552f2945ee7a030e6003fa0567cb547c85
|
4
|
+
data.tar.gz: 3a7d5911b1c8b30074068a299f8841b791b447240141ca3303814a72bb7424ec
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 65b5e1196c0f9900a8444a7282da2f3ebda189605dc5da0b2bcba3ae96cc5e74acc4550ff57d2aebcea0e41b299c13ffa7641268317ca285997626316a785776
|
7
|
+
data.tar.gz: 43f9452d4b33c0b2a0d1211a629bbbc2c1081b5c23c96d878d73eca2561ade654af0c2040ce992527eba6853acee230551f3a2c0bade8b5a2d142a075cbd10fb
|
data/CHANGELOG.md
CHANGED
@@ -2,6 +2,35 @@
|
|
2
2
|
|
3
3
|
## Unreleased
|
4
4
|
|
5
|
+
## 2.5.0
|
6
|
+
|
7
|
+
### New feature
|
8
|
+
- Add option to skip certain responses in coverage calculation
|
9
|
+
```ruby
|
10
|
+
require 'openapi_first'
|
11
|
+
OpenapiFirst::Test.setup do |s|
|
12
|
+
test.register('openapi/openapi.yaml')
|
13
|
+
test.skip_response_coverage { it.status == '401' }
|
14
|
+
end
|
15
|
+
```
|
16
|
+
|
17
|
+
### Minor changes
|
18
|
+
- OpenapiFirst::Test.report_coverage now includes fractional digits when returning a coverage value to avoid reporting "0% / no requests made" even though some requests have been made.
|
19
|
+
- Show details about invalid requests / responses in coverage report
|
20
|
+
|
21
|
+
## 2.4.0
|
22
|
+
|
23
|
+
- Support less verbose test setup without the need to call `OpenapiFirst::Test.report_coverage`, which will be called `at_exit`:
|
24
|
+
```ruby
|
25
|
+
OpenapiFirst::Test.setup do |test|
|
26
|
+
test.register('openapi/openapi.yaml')
|
27
|
+
test.minimum_coverage = 100 # Setting this will lead to an `exit 2` if coverage is below minimum
|
28
|
+
end
|
29
|
+
```
|
30
|
+
- Add `OpenapiFirst::Test::Setup#minimum_coverage=` to control exit behaviour (exit 2 if coverage is below minimum)
|
31
|
+
- Add `verbose` option to `OpenapiFirst::Test.report_coverage(verbose: true)`
|
32
|
+
to see all passing requests/responses
|
33
|
+
|
5
34
|
## 2.3.0
|
6
35
|
|
7
36
|
### New feature
|
data/README.md
CHANGED
@@ -129,9 +129,11 @@ This can be useful in a test or staging environment, especially if you are adopt
|
|
129
129
|
use OpenapiFirst::Middlewares::ResponseValidation, spec: 'openapi.yaml' if ENV['RACK_ENV'] == 'test'
|
130
130
|
|
131
131
|
# Pass `raise_error: false` to not raise an error:
|
132
|
-
use OpenapiFirst::Middlewares::ResponseValidation, raise_error:
|
132
|
+
use OpenapiFirst::Middlewares::ResponseValidation, raise_error: false, spec: 'openapi.yaml'
|
133
133
|
```
|
134
134
|
|
135
|
+
If you are adopting OpenAPI you can use these options together with [hooks](#hooks) to get notified about requests/responses that do match your API description.
|
136
|
+
|
135
137
|
## Contract Testing
|
136
138
|
|
137
139
|
### Coverage
|
@@ -147,23 +149,20 @@ Here is how to set it up for RSpec in your `spec/spec_helper.rb`:
|
|
147
149
|
1. Register all OpenAPI documents to track coverage for and start tracking. This should go at the top of you test helper file before loading application code.
|
148
150
|
```ruby
|
149
151
|
require 'openapi_first'
|
150
|
-
OpenapiFirst::Test.setup do |
|
152
|
+
OpenapiFirst::Test.setup do |s|
|
151
153
|
test.register('openapi/openapi.yaml')
|
154
|
+
test.minimum_coverage = 100 # Setting this will lead to an `exit 2` if coverage is below minimum
|
155
|
+
test.skip_response_coverage { it.status == '500' }
|
152
156
|
end
|
153
157
|
```
|
154
158
|
2. Wrap your app with silent request / response validation. This validates all requets/responses you do during your test run. (✷1)
|
155
159
|
```ruby
|
156
160
|
config.before type: :request do
|
157
161
|
def app
|
158
|
-
OpenapiFirst::Test
|
162
|
+
OpenapiFirst::Test.app(App)
|
159
163
|
end
|
160
164
|
end
|
161
165
|
```
|
162
|
-
3. Check coverage after your test suite has finished
|
163
|
-
```ruby
|
164
|
-
# Prints a coverage report to the terminal
|
165
|
-
config.after(:suite) { OpenapiFirst::Test.report_coverage }
|
166
|
-
```
|
167
166
|
|
168
167
|
(✷1): Instead of using `OpenapiFirstTest.app` to wrap your application, you can use the middlewares or [test assertion method](#test-assertions), but you would have to do that for all requests/responses defined in your API description to make coverage work.
|
169
168
|
|
@@ -26,8 +26,12 @@ module OpenapiFirst
|
|
26
26
|
attr_reader :status, :content_type, :content_schema, :headers, :headers_schema, :key
|
27
27
|
|
28
28
|
def validate(response)
|
29
|
-
parsed_values =
|
30
|
-
error =
|
29
|
+
parsed_values = nil
|
30
|
+
error = catch FAILURE do
|
31
|
+
parsed_values = @parser.parse(response)
|
32
|
+
nil
|
33
|
+
end
|
34
|
+
error ||= @validator.call(parsed_values)
|
31
35
|
ValidatedResponse.new(response, parsed_values:, error:, response_definition: self)
|
32
36
|
end
|
33
37
|
|
@@ -12,20 +12,25 @@ module OpenapiFirst
|
|
12
12
|
class Plan
|
13
13
|
class UnknownRequestError < StandardError; end
|
14
14
|
|
15
|
-
def
|
16
|
-
|
17
|
-
@routes = []
|
18
|
-
@index = {}
|
19
|
-
@filepath = oad.filepath
|
15
|
+
def self.for(oad, skip_response: nil)
|
16
|
+
plan = new(filepath: oad.filepath)
|
20
17
|
oad.routes.each do |route|
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
18
|
+
responses = skip_response ? route.responses.reject(&skip_response) : route.responses
|
19
|
+
plan.add_route request_method: route.request_method,
|
20
|
+
path: route.path,
|
21
|
+
requests: route.requests,
|
22
|
+
responses:
|
25
23
|
end
|
24
|
+
plan
|
26
25
|
end
|
27
26
|
|
28
|
-
|
27
|
+
def initialize(filepath:)
|
28
|
+
@routes = []
|
29
|
+
@index = {}
|
30
|
+
@filepath = filepath
|
31
|
+
end
|
32
|
+
|
33
|
+
attr_reader :filepath, :routes
|
29
34
|
private attr_reader :index
|
30
35
|
|
31
36
|
def track_request(validated_request)
|
@@ -45,15 +50,13 @@ module OpenapiFirst
|
|
45
50
|
return 0 if done.zero?
|
46
51
|
|
47
52
|
all = tasks.count
|
48
|
-
(done / (all.to_f / 100))
|
53
|
+
(done / (all.to_f / 100))
|
49
54
|
end
|
50
55
|
|
51
56
|
def tasks
|
52
57
|
index.values
|
53
58
|
end
|
54
59
|
|
55
|
-
private
|
56
|
-
|
57
60
|
def add_route(request_method:, path:, requests:, responses:)
|
58
61
|
request_tasks = requests.to_a.map do |request|
|
59
62
|
index[request.key] = RequestTask.new(request)
|
@@ -14,13 +14,15 @@ module OpenapiFirst
|
|
14
14
|
def initialize(request_definition)
|
15
15
|
@request = request_definition
|
16
16
|
@requested = false
|
17
|
+
@last_error_message = nil
|
17
18
|
end
|
18
19
|
|
19
|
-
attr_reader :request
|
20
|
+
attr_reader :request, :last_error_message
|
20
21
|
|
21
22
|
def track(validated_request)
|
22
23
|
@requested = true
|
23
24
|
@valid ||= true if validated_request.valid?
|
25
|
+
@last_error_message = validated_request.error.exception_message unless validated_request.valid?
|
24
26
|
end
|
25
27
|
|
26
28
|
def requested?
|
@@ -14,13 +14,15 @@ module OpenapiFirst
|
|
14
14
|
def initialize(response_definition)
|
15
15
|
@response = response_definition
|
16
16
|
@responded = false
|
17
|
+
@last_error_message = nil
|
17
18
|
end
|
18
19
|
|
19
|
-
attr_reader :response
|
20
|
+
attr_reader :response, :last_error_message
|
20
21
|
|
21
22
|
def track(validated_response)
|
22
23
|
@responded = true
|
23
24
|
@valid ||= true if validated_response.valid?
|
25
|
+
@last_error_message = validated_response.error.exception_message unless validated_response.valid?
|
24
26
|
end
|
25
27
|
|
26
28
|
def responded?
|
@@ -5,6 +5,10 @@ module OpenapiFirst
|
|
5
5
|
module Coverage
|
6
6
|
# This is the default formatter
|
7
7
|
class TerminalFormatter
|
8
|
+
def initialize(verbose: false)
|
9
|
+
@verbose = verbose
|
10
|
+
end
|
11
|
+
|
8
12
|
# This takes a list of Coverage::Plan instances and outputs a String
|
9
13
|
def format(coverage_result)
|
10
14
|
@out = StringIO.new
|
@@ -12,7 +16,7 @@ module OpenapiFirst
|
|
12
16
|
@out.string
|
13
17
|
end
|
14
18
|
|
15
|
-
private attr_reader :out
|
19
|
+
private attr_reader :out, :verbose
|
16
20
|
|
17
21
|
private
|
18
22
|
|
@@ -27,10 +31,10 @@ module OpenapiFirst
|
|
27
31
|
def format_plan(plan)
|
28
32
|
filepath = plan.filepath
|
29
33
|
puts ['', "API validation coverage for #{filepath}: #{plan.coverage}%"]
|
30
|
-
return if plan.done?
|
34
|
+
return if plan.done? && !verbose
|
31
35
|
|
32
36
|
plan.routes.each do |route|
|
33
|
-
next if route.finished?
|
37
|
+
next if route.finished? && !verbose
|
34
38
|
|
35
39
|
format_requests(route.requests)
|
36
40
|
next if route.requests.none?(&:requested?)
|
@@ -52,7 +56,7 @@ module OpenapiFirst
|
|
52
56
|
def format_responses(responses)
|
53
57
|
responses.each do |response|
|
54
58
|
if response.finished?
|
55
|
-
puts green " ✓ #{response_label(response)}"
|
59
|
+
puts green " ✓ #{response_label(response)}" if verbose
|
56
60
|
else
|
57
61
|
puts red " ❌ #{response_label(response)} – #{explain_unfinished_response(response)}"
|
58
62
|
end
|
@@ -80,7 +84,7 @@ module OpenapiFirst
|
|
80
84
|
def explain_unfinished_request(request)
|
81
85
|
return 'No requests tracked!' unless request.requested?
|
82
86
|
|
83
|
-
|
87
|
+
"All requests invalid! (#{request.last_error_message.inspect})" unless request.any_valid_request?
|
84
88
|
end
|
85
89
|
|
86
90
|
def response_label(response)
|
@@ -93,7 +97,7 @@ module OpenapiFirst
|
|
93
97
|
def explain_unfinished_response(response)
|
94
98
|
return 'No responses tracked!' unless response.responded?
|
95
99
|
|
96
|
-
|
100
|
+
"All responses invalid! (#{response.last_error_message.inspect})" unless response.any_valid_response?
|
97
101
|
end
|
98
102
|
end
|
99
103
|
end
|
@@ -13,7 +13,11 @@ module OpenapiFirst
|
|
13
13
|
Result = Data.define(:plans, :coverage)
|
14
14
|
|
15
15
|
class << self
|
16
|
-
|
16
|
+
attr_reader :current_run
|
17
|
+
|
18
|
+
def install
|
19
|
+
return if @installed
|
20
|
+
|
17
21
|
@after_request_validation = lambda do |validated_request, oad|
|
18
22
|
track_request(validated_request, oad)
|
19
23
|
end
|
@@ -26,12 +30,21 @@ module OpenapiFirst
|
|
26
30
|
config.after_request_validation(&@after_request_validation)
|
27
31
|
config.after_response_validation(&@after_response_validation)
|
28
32
|
end
|
33
|
+
@installed = true
|
34
|
+
end
|
35
|
+
|
36
|
+
def start(skip_response: nil)
|
37
|
+
@current_run = Test.definitions.values.to_h do |oad|
|
38
|
+
plan = Plan.for(oad, skip_response:)
|
39
|
+
[oad.filepath, plan]
|
40
|
+
end
|
29
41
|
end
|
30
42
|
|
31
|
-
def
|
43
|
+
def uninstall
|
32
44
|
configuration = OpenapiFirst.configuration
|
33
45
|
configuration.hooks[:after_request_validation].delete(@after_request_validation)
|
34
46
|
configuration.hooks[:after_response_validation].delete(@after_response_validation)
|
47
|
+
@installed = nil
|
35
48
|
end
|
36
49
|
|
37
50
|
# Clear current coverage run
|
@@ -51,24 +64,18 @@ module OpenapiFirst
|
|
51
64
|
Result.new(plans:, coverage:)
|
52
65
|
end
|
53
66
|
|
54
|
-
private
|
55
|
-
|
56
67
|
# Returns all plans (Plan) that were registered for this run
|
57
68
|
def plans
|
58
|
-
current_run
|
69
|
+
current_run&.values
|
59
70
|
end
|
60
71
|
|
72
|
+
private
|
73
|
+
|
61
74
|
def coverage
|
62
|
-
return 0
|
75
|
+
return 0 unless plans
|
63
76
|
|
64
77
|
plans.sum(&:coverage) / plans.length
|
65
78
|
end
|
66
|
-
|
67
|
-
def current_run
|
68
|
-
@current_run ||= Test.definitions.values.to_h do |oad|
|
69
|
-
[oad.filepath, Plan.new(oad)]
|
70
|
-
end
|
71
|
-
end
|
72
79
|
end
|
73
80
|
end
|
74
81
|
end
|
data/lib/openapi_first/test.rb
CHANGED
@@ -14,33 +14,73 @@ module OpenapiFirst
|
|
14
14
|
|
15
15
|
# Helper class to setup tests
|
16
16
|
class Setup
|
17
|
+
def initialize
|
18
|
+
@minimum_coverage = 0
|
19
|
+
@coverage_formatter = Coverage::TerminalFormatter
|
20
|
+
@coverage_formatter_options = {}
|
21
|
+
@skip_response_coverage = nil
|
22
|
+
yield self
|
23
|
+
end
|
24
|
+
|
17
25
|
def register(*)
|
18
26
|
Test.register(*)
|
19
27
|
end
|
28
|
+
|
29
|
+
attr_accessor :minimum_coverage, :coverage_formatter_options, :coverage_formatter
|
30
|
+
|
31
|
+
def skip_response_coverage(&block)
|
32
|
+
return @skip_response_coverage unless block_given?
|
33
|
+
|
34
|
+
@skip_response_coverage = block
|
35
|
+
end
|
36
|
+
|
37
|
+
# This called at_exit
|
38
|
+
def handle_exit
|
39
|
+
coverage = Coverage.result.coverage
|
40
|
+
# :nocov:
|
41
|
+
puts 'API Coverage did not detect any API requests for the registered API descriptions' if coverage.zero?
|
42
|
+
if coverage.positive?
|
43
|
+
Test.report_coverage(
|
44
|
+
formatter: coverage_formatter,
|
45
|
+
**coverage_formatter_options
|
46
|
+
)
|
47
|
+
end
|
48
|
+
return unless minimum_coverage > coverage
|
49
|
+
|
50
|
+
puts "API Coverage fails with exit 2, because API coverage of #{coverage}%" \
|
51
|
+
"is below minimum of #{minimum_coverage}%!"
|
52
|
+
exit 2
|
53
|
+
# :nocov:
|
54
|
+
end
|
20
55
|
end
|
21
56
|
|
22
|
-
def self.setup
|
57
|
+
def self.setup(&)
|
23
58
|
unless block_given?
|
24
59
|
raise ArgumentError, "Please provide a block to #{self.class}.setup to register you API descriptions"
|
25
60
|
end
|
26
61
|
|
27
|
-
Coverage.
|
28
|
-
setup = Setup.new
|
29
|
-
|
30
|
-
return unless definitions.empty?
|
62
|
+
Coverage.install
|
63
|
+
setup = Setup.new(&)
|
64
|
+
Coverage.start(skip_response: setup.skip_response_coverage)
|
31
65
|
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
66
|
+
if definitions.empty?
|
67
|
+
raise NotRegisteredError,
|
68
|
+
'No API descriptions have been registered. ' \
|
69
|
+
'Please register your API description via ' \
|
70
|
+
"OpenapiFirst::Test.setup { |test| test.register('myopenapi.yaml') }"
|
71
|
+
end
|
72
|
+
|
73
|
+
@setup ||= at_exit do
|
74
|
+
setup.handle_exit
|
75
|
+
end
|
36
76
|
end
|
37
77
|
|
38
78
|
# Print the coverage report
|
39
79
|
# @param formatter A formatter to define the report.
|
40
80
|
# @output [IO] An output where to puts the report.
|
41
|
-
def self.report_coverage(formatter: Coverage::TerminalFormatter)
|
81
|
+
def self.report_coverage(formatter: Coverage::TerminalFormatter, **)
|
42
82
|
coverage_result = Coverage.result
|
43
|
-
puts formatter.new.format(coverage_result)
|
83
|
+
puts formatter.new(**).format(coverage_result)
|
44
84
|
puts "The overal API validation coverage of this run is: #{coverage_result.coverage}%"
|
45
85
|
end
|
46
86
|
|
metadata
CHANGED
@@ -1,13 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: openapi_first
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 2.
|
4
|
+
version: 2.5.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Andreas Haller
|
8
8
|
bindir: bin
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-
|
10
|
+
date: 2025-03-25 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: hana
|
@@ -160,7 +160,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
160
160
|
- !ruby/object:Gem::Version
|
161
161
|
version: '0'
|
162
162
|
requirements: []
|
163
|
-
rubygems_version: 3.6.
|
163
|
+
rubygems_version: 3.6.5
|
164
164
|
specification_version: 4
|
165
|
-
summary:
|
165
|
+
summary: OpenAPI based request validation, response validation, contract-testing and
|
166
|
+
coverage
|
166
167
|
test_files: []
|