Int range partitioning
Description
Int range partitioning is a technique for dividing a large table into smaller, more manageable chunks based on an integer column. Such that each partition contains a range of integers. This can be particularly useful for tables with large numbers of rows, as it can significantly improve query performance, reduce storage requirements, and simplify maintenance tasks. For this type of partitioning to work well, most queries must access data in a certain int range.
To look at this in more detail, imagine a simplified merge_request_diff_files schema:
CREATE TABLE merge_request_diff_files (
  merge_request_diff_id INT NOT NULL,
  relative_order INT NOT NULL,
  PRIMARY KEY (merge_request_diff_id, relative_order));Now imagine typical queries in the UI would display the data in a certain int range:
SELECT *
FROM merge_request_diff_files
WHERE merge_request_diff_id > 1 AND merge_request_diff_id < 10
LIMIT 100If the table is partitioned on the merge_request_diff_id column the base table would look like:
CREATE TABLE merge_request_diff_files (
  merge_request_diff_id INT NOT NULL,
  relative_order INT NOT NULL,
  PRIMARY KEY (merge_request_diff_id, relative_order))
PARTITION BY RANGE(merge_request_diff_id);The primary key of a partitioned table must include the partition key as part of the primary key definition.
And we might have a list of partitions for the table, such as:
merge_request_diff_files_1 FOR VALUES FROM (1) TO (20)
merge_request_diff_files_20 FOR VALUES FROM (20) TO (40)
merge_request_diff_files_40 FOR VALUES FROM (40) TO (60)Each partition is a separate physical table, with the same structure as
the base merge_request_diff_files table, but contains only data for rows where the
partition key falls in the specified range. For example, the partition
merge_request_diff_files_1 contains rows where the merge_request_diff_id column is
greater than or equal to 1 and less than 20.
Now, if we look at the previous example query again, the database can
use the WHERE to recognize that all matching rows are in the
merge_request_diff_files_1 partition. Rather than searching all the data
in all the partitions. In a large table, this can
dramatically reduce the amount of data the database needs to access.
Creating a new partitioned table
Because this is a new table, this can be done in 1 MR.
ActiveRecord model modification
In your ActiveRecord model, add a partitioned_by call similar to the example below. This is necessary to give instructions
for the PartitionManager to know how to create partitions dynamically.
  PARTITION_SIZE = 2_000_000
  partitioned_by :namespace_id, strategy: :int_range, partition_size: PARTITION_SIZEThe partition size can be decided based on the anticipated growth of the table.
Do not forget to register your model on the partition manager, in the file config/initializers/postgres_partitioning.rb.
Add the database migration
In the database migration that creates the table, we may also need to create the initial partitions, especially if we are partitioning the table by a foreign key that references an existing table.
In this example, we are referencing the namespaces table. You need to adapt it to the referenced table. See
the implementation of method create_partitions in this sample database migration:
class AddKnowledgeGraphEnabledNamespaces < Gitlab::Database::Migration[2.3]
  include Gitlab::Database::PartitioningMigrationHelpers
  disable_ddl_transaction!
  milestone "fill_the_current_milestone"
  TABLE_NAME = :p_knowledge_graph_enabled_namespaces
  PARTITION_SIZE = 2_000_000
  MIN_ID = Namespace.connection
    .select_value("select min_value from pg_sequences where sequencename = 'namespaces_id_seq'") || 1
  def up
    with_lock_retries do
      create_table TABLE_NAME,
        options: 'PARTITION BY RANGE (namespace_id)',
        primary_key: [:id, :namespace_id], if_not_exists: true do |t|
        t.bigserial :id, null: false
        t.bigint :namespace_id, null: false
        t.timestamps_with_timezone null: false
        t.integer :state, null: false, default: 0, limit: 2, index: true
        t.index :namespace_id, unique: true
      end
    end
    create_partitions
  end
  def down
    drop_table TABLE_NAME
  end
  private
  def create_partitions
    max_id = Gitlab::Database::QueryAnalyzers::RestrictAllowedSchemas.with_suppressed do
      Gitlab::Database::QueryAnalyzers::GitlabSchemasValidateConnection.with_suppressed do
        define_batchable_model('namespaces', connection: connection).maximum(:id) || MIN_ID
      end
    end
    create_int_range_partitions(TABLE_NAME, PARTITION_SIZE, MIN_ID, max_id)
  end
