123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- require 'models/application_model_examples'
- require 'models/concerns/can_be_imported_examples'
- require 'models/concerns/has_object_manager_attributes_examples'
- require 'models/concerns/has_collection_update_examples'
- require 'models/concerns/has_xss_sanitized_note_examples'
- require 'models/concerns/has_image_sanitized_note_examples'
- RSpec.describe Group, type: :model do
- subject(:group) { create(:group) }
- it_behaves_like 'ApplicationModel'
- it_behaves_like 'CanBeImported'
- it_behaves_like 'HasObjectManagerAttributes'
- it_behaves_like 'HasCollectionUpdate', collection_factory: :group
- it_behaves_like 'HasXssSanitizedNote', model_factory: :group
- it_behaves_like 'HasImageSanitizedNote', model_factory: :group
- it_behaves_like 'Association clears cache', association: :users
- it_behaves_like 'Association clears cache', association: :roles
- describe 'name compatibility layer' do
- context 'when creating a new group' do
- context 'with name attribute' do
- let(:name) { Faker::Lorem.unique.word.capitalize }
- it 'sets name_last attribute to name' do
- expect(described_class.create(name: name)).to have_attributes(name_last: name)
- end
- context 'when using complete path' do
- let(:group1) { create(:group) }
- let(:group2) { create(:group, parent: group1) }
- let(:name) { "#{group1.name_last}::#{group2.name_last}::#{Faker::Lorem.unique.word.capitalize}" }
- it 'sets parent_id attribute to guessed parent' do
- expect(described_class.create(name: name)).to have_attributes(parent_id: group2.id)
- end
- context 'when path is invalid' do
- let(:name) { Array.new(3) { Faker::Lorem.unique.word.capitalize }.join('::') }
- it 'raises validation error' do
- expect { described_class.create(name: name) }.to raise_error(ActiveRecord::RecordInvalid, 'Validation failed: Name contains invalid path')
- end
- end
- end
- end
- context 'with name_last attribute' do
- let(:name_last) { Faker::Lorem.unique.word.capitalize }
- it 'sets name_last attribute to name_last' do
- expect(described_class.create(name_last: name_last)).to have_attributes(name_last: name_last)
- end
- end
- context 'with both name and name_last attribute' do
- let(:name) { Faker::Lorem.unique.word.capitalize }
- let(:name_last) { Faker::Lorem.unique.word.capitalize }
- it 'sets name_last attribute to name_last' do
- expect(described_class.create(name: name, name_last: name_last)).to have_attributes(name_last: name_last)
- end
- end
- end
- context 'when updating an existing group' do
- let(:group) { create(:group) }
- context 'with name attribute' do
- let(:name) { Faker::Lorem.unique.word.capitalize }
- before do
- group.update!(name: name) if !defined?(skip_before)
- end
- it 'sets name_last attribute to name' do
- expect(group).to have_attributes(name_last: name)
- end
- context 'when using complete path' do
- let(:group1) { create(:group) }
- let(:group2) { create(:group, parent: group1) }
- let(:name) { "#{group1.name_last}::#{group2.name_last}::#{Faker::Lorem.unique.word.capitalize}" }
- it 'sets parent_id attribute to guessed parent' do
- expect(group).to have_attributes(parent_id: group2.id)
- end
- context 'when path is invalid' do
- let(:name) { Array.new(3) { Faker::Lorem.unique.word.capitalize }.join('::') }
- let(:skip_before) { true }
- it 'raises validation error' do
- expect { group.update!(name: name) }.to raise_error(ActiveRecord::RecordInvalid, 'Validation failed: Name contains invalid path')
- end
- end
- end
- end
- context 'with name_last attribute' do
- let(:name_last) { Faker::Lorem.unique.word.capitalize }
- before do
- group.update!(name_last: name_last)
- end
- it 'sets name_last attribute to name_last' do
- expect(described_class.create(name_last: name_last)).to have_attributes(name_last: name_last)
- end
- end
- context 'with both name and name_last attribute' do
- let(:name) { Faker::Lorem.unique.word.capitalize }
- let(:name_last) { Faker::Lorem.unique.word.capitalize }
- before do
- group.update!(name: name, name_last: name_last)
- end
- it 'sets name_last attribute to name_last' do
- expect(described_class.create(name_last: name_last)).to have_attributes(name_last: name_last)
- end
- end
- end
- end
- describe 'tree related functions' do
- let!(:group_1) { create(:group) }
- let!(:group_2) { create(:group, parent: group_1) }
- let!(:group_31) { create(:group, parent: group_2) }
- let!(:group_32) { create(:group, parent: group_2) }
- let!(:group_4) { create(:group, parent: group_31) }
- describe '#all_children' do
- it 'does return all children' do
- expect(group_1.all_children.sort).to eq([group_2, group_31, group_32, group_4].sort)
- end
- end
- describe '#all_parents' do
- it 'does return all parents ids' do
- expect(group_4.all_parents).to eq([group_31, group_2, group_1])
- end
- end
- describe '#depth' do
- it 'does return group depth' do
- expect(group_4.depth).to eq(3)
- end
- end
- describe '#check_max_depth (mysql)', db_adapter: :mysql do
- let(:group_1_1) { create(:group, name: 'tree_group_1_1') }
- let(:group_1_2) { create(:group, name: 'tree_group_1_2', parent: group_1_1) }
- let(:group_1_3) { create(:group, name: 'tree_group_1_3', parent: group_1_2) }
- let(:group_1_4) { create(:group, name: 'tree_group_1_4', parent: group_1_3) }
- let(:group_1_5) { create(:group, name: 'tree_group_1_5', parent: group_1_4) }
- let(:group_1_6) { create(:group, name: 'tree_group_1_6', parent: group_1_5) }
- let(:group_1_7) { create(:group, name: 'tree_group_1_7', parent: group_1_6) }
- let(:group_2_1) { create(:group, name: 'tree_group_2_1') }
- let(:group_2_2) { create(:group, name: 'tree_group_2_2', parent: group_2_1) }
- let(:group_2_3) { create(:group, name: 'tree_group_2_3', parent: group_2_2) }
- let(:group_2_4) { create(:group, name: 'tree_group_2_4', parent: group_2_3) }
- it 'does check depth on creation', :aggregate_failures do
- expect do
- group_1_1
- group_1_2
- group_1_3
- group_1_4
- group_1_5
- group_1_6
- end.not_to raise_error
- expect { group_1_7 }.to raise_error(Exceptions::UnprocessableEntity, 'This group or its children exceed the allowed nesting depth.')
- end
- it 'does check depth on tree merge', :aggregate_failures do
- expect do
- group_1_6
- group_2_4
- end.not_to raise_error
- expect { group_2_1.update(parent: group_1_6) }.to raise_error(Exceptions::UnprocessableEntity, 'This group or its children exceed the allowed nesting depth.')
- end
- end
- describe '#check_max_depth (psql)', db_adapter: :postgresql do
- def groups_with_depth(depth)
- groups = []
- groups << create(:group)
- groups += create_list(:group, depth - 1)
- groups.each_with_index do |group, idx|
- next if idx.zero?
- group.update!(parent: groups[idx - 1])
- end
- groups
- end
- let(:groups_1) { groups_with_depth(10) }
- let(:groups_2) { groups_with_depth(4) }
- let(:group_1_11) { create(:group, parent: groups_1.last) }
- let(:group_2_5) { create(:group, parent: groups_2.last) }
- it 'does check depth on creation', :aggregate_failures do
- expect { groups_1 }.not_to raise_error
- expect { group_1_11 }.to raise_error(Exceptions::UnprocessableEntity, 'This group or its children exceed the allowed nesting depth.')
- expect { group_2_5 }.not_to raise_error
- end
- it 'does check depth on tree merge', :aggregate_failures do
- expect do
- groups_1.last
- groups_2.last
- end.not_to raise_error
- expect { groups_2.last.update!(parent: groups_1.last) }.to raise_error(Exceptions::UnprocessableEntity, 'This group or its children exceed the allowed nesting depth.')
- end
- end
- end
- end
|