123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- module HasHistory
- extend ActiveSupport::Concern
- included do
- attr_accessor :history_changes_last_done
- after_create :history_prefill, :history_create, :history_change_source_clear
- after_update :history_update, :history_change_source_clear
- after_destroy :history_destroy
- end
- def history_prefill
- return if @history_changes_source.blank?
- @history_changes_source.each do |key, value|
- next if !value.is_a?(PostmasterFilter)
- attribute_name = history_attribute_name(key)
- attribute_value = history_attribute_changes(key, [nil, self[key]])
- data = {
- history_attribute: attribute_name,
- }.merge(attribute_value)
- history_log('set', created_by_id, data)
- end
- end
- =begin
- log object create history, if configured - will be executed automatically
- model = Model.find(123)
- model.history_create
- =end
- def history_create
- history_log('created', created_by_id)
- end
- =begin
- log object update history with all updated attributes, if configured - will be executed automatically
- model = Model.find(123)
- model.history_update
- =end
- def history_update
- return if !saved_changes?
- # return if it's no update
- return if new_record?
- # new record also triggers update, so ignore new records
- changes = saved_changes
- history_changes_last_done&.each do |key, value|
- if changes.key?(key) && changes[key] == value
- changes.delete(key)
- end
- end
- self.history_changes_last_done = changes
- # logger.info 'updated ' + self.changes.inspect
- return if changes['id'] && !changes['id'][0]
- ignored_attributes = self.class.instance_variable_get(:@history_attributes_ignored) || []
- ignored_attributes += %i[created_at updated_at created_by_id updated_by_id]
- changes.each do |key, value|
- next if ignored_attributes.include?(key.to_sym)
- # get attribute name
- attribute_name = history_attribute_name(key)
- attribute_value = history_attribute_changes(key, value)
- data = {
- history_attribute: attribute_name,
- }.merge(attribute_value)
- # logger.info "HIST NEW #{self.class.to_s}.find(#{self.id}) #{data.inspect}"
- history_log('updated', updated_by_id, data)
- end
- end
- def history_attribute_name(key)
- attribute_name = key.to_s
- if attribute_name[-3, 3] == '_id'
- attribute_name = attribute_name[ 0, attribute_name.length - 3 ]
- end
- attribute_name
- end
- def history_attribute_changes(key, value_changes)
- attribute_name = history_attribute_name(key)
- value_id = []
- value_str = [ value_changes[0], value_changes[1] ]
- if key.to_s[-3, 3] == '_id'
- value_id[0] = value_changes[0]
- value_id[1] = value_changes[1]
- if respond_to?(attribute_name) && send(attribute_name)
- relation_class = send(attribute_name).class
- if relation_class && value_id[0]
- relation_model = relation_class.lookup(id: value_id[0])
- if relation_model
- if relation_model['name']
- value_str[0] = relation_model['name']
- elsif relation_model.respond_to?(:fullname)
- value_str[0] = relation_model.send(:fullname)
- end
- end
- end
- if relation_class && value_id[1]
- relation_model = relation_class.lookup(id: value_id[1])
- if relation_model
- if relation_model['name']
- value_str[1] = relation_model['name']
- elsif relation_model.respond_to?(:fullname)
- value_str[1] = relation_model.send(:fullname)
- end
- end
- end
- end
- end
- {
- value_from: value_str[0].to_s,
- value_to: value_str[1].to_s,
- id_from: value_id[0],
- id_to: value_id[1],
- }
- end
- =begin
- delete object history, will be executed automatically
- model = Model.find(123)
- model.history_destroy
- =end
- def history_destroy
- History.remove(self.class.to_s, id)
- end
- =begin
- create history entry for this object
- organization = Organization.find(123)
- result = organization.history_log('created', user_id)
- returns
- result = true # false
- =end
- def history_log(type, user_id, attributes = {})
- attributes.merge!(
- o_id: self['id'],
- history_type: type,
- history_object: self.class.name,
- related_o_id: nil,
- related_history_object: nil,
- created_by_id: user_id,
- updated_at: updated_at,
- created_at: updated_at,
- ).merge!(history_log_attributes)
- attributes[:sourceable] = @history_changes_source.try(:delete, attributes[:history_attribute]) || @history_changes_source.try(:delete, "#{attributes[:history_attribute]}_id") || @history_changes_source&.dig(type)
- History.add(attributes)
- end
- # callback function to overwrite
- # default history log attributes
- # gets called from history_log
- def history_log_attributes
- {}
- end
- =begin
- get history log for this object
- organization = Organization.find(123)
- result = organization.history_get
- returns
- result = [
- {
- :type => 'created',
- :object => 'Organization',
- :created_by_id => 3,
- :created_at => "2013-08-19 20:41:33",
- },
- {
- :type => 'updated',
- :object => 'Organization',
- :attribute => 'note',
- :o_id => 1,
- :id_to => nil,
- :id_from => nil,
- :value_from => "some note",
- :value_to => "some other note",
- :created_by_id => 3,
- :created_at => "2013-08-19 20:41:33",
- },
- ]
- to get history log for this object with all assets
- organization = Organization.find(123)
- result = organization.history_get(true)
- returns
- result = {
- :history => [
- { ... },
- { ... },
- ],
- :assets => {
- ...
- }
- }
- =end
- def history_get(fulldata = false)
- relation_object = history_relation_object
- if !fulldata
- return History.list(self.class.name, self['id'], relation_object)
- end
- # get related objects
- history = History.list(self.class.name, self['id'], relation_object, true)
- history[:list].each do |item|
- record = item['object'].constantize.lookup(id: item['o_id'])
- if record.present?
- history[:assets] = record.assets(history[:assets])
- end
- next if !item['related_object']
- record = item['related_object'].constantize.lookup(id: item['related_o_id'])
- if record.present?
- history[:assets] = record.assets(history[:assets])
- end
- end
- {
- history: history[:list],
- assets: history[:assets],
- }
- end
- def history_relation_object
- @history_relation_object ||= self.class.instance_variable_get(:@history_relation_object) || []
- end
- def history_change_source_clear
- @history_changes_source = nil
- @history_changes_source_last = nil
- end
- def history_change_source_attribute(source, attribute)
- return if source.blank?
- return if [Job, Trigger, PostmasterFilter].exclude?(source.class)
- return if !source.persisted?
- @history_changes_source ||= {}
- @history_changes_source[attribute] = source
- @history_changes_source_last = source
- end
- # methods defined here are going to extend the class, not the instance of it
- class_methods do
- =begin
- serve method to ignore model attributes in historization
- class Model < ApplicationModel
- include HasHistory
- history_attributes_ignored :create_article_type_id, :preferences
- end
- =end
- def history_attributes_ignored(*attributes)
- @history_attributes_ignored = attributes
- end
- =begin
- serve method to ignore model attributes in historization
- class Model < ApplicationModel
- include HasHistory
- history_relation_object 'Some::Relation::Object'
- end
- =end
- def history_relation_object(*attributes)
- @history_relation_object ||= []
- @history_relation_object |= attributes
- end
- end
- end
|