diff --git a/app/models/concerns/saved_reply_concern.rb b/app/models/concerns/saved_reply_concern.rb
new file mode 100644
index 0000000000000000000000000000000000000000..139b1b0c9fa3f528705a2e84579e93d6f13d93c5
--- /dev/null
+++ b/app/models/concerns/saved_reply_concern.rb
@@ -0,0 +1,17 @@
+# frozen_string_literal: true
+
+module SavedReplyConcern
+  extend ActiveSupport::Concern
+
+  included do
+    validates namespace_foreign_key, :name, :content, presence: true
+    validates :content, length: { maximum: 10000 }
+    validates :name,
+      length: { maximum: 255 },
+      uniqueness: { scope: [namespace_foreign_key] }
+
+    def self.find_saved_reply(**args)
+      find_by(args)
+    end
+  end
+end
diff --git a/app/models/users/saved_reply.rb b/app/models/users/saved_reply.rb
index f0ae5445a463ba174aece6d2a0cf868c218a23f8..7108def1250bcf1ec1c1e4da8e5322880c09c8aa 100644
--- a/app/models/users/saved_reply.rb
+++ b/app/models/users/saved_reply.rb
@@ -2,18 +2,13 @@
 
 module Users
   class SavedReply < ApplicationRecord
+    def self.namespace_foreign_key
+      :user_id
+    end
     self.table_name = 'saved_replies'
 
-    belongs_to :user
-
-    validates :user_id, :name, :content, presence: true
-    validates :name,
-      length: { maximum: 255 },
-      uniqueness: { scope: [:user_id] }
-    validates :content, length: { maximum: 10000 }
+    include SavedReplyConcern
 
-    def self.find_saved_reply(user_id:, id:)
-      ::Users::SavedReply.find_by(user_id: user_id, id: id)
-    end
+    belongs_to :user
   end
 end
diff --git a/db/docs/group_saved_replies.yml b/db/docs/group_saved_replies.yml
new file mode 100644
index 0000000000000000000000000000000000000000..bd49c377d83acf7fab36e0741c19125712bdcd66
--- /dev/null
+++ b/db/docs/group_saved_replies.yml
@@ -0,0 +1,12 @@
+---
+table_name: group_saved_replies
+classes:
+- Groups::SavedReply
+feature_categories:
+- code_review_workflow
+description: Comment templates used to populate comments
+introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/143515
+milestone: '16.9'
+gitlab_schema: gitlab_main_cell
+sharding_key:
+  group_id: namespaces
diff --git a/db/migrate/20240201112236_create_group_saved_replies_table.rb b/db/migrate/20240201112236_create_group_saved_replies_table.rb
new file mode 100644
index 0000000000000000000000000000000000000000..f9fdbc1230bc27f50d0eccef49d8f640f21197e5
--- /dev/null
+++ b/db/migrate/20240201112236_create_group_saved_replies_table.rb
@@ -0,0 +1,17 @@
+# frozen_string_literal: true
+
+class CreateGroupSavedRepliesTable < Gitlab::Database::Migration[2.2]
+  enable_lock_retries!
+
+  milestone '16.9'
+
+  def change
+    create_table :group_saved_replies do |t|
+      t.references :group, references: :namespaces, null: false,
+        foreign_key: { to_table: :namespaces, on_delete: :cascade }, index: true
+      t.timestamps_with_timezone null: false
+      t.text :name, null: false, limit: 255
+      t.text :content, null: false, limit: 10000
+    end
+  end
+end
diff --git a/db/schema_migrations/20240201112236 b/db/schema_migrations/20240201112236
new file mode 100644
index 0000000000000000000000000000000000000000..2aa90a4c02b58309e5a99833fd574b39dff8314f
--- /dev/null
+++ b/db/schema_migrations/20240201112236
@@ -0,0 +1 @@
+8e21124a691843445077a9d6bc5541eea99f8b2f3f90e1846429ed3e2f97d72d
\ No newline at end of file
diff --git a/db/structure.sql b/db/structure.sql
index c3e70f72ddfcf61e7491c006bb27075f5adfe775..1ebd38f5bffb56897c78efe254d01bb6c2c4dea7 100644
--- a/db/structure.sql
+++ b/db/structure.sql
@@ -17888,6 +17888,26 @@ CREATE SEQUENCE group_repository_storage_moves_id_seq
 
 ALTER SEQUENCE group_repository_storage_moves_id_seq OWNED BY group_repository_storage_moves.id;
 
