diff --git a/config/feature_flags/development/cache_issue_sums.yml b/config/feature_flags/development/cache_issue_sums.yml
new file mode 100644
index 0000000000000000000000000000000000000000..9d045eca2c94a909990a0b2d8779f24b3f1000cd
--- /dev/null
+++ b/config/feature_flags/development/cache_issue_sums.yml
@@ -0,0 +1,8 @@
+---
+name: cache_issue_sums
+introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/95048
+rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/365940
+milestone: '15.3'
+type: development
+group: group::product planning
+default_enabled: false
diff --git a/ee/app/models/ee/epic.rb b/ee/app/models/ee/epic.rb
index 7d4d6bd3536fc5c1e7645e19c49f21ee8819f1f1..b83635544c22532c7c059f5e9f333fdae7a5e408 100644
--- a/ee/app/models/ee/epic.rb
+++ b/ee/app/models/ee/epic.rb
@@ -176,6 +176,7 @@ def etag_caching_enabled?
       before_save :set_fixed_start_date, if: :start_date_is_fixed?
       before_save :set_fixed_due_date, if: :due_date_is_fixed?
       after_create_commit :usage_ping_record_epic_creation
+      after_commit :update_cached_metadata
 
       def epic_tree_root?
         parent_id.nil?
@@ -613,5 +614,75 @@ def blocked_by_epics_for(user)
       unfiltered_epics = self.class.where(id: blocking_epics_ids)
       self.class.epics_readable_by_user(unfiltered_epics, user)
     end
+
+    def total_issue_weight_and_count
+      subepic_sums = subepics_weight_and_count
+      issue_sums = issues_weight_and_count
+
+      {
+        total_opened_issue_weight: subepic_sums[:opened_issue_weight] + issue_sums[:opened_issue_weight],
+        total_closed_issue_weight: subepic_sums[:closed_issue_weight] + issue_sums[:closed_issue_weight],
+        total_opened_issue_count: subepic_sums[:opened_issue_count] + issue_sums[:opened_issue_count],
+        total_closed_issue_count: subepic_sums[:closed_issue_count] + issue_sums[:closed_issue_count]
+      }
+    end
+
+    def subepics_weight_and_count
+      sum = children.select(
+        'SUM(total_opened_issue_weight) AS opened_issue_weight',
+        'SUM(total_closed_issue_weight) AS closed_issue_weight',
+        'SUM(total_opened_issue_count) AS opened_issue_count',
+        'SUM(total_closed_issue_count) AS closed_issue_count'
+      )[0]
+
+      {
+        opened_issue_weight: sum.opened_issue_weight.to_i,
+        closed_issue_weight: sum.closed_issue_weight.to_i,
+        opened_issue_count: sum.opened_issue_count.to_i,
+        closed_issue_count: sum.closed_issue_count.to_i
+      }
+    end
+
+    def issues_weight_and_count
+      state_sums = issues
+        .select('issues.state_id AS issues_state_id',
+                'SUM(COALESCE(issues.weight, 0)) AS issues_weight_sum',
+                'COUNT(issues.id) AS issues_count')
+        .reorder(nil)
+        .group("issues.state_id")
+
+      by_state = state_sums.each_with_object({}) do |state_sum, result|
+        key = ::Issue.available_states.key(state_sum.issues_state_id)
+        result[key] = state_sum
+      end
+
+      {
+        opened_issue_weight: by_state['opened']&.issues_weight_sum.to_i,
+        closed_issue_weight: by_state['closed']&.issues_weight_sum.to_i,
+        opened_issue_count: by_state['opened']&.issues_count.to_i,
+        closed_issue_count: by_state['closed']&.issues_count.to_i
+      }
+    end
+
+    def propagate_issue_metadata_change?
+      return false unless parent_id
+      return true if destroyed?
+
+      attrs = %w[total_opened_issue_weight total_closed_issue_weight
+                 total_opened_issue_count total_closed_issue_count
+                 parent_id]
+
+      (previous_changes.keys & attrs).any?
+    end
+
+    def update_cached_metadata
+      return unless ::Feature.enabled?(:cache_issue_sums)
+
+      ::Epics::UpdateCachedMetadataWorker.perform_async([parent_id]) if propagate_issue_metadata_change?
+
+      if parent_id_previously_changed? && parent_id_previously_was
+        ::Epics::UpdateCachedMetadataWorker.perform_async([parent_id_previously_was])
+      end
+    end
   end
 end
