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

eng: fix bug in type generation for nullable/union props BNCH-111776 #219

Merged
merged 6 commits into from
Dec 6, 2024
Merged
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
Original file line number Diff line number Diff line change
Expand Up @@ -112,3 +112,35 @@ def test_enum_default(self, MyEnum, MyModel):
class TestLiteralEnumDefaults:
def test_default_value(self, MyModel):
assert MyModel().enum_prop == "A"


@with_generated_client_fixture(
"""
# Test the ability to specify a default value for a union type as long as that value is
# supported by at least one of the variants

components:
schemas:
MyModel:
type: object
properties:
simpleTypeProp1:
type: ["integer", "boolean", "string"]
default: 3
simpleTypeProp2:
type: ["integer", "boolean", "string"]
default: true
simpleTypeProp3:
type: ["integer", "boolean", "string"]
default: abc
"""
)
@with_generated_code_imports(".models.MyModel")
class TestUnionDefaults:
def test_simple_type(self, MyModel):
instance = MyModel()
assert instance == MyModel(
simple_type_prop_1=3,
simple_type_prop_2=True,
simple_type_prop_3="abc",
)
Original file line number Diff line number Diff line change
Expand Up @@ -133,47 +133,23 @@ def test_invalid_values(self, MyModel):
"""
components:
schemas:
MyEnum:
type: string
enum: ["a", "b"]
MyEnumIncludingNull:
type: ["string", "null"]
enum: ["a", "b", null]
MyNullOnlyEnum:
EnumOfNullOnly:
Copy link
Author

@eli-bl eli-bl Nov 15, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The changes in this file are just to cut out some test cases that I moved into test_unions.py, because ultimately the issue isn't about the enum itself, it's about the union type (created either explicitly with oneOf, or implicitly by adding "null" as an allowable type).

enum: [null]
MyModel:
properties:
nullableEnumProp:
oneOf:
- {"$ref": "#/components/schemas/MyEnum"}
- type: "null"
enumIncludingNullProp: {"$ref": "#/components/schemas/MyEnumIncludingNull"}
nullOnlyEnumProp: {"$ref": "#/components/schemas/MyNullOnlyEnum"}
nullOnlyEnumProp: {"$ref": "#/components/schemas/EnumOfNullOnly"}
required: ["nullOnlyEnumProp"]
""")
@with_generated_code_imports(
".models.MyEnum",
".models.MyEnumIncludingNullType1", # see comment in test_nullable_enum_prop
".models.MyModel",
".types.Unset",
)
class TestNullableEnums:
def test_nullable_enum_prop(self, MyModel, MyEnum, MyEnumIncludingNullType1):
# Note, MyEnumIncludingNullType1 should be named just MyEnumIncludingNull -
# known bug: https://github.com/openapi-generators/openapi-python-client/issues/1120
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was one of the cases affected by the bug. In the new version of this test that's in test_unions.py, the model class name is correctly rendered as MyEnumIncludingNull.

assert_model_decode_encode(MyModel, {"nullableEnumProp": "b"}, MyModel(nullable_enum_prop=MyEnum.B))
assert_model_decode_encode(MyModel, {"nullableEnumProp": None}, MyModel(nullable_enum_prop=None))
assert_model_decode_encode(
MyModel,
{"enumIncludingNullProp": "a"},
MyModel(enum_including_null_prop=MyEnumIncludingNullType1.A),
)
assert_model_decode_encode( MyModel, {"enumIncludingNullProp": None}, MyModel(enum_including_null_prop=None))
class TestSingleValueNullEnum:
def test_enum_of_null_only(self, MyModel):
assert_model_decode_encode(MyModel, {"nullOnlyEnumProp": None}, MyModel(null_only_enum_prop=None))

def test_type_hints(self, MyModel, MyEnum, Unset):
expected_type = Union[MyEnum, None, Unset]
assert_model_property_type_hint(MyModel, "nullable_enum_prop", expected_type)

def test_type_hints(self, MyModel):
assert_model_property_type_hint(MyModel, "null_only_enum_prop", None)


