# frozen_string_literal: true
module Dynamoid
# Support interface of Rails' ActiveModel::Dirty module
#
# The reason why not just include ActiveModel::Dirty -
# ActiveModel::Dirty conflicts either with @attributes or
# #attributes in different Rails versions.
#
# Separate implementation (or copy-pasting) is the best way to
# avoid endless monkey-patching
#
# Documentation:
# https://api.rubyonrails.org/v4.2/classes/ActiveModel/Dirty.html
module Dirty
extend ActiveSupport::Concern
include ActiveModel::AttributeMethods
included do
attribute_method_suffix '_changed?', '_change', '_will_change!', '_was'
attribute_method_suffix '_previously_changed?', '_previous_change'
attribute_method_affix prefix: 'restore_', suffix: '!'
end
# @private
module ClassMethods
def update_fields(*)
super.tap do |model|
model.clear_changes_information if model
end
end
def upsert(*)
super.tap do |model|
model.clear_changes_information if model
end
end
def from_database(*)
super.tap(&:clear_changes_information)
end
end
# @private
def save(*)
super.tap do |status|
changes_applied if status
end
end
# @private
def save!(*)
super.tap do
changes_applied
end
end
# @private
def update(*)
super.tap do
clear_changes_information
end
end
# @private
def update!(*)
super.tap do
clear_changes_information
end
end
# @private
def reload(*)
super.tap do
clear_changes_information
end
end
# Returns +true+ if any attribute have unsaved changes, +false+ otherwise.
#
# person.changed? # => false
# person.name = 'Bob'
# person.changed? # => true
#
# @return [true|false]
def changed?
changed_attributes.present?
end
# Returns an array with names of the attributes with unsaved changes.
#
# person = Person.new
# person.changed # => []
# person.name = 'Bob'
# person.changed # => ["name"]
#
# @return [Array[String]]
def changed
changed_attributes.keys
end
# Returns a hash of changed attributes indicating their original
# and new values like attr => [original value, new value].
#
# person.changes # => {}
# person.name = 'Bob'
# person.changes # => { "name" => ["Bill", "Bob"] }
#
# @return [ActiveSupport::HashWithIndifferentAccess]
def changes
ActiveSupport::HashWithIndifferentAccess[changed.map { |name| [name, attribute_change(name)] }]
end
# Returns a hash of attributes that were changed before the model was saved.
#
# person.name # => "Bob"
# person.name = 'Robert'
# person.save
# person.previous_changes # => {"name" => ["Bob", "Robert"]}
#
# @return [ActiveSupport::HashWithIndifferentAccess]
def previous_changes
@previously_changed ||= ActiveSupport::HashWithIndifferentAccess.new
end
# Returns a hash of the attributes with unsaved changes indicating their original
# values like attr => original value.
#
# person.name # => "Bob"
# person.name = 'Robert'
# person.changed_attributes # => {"name" => "Bob"}
#
# @return [ActiveSupport::HashWithIndifferentAccess]
def changed_attributes
@changed_attributes ||= ActiveSupport::HashWithIndifferentAccess.new
end
# Clear all dirty data: current changes and previous changes.
def clear_changes_information
@previously_changed = ActiveSupport::HashWithIndifferentAccess.new
@changed_attributes = ActiveSupport::HashWithIndifferentAccess.new
end
# Clears dirty data and moves +changes+ to +previous_changes+.
def changes_applied
@previously_changed = changes
@changed_attributes = ActiveSupport::HashWithIndifferentAccess.new
end
# Remove changes information for the provided attributes.
#
# @param attributes [Array[String]] - a list of attributes to clear changes for
def clear_attribute_changes(names)
attributes_changed_by_setter.except!(*names)
end
# Handle *_changed? for +method_missing+.
#
# person.attribute_changed?(:name) # => true
# person.attribute_changed?(:name, from: 'Alice')
# person.attribute_changed?(:name, to: 'Bob')
# person.attribute_changed?(:name, from: 'Alice', to: 'Bod')
#
# @private
# @param name [Symbol] attribute name
# @param options [Hash] conditions on +from+ and +to+ value (optional)
# @option options [Symbol] :from previous attribute value
# @option options [Symbol] :to current attribute value
def attribute_changed?(name, options = {})
result = changes_include?(name)
result &&= options[:to] == read_attribute(name) if options.key?(:to)
result &&= options[:from] == changed_attributes[name] if options.key?(:from)
result
end
# Handle *_was for +method_missing+.
#
# person = Person.create(name: 'Alice')
# person.name = 'Bob'
# person.attribute_was(:name) # => "Alice"
#
# @private
# @param name [Symbol] attribute name
def attribute_was(name)
attribute_changed?(name) ? changed_attributes[name] : read_attribute(name)
end
# Restore all previous data of the provided attributes.
#
# @param attributes [Array[Symbol]] a list of attribute names
def restore_attributes(names = changed)
names.each { |name| restore_attribute! name }
end
# Handles *_previously_changed? for +method_missing+.
#
# person = Person.create(name: 'Alice')
# person.name = 'Bob'
# person.save
# person.attribute_changed?(:name) # => true
#
# @private
# @param name [Symbol] attribute name
# @return [true|false]
def attribute_previously_changed?(name)
previous_changes_include?(name)
end
# Handles *_previous_change for +method_missing+.
#
# person = Person.create(name: 'Alice')
# person.name = 'Bob'
# person.save
# person.attribute_previously_changed(:name) # => ["Alice", "Bob"]
#
# @private
# @param name [Symbol]
# @return [Array]
def attribute_previous_change(name)
previous_changes[name] if attribute_previously_changed?(name)
end
private
def changes_include?(name)
attributes_changed_by_setter.include?(name)
end
alias attribute_changed_by_setter? changes_include?
# Handle *_change for +method_missing+.
def attribute_change(name)
[changed_attributes[name], read_attribute(name)] if attribute_changed?(name)
end
# Handle *_will_change! for +method_missing+.
def attribute_will_change!(name)
return if attribute_changed?(name)
begin
value = read_attribute(name)
value = value.clone if value.duplicable?
rescue TypeError, NoMethodError
end
set_attribute_was(name, value)
end
# Handle restore_*! for +method_missing+.
def restore_attribute!(name)
if attribute_changed?(name)
write_attribute(name, changed_attributes[name])
clear_attribute_changes([name])
end
end
# Returns +true+ if name were changed before the model was saved,
# +false+ otherwise.
def previous_changes_include?(name)
previous_changes.include?(name)
end
# This is necessary because `changed_attributes` might be overridden in
# other implemntations (e.g. in `ActiveRecord`)
alias attributes_changed_by_setter changed_attributes
# Force an attribute to have a particular "before" value
def set_attribute_was(name, old_value)
attributes_changed_by_setter[name] = old_value
end
end
end