我必须追踪脏东西。它适用于父文档。但是,当我更改 doc 中的嵌入或引用时,必须通过 doc 本身中的嵌入/引用来访问脏。
如何跟踪父文档本身的脏数据?
我整理了 mongoid 扩展来解决这个问题https://github.com/versative/mongoid_relations_dirty_tracking。它只是以类似于 mongoid 对属性进行跟踪的方式跟踪文档关系的更改,以便正确更新时间戳和版本。
我认为 Mongoid 自己的脏属性功能不会跟踪嵌入式文档的更改。但是,自己实现对嵌入式文档的基本脏跟踪并不难,如下所示:
class Bar
include Mongoid::Document
embeds_many :foos
attr_accessor :foos_changed
alias_method :foos_changed?, :foos_changed
# ...
end
class Foo
include Mongoid::Document
embedded_in :bar, :inverse_of => :foos
after_save :notify_parent
def notify_parent
bar.foos_changed = true
end
# ...
end
b = Bar.create
b.foos_changed? # => nil
b.foos.create
b.foos_changed? # => true
如果您需要更多选项,例如在其中包含嵌入式文档更改Bar.changes
或跟踪特定更改,则必须覆盖changes
andchanged?
方法,除非您绝对需要这些功能,否则我不会这样做。
这是一种跟踪嵌入文档中更改内容的方法:
class Bar
include Mongoid::Document
embeds_many :foos
attr_writer :embedded_changes
def embedded_changes
@embedded_changes ||= begin
self._children.inject({}) do |memo, child|
memo.merge(
{child.collection_name => {child.id => child.changes}}
) if child.changed?
end
end
end
def changes
original_value = super
if original_value.blank?
embedded_changes
else
original_value.merge(embedded_changes)
end
end
def changed?
super || self._children.any?(&:changed?)
end
# ...
end
class Foo
include Mongoid::Document
embedded_in :bar, :inverse_of => :foos
# ...
field :hat
end
b = Bar.create
f = b.foos.create
b.changes # => nil
b.changed? # => false
f.hat = 1
f.changes # => {"hat"=>[nil, "1"]}
b.changes # => {"foos"=>{BSON::ObjectId('4cf...')=>{"hat"=>[nil, "1"]}}}
b.changed? # => true