diff --git a/ee/app/models/ee/issue.rb b/ee/app/models/ee/issue.rb
index 8de14ed9fc535d3704f7adfddaff71de0a3bd272..8e1bd31ca3434e42184f19881d7c9218ce461f33 100644
--- a/ee/app/models/ee/issue.rb
+++ b/ee/app/models/ee/issue.rb
@@ -138,6 +138,8 @@ module Issue
 
         ::Dora::DailyMetrics::RefreshWorker.perform_async(related_production_env.id, issue.created_at.to_date.to_s)
       end
+
+      after_commit :update_cached_metadata
     end
 
     class_methods do
@@ -337,5 +339,13 @@ def confidentiality_error
 
       _('This issue cannot be assigned to a confidential epic because it is public.')
     end
+
+    def update_cached_metadata
+      return unless ::Feature.enabled?(:cache_issue_sums)
+      return unless epic_issue
+      return unless weight_previously_changed? || state_id_previously_changed?
+
+      ::Epics::UpdateCachedMetadataWorker.perform_async([epic_issue.epic_id])
+    end
   end
 end
diff --git a/ee/app/models/epic_issue.rb b/ee/app/models/epic_issue.rb
index 6f97629a48c929f0c6c2d1e8b687f88011c7f1f0..20eaa0ebeb3a575632d1bdd1ecaadfc47024f854 100644
--- a/ee/app/models/epic_issue.rb
+++ b/ee/app/models/epic_issue.rb
@@ -18,6 +18,7 @@ class EpicIssue < ApplicationRecord
 
   validate :validate_confidential_epic
   validate :validate_group_structure
+  after_commit :update_cached_metadata
 
   def epic_tree_root?
     false
@@ -48,4 +49,14 @@ def validate_confidential_epic
       errors.add :issue, _('Cannot assign a confidential epic to a non-confidential issue. Make the issue confidential and try again')
     end
   end
+
+  def update_cached_metadata
+    return unless ::Feature.enabled?(:cache_issue_sums)
+
+    ::Epics::UpdateCachedMetadataWorker.perform_async([epic_id])
+
+    if epic_id_previously_changed? && epic_id_previously_was
+      ::Epics::UpdateCachedMetadataWorker.perform_async([epic_id_previously_was])
+    end
+  end
 end
diff --git a/ee/app/workers/all_queues.yml b/ee/app/workers/all_queues.yml
index 77eee3611c683c5a2e7307b1478eb5d25af82a69..f994d0206779b2f38040901ee0a1360c904ee22a 100644
--- a/ee/app/workers/all_queues.yml
+++ b/ee/app/workers/all_queues.yml
@@ -507,6 +507,15 @@
   :weight: 1
   :idempotent: true
   :tags: []
+- :name: epics:epics_update_cached_metadata
+  :worker_name: Epics::UpdateCachedMetadataWorker
+  :feature_category: :portfolio_management
+  :has_external_dependencies: false
+  :urgency: :low
+  :resource_boundary: :unknown
+  :weight: 2
+  :idempotent: true
+  :tags: []
 - :name: epics:epics_update_epics_dates
   :worker_name: Epics::UpdateEpicsDatesWorker
   :feature_category: :portfolio_management
diff --git a/ee/app/workers/epics/update_cached_metadata_worker.rb b/ee/app/workers/epics/update_cached_metadata_worker.rb
new file mode 100644
index 0000000000000000000000000000000000000000..ac9e9458c7c474765949e27427420ec48c5921c2
--- /dev/null
+++ b/ee/app/workers/epics/update_cached_metadata_worker.rb
@@ -0,0 +1,30 @@
+# frozen_string_literal: true
+
+module Epics
+  class UpdateCachedMetadataWorker
+    include ApplicationWorker
+
+    BATCH_SIZE = 100
+
+    data_consistency :delayed
+    idempotent!
+    queue_namespace :epics
+    feature_category :portfolio_management
+
+    def perform(ids)
+      return unless Feature.enabled?(:cache_issue_sums)
+
+      ::Epic.id_in(ids).find_each do |epic|
+        update_epic(epic)
+      end
+    end
+
+    private
+
+    def update_epic(epic)
+      total_sums = epic.total_issue_weight_and_count
+      epic.assign_attributes(total_sums)
+      epic.save!(touch: false)
+    end
+  end
+end
diff --git a/ee/spec/models/epic_issue_spec.rb b/ee/spec/models/epic_issue_spec.rb
index e80af54d3e4e44526d1b35341e6a997617f445a0..28804c2468452c8011d604cc03e0c33f95622017 100644
--- a/ee/spec/models/epic_issue_spec.rb
+++ b/ee/spec/models/epic_issue_spec.rb
@@ -120,4 +120,43 @@ def as_item(item)
       end
     end
   end
