Подтвердить что ты не робот

Внедрение двунаправленных отношений в MongoEngine

Я создаю приложение Django, которое использует MongoDB и MongoEngine для хранения данных. Чтобы представить упрощенную версию моей проблемы, скажем, я хочу иметь два класса: "Пользователь и страница". Каждая страница должна ассоциироваться с пользователем и каждым пользователем страницей.

from mongoengine import *

class Page(Document):
    pass

class User(Document):
    name = StringField()
    page = ReferenceField(Page)

class Page(Document):
    content = StringField()
    user = ReferenceField(User)

(Обратите внимание, что страница должна быть определена до пользователя. Если мне не хватает Pythonic для обработки круговых зависимостей, сообщите мне.) Каждый документ может быть создан и сохранен просто отлично, но назначение страницы пользователю вызывает ошибку.

u = User(name='Jeff')
u.save()
p = Page(content="I'm a page!")
p.save()
p.user = u
p.save()
u.page = p
u.save()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "build\bdist.win32\egg\mongoengine\document.py", line 71, in save
  File "build\bdist.win32\egg\mongoengine\base.py", line 303, in validate
mongoengine.base.ValidationError: Invalid value for field of type "ReferenceField"

Может кто-нибудь объяснить, почему это исключение бросается, что я делаю неправильно, и как я могу его избежать?

4b9b3361

Ответ 1

Это правильное решение:

from mongoengine import *

class User(Document):
    name = StringField()
    page = ReferenceField('Page')

class Page(Document):
    content = StringField()
    user = ReferenceField(User)

Используйте одинарные кавычки ('Страница') для обозначения классов, которые еще не определены.

Ответ 2

Ответ Drew - лучший способ в этом случае, но я хотел бы упомянуть, что вы также можете использовать GenereicReferenceField:

from mongoengine import *

class User(Document):
    name = StringField()
    page = GenericReferenceField()

class Page(Document):
    content = StringField()
    user = ReferenceField(User)

Но опять же, для вашей конкретной проблемы, перейдите с именем класса в одинарные кавычки.