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

Doubly Nested Schemas instantiated via lambdas cannot be referenced with multiple dot notation with onlys/excludes #2165

Open
somethingnew2-0 opened this issue Aug 11, 2023 · 3 comments
Labels

Comments

@somethingnew2-0
Copy link
Contributor

somethingnew2-0 commented Aug 11, 2023

Related to #2164

The following works as intended

from marshmallow import Schema, fields

class ASchema(Schema):
    a = fields.String()
    a2 = fields.String()

class BSchema(Schema):
    b = fields.Nested(lambda: ASchema)

class CSchema(Schema):
    c = fields.Nested(lambda: BSchema)

class DSchema(Schema):
    d = fields.Nested(lambda: CSchema, only=("c.b.a2",))

class A:
    def __init__(self):
        self.a = 'Hello from a'
        self.a2 = 'Hello from a2'

class B:
    def __init__(self, a):
        self.b = a

class C:
    def __init__(self, b):
        self.c = b

class D:
    def __init__(self, c):
        self.d = c

print(DSchema().dump(D(C(B(A())))))

Outputting

{'d': {'c': {'b': {'a2': 'Hello from a2'}}}}

However when changing the CSchema nested field c lambda to return an instantiated BSchema(), the output is different and incorrect

class CSchema(Schema):
    c = fields.Nested(lambda: BSchema())

Outputs

{'d': {'c': {}}}

I've narrowed this issue down to https://github.com/marshmallow-code/marshmallow/blob/dev/src/marshmallow/fields.py#L601 where the self.only values in the instantiated BSchema() will be "b.a2" whereas original = self._schema.fields.keys() will be the top-level parent "b". This results in an empty set intersection meaning the serialized value will have no fields.

@lafrech lafrech added the bug label Aug 11, 2023
@lafrech
Copy link
Member

lafrech commented Aug 11, 2023

Thanks for the research and the comprehensive bug report.

A fix proposal would be very welcome, anyone.

@sloria
Copy link
Member

sloria commented Jan 2, 2025

passing only to the nested ctor also works as expected

class DSchema(Schema):
    d = fields.Nested(lambda: CSchema(only=("c.b.a2",)))

i agree this is a bug since we currently support passing only to Nested with a schema instance, but i wonder if it's something we should drop going forward? it's a bit odd to allow schema kwargs to Nested when they can be passed to the Schema ctor itself 🤔

@sloria
Copy link
Member

sloria commented Jan 19, 2025

i've clarified my above proposal in a new issue: #2794

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

No branches or pull requests

3 participants