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

Fixes querying nested fields in dynamic embedded docs #2647

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 8 additions & 7 deletions mongoengine/base/document.py
Original file line number Diff line number Diff line change
Expand Up @@ -1150,20 +1150,21 @@ def _lookup_field(cls, parts):
if hasattr(getattr(field, "field", None), "lookup_member"):
new_field = field.field.lookup_member(field_name)

# If the parent field is a DynamicField or if it's part of
# a DynamicDocument, mark current field as a DynamicField
# If the parent field is a DynamicField, mark current field as a DynamicField
# with db_name equal to the field name.
elif cls._dynamic and (
isinstance(field, DynamicField)
or getattr(getattr(field, "document_type", None), "_dynamic", None)
):
elif isinstance(field, DynamicField):
new_field = DynamicField(db_field=field_name)

# Else, try to use the parent field's lookup_member method
# Try to use the parent field's lookup_member method
# to find the subfield.
elif hasattr(field, "lookup_member"):
new_field = field.lookup_member(field_name)

# Else, if the current field it's part of a DynamicDocument, mark current field
# as a DynamicField with db_name equal to the field name.
elif cls._dynamic and getattr(getattr(field, "document_type", None), "_dynamic", None):
new_field = DynamicField(db_field=field_name)

# Raise a LookUpError if all the other conditions failed.
else:
raise LookUpError(
Expand Down
20 changes: 14 additions & 6 deletions mongoengine/fields.py
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@
from mongoengine.errors import (
DoesNotExist,
InvalidQueryError,
LookUpError,
ValidationError,
)
from mongoengine.queryset import DO_NOTHING
Expand Down Expand Up @@ -794,9 +795,12 @@ def validate(self, value, clean=True):
def lookup_member(self, member_name):
doc_and_subclasses = [self.document_type] + self.document_type.__subclasses__()
for doc_type in doc_and_subclasses:
field = doc_type._fields.get(member_name)
if field:
return field
try:
field = doc_type._lookup_field(member_name)
if field:
return field[0]
except LookUpError:
return None

def prepare_query_value(self, op, value):
if value is not None and not isinstance(value, self.document_type):
Expand Down Expand Up @@ -854,9 +858,13 @@ def lookup_member(self, member_name):
for document_choice in document_choices:
doc_and_subclasses = [document_choice] + document_choice.__subclasses__()
for doc_type in doc_and_subclasses:
field = doc_type._fields.get(member_name)
if field:
return field
try:
field = doc_type._lookup_field(member_name)
if field:
return field[0]
except LookUpError:
# Try with the next doc_type
pass

def to_mongo(self, document, use_db_field=True, fields=None):
if document is None:
Expand Down
54 changes: 54 additions & 0 deletions tests/document/test_dynamic.py
Original file line number Diff line number Diff line change
Expand Up @@ -223,6 +223,60 @@ class Doc(DynamicDocument):
with pytest.raises(ValidationError):
doc.validate()

def test_complex_embedded_document_query(self):
"""Ensure embedded dynamic documents can be queried"""

class EmbeddedDoc1(DynamicEmbeddedDocument):
email = StringField()

class EmbeddedDoc2(DynamicEmbeddedDocument):
name = StringField()

class Doc(DynamicDocument):
embedded = EmbeddedDocumentField(EmbeddedDoc1)
generic_embedded = GenericEmbeddedDocumentField(choices=(EmbeddedDoc1, EmbeddedDoc2))

Doc.drop_collection()

doc1 = Doc(content='doc1')
doc1.embedded = EmbeddedDoc1(email='[email protected]')
doc1.embedded.dynamic_field = 'dynamic value 1'
doc1.generic_embedded = EmbeddedDoc1(email='[email protected]')
doc1.generic_embedded.another_dynamic_field = 'generic dynamic value 1'
doc1.save()

doc2 = Doc(content='doc2')
doc2.embedded = EmbeddedDoc1(email='[email protected]')
doc2.embedded.dynamic_field = 'dynamic value 2'
doc2.generic_embedded = EmbeddedDoc2(name='doc2 name')
doc2.generic_embedded.another_dynamic_field = 'generic dynamic value 2'
doc2.save()

assert Doc.objects(embedded__dynamic_field='dynamic value 1').first().id == doc1.id
assert Doc.objects(embedded__dynamic_field='dynamic value 2').first().id == doc2.id

assert Doc.objects(generic_embedded__another_dynamic_field='generic dynamic value 1').first().id == doc1.id
assert Doc.objects(generic_embedded__another_dynamic_field='generic dynamic value 2').first().id == doc2.id

def test_complex_embedded_document_with_aliased_field_query(self):
class EmbeddedDoc(DynamicEmbeddedDocument):
first_name = StringField(db_field='firstName')

class Doc(DynamicDocument):
embedded_doc = EmbeddedDocumentField(EmbeddedDoc, db_field='embeddedDoc')

Doc.drop_collection()

doc1 = Doc(content='doc1')
doc1.embedded_doc = EmbeddedDoc(first_name='Alice')
doc1.save()

doc2 = Doc(content='doc1')
doc2.embedded_doc = EmbeddedDoc(first_name='John')
doc2.save()

assert Doc.objects(embedded_doc__first_name='John').first().id == doc2.id

def test_inheritance(self):
"""Ensure that dynamic document plays nice with inheritance"""

Expand Down