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: []
         |