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

Enums: Support for default member / __missing__ #531

Closed
iodbh opened this issue Aug 21, 2023 · 2 comments · Fixed by #532
Closed

Enums: Support for default member / __missing__ #531

iodbh opened this issue Aug 21, 2023 · 2 comments · Fixed by #532

Comments

@iodbh
Copy link

iodbh commented Aug 21, 2023

Description

feature

python enums supports __missing__, which can be used to resolve unknown members on an enum to a preset default member rather than erroring out.

support for this method - or some alternative way of specifying a "default" member when deserializing an enum field - is msgspec would be quite valuable.

use case

In our case, we use msgspec to deserialize deeply nested JSON from an API that heavily uses a "flag" concept. these flags are lists of strings that we map to enums, for instance a user object could have its flags set to ["ACTIVE", "ADMIN"]. New flags are added regularly, and there are multiple occurences of different flag sets at different nesting levels of our models.

As it stands, deserialization will fail with a validation error if any of the flags list contains a newly added flag. We would like to be able to deserialize, and use a default "UNKNOWN" value. The only option we currently have (that i'm aware of) is:

  1. attempt to deserialize the message using the model
  2. catch validation exceptions
  3. check whether the exception is caused by an unknown member in any of the (many) fields where we expect values to be added
  4. deserialize the message again, without using the model

There is of course the possibility of treating these fields as simple strings, but that means losing information about the data model.

example

Ideally, the following would be supported:

class Flag(Enum):
    ACTIVE = "active"
    ADMIN = "admin"
    UNKNOWN_FLAG = "unknown_flag"

    @classmethod
    def __missing__(cls, key):
        return cls.UNKNOWN_FLAG

class User(msgspec.Struct):
    username: str
    flags: List[Flag]

any alternative API allowing this sort of behavior would be a huge improvement.

@jcrist
Copy link
Owner

jcrist commented Aug 24, 2023

Thanks for opening this, this is definitely a feature worth supporting (it's even been brought up before). Should be fixed by #532.

@iodbh
Copy link
Author

iodbh commented Aug 31, 2023

I wasn't expecting such a quick fix ! This is is saving us a huge headache, thank you for your work on this library - it is greatly appreciated :)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants