openapi_parameters 0.6.0 → 0.7.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 +15 -0
- data/lib/openapi_parameters/array_converter.rb +43 -0
- data/lib/openapi_parameters/converter.rb +4 -26
- data/lib/openapi_parameters/converters.rb +2 -6
- data/lib/openapi_parameters/parameter.rb +1 -0
- data/lib/openapi_parameters/query.rb +34 -12
- data/lib/openapi_parameters/version.rb +1 -1
- metadata +3 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 2c8fe7a86aaad928b6277f62969502db824f37ff92624c6f33abfa43dad15454
|
4
|
+
data.tar.gz: f8b36c4416ff14775bf9cb7ca05c5d6ad45589a5a1a390a6bcaa395142027241
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: f447c974255d4f21c3822c975b655ddb3f988a513bdb1684491a4bc0a60406f66d170378b8d5e2bc7d64eed5b94366716fdafa9f4074e90f5f1f1ad4567b0d2d
|
7
|
+
data.tar.gz: f3ac74f5faecfca7aa8513fe0e4ce901a5cca2ca42833e0f0c07e2b844e8e8ded7166ac99b492631d0d2173173cce438f4a8c37daa0b16491db0d7ddaf09aa1f
|
data/CHANGELOG.md
CHANGED
@@ -1,5 +1,20 @@
|
|
1
1
|
## [Unreleased]
|
2
2
|
|
3
|
+
## [0.7.0] - 2025-09-12
|
4
|
+
|
5
|
+
- Correctly handle `style:deepObject` with `explode: true` and parse into array
|
6
|
+
```ruby
|
7
|
+
OpenapiParameters::Query.new(
|
8
|
+
[parameter],
|
9
|
+
explode: true
|
10
|
+
).unpack('ids=1&ids=2')
|
11
|
+
# => { 'ids' => ['1', '2'] }
|
12
|
+
```
|
13
|
+
|
14
|
+
## [0.6.1] - 2025-06-23
|
15
|
+
|
16
|
+
- Fix converting items of array parameters
|
17
|
+
|
3
18
|
## [0.6.0] - 2025-06-23
|
4
19
|
|
5
20
|
- Array parameters without a value ("?ids=" or "?ids") return an empty array instead of nil or an empty string.
|
@@ -0,0 +1,43 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module OpenapiParameters
|
4
|
+
# @visibility private
|
5
|
+
class ArrayConverter
|
6
|
+
def initialize(schema)
|
7
|
+
@schema = schema
|
8
|
+
end
|
9
|
+
|
10
|
+
attr_reader :schema
|
11
|
+
|
12
|
+
def call(value)
|
13
|
+
return [] if value.nil? || value.empty?
|
14
|
+
|
15
|
+
convert_array(value)
|
16
|
+
end
|
17
|
+
|
18
|
+
private
|
19
|
+
|
20
|
+
def convert_array(array)
|
21
|
+
return array unless array.is_a?(Array)
|
22
|
+
|
23
|
+
item_schema = schema['items']
|
24
|
+
prefix_schemas = schema['prefixItems']
|
25
|
+
return convert_array_with_prefixes(array, prefix_schemas, item_schema) if prefix_schemas
|
26
|
+
|
27
|
+
array.map { |item| Converter.convert(item, item_schema) }
|
28
|
+
end
|
29
|
+
|
30
|
+
def convert_array_with_prefixes(array, prefix_schemas, item_schema)
|
31
|
+
prefixes =
|
32
|
+
array
|
33
|
+
.slice(0, prefix_schemas.size)
|
34
|
+
.each_with_index
|
35
|
+
.map { |item, index| Converter.convert(item, prefix_schemas[index]) }
|
36
|
+
array =
|
37
|
+
array[prefix_schemas.size..].map! do |item|
|
38
|
+
Converter.convert(item, item_schema)
|
39
|
+
end
|
40
|
+
prefixes + array
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
@@ -1,11 +1,12 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
+
require_relative 'array_converter'
|
4
|
+
|
3
5
|
module OpenapiParameters
|
4
6
|
# Tries to convert a request parameter value (string) to the type specified in the JSON Schema.
|
5
7
|
# @visibility private
|
6
8
|
module Converter
|
7
9
|
class << self
|
8
|
-
##
|
9
10
|
# @param value [String, Hash, Array] the value to convert
|
10
11
|
# @param schema [Hash] the schema to use for conversion.
|
11
12
|
def convert(value, schema) # rubocop:disable Metrics
|
@@ -34,7 +35,7 @@ module OpenapiParameters
|
|
34
35
|
when 'object'
|
35
36
|
convert_object(value, schema)
|
36
37
|
when 'array'
|
37
|
-
|
38
|
+
ArrayConverter.new(schema).call(value)
|
38
39
|
else
|
39
40
|
if schema['properties']
|
40
41
|
convert_object(value, schema)
|
@@ -48,7 +49,7 @@ module OpenapiParameters
|
|
48
49
|
return value unless value.is_a?(Hash)
|
49
50
|
|
50
51
|
value.each_with_object({}) do |(key, val), hsh|
|
51
|
-
hsh[key] = convert(val, schema['properties']&.fetch(key, nil))
|
52
|
+
hsh[key] = Converter.convert(val, schema['properties']&.fetch(key, nil))
|
52
53
|
end
|
53
54
|
end
|
54
55
|
|
@@ -57,29 +58,6 @@ module OpenapiParameters
|
|
57
58
|
def type(schema)
|
58
59
|
schema && schema['type']
|
59
60
|
end
|
60
|
-
|
61
|
-
def convert_array(array, schema)
|
62
|
-
return array unless array.is_a?(Array)
|
63
|
-
|
64
|
-
item_schema = schema['items']
|
65
|
-
prefix_schemas = schema['prefixItems']
|
66
|
-
return convert_array_with_prefixes(array, prefix_schemas, item_schema) if prefix_schemas
|
67
|
-
|
68
|
-
array.map { |item| convert(item, item_schema) }
|
69
|
-
end
|
70
|
-
|
71
|
-
def convert_array_with_prefixes(array, prefix_schemas, item_schema)
|
72
|
-
prefixes =
|
73
|
-
array
|
74
|
-
.slice(0, prefix_schemas.size)
|
75
|
-
.each_with_index
|
76
|
-
.map { |item, index| convert(item, prefix_schemas[index]) }
|
77
|
-
array =
|
78
|
-
array[prefix_schemas.size..].map! do |item|
|
79
|
-
convert(item, item_schema)
|
80
|
-
end
|
81
|
-
prefixes + array
|
82
|
-
end
|
83
61
|
end
|
84
62
|
end
|
85
63
|
end
|
@@ -15,6 +15,8 @@ module OpenapiParameters
|
|
15
15
|
def [](schema)
|
16
16
|
type = schema && schema['type']
|
17
17
|
converters.fetch(type) do
|
18
|
+
return ArrayConverter.new(schema) if type == 'array'
|
19
|
+
|
18
20
|
->(value) { Converter.convert(value, schema) }
|
19
21
|
end
|
20
22
|
end
|
@@ -26,12 +28,6 @@ module OpenapiParameters
|
|
26
28
|
value
|
27
29
|
end)
|
28
30
|
|
29
|
-
register('array', lambda do |value|
|
30
|
-
break [] if value.nil? || value.empty?
|
31
|
-
|
32
|
-
value
|
33
|
-
end)
|
34
|
-
|
35
31
|
register('number', lambda do |value|
|
36
32
|
Float(value)
|
37
33
|
rescue StandardError
|
@@ -11,16 +11,21 @@ module OpenapiParameters
|
|
11
11
|
@parameters = parameters.map { Parameter.new(_1) }
|
12
12
|
@convert = convert
|
13
13
|
@remove_array_brackets = rack_array_compat
|
14
|
+
@any_deep_object = @parameters.any?(&:deep_object?)
|
14
15
|
end
|
15
16
|
|
16
|
-
def unpack(query_string) # rubocop:disable Metrics/AbcSize
|
17
|
+
def unpack(query_string) # rubocop:disable Metrics/AbcSize,Metrics/MethodLength
|
17
18
|
parsed_query = parse_query(query_string)
|
19
|
+
parsed_nested_query = Rack::Utils.parse_nested_query(query_string) if any_deep_object?
|
18
20
|
parameters.each_with_object({}) do |parameter, result|
|
19
21
|
if parameter.deep_object?
|
20
|
-
parsed_nested_query = Rack::Utils.parse_nested_query(query_string)
|
21
22
|
next unless parsed_nested_query.key?(parameter.name)
|
22
23
|
|
23
|
-
value =
|
24
|
+
value = if parameter.explode?
|
25
|
+
handle_deep_object_explode(parameter, parsed_nested_query[parameter.name], parsed_query)
|
26
|
+
else
|
27
|
+
parsed_nested_query[parameter.name]
|
28
|
+
end
|
24
29
|
else
|
25
30
|
next unless parsed_query.key?(parameter.name)
|
26
31
|
|
@@ -40,6 +45,10 @@ module OpenapiParameters
|
|
40
45
|
|
41
46
|
private
|
42
47
|
|
48
|
+
def any_deep_object?
|
49
|
+
@any_deep_object
|
50
|
+
end
|
51
|
+
|
43
52
|
def parse_query(query_string)
|
44
53
|
Rack::Utils.parse_query(query_string) do |s|
|
45
54
|
Rack::Utils.unescape(s)
|
@@ -48,18 +57,31 @@ module OpenapiParameters
|
|
48
57
|
end
|
49
58
|
end
|
50
59
|
|
51
|
-
def
|
52
|
-
return value unless
|
53
|
-
return value if value == ''
|
60
|
+
def handle_deep_object_explode(parameter, value, parsed_query)
|
61
|
+
return value unless value.is_a?(Hash)
|
54
62
|
|
55
|
-
|
56
|
-
|
63
|
+
schema_props = parameter.schema['properties'] || {}
|
64
|
+
|
65
|
+
array_prop_values = find_prop_matches(parameter.name, schema_props, parsed_query)
|
57
66
|
|
58
|
-
|
59
|
-
|
60
|
-
return value if value == ''
|
67
|
+
schema_props.each_with_object(value) do |(prop, prop_schema), result|
|
68
|
+
next unless prop_schema['type'] == 'array'
|
61
69
|
|
62
|
-
|
70
|
+
arr = array_prop_values[prop]
|
71
|
+
result[prop] = if arr.empty? && value.key?(prop)
|
72
|
+
Array(value[prop])
|
73
|
+
else
|
74
|
+
arr
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
def find_prop_matches(parameter_name, schema_props, parsed_query)
|
80
|
+
schema_props.each_key.with_object({}) do |prop, result|
|
81
|
+
key = "#{parameter_name}[#{prop}]"
|
82
|
+
value = Array(parsed_query[key])
|
83
|
+
result[prop] = value.map { |match| Rack::Utils.unescape(match) } if value.is_a?(Array)
|
84
|
+
end
|
63
85
|
end
|
64
86
|
end
|
65
87
|
end
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: openapi_parameters
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.7.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Andreas Haller
|
8
8
|
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date: 2025-
|
11
|
+
date: 2025-09-12 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: rack
|
@@ -35,6 +35,7 @@ files:
|
|
35
35
|
- CHANGELOG.md
|
36
36
|
- README.md
|
37
37
|
- lib/openapi_parameters.rb
|
38
|
+
- lib/openapi_parameters/array_converter.rb
|
38
39
|
- lib/openapi_parameters/converter.rb
|
39
40
|
- lib/openapi_parameters/converters.rb
|
40
41
|
- lib/openapi_parameters/cookie.rb
|