+CREATE TABLE group_saved_replies (
+    id bigint NOT NULL,
+    group_id bigint NOT NULL,
+    created_at timestamp with time zone NOT NULL,
+    updated_at timestamp with time zone NOT NULL,
+    name text NOT NULL,
+    content text NOT NULL,
+    CONSTRAINT check_13510378d3 CHECK ((char_length(name) <= 255)),
+    CONSTRAINT check_4a96378d43 CHECK ((char_length(content) <= 10000))
+);
+
+CREATE SEQUENCE group_saved_replies_id_seq
+    START WITH 1
+    INCREMENT BY 1
+    NO MINVALUE
+    NO MAXVALUE
+    CACHE 1;
+
+ALTER SEQUENCE group_saved_replies_id_seq OWNED BY group_saved_replies.id;
+
 CREATE TABLE group_ssh_certificates (
     id bigint NOT NULL,
     namespace_id bigint NOT NULL,
@@ -27417,6 +27437,8 @@ ALTER TABLE ONLY group_import_states ALTER COLUMN group_id SET DEFAULT nextval('
 
 ALTER TABLE ONLY group_repository_storage_moves ALTER COLUMN id SET DEFAULT nextval('group_repository_storage_moves_id_seq'::regclass);
 
+ALTER TABLE ONLY group_saved_replies ALTER COLUMN id SET DEFAULT nextval('group_saved_replies_id_seq'::regclass);
+
 ALTER TABLE ONLY group_ssh_certificates ALTER COLUMN id SET DEFAULT nextval('group_ssh_certificates_id_seq'::regclass);
 
 ALTER TABLE ONLY group_wiki_repository_states ALTER COLUMN id SET DEFAULT nextval('group_wiki_repository_states_id_seq'::regclass);
@@ -29729,6 +29751,9 @@ ALTER TABLE ONLY group_merge_request_approval_settings
 ALTER TABLE ONLY group_repository_storage_moves
     ADD CONSTRAINT group_repository_storage_moves_pkey PRIMARY KEY (id);
 
+ALTER TABLE ONLY group_saved_replies
+    ADD CONSTRAINT group_saved_replies_pkey PRIMARY KEY (id);
+
 ALTER TABLE ONLY group_ssh_certificates
     ADD CONSTRAINT group_ssh_certificates_pkey PRIMARY KEY (id);
 
@@ -33970,6 +33995,8 @@ CREATE INDEX index_group_import_states_on_user_id ON group_import_states USING b
 
 CREATE INDEX index_group_repository_storage_moves_on_group_id ON group_repository_storage_moves USING btree (group_id);
 
+CREATE INDEX index_group_saved_replies_on_group_id ON group_saved_replies USING btree (group_id);
+
 CREATE UNIQUE INDEX index_group_ssh_certificates_on_fingerprint ON group_ssh_certificates USING btree (fingerprint);
 
 CREATE INDEX index_group_ssh_certificates_on_namespace_id ON group_ssh_certificates USING btree (namespace_id);
@@ -40693,6 +40720,9 @@ ALTER TABLE ONLY container_registry_protection_rules
 ALTER TABLE ONLY clusters
     ADD CONSTRAINT fk_rails_ac3a663d79 FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE SET NULL;
 
+ALTER TABLE ONLY group_saved_replies
+    ADD CONSTRAINT fk_rails_acd8e1889b FOREIGN KEY (group_id) REFERENCES namespaces(id) ON DELETE CASCADE;
+
 ALTER TABLE ONLY packages_composer_metadata
     ADD CONSTRAINT fk_rails_ad48c2e5bb FOREIGN KEY (package_id) REFERENCES packages_packages(id) ON DELETE CASCADE;
 
diff --git a/ee/app/models/ee/group.rb b/ee/app/models/ee/group.rb
index 25ef7a77acc97e6b3bac39213ab6c9b1a4e1c095..d4be421d36febfa16335c6a1a86f035842bc5052 100644
--- a/ee/app/models/ee/group.rb
+++ b/ee/app/models/ee/group.rb
@@ -84,6 +84,8 @@ module Group
       belongs_to :push_rule, inverse_of: :group
       has_many :approval_rules, class_name: 'ApprovalRules::ApprovalGroupRule', inverse_of: :group
 
+      has_many :saved_replies, class_name: 'Groups::SavedReply'
+
       delegate :deleting_user, :marked_for_deletion_on, to: :deletion_schedule, allow_nil: true
 
       delegate :repository_read_only,
diff --git a/ee/app/models/groups/saved_reply.rb b/ee/app/models/groups/saved_reply.rb
new file mode 100644
index 0000000000000000000000000000000000000000..f771bcce5e4f1b52d913daf2530b58248aae45e9
--- /dev/null
+++ b/ee/app/models/groups/saved_reply.rb
@@ -0,0 +1,14 @@
+# frozen_string_literal: true
+
+module Groups
+  class SavedReply < ApplicationRecord
+    def self.namespace_foreign_key
+      :group_id
+    end
+    self.table_name = :group_saved_replies
+
+    include SavedReplyConcern
+
+    belongs_to :group
+  end
+end
diff --git a/ee/spec/factories/groups/saved_replies.rb b/ee/spec/factories/groups/saved_replies.rb
new file mode 100644
index 0000000000000000000000000000000000000000..3efc89f6ab1615104667e9a3ad12d55ed9150873
--- /dev/null
+++ b/ee/spec/factories/groups/saved_replies.rb
@@ -0,0 +1,10 @@
+# frozen_string_literal: true
+
+FactoryBot.define do
+  factory :group_saved_reply, class: 'Groups::SavedReply' do
+    sequence(:name) { |n| "saved_reply_#{n}" }
+    content { 'Saved Reply Content' }
+
+    group
+  end
+end
diff --git a/ee/spec/models/groups/saved_reply_spec.rb b/ee/spec/models/groups/saved_reply_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..a0683e75a2931e27507a3f2e5a5c9e0bcd3b76a4
--- /dev/null
+++ b/ee/spec/models/groups/saved_reply_spec.rb
@@ -0,0 +1,16 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Groups::SavedReply, feature_category: :code_review_workflow do
+  let_it_be(:saved_reply) { create(:group_saved_reply) }
+
+  describe 'validations' do
+    it { is_expected.to validate_presence_of(:group_id) }
+    it { is_expected.to validate_presence_of(:name) }
+    it { is_expected.to validate_presence_of(:content) }
+    it { is_expected.to validate_uniqueness_of(:name).scoped_to([:group_id]) }
+    it { is_expected.to validate_length_of(:name).is_at_most(255) }
+    it { is_expected.to validate_length_of(:content).is_at_most(10000) }
+  end
+end