diff --git a/ee/app/models/gitlab_subscription.rb b/ee/app/models/gitlab_subscription.rb
index 595a465abf5f24909c8d155e1b53f90c2189d9c8..d40aa6e41f1971836ffefc84e0d2271ccc3a1e92 100644
--- a/ee/app/models/gitlab_subscription.rb
+++ b/ee/app/models/gitlab_subscription.rb
@@ -16,6 +16,9 @@ class GitlabSubscription < ApplicationRecord
 
   after_save :set_prevent_sharing_groups_outside_hierarchy
 
+  # Needs to run after_commit because workers can be spawned that can't be run within a transaction
+  after_commit :activate_users_post_subscription_upgrade
+
   after_commit :index_namespace, on: [:create, :update]
   after_destroy_commit :log_previous_state_for_destroy
 
@@ -214,4 +217,15 @@ def set_prevent_sharing_groups_outside_hierarchy
     # going from paid to free - prevent
     namespace.update_attribute(:prevent_sharing_groups_outside_hierarchy, prevent_sharing)
   end
+
+  # When free groups where `free_user_cap` was enabled upgrade to a paid
+  # plan we want to activate all the memberships that got set to
+  # awaiting due to the limit.
+  def activate_users_post_subscription_upgrade
+    # rubocop: disable CodeReuse/ServiceClass
+    GitlabSubscriptions::ActivateAwaitingUsersService
+      .new(gitlab_subscription: self, previous_plan_id: hosted_plan_id_before_last_save)
+      .execute
+    # rubocop: enable CodeReuse/ServiceClass
+  end
 end
diff --git a/ee/app/services/gitlab_subscriptions/activate_awaiting_users_service.rb b/ee/app/services/gitlab_subscriptions/activate_awaiting_users_service.rb
new file mode 100644
index 0000000000000000000000000000000000000000..c7f4dcff93ab85edcba71167a19d7b4a2616b8cf
--- /dev/null
+++ b/ee/app/services/gitlab_subscriptions/activate_awaiting_users_service.rb
@@ -0,0 +1,41 @@
+# frozen_string_literal: true
+
+module GitlabSubscriptions
+  class ActivateAwaitingUsersService
+    def initialize(gitlab_subscription:, previous_plan_id:)
+      @gitlab_subscription = gitlab_subscription
+      @previous_plan_id = previous_plan_id
+    end
+
+    def execute
+      return unless previous_plan_id
+      return unless namespace.group_namespace?
+      return unless ::Namespaces::FreeUserCap::Standard.new(namespace).feature_enabled?
+      return unless upgrade_from_free_to_paid?
+
+      activate
+    end
+
+    private
+
+    attr_reader :gitlab_subscription, :previous_plan_id
+
+    def namespace
+      @namespace ||= gitlab_subscription.namespace
+    end
+
+    def upgrade_from_free_to_paid?
+      !Plan.find(previous_plan_id).paid? && Plan.find(gitlab_subscription.hosted_plan_id).paid?
+    end
+
+    def activate
+      awaiting_user_ids = namespace.awaiting_user_ids
+      return if awaiting_user_ids.empty?
+      return if awaiting_user_ids.count > gitlab_subscription.seats_remaining
+
+      ::Members::ActivateService
+        .for_users(namespace, users: awaiting_user_ids)
+        .execute(current_user: User.automation_bot, skip_authorization: true)
+    end
+  end
+end
diff --git a/ee/app/services/members/activate_service.rb b/ee/app/services/members/activate_service.rb
index 9cc70d3cf0af1b61bff0947e76a785e70cf7bc26..34b69c9686c6b455c8ab5f3989d45ca9b42d38a3 100644
--- a/ee/app/services/members/activate_service.rb
+++ b/ee/app/services/members/activate_service.rb
@@ -39,8 +39,9 @@ def initialize(group, memberships:)
 
     private_class_method :new
 
-    def execute(current_user:)
+    def execute(current_user:, skip_authorization: false)
       @current_user = current_user
