package liquibase.ext.percona; import java.util.Collections; import java.util.List; import liquibase.change.Change; import liquibase.change.ChangeMetaData; import liquibase.change.DatabaseChange; import liquibase.change.core.AddForeignKeyConstraintChange; import liquibase.database.Database; import liquibase.statement.SqlStatement; import liquibase.util.StringUtils; /** * Subclasses the original {@link liquibase.change.core.AddForeignKeyConstraintChange} to * integrate with pt-online-schema-change. * @see PTOnlineSchemaChangeStatement */ @DatabaseChange(name = PerconaAddForeignKeyConstraintChange.NAME, description = "Adds a foreign key constraint to an existing column", priority = PerconaAddForeignKeyConstraintChange.PRIORITY, appliesTo = "column") public class PerconaAddForeignKeyConstraintChange extends AddForeignKeyConstraintChange implements PerconaChange { public static final String NAME = "addForeignKeyConstraint"; public static final int PRIORITY = ChangeMetaData.PRIORITY_DEFAULT + 50; private Boolean usePercona; /** * Generates the statements required for the add foreign key constraint change. * In case of a MySQL database, percona toolkit will be used. * In case of generating the SQL statements for review (updateSQL) the command * will be added as a comment. * @param database the database * @return the list of statements * @see PTOnlineSchemaChangeStatement */ @Override public SqlStatement[] generateStatements(Database database) { return PerconaChangeUtil.generateStatements(this, database, super.generateStatements(database)); } @Override public String generateAlterStatement(Database database) { StringBuilder alter = new StringBuilder(); alter.append("ADD CONSTRAINT "); if (StringUtil.isNotEmpty(getConstraintName())) { alter.append(database.escapeConstraintName(getConstraintName())).append(" "); } alter.append("FOREIGN KEY "); alter.append("("); List<String> baseColumns = StringUtils.splitAndTrim(getBaseColumnNames(), ","); if (baseColumns == null) baseColumns = Collections.emptyList(); alter.append(database.escapeColumnNameList(StringUtils.join(baseColumns, ", "))); alter.append(") "); alter.append("REFERENCES "); String referencedTable = PerconaChangeUtil.resolveReferencedPerconaTableName(getBaseTableName(), getReferencedTableName()); alter.append(database.escapeTableName(getReferencedTableCatalogName(), getReferencedTableSchemaName(), referencedTable)).append(" "); alter.append("("); List<String> referencedColumns = StringUtils.splitAndTrim(getReferencedColumnNames(), ","); if (referencedColumns == null) referencedColumns = Collections.emptyList(); alter.append(database.escapeColumnNameList(StringUtils.join(referencedColumns, ", "))); alter.append(")"); if (getOnDelete() != null) { alter.append(" ON DELETE ").append(getOnDelete()); } if (getOnUpdate() != null) { alter.append(" ON UPDATE ").append(getOnUpdate()); } if (getDeferrable() != null && getDeferrable()) { alter.append(" DEFERRABLE"); } if (getInitiallyDeferred() != null && getInitiallyDeferred()) { alter.append(" INITIALLY DEFERRED"); } return alter.toString(); } @Override protected Change[] createInverses() { // that's the percona drop foreign key constraint change PerconaDropForeignKeyConstraintChange inverse = new PerconaDropForeignKeyConstraintChange(); inverse.setBaseTableSchemaName(getBaseTableSchemaName()); inverse.setBaseTableName(getBaseTableName()); inverse.setConstraintName(getConstraintName()); return new Change[] { inverse }; } @Override public Boolean getUsePercona() { return usePercona; } public void setUsePercona(Boolean usePercona) { this.usePercona = usePercona; } @Override public String getChangeSkipName() { return NAME; } @Override public String getTargetTableName() { return getBaseTableName(); } }