diff --git a/db/migrations/20220210_missing_indices_v50.php b/db/migrations/20220210_missing_indices_v50.php
index 39056fa4402435da964466614844ac595bb85318..66af9c36def5819cc56051731529b7aa8cdc2bd3 100644
--- a/db/migrations/20220210_missing_indices_v50.php
+++ b/db/migrations/20220210_missing_indices_v50.php
@@ -1,6 +1,8 @@
 <?php
 final class MissingIndicesV50 extends Migration
 {
+    use DatabaseMigrationTrait;
+
     public function description()
     {
         return 'Add missing indices on some tables';
@@ -8,6 +10,11 @@ final class MissingIndicesV50 extends Migration
 
     public function up()
     {
+        // avoid running this migration twice
+        if ($this->keyExists('mvv_files_ranges', 'range_id')) {
+            return;
+        }
+
         $query = "CREATE INDEX `range_id` ON `mvv_files_ranges` (`range_id`)";
         DBManager::get()->exec($query);
 
diff --git a/db/migrations/20220301_biest_348.php b/db/migrations/20220301_biest_348.php
index 67f854fc69b18023fb3552f56642dbdfaf938465..dd15560055113676be74e44c9954bf24ac12287c 100644
--- a/db/migrations/20220301_biest_348.php
+++ b/db/migrations/20220301_biest_348.php
@@ -1,6 +1,8 @@
 <?php
 class Biest348 extends Migration
 {
+    use DatabaseMigrationTrait;
+
     public function description ()
     {
         return 'Adds a column to the resources table to mark resources as lockable, default is 1.';
@@ -8,7 +10,7 @@ class Biest348 extends Migration
 
     public function up()
     {
-        if ($this->columnExists()) {
+        if ($this->columnExists('resources', 'lockable')) {
             return;
         }
 
@@ -22,9 +24,4 @@ class Biest348 extends Migration
         $query = 'ALTER TABLE `resources` DROP `lockable`';
         DBManager::get()->exec($query);
     }
-
-    private function columnExists()
-    {
-        return DBManager::get()->fetchFirst("SHOW COLUMNS FROM `resources` LIKE 'lockable'");
-    }
 }
diff --git a/db/migrations/20220401_add_index_to_cw_user_progresses_v50.php b/db/migrations/20220401_add_index_to_cw_user_progresses_v50.php
index fe86201d8d7be964de17e6897c92e860ccaa8a1e..bceb74525597fde634fc9aad726c4490436eec0a 100644
--- a/db/migrations/20220401_add_index_to_cw_user_progresses_v50.php
+++ b/db/migrations/20220401_add_index_to_cw_user_progresses_v50.php
@@ -1,6 +1,8 @@
 <?php
 final class AddIndexToCwUserProgressesV50 extends Migration
 {
+    use DatabaseMigrationTrait;
+
     public function description()
     {
         return 'Alter cw_user_progresses table, add index for block_id';
@@ -9,10 +11,7 @@ final class AddIndexToCwUserProgressesV50 extends Migration
     public function up()
     {
         // avoid running this migration twice
-        $query = "SHOW INDEX FROM `cw_user_progresses` WHERE Key_name = 'block_id'";
-        $result = DBManager::get()->query($query);
-
-        if ($result && $result->rowCount() > 0) {
+        if ($this->keyExists('cw_user_progresses', 'block_id')) {
             return;
         }
 
diff --git a/db/migrations/20220505_add_index_resource_booking_intervals.php b/db/migrations/20220505_add_index_resource_booking_intervals.php
index d819125c38189ea51305fa82385482a6c969b486..e3c655753771f58e396200f98e166d846ae7d4a8 100644
--- a/db/migrations/20220505_add_index_resource_booking_intervals.php
+++ b/db/migrations/20220505_add_index_resource_booking_intervals.php
@@ -2,6 +2,8 @@
 
 class AddIndexResourceBookingIntervals extends Migration
 {
+    use DatabaseMigrationTrait;
+
     public function description()
     {
         return 'add index for booking_id to resource_booking_intervals';
@@ -9,34 +11,24 @@ class AddIndexResourceBookingIntervals extends Migration
 
     public function up()
     {
-        $db = DBManager::get();
-
         // avoid running this migration twice
-        $sql = "SHOW INDEX FROM resource_booking_intervals WHERE Key_name = 'booking_id'";
-        $result = $db->query($sql);
-
-        if ($result && $result->rowCount() > 0) {
+        if ($this->keyExists('resource_booking_intervals', 'booking_id')) {
             return;
         }
 
         // index "assign_object_id" may not exist (depending on upgrade path)
-        $sql = "SHOW INDEX FROM resource_booking_intervals WHERE Key_name = 'assign_object_id'";
-        $result = $db->query($sql);
-
-        if ($result && $result->rowCount() > 0) {
-            $sql = 'ALTER TABLE resource_booking_intervals DROP INDEX assign_object_id';
-            $db->exec($sql);
+        if ($this->keyExists('resource_booking_intervals', 'assign_object_id')) {
+            $sql = "ALTER TABLE resource_booking_intervals DROP INDEX assign_object_id";
+            DBManager::get()->exec($sql);
         }
 
-        $sql = 'ALTER TABLE resource_booking_intervals ADD INDEX booking_id (booking_id)';
-        $db->exec($sql);
+        $sql = "ALTER TABLE resource_booking_intervals ADD INDEX booking_id (booking_id)";
+        DBManager::get()->exec($sql);
     }
 
     public function down()
     {
-        $db = DBManager::get();
-
-        $query = 'ALTER TABLE resource_booking_intervals DROP INDEX booking_id';
-        $db->exec($query);
+        $query = "ALTER TABLE resource_booking_intervals DROP INDEX booking_id";
+        DBManager::get()->exec($query);
     }
 }
diff --git a/db/migrations/20220524_remove_column_termine_topic_id.php b/db/migrations/20220524_remove_column_termine_topic_id.php
index 992e765e2d8481e2313b6d1811a69c0239c4686b..23972c37f2cb88b0ddb543ca094373bb4dd410be 100644
--- a/db/migrations/20220524_remove_column_termine_topic_id.php
+++ b/db/migrations/20220524_remove_column_termine_topic_id.php
@@ -4,6 +4,8 @@
  */
 final class RemoveColumnTermineTopicId extends Migration
 {
+    use DatabaseMigrationTrait;
+
     public function description()
     {
         return 'Removes unused column topic_id from table termine.';
@@ -32,10 +34,4 @@ final class RemoveColumnTermineTopicId extends Migration
                   ADD COLUMN `topic_id` VARCHAR(32) COLLATE latin1_bin DEFAULT NULL";
         DBManager::get()->exec($query);
     }
-
-    protected function columnExists(string $table, string $column): bool
-    {
-        $query = "SHOW COLUMNS FROM `{$table}` LIKE ?";
-        return (bool) DBManager::get()->fetchOne($query, [$column]);
-    }
 }
diff --git a/db/migrations/20220629_remove_column_ex_termine_topic_id.php b/db/migrations/20220629_remove_column_ex_termine_topic_id.php
index 57e6c3dcfdabd99d9b88539308c21674d9b38da9..363c2d5c9021b596f7c91d5718fd5a9772b01fce 100644
--- a/db/migrations/20220629_remove_column_ex_termine_topic_id.php
+++ b/db/migrations/20220629_remove_column_ex_termine_topic_id.php
@@ -5,6 +5,8 @@
  */
 final class RemoveColumnExTermineTopicId extends Migration
 {
+    use DatabaseMigrationTrait;
+
     public function description()
     {
         return 'Removes unused column topic_id from table ex_termine.';
@@ -33,10 +35,4 @@ final class RemoveColumnExTermineTopicId extends Migration
                   ADD COLUMN `topic_id` VARCHAR(32) COLLATE latin1_bin DEFAULT NULL";
         DBManager::get()->exec($query);
     }
-
-    protected function columnExists(string $table, string $column): bool
-    {
-        $query = "SHOW COLUMNS FROM `{$table}` LIKE ?";
-        return (bool) DBManager::get()->fetchOne($query, [$column]);
-    }
 }
diff --git a/lib/migrations/DatabaseMigrationTrait.php b/lib/migrations/DatabaseMigrationTrait.php
new file mode 100644
index 0000000000000000000000000000000000000000..f8fad728900696d93db769aaace0808c9799cacd
--- /dev/null
+++ b/lib/migrations/DatabaseMigrationTrait.php
@@ -0,0 +1,22 @@
+<?php
+trait DatabaseMigrationTrait
+{
+    /**
+     * Returns whether a key/index with the given name exists on the given
+     * table.
+     */
+    protected function keyExists(string $table, string $key): bool
+    {
+        $query = "SHOW INDEX FROM `$table` WHERE Key_name = ?";
+        return (bool) DBManager::get()->fetchOne($query, [$key]);
+    }
+
+    /**
+     * Returns whether a column with the given name exists on the given table.
+     */
+    protected function columnExists(string $table, string $column): bool
+    {
+        $query = "SHOW COLUMNS FROM `{$table}` LIKE ?";
+        return (bool) DBManager::get()->fetchOne($query, [$column]);
+    }
+}