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

Dataclass kw_only attribute incorrectly structured without detailed validation enabled #637

Closed
moltob opened this issue Mar 25, 2025 · 3 comments · Fixed by #638
Closed

Comments

@moltob
Copy link

moltob commented Mar 25, 2025

With cattrs version v24.1.2 the following test fails:

def test__cattrs_kw_only_propagation():
    import dataclasses

    import cattrs

    @dataclasses.dataclass
    class PartialKeywords:
        a1: str = 'Default'
        a2: str = dataclasses.field(kw_only=True)

    converter = cattrs.Converter(detailed_validation=False)

    assert converter.structure(
        {
            'a2': 'Value',
        },
        PartialKeywords,
    ) == PartialKeywords(
        a1='Default',
        a2='Value',
    )

What is the test showing?

A dataclass class is defined, with the first attribute possibly positional, and the second attribute with kw_only=True. The first attribute has a default value, so it is optional.

When structuring the passed dict, cattrs is expected to honor the kw_only attribute setting. It does not.

Suggested root cause

The _compat module has functions to support dataclasses by converting some features to the corresponding attrs concepts. The function adapted_fields() does this for class attribute descriptors by creating cattrs Attribute instances from dataclass fields.

The Attribute creation lacks the kw_only argument and therefore uses the default kw_only=False.

"Unfortunately" the defect hits only in one scenario:

  • When using BaseConverter, this does not hit, as structure_attrs_fromdict always uses kw-based creation.
  • When using Converter with detailed validation enabled, the generated instantiation script is also always using kw-based creation.
  • Only when using Converter without detailed validation support, the generated instantion script depends on the kw_only setting of an attribute.

The latter distinction is part of Converter.make_dict_structure_n_from_attrs, where you find code like:

if a.kw_only:
    invocation_line = f"{a.alias}={invocation_line}"

Options to fix

As only the non-detailed-validation case is currently respecting kw_only at all, this could could be removed to even in that case always use kw-based class creation in the instantiation script.

If that code should be kept, adapted_fields can possibly be fixed easily by propagating the kw_only setting:

Attribute(
    attr.name,
    ...,

    # FIX:
    kw_only=attr.kw_only,
)
for attr in attrs
]

This patch fixes the test above.

@moltob moltob changed the title Dataclass kw_only attribute incorrectly structured without detailed validation enbaled Dataclass kw_only attribute incorrectly structured without detailed validation enabled Mar 25, 2025
@Tinche
Copy link
Member

Tinche commented Mar 25, 2025

Mmm good catch. Let me see if I can release this as 24.1.3 soon.

@Tinche Tinche linked a pull request Mar 25, 2025 that will close this issue
@Tinche
Copy link
Member

Tinche commented Mar 25, 2025

New version is out! Great job on the analysis.

@Tinche Tinche closed this as completed Mar 25, 2025
@moltob
Copy link
Author

moltob commented Mar 26, 2025

Thank you for releasing this so quickly! This is a terrific library.

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