123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- class History < ApplicationModel
- include CanBeImported
- include History::Assets
- self.table_name = 'histories'
- belongs_to :history_type, class_name: 'History::Type', optional: true
- belongs_to :history_object, class_name: 'History::Object', optional: true
- belongs_to :history_attribute, class_name: 'History::Attribute', optional: true
- belongs_to :sourceable, polymorphic: true, optional: true
- =begin
- add a new history entry for an object
- History.add(
- history_type: 'updated',
- history_object: 'Ticket',
- history_attribute: 'state',
- o_id: ticket.id,
- id_to: 3,
- id_from: 2,
- value_from: 'open',
- value_to: 'pending reminder',
- created_by_id: 1,
- created_at: '2013-06-04 10:00:00',
- updated_at: '2013-06-04 10:00:00'
- )
- =end
- def self.add(data)
- # return if we run import mode
- return if Setting.get('import_mode') && !data[:id]
- # lookups
- if data[:history_type].present?
- history_type = type_lookup(data[:history_type])
- end
- if data[:history_object].present?
- history_object = object_lookup(data[:history_object])
- end
- related_history_object_id = nil
- if data[:related_history_object].present?
- related_history_object = object_lookup(data[:related_history_object])
- related_history_object_id = related_history_object.id
- end
- history_attribute_id = nil
- if data[:history_attribute].present?
- history_attribute = attribute_lookup(data[:history_attribute])
- history_attribute_id = history_attribute.id
- end
- # create history
- record = {
- id: data[:id],
- o_id: data[:o_id],
- history_type_id: history_type.id,
- history_object_id: history_object.id,
- history_attribute_id: history_attribute_id,
- sourceable: data[:sourceable],
- sourceable_name: data[:sourceable].try(:name),
- related_history_object_id: related_history_object_id,
- related_o_id: data[:related_o_id],
- value_from: data[:value_from],
- value_to: data[:value_to],
- id_from: data[:id_from],
- id_to: data[:id_to],
- created_at: data[:created_at],
- created_by_id: data[:created_by_id]
- }
- history_record = nil
- if data[:id]
- history_record = History.find_by(id: data[:id])
- end
- if history_record
- history_record.update!(record)
- else
- record_new = History.create!(record)
- if record[:id]
- record_new.id = record[:id]
- end
- record_new.save!
- end
- end
- =begin
- remove whole history entries of an object
- History.remove('Ticket', 123)
- =end
- def self.remove(requested_object, requested_object_id)
- history_object = History::Object.find_by(name: requested_object)
- return if !history_object
- History.where(
- history_object_id: history_object.id,
- o_id: requested_object_id,
- ).destroy_all
- end
- =begin
- return all history entries of an object
- history_list = History.list('Ticket', 123)
- returns
- history_list = [
- { ... },
- { ... },
- { ... },
- { ... },
- ]
- return all history entries of an object and it's related history objects
- history_list = History.list('Ticket', 123, 'Ticket::Article')
- returns
- history_list = [
- { ... },
- { ... },
- { ... },
- { ... },
- ]
- return all history entries of an object and it's assets
- history = History.list('Ticket', 123, nil, ['Ticket::Article'])
- returns
- history = {
- list: list,
- assets: assets,
- }
- =end
- def self.list(requested_object, requested_object_id, related_history_object = [], assets = nil, raw: false)
- histories = History.where(
- history_object_id: object_lookup(requested_object).id,
- o_id: requested_object_id
- )
- if related_history_object.present?
- object_ids = Array(related_history_object).map do |object|
- object_lookup(object).id
- end
- histories = histories.or(
- History.where(
- history_object_id: object_ids,
- related_o_id: requested_object_id
- )
- )
- end
- histories = histories.reorder(:created_at, :id)
- return histories if raw
- list = histories.map(&:attributes).each do |data|
- data['object'] = History::Object.lookup(id: data.delete('history_object_id'))&.name
- data['type'] = History::Type.lookup(id: data.delete('history_type_id'))&.name
- if data['history_attribute_id']
- data['attribute'] = History::Attribute.lookup(id: data.delete('history_attribute_id'))&.name
- end
- if data['related_history_object_id']
- data['related_object'] = History::Object.lookup(id: data.delete('related_history_object_id'))&.name
- end
- data.delete('updated_at')
- data.delete('related_o_id') if data['related_o_id'].nil?
- if data['id_to'].nil? && data['id_from'].nil?
- data.delete('id_from')
- data.delete('id_to')
- end
- if data['value_to'].nil? && data['value_from'].nil?
- data.delete('value_from')
- data.delete('value_to')
- end
- end
- return list if !assets
- {
- list: list,
- assets: histories.reduce({}) { |memo, obj| obj.assets(memo) }
- }
- end
- def self.type_lookup(name)
- # lookup
- history_type = History::Type.lookup(name: name)
- return history_type if history_type
- # create
- History::Type.create!(name: name)
- end
- def self.object_lookup(name)
- # lookup
- history_object = History::Object.lookup(name: name)
- return history_object if history_object
- # create
- History::Object.create!(name: name)
- end
- def self.attribute_lookup(name)
- # lookup
- history_attribute = History::Attribute.lookup(name: name)
- return history_attribute if history_attribute
- # create
- History::Attribute.create!(name: name)
- end
- end
|