0

我正在尝试使用LiteralPydantic 创建一个有区别的联合。有关于 Job 资源的事件,我想用event_name. 因为JobPublishedEvents我想确保extra_field存在一些。

class GenericJobEvent(BaseModel):
    event_name: str
    id: int


class JobPublishedEvent(GenericJobEvent):
    event_name: Literal['job.published']
    extra_field: str


class Wrapper(BaseModel):
    wrapped: Union[JobPublishedEvent, GenericJobEvent]

print(type(Wrapper(wrapped={'event_name': 'some.event', 'id': 1}).wrapped))  # GenericJobEvent
print(type(Wrapper(wrapped={'event_name': 'job.published', 'id': 1, 'extra_field': 'extra'}).wrapped))  # JobPublishedEvent
print(type(Wrapper(wrapped={'event_name': 'job.published', 'id': 1}).wrapped))  # GenericJobEvent

前两种情况的行为符合预期,对于第三种情况,我想要一个验证错误,因为文字匹配,但架构没有实现。不过,我明白为什么回退到 GenericJobEvent 是有效的。

有人知道如何实现这一目标吗?

4

1 回答 1

1

问题是由于虽然JobPublishedEvent失败,GenericJobEvent但没有。想一想,event_nameattribute是一个string,theid是一个int。一切都匹配。

所以,你可以做的是验证event_nameGenericJobEvent拒绝事件名称job.published

您可以按照文档中的说明对其进行验证

@validator('event_name')
    def event_name_filter(cls, v):
        if 'job.published' == v:
            raise ValueError('GenericJobEvent cannot have event_name equal to job.published')
        return v.title()
于 2020-06-23T09:11:54.677 回答