Skip to content

Migration helpers to manage FKs for partitioned tables

What does this MR do?

This MR is the second step of implementing a new set of migration helpers, which will handle management of foreign key-like behavior for partitioned tables, which do not support some types of foreign keys natively. This issue implements the new migration helpers that create the database objects which will handle the cascading operations.

Related issue: #201872 (closed)

MR for part 1: !29064 (merged)

MR for part 3: !29525 (merged)

Overview

Postgres 11 does not support foreign keys that reference partitioned tables, so a similar solution needs to be provided by other means. This MR implements part of the features that are given by foreign keys, which is automatic cleanup when a record pointed to be a foreign key is deleted. The most common action taken in that scenario is a cascading delete, but also it's not unusual to set the nullify the foreign key.

A new set of migration helpers are added, which create a function in the database that performs the cleanup operation, and an AFTER DELETE trigger that will call the function. The foreign key definitions are tracked in a new table, so that when a key is added or removed by a migration, the database has the information to rebuild the cleanup function with the correct operations.

A sample migration might look something like:

class AddIssueToProjectForeignKey < ActiveRecord::Migration[6.0]
  include Gitlab::Database::PartitioningMigrationHelpers

  DOWNTIME = false

  def up
    add_foreign_key :issues, :projects
  end

  def down
    remove_foreign_key :issues, :projects
  end
end

Then, when the migration is run, it will do the following:

  1. Read the current foreign keys that reference projects from the partitioned_foreign_keys table
  2. Verify that the added foreign key references valid tables and columns
  3. Drop the existing trigger on projects (if exists)
  4. Creates or replaces the trigger function with the correct body based on foreign key definitions
  5. Re-create a trigger on projects to call the function

An example function for the above migration would look like:

CREATE OR REPLACE FUNCTION fk_cascade_function_b4db0d065d()
RETURNS TRIGGER AS
$$
BEGIN
DELETE FROM issues WHERE project_id = OLD.id;
RETURN OLD;
END
$$ LANGUAGE PLPGSQL

While the created trigger would look like:

CREATE TRIGGER fk_cascade_trigger_e59525315e
AFTER DELETE ON projects
FOR EACH ROW
EXECUTE PROCEDURE fk_cascade_function_b4db0d065d()

A remove_foreign_key will be similar in logic, with the difference being that if the last FK from a table is removed, both the trigger and trigger function will be dropped and not re-created.

Originally the entire migration helper was expected to be run inside a transactional migration, but adding a trigger to a table requires a ShareRowExclusiveLock. To limit the lock the changes where broken up, and only applied around the window of changes to the function and trigger.

Limitations

  1. At some point in the future we would likely add support for multi-column foreign keys, since we would want to be able to include the partitioning key in the foreign key.
  2. Currently only two options for ON DELETE are allowed: CASCADE and SET NULL. It was mentioned here: !29525 (comment 325426412) about needing support for other options, which is a valid point. Given that this is already a sizable change of several MR in total, I think it makes sense to try what we have now in a real scenario, and add in those additional options later once we have tested the approach.

Does this MR meet the acceptance criteria?

Conformity

Availability and Testing

Edited by 🤖 GitLab Bot 🤖

Merge request reports

Loading