更新mongoengine中的嵌入文档列表

时间:2012-09-12 11:47:06

标签: python mongodb pymongo mongoengine

我正在努力使用mongoengine语法。

我有以下型号......

class Post(EmbeddedDocument):
    uid = StringField(required=True)
    text = StringField(required=True)
    when = DateTimeField(required=True)


class Feed(Document):
    label = StringField(required=True)
    feed_url = StringField(required=True)
    posts = ListField(EmbeddedDocumentField(Post))

    def my_method(self, post):
        pass

...并且将post对象传递给my_method,我想更新现有帖子,如果它存在于具有匹配uid的self.posts中,或者如果不存在则推送到self.posts。

在mongoengine的一次调用中是否有语法?

2 个答案:

答案 0 :(得分:15)

否,对于列表字段,您无法在单个查询中将列表插入到列表中。 $addToSet因为您已更改post而无法正常工作,因此您无法匹配。您可以对此进行编码,但它确实会创建一个竞争条件,其中存在错误的小机会窗口,例如:

    class Post(EmbeddedDocument):
        uid = StringField(required=True)
        text = StringField(required=True)

    class Feed(Document):
        label = StringField(required=True)
        feed_url = StringField(required=True)
        posts = ListField(EmbeddedDocumentField(Post))

    Feed.drop_collection()

    Feed(
        label="label",
        feed_url="www.feed.com"
    ).save()

    post = Post(uid='1', text="hi")
    updated = Feed.objects(posts__uid=post.uid).update_one(set__posts__S=post)
    if not updated:
        Feed.objects.update_one(push__posts=post)

首先我们尝试更新,如果它不存在,我们推送到列表 - 这是另一个进程运行的机会窗口,可能会推送列表上的post

风险可能是可以接受的,但实际上,我认为更改模式会更好,可能会将Post分成自己的集合。然后,您可以使用更新语句并设置整个对象。费用将是获取Feed数据的额外查询。

答案 1 :(得分:0)

Feed.objects.filter(posts__uid=post.uid).\
          update_one(push__posts__S__comments='comment demo')