如何在Ecto MongoDB中执行像$ push $ pull $ set等原子操作

时间:2016-12-01 16:10:55

标签: mongodb elixir ecto

我使用的是mongodb_ecto,我想知道如何在深度嵌套的字段上执行$ push或$ pull等操作?目前,我回写整个文档,由于竞争条件,有时会导致数据库中存在错误数据。

1 个答案:

答案 0 :(得分:0)

好的,我有点想通了。不要使用Ecto。在某些情况下,您需要MongoDB positional operator,这只能通过Mongo-Adapter直接完成。现在来看一些用法示例:

我有一个选项列表的doucment。选项具有ID,标签和投票选择此选项的用户ID列表。

BTW生成一个ObjectID(直接与MongoDB-Adapter交谈时需要)使用:

id = "584a5b9419d51d724d146e3f" # string form
value = (for <<hex::16 <- id>>, into: <<>>, do: <<String.to_integer(<<hex::16>>, 16)::8>>)
object_id = %BSON.ObjectId{value: value}

现在举一些例子:

# update label of option
Mongo.update_one(Repo.Pool, "polls",
  %{"_id" => document_id, "options.id" => option_id}, # query
  %{"$set" => %{"data.options.$.label" => new_label}} # update
)

# add new option to poll
Mongo.update_one(Repo.Pool, "polls",
  %{"_id" => document_id},
  %{"$addToSet" => %{"options" => %{label: label, id: random_new_id, votes: []}}}
)

# add user_id to option
Mongo.update_one(Repo.Pool, "polls",
  %{"_id" => document_id, "options.id" => option_id},
  %{"$addToSet" => %{"options.$.votes" => user_id}}
)

# remove user_id form option
Mongo.update_one(Repo.Pool, "polls",
  %{"_id" => document_id, "options.id" => option_id},
  %{"$pull" => %{"data.options.$.votes" => user_id}}
)