liquibase.change.ChangeParameterMetaData Maven / Gradle / Ivy
package liquibase.change;
import liquibase.change.core.LoadDataColumnConfig;
import liquibase.database.Database;
import liquibase.database.DatabaseFactory;
import liquibase.database.DatabaseList;
import liquibase.exception.UnexpectedLiquibaseException;
import liquibase.exception.ValidationErrors;
import liquibase.serializer.LiquibaseSerializable;
import liquibase.sqlgenerator.SqlGeneratorFactory;
import liquibase.statement.DatabaseFunction;
import liquibase.statement.SequenceNextValueFunction;
import liquibase.statement.SqlStatement;
import liquibase.util.ObjectUtil;
import liquibase.util.StringUtil;
import lombok.Getter;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.math.BigInteger;
import java.util.*;
/**
* Static metadata about a {@link Change} parameter.
* Instances of this class are tracked within {@link ChangeMetaData} and are immutable.
*/
public class ChangeParameterMetaData {
public static final String COMPUTE = "COMPUTE";
public static final String ALL = "all";
public static final String NONE = "none";
private final Change change;
private final String parameterName;
private final String description;
private final Map exampleValues;
private final String displayName;
private String dataType;
@Getter
private Class dataTypeClass;
@Getter
private Type[] dataTypeClassParameters = new Type[0];
@Getter
private final String since;
private Set requiredForDatabase;
private Set supportedDatabases;
private final String mustEqualExisting;
private final LiquibaseSerializable.SerializationType serializationType;
private final String[] requiredForDatabaseArg;
private final String[] supportedDatabasesArg;
private Optional readMethodRef = Optional.empty();
private Optional writeMethodRef = Optional.empty();
@Getter
private final String[] alternateParameterNames;
public ChangeParameterMetaData(Change change, String parameterName, String displayName, String description,
Map exampleValues, String since, Type dataType,
String[] requiredForDatabase, String[] supportedDatabases, String mustEqualExisting,
LiquibaseSerializable.SerializationType serializationType) {
this(change, parameterName, displayName, description, exampleValues, since, dataType, requiredForDatabase,
supportedDatabases, mustEqualExisting, serializationType, null);
}
public ChangeParameterMetaData(Change change, String parameterName, String displayName, String description,
Map exampleValues, String since, Type dataType,
String[] requiredForDatabase, String[] supportedDatabases, String mustEqualExisting,
LiquibaseSerializable.SerializationType serializationType, String[] alternateParameterNames) {
if (parameterName == null) {
throw new UnexpectedLiquibaseException("Unexpected null parameterName");
}
if (parameterName.contains(" ")) {
throw new UnexpectedLiquibaseException("Unexpected space in parameterName");
}
if (displayName == null) {
throw new UnexpectedLiquibaseException("Unexpected null displayName");
}
if (dataType == null) {
throw new UnexpectedLiquibaseException("Unexpected null dataType");
}
this.change = change;
this.parameterName = parameterName;
this.displayName = displayName;
this.description = description;
this.exampleValues = exampleValues;
if (dataType instanceof Class) {
this.dataType = StringUtil.lowerCaseFirst(((Class>) dataType).getSimpleName());
this.dataTypeClass = (Class) dataType;
} else if (dataType instanceof ParameterizedType) {
this.dataType = StringUtil.lowerCaseFirst(
((Class>) ((ParameterizedType) dataType).getRawType()).getSimpleName() +
" of " +
StringUtil.lowerCaseFirst(
((Class>) ((ParameterizedType) dataType).getActualTypeArguments()[0]).getSimpleName()
)
);
this.dataTypeClass = (Class) ((ParameterizedType) dataType).getRawType();
this.dataTypeClassParameters = ((ParameterizedType) dataType).getActualTypeArguments();
}
this.mustEqualExisting = mustEqualExisting;
this.serializationType = serializationType;
this.since = since;
this.supportedDatabasesArg = supportedDatabases;
this.requiredForDatabaseArg = requiredForDatabase;
this.alternateParameterNames = alternateParameterNames;
}
public ChangeParameterMetaData withAccessors(Method readMethod, Method writeMethod) {
this.readMethodRef = Optional.ofNullable(readMethod);
this.writeMethodRef = Optional.ofNullable(writeMethod);
return this;
}
protected Set analyzeSupportedDatabases(String[] supportedDatabases) {
if (supportedDatabases == null) {
supportedDatabases = new String[]{COMPUTE};
}
Set computedDatabases = new HashSet<>();
if ((supportedDatabases.length == 1)
&& StringUtil.join(supportedDatabases, ",").equals(COMPUTE)) {
int validDatabases = 0;
for (Database database : DatabaseFactory.getInstance().getImplementedDatabases()) {
if ((database.getShortName() == null) || "unsupported".equals(database.getShortName())) {
continue;
}
if (!change.supports(database)) {
continue;
}
try {
if (!change.generateStatementsVolatile(database)) {
Change testChange = change.getClass().getConstructor().newInstance();
ValidationErrors originalErrors = getStatementErrors(testChange, database);
this.setValue(testChange, this.getExampleValue(database));
ValidationErrors finalErrors = getStatementErrors(testChange, database);
if (finalErrors.getUnsupportedErrorMessages().isEmpty() || (finalErrors
.getUnsupportedErrorMessages().size() == originalErrors.getUnsupportedErrorMessages()
.size())) {
computedDatabases.add(database.getShortName());
}
validDatabases++;
}
} catch (Exception ignore) {
// Do nothing
}
}
if (validDatabases == 0) {
return new HashSet<>(Collections.singletonList(ALL));
} else if (computedDatabases.size() == validDatabases) {
computedDatabases = new HashSet<>(Collections.singletonList(ALL));
}
computedDatabases.remove(NONE);
return computedDatabases;
} else {
return new HashSet<>(Arrays.asList(supportedDatabases));
}
}
protected Set analyzeRequiredDatabases(String[] requiredDatabases) {
if (requiredDatabases == null) {
requiredDatabases = new String[]{COMPUTE};
}
Set computedDatabases = new HashSet<>();
if ((requiredDatabases.length == 1)
&& StringUtil.join(requiredDatabases, ",").equals(COMPUTE)) {
int validDatabases = 0;
for (Database database : DatabaseFactory.getInstance().getImplementedDatabases()) {
try {
if (!change.generateStatementsVolatile(database)) {
Change testChange = change.getClass().getConstructor().newInstance();
ValidationErrors originalErrors = getStatementErrors(testChange, database);
this.setValue(testChange, this.getExampleValue(database));
ValidationErrors finalErrors = getStatementErrors(testChange, database);
if (!originalErrors.getRequiredErrorMessages().isEmpty() && (finalErrors
.getRequiredErrorMessages().size() < originalErrors.getRequiredErrorMessages().size())
) {
computedDatabases.add(database.getShortName());
}
validDatabases++;
}
} catch (Exception ignore) {
// Do nothing
}
}
if (validDatabases == 0) {
return new HashSet<>();
} else if (computedDatabases.size() == validDatabases) {
computedDatabases = new HashSet<>(Collections.singletonList(ALL));
}
computedDatabases.remove(NONE);
} else {
computedDatabases = new HashSet<>(Arrays.asList(requiredDatabases));
}
computedDatabases.remove(NONE);
return computedDatabases;
}
private static ValidationErrors getStatementErrors(Change testChange, Database database) {
ValidationErrors errors = new ValidationErrors();
SqlStatement[] statements = testChange.generateStatements(database);
for (SqlStatement statement : statements) {
errors.addAll(SqlGeneratorFactory.getInstance().validate(statement, database));
}
return errors;
}
/**
* Programmatic Name of the parameter. Will not contain spaces so it can be used for XMl tag names etc.
* By convention, Change names should start be camel case starting with a lower case letter.
*/
public String getParameterName() {
return parameterName;
}
/**
* A more friendly name of the parameter.
*/
public String getDisplayName() {
return displayName;
}
/**
* Return the data type of value stored in this parameter. Used for documentation and integration purposes as well
* as validation.
*/
public String getDataType() {
return dataType;
}
/**
* Return the database types for which this parameter is required. The strings returned correspond to the values
* returned by {@link liquibase.database.Database#getShortName()}.
* If the parameter is required for all databases, this will return the string "all" as an element.
* If the parameter is required for no databases, this will return an empty set. Passing the string "none" to the
* constructor also results in an empty set.
* This method will never return a null value
*/
public Set getRequiredForDatabase() {
if (requiredForDatabase == null) {
requiredForDatabase = Collections.unmodifiableSet(analyzeRequiredDatabases(requiredForDatabaseArg));
}
return requiredForDatabase;
}
public Set getSupportedDatabases() {
if (supportedDatabases == null) {
supportedDatabases = Collections.unmodifiableSet(analyzeSupportedDatabases(supportedDatabasesArg));
}
return supportedDatabases;
}
/**
* A convenience method for testing the value returned by {@link #getRequiredForDatabase()} against a given database.
* Returns true if the {@link Database#getShortName()} method is contained in the required databases or the
* required database list contains the string "all"
*/
public boolean isRequiredFor(Database database) {
return getRequiredForDatabase().contains(ALL) || getRequiredForDatabase().contains(database.getShortName());
}
public boolean supports(Database database) {
return getSupportedDatabases().contains(ALL) || getSupportedDatabases().contains(database.getShortName());
}
/**
* Returns the current value of this parameter for the given Change.
*/
public Object getCurrentValue(Change change) {
try {
return getReadMethod(change).invoke(change);
} catch (UnexpectedLiquibaseException e) {
throw e;
} catch (Exception e) {
throw new UnexpectedLiquibaseException(e);
}
}
private Method getReadMethod(Change change) {
if (readMethodRef.isPresent()) {
return readMethodRef.orElseThrow(() -> new UnexpectedLiquibaseException("No readMethod for " + parameterName));
}
try {
readMethodRef = Optional.empty();
for (PropertyDescriptor descriptor : ObjectUtil.getDescriptors(change.getClass())) {
if (descriptor.getDisplayName().equals(this.parameterName)) {
Method readMethod = descriptor.getReadMethod();
if (readMethod == null) {
readMethod = change.getClass().getMethod(
"is" + StringUtil.upperCaseFirst(descriptor.getName())
);
}
readMethodRef = Optional.of(readMethod);
return readMethod;
}
}
} catch (Exception e) {
throw new UnexpectedLiquibaseException(e);
}
throw new UnexpectedLiquibaseException("Could not find readMethod for " + this.parameterName);
}
/**
* Sets the value of this parameter on the given change.
*/
public void setValue(Change change, Object value) {
if ((value instanceof String) && (!"string".equals(dataType))) {
try {
switch (dataType) {
case "bigInteger":
value = new BigInteger((String) value);
break;
case "databaseFunction":
value = new DatabaseFunction((String) value);
break;
default:
throw new UnexpectedLiquibaseException("Unknown data type: " + dataType);
}
} catch (Exception e) {
throw new UnexpectedLiquibaseException("Cannot convert string value '" + value + "' to " +
dataType + ": " + e.getMessage());
}
}
try {
Method writeMethod = getWriteMethod(change);
Class> expectedWriteType = writeMethod.getParameterTypes()[0];
if ((value != null)) {
Class> actualType = value.getClass();
if (!expectedWriteType.isAssignableFrom(actualType)) {
if (expectedWriteType.equals(String.class)) {
value = value.toString();
} else if (!allowUnboxing(actualType, expectedWriteType)) {
throw new UnexpectedLiquibaseException(
"Could not convert " + actualType.getName() +
" to " +
expectedWriteType.getName()
);
}
}
}
writeMethod.invoke(change, value);
} catch (UnexpectedLiquibaseException e) {
throw e;
} catch (Exception e) {
throw new UnexpectedLiquibaseException("Error setting " + this.parameterName + " to " + value, e);
}
}
private static boolean allowUnboxing(Class> actual, Class> expected) {
if (!expected.isPrimitive()) {
return false;
}
return
actual.equals(Boolean.class) && expected.equals(boolean.class) ||
actual.equals(Byte.class) && expected.equals(byte.class) ||
actual.equals(Character.class) && expected.equals(char.class) ||
actual.equals(Double.class) && expected.equals(double.class) ||
actual.equals(Float.class) && expected.equals(float.class) ||
actual.equals(Integer.class) && expected.equals(int.class) ||
actual.equals(Long.class) && expected.equals(long.class) ||
actual.equals(Short.class) && expected.equals(short.class);
}
private Method getWriteMethod(Change change) {
if (writeMethodRef.isPresent()) {
return writeMethodRef.orElseThrow(() -> new UnexpectedLiquibaseException("No writeMethod for " + parameterName));
}
try {
writeMethodRef = Optional.empty();
for (PropertyDescriptor descriptor : ObjectUtil.getDescriptors(change.getClass())) {
if (descriptor.getDisplayName().equals(this.parameterName)) {
Method writeMethod = descriptor.getWriteMethod();
if (writeMethod == null) {
break;
}
writeMethodRef = Optional.of(writeMethod);
return writeMethod;
}
}
} catch (Exception e) {
throw new UnexpectedLiquibaseException(e);
}
throw new UnexpectedLiquibaseException("Could not find writeMethod for " + this.parameterName);
}
/**
* Returns a dot-delimited chain of {@link liquibase.structure.DatabaseObject} fields describing what existing
* value this parameter would need to be set if applying the Change to a particular DatabaseObject.
*
* For example, in an addColumn Change, the "name" parameter would return "column.name" because if you know of an
* existing Column object, the "name" parameter needs to be set to the column's name.
* In the addColumn's "tableName" parameter, this method would return "column.table.name".
*
* The values of the chain correspond to the {@link liquibase.structure.DatabaseObject#getObjectTypeName()} and
* {@link liquibase.structure.DatabaseObject#getAttributes()}
*
* This method is used by integrations that want to generate Change instances or configurations pre-filled with
* data required to apply to an existing database object.
*/
public String getMustEqualExisting() {
return mustEqualExisting;
}
/**
* Return the {@link liquibase.serializer.LiquibaseSerializable.SerializationType}
* to use when serializing this object.
*/
public liquibase.serializer.LiquibaseSerializable.SerializationType getSerializationType() {
return serializationType;
}
public Object getExampleValue(Database database) {
if (exampleValues != null) {
Object exampleValue = null;
for (Map.Entry entry : exampleValues.entrySet()) {
if (ALL.equalsIgnoreCase(entry.getKey())) {
exampleValue = entry.getValue();
} else if (DatabaseList.definitionMatches(entry.getKey(), database, false)) {
return entry.getValue();
}
}
if (exampleValue != null) {
return exampleValue;
}
}
Map standardExamples = new HashMap<>();
standardExamples.put("tableName", "person");
standardExamples.put("schemaName", "public");
standardExamples.put("tableSchemaName", "public");
standardExamples.put("catalogName", "cat");
standardExamples.put("tableCatalogName", "cat");
standardExamples.put("columnName", "id");
standardExamples.put("columnNames", "id, name");
standardExamples.put("indexName", "idx_address");
standardExamples.put("columnDataType", "int");
standardExamples.put("dataType", "int");
standardExamples.put("sequenceName", "seq_id");
standardExamples.put("viewName", "v_person");
standardExamples.put("constraintName", "const_name");
standardExamples.put("primaryKey", "pk_id");
if (standardExamples.containsKey(parameterName)) {
return standardExamples.get(parameterName);
}
for (String prefix : new String[]{"base", "referenced", "new", "old"}) {
if (parameterName.startsWith(prefix)) {
String mainName = StringUtil.lowerCaseFirst(parameterName.replaceFirst("^" + prefix, ""));
if (standardExamples.containsKey(mainName)) {
return standardExamples.get(mainName);
}
}
}
switch (dataType) {
case "string":
return "A String";
case "integer":
return 3;
case "boolean":
return true;
case "bigInteger":
return new BigInteger("371717");
case "list":
return null; // TODO
case "sequenceNextValueFunction":
return new SequenceNextValueFunction("seq_name");
case "databaseFunction":
return new DatabaseFunction("now");
case "list of columnConfig": {
ArrayList list = new ArrayList<>();
list.add(new ColumnConfig().setName("id").setType("int"));
return list;
}
case "list of addColumnConfig": {
ArrayList list = new ArrayList<>();
list.add(new AddColumnConfig().setName("id").setType("int"));
return list;
}
case "list of loadDataColumnConfig": {
ArrayList list = new ArrayList<>();
list.add(new LoadDataColumnConfig().setName("id").setType("int"));
return list;
}
default:
throw new UnexpectedLiquibaseException("Unknown dataType " + dataType + " for " + getParameterName());
}
}
public String getDescription() {
if (description != null) {
return description;
}
Map standardDescriptions = new HashMap<>();
standardDescriptions.put("tableName", "Name of the table");
standardDescriptions.put("schemaName", "Name of the schema");
standardDescriptions.put("catalogName", "Name of the catalog");
standardDescriptions.put("columnName", "Name of the column");
return StringUtil.trimToEmpty(standardDescriptions.get(parameterName));
}
@Override
public String toString() {
return (change != null ? (change + ".") : "") + getParameterName();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy