debian-mirror-gitlab/spec/models/broadcast_message_spec.rb

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

408 lines
13 KiB
Ruby
Raw Normal View History

2019-07-07 11:18:12 +05:30
# frozen_string_literal: true
2014-09-02 18:07:02 +05:30
require 'spec_helper'
2020-07-28 23:09:34 +05:30
RSpec.describe BroadcastMessage do
2017-08-17 22:00:37 +05:30
subject { build(:broadcast_message) }
2014-09-02 18:07:02 +05:30
2015-04-26 12:48:37 +05:30
it { is_expected.to be_valid }
2014-09-02 18:07:02 +05:30
2015-12-23 02:04:40 +05:30
describe 'validations' do
let(:triplet) { '#000' }
let(:hex) { '#AABBCC' }
it { is_expected.to allow_value(nil).for(:color) }
it { is_expected.to allow_value(triplet).for(:color) }
it { is_expected.to allow_value(hex).for(:color) }
it { is_expected.not_to allow_value('000').for(:color) }
it { is_expected.to allow_value(nil).for(:font) }
it { is_expected.to allow_value(triplet).for(:font) }
it { is_expected.to allow_value(hex).for(:font) }
it { is_expected.not_to allow_value('000').for(:font) }
2020-01-01 13:55:28 +05:30
it { is_expected.to allow_value(1).for(:broadcast_type) }
it { is_expected.not_to allow_value(nil).for(:broadcast_type) }
2022-05-07 20:08:51 +05:30
it { is_expected.not_to allow_value(nil).for(:target_access_levels) }
2023-07-09 08:55:56 +05:30
it do
is_expected.to validate_inclusion_of(:target_access_levels)
.in_array(described_class::ALLOWED_TARGET_ACCESS_LEVELS)
end
2015-12-23 02:04:40 +05:30
end
2023-01-13 00:05:48 +05:30
describe 'default values' do
subject(:message) { described_class.new }
it { expect(message.color).to eq('#E75E40') }
it { expect(message.font).to eq('#FFFFFF') }
end
2020-01-01 13:55:28 +05:30
shared_examples 'time constrainted' do |broadcast_type|
2017-09-10 17:25:29 +05:30
it 'returns message if time match' do
2020-01-01 13:55:28 +05:30
message = create(:broadcast_message, broadcast_type: broadcast_type)
2020-01-01 13:55:28 +05:30
expect(subject.call).to include(message)
2014-09-02 18:07:02 +05:30
end
2017-09-10 17:25:29 +05:30
it 'returns multiple messages if time match' do
2020-01-01 13:55:28 +05:30
message1 = create(:broadcast_message, broadcast_type: broadcast_type)
message2 = create(:broadcast_message, broadcast_type: broadcast_type)
2017-09-10 17:25:29 +05:30
2020-01-01 13:55:28 +05:30
expect(subject.call).to contain_exactly(message1, message2)
2017-09-10 17:25:29 +05:30
end
it 'returns empty list if time not come' do
2020-01-01 13:55:28 +05:30
create(:broadcast_message, :future, broadcast_type: broadcast_type)
2020-01-01 13:55:28 +05:30
expect(subject.call).to be_empty
2014-09-02 18:07:02 +05:30
end
2017-09-10 17:25:29 +05:30
it 'returns empty list if time has passed' do
2020-01-01 13:55:28 +05:30
create(:broadcast_message, :expired, broadcast_type: broadcast_type)
2020-01-01 13:55:28 +05:30
expect(subject.call).to be_empty
2017-09-10 17:25:29 +05:30
end
2020-01-01 13:55:28 +05:30
end
2017-09-10 17:25:29 +05:30
2020-01-01 13:55:28 +05:30
shared_examples 'message cache' do |broadcast_type|
2019-09-30 21:07:59 +05:30
it 'caches the output of the query for two weeks' do
2020-01-01 13:55:28 +05:30
create(:broadcast_message, broadcast_type: broadcast_type)
2017-09-10 17:25:29 +05:30
2019-09-30 21:07:59 +05:30
expect(described_class).to receive(:current_and_future_messages).and_call_original.twice
2017-09-10 17:25:29 +05:30
2020-01-01 13:55:28 +05:30
subject.call
2018-10-15 14:42:47 +05:30
2022-05-07 20:08:51 +05:30
travel_to(3.weeks.from_now) do
2020-01-01 13:55:28 +05:30
subject.call
2018-10-15 14:42:47 +05:30
end
2017-09-10 17:25:29 +05:30
end
2020-03-13 15:44:24 +05:30
it 'expires the value if a broadcast message has ended', :request_store do
2020-06-23 00:09:42 +05:30
message = create(:broadcast_message, broadcast_type: broadcast_type, ends_at: Time.current.utc + 1.day)
2020-03-13 15:44:24 +05:30
expect(subject.call).to match_array([message])
expect(described_class.cache).to receive(:expire).and_call_original
2022-05-07 20:08:51 +05:30
travel_to(1.week.from_now) do
2020-03-13 15:44:24 +05:30
2.times { expect(subject.call).to be_empty }
end
end
2019-02-15 15:39:39 +05:30
it 'does not create new records' do
2020-01-01 13:55:28 +05:30
create(:broadcast_message, broadcast_type: broadcast_type)
2019-02-15 15:39:39 +05:30
2020-01-01 13:55:28 +05:30
expect { subject.call }.not_to change { described_class.count }
2019-02-15 15:39:39 +05:30
end
2017-09-10 17:25:29 +05:30
it 'includes messages that need to be displayed in the future' do
2020-01-01 13:55:28 +05:30
create(:broadcast_message, broadcast_type: broadcast_type)
2017-09-10 17:25:29 +05:30
future = create(
:broadcast_message,
2020-06-23 00:09:42 +05:30
starts_at: Time.current + 10.minutes,
ends_at: Time.current + 20.minutes,
2020-01-01 13:55:28 +05:30
broadcast_type: broadcast_type
2017-09-10 17:25:29 +05:30
)
2020-01-01 13:55:28 +05:30
expect(subject.call.length).to eq(1)
2017-09-10 17:25:29 +05:30
2022-05-07 20:08:51 +05:30
travel_to(future.starts_at + 1.second) do
2020-01-01 13:55:28 +05:30
expect(subject.call.length).to eq(2)
2017-09-10 17:25:29 +05:30
end
end
it 'does not clear the cache if only a future message should be displayed' do
create(:broadcast_message, :future)
2019-02-15 15:39:39 +05:30
expect(Rails.cache).not_to receive(:delete).with(described_class::CACHE_KEY)
2020-01-01 13:55:28 +05:30
expect(subject.call.length).to eq(0)
end
end
shared_examples "matches with current path" do |broadcast_type|
it 'returns message if it matches the target path' do
message = create(:broadcast_message, target_path: "*/onboarding_completed", broadcast_type: broadcast_type)
expect(subject.call('/users/onboarding_completed')).to include(message)
end
it 'returns message if part of the target path matches' do
create(:broadcast_message, target_path: "/users/*/issues", broadcast_type: broadcast_type)
expect(subject.call('/users/name/issues').length).to eq(1)
end
2021-06-08 01:23:25 +05:30
it 'returns message if provided a path without a preceding slash' do
create(:broadcast_message, target_path: "/users/*/issues", broadcast_type: broadcast_type)
expect(subject.call('users/name/issues').length).to eq(1)
end
2020-01-01 13:55:28 +05:30
it 'returns the message for empty target path' do
create(:broadcast_message, target_path: "", broadcast_type: broadcast_type)
expect(subject.call('/users/name/issues').length).to eq(1)
end
it 'returns the message if target path is nil' do
create(:broadcast_message, target_path: nil, broadcast_type: broadcast_type)
expect(subject.call('/users/name/issues').length).to eq(1)
end
it 'does not return message if target path does not match' do
create(:broadcast_message, target_path: "/onboarding_completed", broadcast_type: broadcast_type)
expect(subject.call('/welcome').length).to eq(0)
end
it 'does not return message if target path does not match when using wildcard' do
create(:broadcast_message, target_path: "/users/*/issues", broadcast_type: broadcast_type)
expect(subject.call('/group/groupname/issues').length).to eq(0)
end
2020-05-24 23:13:21 +05:30
it 'does not return message if target path has no wild card at the end' do
create(:broadcast_message, target_path: "*/issues", broadcast_type: broadcast_type)
expect(subject.call('/group/issues/test').length).to eq(0)
end
it 'does not return message if target path has wild card at the end' do
create(:broadcast_message, target_path: "/issues/*", broadcast_type: broadcast_type)
expect(subject.call('/group/issues/test').length).to eq(0)
end
it 'does return message if target path has wild card at the beginning and the end' do
create(:broadcast_message, target_path: "*/issues/*", broadcast_type: broadcast_type)
expect(subject.call('/group/issues/test').length).to eq(1)
end
2021-01-29 00:20:46 +05:30
it "does not return message if the target path is set but no current path is provided" do
create(:broadcast_message, target_path: "*/issues/*", broadcast_type: broadcast_type)
expect(subject.call.length).to eq(0)
end
2020-01-01 13:55:28 +05:30
end
2022-05-07 20:08:51 +05:30
shared_examples "matches with user access level" do |broadcast_type|
let_it_be(:target_access_levels) { [Gitlab::Access::GUEST] }
context 'when target_access_levels is empty' do
let_it_be(:message) { create(:broadcast_message, target_access_levels: [], broadcast_type: broadcast_type) }
it 'returns the message if user access level is not nil' do
expect(subject.call(nil, Gitlab::Access::MINIMAL_ACCESS)).to include(message)
end
it 'returns the message if user access level is nil' do
expect(subject.call).to include(message)
end
end
context 'when target_access_levels is not empty' do
2023-07-09 08:55:56 +05:30
let_it_be(:message) do
create(:broadcast_message, target_access_levels: target_access_levels, broadcast_type: broadcast_type)
end
2022-05-07 20:08:51 +05:30
it "does not return the message if user access level is nil" do
expect(subject.call).to be_empty
end
it "returns the message if user access level is in target_access_levels" do
expect(subject.call(nil, Gitlab::Access::GUEST)).to include(message)
end
it "does not return the message if user access level is not in target_access_levels" do
expect(subject.call(nil, Gitlab::Access::MINIMAL_ACCESS)).to be_empty
end
end
end
shared_examples "handles stale cache data gracefully" do
# Regression test for https://gitlab.com/gitlab-org/gitlab/-/issues/353076
context 'when cache returns stale data (e.g. nil target_access_levels)' do
let(:message) { build(:broadcast_message, :banner, target_access_levels: nil) }
let(:cache) { Gitlab::JsonCache.new }
before do
cache.write(described_class::BANNER_CACHE_KEY, [message])
2023-07-09 08:55:56 +05:30
allow(described_class).to receive(:cache) { cache }
2022-05-07 20:08:51 +05:30
end
it 'does not raise error (e.g. NoMethodError from nil.empty?)' do
expect { subject.call }.not_to raise_error
end
end
end
2020-01-01 13:55:28 +05:30
describe '.current', :use_clean_rails_memory_store_caching do
2022-05-07 20:08:51 +05:30
subject do
2023-07-09 08:55:56 +05:30
->(path = nil, user_access_level = nil) do
2022-05-07 20:08:51 +05:30
described_class.current(current_path: path, user_access_level: user_access_level)
end
end
2020-01-01 13:55:28 +05:30
it_behaves_like 'time constrainted', :banner
it_behaves_like 'message cache', :banner
it_behaves_like 'matches with current path', :banner
2022-05-07 20:08:51 +05:30
it_behaves_like 'matches with user access level', :banner
it_behaves_like 'handles stale cache data gracefully'
context 'when message is from cache' do
before do
subject.call
end
it_behaves_like 'matches with current path', :banner
it_behaves_like 'matches with user access level', :banner
it_behaves_like 'matches with current path', :notification
it_behaves_like 'matches with user access level', :notification
end
2020-01-01 13:55:28 +05:30
it 'returns both types' do
banner_message = create(:broadcast_message, broadcast_type: :banner)
notification_message = create(:broadcast_message, broadcast_type: :notification)
expect(subject.call).to contain_exactly(banner_message, notification_message)
end
end
describe '.current_banner_messages', :use_clean_rails_memory_store_caching do
2022-05-07 20:08:51 +05:30
subject do
2023-07-09 08:55:56 +05:30
->(path = nil, user_access_level = nil) do
2022-05-07 20:08:51 +05:30
described_class.current_banner_messages(current_path: path, user_access_level: user_access_level)
end
end
2020-01-01 13:55:28 +05:30
it_behaves_like 'time constrainted', :banner
it_behaves_like 'message cache', :banner
it_behaves_like 'matches with current path', :banner
2022-05-07 20:08:51 +05:30
it_behaves_like 'matches with user access level', :banner
it_behaves_like 'handles stale cache data gracefully'
context 'when message is from cache' do
before do
subject.call
end
it_behaves_like 'matches with current path', :banner
it_behaves_like 'matches with user access level', :banner
end
2020-01-01 13:55:28 +05:30
it 'only returns banners' do
banner_message = create(:broadcast_message, broadcast_type: :banner)
create(:broadcast_message, broadcast_type: :notification)
expect(subject.call).to contain_exactly(banner_message)
end
end
describe '.current_notification_messages', :use_clean_rails_memory_store_caching do
2022-05-07 20:08:51 +05:30
subject do
2023-07-09 08:55:56 +05:30
->(path = nil, user_access_level = nil) do
2022-05-07 20:08:51 +05:30
described_class.current_notification_messages(current_path: path, user_access_level: user_access_level)
end
end
2020-01-01 13:55:28 +05:30
it_behaves_like 'time constrainted', :notification
it_behaves_like 'message cache', :notification
it_behaves_like 'matches with current path', :notification
2022-05-07 20:08:51 +05:30
it_behaves_like 'matches with user access level', :notification
it_behaves_like 'handles stale cache data gracefully'
context 'when message is from cache' do
before do
subject.call
end
it_behaves_like 'matches with current path', :notification
it_behaves_like 'matches with user access level', :notification
end
2020-01-01 13:55:28 +05:30
it 'only returns notifications' do
notification_message = create(:broadcast_message, broadcast_type: :notification)
create(:broadcast_message, broadcast_type: :banner)
expect(subject.call).to contain_exactly(notification_message)
2019-02-15 15:39:39 +05:30
end
2014-09-02 18:07:02 +05:30
end
2019-07-07 11:18:12 +05:30
describe '#attributes' do
it 'includes message_html field' do
expect(subject.attributes.keys).to include("cached_markdown_version", "message_html")
end
end
describe '#active?' do
it 'is truthy when started and not ended' do
message = build(:broadcast_message)
expect(message).to be_active
end
it 'is falsey when ended' do
message = build(:broadcast_message, :expired)
expect(message).not_to be_active
end
it 'is falsey when not started' do
message = build(:broadcast_message, :future)
expect(message).not_to be_active
end
end
describe '#started?' do
it 'is truthy when starts_at has passed' do
message = build(:broadcast_message)
travel_to(3.days.from_now) do
expect(message).to be_started
end
end
it 'is falsey when starts_at is in the future' do
message = build(:broadcast_message)
travel_to(3.days.ago) do
expect(message).not_to be_started
end
end
end
describe '#ended?' do
it 'is truthy when ends_at has passed' do
message = build(:broadcast_message)
travel_to(3.days.from_now) do
expect(message).to be_ended
end
end
it 'is falsey when ends_at is in the future' do
message = build(:broadcast_message)
travel_to(3.days.ago) do
expect(message).not_to be_ended
end
end
end
2017-09-10 17:25:29 +05:30
describe '#flush_redis_cache' do
it 'flushes the Redis cache' do
message = create(:broadcast_message)
2022-05-07 20:08:51 +05:30
expect(Rails.cache).to receive(:delete).with("#{described_class::CACHE_KEY}:#{Gitlab.revision}")
expect(Rails.cache).to receive(:delete).with("#{described_class::BANNER_CACHE_KEY}:#{Gitlab.revision}")
expect(Rails.cache).to receive(:delete).with("#{described_class::NOTIFICATION_CACHE_KEY}:#{Gitlab.revision}")
2017-09-10 17:25:29 +05:30
message.flush_redis_cache
end
end
2014-09-02 18:07:02 +05:30
end