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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: b798ca1e80a551797f19f1b1f2242b7785af3158446c898d9b0aadaa9c520777
4
- data.tar.gz: d8540ac316847320ace7b8379395514c5f2deee2b711ab097e8b73c22b2de14f
3
+ metadata.gz: 2c8fe7a86aaad928b6277f62969502db824f37ff92624c6f33abfa43dad15454
4
+ data.tar.gz: f8b36c4416ff14775bf9cb7ca05c5d6ad45589a5a1a390a6bcaa395142027241
5
5
  SHA512:
6
- metadata.gz: c8c9011ca61153e8fb20f375aa3bda9f11f69a4bac614726cff883f1811e219eece7a4cad2ee384ead8d0688a6f327783ed1c156cc94505ad791341e22d0fc97
7
- data.tar.gz: 1e5fe939d22288b0196ea0a85a7152f022f2f3e96a2f5ac65329f97cb7448517b1df0b13926cf0d4ef227346ca4f002e2727bc8a2e5ceacfadca857b9ffa3dbf
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
- convert_array(value, schema)
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
@@ -9,6 +9,7 @@ module OpenapiParameters
9
9
  @definition = definition
10
10
  @name = definition['name']
11
11
  @is_deep_object = style == 'deepObject'
12
+ @is_explode = explode?
12
13
  @converter = Converters[schema]
13
14
  check_supported!
14
15
  end
@@ -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 = parsed_nested_query[parameter.name]
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 convert_primitive(value, parameter)
52
- return value unless @convert
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
- Converter.convert_primitive(value, parameter.schema)
56
- end
63
+ schema_props = parameter.schema['properties'] || {}
64
+
65
+ array_prop_values = find_prop_matches(parameter.name, schema_props, parsed_query)
57
66
 
58
- def convert(value, parameter)
59
- return value unless @convert
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
- Converter.convert(value, parameter.schema)
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
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module OpenapiParameters
4
- VERSION = '0.6.0'
4
+ VERSION = '0.7.0'
5
5
  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.6.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-06-26 00:00:00.000000000 Z
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