通过指定其“id”(而不是通过其父级查找)来销毁嵌入的Mongoid文档

时间:2013-04-14 18:52:14

标签: ruby-on-rails ruby-on-rails-3 mongoid

我有这两个模型:

class Presentation
  include Mongoid::Document
  embeds_many :presentation_rows
end

class PresentationRow
  include Mongoid::Document
  embedded_in :presentation
end

在我的presentation_rows_controller.rb中,我有以下几行代码:

def show
  @presentation = Presentation.find(params[:id])
  @presentation_rows = @presentation.presentation_rows
end

def destroy
  ...
  ...
end

在我的presentation_rows / show.html.haml中,我有以下几行代码:

- @presentation_rows.each do |presentation_row|
  = link_to "Delete", presentation_row, method: :delete

我在destroy控制器动作中尝试了很多方法,但它们都指向了一个显而易见的事实,即我试图破坏嵌入式文档而不通过它的父级。但是现在我在我的视图文件中有了它的id的presentation_row,我不应该被允许销毁它。这似乎很愚蠢。

带有空的销毁操作的错误消息,fyi:

Started DELETE "/en/presentation_rows/516af0a983c336708300000f" for 127.0.0.1 at 2013-04-14 20:08:47 +0200
Processing by PresentationRowsController#destroy as HTML
  Parameters: {"authenticity_token"=>"KHGG2dsTseCl88okOKW9JAlHb+VaK2lKIxb0ptAIC7A=", "locale"=>"en", "id"=>"516af0a983c336708300000f"}
  MOPED: 127.0.0.1:27017 QUERY        database=shop_import_development collection=users selector={"$query"=>{"_id"=>"511a813a83c336a0ea000001"}, "$orderby"=>{:_id=>1}} flags=[:slave_ok] limit=-1 skip=0 fields=nil (0.3152ms)
  MOPED: 127.0.0.1:27017 QUERY        database=shop_import_development collection=presentations selector={"_id"=>"516af0a983c336708300000f"} flags=[:slave_ok] limit=0 skip=0 fields=nil (0.2129ms)
Completed 500 Internal Server Error in 2ms

Mongoid::Errors::DocumentNotFound (
Problem:
  Document(s) not found for class Presentation with id(s) 516af0a983c336708300000f.
Summary:
  When calling Presentation.find with an id or array of ids, each parameter must match a document in the database or this error will be raised. The search was for the id(s): 516af0a983c336708300000f ... (1 total) and the following ids were not found: 516af0a983c336708300000f.
Resolution:
  Search for an id that is in the database or set the Mongoid.raise_not_found_error configuration option to false, which will cause a nil to be returned instead of raising this error when searching for a single id, or only the matched documents when searching for multiples.):
  app/controllers/presentation_rows_controller.rb:16:in `correct_user?'


  Rendered /Users/christoffer/.rvm/gems/ruby-1.9.3-p385/gems/actionpack-3.2.8/lib/action_dispatch/middleware/templates/rescues/_trace.erb (1.0ms)
  Rendered /Users/christoffer/.rvm/gems/ruby-1.9.3-p385/gems/actionpack-3.2.8/lib/action_dispatch/middleware/templates/rescues/_request_and_response.erb (0.8ms)
  Rendered /Users/christoffer/.rvm/gems/ruby-1.9.3-p385/gems/actionpack-3.2.8/lib/action_dispatch/middleware/templates/rescues/diagnostics.erb within rescues/layout (9.2ms

我应该在摧毁行动中投入什么?

1 个答案:

答案 0 :(得分:7)

这是mongoid的限制。贡献者“必须始终通过父级访问嵌入式文档。” https://github.com/mongoid/mongoid/issues/348

请注意,这不是mongodb的限制。从其id获取PresentationRow对象的解决方法是......

pres = Presentation.where('presentation_rows._id' => Moped::BSON::ObjectId(params[:id])).first
row = pres.presentation_rows.detect { |pr| pr.id.to_s == params[:id] }
row.destroy

如果您有很多索引,请为presentation_rows._id添加索引。