Commit c7a8a89c authored by Eugie Limpin's avatar Eugie Limpin Committed by Adam Hegyi

Re-introduce role-targeted broadcast messages

This reverts commit ac8cc851.

Changelog: changed
parent efe67a3b
...@@ -65,6 +65,6 @@ class Admin::BroadcastMessagesController < Admin::ApplicationController ...@@ -65,6 +65,6 @@ class Admin::BroadcastMessagesController < Admin::ApplicationController
target_path target_path
broadcast_type broadcast_type
dismissable dismissable
)) ), target_access_levels: []).reverse_merge!(target_access_levels: [])
end end
end end
# frozen_string_literal: true # frozen_string_literal: true
module BroadcastMessagesHelper module BroadcastMessagesHelper
include Gitlab::Utils::StrongMemoize
def current_broadcast_banner_messages def current_broadcast_banner_messages
BroadcastMessage.current_banner_messages(request.path).select do |message| BroadcastMessage.current_banner_messages(
current_path: request.path,
user_access_level: current_user_access_level_for_project_or_group
).select do |message|
cookies["hide_broadcast_message_#{message.id}"].blank? cookies["hide_broadcast_message_#{message.id}"].blank?
end end
end end
def current_broadcast_notification_message def current_broadcast_notification_message
not_hidden_messages = BroadcastMessage.current_notification_messages(request.path).select do |message| not_hidden_messages = BroadcastMessage.current_notification_messages(
current_path: request.path,
user_access_level: current_user_access_level_for_project_or_group
).select do |message|
cookies["hide_broadcast_message_#{message.id}"].blank? cookies["hide_broadcast_message_#{message.id}"].blank?
end end
not_hidden_messages.last not_hidden_messages.last
...@@ -61,4 +69,35 @@ module BroadcastMessagesHelper ...@@ -61,4 +69,35 @@ module BroadcastMessagesHelper
def broadcast_type_options def broadcast_type_options
BroadcastMessage.broadcast_types.keys.map { |w| [w.humanize, w] } BroadcastMessage.broadcast_types.keys.map { |w| [w.humanize, w] }
end end
def target_access_level_options
BroadcastMessage::ALLOWED_TARGET_ACCESS_LEVELS.map do |access_level|
[Gitlab::Access.human_access(access_level), access_level]
end
end
def target_access_levels_display(access_levels)
access_levels.map do |access_level|
Gitlab::Access.human_access(access_level)
end.join(', ')
end
private
def current_user_access_level_for_project_or_group
return if Feature.disabled?(:role_targeted_broadcast_messages, default_enabled: :yaml)
return unless current_user.present?
strong_memoize(:current_user_access_level_for_project_or_group) do
if controller.is_a? Projects::ApplicationController
next unless @project
@project.team.max_member_access(current_user.id)
elsif controller.is_a? Groups::ApplicationController
next unless @group
@group.max_member_access_for_user(current_user)
end
end
end
end end
...@@ -4,12 +4,21 @@ class BroadcastMessage < ApplicationRecord ...@@ -4,12 +4,21 @@ class BroadcastMessage < ApplicationRecord
include CacheMarkdownField include CacheMarkdownField
include Sortable include Sortable
ALLOWED_TARGET_ACCESS_LEVELS = [
Gitlab::Access::GUEST,
Gitlab::Access::REPORTER,
Gitlab::Access::DEVELOPER,
Gitlab::Access::MAINTAINER,
Gitlab::Access::OWNER
].freeze
cache_markdown_field :message, pipeline: :broadcast_message, whitelisted: true cache_markdown_field :message, pipeline: :broadcast_message, whitelisted: true
validates :message, presence: true validates :message, presence: true
validates :starts_at, presence: true validates :starts_at, presence: true
validates :ends_at, presence: true validates :ends_at, presence: true
validates :broadcast_type, presence: true validates :broadcast_type, presence: true
validates :target_access_levels, inclusion: { in: ALLOWED_TARGET_ACCESS_LEVELS }
validates :color, allow_blank: true, color: true validates :color, allow_blank: true, color: true
validates :font, allow_blank: true, color: true validates :font, allow_blank: true, color: true
...@@ -29,20 +38,20 @@ class BroadcastMessage < ApplicationRecord ...@@ -29,20 +38,20 @@ class BroadcastMessage < ApplicationRecord
} }
class << self class << self
def current_banner_messages(current_path = nil) def current_banner_messages(current_path: nil, user_access_level: nil)
fetch_messages BANNER_CACHE_KEY, current_path do fetch_messages BANNER_CACHE_KEY, current_path, user_access_level do
current_and_future_messages.banner current_and_future_messages.banner
end end
end end
def current_notification_messages(current_path = nil) def current_notification_messages(current_path: nil, user_access_level: nil)
fetch_messages NOTIFICATION_CACHE_KEY, current_path do fetch_messages NOTIFICATION_CACHE_KEY, current_path, user_access_level do
current_and_future_messages.notification current_and_future_messages.notification
end end
end end
def current(current_path = nil) def current(current_path: nil, user_access_level: nil)
fetch_messages CACHE_KEY, current_path do fetch_messages CACHE_KEY, current_path, user_access_level do
current_and_future_messages current_and_future_messages
end end
end end
...@@ -63,7 +72,7 @@ class BroadcastMessage < ApplicationRecord ...@@ -63,7 +72,7 @@ class BroadcastMessage < ApplicationRecord
private private
def fetch_messages(cache_key, current_path) def fetch_messages(cache_key, current_path, user_access_level)
messages = cache.fetch(cache_key, as: BroadcastMessage, expires_in: cache_expires_in) do messages = cache.fetch(cache_key, as: BroadcastMessage, expires_in: cache_expires_in) do
yield yield
end end
...@@ -74,7 +83,13 @@ class BroadcastMessage < ApplicationRecord ...@@ -74,7 +83,13 @@ class BroadcastMessage < ApplicationRecord
# displaying we'll refresh the cache so we don't need to keep filtering. # displaying we'll refresh the cache so we don't need to keep filtering.
cache.expire(cache_key) if now_or_future != messages cache.expire(cache_key) if now_or_future != messages
now_or_future.select(&:now?).select { |message| message.matches_current_path(current_path) } messages = now_or_future.select(&:now?)
messages = messages.select do |message|
message.matches_current_user_access_level?(user_access_level)
end
messages.select do |message|
message.matches_current_path(current_path)
end
end end
end end
...@@ -102,6 +117,13 @@ class BroadcastMessage < ApplicationRecord ...@@ -102,6 +117,13 @@ class BroadcastMessage < ApplicationRecord
now? || future? now? || future?
end end
def matches_current_user_access_level?(user_access_level)
return false if target_access_levels.present? && Feature.disabled?(:role_targeted_broadcast_messages, default_enabled: :yaml)
return true unless target_access_levels.present?
target_access_levels.include? user_access_level
end
def matches_current_path(current_path) def matches_current_path(current_path)
return false if current_path.blank? && target_path.present? return false if current_path.blank? && target_path.present?
return true if current_path.blank? || target_path.blank? return true if current_path.blank? || target_path.blank?
......
...@@ -86,7 +86,7 @@ class PostReceiveService ...@@ -86,7 +86,7 @@ class PostReceiveService
banner = nil banner = nil
if project if project
scoped_messages = BroadcastMessage.current_banner_messages(project.full_path).select do |message| scoped_messages = BroadcastMessage.current_banner_messages(current_path: project.full_path).select do |message|
message.target_path.present? && message.matches_current_path(project.full_path) message.target_path.present? && message.matches_current_path(project.full_path)
end end
......
...@@ -55,6 +55,14 @@ ...@@ -55,6 +55,14 @@
= f.check_box :dismissable = f.check_box :dismissable
= f.label :dismissable do = f.label :dismissable do
= _('Allow users to dismiss the broadcast message') = _('Allow users to dismiss the broadcast message')
- if Feature.enabled?(:role_targeted_broadcast_messages, default_enabled: :yaml)
.form-group.row
.col-sm-2.col-form-label
= f.label :target_access_levels, _('Target roles')
.col-sm-10
= f.select :target_access_levels, target_access_level_options, { include_hidden: false }, multiple: true, class: 'form-control'
.form-text.text-muted
= _('The broadcast message displays only to users in projects and groups who have these roles.')
.form-group.row.js-toggle-colors-container.toggle-colors.hide .form-group.row.js-toggle-colors-container.toggle-colors.hide
.col-sm-2.col-form-label .col-sm-2.col-form-label
= f.label :font, _("Font Color") = f.label :font, _("Font Color")
......
- breadcrumb_title _("Messages") - breadcrumb_title _("Messages")
- page_title _("Broadcast Messages") - page_title _("Broadcast Messages")
- targeted_broadcast_messages_enabled = Feature.enabled?(:role_targeted_broadcast_messages, default_enabled: :yaml)
%h3.page-title %h3.page-title
= _('Broadcast Messages') = _('Broadcast Messages')
%p.light %p.light
= _('Broadcast messages are displayed for every user and can be used to notify users about scheduled maintenance, recent upgrades and more.') = _('Use banners and notifications to notify your users about scheduled maintenance, recent upgrades, and more.')
= render 'form' = render 'form'
...@@ -19,8 +20,10 @@ ...@@ -19,8 +20,10 @@
%th= _('Preview') %th= _('Preview')
%th= _('Starts') %th= _('Starts')
%th= _('Ends') %th= _('Ends')
%th= _(' Target Path') - if targeted_broadcast_messages_enabled
%th= _(' Type') %th= _('Target roles')
%th= _('Target Path')
%th= _('Type')
%th &nbsp; %th &nbsp;
%tbody %tbody
- @broadcast_messages.each do |message| - @broadcast_messages.each do |message|
...@@ -33,6 +36,9 @@ ...@@ -33,6 +36,9 @@
= message.starts_at = message.starts_at
%td %td
= message.ends_at = message.ends_at
- if targeted_broadcast_messages_enabled
%td
= target_access_levels_display(message.target_access_levels)
%td %td
= message.target_path = message.target_path
%td %td
......
---
name: role_targeted_broadcast_messages
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/81232
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/351736
milestone: '14.9'
type: development
group: group::activation
default_enabled: false
# frozen_string_literal: true
class AddTargetAccessLevelsToBroadcastMessages < Gitlab::Database::Migration[1.0]
def up
add_column :broadcast_messages, :target_access_levels, :integer, if_not_exists: true, array: true, null: false, default: []
end
def down
remove_column :broadcast_messages, :target_access_levels, if_exists: true
end
end
d7ddc369818f0a2403abefea2ac1da5abd1ca41199d3948166f10dfdf9d2fa9d
\ No newline at end of file
...@@ -11972,7 +11972,8 @@ CREATE TABLE broadcast_messages ( ...@@ -11972,7 +11972,8 @@ CREATE TABLE broadcast_messages (
cached_markdown_version integer, cached_markdown_version integer,
target_path character varying(255), target_path character varying(255),
broadcast_type smallint DEFAULT 1 NOT NULL, broadcast_type smallint DEFAULT 1 NOT NULL,
dismissable boolean dismissable boolean,
target_access_levels integer[] DEFAULT '{}'::integer[] NOT NULL
); );
CREATE SEQUENCE broadcast_messages_id_seq CREATE SEQUENCE broadcast_messages_id_seq
...@@ -28,15 +28,9 @@ msgstr "" ...@@ -28,15 +28,9 @@ msgstr ""
msgid " Please sign in." msgid " Please sign in."
msgstr "" msgstr ""
msgid " Target Path"
msgstr ""
msgid " Try to %{action} this file again." msgid " Try to %{action} this file again."
msgstr "" msgstr ""
msgid " Type"
msgstr ""
msgid " You need to do this before %{grace_period_deadline}." msgid " You need to do this before %{grace_period_deadline}."
msgstr "" msgstr ""
...@@ -6145,9 +6139,6 @@ msgstr "" ...@@ -6145,9 +6139,6 @@ msgstr ""
msgid "Broadcast Messages" msgid "Broadcast Messages"
msgstr "" msgstr ""
msgid "Broadcast messages are displayed for every user and can be used to notify users about scheduled maintenance, recent upgrades and more."
msgstr ""
msgid "Browse Directory" msgid "Browse Directory"
msgstr "" msgstr ""
...@@ -36044,6 +36035,9 @@ msgstr "" ...@@ -36044,6 +36035,9 @@ msgstr ""
msgid "Target branch or tag" msgid "Target branch or tag"
msgstr "" msgstr ""
msgid "Target roles"
msgstr ""
msgid "Target-Branch" msgid "Target-Branch"
msgstr "" msgstr ""
...@@ -36532,6 +36526,9 @@ msgstr "" ...@@ -36532,6 +36526,9 @@ msgstr ""
msgid "The branch or tag does not exist" msgid "The branch or tag does not exist"
msgstr "" msgstr ""
msgid "The broadcast message displays only to users in projects and groups who have these roles."
msgstr ""
msgid "The character highlighter helps you keep the subject line to %{titleLength} characters and wrap the body at %{bodyLength} so they are readable in git." msgid "The character highlighter helps you keep the subject line to %{titleLength} characters and wrap the body at %{bodyLength} so they are readable in git."
msgstr "" msgstr ""
...@@ -39796,6 +39793,9 @@ msgstr "" ...@@ -39796,6 +39793,9 @@ msgstr ""
msgid "Use authorized_keys file to authenticate SSH keys" msgid "Use authorized_keys file to authenticate SSH keys"
msgstr "" msgstr ""
msgid "Use banners and notifications to notify your users about scheduled maintenance, recent upgrades, and more."
msgstr ""
msgid "Use cURL" msgid "Use cURL"
msgstr "" msgstr ""
......
...@@ -7,7 +7,12 @@ RSpec.describe 'Admin Broadcast Messages' do ...@@ -7,7 +7,12 @@ RSpec.describe 'Admin Broadcast Messages' do
admin = create(:admin) admin = create(:admin)
sign_in(admin) sign_in(admin)
gitlab_enable_admin_mode_sign_in(admin) gitlab_enable_admin_mode_sign_in(admin)
create(:broadcast_message, :expired, message: 'Migration to new server') create(
:broadcast_message,
:expired,
message: 'Migration to new server',
target_access_levels: [Gitlab::Access::DEVELOPER]
)
visit admin_broadcast_messages_path visit admin_broadcast_messages_path
end end
...@@ -21,10 +26,13 @@ RSpec.describe 'Admin Broadcast Messages' do ...@@ -21,10 +26,13 @@ RSpec.describe 'Admin Broadcast Messages' do
fill_in 'broadcast_message_target_path', with: '*/user_onboarded' fill_in 'broadcast_message_target_path', with: '*/user_onboarded'
fill_in 'broadcast_message_font', with: '#b94a48' fill_in 'broadcast_message_font', with: '#b94a48'
select Date.today.next_year.year, from: 'broadcast_message_ends_at_1i' select Date.today.next_year.year, from: 'broadcast_message_ends_at_1i'
select 'Guest', from: 'broadcast_message_target_access_levels'
select 'Owner', from: 'broadcast_message_target_access_levels'
click_button 'Add broadcast message' click_button 'Add broadcast message'
expect(page).to have_current_path admin_broadcast_messages_path, ignore_query: true expect(page).to have_current_path admin_broadcast_messages_path, ignore_query: true
expect(page).to have_content 'Application update from 4:00 CST to 5:00 CST' expect(page).to have_content 'Application update from 4:00 CST to 5:00 CST'
expect(page).to have_content 'Guest, Owner'
expect(page).to have_content '*/user_onboarded' expect(page).to have_content '*/user_onboarded'
expect(page).to have_selector 'strong', text: '4:00 CST to 5:00 CST' expect(page).to have_selector 'strong', text: '4:00 CST to 5:00 CST'
expect(page).to have_selector %(div[style="background-color: #f2dede; color: #b94a48"]) expect(page).to have_selector %(div[style="background-color: #f2dede; color: #b94a48"])
...@@ -35,10 +43,14 @@ RSpec.describe 'Admin Broadcast Messages' do ...@@ -35,10 +43,14 @@ RSpec.describe 'Admin Broadcast Messages' do
fill_in 'broadcast_message_target_path', with: '*/user_onboarded' fill_in 'broadcast_message_target_path', with: '*/user_onboarded'
select 'Notification', from: 'broadcast_message_broadcast_type' select 'Notification', from: 'broadcast_message_broadcast_type'
select Date.today.next_year.year, from: 'broadcast_message_ends_at_1i' select Date.today.next_year.year, from: 'broadcast_message_ends_at_1i'
select 'Reporter', from: 'broadcast_message_target_access_levels'
select 'Developer', from: 'broadcast_message_target_access_levels'
select 'Maintainer', from: 'broadcast_message_target_access_levels'
click_button 'Add broadcast message' click_button 'Add broadcast message'
expect(page).to have_current_path admin_broadcast_messages_path, ignore_query: true expect(page).to have_current_path admin_broadcast_messages_path, ignore_query: true
expect(page).to have_content 'Application update from 4:00 CST to 5:00 CST' expect(page).to have_content 'Application update from 4:00 CST to 5:00 CST'
expect(page).to have_content 'Reporter, Developer, Maintainer'
expect(page).to have_content '*/user_onboarded' expect(page).to have_content '*/user_onboarded'
expect(page).to have_content 'Notification' expect(page).to have_content 'Notification'
expect(page).to have_selector 'strong', text: '4:00 CST to 5:00 CST' expect(page).to have_selector 'strong', text: '4:00 CST to 5:00 CST'
...@@ -47,10 +59,15 @@ RSpec.describe 'Admin Broadcast Messages' do ...@@ -47,10 +59,15 @@ RSpec.describe 'Admin Broadcast Messages' do
it 'edit an existing broadcast message' do it 'edit an existing broadcast message' do
click_link 'Edit' click_link 'Edit'
fill_in 'broadcast_message_message', with: 'Application update RIGHT NOW' fill_in 'broadcast_message_message', with: 'Application update RIGHT NOW'
select 'Reporter', from: 'broadcast_message_target_access_levels'
click_button 'Update broadcast message' click_button 'Update broadcast message'
expect(page).to have_current_path admin_broadcast_messages_path, ignore_query: true expect(page).to have_current_path admin_broadcast_messages_path, ignore_query: true
expect(page).to have_content 'Application update RIGHT NOW' expect(page).to have_content 'Application update RIGHT NOW'
page.within('.table-responsive') do
expect(page).to have_content 'Reporter, Developer'
end
end end
it 'remove an existing broadcast message' do it 'remove an existing broadcast message' do
......
...@@ -3,6 +3,71 @@ ...@@ -3,6 +3,71 @@
require 'spec_helper' require 'spec_helper'
RSpec.describe BroadcastMessagesHelper do RSpec.describe BroadcastMessagesHelper do
include Gitlab::Routing.url_helpers
let_it_be(:user) { create(:user) }
before do
allow(helper).to receive(:current_user).and_return(user)
end
shared_examples 'returns role-targeted broadcast message when in project, group, or sub-group URL' do
let(:feature_flag_state) { true }
before do
stub_feature_flags(role_targeted_broadcast_messages: feature_flag_state)
allow(helper).to receive(:cookies) { {} }
end
context 'when in a project page' do
let_it_be(:project) { create(:project) }
before do
project.add_developer(user)
assign(:project, project)
allow(helper).to receive(:controller) { ProjectsController.new }
end
it { is_expected.to eq message }
context 'when feature flag is disabled' do
let(:feature_flag_state) { false }
it { is_expected.to be_nil }
end
end
context 'when in a group page' do
let_it_be(:group) { create(:group) }
before do
group.add_developer(user)
assign(:group, group)
allow(helper).to receive(:controller) { GroupsController.new }
end
it { is_expected.to eq message }
context 'when feature flag is disabled' do
let(:feature_flag_state) { false }
it { is_expected.to be_nil }
end
end
context 'when not in a project, group, or sub-group page' do
it { is_expected.to be_nil }
context 'when feature flag is disabled' do
let(:feature_flag_state) { false }
it { is_expected.to be_nil }
end
end
end
describe 'current_broadcast_notification_message' do describe 'current_broadcast_notification_message' do
subject { helper.current_broadcast_notification_message } subject { helper.current_broadcast_notification_message }
...@@ -24,16 +89,26 @@ RSpec.describe BroadcastMessagesHelper do ...@@ -24,16 +89,26 @@ RSpec.describe BroadcastMessagesHelper do
context 'without broadcast notification messages' do context 'without broadcast notification messages' do
it { is_expected.to be_nil } it { is_expected.to be_nil }
end end
describe 'user access level targeted messages' do
let_it_be(:message) { create(:broadcast_message, broadcast_type: 'notification', starts_at: Time.now, target_access_levels: [Gitlab::Access::DEVELOPER]) }
include_examples 'returns role-targeted broadcast message when in project, group, or sub-group URL'
end
end end
describe 'broadcast_message' do describe 'current_broadcast_banner_messages' do
let_it_be(:user) { create(:user) } describe 'user access level targeted messages' do
let_it_be(:message) { create(:broadcast_message, broadcast_type: 'banner', starts_at: Time.now, target_access_levels: [Gitlab::Access::DEVELOPER]) }
let(:current_broadcast_message) { BroadcastMessage.new(message: 'Current Message') } subject { helper.current_broadcast_banner_messages.first }
before do include_examples 'returns role-targeted broadcast message when in project, group, or sub-group URL'
allow(helper).to receive(:current_user).and_return(user)
end end
end
describe 'broadcast_message' do
let(:current_broadcast_message) { BroadcastMessage.new(message: 'Current Message') }
it 'returns nil when no current message' do it 'returns nil when no current message' do
expect(helper.broadcast_message(nil)).to be_nil expect(helper.broadcast_message(nil)).to be_nil
......
...@@ -23,6 +23,8 @@ RSpec.describe BroadcastMessage do ...@@ -23,6 +23,8 @@ RSpec.describe BroadcastMessage do
it { is_expected.to allow_value(1).for(:broadcast_type) } it { is_expected.to allow_value(1).for(:broadcast_type) }
it { is_expected.not_to allow_value(nil).for(:broadcast_type) } it { is_expected.not_to allow_value(nil).for(:broadcast_type) }
it { is_expected.not_to allow_value(nil).for(:target_access_levels) }
it { is_expected.to validate_inclusion_of(:target_access_levels).in_array(described_class::ALLOWED_TARGET_ACCESS_LEVELS) }
end end
shared_examples 'time constrainted' do |broadcast_type| shared_examples 'time constrainted' do |broadcast_type|
...@@ -175,12 +177,112 @@ RSpec.describe BroadcastMessage do ...@@ -175,12 +177,112 @@ RSpec.describe BroadcastMessage do
end end
end end
shared_examples "matches with user access level" do |broadcast_type|
let_it_be(:target_access_levels) { [Gitlab::Access::GUEST] }
let(:feature_flag_state) { true }
before do
stub_feature_flags(role_targeted_broadcast_messages: feature_flag_state)
end
context 'when feature flag is disabled' do
let(:feature_flag_state) { false }
context 'when message is role-targeted' do
let_it_be(:message) { create(:broadcast_message, target_access_levels: target_access_levels, broadcast_type: broadcast_type) }
it 'does not return the message' do
expect(subject.call(nil, Gitlab::Access::GUEST)).to be_empty
end
end
context 'when message is not role-targeted' do
let_it_be(:message) { create(:broadcast_message, target_access_levels: [], broadcast_type: broadcast_type) }
it 'returns the message' do
expect(subject.call(nil, Gitlab::Access::GUEST)).to include(message)
end
end
end
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
let_it_be(:message) { create(:broadcast_message, target_access_levels: target_access_levels, broadcast_type: broadcast_type) }
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])
allow(BroadcastMessage).to receive(:cache) { cache }
end
it 'does not raise error (e.g. NoMethodError from nil.empty?)' do
expect { subject.call }.not_to raise_error
end
context 'when feature flag is disabled' do
it 'does not raise error (e.g. NoMethodError from nil.empty?)' do
stub_feature_flags(role_targeted_broadcast_messages: false)
expect { subject.call }.not_to raise_error
end
end
end
end
describe '.current', :use_clean_rails_memory_store_caching do describe '.current', :use_clean_rails_memory_store_caching do
subject { -> (path = nil) { described_class.current(path) } } subject do
-> (path = nil, user_access_level = nil) do
described_class.current(current_path: path, user_access_level: user_access_level)
end
end
it_behaves_like 'time constrainted', :banner it_behaves_like 'time constrainted', :banner
it_behaves_like 'message cache', :banner it_behaves_like 'message cache', :banner
it_behaves_like 'matches with current path', :banner it_behaves_like 'matches with current path', :banner
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
it 'returns both types' do it 'returns both types' do
banner_message = create(:broadcast_message, broadcast_type: :banner) banner_message = create(:broadcast_message, broadcast_type: :banner)
...@@ -191,11 +293,26 @@ RSpec.describe BroadcastMessage do ...@@ -191,11 +293,26 @@ RSpec.describe BroadcastMessage do
end end
describe '.current_banner_messages', :use_clean_rails_memory_store_caching do describe '.current_banner_messages', :use_clean_rails_memory_store_caching do
subject { -> (path = nil) { described_class.current_banner_messages(path) } } subject do
-> (path = nil, user_access_level = nil) do
described_class.current_banner_messages(current_path: path, user_access_level: user_access_level)
end
end
it_behaves_like 'time constrainted', :banner it_behaves_like 'time constrainted', :banner
it_behaves_like 'message cache', :banner it_behaves_like 'message cache', :banner
it_behaves_like 'matches with current path', :banner it_behaves_like 'matches with current path', :banner
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
it 'only returns banners' do it 'only returns banners' do
banner_message = create(:broadcast_message, broadcast_type: :banner) banner_message = create(:broadcast_message, broadcast_type: :banner)
...@@ -206,11 +323,26 @@ RSpec.describe BroadcastMessage do ...@@ -206,11 +323,26 @@ RSpec.describe BroadcastMessage do
end end
describe '.current_notification_messages', :use_clean_rails_memory_store_caching do describe '.current_notification_messages', :use_clean_rails_memory_store_caching do
subject { -> (path = nil) { described_class.current_notification_messages(path) } } subject do
-> (path = nil, user_access_level = nil) do
described_class.current_notification_messages(current_path: path, user_access_level: user_access_level)
end
end
it_behaves_like 'time constrainted', :notification it_behaves_like 'time constrainted', :notification
it_behaves_like 'message cache', :notification it_behaves_like 'message cache', :notification
it_behaves_like 'matches with current path', :notification it_behaves_like 'matches with current path', :notification
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
it 'only returns notifications' do it 'only returns notifications' do
notification_message = create(:broadcast_message, broadcast_type: :notification) notification_message = create(:broadcast_message, broadcast_type: :notification)
......
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe 'admin/broadcast_messages/index' do
describe 'Target roles select and table column' do
let(:feature_flag_state) { true }
let_it_be(:message) { create(:broadcast_message, broadcast_type: 'banner', target_access_levels: [Gitlab::Access::GUEST, Gitlab::Access::DEVELOPER]) }
before do
assign(:broadcast_messages, BroadcastMessage.page(1))
assign(:broadcast_message, BroadcastMessage.new)
stub_feature_flags(role_targeted_broadcast_messages: feature_flag_state)
render
end
it 'rendered' do
expect(rendered).to have_content('Target roles')
expect(rendered).to have_content('Owner')
expect(rendered).to have_content('Guest, Developer')
end
context 'when feature flag is off' do
let(:feature_flag_state) { false }
it 'is not rendered' do
expect(rendered).not_to have_content('Target roles')
expect(rendered).not_to have_content('Owner')
expect(rendered).not_to have_content('Guest, Developer')
end
end
end
end
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment