如何使用mongoid / moped进行upsert / push

时间:2012-07-18 17:39:39

标签: ruby mongodb mongoid mongoid3 moped

我使用Mongoid(v3)访问MongoDB,并希望执行此操作:

db.sessionlogs.update( 
    {sessionid: '12345'}, /* selection criteria */
    {'$push':{rows: "new set of data"}},  /* modification */
    true /* upsert */
);

这在mongo shell中运行良好。它也正是我想要的,因为它是一个对我来说很重要的原子操作,因为我会调用它。我不想做两个操作 - 获取然后更新。我通过mongoid尝试了很多东西,但无法让它工作。

如何让MongoID不受影响,只需将此命令发送给MongoDB?我猜测在Moped级别有一些方法可以做到这一点,但该库的文档基本上不存在。

3 个答案:

答案 0 :(得分:10)

[在写这个问题时找到了答案......]

criteria = Sessionlogs.collection.find(:sessionid => sessionid)
criteria.upsert("$push" => {"rows" => datarow})

答案 1 :(得分:2)

这是一种方法:

session_log = SessionLog.new(session_id: '12345')
session_log.upsert
session_log.push(:rows, "new set of data")

或另一个:

SessionLog.find_or_create_by(session_id: '12345').
  push(:rows, "new set of data")

#push在该字段上执行原子$push。它是在解释 Atomic Persistence页。

(注意:这些示例使用UpperCamelCase和snake_case,就像Ruby惯例一样。)

答案 2 :(得分:0)

暂时不要去轻便摩托车,你可以使用查找和修改操作来实现同样的事情(所有默认范围和继承好东西)

如果不存在则在图表中保存边缘的示例

edge = {source_id: session[:user_id],dest_id:product._id, name: edge_name}
ProductEdge.where(edge).find_and_modify(ProductEdge.new(edge).as_document,{upsert:true})