123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- require 'models/application_model_examples'
- require 'models/concerns/can_csv_import_examples'
- require 'models/concerns/can_csv_import_organization_examples'
- require 'models/concerns/has_history_examples'
- require 'models/concerns/has_search_index_backend_examples'
- require 'models/concerns/has_xss_sanitized_note_examples'
- require 'models/concerns/has_image_sanitized_note_examples'
- require 'models/concerns/has_object_manager_attributes_examples'
- require 'models/concerns/has_taskbars_examples'
- RSpec.describe Organization, type: :model do
- subject(:organization) { create(:organization) }
- it_behaves_like 'ApplicationModel', can_assets: { associations: :members }
- it_behaves_like 'CanCsvImport', unique_attributes: 'name'
- include_examples 'CanCsvImport - Organization specific tests'
- it_behaves_like 'HasHistory'
- it_behaves_like 'HasSearchIndexBackend', indexed_factory: :organization
- it_behaves_like 'HasXssSanitizedNote', model_factory: :organization
- it_behaves_like 'HasImageSanitizedNote', model_factory: :organization
- it_behaves_like 'HasObjectManagerAttributes'
- it_behaves_like 'HasTaskbars'
- describe 'Class methods:' do
- describe '.where_or_cis' do
- it 'finds instance by querying multiple attributes case insensitive' do
- # search for Zammad Foundation
- organizations = described_class.where_or_cis(%i[name note], '%zammad%')
- expect(organizations).not_to be_blank
- end
- end
- describe '.destroy' do
- let!(:refs_known) { { 'Ticket' => { 'organization_id'=> 1 }, 'User' => { 'organization_id'=> 1 } } }
- let!(:user) { create(:customer, organization: organization) }
- let!(:ticket) { create(:ticket, organization: organization, customer: user) }
- it 'checks known refs' do
- refs_organization = Models.references('Organization', organization.id, true)
- expect(refs_organization).to eq(refs_known)
- end
- context 'with associations' do
- it 'checks user deletion' do
- organization.destroy(associations: true)
- expect { user.reload }.to raise_exception(ActiveRecord::RecordNotFound)
- end
- it 'checks ticket deletion' do
- organization.destroy(associations: true)
- expect { ticket.reload }.to raise_exception(ActiveRecord::RecordNotFound)
- end
- end
- context 'without associations' do
- it 'checks user deletion' do
- organization.destroy
- expect(user.reload.organization_id).to be_nil
- end
- it 'checks ticket deletion' do
- organization.destroy
- expect(ticket.reload.organization_id).to be_nil
- end
- end
- describe 'when changes for member_ids' do
- let(:agent1) { create(:agent) }
- let(:agent2) { create(:agent) }
- let(:agent3) { create(:agent) }
- let(:organization_agents) { create(:organization, member_ids: [agent1.id, agent2.id, agent3.id]) }
- it 'does not delete users' do
- organization_agents.update(member_ids: [agent1.id, agent2.id])
- expect { agent3.reload }.not_to raise_error
- end
- end
- end
- end
- describe 'Callbacks, Observers, & Async Transactions -' do
- describe 'Touching associations on update:' do
- let!(:member) { create(:customer, organization: organization) }
- let!(:member_ticket) { create(:ticket, customer: member) }
- context 'when member associations are added' do
- let(:user) { create(:customer) }
- it 'is touched, and touches its other members (but not their tickets)' do
- expect { organization.members.push(user) }
- .to change { organization.reload.updated_at }
- end
- end
- end
- end
- describe '#domain_assignment' do
- it 'fails if enabled and domain is missing' do
- organization.domain_assignment = true
- organization.domain = nil
- organization.valid?
- expect(organization.errors[:domain]).to be_present
- end
- it 'succeeds if enabled and domain is present' do
- organization.domain_assignment = true
- organization.domain = 'example.org'
- organization.valid?
- expect(organization.errors[:domain]).to be_empty
- end
- it 'succeeds if disabled and domain is missing' do
- organization.domain_assignment = false
- organization.domain = nil
- organization.valid?
- expect(organization.errors[:domain]).to be_empty
- end
- end
- describe 'Updating organization members' do
- context 'when member gets removed' do
- let(:customer) { create(:customer, organization: organization) }
- before do
- customer.attributes_with_association_ids
- organization.attributes_with_association_ids
- end
- it 'does clear cache of customer after user unassignment' do
- organization.update(member_ids: [])
- expect(customer.reload.attributes_with_association_ids['organization_id']).to be_nil
- end
- it 'does touch customer after user unassignment' do
- expect { organization.update(member_ids: []) }.to change { customer.reload.updated_at }
- end
- it 'does clear cache of organization after user unassignment' do
- organization.update(member_ids: [])
- expect(organization.reload.attributes_with_association_ids['member_ids']).not_to include(customer.id)
- end
- it 'does touch organization after user unassignment' do
- expect { organization.update(member_ids: []) }.to change { organization.reload.updated_at }
- end
- end
- context 'when member gets added' do
- let(:customer) { create(:customer) }
- before do
- customer.attributes_with_association_ids
- organization.attributes_with_association_ids
- end
- it 'does clear cache of customer after user assignment' do
- organization.update(member_ids: [customer.id])
- expect(customer.reload.attributes_with_association_ids['organization_id']).not_to be_nil
- end
- it 'does touch customer after user assignment' do
- expect { organization.update(member_ids: [customer.id]) }.to change { customer.reload.updated_at }
- end
- it 'does clear cache of organization after user assignment' do
- organization.update(member_ids: [customer.id])
- expect(organization.reload.attributes_with_association_ids['member_ids']).to include(customer.id)
- end
- it 'does touch organization after user assignment' do
- expect { organization.update(member_ids: [customer.id]) }.to change { organization.reload.updated_at }
- end
- end
- end
- describe 'Updating secondary organization members' do
- context 'when member gets removed' do
- let(:customer) { create(:customer, organization: organization) }
- let(:secondary_organization) { create(:organization) }
- before do
- secondary_organization.update(member_ids: [customer.id])
- customer.attributes_with_association_ids
- organization.attributes_with_association_ids
- secondary_organization.attributes_with_association_ids
- end
- it 'does clear cache of customer after user unassignment' do
- secondary_organization.update(member_ids: [])
- expect(customer.reload.attributes_with_association_ids['organization_id']).to be_nil
- end
- it 'does touch customer after user unassignment' do
- expect { secondary_organization.update(member_ids: []) }.to change { customer.reload.updated_at }
- end
- it 'does clear cache of organization after user unassignment' do
- secondary_organization.update(member_ids: [])
- expect(secondary_organization.reload.attributes_with_association_ids['member_ids']).not_to include(customer.id)
- end
- it 'does touch organization after user unassignment' do
- expect { secondary_organization.update(member_ids: []) }.to change { secondary_organization.reload.updated_at }
- end
- end
- context 'when member gets added' do
- let(:customer) { create(:customer, organization: organization) }
- let(:secondary_organization) { create(:organization) }
- before do
- customer.attributes_with_association_ids
- organization.attributes_with_association_ids
- secondary_organization.attributes_with_association_ids
- end
- it 'does clear cache of customer after user assignment' do
- secondary_organization.update(secondary_member_ids: [customer.id])
- expect(customer.reload.attributes_with_association_ids['organization_id']).not_to be_nil
- end
- it 'does touch customer after user assignment' do
- expect { secondary_organization.update(secondary_member_ids: [customer.id]) }.to change { customer.reload.updated_at }
- end
- it 'does clear cache of organization after user assignment' do
- secondary_organization.update(member_ids: [customer.id])
- expect(secondary_organization.reload.attributes_with_association_ids['member_ids']).to include(customer.id)
- end
- it 'does touch organization after user assignment' do
- expect { secondary_organization.update(member_ids: [customer.id]) }.to change { secondary_organization.reload.updated_at }
- end
- end
- end
- describe '#all_members' do
- let!(:primary_user) { create(:user, organization:, organizations: create_list(:organization, 3)) }
- let!(:secondary_user) { create(:user, organization: create(:organization), organizations: [organization]) }
- it 'lists all assigned members' do
- expect(organization.all_members).to contain_exactly(primary_user, secondary_user)
- end
- end
- end
|