123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 |
- # Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- require 'rails_helper'
- RSpec.describe Setting, type: :model do
- subject(:setting) { create(:setting) }
- describe '.get' do
- context 'when given a valid Setting#name' do
- it 'returns #state_current[:value]' do
- expect { setting.update(state_current: { value: 'foo' }) }
- .to change { described_class.get(setting.name) }.to('foo')
- end
- end
- context 'when interpolated value was set and cache is still valid' do
- it 'stores interpolated value' do
- create(:setting, name: 'broadcast_test', state: 'test')
- described_class.send(:load) # prewarm cache
- described_class.set('broadcast_test', 'test #{config.fqdn}') # rubocop:disable Lint/InterpolationCheck
- expect(described_class.get('broadcast_test'))
- .to eq("test #{described_class.get('fqdn')}")
- end
- end
- end
- describe '.set' do
- context 'when given a valid Setting#name' do
- it 'sets #state_current = { value: <arg> }' do
- expect { described_class.set(setting.name, 'foo') }
- .to change { setting.reload.state_current }.to({ 'value' => 'foo' })
- end
- it 'logs the value' do
- allow(described_class.logger).to receive(:info)
- described_class.set(setting.name, 'foo')
- expect(described_class.logger).to have_received(:info).with("Setting.set('#{setting.name}', \"foo\")")
- end
- end
- context 'when given a sensitive Setting#name' do
- subject(:setting) { create(:setting, name: 'my_token') }
- it 'masks the value' do
- allow(described_class.logger).to receive(:info)
- described_class.set(setting.name, 'foo')
- expect(described_class.logger).to have_received(:info).with("Setting.set('#{setting.name}', \"[FILTERED]\")")
- end
- end
- context 'when #preferences hash includes a :cache key' do
- subject(:setting) { create(:setting, preferences: { cache: ['foo'] }) }
- before { Rails.cache.write('foo', 'bar') }
- it 'resets the cache key' do
- expect { described_class.set(setting.name, 'baz') }
- .to change { Rails.cache.read('foo') }.to(nil)
- end
- end
- end
- describe '.reset' do
- context 'when given a valid Setting#name' do
- it 'sets #state_current = { value: <orig> } (via #state_initial[:value])' do
- setting.update(state_initial: { value: 'foo' })
- described_class.set(setting.name, 'bar')
- expect { described_class.reset(setting.name) }
- .to change { setting.reload.state_current }.to({ value: 'foo' })
- end
- it 'logs the value' do
- setting.update(state_initial: { value: 'foo' })
- allow(described_class.logger).to receive(:info)
- described_class.reset(setting.name)
- expect(described_class.logger).to have_received(:info).with("Setting.reset('#{setting.name}', {\"value\"=>\"foo\"})")
- end
- end
- context 'when given a sensitive Setting#name' do
- subject(:setting) { create(:setting, name: 'my_token') }
- it 'masks the value' do
- setting.update(state_initial: { value: 'foo' })
- allow(described_class.logger).to receive(:info)
- described_class.reset(setting.name)
- expect(described_class.logger).to have_received(:info).with("Setting.reset('#{setting.name}', \"[FILTERED]\")")
- end
- end
- end
- describe '.cache_valid?' do
- context 'when loading first time' do
- before do
- # ensure no cache checks are set
- described_class.class_variable_set(:@@lookup_at, nil) # rubocop:disable Style/ClassVars
- described_class.class_variable_set(:@@query_cache_key, nil) # rubocop:disable Style/ClassVars
- end
- it 'cache is not valid' do
- expect(described_class).not_to be_cache_valid
- end
- end
- context 'when cache is valid' do
- before do
- # ensure cache is warm
- described_class.send(:load)
- # ensure cache is not touched by broadcasting the new value
- allow_any_instance_of(described_class).to receive(:broadcast_frontend)
- end
- it 'cache is valid' do
- expect(described_class).to be_cache_valid
- end
- it 'cache is still valid after some time' do
- travel 1.minute
- expect(described_class).to be_cache_valid
- end
- context 'when Setting is updated in the same process' do
- before { described_class.set('maintenance_login', 'sample message') }
- it 'cache is not valid' do
- expect(described_class).not_to be_cache_valid
- end
- end
- context 'when Setting updated outside of the process and class variables were not touched' do
- before { described_class.find_by(name: 'maintenance_login').touch }
- it 'cache is seen as valid' do
- expect(described_class).to be_cache_valid
- end
- it 'cache is seen as invalid after some time' do
- travel 1.minute
- expect(described_class).not_to be_cache_valid
- end
- end
- end
- end
- describe 'attributes' do
- describe '#state_initial' do
- subject(:setting) { build(:setting, state: 'foo') }
- it 'is set on creation, based on #state' do
- expect { setting.save }
- .to change(setting, :state_initial).from({}).to({ value: 'foo' })
- end
- end
- end
- describe 'broadcast_frontend' do
- subject(:setting) do
- build(:setting, name: 'broadcast_test', state: value, frontend: frontend)
- .tap { |setting| setting.preferences = { authentication: true } if authentication_required }
- end
- let(:value) { 'foo' }
- let(:frontend) { true }
- let(:authentication_required) { false }
- context 'when setting is non-frontend' do
- let(:frontend) { false }
- it 'does not broadcast' do
- allow(Sessions).to receive(:broadcast)
- setting.save
- expect(Sessions).not_to have_received(:broadcast)
- end
- it 'does not trigger subscription' do
- allow(Gql::Subscriptions::ConfigUpdates).to receive(:trigger)
- setting.save
- expect(Gql::Subscriptions::ConfigUpdates).not_to have_received(:trigger).with(setting)
- end
- end
- context 'when setting is public' do
- it 'broadcasts to public' do
- allow(Sessions).to receive(:broadcast)
- setting.save
- expect(Sessions).to have_received(:broadcast)
- .with({ data: { name: 'broadcast_test', value: 'foo' }, event: 'config_update' }, 'public')
- end
- it 'triggers subscription' do
- allow(Gql::Subscriptions::ConfigUpdates).to receive(:trigger)
- setting.save
- expect(Gql::Subscriptions::ConfigUpdates).to have_received(:trigger).with(setting)
- end
- end
- context 'when setting requires authentication' do
- let(:authentication_required) { true }
- it 'broadcasts to authenticated only' do
- allow(Sessions).to receive(:broadcast)
- setting.save
- expect(Sessions).to have_received(:broadcast)
- .with({ data: { name: 'broadcast_test', value: 'foo' }, event: 'config_update' }, 'authenticated')
- end
- it 'triggers subscription' do
- allow(Gql::Subscriptions::ConfigUpdates).to receive(:trigger)
- setting.save
- expect(Gql::Subscriptions::ConfigUpdates).to have_received(:trigger).with(setting)
- end
- end
- context 'when setting uses interpolation' do
- let(:value) { 'test #{config.fqdn}' } # rubocop:disable Lint/InterpolationCheck
- it 'broadcasts to authenticated only' do
- allow(Sessions).to receive(:broadcast)
- setting.save
- expect(Sessions)
- .to have_received(:broadcast)
- .with(
- { data: { name: 'broadcast_test', value: "test #{described_class.get('fqdn')}" }, event: 'config_update' },
- 'public'
- )
- end
- it 'triggers subscription' do
- allow(Gql::Subscriptions::ConfigUpdates).to receive(:trigger)
- setting.save
- expect(Gql::Subscriptions::ConfigUpdates).to have_received(:trigger).with(setting)
- end
- end
- end
- end
|