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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 851414a9f2a8d64df210a610ed9577ad6608c85173a06221b96c3d390dcca1e5
4
- data.tar.gz: 815e970727a8e683740b622768ccd031a138e4ef4ca38c8aebcd73bf2a7f4230
3
+ metadata.gz: be641d2241f6650f7eb0c9bc4a5d78552f2945ee7a030e6003fa0567cb547c85
4
+ data.tar.gz: 3a7d5911b1c8b30074068a299f8841b791b447240141ca3303814a72bb7424ec
5
5
  SHA512:
6
- metadata.gz: 8696eaabb5455c0e009233a3f8099198aa222cb1a442ef53314e5ed756373f41eff3747f69c0036cb43c708a746b21289866b9d7cfeace5649a68a686ad302ef
7
- data.tar.gz: 9bb2db9c0443eb75299c755a79a0e9c6d696bc27a5683bc8fe4994e3e7e965e336afe855bbe41666346697668ae312697f5b0e307ec89a13c1c9ecf6c4087134
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: true, spec: 'openapi.yaml'
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 |test|
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::(App)
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 = @parser.parse(response)
30
- error = @validator.call(parsed_values)
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 initialize(oad)
16
- @oad = oad
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
- add_route request_method: route.request_method,
22
- path: route.path,
23
- requests: route.requests,
24
- responses: route.responses
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
- attr_reader :filepath, :oad, :routes
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)).to_i
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
- 'All requests invalid!' unless request.any_valid_request?
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
- 'All responses invalid!' unless response.any_valid_response?
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
- def start
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 stop
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.values
69
+ current_run&.values
59
70
  end
60
71
 
72
+ private
73
+
61
74
  def coverage
62
- return 0 if plans.empty?
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
@@ -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.start
28
- setup = Setup.new
29
- yield setup
30
- return unless definitions.empty?
62
+ Coverage.install
63
+ setup = Setup.new(&)
64
+ Coverage.start(skip_response: setup.skip_response_coverage)
31
65
 
32
- raise NotRegisteredError,
33
- 'No API descriptions have been registered. ' \
34
- 'Please register your API description via ' \
35
- "OpenapiFirst::Test.setup { |test| test.register('myopenapi.yaml') }"
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
 
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module OpenapiFirst
4
- VERSION = '2.3.0'
4
+ VERSION = '2.5.0'
5
5
  end
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.3.0
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-02-14 00:00:00.000000000 Z
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.2
163
+ rubygems_version: 3.6.5
164
164
  specification_version: 4
165
- summary: Implement HTTP APIs based on OpenApi 3.x
165
+ summary: OpenAPI based request validation, response validation, contract-testing and
166
+ coverage
166
167
  test_files: []