+
+  describe '#update_cached_metadata' do
+    it 'schedules cache update for epic when new issue is added' do
+      expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([epic.id]).once
+
+      create(:epic_issue, epic: epic, issue: issue)
+    end
+
+    context 'when epic issue already exists' do
+      let_it_be_with_reload(:epic_issue) { create(:epic_issue, epic: epic, issue: issue) }
+
+      it 'schedules cache update for epic when epic issue is updated' do
+        new_epic = create(:epic, group: group)
+
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([epic.id]).once
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([new_epic.id]).once
+
+        epic_issue.update!(epic: new_epic)
+      end
+
+      it 'schedules cache update for epic when epic issue is destroyed' do
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([epic.id]).once
+
+        epic_issue.destroy!
+      end
+
+      context 'when cache_issue_sums flag is disabled' do
+        before do
+          stub_feature_flags(cache_issue_sums: false)
+        end
+
+        it 'does nothing' do
+          expect(::Epics::UpdateCachedMetadataWorker).not_to receive(:perform_async)
+
+          epic_issue.destroy!
+        end
+      end
+    end
+  end
 end
diff --git a/ee/spec/models/epic_spec.rb b/ee/spec/models/epic_spec.rb
index aad1a4507e6198da3fee05e11cfa2d0160820a19..c30e84718c3d24ec9c1989f36dff18501b6ec9ce 100644
--- a/ee/spec/models/epic_spec.rb
+++ b/ee/spec/models/epic_spec.rb
@@ -1143,4 +1143,87 @@ def as_item(item)
       let(:params) { { noteable: issuable } }
     end
   end
+
+  describe '#total_issue_weight_and_count' do
+    let_it_be(:group) { create(:group) }
+    let_it_be(:project) { create(:project, group: group) }
+    let_it_be(:epic) { create(:epic, group: group) }
+    let_it_be(:issue) { create(:issue, weight: 20, project: project) }
+    let_it_be(:issue2) { create(:issue, :closed, weight: 30, project: project) }
+    let_it_be(:epic_issue) { create(:epic_issue, epic: epic, issue: issue) }
+    let_it_be(:epic_issue2) { create(:epic_issue, epic: epic, issue: issue2) }
+
+    let_it_be(:subepic) do
+      create(:epic, parent: epic, group: group, total_opened_issue_weight: 10,
+                    total_closed_issue_weight: 20, total_opened_issue_count: 2,
+                    total_closed_issue_count: 3)
+    end
+
+    it 'returns hash of total issue weight and count including its subepics' do
+      expect(epic.total_issue_weight_and_count).to eq(
+        {
+          total_opened_issue_weight: 30,
+          total_closed_issue_weight: 50,
+          total_opened_issue_count: 3,
+          total_closed_issue_count: 4
+        }
+      )
+    end
+  end
+
+  describe '#update_cached_metadata' do
+    let_it_be(:parent_epic) { create(:epic, group: group) }
+
+    it 'schedules cache update for parent epic when new subepic is created' do
+      expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([parent_epic.id]).once
+
+      create(:epic, parent: parent_epic, group: group)
+    end
+
+    context 'when adding existing subepic' do
+      let_it_be_with_reload(:subepic) { create(:epic, group: group) }
+
+      it 'schedules cache update for parent epic' do
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([parent_epic.id]).once
+
+        subepic.update!(parent: parent_epic)
+      end
+    end
+
+    context 'when epic is already assigned to other epic' do
+      let_it_be(:old_parent) { create(:epic, group: group) }
+      let_it_be_with_reload(:subepic) { create(:epic, group: group, parent: old_parent) }
+
+      it 'schedules cache update for old parent and new parent epics' do
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([parent_epic.id]).once
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([old_parent.id]).once
+
+        subepic.update!(parent: parent_epic)
+      end
+
+      it 'schedules cache update for parent epic when removing subepic parent' do
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([old_parent.id]).once
+
+        subepic.update!(parent: nil)
+      end
+
+      it 'schedules cache update for parent epic when subepic is destroyed' do
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([old_parent.id]).once
+
+        subepic.destroy!
+      end
+
+      context 'when cache_issue_sums flag is disabled' do
+        before do
+          stub_feature_flags(cache_issue_sums: false)
+        end
+
+        it 'does nothing' do
+          expect(::Epics::UpdateCachedMetadataWorker).not_to receive(:perform_async)
+
+          subepic.destroy!
+        end
+      end
+    end
+  end
 end
