Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

improve UnionProperty behavior for anyOf/oneOf, lists of types, and nullable enums #1121

Open
wants to merge 7 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions .changeset/union_fixes.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
---
default: patch
---

# Fix class generation for some union types

Fixed issue #1120, where certain combinations of types-- such as a `oneOf` between a model or an enum and null, or the OpenAPI 3.0 equivalent of using `nullable: true`-- could cause unnecessary suffixes like "Type0" to be added to the class name, and/or could cause extra copies of the class to be generated.
13 changes: 12 additions & 1 deletion end_to_end_tests/baseline_openapi_3.0.json
Original file line number Diff line number Diff line change
Expand Up @@ -1762,7 +1762,9 @@
"model",
"nullable_model",
"one_of_models",
"nullable_one_of_models"
"nullable_one_of_models",
"nullable_enum_as_ref",
"nullable_enum_inline"
],
"type": "object",
"properties": {
Expand Down Expand Up @@ -1951,6 +1953,14 @@
}
],
"nullable": true
},
"nullable_enum_as_ref": {
"$ref": "#/components/schemas/AnEnumWithNull"
},
"nullable_enum_inline": {
"type": "string",
"enum": ["FIRST_VALUE", "SECOND_VALUE", null],
"nullable": true
}
},
"description": "A Model for testing all the ways custom objects can be used ",
Expand All @@ -1971,6 +1981,7 @@
"SECOND_VALUE",
null
],
"nullable": true,
"description": "For testing Enums with mixed string / null values "
},
"AnEnumWithOnlyNull": {
Expand Down
12 changes: 11 additions & 1 deletion end_to_end_tests/baseline_openapi_3.1.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -1747,7 +1747,9 @@ info:
"model",
"nullable_model",
"one_of_models",
"nullable_one_of_models"
"nullable_one_of_models",
"nullable_enum_as_ref",
"nullable_enum_inline"
],
"type": "object",
"properties": {
Expand Down Expand Up @@ -1950,6 +1952,13 @@ info:
"$ref": "#/components/schemas/ModelWithUnionProperty"
}
]
},
"nullable_enum_as_ref": {
"$ref": "#/components/schemas/AnEnumWithNull"
},
"nullable_enum_inline": {
"type": ["string", "null"],
"enum": ["FIRST_VALUE", "SECOND_VALUE", null]
}
},
"description": "A Model for testing all the ways custom objects can be used ",
Expand All @@ -1965,6 +1974,7 @@ info:
},
"AnEnumWithNull": {
"title": "AnEnumWithNull",
"type": ["string", "null"],
"enum": [
"FIRST_VALUE",
"SECOND_VALUE",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
from .a_discriminated_union_type_2 import ADiscriminatedUnionType2
from .a_form_data import AFormData
from .a_model import AModel
from .a_model_nullable_enum_inline import AModelNullableEnumInline
from .a_model_with_properties_reference_that_are_not_object import AModelWithPropertiesReferenceThatAreNotObject
from .all_of_has_properties_but_no_type import AllOfHasPropertiesButNoType
from .all_of_has_properties_but_no_type_type_enum import AllOfHasPropertiesButNoTypeTypeEnum
Expand Down Expand Up @@ -54,7 +55,7 @@
)
from .model_with_additional_properties_refed import ModelWithAdditionalPropertiesRefed
from .model_with_any_json_properties import ModelWithAnyJsonProperties
from .model_with_any_json_properties_additional_property_type_0 import ModelWithAnyJsonPropertiesAdditionalPropertyType0
from .model_with_any_json_properties_additional_property import ModelWithAnyJsonPropertiesAdditionalProperty
from .model_with_backslash_in_description import ModelWithBackslashInDescription
from .model_with_circular_ref_a import ModelWithCircularRefA
from .model_with_circular_ref_b import ModelWithCircularRefB
Expand Down Expand Up @@ -82,8 +83,8 @@
from .post_naming_property_conflict_with_import_body import PostNamingPropertyConflictWithImportBody
from .post_naming_property_conflict_with_import_response_200 import PostNamingPropertyConflictWithImportResponse200
from .post_responses_unions_simple_before_complex_response_200 import PostResponsesUnionsSimpleBeforeComplexResponse200
from .post_responses_unions_simple_before_complex_response_200a_type_1 import (
PostResponsesUnionsSimpleBeforeComplexResponse200AType1,
from .post_responses_unions_simple_before_complex_response_200a import (
PostResponsesUnionsSimpleBeforeComplexResponse200A,
)
from .test_inline_objects_body import TestInlineObjectsBody
from .test_inline_objects_response_200 import TestInlineObjectsResponse200
Expand All @@ -98,6 +99,7 @@
"AllOfSubModel",
"AllOfSubModelTypeEnum",
"AModel",
"AModelNullableEnumInline",
"AModelWithPropertiesReferenceThatAreNotObject",
"AnAllOfEnum",
"AnArrayWithACircularRefInItemsObjectAdditionalPropertiesAItem",
Expand Down Expand Up @@ -136,7 +138,7 @@
"ModelWithAdditionalPropertiesInlinedAdditionalProperty",
"ModelWithAdditionalPropertiesRefed",
"ModelWithAnyJsonProperties",
"ModelWithAnyJsonPropertiesAdditionalPropertyType0",
"ModelWithAnyJsonPropertiesAdditionalProperty",
"ModelWithBackslashInDescription",
"ModelWithCircularRefA",
"ModelWithCircularRefB",
Expand Down Expand Up @@ -164,7 +166,7 @@
"PostNamingPropertyConflictWithImportBody",
"PostNamingPropertyConflictWithImportResponse200",
"PostResponsesUnionsSimpleBeforeComplexResponse200",
"PostResponsesUnionsSimpleBeforeComplexResponse200AType1",
"PostResponsesUnionsSimpleBeforeComplexResponse200A",
"TestInlineObjectsBody",
"TestInlineObjectsResponse200",
"ValidationError",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,10 @@
from attrs import define as _attrs_define
from dateutil.parser import isoparse

from ..models.a_model_nullable_enum_inline import AModelNullableEnumInline
from ..models.an_all_of_enum import AnAllOfEnum
from ..models.an_enum import AnEnum
from ..models.an_enum_with_null import AnEnumWithNull
from ..models.different_enum import DifferentEnum
from ..types import UNSET, Unset

Expand Down Expand Up @@ -36,6 +38,8 @@ class AModel:
nullable_one_of_models (Union['FreeFormModel', 'ModelWithUnionProperty', None]):
model (ModelWithUnionProperty):
nullable_model (Union['ModelWithUnionProperty', None]):
nullable_enum_as_ref (Union[AnEnumWithNull, None]): For testing Enums with mixed string / null values
nullable_enum_inline (Union[AModelNullableEnumInline, None]):
any_value (Union[Unset, Any]): Default: 'default'.
an_optional_allof_enum (Union[Unset, AnAllOfEnum]):
nested_list_of_enums (Union[Unset, list[list[DifferentEnum]]]):
Expand All @@ -62,6 +66,8 @@ class AModel:
nullable_one_of_models: Union["FreeFormModel", "ModelWithUnionProperty", None]
model: "ModelWithUnionProperty"
nullable_model: Union["ModelWithUnionProperty", None]
nullable_enum_as_ref: Union[AnEnumWithNull, None]
nullable_enum_inline: Union[AModelNullableEnumInline, None]
an_allof_enum_with_overridden_default: AnAllOfEnum = AnAllOfEnum.OVERRIDDEN_DEFAULT
a_nullable_uuid: Union[None, UUID] = UUID("07EF8B4D-AA09-4FFA-898D-C710796AFF41")
any_value: Union[Unset, Any] = "default"
Expand Down Expand Up @@ -137,6 +143,18 @@ def to_dict(self) -> dict[str, Any]:
else:
nullable_model = self.nullable_model

nullable_enum_as_ref: Union[None, str]
if isinstance(self.nullable_enum_as_ref, AnEnumWithNull):
nullable_enum_as_ref = self.nullable_enum_as_ref.value
else:
nullable_enum_as_ref = self.nullable_enum_as_ref

nullable_enum_inline: Union[None, str]
if isinstance(self.nullable_enum_inline, AModelNullableEnumInline):
nullable_enum_inline = self.nullable_enum_inline.value
else:
nullable_enum_inline = self.nullable_enum_inline

any_value = self.any_value

an_optional_allof_enum: Union[Unset, str] = UNSET
Expand Down Expand Up @@ -220,6 +238,8 @@ def to_dict(self) -> dict[str, Any]:
"nullable_one_of_models": nullable_one_of_models,
"model": model,
"nullable_model": nullable_model,
"nullable_enum_as_ref": nullable_enum_as_ref,
"nullable_enum_inline": nullable_enum_inline,
}
)
if any_value is not UNSET:
Expand Down Expand Up @@ -373,15 +393,45 @@ def _parse_nullable_model(data: object) -> Union["ModelWithUnionProperty", None]
try:
if not isinstance(data, dict):
raise TypeError()
nullable_model_type_1 = ModelWithUnionProperty.from_dict(data)
nullable_model = ModelWithUnionProperty.from_dict(data)

