Skip to content
Advertisement

Extending frozen dataclass and take all data from base class instance

Suppose we have a class coming from a library,

JavaScript

I have a dog constructor coming from an external library.

JavaScript

I would like this dog to have a new member, let’s say ‘bite‘. Obviously pluto['bite'] = True will fail, since dataclass is frozen.

So my idea is to make a subclass from Dog and get all the data from the ‘pluto’ instance.

JavaScript

Is there a way to avoid manually put all the class Dog parameters in init ? Something like a copy constructor.

ideally:

JavaScript

Advertisement

Answer

If you want to use inheritance to solve your problem, you need to start off with writing a proper AngryDog subclass that you can use to build sane instances from.

The next step would be to add a from_dog classmethod, something like this maybe:

JavaScript

But following this pattern, you’ll face a specific edge case, which you yourself already pointed out through the whatever parameter. When re-calling the Dog constructor, any InitVar will be missing in an asdict call, since they are not a proper member of the class. In fact, anything that takes place in a dataclass’ __post_init__, which is where InitVars go, might lead to bugs or unexpected behavior.

If it’s only minor stuff like filtering or deleting known parameters from the cls call and the parent class is not expected to change, you can just try to handle it in from_dog. But there is conceptually no way to provide a general solution for this kind of from_instance problem.


Composition would work bug-free from a data-integrity perspective, but might be unidiomatic or clunky given the exact matter at hand. Such a dog-extension wouldn’t be usable in-place of a proper dog-instance, but we could duck-type it into the right shape in case it’s necessary:

JavaScript

Usage:

JavaScript

But ultimately, the point remains that isinstance(dog_e, Dog) will return False. If you’re committed to make that call return True, there is some advanced trickery to help you out, and make anyone who inherits your code hate you:

JavaScript

Usage:

JavaScript

Most of the dataclass-added methods, like __repr__, will be broken though, including plugging doppelganger instances in things like dataclass.asdict or even just vars – so use at own risk.

User contributions licensed under: CC BY-SA
10 People found this is helpful
Advertisement