diff --git a/ee/spec/models/issue_spec.rb b/ee/spec/models/issue_spec.rb
index bddeea6f0e3f88a01c6349075b33d25731d19a7a..71ada60898a91574a333d402240b45f47b07c4b8 100644
--- a/ee/spec/models/issue_spec.rb
+++ b/ee/spec/models/issue_spec.rb
@@ -1188,4 +1188,74 @@
         .to contain_exactly(issue, test_case)
     end
   end
+
+  describe '#update_cached_metadata' do
+    let_it_be(:group) { create(:group) }
+    let_it_be(:project) { create(:project, group: group) }
+    let_it_be(:epic) { create(:epic, group: group) }
+
+    before do
+      stub_licensed_features(epics: true)
+    end
+
+    context 'when epic is not assigned' do
+      let(:issue) { build(:issue, project: project) }
+
+      it 'does nothing' do
+        expect(::Epics::UpdateCachedMetadataWorker).not_to receive(:perform_async)
+
+        issue.save!
+      end
+    end
+
+    context 'when creating new issue' do
+      let(:issue) { build(:issue, project: project, epic: epic) }
+
+      it 'schedules cache update for epic' do
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([epic.id]).once
+
+        issue.save!
+      end
+
+      context 'when cache_issue_sums flag is disabled' do
+        before do
+          stub_feature_flags(cache_issue_sums: false)
+        end
+
+        it 'does nothing' do
+          expect(::Epics::UpdateCachedMetadataWorker).not_to receive(:perform_async)
+
+          issue.save!
+        end
+      end
+    end
+
+    context 'when updating an existing issue' do
+      let_it_be(:issue) { create(:issue, project: project, epic: epic) }
+
+      it 'schedules cache update for epic if state is changed' do
+        issue.state = :closed
+
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([epic.id]).once
+
+        issue.save!
+      end
+
+      it 'schedules cache update for epic if weight is changed' do
+        issue.weight = 2
+
+        expect(::Epics::UpdateCachedMetadataWorker).to receive(:perform_async).with([epic.id]).once
+
+        issue.save!
+      end
+
+      it 'does nothing when unrelated attributes are changed' do
+        issue.title = 'new title'
+
+        expect(::Epics::UpdateCachedMetadataWorker).not_to receive(:perform_async)
+
+        issue.save!
+      end
+    end
+  end
 end
diff --git a/ee/spec/services/ee/issues/update_service_spec.rb b/ee/spec/services/ee/issues/update_service_spec.rb
index c5eff9d146e26b477f0b146c98c66c66d002abb1..7ea85a8d642108714980a744724e17816fcfadc2 100644
--- a/ee/spec/services/ee/issues/update_service_spec.rb
+++ b/ee/spec/services/ee/issues/update_service_spec.rb
@@ -317,6 +317,10 @@ def update_issue(opts)
               let(:property) { Gitlab::UsageDataCounters::IssueActivityUniqueCounter::ISSUE_ADDED_TO_EPIC }
             end
           end
+
+          it 'updates epic cache counts', :sidekiq_inline do
+            expect { subject }.to change { epic.reload.total_opened_issue_count }.by(1)
+          end
         end
 
         context 'when issue belongs to another epic' do
@@ -352,6 +356,11 @@ def update_issue(opts)
               let(:property) { Gitlab::UsageDataCounters::IssueActivityUniqueCounter::ISSUE_CHANGED_EPIC }
             end
           end