+      @skip_authorization = skip_authorization
 
       return error(_('You do not have permission to approve a member'), :forbidden) unless allowed?
       return error(_('There is no seat left to activate the member')) unless has_capacity_left?
@@ -56,7 +57,7 @@ def execute(current_user:)
 
     private
 
-    attr_reader :group, :memberships, :current_user, :affected_members
+    attr_reader :group, :memberships, :current_user, :affected_members, :skip_authorization
 
     def activate_memberships
       @affected_members = memberships.to_a
@@ -65,6 +66,8 @@ def activate_memberships
     end
 
     def allowed?
+      return true if skip_authorization
+
       can?(current_user, :admin_group_member, group)
     end
 
diff --git a/ee/spec/models/gitlab_subscription_spec.rb b/ee/spec/models/gitlab_subscription_spec.rb
index 7b2c1d1b95b4cac9398066cd7092c9e992faea65..c51f17edb15de15d8a8848fac4185dbfb7c03834 100644
--- a/ee/spec/models/gitlab_subscription_spec.rb
+++ b/ee/spec/models/gitlab_subscription_spec.rb
@@ -420,93 +420,111 @@
   end
 
   describe 'callbacks' do
-    context 'after_commit :index_namespace', :saas do
-      let_it_be(:namespace) { create(:namespace) }
-
-      let(:gitlab_subscription) { build(:gitlab_subscription, plan, namespace: namespace) }
-      let(:expiration_date) { Date.today + 10 }
-      let(:plan) { :bronze }
+    context 'after_commit', :saas do
+      context 'activate_users_post_subscription_upgrade' do
+        let_it_be(:gitlab_subscription) { create(:gitlab_subscription, hosted_plan: free_plan) }
+
+        it 'calls ActivateAwaitingUsersService' do
+          expect_next_instance_of(
+            GitlabSubscriptions::ActivateAwaitingUsersService,
+            gitlab_subscription: gitlab_subscription,
+            previous_plan_id: free_plan.id
+          ) do |instance|
+            expect(instance).to receive(:execute)
+          end
 
-      before do
-        gitlab_subscription.end_date = expiration_date
+          gitlab_subscription.update!(hosted_plan: ultimate_plan)
+        end
       end
 
-      it 'indexes the namespace' do
-        expect(ElasticsearchIndexedNamespace).to receive(:safe_find_or_create_by!).with(namespace_id: gitlab_subscription.namespace_id)
+      context 'index_namespace' do
+        let_it_be(:namespace) { create(:namespace) }
 
-        gitlab_subscription.save!
-      end
+        let(:gitlab_subscription) { build(:gitlab_subscription, plan, namespace: namespace) }
+        let(:expiration_date) { Date.today + 10 }
+        let(:plan) { :bronze }
 
-      context 'when seats is 0' do
-        let(:gitlab_subscription) { build(:gitlab_subscription, namespace: namespace, seats: 0) }
+        before do
+          gitlab_subscription.end_date = expiration_date
+        end
 
-        it 'does not index the namespace' do
-          expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
+        it 'indexes the namespace' do
+          expect(ElasticsearchIndexedNamespace).to receive(:safe_find_or_create_by!).with(namespace_id: gitlab_subscription.namespace_id)
 
           gitlab_subscription.save!
         end
-      end
 
-      context 'when it is a trial' do
-        let(:seats) { 10 }
-        let(:gitlab_subscription) { build(:gitlab_subscription, :active_trial, namespace: namespace, seats: seats) }
+        context 'when seats is 0' do
+          let(:gitlab_subscription) { build(:gitlab_subscription, namespace: namespace, seats: 0) }
 
-        it 'indexes the namespace' do
-          expect(ElasticsearchIndexedNamespace).to receive(:safe_find_or_create_by!).with(namespace_id: gitlab_subscription.namespace_id)
+          it 'does not index the namespace' do
+            expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
 
