openapi_first 3.2.0 → 3.3.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 +9 -0
- data/lib/openapi_first/failure.rb +10 -12
- data/lib/openapi_first/file_loader.rb +21 -6
- data/lib/openapi_first/response_parser.rb +1 -1
- data/lib/openapi_first/router/find_content.rb +4 -2
- data/lib/openapi_first/router/path_template.rb +6 -7
- data/lib/openapi_first/router.rb +6 -6
- data/lib/openapi_first/test/configuration.rb +1 -1
- data/lib/openapi_first/test.rb +5 -8
- data/lib/openapi_first/validated_request.rb +1 -0
- data/lib/openapi_first/version.rb +1 -1
- data/lib/openapi_first.rb +5 -0
- metadata +2 -2
checksums.yaml
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
---
|
|
2
2
|
SHA256:
|
|
3
|
-
metadata.gz:
|
|
4
|
-
data.tar.gz:
|
|
3
|
+
metadata.gz: 7d164451de439181b88fc45c5735f564cc802f5f53e13e245d224ca474a12c03
|
|
4
|
+
data.tar.gz: c5adc0092bccf3de1248799ddde58bf77dce9c58858e588dd72e8c8ed4a5c6cd
|
|
5
5
|
SHA512:
|
|
6
|
-
metadata.gz:
|
|
7
|
-
data.tar.gz:
|
|
6
|
+
metadata.gz: ca4c5185152f49fe83977856d585f31ab1490f6b813a3f9878fc42e266f84dddc445baaf6865868e4a2b9c24f9ab7dccc2732ccc74f15aea43363a2f2b519e80
|
|
7
|
+
data.tar.gz: 43e0abb51e9d78fe86d0fb0903e372dbdfe480e73906148dd9a1da15261feb41d3614c3a6d337e11796baceff375c7ea887cbb8e7bbbb7098e6f00ec3825c5c8
|
data/CHANGELOG.md
CHANGED
|
@@ -2,6 +2,15 @@
|
|
|
2
2
|
|
|
3
3
|
## Unreleased
|
|
4
4
|
|
|
5
|
+
## 3.3.0
|
|
6
|
+
|
|
7
|
+
- OpenapiFirst will now cache the contents of files that have been loaded. If you need to reload your OpenAPI definition for tests or server hot reloading, you can call `OpenapiFirst.clear_cache!`.
|
|
8
|
+
- Optimized `OpenapiFirst::Router#match` for faster path matching and reduced memory allocation.
|
|
9
|
+
|
|
10
|
+
## 3.2.1
|
|
11
|
+
|
|
12
|
+
- Don't raise `UnknownQueryParameterError` if request is ignored in tests. Fixes [#441](https://github.com/ahx/openapi_first/issues/441).
|
|
13
|
+
|
|
5
14
|
## 3.2.0
|
|
6
15
|
|
|
7
16
|
### Changed
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
module OpenapiFirst
|
|
4
4
|
# A failure object returned when validation or parsing of a request or response has failed.
|
|
5
5
|
# This returned in ValidatedRequest#error and ValidatedResponse#error.
|
|
6
|
-
class Failure
|
|
6
|
+
class Failure < Data.define(:type, :message, :errors) # rubocop:disable Style/DataInheritance
|
|
7
7
|
TYPES = {
|
|
8
8
|
not_found: [NotFoundError, 'Not found.'],
|
|
9
9
|
method_not_allowed: [RequestInvalidError, 'Request method is not defined.'],
|
|
@@ -33,27 +33,25 @@ module OpenapiFirst
|
|
|
33
33
|
# @param type [Symbol] See TYPES.keys
|
|
34
34
|
# @param message [String] A generic error message
|
|
35
35
|
# @param errors [Array<OpenapiFirst::Schema::ValidationError>]
|
|
36
|
-
def
|
|
36
|
+
def self.new(type, message: nil, errors: nil)
|
|
37
37
|
unless TYPES.key?(type)
|
|
38
38
|
raise ArgumentError,
|
|
39
39
|
"type must be one of #{TYPES.keys} but was #{type.inspect}"
|
|
40
40
|
end
|
|
41
|
-
|
|
42
|
-
@type = type
|
|
43
|
-
@message = message
|
|
44
|
-
@errors = errors
|
|
41
|
+
super(type:, message:, errors:)
|
|
45
42
|
end
|
|
46
43
|
|
|
47
|
-
# @
|
|
44
|
+
# @method type [Symbol] type The type of the failure. See TYPES.keys.
|
|
48
45
|
# Example: :invalid_body
|
|
49
|
-
attr_reader :type
|
|
50
46
|
|
|
51
|
-
# @
|
|
52
|
-
|
|
47
|
+
# @method errors [Array<OpenapiFirst::Schema::ValidationError>, nil] errors Schema validation errors
|
|
48
|
+
|
|
49
|
+
alias original_message message
|
|
50
|
+
private :original_message
|
|
53
51
|
|
|
54
52
|
# A generic error message
|
|
55
53
|
def message
|
|
56
|
-
|
|
54
|
+
original_message || exception_message
|
|
57
55
|
end
|
|
58
56
|
|
|
59
57
|
def exception(context = nil)
|
|
@@ -63,7 +61,7 @@ module OpenapiFirst
|
|
|
63
61
|
def exception_message
|
|
64
62
|
_, message_prefix = TYPES.fetch(type)
|
|
65
63
|
|
|
66
|
-
[message_prefix,
|
|
64
|
+
[message_prefix, original_message || generate_message].compact.join(' ')
|
|
67
65
|
end
|
|
68
66
|
|
|
69
67
|
private
|
|
@@ -6,17 +6,32 @@ require 'yaml'
|
|
|
6
6
|
module OpenapiFirst
|
|
7
7
|
# @!visibility private
|
|
8
8
|
module FileLoader
|
|
9
|
+
@cache = {}
|
|
10
|
+
@mutex = Mutex.new
|
|
11
|
+
|
|
9
12
|
module_function
|
|
10
13
|
|
|
11
14
|
def load(file_path)
|
|
12
|
-
|
|
15
|
+
@cache[file_path] || @mutex.synchronize do
|
|
16
|
+
@cache[file_path] ||= begin
|
|
17
|
+
raise FileNotFoundError, "File not found #{file_path.inspect}" unless File.exist?(file_path)
|
|
18
|
+
|
|
19
|
+
body = File.read(file_path)
|
|
20
|
+
extname = File.extname(file_path)
|
|
13
21
|
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
22
|
+
if extname == '.json'
|
|
23
|
+
::JSON.parse(body)
|
|
24
|
+
elsif ['.yaml', '.yml'].include?(extname)
|
|
25
|
+
YAML.unsafe_load(body)
|
|
26
|
+
else
|
|
27
|
+
body
|
|
28
|
+
end
|
|
29
|
+
end
|
|
30
|
+
end
|
|
31
|
+
end
|
|
18
32
|
|
|
19
|
-
|
|
33
|
+
def clear_cache!
|
|
34
|
+
@mutex.synchronize { @cache.clear }
|
|
20
35
|
end
|
|
21
36
|
end
|
|
22
37
|
end
|
|
@@ -25,7 +25,7 @@ module OpenapiFirst
|
|
|
25
25
|
buffered_body = +''
|
|
26
26
|
|
|
27
27
|
if rack_response.body.respond_to?(:each)
|
|
28
|
-
rack_response.body.each { |chunk| buffered_body
|
|
28
|
+
rack_response.body.each { |chunk| buffered_body << chunk }
|
|
29
29
|
return buffered_body
|
|
30
30
|
end
|
|
31
31
|
rack_response.body
|
|
@@ -8,8 +8,10 @@ module OpenapiFirst
|
|
|
8
8
|
return contents[nil] if content_type.nil? || content_type.empty?
|
|
9
9
|
|
|
10
10
|
contents.fetch(content_type) do
|
|
11
|
-
|
|
12
|
-
|
|
11
|
+
semi = content_type.index(';')
|
|
12
|
+
type = semi ? content_type[0, semi] : content_type
|
|
13
|
+
slash = type.index('/') || type.length
|
|
14
|
+
contents[type] || contents["#{type[0, slash]}/*"] || contents['*/*'] || contents[nil]
|
|
13
15
|
end
|
|
14
16
|
end
|
|
15
17
|
end
|
|
@@ -6,8 +6,6 @@ module OpenapiFirst
|
|
|
6
6
|
class PathTemplate
|
|
7
7
|
# See also https://spec.openapis.org/oas/v3.1.0#path-templating
|
|
8
8
|
TEMPLATE_EXPRESSION = /(\{[^{}]+\})/
|
|
9
|
-
TEMPLATE_EXPRESSION_NAME = /\{([^{}]+)\}/
|
|
10
|
-
ALLOWED_PARAMETER_CHARACTERS = %r{([^/?#]+)}
|
|
11
9
|
|
|
12
10
|
def self.template?(string)
|
|
13
11
|
string.include?('{')
|
|
@@ -15,7 +13,6 @@ module OpenapiFirst
|
|
|
15
13
|
|
|
16
14
|
def initialize(template)
|
|
17
15
|
@template = template
|
|
18
|
-
@names = template.scan(TEMPLATE_EXPRESSION_NAME).flatten
|
|
19
16
|
@pattern = build_pattern(template)
|
|
20
17
|
end
|
|
21
18
|
|
|
@@ -25,20 +22,22 @@ module OpenapiFirst
|
|
|
25
22
|
|
|
26
23
|
def match(path)
|
|
27
24
|
return {} if path == @template
|
|
28
|
-
return if @names.empty?
|
|
29
25
|
|
|
30
26
|
matches = path.match(@pattern)
|
|
31
27
|
return unless matches
|
|
32
28
|
|
|
33
|
-
|
|
34
|
-
@names.zip(values).to_h
|
|
29
|
+
matches.named_captures
|
|
35
30
|
end
|
|
36
31
|
|
|
37
32
|
private
|
|
38
33
|
|
|
39
34
|
def build_pattern(template)
|
|
40
35
|
parts = template.split(TEMPLATE_EXPRESSION).map! do |part|
|
|
41
|
-
part.start_with?('{')
|
|
36
|
+
if part.start_with?('{')
|
|
37
|
+
"(?<#{part[1..-2]}>[^/?#]+)"
|
|
38
|
+
else
|
|
39
|
+
Regexp.escape(part)
|
|
40
|
+
end
|
|
42
41
|
end
|
|
43
42
|
|
|
44
43
|
/^#{parts.join}$/
|
data/lib/openapi_first/router.rb
CHANGED
|
@@ -97,15 +97,15 @@ module OpenapiFirst
|
|
|
97
97
|
found = @static[request_path]
|
|
98
98
|
return [found, {}] if found
|
|
99
99
|
|
|
100
|
-
|
|
100
|
+
@dynamic.each_value.reduce(nil) do |best, path_item|
|
|
101
101
|
params = path_item[:template].match(request_path)
|
|
102
|
-
next unless params
|
|
102
|
+
next best unless params
|
|
103
103
|
|
|
104
|
-
[path_item, params]
|
|
105
|
-
|
|
106
|
-
return matches.first if matches.length == 1
|
|
104
|
+
candidate = [path_item, params]
|
|
105
|
+
next candidate unless best
|
|
107
106
|
|
|
108
|
-
|
|
107
|
+
params.values.sum(&:length) < best[1].values.sum(&:length) ? candidate : best
|
|
108
|
+
end
|
|
109
109
|
end
|
|
110
110
|
end
|
|
111
111
|
end
|
|
@@ -88,7 +88,7 @@ module OpenapiFirst
|
|
|
88
88
|
return false if @ignore_request_error&.call(validated_request)
|
|
89
89
|
return false if ignore_unknown_requests? && validated_request.unknown?
|
|
90
90
|
|
|
91
|
-
|
|
91
|
+
true
|
|
92
92
|
end
|
|
93
93
|
|
|
94
94
|
def raise_response_error?(validated_response, rack_request)
|
data/lib/openapi_first/test.rb
CHANGED
|
@@ -106,9 +106,12 @@ module OpenapiFirst
|
|
|
106
106
|
OpenapiFirst.configure do |config|
|
|
107
107
|
@after_request_validation = config.after_request_validation do |validated_request, oad|
|
|
108
108
|
next unless registered?(oad)
|
|
109
|
-
raise validated_request.error.exception if raise_request_error?(validated_request)
|
|
110
109
|
|
|
111
|
-
|
|
110
|
+
if configuration.raise_request_error?(validated_request)
|
|
111
|
+
raise validated_request.error.exception if validated_request.unknown?
|
|
112
|
+
|
|
113
|
+
check_unknown_query_parameters(validated_request)
|
|
114
|
+
end
|
|
112
115
|
|
|
113
116
|
Coverage.track_request(validated_request, oad)
|
|
114
117
|
end
|
|
@@ -152,12 +155,6 @@ module OpenapiFirst
|
|
|
152
155
|
"Unknown query parameter#{s} #{list} for #{validated_request.fullpath}"
|
|
153
156
|
end
|
|
154
157
|
|
|
155
|
-
def raise_request_error?(validated_request)
|
|
156
|
-
return false if validated_request.valid?
|
|
157
|
-
|
|
158
|
-
configuration.raise_request_error?(validated_request)
|
|
159
|
-
end
|
|
160
|
-
|
|
161
158
|
def many?(array) = array.length > 1
|
|
162
159
|
|
|
163
160
|
def raise_response_error?(validated_response, rack_request)
|
|
@@ -33,6 +33,7 @@ module OpenapiFirst
|
|
|
33
33
|
def_delegator :request_definition, :operation
|
|
34
34
|
|
|
35
35
|
# @return [Hash] Query parameters and values that are not defined in the OpenAPI spec.
|
|
36
|
+
# FIXME: Optimize this so it does not have to parse the query string a second time to find unknown parameters
|
|
36
37
|
def unknown_query_parameters
|
|
37
38
|
@query_parser&.unknown_values(query_string)
|
|
38
39
|
end
|
data/lib/openapi_first.rb
CHANGED
|
@@ -22,6 +22,11 @@ module OpenapiFirst
|
|
|
22
22
|
|
|
23
23
|
FAILURE = :openapi_first_validation_failure
|
|
24
24
|
|
|
25
|
+
# Clears cached files
|
|
26
|
+
def self.clear_cache!
|
|
27
|
+
FileLoader.clear_cache!
|
|
28
|
+
end
|
|
29
|
+
|
|
25
30
|
# @return [Configuration]
|
|
26
31
|
def self.configuration
|
|
27
32
|
@configuration ||= Configuration.new
|
metadata
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
|
2
2
|
name: openapi_first
|
|
3
3
|
version: !ruby/object:Gem::Version
|
|
4
|
-
version: 3.
|
|
4
|
+
version: 3.3.0
|
|
5
5
|
platform: ruby
|
|
6
6
|
authors:
|
|
7
7
|
- Andreas Haller
|
|
@@ -186,7 +186,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
|
186
186
|
- !ruby/object:Gem::Version
|
|
187
187
|
version: '0'
|
|
188
188
|
requirements: []
|
|
189
|
-
rubygems_version:
|
|
189
|
+
rubygems_version: 4.0.6
|
|
190
190
|
specification_version: 4
|
|
191
191
|
summary: OpenAPI based request validation, response validation, contract-testing and
|
|
192
192
|
coverage
|