Skip to content

Commit e7972fc

Browse files
committed
AdditionalProperties free form nullable fix
1 parent 027860e commit e7972fc

File tree

3 files changed

+15
-10
lines changed

3 files changed

+15
-10
lines changed

openapi_core/schema/schemas.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,15 +4,15 @@
44
from openapi_core.spec import Spec
55

66

7-
def get_all_properties(schema: Spec) -> Dict[str, Any]:
7+
def get_schema_properties(schema: Spec) -> Dict[str, Any]:
88
properties = schema.get("properties", {})
99
properties_dict = dict(list(properties.items()))
1010

1111
if "allOf" not in schema:
1212
return properties_dict
1313

1414
for subschema in schema / "allOf":
15-
subschema_props = get_all_properties(subschema)
15+
subschema_props = get_schema_properties(subschema)
1616
properties_dict.update(subschema_props)
1717

1818
return properties_dict

openapi_core/unmarshalling/schemas/unmarshallers.py

+11-8
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@
2020
from openapi_schema_validator._types import is_string
2121

2222
from openapi_core.extensions.models.factories import ModelPathFactory
23-
from openapi_core.schema.schemas import get_all_properties
23+
from openapi_core.schema.schemas import get_schema_properties
2424
from openapi_core.spec import Spec
2525
from openapi_core.unmarshalling.schemas.datatypes import FormattersDict
2626
from openapi_core.unmarshalling.schemas.enums import UnmarshalContext
@@ -268,7 +268,9 @@ def _unmarshal_properties(self, value: Any) -> Any:
268268
else:
269269
properties.update(any_of_properties)
270270

271-
for prop_name, prop in get_all_properties(self.schema).items():
271+
# unmarshal schema properties
272+
schema_properties = get_schema_properties(self.schema)
273+
for prop_name, prop in schema_properties.items():
272274
read_only = prop.getkey("readOnly", False)
273275
if self.context == UnmarshalContext.REQUEST and read_only:
274276
continue
@@ -286,23 +288,24 @@ def _unmarshal_properties(self, value: Any) -> Any:
286288
prop_value
287289
)
288290

291+
# unmarshal additional properties
289292
additional_properties = self.schema.getkey(
290293
"additionalProperties", True
291294
)
292295
if additional_properties is not False:
293296
# free-form object
294297
if additional_properties is True:
295-
additional_prop_schema = Spec.from_dict({})
298+
unmarshal_func = lambda x: x
296299
# defined schema
297300
else:
298301
additional_prop_schema = self.schema / "additionalProperties"
299-
additional_prop_unmarshaler = self.unmarshallers_factory.create(
300-
additional_prop_schema
301-
)
302+
unmarshal_func = (
303+
self.unmarshallers_factory.create(additional_prop_schema)
304+
)
302305
for prop_name, prop_value in value.items():
303-
if prop_name in properties:
306+
if prop_name in schema_properties:
304307
continue
305-
properties[prop_name] = additional_prop_unmarshaler(prop_value)
308+
properties[prop_name] = unmarshal_func(prop_value)
306309

307310
return properties
308311

tests/integration/validation/test_petstore.py

+2
Original file line numberDiff line numberDiff line change
@@ -752,6 +752,7 @@ def test_post_cats(self, spec, spec_dict):
752752
"ears": {
753753
"healthy": pet_healthy,
754754
},
755+
"extra": None,
755756
}
756757
data = json.dumps(data_json)
757758
headers = {
@@ -799,6 +800,7 @@ def test_post_cats(self, spec, spec_dict):
799800
assert result.body.address.street == pet_street
800801
assert result.body.address.city == pet_city
801802
assert result.body.healthy == pet_healthy
803+
assert result.body.extra is None
802804

803805
def test_post_cats_boolean_string(self, spec, spec_dict):
804806
host_url = "https://staging.gigantic-server.com/v1"

0 commit comments

Comments
 (0)