From c60d65c180a1a9d6c1416aee4cb31125483f5d24 Mon Sep 17 00:00:00 2001 From: Kent Friesen Date: Wed, 1 May 2024 17:48:07 -0700 Subject: [PATCH 1/2] Fixes pydantic#9235 --- src/serializers/type_serializers/literal.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/serializers/type_serializers/literal.rs b/src/serializers/type_serializers/literal.rs index d0b751e72..258f35ed9 100644 --- a/src/serializers/type_serializers/literal.rs +++ b/src/serializers/type_serializers/literal.rs @@ -89,7 +89,7 @@ impl LiteralSerializer { if let Ok(py_str) = value.downcast::() { let s = py_str.to_str()?; if self.expected_str.contains(s) { - return Ok(OutputValue::OkStr(py_str.clone())); + return Ok(OutputValue::OkStr(PyString::new_bound(value.py(), s))); } } } From 1e6912177a4007de2bf8cf6eab7cb79ba8a917ea Mon Sep 17 00:00:00 2001 From: Kent Friesen Date: Thu, 11 Jul 2024 07:12:38 -0300 Subject: [PATCH 2/2] Adds test --- tests/serializers/test_literal.py | 76 +++++++++++++++++++++++++++++++ 1 file changed, 76 insertions(+) diff --git a/tests/serializers/test_literal.py b/tests/serializers/test_literal.py index d3cc9f262..6ce75ed96 100644 --- a/tests/serializers/test_literal.py +++ b/tests/serializers/test_literal.py @@ -1,3 +1,7 @@ +from dataclasses import dataclass +from enum import Enum +from typing import Literal, Union + import pytest from pydantic_core import SchemaError, SchemaSerializer, core_schema @@ -71,3 +75,75 @@ def test_bool_literal(): assert s.to_python(False, mode='json') is False assert s.to_python(True) is True assert s.to_json(False) == b'false' + + +def test_literal_with_enum() -> None: + class SomeEnum(str, Enum): + CAT = 'cat' + DOG = 'dog' + + @dataclass + class Dog: + name: str + type: Literal[SomeEnum.DOG] = SomeEnum.DOG + + @dataclass + class Cat: + name: str + type: Literal[SomeEnum.CAT] = SomeEnum.CAT + + @dataclass + class Yard: + pet: Union[Dog, Cat] + + serializer = SchemaSerializer( + core_schema.model_schema( + cls=Yard, + schema=core_schema.model_fields_schema( + fields={ + 'pet': core_schema.model_field( + schema=core_schema.tagged_union_schema( + choices={ + SomeEnum.DOG: core_schema.model_schema( + cls=Dog, + schema=core_schema.model_fields_schema( + fields={ + 'type': core_schema.model_field( + schema=core_schema.with_default_schema( + schema=core_schema.literal_schema([SomeEnum.DOG]), + default=SomeEnum.DOG, + ) + ), + 'name': core_schema.model_field(schema=core_schema.str_schema()), + }, + model_name='Dog', + ), + ), + SomeEnum.CAT: core_schema.model_schema( + cls=Cat, + schema=core_schema.model_fields_schema( + fields={ + 'type': core_schema.model_field( + schema=core_schema.with_default_schema( + schema=core_schema.literal_schema([SomeEnum.CAT]), + default=SomeEnum.CAT, + ) + ), + 'name': core_schema.model_field(schema=core_schema.str_schema()), + }, + model_name='Cat', + ), + ), + }, + discriminator='type', + strict=False, + from_attributes=True, + ) + ) + } + ), + ) + ) + + yard = Yard(pet=Dog(name='Rex')) + assert serializer.to_python(yard, mode='json') == {'pet': {'type': 'dog', 'name': 'Rex'}}