-
-
Notifications
You must be signed in to change notification settings - Fork 118
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
Structuring a dict containing class instances #634
Comments
Definitely an interesting problem. First question: is your data definition recursive (in particular, for |
I am not sure that I understand the question properly, but I guess the answer is no. B's attributes are all standard types (int, str, list) except for the attribute/s containing references to A. |
I asked because there's a cute trick you can use if you don't want to change an existing converter: you can generate a hook but not register it. You can just call it directly. This doesn't work if your data is recursive and your hook expects to be able to call itself through the converter. Another solution might be copying an existing converter using Playing around with your problem, here's how I would possibly solve it: from threading import local
import attrs
import cattrs
@attrs.define
class A:
id: str
@attrs.define
class B:
member: A
data_A = [{"id": "A_1"}, {"id": "A_2"}]
data_B = [{"member": "A_1"}, {"member": "A_2"}]
converter = cattrs.Converter()
registry_of_a = local()
hook_for_b = cattrs.gen.make_dict_structure_fn(
B,
converter,
member=cattrs.override(struct_hook=lambda a_id, _: registry_of_a.a_dict[a_id]),
)
def structure_data():
a_instances = converter.structure(data_A, list[A])
a_dict = {a.id: a for a in a_instances}
registry_of_a.a_dict = a_dict
try:
b_instances = [hook_for_b(val) for val in data_B]
print(b_instances)
finally:
del registry_of_a.a_dict
structure_data() We use a thread local variable to keep our registry of As during the structuring process. If you don't need it to be threadsafe you can just use a module-level variable. We generate a custom hook for Then we structure the entire thing in What do you think? |
If I understand correctly, with this hook we are overriding how the It seems to me that using the In the case I don't want to use a global variable to store de Thanks in advance. |
I am dealing with some unstructured data that is supposed to be converted to some object instances of classes A and B.
This is an example of how the unstructured data might look:
In this example the desire output would be to obtain a list of instances of B where both instances contain a reference to the same instance of A. I will use cattrs to structure instances of A and B. To do that I write the following:
Now this does not work because cattrs will try to structure the A instances in the new_element dictionaries. I can write a custom hook to solve this and use the custom converter to structure both A and B.
This works but have some downsides:
Some workarounds that I have thought of:
Is there a better way of solving this problem?
The text was updated successfully, but these errors were encountered: