Обратный вызов для измененных атрибутов ActiveRecord?

Я знаю ActiveRecord:: Dirty и связанные с ним методы, но я не вижу средств, с помощью которых я могу подписаться на событие с измененным атрибутом. Что-то вроде:

class Person < ActiveRecord::Base
  def attribute_changed(attribute_name, old_value, new_value)
  end

  #or

  attribute_changed do |attribute_name, old_value, new_value|
  end
end

Есть ли для этого стандарт Rails или плагин? Я чувствую, что он должен быть где-то, и я просто пропустил его.

Ответы

Ответ 1

Ответ cwninja должен делать трюк, но есть немного больше.

Прежде всего, обработка базового атрибута выполняется с помощью метода write_attribute, поэтому вы должны использовать это.

Rails также имеет встроенную структуру обратного вызова, к которой можно было бы подключиться, хотя она не позволяет передавать аргументы, которые немного раздражают.

Используя пользовательские обратные вызовы, вы можете сделать это следующим образом:

class Person < ActiveRecord::Base

  def write_attribute(attr_name, value)
    attribute_changed(attr_name, read_attribute(attr_name), value)
    super
  end

  private

    def attribute_changed(attr, old_val, new_val)
      logger.info "Attribute Changed: #{attr} from #{old_val} to #{new_val}"
    end

 end

Если вы хотите попробовать использовать обратные вызовы Rails (особенно полезно, если у вас есть несколько обратных вызовов и/или подклассов), вы можете сделать что-то вроде этого:

class Person < ActiveRecord::Base
  define_callbacks :attribute_changed

  attribute_changed :notify_of_attribute_change

  def write_attribute(attr_name, value)
    returning(super) do
      @last_changed_attr = attr_name
      run_callbacks(:attribute_changed)
    end
  end

  private

    def notify_of_attribute_change
      attr = @last_changed_attr
      old_val, new_val = send("#{attr}_change")
      logger.info "Attribute Changed: #{attr} from #{old_val} to #{new_val}"
    end

end

Ответ 2

Для Rails 3:

class MyModel < ActiveRecord::Base
  after_update :my_listener, :if => :my_attribute_changed?

  def my_listener
    puts "Attribute 'my_attribute' has changed"
  end
end

Добавил комментарий: fooobar.com/questions/298155/...

Я не могу найти официальную документацию об этом.

Ответ 3

Для Rails 4

def attribute=(value)
  super(value)
  your_callback(self.attribute)
end

Если вы хотите перезаписать значение атрибута, вы должны использовать write_attribute(:attribute, your_callback(self.attribute)), а не attribute=, или вы будете перебирать его, пока не получите слишком глубокое исключение стека.

Ответ 4

попробовать:

def attribute_changed(attribute_name, old_value, new_value)
end

def attribute=(attribute_name, value)
  returning(super) do
    attribute_changed(attribute_name, attribute_was(attribute_name), attribute(attribute_name))
  end
end

Просто придумал это сейчас, но он должен работать.

Ответ 5

Вы всегда можете получить доступ к частному методу changed_attributes и проверить ключи там, используя before_save, и делать с ним то, что вы будете делать.

Ответ 6

Самый простой способ, который я нашел:

after_save :my_method, :if => Proc.new{ self.my_attribute_changed? }

def my_method
  ... do stuff...
end