return nullable_model_type_1
return nullable_model
except: # noqa: E722
pass
return cast(Union["ModelWithUnionProperty", None], data)

nullable_model = _parse_nullable_model(d.pop("nullable_model"))

def _parse_nullable_enum_as_ref(data: object) -> Union[AnEnumWithNull, None]:
if data is None:
return data
try:
if not isinstance(data, str):
raise TypeError()
componentsschemas_an_enum_with_null = AnEnumWithNull(data)

return componentsschemas_an_enum_with_null
except: # noqa: E722
pass
return cast(Union[AnEnumWithNull, None], data)

nullable_enum_as_ref = _parse_nullable_enum_as_ref(d.pop("nullable_enum_as_ref"))

def _parse_nullable_enum_inline(data: object) -> Union[AModelNullableEnumInline, None]:
if data is None:
return data
try:
if not isinstance(data, str):
raise TypeError()
nullable_enum_inline = AModelNullableEnumInline(data)

return nullable_enum_inline
except: # noqa: E722
pass
return cast(Union[AModelNullableEnumInline, None], data)

nullable_enum_inline = _parse_nullable_enum_inline(d.pop("nullable_enum_inline"))

any_value = d.pop("any_value", UNSET)

_an_optional_allof_enum = d.pop("an_optional_allof_enum", UNSET)
Expand Down Expand Up @@ -495,9 +545,9 @@ def _parse_not_required_nullable_model(data: object) -> Union["ModelWithUnionPro
try:
if not isinstance(data, dict):
raise TypeError()
not_required_nullable_model_type_1 = ModelWithUnionProperty.from_dict(data)
not_required_nullable_model = ModelWithUnionProperty.from_dict(data)

return not_required_nullable_model_type_1
return not_required_nullable_model
except: # noqa: E722
pass
return cast(Union["ModelWithUnionProperty", None, Unset], data)
Expand All @@ -518,6 +568,8 @@ def _parse_not_required_nullable_model(data: object) -> Union["ModelWithUnionPro
nullable_one_of_models=nullable_one_of_models,
model=model,
nullable_model=nullable_model,
nullable_enum_as_ref=nullable_enum_as_ref,
nullable_enum_inline=nullable_enum_inline,
any_value=any_value,
an_optional_allof_enum=an_optional_allof_enum,
nested_list_of_enums=nested_list_of_enums,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
from enum import Enum


class AModelNullableEnumInline(str, Enum):
FIRST_VALUE = "FIRST_VALUE"
SECOND_VALUE = "SECOND_VALUE"

def __str__(self) -> str:
return str(self.value)
Original file line number Diff line number Diff line change
Expand Up @@ -304,9 +304,9 @@ def _parse_some_nullable_object(data: object) -> Union["BodyUploadFileTestsUploa
try:
if not isinstance(data, dict):
raise TypeError()
some_nullable_object_type_0 = BodyUploadFileTestsUploadPostSomeNullableObject.from_dict(data)
some_nullable_object = BodyUploadFileTestsUploadPostSomeNullableObject.from_dict(data)

return some_nullable_object_type_0
return some_nullable_object
except: # noqa: E722
pass
return cast(Union["BodyUploadFileTestsUploadPostSomeNullableObject", None], data)
Expand Down
Loading
Loading