mirror of
https://github.com/immich-app/immich.git
synced 2025-12-25 01:11:43 +03:00
166 lines
5.0 KiB
TypeScript
166 lines
5.0 KiB
TypeScript
import { asRenameKey, haveEqualColumns } from 'src/sql-tools/helpers';
|
|
import {
|
|
CompareFunction,
|
|
Comparer,
|
|
ConstraintType,
|
|
DatabaseCheckConstraint,
|
|
DatabaseConstraint,
|
|
DatabaseForeignKeyConstraint,
|
|
DatabasePrimaryKeyConstraint,
|
|
DatabaseUniqueConstraint,
|
|
Reason,
|
|
SchemaDiff,
|
|
} from 'src/sql-tools/types';
|
|
|
|
export const compareConstraints: Comparer<DatabaseConstraint> = {
|
|
getRenameKey: (constraint) => {
|
|
switch (constraint.type) {
|
|
case ConstraintType.PRIMARY_KEY:
|
|
case ConstraintType.UNIQUE: {
|
|
return asRenameKey([constraint.type, constraint.tableName, ...constraint.columnNames.toSorted()]);
|
|
}
|
|
|
|
case ConstraintType.FOREIGN_KEY: {
|
|
return asRenameKey([
|
|
constraint.type,
|
|
constraint.tableName,
|
|
...constraint.columnNames.toSorted(),
|
|
constraint.referenceTableName,
|
|
...constraint.referenceColumnNames.toSorted(),
|
|
]);
|
|
}
|
|
|
|
case ConstraintType.CHECK: {
|
|
const expression = constraint.expression.replaceAll('(', '').replaceAll(')', '');
|
|
return asRenameKey([constraint.type, constraint.tableName, expression]);
|
|
}
|
|
}
|
|
},
|
|
onRename: (source, target) => [
|
|
{
|
|
type: 'ConstraintRename',
|
|
tableName: target.tableName,
|
|
oldName: target.name,
|
|
newName: source.name,
|
|
reason: Reason.Rename,
|
|
},
|
|
],
|
|
onMissing: (source) => [
|
|
{
|
|
type: 'ConstraintAdd',
|
|
constraint: source,
|
|
reason: Reason.MissingInTarget,
|
|
},
|
|
],
|
|
onExtra: (target) => [
|
|
{
|
|
type: 'ConstraintDrop',
|
|
tableName: target.tableName,
|
|
constraintName: target.name,
|
|
reason: Reason.MissingInSource,
|
|
},
|
|
],
|
|
onCompare: (source, target) => {
|
|
switch (source.type) {
|
|
case ConstraintType.PRIMARY_KEY: {
|
|
return comparePrimaryKeyConstraint(source, target as DatabasePrimaryKeyConstraint);
|
|
}
|
|
|
|
case ConstraintType.FOREIGN_KEY: {
|
|
return compareForeignKeyConstraint(source, target as DatabaseForeignKeyConstraint);
|
|
}
|
|
|
|
case ConstraintType.UNIQUE: {
|
|
return compareUniqueConstraint(source, target as DatabaseUniqueConstraint);
|
|
}
|
|
|
|
case ConstraintType.CHECK: {
|
|
return compareCheckConstraint(source, target as DatabaseCheckConstraint);
|
|
}
|
|
|
|
default: {
|
|
return [];
|
|
}
|
|
}
|
|
},
|
|
};
|
|
|
|
const comparePrimaryKeyConstraint: CompareFunction<DatabasePrimaryKeyConstraint> = (source, target) => {
|
|
if (!haveEqualColumns(source.columnNames, target.columnNames)) {
|
|
return dropAndRecreateConstraint(
|
|
source,
|
|
target,
|
|
`Primary key columns are different: (${source.columnNames} vs ${target.columnNames})`,
|
|
);
|
|
}
|
|
|
|
return [];
|
|
};
|
|
|
|
const compareForeignKeyConstraint: CompareFunction<DatabaseForeignKeyConstraint> = (source, target) => {
|
|
let reason = '';
|
|
|
|
const sourceDeleteAction = source.onDelete ?? 'NO ACTION';
|
|
const targetDeleteAction = target.onDelete ?? 'NO ACTION';
|
|
|
|
const sourceUpdateAction = source.onUpdate ?? 'NO ACTION';
|
|
const targetUpdateAction = target.onUpdate ?? 'NO ACTION';
|
|
|
|
if (!haveEqualColumns(source.columnNames, target.columnNames)) {
|
|
reason = `columns are different (${source.columnNames} vs ${target.columnNames})`;
|
|
} else if (!haveEqualColumns(source.referenceColumnNames, target.referenceColumnNames)) {
|
|
reason = `reference columns are different (${source.referenceColumnNames} vs ${target.referenceColumnNames})`;
|
|
} else if (source.referenceTableName !== target.referenceTableName) {
|
|
reason = `reference table is different (${source.referenceTableName} vs ${target.referenceTableName})`;
|
|
} else if (sourceDeleteAction !== targetDeleteAction) {
|
|
reason = `ON DELETE action is different (${sourceDeleteAction} vs ${targetDeleteAction})`;
|
|
} else if (sourceUpdateAction !== targetUpdateAction) {
|
|
reason = `ON UPDATE action is different (${sourceUpdateAction} vs ${targetUpdateAction})`;
|
|
}
|
|
|
|
if (reason) {
|
|
return dropAndRecreateConstraint(source, target, reason);
|
|
}
|
|
|
|
return [];
|
|
};
|
|
|
|
const compareUniqueConstraint: CompareFunction<DatabaseUniqueConstraint> = (source, target) => {
|
|
let reason = '';
|
|
|
|
if (!haveEqualColumns(source.columnNames, target.columnNames)) {
|
|
reason = `columns are different (${source.columnNames} vs ${target.columnNames})`;
|
|
}
|
|
|
|
if (reason) {
|
|
return dropAndRecreateConstraint(source, target, reason);
|
|
}
|
|
|
|
return [];
|
|
};
|
|
|
|
const compareCheckConstraint: CompareFunction<DatabaseCheckConstraint> = (source, target) => {
|
|
if (source.expression !== target.expression) {
|
|
// comparing expressions is hard because postgres reconstructs it with different formatting
|
|
// for now if the constraint exists with the same name, we will just skip it
|
|
}
|
|
|
|
return [];
|
|
};
|
|
|
|
const dropAndRecreateConstraint = (
|
|
source: DatabaseConstraint,
|
|
target: DatabaseConstraint,
|
|
reason: string,
|
|
): SchemaDiff[] => {
|
|
return [
|
|
{
|
|
type: 'ConstraintDrop',
|
|
tableName: target.tableName,
|
|
constraintName: target.name,
|
|
reason,
|
|
},
|
|
{ type: 'ConstraintAdd', constraint: source, reason },
|
|
];
|
|
};
|