1

使用 DataMapper,我注意到除非我修改整个Object,否则属性不会更新 - 更新对象的字段然后保存不会持久更改。

我正在更新我的类的value属性,UnitValue我将其作为Object属性存储在我的ProductQuantity类中。

require 'data_mapper'

class ProductQuantity
  include DataMapper::Resource
  property :id, Serial
  property :quantity, Object, :required => true
end

# RSpec:

# Create a UnitValue with value 300, save as the Object property of this ProductQuantity
prod_quantity1 = ProductQuantity.create(:quantity => UnitValue.new(300, 'kg'))
expect(prod_quantity1.quantity.value).to eql(300) # true

# Replace the entire UnitValue object
prod_quantity1.quantity = UnitValue.new(400, 'kg')
expect(prod_quantity1.quantity.value).to eql(400)
prod_quantity1.save

# Check save worked when modifying the entire object
expect(ProductQuantity.get(prod_quantity1.id).quantity.value).to eql(400) # true

# Modify only a single field
prod_quantity1.quantity.value = 500
prod_quantity1.save

# Check save worked when modifying a single object - this FAILS
expect(ProductQuantity.get(prod_quantity1.id).quantity.value).to eql(500) # false
4

1 回答 1

0

在查看文档并找到这个过时的线程后,我相信我已经找到了解决方案。DataMapper::Resource您可以通过更改实例的 persistence_state 将属性标记为脏。

DirtyState = DataMapper::Resource::PersistenceState::Dirty
quantity_property = ProductQuantity.properties[:quantity]

old_quantity = prod_quantity1.quantity
prod_quantity1.quantity.value = 500
dirty_state = DataMapper::Resource::PersistenceState::Dirty.new(prod_quantity1)
dirty_state.original_attributes[quantity_property] = old_quantity
prod_quantity1.persistence_state = dirty_state
expect(prod_quantity1.persistence_state.is_a? DirtyState).to eql(true)
expect(prod_quantity1.dirty?).to eql(true)
prod_quantity1.save
expect(prod_quantity1.dirty?).to eql(false)

基本上我们创建一个DataMapper::Resource::PersistenceState::Dirty对象,然后使用我的 ProductQuantity 的“数量”属性作为键,我们将值设置为original_attributes. 任何非空地图都将为prod_quantity1.dirty?.

我已经把它做成了下面的一个模块。只需调用<resource_instance>.make_dirty(*attributes)一些属性名称即可。

require 'data_mapper'

module DataMapper
  module Resource

    # Make the give attributes dirty
    def make_dirty(*attributes)
      if attributes.empty?
        return
      end
      unless self.clean?
        self.save
      end
      dirty_state = DataMapper::Resource::PersistenceState::Dirty.new(self)
      attributes.each do |attribute|
        property = self.class.properties[attribute]
        # Any value will do here and return true for self.dirty?, but it expects the old version of this attribute.
        dirty_state.original_attributes[property] = nil
        self.persistence_state = dirty_state
      end
    end

  end
end

我已经在下面进行了测试:

describe DataMapper::Resource do

  before(:all) do
    DataMapper.auto_migrate!
    @prod1 = Product.create(:name => 'Snickers')
  end

  after(:all) do
    DataMapper.auto_migrate!
  end

  it 'should fail when an attribute is not dirty' do
    prod_quantity1 = ProductQuantity.create(:product => @prod1, :quantity => UnitValue.new(300, 'kg'))
    prod_quantity1.quantity.value = 400
    expect(prod_quantity1.dirty?).to eql(false)
    prod_quantity1.save
    prod_quantity1 = ProductQuantity.get(prod_quantity1.id)
    expect(prod_quantity1.quantity.value).to eql(300)
  end

  it 'should mark an attribute as dirty' do
    prod_quantity1 = ProductQuantity.create(:product => @prod1, :quantity => UnitValue.new(300, 'kg'))
    prod_quantity1.quantity.value = 400
    expect(prod_quantity1.dirty?).to eql(false)
    prod_quantity1.make_dirty(:quantity)
    expect(prod_quantity1.dirty?).to eql(true)
    prod_quantity1.save
    expect(prod_quantity1.quantity.value).to eql(400)
    expect(prod_quantity1.dirty?).to eql(false)
  end

end
于 2013-09-09T15:20:52.690 回答