org.flywaydb.ant.AbstractFlywayTask Maven / Gradle / Ivy
/*
* Copyright 2010-2018 Boxfuse GmbH
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.flywaydb.ant;
import java.io.File;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import javax.sql.DataSource;
import org.apache.tools.ant.AntClassLoader;
import org.apache.tools.ant.BuildException;
import org.apache.tools.ant.Task;
import org.apache.tools.ant.types.Path;
import org.apache.tools.ant.types.Reference;
import org.flywaydb.core.Flyway;
import org.flywaydb.core.api.Location;
import org.flywaydb.core.api.logging.Log;
import org.flywaydb.core.api.logging.LogFactory;
import org.flywaydb.core.internal.util.ExceptionUtils;
import org.flywaydb.core.internal.util.StringUtils;
import org.flywaydb.core.internal.util.jdbc.DriverDataSource;
/**
* Base class for all Flyway Ant tasks.
*/
@SuppressWarnings({"UnusedDeclaration"})
public abstract class AbstractFlywayTask extends Task {
/**
* Property name prefix for placeholders that are configured through properties.
*/
private static final String PLACEHOLDERS_PROPERTY_PREFIX = "flyway.placeholders.";
private final Flyway flyway;
/**
* Logger.
*/
protected Log log;
/**
* The classpath used to load the JDBC driver and the migrations.
*/
private Path classPath;
/**
* The fully qualified classname of the jdbc driver to use to connect to the database.
Also configurable with Ant Property: ${flyway.driver}
*/
private String driver;
/**
* The jdbc url to use to connect to the database.
Also configurable with Ant Property: ${flyway.url}
*/
private String url;
/**
* The user to use to connect to the database. (default: blank)
Also configurable with Ant Property: ${flyway.user}
The credentials can be
* specified by user/password or serverId from settings.xml
*/
private String user;
/**
* The password to use to connect to the database. (default: blank)
Also configurable with Ant Property: ${flyway.password}
*/
private String password;
/**
* Locations on the classpath to scan recursively for migrations. Locations may contain both sql and java-based migrations. (default: db.migration)
Also
* configurable with Ant Property: ${flyway.locations}
*/
private String[] locations;
/**
* The custom MigrationResolvers to be used in addition or as replacement to the built-in (as determined by the skipDefaultResolvers property) ones for
* resolving Migrations to apply. (default: none)
*/
private String[] resolvers;
/**
* The callbacks for lifecycle notifications. (default: none)
*/
private String[] callbacks;
/**
* A map of <placeholder, replacementValue> to apply to sql migration scripts.
*/
private Map placeholders;
/**
* Default constructor.
*/
public AbstractFlywayTask() {
this(new Flyway(Thread.currentThread().getContextClassLoader()));
}
AbstractFlywayTask(Flyway flyway) {
this.flyway = flyway;
this.locations = locationsToStrings(flyway.getLocations());
this.placeholders = flyway.getPlaceholders();
}
private String[] locationsToStrings(Location[] locations) {
String[] locationsString = new String[locations.length];
for (int i = 0; i < locations.length; i++) {
locationsString[i] = locations[i].getDescriptor();
}
return locationsString;
}
/**
* @param classpath The classpath used to load the JDBC driver and the migrations.
Also configurable with Ant Property: ${flyway.classpath}
*/
public void setClasspath(Path classpath) {
this.classPath = classpath;
}
/**
* @param classpathref The reference to the classpath used to load the JDBC driver and the migrations.
Also configurable with Ant Property:
* ${flyway.classpathref}
*/
public void setClasspathref(Reference classpathref) {
Path classPath = new Path(getProject());
classPath.setRefid(classpathref);
this.classPath = classPath;
}
/**
* @param driver The fully qualified classname of the jdbc driver to use to connect to the database.
By default, the driver is autodetected based on the
* url.
Also configurable with Ant Property: ${flyway.driver}
*/
public void setDriver(String driver) {
this.driver = driver;
}
/**
* @param url The jdbc url to use to connect to the database.
Also configurable with Ant Property: ${flyway.url}
*/
public void setUrl(String url) {
this.url = url;
}
/**
* @param user The user to use to connect to the database.
Also configurable with Ant Property: ${flyway.user}
*/
public void setUser(String user) {
this.user = user;
}
/**
* @param password The password to use to connect to the database. (default: blank)
Also configurable with Ant Property: ${flyway.password}
*/
public void setPassword(String password) {
this.password = password;
}
/**
* @param schemas Comma-separated list of the schemas managed by Flyway. These schema names are case-sensitive.
(default: The default schema for the
* datasource connection) Consequences:
- The first schema in the list will be automatically set as the default one during the
* migration.
- The first schema in the list will also be the one containing the metadata table.
- The schemas will be cleaned
* in the order of this list.
Also configurable with Ant Property: ${flyway.schemas}
*/
public void setSchemas(String schemas) {
flyway.setSchemas(StringUtils.tokenizeToStringArray(schemas, ","));
}
/**
* @param resolvers The custom MigrationResolvers to be used in addition to the built-in ones for resolving Migrations to apply. (default: none)
*/
public void setResolvers(String resolvers) {
this.resolvers = StringUtils.tokenizeToStringArray(resolvers, ",");
}
/**
* @param skipDefaultResolvers Whether built-int resolvers should be skipped. If true, only custom resolvers are used.(default: false)
Also
* configurable with Ant Property: ${flyway.skipDefaultResolvers}
*/
public void setSkipDefaultResolvers(boolean skipDefaultResolvers) {
flyway.setSkipDefaultResolvers(skipDefaultResolvers);
}
/**
* @param callbacks A comma-separated list of fully qualified FlywayCallback implementation class names. These classes will be instantiated and wired into
* the Flyway lifecycle notification events.
*/
public void setCallbacks(String callbacks) {
this.callbacks = StringUtils.tokenizeToStringArray(callbacks, ",");
}
/**
* @param skipDefaultCallbacks Whether built-int callbacks should be skipped. If true, only custom callbacks are used.(default: false)
Also
* configurable with Ant Property: ${flyway.skipDefaultCallbacks}
*/
public void setSkipDefaultCallbacks(boolean skipDefaultCallbacks) {
flyway.setSkipDefaultCallbacks(skipDefaultCallbacks);
}
/**
* @param table The name of the schema metadata table that will be used by Flyway.
By default (single-schema mode) the metadata table is placed in
* the default schema for the connection provided by the datasource.
When the flyway.schemas property is set (multi-schema
* mode), the metadata table is placed in the first schema of the list.
(default: schema_version)
Also configurable with Ant Property:
* ${flyway.table}
*/
public void setTable(String table) {
flyway.setTable(table);
}
/**
* @param baselineVersion The version to tag an existing schema with when executing baseline. (default: 1)
Also configurable with Ant Property:
* ${flyway.baselineVersion}
*/
public void setBaselineVersion(String baselineVersion) {
flyway.setBaselineVersionAsString(baselineVersion);
}
/**
* @param baselineDescription The description to tag an existing schema with when executing baseline. (default: << Flyway Baseline >>)
Also
* configurable with Ant Property: ${flyway.baselineDescription}
*/
public void setBaselineDescription(String baselineDescription) {
flyway.setBaselineDescription(baselineDescription);
}
/**
* Whether to allow mixing transactional and non-transactional statements within the same migration.
Also configurable with Ant Property:
* ${flyway.mixed}
*
* @param mixed {@code true} if mixed migrations should be allowed. {@code false} if an error should be thrown instead. (default: {@code false})
*/
public void setMixed(boolean mixed) {
flyway.setMixed(mixed);
}
/**
* Whether to group all pending migrations together in the same transaction when applying them (only recommended for databases with support for DDL
* transactions).
*
* Also configurable with Ant Property: ${flyway.group}
*
* @param group {@code true} if migrations should be grouped. {@code false} if they should be applied individually instead. (default: {@code false})
*/
public void setGroup(boolean group) {
flyway.setGroup(group);
}
/**
* The username that will be recorded in the metadata table as having applied the migration.
Also configurable with Ant Property:
* ${flyway.installedBy}
*
* @param installedBy The username or blank for the current database user of the connection. (default: blank).
*/
public void setInstalledBy(String installedBy) {
flyway.setInstalledBy(installedBy);
}
/**
* Creates the datasource base on the provided parameters.
*
* @return The fully configured datasource.
* @throws Exception Thrown when the datasource could not be created.
*/
/* private -> for testing */ DataSource createDataSource() throws Exception {
String driverValue = useValueIfPropertyNotSet(driver, "driver");
String urlValue = useValueIfPropertyNotSet(url, "url");
String userValue = useValueIfPropertyNotSet(user, "user");
String passwordValue = useValueIfPropertyNotSet(password, "password");
return new DriverDataSource(Thread.currentThread().getContextClassLoader(), driverValue, urlValue, userValue, passwordValue, null);
}
/**
* Retrieves a value either from an Ant property or if not set, directly.
*
* @param value The value to check.
* @param flywayProperty The flyway Ant property. Ex. 'url' for 'flyway.url'
* @return The value.
*/
protected String useValueIfPropertyNotSet(String value, String flywayProperty) {
String propertyValue = getProject().getProperty("flyway." + flywayProperty);
if (propertyValue != null) {
return propertyValue;
}
return value;
}
/**
* Retrieves a boolean value either from an Ant property or if not set, directly.
*
* @param value The boolean value to check.
* @param flywayProperty The flyway Ant property. Ex. 'url' for 'flyway.url'
* @return The boolean value.
*/
protected boolean useValueIfPropertyNotSet(boolean value, String flywayProperty) {
String propertyValue = getProject().getProperty("flyway." + flywayProperty);
if (propertyValue != null) {
return Boolean.parseBoolean(propertyValue);
}
return value;
}
/**
* Prepares the classpath this task runs in, so that it includes both the classpath for Flyway and the classpath for the JDBC drivers and migrations.
*/
private void prepareClassPath() {
Path classpath = (Path) getProject().getReference("flyway.classpath");
if (classpath != null) {
setClasspath(classpath);
} else {
Reference classpathRef = (Reference) getProject().getReference("flyway.classpathref");
if (classpathRef != null) {
setClasspathref(classpathRef);
}
}
ClassLoader classLoader =
new AntClassLoader(getClass().getClassLoader(), getProject(), classPath);
Thread.currentThread().setContextClassLoader(classLoader);
}
/**
* Do not use. For Ant itself.
*
* @param locationsElement The locations on the classpath.
*/
public void addConfiguredLocations(LocationsElement locationsElement) {
this.locations = locationsElement.locations.toArray(new String[locationsElement.locations.size()]);
}
/**
* Do not use. For Ant itself.
*
* @param resolversElement The resolvers on the classpath.
*/
public void addConfiguredResolvers(ResolversElement resolversElement) {
this.resolvers = resolversElement.resolvers.toArray(new String[resolversElement.resolvers.size()]);
}
/**
* Do not use. For Ant itself.
*
* @param callbacksElement The callbacks on the classpath.
*/
public void addConfiguredCallbacks(CallbacksElement callbacksElement) {
this.callbacks = callbacksElement.callbacks.toArray(new String[callbacksElement.callbacks.size()]);
}
/**
* Do not use. For Ant itself.
*
* @param schemasElement The schemas.
*/
public void addConfiguredSchemas(SchemasElement schemasElement) {
flyway.setSchemas(schemasElement.schemas.toArray(new String[schemasElement.schemas.size()]));
}
/**
* @param encoding The encoding of Sql migrations. (default: UTF-8)
Also configurable with Ant Property: ${flyway.encoding}
*/
public void setEncoding(String encoding) {
flyway.setEncoding(encoding);
}
/**
* Sql migrations have the following file name structure: prefixVERSIONseparatorDESCRIPTIONsuffix , which using the defaults translates to
* V1_1__My_description.sql
*
* @param sqlMigrationPrefix The file name prefix for Sql migrations (default: V)
Also configurable with Ant Property: ${flyway.sqlMigrationPrefix}
*/
public void setSqlMigrationPrefix(String sqlMigrationPrefix) {
flyway.setSqlMigrationPrefix(sqlMigrationPrefix);
}
/**
* Repeatable sql migrations have the following file name structure: prefixSeparatorDESCRIPTIONsuffix , which using the defaults translates to
* R__My_description.sql
*
* @param repeatableSqlMigrationPrefix The file name prefix for repeatable sql migrations (default: R)
Also configurable with Ant Property:
* ${flyway.repeatableSqlMigrationPrefix}
*/
public void setRepeatableSqlMigrationPrefix(String repeatableSqlMigrationPrefix) {
flyway.setRepeatableSqlMigrationPrefix(repeatableSqlMigrationPrefix);
}
/**
* Sql migrations have the following file name structure: prefixVERSIONseparatorDESCRIPTIONsuffix , which using the defaults translates to
* V1_1__My_description.sql
*
* @param sqlMigrationSeparator The file name separator for Sql migrations (default: V)
Also configurable with Ant Property:
* ${flyway.sqlMigrationPrefix}
*/
public void setSqlMigrationSeparator(String sqlMigrationSeparator) {
flyway.setSqlMigrationSeparator(sqlMigrationSeparator);
}
/**
* Sql migrations have the following file name structure: prefixVERSIONseparatorDESCRIPTIONsuffix , which using the defaults translates to
* V1_1__My_description.sql
Multiple suffixes (like .sql,.pkg,.pkb) can be specified for easier compatibility with other tools such as editors with
* specific file associations.
*
* @param sqlMigrationSuffixes The file name suffixes for SQL migrations, comma-separated. (default: .sql)
Also configurable with Ant Property:
* ${flyway.sqlMigrationSuffixes}
*/
public void setSqlMigrationSuffixes(String sqlMigrationSuffixes) {
flyway.setSqlMigrationSuffixes(StringUtils.tokenizeToStringArray(sqlMigrationSuffixes, ","));
}
/**
* @param target The target version up to which Flyway should consider migrations. Migrations with a higher version number will be ignored. The special
* value {@code current} designates the current version of the schema. (default: the latest version)
Also configurable with Ant Property:
* ${flyway.target}
*/
public void setTarget(String target) {
flyway.setTargetAsString(target);
}
/**
* @param cleanOnValidationError Whether to automatically call clean or not when a validation error occurs. (default: {@code false})
This is
* exclusively intended as a convenience for development. Even tough we strongly recommend not to change migration scripts
* once they have been checked into SCM and run, this provides a way of dealing with this case in a smooth manner. The
* database will be wiped clean automatically, ensuring that the next migration will bring you back to the state checked into
* SCM.
Warning ! Do not enable in production !
Also configurable with Ant Property:
* ${flyway.cleanOnValidationError}
*/
public void setCleanOnValidationError(boolean cleanOnValidationError) {
flyway.setCleanOnValidationError(cleanOnValidationError);
}
/**
* @param cleanDisabled Whether to disable clean. (default: {@code false}) This is especially useful for production environments where running clean can
* be quite a career limiting move.
*/
public void setCleanDisabled(boolean cleanDisabled) {
flyway.setCleanDisabled(cleanDisabled);
}
/**
* @param outOfOrder Allows migrations to be run "out of order" (default: {@code false}). If you already have versions 1 and 3 applied, and now a version
* 2 is found, it will be applied too instead of being ignored.
Also configurable with Ant Property: ${flyway.outOfOrder}
*/
public void setOutOfOrder(boolean outOfOrder) {
flyway.setOutOfOrder(outOfOrder);
}
/**
* @param placeholderReplacement Whether placeholders should be replaced. (default: true)
Also configurable with Ant Property:
* ${flyway.placeholderReplacement}
*/
public void setPlaceholderReplacement(boolean placeholderReplacement) {
flyway.setPlaceholderReplacement(placeholderReplacement);
}
/**
* @param placeholderPrefix The prefix of every placeholder. (default: ${ )
Also configurable with Ant Property: ${flyway.placeholderPrefix}
*/
public void setPlaceholderPrefix(String placeholderPrefix) {
flyway.setPlaceholderPrefix(placeholderPrefix);
}
/**
* @param placeholderSuffix The suffix of every placeholder. (default: } )
Also configurable with Ant Property: ${flyway.placeholderSuffix}
*/
public void setPlaceholderSuffix(String placeholderSuffix) {
flyway.setPlaceholderSuffix(placeholderSuffix);
}
/**
* Adds placeholders from a nested <placeholders> element. Called by Ant.
*
* @param placeholders The fully configured placeholders element.
*/
public void addConfiguredPlaceholders(PlaceholdersElement placeholders) {
this.placeholders = placeholders.placeholders;
}
/**
* Ignore missing migrations when reading the metadata table. These are migrations that were performed by an older deployment of the application that are no
* longer available in this version. For example: we have migrations available on the classpath with versions 1.0 and 3.0. The metadata table indicates that
* a migration with version 2.0 (unknown to us) has also been applied. Instead of bombing out (fail fast) with an exception, a warning is logged and Flyway
* continues normally. This is useful for situations where one must be able to deploy a newer version of the application even though it doesn't contain
* migrations included with an older one anymore.
*
* @param ignoreMissingMigrations {@code true} to continue normally and log a warning, {@code false} to fail fast with an exception. (default: {@code
* false})
*/
public void setIgnoreMissingMigrations(boolean ignoreMissingMigrations) {
flyway.setIgnoreMissingMigrations(ignoreMissingMigrations);
}
/**
* Whether to ignore future migrations when reading the metadata table. These are migrations that were performed by a newer deployment of the application
* that are not yet available in this version. For example: we have migrations available on the classpath up to version 3.0. The metadata table indicates
* that a migration to version 4.0 (unknown to us) has already been applied. Instead of bombing out (fail fast) with an exception, a warning is logged and
* Flyway continues normally. This is useful for situations where one must be able to redeploy an older version of the application after the database has
* been migrated by a newer one.
Also configurable with Ant Property: ${flyway.ignoreFutureMigrations}
*
* @param ignoreFutureMigrations {@code true} to continue normally and log a warning, {@code false} to fail fast with an exception. (default: {@code true})
*/
public void setIgnoreFutureMigrations(boolean ignoreFutureMigrations) {
flyway.setIgnoreFutureMigrations(ignoreFutureMigrations);
}
/**
* @param validateOnMigrate Whether to automatically call validate or not when running migrate. (default: {@code true})
Also configurable with Ant
* Property: ${flyway.validateOnMigrate}
*/
public void setValidateOnMigrate(boolean validateOnMigrate) {
flyway.setValidateOnMigrate(validateOnMigrate);
}
/**
* Whether to automatically call baseline when migrate is executed against a non-empty schema with no metadata table. This schema will then be baselined
* with the {@code initialVersion} before executing the migrations. Only migrations above {@code initialVersion} will then be applied.
This is
* useful for initial Flyway production deployments on projects with an existing DB.
Be careful when enabling this as it removes the safety net
* that ensures Flyway does not migrate the wrong database in case of a configuration mistake!
Also configurable with Ant Property:
* ${flyway.baselineOnMigrate}
*
* @param baselineOnMigrate {@code true} if baseline should be called on migrate for non-empty schemas, {@code false} if not. (default: {@code false})
*/
public void setBaselineOnMigrate(boolean baselineOnMigrate) {
flyway.setBaselineOnMigrate(baselineOnMigrate);
}
@Override
public void init() throws BuildException {
AntLogCreator.INSTANCE.setAntProject(getProject());
LogFactory.setLogCreator(AntLogCreator.INSTANCE);
log = LogFactory.getLog(getClass());
}
@Override
public void execute() throws BuildException {
prepareClassPath();
try {
flyway.setDataSource(createDataSource());
if (resolvers != null) {
flyway.setResolversAsClassNames(resolvers);
}
if (callbacks != null) {
flyway.setCallbacksAsClassNames(callbacks);
}
Properties projectProperties = new Properties();
projectProperties.putAll(getProject().getProperties());
flyway.configure(projectProperties);
flyway.configure(System.getProperties());
flyway.setLocations(getLocations());
addPlaceholdersFromProperties(placeholders, getProject().getProperties());
flyway.setPlaceholders(placeholders);
doExecute(flyway);
} catch (Exception e) {
throw new BuildException("Flyway Error: " + e.toString(), ExceptionUtils.getRootCause(e));
}
}
/**
* Executes this task.
*
* @param flyway The flyway instance to operate on.
* @throws Exception any exception
*/
protected abstract void doExecute(Flyway flyway) throws Exception;
/**
* @return The locations configured through Ant.
*/
private String[] getLocations() {
String[] locationsVal = locations;
String locationsProperty = getProject().getProperty("flyway.locations");
if (locationsProperty != null) {
String[] locationsString = StringUtils.tokenizeToStringArray(locationsProperty, ",");
}
//Adjust relative locations to be relative from Ant's basedir.
File baseDir = getProject().getBaseDir();
for (int i = 0; i < locationsVal.length; i++) {
locationsVal[i] = adjustRelativeFileSystemLocationToBaseDir(baseDir, locationsVal[i]);
}
return locationsVal;
}
/**
* Adjusts a relative filesystem location to Ant's basedir. All other locations are left untouched.
*
* @param baseDir Ant's basedir.
* @param locationStr The location to adjust.
* @return The adjusted location.
*/
/* private -> testing */
static String adjustRelativeFileSystemLocationToBaseDir(File baseDir, String locationStr) {
Location location = new Location(locationStr);
if (location.isFileSystem() && !new File(location.getPath()).isAbsolute()) {
return Location.FILESYSTEM_PREFIX + baseDir.getAbsolutePath() + "/" + location.getPath();
}
return locationStr;
}
/**
* Adds the additional placeholders contained in these properties to the existing list.
*
* @param placeholders The existing list of placeholders.
* @param properties The properties containing additional placeholders.
*/
private static void addPlaceholdersFromProperties(Map placeholders, Hashtable properties) {
for (Object property : properties.keySet()) {
String propertyName = (String) property;
if (propertyName.startsWith(PLACEHOLDERS_PROPERTY_PREFIX)
&& propertyName.length() > PLACEHOLDERS_PROPERTY_PREFIX.length()) {
String placeholderName = propertyName.substring(PLACEHOLDERS_PROPERTY_PREFIX.length());
String placeholderValue = (String) properties.get(propertyName);
placeholders.put(placeholderName, placeholderValue);
}
}
}
/**
* The nested <locations> element of the task. Contains 1 or more <location> sub-elements.
*/
public static class LocationsElement {
/**
* The classpath locations.
*/
List locations = new ArrayList();
/**
* Do not use. For Ant itself.
*
* @param location A location on the classpath.
*/
public void addConfiguredLocation(LocationElement location) {
locations.add(location.path);
}
}
/**
* One <location> sub-element within the <locations> element.
*/
public static class LocationElement {
/**
* The path of the location.
*/
private String path;
/**
* Do not use. For Ant itself.
*
* @param path The path of the location.
*/
public void setPath(String path) {
this.path = path;
}
}
/**
* The nested <schemas> element of the task. Contains 1 or more <schema> sub-elements.
*/
public static class SchemasElement {
/**
* The schema names.
*/
List schemas = new ArrayList();
/**
* Do not use. For Ant itself.
*
* @param schema A schema.
*/
public void addConfiguredLocation(SchemaElement schema) {
schemas.add(schema.name);
}
}
/**
* One <location> sub-element within the <locations> element.
*/
public static class SchemaElement {
/**
* The name of the schema.
*/
private String name;
/**
* Do not use. For Ant itself.
*
* @param name The name of the schema.
*/
public void setPath(String name) {
this.name = name;
}
}
/**
* The nested <resolvers> element of the task. Contains 1 or more <resolver> sub-elements.
*/
public static class ResolversElement {
/**
* The classpath locations.
*/
List resolvers = new ArrayList();
/**
* Do not use. For Ant itself.
*
* @param resolver A resolver on the classpath.
*/
public void addConfiguredResolver(ResolverElement resolver) {
resolvers.add(resolver.clazz);
}
}
/**
* One <resolver> sub-element within the <resolvers> element.
*/
public static class ResolverElement {
/**
* The fully qualified class name of the resolver.
*/
private String clazz;
/**
* Do not use. For Ant itself.
*
* @param clazz The fully qualified class name of the resolver.
*/
public void setClass(String clazz) {
this.clazz = clazz;
}
}
/**
* The nested <callbacks> element of the task. Contains 1 or more <callback> sub-elements.
*/
public static class CallbacksElement {
/**
* The classpath locations.
*/
List callbacks = new ArrayList();
/**
* Do not use. For Ant itself.
*
* @param callback A callback on the classpath.
*/
public void addConfiguredCallback(CallbackElement callback) {
callbacks.add(callback.clazz);
}
}
/**
* One <callback> sub-element within the <callbacks> element.
*/
public static class CallbackElement {
/**
* The fully qualified class name of the callback.
*/
private String clazz;
/**
* Do not use. For Ant itself.
*
* @param clazz The fully qualified class name of the callback.
*/
public void setClass(String clazz) {
this.clazz = clazz;
}
}
/**
* Nested <placeholders> element of the migrate Ant task.
*/
public static class PlaceholdersElement {
/**
* A map of <placeholder, replacementValue> to apply to sql migration scripts.
*/
Map placeholders = new HashMap();
/**
* Adds a placeholder from a nested <placeholder> element. Called by Ant.
*
* @param placeholder The fully configured placeholder element.
*/
public void addConfiguredPlaceholder(PlaceholderElement placeholder) {
placeholders.put(placeholder.name, placeholder.value);
}
}
/**
* Nested <placeholder> element inside the <placeholders> element of the migrate Ant task.
*/
public static class PlaceholderElement {
/**
* The name of the placeholder.
*/
private String name;
/**
* The value of the placeholder.
*/
private String value;
/**
* @param name The name of the placeholder.
*/
public void setName(String name) {
this.name = name;
}
/**
* @param value The value of the placeholder.
*/
public void setValue(String value) {
this.value = value;
}
}
}