@with_generated_client_fixture(
"""
Expand Down Expand Up @@ -217,6 +193,8 @@ def test_invalid_int(self, MyModel):

@with_generated_client_fixture(
"""
# Tests of literal_enums mode, where enums become a typing.Literal type instead of a class

components:
schemas:
MyEnum:
Expand Down Expand Up @@ -261,6 +239,8 @@ def test_invalid_values(self, MyModel):

@with_generated_client_fixture(
"""
# Tests of literal_enums mode, where enums become a typing.Literal type instead of a class

components:
schemas:
MyEnum:
Expand Down Expand Up @@ -305,6 +285,8 @@ def test_invalid_values(self, MyModel):

@with_generated_client_fixture(
"""
# Similar to some of the "union with null" tests in test_unions.py, but in literal_enums mode

components:
schemas:
MyEnum:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,32 +12,8 @@

@with_generated_client_fixture(
"""
components:
schemas:
StringOrInt:
type: ["string", "integer"]
MyModel:
type: object
properties:
stringOrIntProp:
type: ["string", "integer"]
"""
)
@with_generated_code_imports(
".models.MyModel",
".types.Unset"
)
class TestSimpleTypeList:
def test_decode_encode(self, MyModel):
assert_model_decode_encode(MyModel, {"stringOrIntProp": "a"}, MyModel(string_or_int_prop="a"))
assert_model_decode_encode(MyModel, {"stringOrIntProp": 1}, MyModel(string_or_int_prop=1))

def test_type_hints(self, MyModel, Unset):
assert_model_property_type_hint(MyModel, "string_or_int_prop", Union[str, int, Unset])

# Various use cases for oneOf

@with_generated_client_fixture(
"""
components:
schemas:
ThingA:
Expand Down Expand Up @@ -154,6 +130,123 @@ def test_type_hints(self, ModelWithUnion, ModelWithRequiredUnion, ModelWithUnion

@with_generated_client_fixture(
"""
# Various use cases for a oneOf where one of the variants is null, since these are handled
# a bit differently in the generator

components:
schemas:
MyEnum:
type: string
enum: ["a", "b"]
MyObject:
type: object
properties:
name:
type: string
MyModel:
properties:
nullableEnumProp:
oneOf:
- {"$ref": "#/components/schemas/MyEnum"}
- type: "null"
nullableObjectProp:
oneOf:
- {"$ref": "#/components/schemas/MyObject"}
- type: "null"
inlineNullableObject:
# Note, the generated class for this should be called "MyModelInlineNullableObject",
# since the generator's rule for inline schemas that require their own class is to
# concatenate the property name to the parent schema name.
oneOf:
- type: object
properties:
name:
type: string
- type: "null"
""")
@with_generated_code_imports(
".models.MyEnum",
".models.MyObject",
".models.MyModel",
".models.MyModelInlineNullableObject",
".types.Unset",
)
class TestUnionsWithNull:
def test_nullable_enum_prop(self, MyModel, MyEnum):
assert_model_decode_encode(MyModel, {"nullableEnumProp": "b"}, MyModel(nullable_enum_prop=MyEnum.B))
assert_model_decode_encode(MyModel, {"nullableEnumProp": None}, MyModel(nullable_enum_prop=None))

def test_nullable_object_prop(self, MyModel, MyObject):
assert_model_decode_encode( MyModel, {"nullableObjectProp": None}, MyModel(nullable_object_prop=None))
assert_model_decode_encode( MyModel, {"nullableObjectProp": None}, MyModel(nullable_object_prop=None))

def test_nullable_object_prop_with_inline_schema(self, MyModel, MyModelInlineNullableObject):
assert_model_decode_encode(
MyModel,
{"inlineNullableObject": {"name": "a"}},
MyModel(inline_nullable_object=MyModelInlineNullableObject(name="a")),
)
assert_model_decode_encode( MyModel, {"inlineNullableObject": None}, MyModel(inline_nullable_object=None))

def test_type_hints(self, MyModel, MyEnum, Unset):
assert_model_property_type_hint(MyModel, "nullable_enum_prop", Union[MyEnum, None, Unset])
assert_model_property_type_hint(MyModel, "nullable_object_prop", Union[ForwardRef("MyObject"), None, Unset])
assert_model_property_type_hint(
MyModel,
"inline_nullable_object",
Union[ForwardRef("MyModelInlineNullableObject"), None, Unset],
)


@with_generated_client_fixture(
"""
# Tests for combining the OpenAPI 3.0 "nullable" attribute with an enum

openapi: 3.0.0

components:
schemas:
MyEnum:
type: string
enum: ["a", "b"]
MyEnumIncludingNull:
type: string
nullable: true
enum: ["a", "b", null]
MyModel:
properties:
nullableEnumProp:
allOf:
- {"$ref": "#/components/schemas/MyEnum"}
nullable: true
enumIncludingNullProp: {"$ref": "#/components/schemas/MyEnumIncludingNull"}
""")
@with_generated_code_imports(
".models.MyEnum",
".models.MyEnumIncludingNull",
".models.MyModel",
".types.Unset",
)
class TestNullableEnumsInOpenAPI30:
def test_nullable_enum_prop(self, MyModel, MyEnum, MyEnumIncludingNull):
assert_model_decode_encode(MyModel, {"nullableEnumProp": "b"}, MyModel(nullable_enum_prop=MyEnum.B))
assert_model_decode_encode(MyModel, {"nullableEnumProp": None}, MyModel(nullable_enum_prop=None))
assert_model_decode_encode(
MyModel,
{"enumIncludingNullProp": "a"},
MyModel(enum_including_null_prop=MyEnumIncludingNull.A),
)
assert_model_decode_encode( MyModel, {"enumIncludingNullProp": None}, MyModel(enum_including_null_prop=None))

def test_type_hints(self, MyModel, MyEnum, MyEnumIncludingNull, Unset):
assert_model_property_type_hint(MyModel, "nullable_enum_prop", Union[MyEnum, None, Unset])
assert_model_property_type_hint(MyModel, "enum_including_null_prop", Union[MyEnumIncludingNull, None, Unset])


@with_generated_client_fixture(
"""
# Tests for using a discriminator property

components:
schemas:
ModelType1:
Expand Down Expand Up @@ -304,3 +397,112 @@ def test_nested_with_different_property(self, ModelType1, Schnauzer, WithNestedD
{"unionProp": {"modelType": "irrelevant", "dogType": "Schnauzer", "name": "a"}},
WithNestedDiscriminatorsDifferentProperty(union_prop=Schnauzer(model_type="irrelevant", dog_type="Schnauzer", name="a")),
)


@with_generated_client_fixture(
"""
# Tests for using multiple values of "type:" in one schema (OpenAPI 3.1)

components:
schemas:
StringOrInt:
type: ["string", "integer"]
MyModel:
type: object
properties:
stringOrIntProp:
type: ["string", "integer"]
"""
)
@with_generated_code_imports(
".models.MyModel",
".types.Unset"
)
class TestListOfSimpleTypes:
def test_decode_encode(self, MyModel):
assert_model_decode_encode(MyModel, {"stringOrIntProp": "a"}, MyModel(string_or_int_prop="a"))
assert_model_decode_encode(MyModel, {"stringOrIntProp": 1}, MyModel(string_or_int_prop=1))

def test_type_hints(self, MyModel, Unset):
assert_model_property_type_hint(MyModel, "string_or_int_prop", Union[str, int, Unset])


@with_generated_client_fixture(
"""
# Test cases where there's a union of types *and* an explicit list of multiple "type:"s -
# there was a bug where this could cause enum/model classes to be generated incorrectly

components:
schemas:
MyStringEnum:
type: string
enum: ["a", "b"]
MyIntEnum:
type: integer
enum: [1, 2]
MyEnumIncludingNull:
type: ["string", "null"]
enum: ["a", "b", null]
MyObject:
type: object
properties:
name:
type: string
MyModel:
properties:
enumsWithListOfTypesProp:
type: ["string", "integer"]
oneOf:
- {"$ref": "#/components/schemas/MyStringEnum"}
- {"$ref": "#/components/schemas/MyIntEnum"}
enumIncludingNullProp: {"$ref": "#/components/schemas/MyEnumIncludingNull"}
nullableObjectWithListOfTypesProp:
type: ["string", "object"]
oneOf:
- {"$ref": "#/components/schemas/MyObject"}
- type: "null"
""")
@with_generated_code_imports(
".models.MyStringEnum",
".models.MyIntEnum",
".models.MyEnumIncludingNull",
".models.MyObject",
".models.MyModel",
".types.Unset",
)
class TestUnionsWithListOfSimpleTypes:
def test_union_of_enums(self, MyModel, MyStringEnum, MyIntEnum):
assert_model_decode_encode(
MyModel,
{"enumsWithListOfTypesProp": "b"},
MyModel(enums_with_list_of_types_prop=MyStringEnum.B),
)
assert_model_decode_encode(
MyModel,
{"enumsWithListOfTypesProp": 2},
MyModel(enums_with_list_of_types_prop=MyIntEnum.VALUE_2),
)

def test_union_of_enum_with_null(self, MyModel, MyEnumIncludingNull):
assert_model_decode_encode(
MyModel,
{"enumIncludingNullProp": "b"},
MyModel(enum_including_null_prop=MyEnumIncludingNull.B),
)
assert_model_decode_encode(
MyModel,
{"enumIncludingNullProp": None},
MyModel(enum_including_null_prop=None),
)

def test_nullable_object_with_list_of_types(self, MyModel, MyObject):
assert_model_decode_encode(
MyModel,
{"nullableObjectWithListOfTypesProp": {"name": "a"}},
MyModel(nullable_object_with_list_of_types_prop=MyObject(name="a")),
)
assert_model_decode_encode(
MyModel,
{"nullableObjectWithListOfTypesProp": None},
MyModel(nullable_object_with_list_of_types_prop=None),
)
3 changes: 1 addition & 2 deletions end_to_end_tests/functional_tests/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -47,8 +47,7 @@ def _decorator(cls):
nonlocal alias

def _func(self, generated_client):
module = generated_client.import_module(module_name)
return getattr(module, import_name)
return generated_client.import_symbol(module_name, import_name)

alias = alias or import_name
_func.__name__ = alias
Expand Down
Loading