123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- RSpec.shared_examples 'HasHistory' do |history_relation_object: []|
- describe 'auto-creation of history records' do
- let(:histories) { History.where(history_object_id: History::Object.find_by(name: described_class.name)) }
- context 'on creation' do
- it 'creates a History record for it' do
- expect { subject }.to change(histories, :count).by(1)
- expect(histories.last.history_type.name).to eq('created')
- end
- end
- context 'on update' do
- let(:histories) do
- History.where(history_object_id: History::Object.lookup(name: described_class.name).id,
- history_type_id: History::Type.lookup(name: 'updated').id,
- history_attribute_id: History::Attribute.find_or_create_by(name: attribute).id)
- end
- let!(:old_value) { subject.send(attribute) }
- shared_examples 'attribute update' do
- it 'creates a History record for it' do
- expect { subject.update(attribute => new_value) }.to change(histories, :count).by(1)
- expect(histories.last.attributes).to include(attributes)
- end
- end
- describe 'of #active', if: described_class.attribute_names.include?('active') do
- let(:attribute) { 'active' }
- let(:new_value) { !subject.active }
- let(:attributes) { { 'value_from' => old_value.to_s, 'value_to' => new_value.to_s } }
- include_examples 'attribute update'
- end
- describe 'of #body', if: described_class.attribute_names.include?('body') do
- let(:attribute) { 'body' }
- let(:new_value) { 'Lorem ipsum dolor' }
- let(:attributes) { { 'value_from' => old_value, 'value_to' => new_value } }
- include_examples 'attribute update'
- end
- describe 'of #email', if: described_class.attribute_names.include?('email') do
- let(:attribute) { 'email' }
- let(:new_value) { Faker::Internet.email }
- let(:attributes) { { 'value_from' => old_value, 'value_to' => new_value } }
- include_examples 'attribute update'
- end
- describe 'of #lastname', if: described_class.attribute_names.include?('lastname') do
- let(:attribute) { 'lastname' }
- let(:new_value) { 'Foo' }
- let(:attributes) { { 'value_from' => old_value, 'value_to' => new_value } }
- include_examples 'attribute update'
- end
- describe 'of #name', if: described_class.attribute_names.include?('name') do
- let(:attribute) { 'name' }
- let(:new_value) { 'Foo' }
- let(:attributes) { { 'value_from' => old_value, 'value_to' => new_value } }
- include_examples 'attribute update'
- end
- describe 'of #state', if: described_class.attribute_names.include?('state_id') do
- let(:attribute) { 'state' }
- let(:new_value) { state_class.where.not(id: old_value.id).first }
- let(:state_class) { "#{described_class.name}::State".constantize }
- let(:attributes) { { 'value_from' => old_value.name, 'value_to' => new_value.name } }
- include_examples 'attribute update'
- end
- describe 'of #title', if: described_class.attribute_names.include?('title') do
- let(:attribute) { 'title' }
- let(:new_value) { 'foo' }
- let(:attributes) { { 'value_from' => old_value, 'value_to' => new_value } }
- include_examples 'attribute update'
- end
- context 'when validations or callbacks prevent update' do
- shared_examples 'failed attribute update' do
- it 'does not create a History record for it' do
- expect { subject.update(attribute => new_value) }.not_to change(histories, :count)
- end
- end
- describe 'of #owner', if: described_class.attribute_names.include?('owner_id') do
- let(:attribute) { 'owner' }
- let(:new_value) { create(:customer) } # Ticket#owner is restricted to active agents of the same group
- include_examples 'failed attribute update'
- end
- end
- end
- end
- describe '#history_get' do
- context 'without "full" flag' do
- it 'delegates to History.list for self' do
- expect(History).to receive(:list).with(described_class.name, subject.id, history_relation_object)
- subject.history_get
- end
- end
- context 'with "full" flag' do
- it 'returns a hash including History.list for self' do
- expect(subject.history_get(true))
- .to include(history: History.list(described_class.name, subject.id, history_relation_object))
- end
- it 'returns a hash including FE assets of self and related objects' do
- expect(subject.history_get(true)[:assets][described_class.to_app_model]).to include(subject.assets({})[described_class.to_app_model])
- end
- end
- end
- end
|