-          gitlab_subscription.save!
+            gitlab_subscription.save!
+          end
         end
 
-        context 'when seats is zero' do
-          let(:seats) { 0 }
+        context 'when it is a trial' do
+          let(:seats) { 10 }
+          let(:gitlab_subscription) { build(:gitlab_subscription, :active_trial, namespace: namespace, seats: seats) }
 
           it 'indexes the namespace' do
             expect(ElasticsearchIndexedNamespace).to receive(:safe_find_or_create_by!).with(namespace_id: gitlab_subscription.namespace_id)
 
             gitlab_subscription.save!
           end
-        end
 
-        context 'when in free plan' do
-          let(:gitlab_subscription) { build(:gitlab_subscription, :active_trial, namespace: namespace, seats: seats, hosted_plan_id: nil) }
+          context 'when seats is zero' do
+            let(:seats) { 0 }
 
-          it 'does not index the namespace' do
-            expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
+            it 'indexes the namespace' do
+              expect(ElasticsearchIndexedNamespace).to receive(:safe_find_or_create_by!).with(namespace_id: gitlab_subscription.namespace_id)
 
-            gitlab_subscription.save!
+              gitlab_subscription.save!
+            end
           end
-        end
-      end
 
-      context 'when not ::Gitlab.com?' do
-        before do
-          allow(::Gitlab).to receive(:com?).and_return(false)
+          context 'when in free plan' do
+            let(:gitlab_subscription) { build(:gitlab_subscription, :active_trial, namespace: namespace, seats: seats, hosted_plan_id: nil) }
+
+            it 'does not index the namespace' do
+              expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
+
+              gitlab_subscription.save!
+            end
+          end
         end
 
-        it 'does not index the namespace' do
-          expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
+        context 'when not ::Gitlab.com?' do
+          before do
+            allow(::Gitlab).to receive(:com?).and_return(false)
+          end
+
+          it 'does not index the namespace' do
+            expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
 
-          gitlab_subscription.save!
+            gitlab_subscription.save!
+          end
         end
-      end
 
-      context 'when the plan has expired' do
-        let(:expiration_date) { Date.today - 8.days }
+        context 'when the plan has expired' do
+          let(:expiration_date) { Date.today - 8.days }
 
-        it 'does not index the namespace' do
-          expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
+          it 'does not index the namespace' do
+            expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
 
-          gitlab_subscription.save!
+            gitlab_subscription.save!
+          end
         end
-      end
 
-      context 'when it is a free plan' do
-        let(:plan) { :free }
+        context 'when it is a free plan' do
+          let(:plan) { :free }
 
-        it 'does not index the namespace' do
-          expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
+          it 'does not index the namespace' do
+            expect(ElasticsearchIndexedNamespace).not_to receive(:safe_find_or_create_by!)
 
-          gitlab_subscription.save!
+            gitlab_subscription.save!
+          end
         end
       end
     end
