# Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ RSpec.shared_examples 'ApplicationModel::CanCreatesAndUpdates' do |unique_name: true| describe '.create_if_not_exists' do let!(:record) { create(described_class.name.underscore) } context 'when given a valid #id' do let(:id) { record.id } it 'returns that record' do expect(described_class.create_if_not_exists(id: id)).to eq(record) end it 'does not create a new record' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(id: id) expect(described_class).not_to have_received(:create!).with(id: id) end end context 'when given an invalid #id' do let(:id) { described_class.pluck(:id).max + 1 } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(id: id) expect(described_class).to have_received(:create!).with(id: id) end end shared_examples 'for #name attribute' do context 'when given a valid #name' do let(:name) { record.name } it 'returns that record' do expect(described_class.create_if_not_exists(name: name)).to eq(record) end it 'does not create a new record for matching identifier' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(name: name) expect(described_class).not_to have_received(:create!).with(name: name) end if unique_name it 'raises uniqueness error if identifier is different in case only' do swapcase_attibutes = attributes_for(described_class.name.underscore) .merge(name: record.name.swapcase) expect { described_class.create_if_not_exists(**swapcase_attibutes) } .to raise_error(ActiveRecord::RecordInvalid, %r{has already been taken}) end end end context 'when given an invalid #name' do let(:name) { "#{described_class.pluck(:name).max}foo" } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(name: name) expect(described_class).to have_received(:create!).with(name: name) end end end shared_examples 'for #login attribute' do context 'when given a valid #login' do let(:login) { record.login } it 'returns that record' do expect(described_class.create_if_not_exists(login: login)).to eq(record) end it 'does not create a new record' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(login: login) expect(described_class).not_to have_received(:create!).with(login: login) end end context 'when given an invalid #login' do let(:login) { "#{described_class.pluck(:login).max}foo" } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(login: login) expect(described_class).to have_received(:create!).with(login: login) end end end shared_examples 'for #email attribute' do context 'when given a valid #email' do let(:email) { record.email } it 'returns that record' do expect(described_class.create_if_not_exists(email: email)).to eq(record) end it 'does not create a new record' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(email: email) expect(described_class).not_to have_received(:create!).with(email: email) end end context 'when given an invalid #email' do let(:email) { "#{described_class.pluck(:email).max}foo" } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(email: email) expect(described_class).to have_received(:create!).with(email: email) end end end shared_examples 'for #source and #locale attributes' do context 'when given a valid #source and #locale' do let(:source) { record.source } let(:locale) { record.locale } it 'returns that record' do expect(described_class.create_if_not_exists(source: source, locale: locale)).to eq(record) end it 'does not create a new record' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(source: source, locale: locale) expect(described_class).not_to have_received(:create!).with(source: source, locale: locale) end end context 'when given an invalid #source or #locale' do let(:source) { "#{described_class.pluck(:source).max}foo" } let(:locale) { record.locale } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_if_not_exists(source: source, locale: locale) expect(described_class).to have_received(:create!).with(source: source, locale: locale) end end end include_examples 'for #name attribute' if described_class.attribute_names.include?('name') include_examples 'for #login attribute' if described_class.attribute_names.include?('login') include_examples 'for #email attribute' if described_class.attribute_names.include?('email') include_examples 'for #source and #locale attributes' if (described_class.attribute_names & %w[source locale]).many? end describe '.create_or_update' do let!(:record) { create(described_class.name.underscore) } let(:yesterday) { 1.day.ago } context 'when given a valid #id' do let(:id) { record.id } it 'updates other attributes on (and returns) that record' do expect { described_class.create_or_update(id: id, updated_at: yesterday) } .to change { record.reload.updated_at.to_i }.to(yesterday.to_i) end end context 'when given an invalid #id' do let(:id) { described_class.pluck(:id).max + 1 } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_or_update(id: id) expect(described_class).to have_received(:create!).with(id: id) end end shared_examples 'for #name attribute' do context 'when given a valid #name' do let(:name) { record.name } it 'updates other attributes on (and returns) that record' do expect { described_class.create_or_update(name: name, updated_at: yesterday) } .to change { record.reload.updated_at.to_i }.to(yesterday.to_i) end if unique_name it 'raises uniqueness error if identifier is different in case only' do swapcase_attibutes = attributes_for(described_class.name.underscore) .merge(name: record.name.swapcase) expect { described_class.create_or_update(**swapcase_attibutes) } .to raise_error(ActiveRecord::RecordInvalid, %r{has already been taken}) end end end context 'when given an invalid #name' do let(:name) { "#{described_class.pluck(:name).max}foo" } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_or_update(name: name) expect(described_class).to have_received(:create!).with(name: name) end end end shared_examples 'for #login attribute' do context 'when given a valid #login' do let(:login) { record.login } it 'updates other attributes on (and returns) that record' do expect { described_class.create_or_update(login: login, updated_at: yesterday) } .to change { record.reload.updated_at.to_i }.to(yesterday.to_i) end end context 'when given an invalid #login' do let(:login) { "#{described_class.pluck(:login).max}foo" } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_or_update(login: login) expect(described_class).to have_received(:create!).with(login: login) end end end shared_examples 'for #email attribute' do context 'when given a valid #email' do let(:email) { record.email } it 'updates other attributes on (and returns) that record' do expect { described_class.create_or_update(email: email, updated_at: yesterday) } .to change { record.reload.updated_at.to_i }.to(yesterday.to_i) end end context 'when given an invalid #email' do let(:email) { "#{described_class.pluck(:email).max}foo" } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_or_update(email: email) expect(described_class).to have_received(:create!).with(email: email) end end end shared_examples 'for #locale attribute' do context 'when given a valid #locale' do let(:locale) { record.locale } it 'updates other attributes on (and returns) that record' do expect { described_class.create_or_update(locale: locale, updated_at: yesterday) } .to change { record.reload.updated_at.to_i }.to(yesterday.to_i) end end context 'when given an invalid #locale' do let(:locale) { record.locale } it 'attempts to create a new record' do allow(described_class).to receive(:create!) described_class.create_or_update(locale: locale) expect(described_class).to have_received(:create!).with(locale: locale) end end end include_examples 'for #name attribute' if described_class.attribute_names.include?('name') include_examples 'for #login attribute' if described_class.attribute_names.include?('login') include_examples 'for #email attribute' if described_class.attribute_names.include?('email') include_examples 'for #locale attribute' if described_class.attribute_names.include?('locale') end describe '.create_or_update_with_ref' do before do allow(described_class) .to receive(:association_name_to_id_convert).and_return(converted_params) allow(described_class) .to receive(:create_or_update) end let(:given_params) { { given: 'attr' } } let(:converted_params) { { converted: 'attr' } } it 'calls create_or_update with given data', aggregate_failures: true do described_class.create_or_update_with_ref(given_params) expect(described_class).to have_received(:association_name_to_id_convert).with(given_params) expect(described_class).to have_received(:create_or_update).with(converted_params) end end end