Commit 2e94399f authored by Fabio Pitino's avatar Fabio Pitino Committed by Marius Bobin

Separate CI minutes notification levels between new and legacy

Track notification levels separately for new and legacy tracking.
Use the feature flag to decide which one betwee new or legacy
tracking to send out the email. Each notification level is set
idependently from the other tracking.

Changelog: changed
EE: true
parent b5848df6
...@@ -8,10 +8,11 @@ module Ci ...@@ -8,10 +8,11 @@ module Ci
attr_reader :level attr_reader :level
def initialize(project, namespace) def initialize(project, namespace, tracking_strategy: nil)
@project = project @project = project
@namespace = project&.shared_runners_limit_namespace || namespace @namespace = project&.shared_runners_limit_namespace || namespace
@level = project || namespace @level = project || namespace
@tracking_strategy = tracking_strategy
end end
def percent_total_minutes_remaining def percent_total_minutes_remaining
...@@ -23,7 +24,7 @@ module Ci ...@@ -23,7 +24,7 @@ module Ci
attr_reader :project, :namespace attr_reader :project, :namespace
def quota def quota
@quota ||= ::Ci::Minutes::Quota.new(namespace) @quota ||= ::Ci::Minutes::Quota.new(namespace, tracking_strategy: @tracking_strategy)
end end
end end
end end
......
...@@ -10,8 +10,8 @@ module Ci ...@@ -10,8 +10,8 @@ module Ci
exceeded: 0 exceeded: 0
}.freeze }.freeze
def initialize(project, namespace) def initialize(project, namespace, tracking_strategy: nil)
@context = Ci::Minutes::Context.new(project, namespace) @context = Ci::Minutes::Context.new(project, namespace, tracking_strategy: tracking_strategy)
@stage = calculate_notification_stage if eligible_for_notifications? @stage = calculate_notification_stage if eligible_for_notifications?
end end
......
...@@ -11,9 +11,12 @@ module Ci ...@@ -11,9 +11,12 @@ module Ci
attr_reader :namespace, :limit attr_reader :namespace, :limit
def initialize(namespace) def initialize(namespace, tracking_strategy: nil)
@namespace = namespace @namespace = namespace
@limit = ::Ci::Minutes::Limit.new(namespace) @limit = ::Ci::Minutes::Limit.new(namespace)
# TODO: remove `tracking_strategy` after `ci_use_new_monthly_minutes` feature flag
# https://gitlab.com/gitlab-org/gitlab/-/issues/341730
@tracking_strategy = tracking_strategy
end end
def enabled? def enabled?
...@@ -36,21 +39,19 @@ module Ci ...@@ -36,21 +39,19 @@ module Ci
def total_minutes_used def total_minutes_used
strong_memoize(:total_minutes_used) do strong_memoize(:total_minutes_used) do
if namespace.new_monthly_ci_minutes_enabled? conditional_value(
current_usage.amount_used.to_i when_new_strategy: -> { current_usage.amount_used.to_i },
else when_legacy_strategy: -> { namespace.namespace_statistics&.shared_runners_seconds.to_i / 60 }
namespace.namespace_statistics&.shared_runners_seconds.to_i / 60 )
end
end end
end end
def reset_date def reset_date
strong_memoize(:reset_date) do strong_memoize(:reset_date) do
if namespace.new_monthly_ci_minutes_enabled? conditional_value(
current_usage.date when_new_strategy: -> { current_usage.date },
else when_legacy_strategy: -> { namespace.namespace_statistics&.shared_runners_seconds_last_reset }
namespace.namespace_statistics&.shared_runners_seconds_last_reset )
end
end end
end end
...@@ -90,6 +91,18 @@ module Ci ...@@ -90,6 +91,18 @@ module Ci
def total_minutes_remaining def total_minutes_remaining
[current_balance, 0].max [current_balance, 0].max
end end
def conditional_value(when_new_strategy:, when_legacy_strategy:)
if @tracking_strategy == :new
when_new_strategy.call
elsif @tracking_strategy == :legacy
when_legacy_strategy.call
elsif namespace.new_monthly_ci_minutes_enabled?
when_new_strategy.call
else
when_legacy_strategy.call
end
end
end end
end end
end end
...@@ -3,58 +3,69 @@ ...@@ -3,58 +3,69 @@
module Ci module Ci
module Minutes module Minutes
class EmailNotificationService < ::BaseService class EmailNotificationService < ::BaseService
include Gitlab::Utils::StrongMemoize
def execute def execute
return unless notification.eligible_for_notifications? return unless notification.eligible_for_notifications?
legacy_notify
notify notify
end end
private private
# We use 2 notification objects for new and legacy tracking side-by-side.
# We read and write data to each tracking using the respective data but we alert
# only based on the currently active tracking.
def notification def notification
@notification ||= ::Ci::Minutes::Notification.new(project, nil) @notification ||= ::Ci::Minutes::Notification.new(project, nil, tracking_strategy: :new)
end
def legacy_notification
@legacy_notification ||= ::Ci::Minutes::Notification.new(project, nil, tracking_strategy: :legacy)
end end
def notify def notify
if notification.no_remaining_minutes? if notification.no_remaining_minutes?
notify_total_usage return if namespace_usage.total_usage_notified?
namespace_usage.update!(notification_level: current_alert_percentage)
if ci_minutes_use_notification_level?
CiMinutesUsageMailer.notify(namespace, recipients).deliver_later
end
elsif notification.running_out? elsif notification.running_out?
notify_partial_usage return if namespace_usage.usage_notified?(current_alert_percentage)
namespace_usage.update!(notification_level: current_alert_percentage)
if ci_minutes_use_notification_level?
CiMinutesUsageMailer.notify_limit(namespace, recipients, current_alert_percentage).deliver_later
end
end end
end end
def notify_total_usage def legacy_notify
# TODO: Enable the FF on the month after this is released. if legacy_notification.no_remaining_minutes?
# https://gitlab.com/gitlab-org/gitlab/-/issues/339324
if Feature.enabled?(:ci_minutes_use_notification_level, namespace, default_enabled: :yaml)
return if namespace_usage.total_usage_notified?
else
return if namespace.last_ci_minutes_notification_at return if namespace.last_ci_minutes_notification_at
end
legacy_track_total_usage
namespace_usage.update!(notification_level: current_alert_percentage)
CiMinutesUsageMailer.notify(namespace, recipients).deliver_later namespace.update_columns(last_ci_minutes_notification_at: Time.current)
end
def notify_partial_usage unless ci_minutes_use_notification_level?
# TODO: Enable the FF on the month after this is released. CiMinutesUsageMailer.notify(namespace, recipients).deliver_later
# https://gitlab.com/gitlab-org/gitlab/-/issues/339324 end
if Feature.enabled?(:ci_minutes_use_notification_level, namespace, default_enabled: :yaml) elsif legacy_notification.running_out?
return if namespace_usage.usage_notified?(current_alert_percentage) current_alert_percentage = legacy_notification.stage_percentage
else
return if already_notified_running_out
end
legacy_track_partial_usage # exit if we have already sent a notification for the same level
namespace_usage.update!(notification_level: current_alert_percentage) return if namespace.last_ci_minutes_usage_notification_level == current_alert_percentage
CiMinutesUsageMailer.notify_limit(namespace, recipients, current_alert_percentage).deliver_later namespace.update_columns(last_ci_minutes_usage_notification_level: current_alert_percentage)
end
def already_notified_running_out unless ci_minutes_use_notification_level?
namespace.last_ci_minutes_usage_notification_level == current_alert_percentage CiMinutesUsageMailer.notify_limit(namespace, recipients, current_alert_percentage).deliver_later
end
end
end end
def recipients def recipients
...@@ -74,14 +85,10 @@ module Ci ...@@ -74,14 +85,10 @@ module Ci
notification.stage_percentage notification.stage_percentage
end end
# TODO: delete this method after full rollout of ci_minutes_use_notification_level Feature Flag def ci_minutes_use_notification_level?
def legacy_track_total_usage strong_memoize(:ci_minutes_use_notification_level) do
namespace.update_columns(last_ci_minutes_notification_at: Time.current) Feature.enabled?(:ci_minutes_use_notification_level, namespace, default_enabled: :yaml)
end end
# TODO: delete this method after full rollout of ci_minutes_use_notification_level Feature Flag
def legacy_track_partial_usage
namespace.update_columns(last_ci_minutes_usage_notification_level: current_alert_percentage)
end end
end end
end end
......
...@@ -92,6 +92,8 @@ RSpec.describe Ci::Minutes::Quota do ...@@ -92,6 +92,8 @@ RSpec.describe Ci::Minutes::Quota do
end end
describe '#total_minutes_used' do describe '#total_minutes_used' do
let(:namespace) { create(:namespace, :with_ci_minutes, ci_minutes_used: minutes_used) }
subject { quota.total_minutes_used } subject { quota.total_minutes_used }
where(:minutes_used, :expected_minutes) do where(:minutes_used, :expected_minutes) do
...@@ -103,10 +105,30 @@ RSpec.describe Ci::Minutes::Quota do ...@@ -103,10 +105,30 @@ RSpec.describe Ci::Minutes::Quota do
end end
with_them do with_them do
let(:namespace) { create(:namespace, :with_ci_minutes, ci_minutes_used: minutes_used) }
it { is_expected.to eq(expected_minutes) } it { is_expected.to eq(expected_minutes) }
end end
context 'with tracking_strategy' do
where(:minutes_used, :legacy_minutes_used, :tracking_strategy, :ff_enabled, :expected_minutes) do
0 | 100 | nil | true | 0
0 | 100 | nil | false | 100
0 | 100 | :new | true | 0
0 | 100 | :new | false | 0
0 | 100 | :legacy | true | 100
0 | 100 | :legacy | false | 100
end
with_them do
let(:quota) { described_class.new(namespace, tracking_strategy: tracking_strategy) }
before do
stub_feature_flags(ci_use_new_monthly_minutes: ff_enabled)
namespace.namespace_statistics.update!(shared_runners_seconds: legacy_minutes_used.minutes)
end
it { is_expected.to eq(expected_minutes) }
end
end
end end
describe '#percent_total_minutes_remaining' do describe '#percent_total_minutes_remaining' do
...@@ -214,6 +236,9 @@ RSpec.describe Ci::Minutes::Quota do ...@@ -214,6 +236,9 @@ RSpec.describe Ci::Minutes::Quota do
end end
describe '#reset_date' do describe '#reset_date' do
let(:quota) { described_class.new(namespace, tracking_strategy: tracking_strategy) }
let(:tracking_strategy) { nil }
subject(:reset_date) { quota.reset_date } subject(:reset_date) { quota.reset_date }
around do |example| around do |example|
...@@ -228,6 +253,22 @@ RSpec.describe Ci::Minutes::Quota do ...@@ -228,6 +253,22 @@ RSpec.describe Ci::Minutes::Quota do
expect(reset_date).to eq(Date.new(2021, 07, 1)) expect(reset_date).to eq(Date.new(2021, 07, 1))
end end
context 'when tracking_strategy: :new' do
let(:tracking_strategy) { :new }
it 'corresponds to the beginning of the current month' do
expect(reset_date).to eq(Date.new(2021, 07, 1))
end
end
context 'when tracking_strategy: :legacy' do
let(:tracking_strategy) { :legacy }
it 'corresponds to the current time' do
expect(reset_date).to eq(Date.new(2021, 07, 14))
end
end
context 'when feature flag ci_use_new_monthly_minutes is disabled' do context 'when feature flag ci_use_new_monthly_minutes is disabled' do
before do before do
stub_feature_flags(ci_use_new_monthly_minutes: false) stub_feature_flags(ci_use_new_monthly_minutes: false)
......
...@@ -10,73 +10,86 @@ RSpec.describe Ci::Minutes::EmailNotificationService do ...@@ -10,73 +10,86 @@ RSpec.describe Ci::Minutes::EmailNotificationService do
subject { described_class.new(project).execute } subject { described_class.new(project).execute }
def expect_warning_usage_notification(new_notification_level) where(:ff_enabled, :monthly_minutes_limit, :minutes_used, :current_notification_level, :new_notification_level, :legacy_minutes_used, :legacy_current_notification_level, :legacy_new_notification_level, :result) do
expect(CiMinutesUsageMailer) # when legacy and new tracking usage matches
.to receive(:notify_limit) true | 1000 | 500 | 100 | 100 | 500 | 100 | 100 | [false]
.with(namespace, match_array(recipients), new_notification_level) false | 1000 | 500 | 100 | 100 | 500 | 100 | 100 | [false]
.and_call_original true | 1000 | 800 | 100 | 30 | 800 | 100 | 30 | [true, 30]
false | 1000 | 800 | 100 | 30 | 800 | 100 | 30 | [true, 30]
subject true | 1000 | 800 | 30 | 30 | 800 | 30 | 30 | [false]
false | 1000 | 800 | 30 | 30 | 800 | 30 | 30 | [false]
expect(namespace_usage.reload.notification_level).to eq(new_notification_level) true | 1000 | 950 | 100 | 5 | 950 | 100 | 5 | [true, 5]
expect(namespace.reload.last_ci_minutes_usage_notification_level).to eq(new_notification_level) false | 1000 | 950 | 100 | 5 | 950 | 100 | 5 | [true, 5]
end true | 1000 | 950 | 30 | 5 | 950 | 30 | 5 | [true, 5]
false | 1000 | 950 | 30 | 5 | 950 | 30 | 5 | [true, 5]
def expect_quota_exceeded_notification true | 1000 | 950 | 5 | 5 | 950 | 5 | 5 | [false]
expect(CiMinutesUsageMailer) false | 1000 | 950 | 5 | 5 | 950 | 5 | 5 | [false]
.to receive(:notify) true | 1000 | 1000 | 100 | 0 | 1000 | 100 | 0 | [true, 0]
.with(namespace, match_array(recipients)) false | 1000 | 1000 | 100 | 0 | 1000 | 100 | 0 | [true, 0]
.and_call_original true | 1000 | 1000 | 30 | 0 | 1000 | 30 | 0 | [true, 0]
false | 1000 | 1000 | 30 | 0 | 1000 | 30 | 0 | [true, 0]
subject true | 1000 | 1000 | 5 | 0 | 1000 | 5 | 0 | [true, 0]
false | 1000 | 1000 | 5 | 0 | 1000 | 5 | 0 | [true, 0]
expect(namespace_usage.reload.notification_level).to eq(0) true | 1000 | 1001 | 5 | 0 | 1001 | 5 | 0 | [true, 0]
expect(namespace.reload.last_ci_minutes_notification_at).to be_present false | 1000 | 1001 | 5 | 0 | 1001 | 5 | 0 | [true, 0]
end true | 1000 | 1000 | 0 | 0 | 1000 | 0 | 0 | [false]
false | 1000 | 1000 | 0 | 0 | 1000 | 0 | 0 | [false]
def expect_no_notification(current_notification_level) true | 0 | 1000 | 100 | 100 | 1000 | 100 | 100 | [false]
expect(CiMinutesUsageMailer) false | 0 | 1000 | 100 | 100 | 1000 | 100 | 100 | [false]
.not_to receive(:notify_limit)
expect(CiMinutesUsageMailer) # when legacy and new tracking usage doesn't match we send notifications
.not_to receive(:notify) # based on the feature flag.
true | 1000 | 500 | 100 | 100 | 800 | 100 | 30 | [false]
subject false | 1000 | 500 | 100 | 100 | 800 | 100 | 30 | [true, 30]
true | 1000 | 800 | 100 | 30 | 500 | 100 | 100 | [true, 30]
# notification level remains the same false | 1000 | 800 | 100 | 30 | 500 | 100 | 100 | [false]
expect(namespace_usage.reload.notification_level).to eq(current_notification_level) true | 1000 | 950 | 100 | 5 | 800 | 100 | 30 | [true, 5]
end false | 1000 | 950 | 100 | 5 | 800 | 100 | 30 | [true, 30]
true | 1000 | 950 | 100 | 5 | 1001 | 30 | 0 | [true, 5]
where(:monthly_minutes_limit, :minutes_used, :current_notification_level, :result) do false | 1000 | 950 | 100 | 5 | 1001 | 30 | 0 | [true, 0]
1000 | 500 | 100 | [:not_notified]
1000 | 800 | 100 | [:notified, 30]
1000 | 800 | 30 | [:not_notified]
1000 | 950 | 100 | [:notified, 5]
1000 | 950 | 30 | [:notified, 5]
1000 | 950 | 5 | [:not_notified]
1000 | 1000 | 100 | [:notified, 0]
1000 | 1000 | 30 | [:notified, 0]
1000 | 1000 | 5 | [:notified, 0]
1000 | 1001 | 5 | [:notified, 0]
1000 | 1000 | 0 | [:not_notified]
0 | 1000 | 100 | [:not_notified]
end end
with_them do with_them do
shared_examples 'matches the expectation' do shared_examples 'matches the expectations' do
it 'matches the expectation' do it 'matches the expectation on the email sent' do
expectation, new_notification_level = result email_sent, level_notified = result
if email_sent
if level_notified > 0
expect(CiMinutesUsageMailer)
.to receive(:notify_limit)
.with(namespace, match_array(recipients), level_notified)
.and_call_original
else
expect(CiMinutesUsageMailer)
.to receive(:notify)
.with(namespace, match_array(recipients))
.and_call_original
end
else
expect(CiMinutesUsageMailer).not_to receive(:notify_limit)
expect(CiMinutesUsageMailer).not_to receive(:notify)
end
if expectation == :notified && new_notification_level > 0 subject
expect_warning_usage_notification(new_notification_level) end
elsif expectation == :notified && new_notification_level == 0 it 'matches the updated notification level' do
expect_quota_exceeded_notification subject
elsif expectation == :not_notified expect(namespace_usage.reload.notification_level).to eq(new_notification_level)
expect_no_notification(current_notification_level) end
it 'matches the updated legacy notification level' do
subject
if legacy_new_notification_level == 0
expect(namespace.reload.last_ci_minutes_notification_at).to be_present
elsif legacy_new_notification_level == 100
expect(namespace.reload.last_ci_minutes_notification_at).to be_nil
expect(namespace.reload.last_ci_minutes_usage_notification_level).to be_nil
else else
raise 'unexpected test scenario' expect(namespace.reload.last_ci_minutes_usage_notification_level).to eq(legacy_new_notification_level)
end end
end end
end end
...@@ -91,15 +104,23 @@ RSpec.describe Ci::Minutes::EmailNotificationService do ...@@ -91,15 +104,23 @@ RSpec.describe Ci::Minutes::EmailNotificationService do
end end
before do before do
set_ci_minutes_used(namespace, minutes_used) stub_feature_flags(ci_minutes_use_notification_level: ff_enabled)
if namespace.namespace_statistics
namespace.namespace_statistics.update!(shared_runners_seconds: legacy_minutes_used.minutes)
else
namespace.create_namespace_statistics(shared_runners_seconds: legacy_minutes_used.minutes)
end
namespace_usage.update!(amount_used: minutes_used)
namespace_usage.update_column(:notification_level, current_notification_level) namespace_usage.update_column(:notification_level, current_notification_level)
namespace.update_column(:shared_runners_minutes_limit, monthly_minutes_limit) namespace.update_column(:shared_runners_minutes_limit, monthly_minutes_limit)
if current_notification_level == 0 if legacy_current_notification_level == 0
namespace.update_column(:last_ci_minutes_notification_at, Time.current) namespace.update_column(:last_ci_minutes_notification_at, Time.current)
elsif current_notification_level != 100 elsif current_notification_level != 100
namespace.update_column(:last_ci_minutes_usage_notification_level, current_notification_level) namespace.update_column(:last_ci_minutes_usage_notification_level, legacy_current_notification_level)
end end
end end
...@@ -107,15 +128,7 @@ RSpec.describe Ci::Minutes::EmailNotificationService do ...@@ -107,15 +128,7 @@ RSpec.describe Ci::Minutes::EmailNotificationService do
let(:namespace) { create(:namespace, owner: user) } let(:namespace) { create(:namespace, owner: user) }
let(:recipients) { [user.email] } let(:recipients) { [user.email] }
it_behaves_like 'matches the expectation' it_behaves_like 'matches the expectations'
context 'when feature flag ci_minutes_use_notification_level is disabled' do
before do
stub_feature_flags(ci_minutes_use_notification_level: false)
end
it_behaves_like 'matches the expectation'
end
end end
context 'when on group' do context 'when on group' do
...@@ -127,14 +140,143 @@ RSpec.describe Ci::Minutes::EmailNotificationService do ...@@ -127,14 +140,143 @@ RSpec.describe Ci::Minutes::EmailNotificationService do
namespace.add_owner(user_2) namespace.add_owner(user_2)
end end
it_behaves_like 'matches the expectation' it_behaves_like 'matches the expectations'
end
end
context 'legacy path - to remove' do
def expect_warning_usage_notification(new_notification_level)
expect(CiMinutesUsageMailer)
.to receive(:notify_limit)
.with(namespace, match_array(recipients), new_notification_level)
.and_call_original
subject
expect(namespace_usage.reload.notification_level).to eq(new_notification_level)
expect(namespace.reload.last_ci_minutes_usage_notification_level).to eq(new_notification_level)
end
def expect_quota_exceeded_notification
expect(CiMinutesUsageMailer)
.to receive(:notify)
.with(namespace, match_array(recipients))
.and_call_original
subject
expect(namespace_usage.reload.notification_level).to eq(0)
expect(namespace.reload.last_ci_minutes_notification_at).to be_present
end
def expect_no_notification(current_notification_level)
expect(CiMinutesUsageMailer)
.not_to receive(:notify_limit)
expect(CiMinutesUsageMailer)
.not_to receive(:notify)
subject
# notification level remains the same
expect(namespace_usage.reload.notification_level).to eq(current_notification_level)
end
where(:monthly_minutes_limit, :minutes_used, :legacy_minutes_used, :current_notification_level, :legacy_notification_level, :result, :legacy_result) do
1000 | 500 | 500 | 100 | 100 | [:not_notified] | [:not_notified]
1000 | 800 | 800 | 100 | 100 | [:notified, 30] | [:notified, 30]
1000 | 800 | 800 | 30 | 30 | [:not_notified] | [:not_notified]
1000 | 950 | 950 | 100 | 100 | [:notified, 5] | [:notified, 5]
1000 | 950 | 950 | 30 | 30 | [:notified, 5] | [:notified, 5]
1000 | 950 | 950 | 5 | 5 | [:not_notified] | [:not_notified]
1000 | 1000 | 1000 | 100 | 100 | [:notified, 0] | [:notified, 0]
1000 | 1000 | 1000 | 30 | 30 | [:notified, 0] | [:notified, 0]
1000 | 1000 | 1000 | 5 | 5 | [:notified, 0] | [:notified, 0]
1000 | 1001 | 1001 | 5 | 5 | [:notified, 0] | [:notified, 0]
1000 | 1000 | 1000 | 0 | 0 | [:not_notified] | [:not_notified]
0 | 1000 | 1000 | 100 | 100 | [:not_notified] | [:not_notified]
end
with_them do
shared_examples 'matches the expectation' do
it 'matches the expectation' do
expectation, new_notification_level = result
if expectation == :notified && new_notification_level > 0
expect_warning_usage_notification(new_notification_level)
elsif expectation == :notified && new_notification_level == 0
expect_quota_exceeded_notification
elsif expectation == :not_notified
expect_no_notification(current_notification_level)
else
raise 'unexpected test scenario'
end
end
end
let_it_be(:user) { create(:user) }
let_it_be(:user_2) { create(:user) }
let(:project) { create(:project, namespace: namespace) }
let(:namespace_usage) do
Ci::Minutes::NamespaceMonthlyUsage.find_or_create_current(namespace_id: namespace.id)
end
before do
if namespace.namespace_statistics
namespace.namespace_statistics.update!(shared_runners_seconds: legacy_minutes_used.minutes)
else
namespace.create_namespace_statistics(shared_runners_seconds: legacy_minutes_used.minutes)
end
namespace_usage.update!(amount_used: minutes_used)
namespace_usage.update_column(:notification_level, current_notification_level)
namespace.update_column(:shared_runners_minutes_limit, monthly_minutes_limit)
if current_notification_level == 0
namespace.update_column(:last_ci_minutes_notification_at, Time.current)
elsif current_notification_level != 100
namespace.update_column(:last_ci_minutes_usage_notification_level, current_notification_level)
end
end
context 'when on personal namespace' do
let(:namespace) { create(:namespace, owner: user) }
let(:recipients) { [user.email] }
it_behaves_like 'matches the expectation'
context 'when feature flag ci_minutes_use_notification_level is disabled' do
before do
stub_feature_flags(ci_minutes_use_notification_level: false)
end
it_behaves_like 'matches the expectation'
end
end
context 'when on group' do
let(:namespace) { create(:group) }
let(:recipients) { [user.email, user_2.email] }
context 'when feature flag ci_minutes_use_notification_level is disabled' do
before do before do
stub_feature_flags(ci_minutes_use_notification_level: false) namespace.add_owner(user)
namespace.add_owner(user_2)
end end
it_behaves_like 'matches the expectation' it_behaves_like 'matches the expectation'
context 'when feature flag ci_minutes_use_notification_level is disabled' do
before do
stub_feature_flags(ci_minutes_use_notification_level: false)
end
it_behaves_like 'matches the expectation'
end
end end
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