diff --git a/ee/spec/services/gitlab_subscriptions/activate_awaiting_users_service_spec.rb b/ee/spec/services/gitlab_subscriptions/activate_awaiting_users_service_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..6f18d91a0742c025b8fab45019ac3c9708b63b5c
--- /dev/null
+++ b/ee/spec/services/gitlab_subscriptions/activate_awaiting_users_service_spec.rb
@@ -0,0 +1,110 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe GitlabSubscriptions::ActivateAwaitingUsersService, :saas do
+  shared_examples 'skips user activation' do
+    it 'does not activate users' do
+      execute
+
+      expect(member1.reload).to be_awaiting
+      expect(member2.reload).to be_awaiting
+    end
+  end
+
+  describe '#execute' do
+    let_it_be(:namespace) { create(:group) }
+    let_it_be(:free_plan) { create(:free_plan) }
+    let_it_be(:ultimate_plan) { create(:ultimate_plan) }
+    let_it_be(:user1) { create(:user) }
+    let_it_be(:user2) { create(:user) }
+    let_it_be(:member1) { create(:group_member, :awaiting, user: user1, source: namespace) }
+    let_it_be(:member2) { create(:group_member, :awaiting, user: user2, source: namespace) }
+
+    let(:current_plan) { free_plan }
+    let(:free_user_cap_enabled) { true }
+    let(:seats_remaining) { 2 }
+    let(:gitlab_subscription) do
+      create(:gitlab_subscription, hosted_plan: current_plan, seats: seats_remaining, namespace: namespace)
+    end
+
+    subject(:execute) do
+      described_class.new(gitlab_subscription: gitlab_subscription, previous_plan_id: previous_plan&.id).execute
+    end
+
+    before do
+      allow_next_instance_of(::Namespaces::FreeUserCap::Standard, namespace) do |instance|
+        allow(instance).to receive(:feature_enabled?).and_return(free_user_cap_enabled)
+      end
+    end
+
+    context 'without a previous plan' do
+      let(:previous_plan) { nil }
+
+      context 'when switching to a free plan' do
+        let_it_be(:current_plan) { free_plan }
+
+        it_behaves_like 'skips user activation'
+      end
+
+      context 'when switching to a paid plan' do
+        let_it_be(:current_plan) { ultimate_plan }
+
+        it_behaves_like 'skips user activation'
+      end
+    end
+
+    context 'with a previous free plan' do
+      let(:previous_plan) { free_plan }
+
+      context 'when switching to a paid plan' do
+        let_it_be_with_reload(:current_plan) { ultimate_plan }
+
+        context 'when not a group namespace' do
+          let_it_be(:namespace) { create(:namespace) }
+
+          it_behaves_like 'skips user activation'
+        end
+
+        context 'when there are enough seats remaining' do
+          it 'activates users' do
+            execute
+
+            expect(member1.reload).to be_active
+            expect(member2.reload).to be_active
+          end
+
+          it 'audits the event using the automation bot user as author' do
+            execute
+
+            audit_event = AuditEvent.find_by(target_id: member1.id)
+            expect(audit_event.author).to eq(User.automation_bot)
+            expect(audit_event.details[:custom_message]).to eq('Changed the membership state to active')
+          end
+
+          context 'when free user cap is not enabled' do
+            let(:free_user_cap_enabled) { false }
+
+            it_behaves_like 'skips user activation'
+          end
+        end
+
+        context 'when there are not enough seats remaining' do
+          let(:seats_remaining) { 0 }
+
+          it_behaves_like 'skips user activation'
+        end
+      end
+    end
+
+    context 'with a previous paid plan' do
+      let(:previous_plan) { ultimate_plan }
+
+      context 'changes from paid plan to free plan' do
+        let(:current_plan) { free_plan }
+
+        it_behaves_like 'skips user activation'
+      end
+    end
+  end
+end
diff --git a/ee/spec/services/members/activate_service_spec.rb b/ee/spec/services/members/activate_service_spec.rb
index 2e754583ab8ecae2b3b0da4c05a9d575c0ddad0e..3615c52529f75ec36960bf0c9139ff36e15c8d83 100644
--- a/ee/spec/services/members/activate_service_spec.rb
+++ b/ee/spec/services/members/activate_service_spec.rb
@@ -252,6 +252,22 @@
       it_behaves_like 'returns an error', 'You do not have permission to approve a member'
     end
 
+    context 'when current_user is nil' do
+      let_it_be(:current_user) { nil }
+
+      it_behaves_like 'returns an error', 'You do not have permission to approve a member'
+    end
+
+    context 'when skipping authorization' do
+      let_it_be(:current_user) { User.automation_bot }
+
+      let_it_be(:members) { [create(:group_member, :awaiting, group: root_group, user: user)] }
+
+      subject(:execute) { described_class.for_group(root_group).execute(current_user: current_user, skip_authorization: true) }
+
+      it_behaves_like 'successful member activation'
+    end
+
     context 'when authorized' do
       before do
         root_group.add_owner(current_user)