diff --git a/app/assets/javascripts/persistent_user_callouts.js b/app/assets/javascripts/persistent_user_callouts.js index 120c71273fae941e40f032e8842981c6b06e0032..7e996223f9ed033c00dad856f34e31224001b76d 100644 --- a/app/assets/javascripts/persistent_user_callouts.js +++ b/app/assets/javascripts/persistent_user_callouts.js @@ -26,6 +26,7 @@ const PERSISTENT_USER_CALLOUTS = [ '.js-joining-a-project-alert', '.js-duo-pro-trial-alert', '.js-duo-chat-ga-alert', + '.js-all-seats-used', ]; const initCallouts = () => { diff --git a/app/models/users/group_callout.rb b/app/models/users/group_callout.rb index 74b653b57776d3b5972521da276842e41c3e46d7..5dc5140f386b6a555ef4a520bc4f6772b0f3ce05 100644 --- a/app/models/users/group_callout.rb +++ b/app/models/users/group_callout.rb @@ -30,7 +30,8 @@ class GroupCallout < ApplicationRecord project_repository_limit_alert_warning_threshold: 20, # EE-only project_repository_limit_alert_alert_threshold: 21, # EE-only project_repository_limit_alert_error_threshold: 22, # EE-only - namespace_over_storage_users_combined_alert: 23 # EE-only + namespace_over_storage_users_combined_alert: 23, # EE-only + all_seats_used_alert: 24 # EE-only } validates :group, presence: true diff --git a/app/views/layouts/_page.html.haml b/app/views/layouts/_page.html.haml index d854190b73454cf92278bd4e6a8d6de109d7b856..f4428f1ccc7c7ae80132d193b866da6b58a575ed 100644 --- a/app/views/layouts/_page.html.haml +++ b/app/views/layouts/_page.html.haml @@ -1,6 +1,7 @@ .layout-page{ class: page_with_sidebar_class } -# Render the parent group sidebar while creating a new subgroup/project, see GroupsController#new. - group = @parent_group || @group + - context = group || @project - sidebar_panel = super_sidebar_nav_panel(nav: nav, user: current_user, group: group, project: @project, current_ref: current_ref, ref_type: @ref_type, viewed_user: @user, organization: @organization) - sidebar_data = super_sidebar_context(current_user, group: group, project: @project, panel: sidebar_panel, panel_type: nav).to_json @@ -28,6 +29,7 @@ = dispensable_render "shared/service_ping_consent" = dispensable_render_if_exists "layouts/header/ee_subscribable_banner" = dispensable_render_if_exists "layouts/header/seat_count_alert" + = dispensable_render_if_exists "layouts/header/all_seats_used_alert", context: context = dispensable_render_if_exists "shared/namespace_user_cap_reached_alert" = dispensable_render_if_exists "shared/new_user_signups_cap_reached_alert" = dispensable_render_if_exists "shared/silent_mode_banner" diff --git a/ee/app/components/namespaces/block_seat_overages/all_seats_used_alert_component.html.haml b/ee/app/components/namespaces/block_seat_overages/all_seats_used_alert_component.html.haml new file mode 100644 index 0000000000000000000000000000000000000000..d4e99d6b33211f89ed68f5e40e734fc5410afe7e --- /dev/null +++ b/ee/app/components/namespaces/block_seat_overages/all_seats_used_alert_component.html.haml @@ -0,0 +1,13 @@ +%div{ class: [@content_class, 'gl-pt-5!'] } + = render Pajamas::AlertComponent.new(alert_options: { class: 'js-all-seats-used', + data: { dismiss_endpoint: group_callouts_path, + feature_id: EE::Users::GroupCalloutsHelper::ALL_SEATS_USED_ALERT, + group_id: root_namespace.id, + testid: 'all-seats-used-alert' }}, + title: _('No more seats in subscription'), + variant: :warning) do |c| + - c.with_body do + %p= _('Your namespace has used all the seats in your subscription and users can no longer be invited or added to the namespace.') + - c.with_actions do + = render Pajamas::ButtonComponent.new(variant: :confirm, href: help_page_path('subscriptions/gitlab_com/index', anchor: 'add-seats-to-your-subscription')) do + = s_('Purchase more seats') diff --git a/ee/app/components/namespaces/block_seat_overages/all_seats_used_alert_component.rb b/ee/app/components/namespaces/block_seat_overages/all_seats_used_alert_component.rb new file mode 100644 index 0000000000000000000000000000000000000000..23e986bc5c2b3f0b908676ba807223e38064e0a9 --- /dev/null +++ b/ee/app/components/namespaces/block_seat_overages/all_seats_used_alert_component.rb @@ -0,0 +1,55 @@ +# frozen_string_literal: true + +module Namespaces + module BlockSeatOverages + class AllSeatsUsedAlertComponent < ViewComponent::Base + def initialize(context:, content_class:, current_user:) + @root_namespace = context&.root_ancestor + @content_class = content_class + @current_user = current_user + end + + attr_reader :root_namespace, :content_class, :current_user + + def render? + return false unless group_namespace? && owner? && block_seat_overages? && !user_dismissed_alert? + + all_seats_used? + end + + private + + def group_namespace? + root_namespace&.group_namespace? + end + + def block_seat_overages? + subscription&.has_a_paid_hosted_plan? && root_namespace.block_seat_overages? + end + + def subscription + root_namespace.gitlab_subscription + end + + def owner? + Ability.allowed?(current_user, :owner_access, root_namespace) + end + + def all_seats_used? + billable_members_count = root_namespace.billable_members_count_with_reactive_cache + + return false if billable_members_count.blank? + + # We use `==` here because there is another banner for seat overage + subscription.seats == billable_members_count + end + + def user_dismissed_alert? + current_user.dismissed_callout_for_group?( + feature_name: EE::Users::GroupCalloutsHelper::ALL_SEATS_USED_ALERT, + group: root_namespace + ) + end + end + end +end diff --git a/ee/app/helpers/ee/users/group_callouts_helper.rb b/ee/app/helpers/ee/users/group_callouts_helper.rb index 87a72d79585960c99868c5efab1333a1c6b3d084..066fa4d33a414402bfd5ccbe80e5e158cebc2703 100644 --- a/ee/app/helpers/ee/users/group_callouts_helper.rb +++ b/ee/app/helpers/ee/users/group_callouts_helper.rb @@ -4,6 +4,7 @@ module EE module Users module GroupCalloutsHelper UNLIMITED_MEMBERS_DURING_TRIAL_ALERT = 'unlimited_members_during_trial_alert' + ALL_SEATS_USED_ALERT = 'all_seats_used_alert' def show_unlimited_members_during_trial_alert?(group) ::Namespaces::FreeUserCap::Enforcement.new(group).qualified_namespace? && diff --git a/ee/app/services/gitlab_subscriptions/reconciliations/calculate_seat_count_data_service.rb b/ee/app/services/gitlab_subscriptions/reconciliations/calculate_seat_count_data_service.rb index 6c895cebfe75ad2fd9379153b8c1c6fa77534793..3980426b57b057d64f821ef451bdef2b4d08d182 100644 --- a/ee/app/services/gitlab_subscriptions/reconciliations/calculate_seat_count_data_service.rb +++ b/ee/app/services/gitlab_subscriptions/reconciliations/calculate_seat_count_data_service.rb @@ -1,5 +1,5 @@ # frozen_string_literal: true -# + module GitlabSubscriptions module Reconciliations class CalculateSeatCountDataService diff --git a/ee/app/views/layouts/header/_all_seats_used_alert.html.haml b/ee/app/views/layouts/header/_all_seats_used_alert.html.haml new file mode 100644 index 0000000000000000000000000000000000000000..b54114bda9a93bc31403eba45414d8c88e4fa927 --- /dev/null +++ b/ee/app/views/layouts/header/_all_seats_used_alert.html.haml @@ -0,0 +1 @@ += render Namespaces::BlockSeatOverages::AllSeatsUsedAlertComponent.new(context: context, content_class: full_content_class, current_user: current_user) diff --git a/ee/spec/components/namespaces/block_seat_overages/all_seats_used_alert_component_spec.rb b/ee/spec/components/namespaces/block_seat_overages/all_seats_used_alert_component_spec.rb new file mode 100644 index 0000000000000000000000000000000000000000..b56bd974c6c68c9466d6387efedb61fd6a0956f5 --- /dev/null +++ b/ee/spec/components/namespaces/block_seat_overages/all_seats_used_alert_component_spec.rb @@ -0,0 +1,92 @@ +# frozen_string_literal: true + +require "spec_helper" + +RSpec.describe Namespaces::BlockSeatOverages::AllSeatsUsedAlertComponent, type: :component, feature_category: :consumables_cost_management do + include ReactiveCachingHelpers + + let_it_be(:current_user) { build(:user) } + let_it_be(:namespace) { build(:group) } + + let(:billable_members_count) { 2 } + let(:permission_owner) { true } + + before do + allow(namespace).to receive(:billable_members_count).and_return(billable_members_count) + allow(Ability).to receive(:allowed?).with(current_user, :owner_access, namespace).and_return(permission_owner) + + build(:gitlab_subscription, namespace: namespace, plan_code: Plan::ULTIMATE, seats: 2) + end + + describe '#render?' do + subject { component.render? } + + before do + allow(current_user).to receive(:dismissed_callout_for_group?).and_return(false) + end + + context 'in a saas environment', :saas do + context 'with a reactive cache hit' do + before do + synchronous_reactive_cache(namespace) + end + + describe 'when user has dismissed alert' do + before do + allow(current_user).to receive(:dismissed_callout_for_group?).and_return(true) + end + + it { is_expected.to be false } + end + + describe 'when namespace has no paid plan' do + before do + build(:gitlab_subscription, namespace: namespace, plan_code: Plan::FREE) + end + + it { is_expected.to be false } + end + + describe 'when user is not a owner' do + let(:permission_owner) { false } + + it { is_expected.to be false } + end + + describe 'when block seats overages is false' do + before do + stub_feature_flags(block_seat_overages: false) + end + + it { is_expected.to be false } + end + + describe 'with no billable members' do + let(:billable_members_count) { 0 } + + it { is_expected.to be false } + end + + describe 'when namespace is personal' do + let_it_be(:namespace) { build(:user).namespace } + + it { is_expected.to be false } + end + + it { is_expected.to be true } + end + + context 'with a reactive cache miss' do + before do + stub_reactive_cache(namespace, nil) + end + + it { is_expected.to be false } + end + end + end + + def component(context = namespace) + described_class.new(context: context, content_class: '', current_user: current_user) + end +end diff --git a/ee/spec/features/groups/group_overview_spec.rb b/ee/spec/features/groups/group_overview_spec.rb index 3fcc6c4f85151034fa7c7101625685ae919b9257..ea23d3fcb2eeb0a4bc0bea873b455e348909a6cc 100644 --- a/ee/spec/features/groups/group_overview_spec.rb +++ b/ee/spec/features/groups/group_overview_spec.rb @@ -4,6 +4,7 @@ RSpec.describe 'Group information', :js, :aggregate_failures, feature_category: :groups_and_projects do include BillableMembersHelpers + using RSpec::Parameterized::TableSyntax let_it_be(:user) { create(:user) } let_it_be(:group) { create(:group) } @@ -45,8 +46,8 @@ page.within(find('.content')) do expect(page).to have_content s_("SecurityReports|Either you don't have permission to view this dashboard or "\ - 'the dashboard has not been setup. Please check your permission settings '\ - 'with your administrator or check your dashboard configurations to proceed.') + 'the dashboard has not been setup. Please check your permission settings '\ + 'with your administrator or check your dashboard configurations to proceed.') end end end @@ -92,6 +93,75 @@ it_behaves_like 'over the free user limit alert' end + context 'with all seats used alert', :saas, :use_clean_rails_memory_store_caching do + context 'when all seats are used' do + let_it_be(:subscription) { create(:gitlab_subscription, :premium, namespace: group, seats: 1) } + + before do + stub_billable_members_reactive_cache(group) + end + + context 'when the user is an owner' do + it 'displays the all seats used alert' do + visit_page + + expect(page).to have_css '[data-testid="all-seats-used-alert"].gl-alert-warning' + + within_testid('all-seats-used-alert') do + expect(page).to have_css('[data-testid="close-icon"]') + expect(page).to have_text "No more seats in subscription" + expect(page).to have_text "Your namespace has used all the seats in your subscription and users can " \ + "no longer be invited or added to the namespace." + expect(page).to have_link 'Purchase more seats', href: + help_page_path('subscriptions/gitlab_com/index', anchor: 'add-seats-to-your-subscription') + end + end + + context 'when the user is not an owner' do + where(:role) do + ::Gitlab::Access.sym_options.keys.map(&:to_sym) + end + + with_them do + it 'does not display the all seats used alert' do + visit_page + + expect(page).not_to have_css '[data-testid="all-seats-used-alert"].gl-alert-warning' + end + end + end + end + end + + context 'when not all seats are used' do + let_it_be(:subscription) { create(:gitlab_subscription, :premium, namespace: group, seats: 5) } + + before do + stub_billable_members_reactive_cache(group) + end + + it 'does not display the all seats used alert' do + visit_page + + expect(page).not_to have_css '[data-testid="all-seats-used-alert"].gl-alert-warning' + end + end + + context 'with a free plan' do + let_it_be(:subscription) { create(:gitlab_subscription, :free, namespace: group, seats: 1) } + + before do + stub_billable_members_reactive_cache(group) + end + + it 'does not display the all seats used alert' do + visit_page + + expect(page).not_to have_css '[data-testid="all-seats-used-alert"].gl-alert-warning' + end + end + end + context 'when there is a seat overage', :saas, :use_clean_rails_memory_store_caching do let_it_be(:subscription) { create(:gitlab_subscription, :premium, namespace: group, seats: 1) } @@ -100,8 +170,6 @@ end before do - stub_feature_flags(block_seat_overages: true) - stub_billable_members_reactive_cache(group) end diff --git a/ee/spec/features/projects/show_project_spec.rb b/ee/spec/features/projects/show_project_spec.rb index 600b5dc669ab5dddcb665fe92307bb70e39a79e4..79316fcd29b5664425019e9104828e3685b30d0f 100644 --- a/ee/spec/features/projects/show_project_spec.rb +++ b/ee/spec/features/projects/show_project_spec.rb @@ -143,4 +143,79 @@ expect(page).to have_selector('[data-testid="settings-project-link"]') end end + + describe 'all seats used alert', :saas, :use_clean_rails_memory_store_caching do + let_it_be(:group) { create(:group) } + let_it_be(:project) { create(:project, namespace: group) } + + before do + group.add_member(create(:user), GroupMember::DEVELOPER) + sign_in(user) + end + + context 'when all seats are used' do + let_it_be(:subscription) { create(:gitlab_subscription, :premium, namespace: group, seats: 1) } + + context 'when the user is an owner' do + before do + stub_billable_members_reactive_cache(group) + + group.add_owner(user) + end + + it 'displays the all seats used alert' do + visit project_path(project) + + expect(page).to have_css '[data-testid="all-seats-used-alert"].gl-alert-warning' + + within_testid('all-seats-used-alert') do + expect(page).to have_css('[data-testid="close-icon"]') + expect(page).to have_text "No more seats in subscription" + expect(page).to have_text "Your namespace has used all the seats in your subscription and users can " \ + "no longer be invited or added to the namespace." + expect(page).to have_link 'Purchase more seats', href: + help_page_path('subscriptions/gitlab_com/index', anchor: 'add-seats-to-your-subscription') + end + end + end + + context 'when the user is not an owner' do + let(:role) { :developer } + + it 'does not display the all seats used alert' do + visit project_path(project) + + expect(page).not_to have_css '[data-testid="all-seats-used-alert"].gl-alert-warning' + end + end + end + + context 'with a free plan' do + let_it_be(:subscription) { create(:gitlab_subscription, :free, namespace: group, seats: 1) } + + before do + stub_billable_members_reactive_cache(group) + end + + it 'does not display the all seats used alert' do + visit project_path(project) + + expect(page).not_to have_css '[data-testid="all-seats-used-alert"].gl-alert-warning' + end + end + + context 'when not all seats are used' do + let_it_be(:subscription) { create(:gitlab_subscription, :premium, namespace: group, seats: 3) } + + before do + stub_billable_members_reactive_cache(group) + end + + it 'does not display the all seats used alert' do + visit project_path(project) + + expect(page).not_to have_css '[data-testid="all-seats-used-alert"].gl-alert-warning' + end + end + end end diff --git a/locale/gitlab.pot b/locale/gitlab.pot index 71287b27e269e846f0bf39a44a55d3d559510b9f..35c680722c9c555684a58d834c97e507ff8abf15 100644 --- a/locale/gitlab.pot +++ b/locale/gitlab.pot @@ -33989,6 +33989,9 @@ msgstr "" msgid "No milestone" msgstr "" +msgid "No more seats in subscription" +msgstr "" + msgid "No more than %{max_issues} issues can be updated at the same time" msgstr "" @@ -59791,6 +59794,9 @@ msgstr "" msgid "Your name" msgstr "" +msgid "Your namespace has used all the seats in your subscription and users can no longer be invited or added to the namespace." +msgstr "" + msgid "Your namespace storage is full. This merge request cannot be merged. To continue, %{link_start}manage your storage usage%{link_end}." msgstr ""