diff --git a/app/models/todo.rb b/app/models/todo.rb
index 5f4fa83ba645aa1844e4d01b0d3fa3e822cedc5e..26911929ecce872cfe3bb726c08018c92507b6f3 100644
--- a/app/models/todo.rb
+++ b/app/models/todo.rb
@@ -68,6 +68,8 @@ class Todo < ApplicationRecord
   validates :group, presence: true, unless: :project_id
 
   scope :pending, -> { with_state(:pending) }
+  scope :snoozed, -> { where(arel_table[:snoozed_until].gt(Time.current)) }
+  scope :not_snoozed, -> { where(arel_table[:snoozed_until].lteq(Time.current)).or(where(snoozed_until: nil)) }
   scope :done, -> { with_state(:done) }
   scope :for_action, ->(action) { where(action: action) }
   scope :for_author, ->(author) { where(author: author) }
diff --git a/db/migrate/20240820125541_add_snoozed_until_to_todos.rb b/db/migrate/20240820125541_add_snoozed_until_to_todos.rb
new file mode 100644
index 0000000000000000000000000000000000000000..c3b591d91dbb35ac0703c7dd872486d7bc2c9666
--- /dev/null
+++ b/db/migrate/20240820125541_add_snoozed_until_to_todos.rb
@@ -0,0 +1,9 @@
+# frozen_string_literal: true
+
+class AddSnoozedUntilToTodos < Gitlab::Database::Migration[2.2]
+  milestone '17.4'
+
+  def change
+    add_column :todos, :snoozed_until, :datetime_with_timezone
+  end
+end
diff --git a/db/schema_migrations/20240820125541 b/db/schema_migrations/20240820125541
new file mode 100644
index 0000000000000000000000000000000000000000..828e350039e2663acbd220113a579ccc45993dba
--- /dev/null
+++ b/db/schema_migrations/20240820125541
@@ -0,0 +1 @@
+acb0240ec26db6e159d68440bec330d045eba9a1fd7738ed6a23b7f34251f2b6
\ No newline at end of file
diff --git a/db/structure.sql b/db/structure.sql
index 6f13524acc12d00f65207da9ea6a4a57a73ae380..6971ba5aebb2e3208e50a005a410c75275a733c7 100644
--- a/db/structure.sql
+++ b/db/structure.sql
@@ -18833,7 +18833,8 @@ CREATE TABLE todos (
     commit_id character varying,
     group_id bigint,
     resolved_by_action smallint,
-    note_id bigint
+    note_id bigint,
+    snoozed_until timestamp with time zone
 );
 
 CREATE SEQUENCE todos_id_seq
diff --git a/spec/models/todo_spec.rb b/spec/models/todo_spec.rb
index a3bead7351c171871a49f6dcef6d0e3057d56b55..4037fe0e071dba7f75b0fbc613ac9902a55c765b 100644
--- a/spec/models/todo_spec.rb
+++ b/spec/models/todo_spec.rb
@@ -700,6 +700,24 @@
     end
   end
 
+  describe 'snoozed and not_snoozed scopes' do
+    let_it_be(:snoozed_todo) { create(:todo, snoozed_until: 1.day.from_now) }
+    let_it_be(:unsnoozed_todo) { create(:todo, snoozed_until: 1.day.ago) }
+    let_it_be(:never_snoozed_todo) { create(:todo, snoozed_until: nil) }
+
+    describe '.snoozed' do
+      it 'only returns todos that are currently snoozed' do
+        expect(described_class.snoozed).to contain_exactly(snoozed_todo)
+      end
+    end
+
+    describe '.not_snoozed' do
+      it 'returns todos that are not snoozed anymore or never were snoozed' do
+        expect(described_class.not_snoozed).to contain_exactly(unsnoozed_todo, never_snoozed_todo)
+      end
+    end
+  end
+
   describe '#access_request_url' do
     shared_examples 'returns member access requests tab url/path' do
       it 'returns group access requests tab url/path if target is group' do