endIt’s worth noting that we cannot assume that tables sequences start from 1, that is because
in Cells, each Cell will dedicate a range of sequences for the database tables.
Converting an existing table into a partitioned table
Step 1: Creating the partitioned copy (Release N)
The first step is to add a migration to create the partitioned copy of the original table. This migration creates the appropriate partitions based on the data in the original table, and install a trigger that syncs writes from the original table into the partitioned copy.
An example migration of partitioning the merge_request_diff_commits table by its
merge_request_diff_id column would look like:
class PartitionMergeRequestDiffCommits < Gitlab::Database::Migration[2.1]
  include Gitlab::Database::PartitioningMigrationHelpers
  disable_ddl_transaction!
  def up
    partition_table_by_int_range(
      'merge_request_diff_commits',
      'merge_request_diff_id',
      partition_size: 10_000_000,
      primary_key: %w[merge_request_diff_id relative_order]
    )
  end
  def down
    drop_partitioned_table_for('merge_request_diff_commits')
  end
endAfter this has executed, any inserts, updates, or deletes in the original table are also duplicated in the new table. For updates and deletes, the operation only has an effect if the corresponding row exists in the partitioned table.
Step 2: Backfill the partitioned copy (Release N)
The second step is to add a post-deployment migration that schedules the background jobs that backfill existing data from the original table into the partitioned copy.
Continuing the above example, the migration would look like:
class BackfillPartitionMergeRequestDiffCommits < Gitlab::Database::Migration[2.2]
  include Gitlab::Database::PartitioningMigrationHelpers
  milestone '16.10'
  disable_ddl_transaction!
  restrict_gitlab_migration gitlab_schema: :gitlab_main_org
  def up
    enqueue_partitioning_data_migration :merge_request_diff_commits
  end
  def down
    cleanup_partitioning_data_migration :merge_request_diff_commits
  end
endThis step queues a batched background migration internally with BATCH_SIZE and SUB_BATCH_SIZE as 50,000 and 2,500. Refer Batched Background migrations guide for more details.
Step 3: Post-backfill cleanup (Release N+1)
This step must occur at least one release after the release that includes step (2). This gives time for the background migration to execute properly in GitLab Self-Managed instances. In this step, add another post-deployment migration that cleans up after the background migration. This includes forcing any remaining jobs to execute, and copying data that may have been missed, due to dropped or failed jobs.
A required stop must occur between steps 2 and 3 to allow the background migration from step 2 to complete successfully.
Once again, continuing the example, this migration would look like:
class CleanupPartitionMergeRequestDiffCommitsBackfill < Gitlab::Database::Migration[2.1]
  include Gitlab::Database::PartitioningMigrationHelpers
  disable_ddl_transaction!
  restrict_gitlab_migration gitlab_schema: :gitlab_main_org
  def up
    finalize_backfilling_partitioned_table :merge_request_diff_commits
  end
  def down
    # no op
  end
endAfter this migration completes, the original table and partitioned table should contain identical data. The trigger installed on the original table guarantees that the data remains in sync going forward.
Step 4: Swap the partitioned and non-partitioned tables (Release N+1)
This step replaces the non-partitioned table with its partitioned copy, this should be used only after all other migration steps have completed successfully.
Some limitations to this method MUST be handled before, or during, the swap migration:
- Secondary indexes and foreign keys are not automatically recreated on the partitioned table.
- Some types of constraints (UNIQUE and EXCLUDE) which rely on indexes, are not automatically recreated on the partitioned table, since the underlying index will not be present.
- Foreign keys referencing the original non-partitioned table should be updated to reference the partitioned table. This is not supported in PostgreSQL 11.
- Views referencing the original table are not automatically updated to reference the partitioned table.
# frozen_string_literal: true
class SwapPartitionMergeRequestDiffCommits < ActiveRecord::Migration[6.0]
  include Gitlab::Database::PartitioningMigrationHelpers
  def up
    replace_with_partitioned_table :audit_events
  end
  def down
    rollback_replace_with_partitioned_table :audit_events
  end
endAfter this migration completes:
- The partitioned table replaces the non-partitioned (original) table.
- The sync trigger created earlier is dropped.
The partitioned table is now ready for use by the application.