+
+          it 'updates epic cache counts', :sidekiq_inline do
+            expect { subject }.to change { epic.reload.total_opened_issue_count }.by(1)
+              .and(change { epic2.reload.total_opened_issue_count }.by(-1))
+          end
         end
 
         context 'when updating issue epic and milestone and assignee attributes' do
diff --git a/ee/spec/workers/epics/update_cached_metadata_worker_spec.rb b/ee/spec/workers/epics/update_cached_metadata_worker_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..bad53318d30cc0b2892e0aea5216b748e652e6ce
--- /dev/null
+++ b/ee/spec/workers/epics/update_cached_metadata_worker_spec.rb
@@ -0,0 +1,91 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Epics::UpdateCachedMetadataWorker do
+  describe '#perform', :sidekiq_inline do
+    let_it_be(:group) { create(:group) }
+    let_it_be(:project) { create(:project, group: group) }
+    let_it_be_with_reload(:parent_epic) { create(:epic, group: group) }
+    let_it_be_with_reload(:epic) { create(:epic, parent: parent_epic, group: group) }
+    let_it_be_with_reload(:other_epic) { create(:epic) }
+    let_it_be(:issue1) { create(:issue, weight: 20, project: project) }
+    let_it_be(:issue2) { create(:issue, weight: 30, project: project) }
+    let_it_be(:issue3) { create(:issue, :closed, weight: 10, project: project) }
+    let_it_be(:epic_issue1) { create(:epic_issue, epic: epic, issue: issue1) }
+    let_it_be(:epic_issue2) { create(:epic_issue, epic: parent_epic, issue: issue2) }
+    let_it_be(:epic_issue3) { create(:epic_issue, epic: epic, issue: issue3) }
+
+    let(:epic_ids) { [epic.id] }
+    let(:worker) { described_class.new }
+
+    subject(:perform) { worker.perform(epic_ids) }
+
+    shared_examples_for 'successful metadata update' do
+      it 'updates epic issue cached metadata and changes are propagated to ancestors', :aggregate_failures do
+        expect(epic.total_opened_issue_weight).to eq(0)
+        expect(epic.total_closed_issue_weight).to eq(0)
+        expect(epic.total_opened_issue_count).to eq(0)
+        expect(epic.total_closed_issue_count).to eq(0)
+        expect(parent_epic.total_opened_issue_weight).to eq(0)
+        expect(parent_epic.total_closed_issue_weight).to eq(0)
+        expect(parent_epic.total_opened_issue_count).to eq(0)
+        expect(parent_epic.total_closed_issue_count).to eq(0)
+
+        subject
+
+        epic.reload
+        parent_epic.reload
+        expect(epic.total_opened_issue_weight).to eq(20)
+        expect(epic.total_closed_issue_weight).to eq(10)
+        expect(epic.total_opened_issue_count).to eq(1)
+        expect(epic.total_closed_issue_count).to eq(1)
+        expect(parent_epic.total_opened_issue_weight).to eq(50)
+        expect(parent_epic.total_closed_issue_weight).to eq(10)
+        expect(parent_epic.total_opened_issue_count).to eq(2)
+        expect(parent_epic.total_closed_issue_count).to eq(1)
+      end
+    end
+
+    it_behaves_like 'successful metadata update'
+
+    include_examples 'an idempotent worker' do
+      let(:job_args) { epic_ids }
+
+      it_behaves_like 'successful metadata update'
+    end
+
+    context 'when cache_issue_sums flag is disabled' do
+      before do
+        stub_feature_flags(cache_issue_sums: false)
+      end
+
+      it 'does nothing' do
+        expect(worker).not_to receive(:update_epic)
+
+        perform
+      end
+    end
+
+    context 'when epic id not found' do
+      let(:epic_ids) { [non_existing_record_id] }
+
+      it 'does nothing' do
+        expect(worker).not_to receive(:update_epic)
+
+        perform
+      end
+    end
+
+    context 'when multiple epic ids are passed' do
+      let(:epic_ids) { [epic.id, other_epic.id] }
+
+      it 'updates epic issue cached metadata for each epic' do
+        expect(worker).to receive(:update_epic).with(epic)
+        expect(worker).to receive(:update_epic).with(other_epic)
+
+        perform
+      end
+    end
+  end
+end