org.flywaydb.commandline.Main 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.commandline;
import org.flywaydb.core.Flyway;
import org.flywaydb.core.api.FlywayException;
import org.flywaydb.core.api.MigrationInfo;
import org.flywaydb.core.api.MigrationInfoService;
import org.flywaydb.core.api.MigrationVersion;
import org.flywaydb.core.api.logging.Log;
import org.flywaydb.core.api.logging.LogFactory;
import org.flywaydb.core.internal.configuration.ConfigUtils;
import org.flywaydb.core.internal.info.MigrationInfoDumper;
import org.flywaydb.core.internal.util.ClassUtils;
import org.flywaydb.core.internal.util.StringUtils;
import org.flywaydb.core.internal.util.VersionPrinter;
import org.flywaydb.core.internal.util.logging.console.ConsoleLog.Level;
import org.flywaydb.core.internal.util.logging.console.ConsoleLogCreator;
import java.io.Console;
import java.io.File;
import java.io.FilenameFilter;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.Properties;
/**
* Main class and central entry point of the Flyway command-line tool.
*/
public class Main {
private static Log LOG;
private static List VALID_OPERATIONS_AND_FLAGS = Arrays.asList("-X", "-q", "-n", "-v", "-?",
"help", "migrate", "clean", "info", "validate", "undo", "baseline", "repair");
/**
* Initializes the logging.
*
* @param level The minimum level to log at.
*/
static void initLogging(Level level) {
LogFactory.setFallbackLogCreator(new ConsoleLogCreator(level));
LOG = LogFactory.getLog(Main.class);
}
/**
* Main method.
*
* @param args The command-line arguments.
*/
public static void main(String[] args) {
Level logLevel = getLogLevel(args);
initLogging(logLevel);
try {
printVersion();
if (isPrintVersionAndExit(args)) {
System.exit(0);
}
List operations = determineOperations(args);
if (operations.isEmpty() || operations.contains("help") || isFlagSet(args, "-?")) {
printUsage();
return;
}
validateArgs(args);
Map envVars = ConfigUtils.environmentVariablesToPropertyMap();
Properties properties = new Properties();
initializeDefaults(properties);
loadConfigurationFromConfigFiles(properties, args, envVars);
properties.putAll(envVars);
overrideConfigurationWithArgs(properties, args);
if (!isSuppressPrompt(args)) {
promptForCredentialsIfMissing(properties);
}
dumpConfiguration(properties);
ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
classLoader = loadJdbcDrivers(classLoader);
classLoader = loadJavaMigrationsFromJarDirs(classLoader, properties);
filterProperties(properties);
Flyway flyway = new Flyway(classLoader);
flyway.configure(properties);
for (String operation : operations) {
executeOperation(flyway, operation);
}
} catch (Exception e) {
if (logLevel == Level.DEBUG) {
LOG.error("Unexpected error", e);
} else {
if (e instanceof FlywayException) {
LOG.error(e.getMessage());
} else {
LOG.error(e.toString());
}
}
System.exit(1);
}
}
static void validateArgs(String[] args) {
for (String arg : args) {
if (!isPropertyArgument(arg) && !VALID_OPERATIONS_AND_FLAGS.contains(arg)) {
throw new FlywayException("Invalid argument: " + arg);
}
}
}
private static boolean isPrintVersionAndExit(String[] args) {
return isFlagSet(args, "-v");
}
private static boolean isSuppressPrompt(String[] args) {
return isFlagSet(args, "-n");
}
private static boolean isFlagSet(String[] args, String flag) {
for (String arg : args) {
if (flag.equals(arg)) {
return true;
}
}
return false;
}
/**
* Executes this operation on this Flyway instance.
*
* @param flyway The Flyway instance.
* @param operation The operation to execute.
*/
private static void executeOperation(Flyway flyway, String operation) {
if ("clean".equals(operation)) {
flyway.clean();
} else if ("baseline".equals(operation)) {
flyway.baseline();
} else if ("migrate".equals(operation)) {
flyway.migrate();
} else if ("undo".equals(operation)) {
flyway.undo();
} else if ("validate".equals(operation)) {
flyway.validate();
} else if ("info".equals(operation)) {
MigrationInfoService info = flyway.info();
MigrationInfo current = info.current();
MigrationVersion currentSchemaVersion = current == null ? MigrationVersion.EMPTY : current.getVersion();
LOG.info("Schema version: " + currentSchemaVersion);
LOG.info("");
LOG.info(MigrationInfoDumper.dumpToAsciiTable(info.all()));
} else if ("repair".equals(operation)) {
flyway.repair();
} else {
LOG.error("Invalid operation: " + operation);
printUsage();
System.exit(1);
}
}
/**
* Checks the desired log level.
*
* @param args The command-line arguments.
* @return The desired log level.
*/
private static Level getLogLevel(String[] args) {
for (String arg : args) {
if ("-X".equals(arg)) {
return Level.DEBUG;
}
if ("-q".equals(arg)) {
return Level.WARN;
}
}
return Level.INFO;
}
/**
* Initializes the properties with the default configuration for the command-line tool.
*
* @param properties The properties object to initialize.
*/
private static void initializeDefaults(Properties properties) {
properties.put(ConfigUtils.LOCATIONS, "filesystem:" + new File(getInstallationDir(), "sql").getAbsolutePath());
properties.put(ConfigUtils.JAR_DIRS, new File(getInstallationDir(), "jars").getAbsolutePath());
}
/**
* Filters there properties to remove the Flyway Commandline-specific ones.
*
* @param properties The properties to filter.
*/
private static void filterProperties(Properties properties) {
properties.remove(ConfigUtils.JAR_DIRS);
properties.remove(ConfigUtils.CONFIG_FILE);
properties.remove(ConfigUtils.CONFIG_FILES);
properties.remove(ConfigUtils.CONFIG_FILE_ENCODING);
}
/**
* Prints the version number on the console.
*/
private static void printVersion() {
VersionPrinter.printVersion();
LOG.info("");
LOG.debug("Java " + System.getProperty("java.version") + " (" + System.getProperty("java.vendor") + ")");
LOG.debug(System.getProperty("os.name") + " " + System.getProperty("os.version") + " " + System.getProperty("os.arch") + "\n");
}
/**
* Prints the usage instructions on the console.
*/
private static void printUsage() {
LOG.info("Usage");
LOG.info("=====");
LOG.info("");
LOG.info("flyway [options] command");
LOG.info("");
LOG.info("By default, the configuration will be read from conf/flyway.conf.");
LOG.info("Options passed from the command-line override the configuration.");
LOG.info("");
LOG.info("Commands");
LOG.info("--------");
LOG.info("migrate : Migrates the database");
LOG.info("clean : Drops all objects in the configured schemas");
LOG.info("info : Prints the information about applied, current and pending migrations");
LOG.info("validate : Validates the applied migrations against the ones on the classpath");
LOG.info("undo : [" + "pro] Undoes the most recently applied versioned migration");
LOG.info("baseline : Baselines an existing database at the baselineVersion");
LOG.info("repair : Repairs the schema history table");
LOG.info("");
LOG.info("Options (Format: -key=value)");
LOG.info("-------");
LOG.info("driver : Fully qualified classname of the JDBC driver");
LOG.info("url : Jdbc url to use to connect to the database");
LOG.info("user : User to use to connect to the database");
LOG.info("password : Password to use to connect to the database");
LOG.info("schemas : Comma-separated list of the schemas managed by Flyway");
LOG.info("table : Name of Flyway's schema history table");
LOG.info("locations : Classpath locations to scan recursively for migrations");
LOG.info("resolvers : Comma-separated list of custom MigrationResolvers");
LOG.info("skipDefaultResolvers : Skips default resolvers (jdbc, sql and Spring-jdbc)");
LOG.info("sqlMigrationPrefix : File name prefix for versioned SQL migrations");
LOG.info("undoSqlMigrationPrefix : [" + "pro] File name prefix for undo SQL migrations");
LOG.info("repeatableSqlMigrationPrefix : File name prefix for repeatable SQL migrations");
LOG.info("sqlMigrationSeparator : File name separator for SQL migrations");
LOG.info("sqlMigrationSuffixes : Comma-separated list of file name suffixes for SQL migrations");
LOG.info("stream : [" + "pro] Stream SQL migrations when executing them");
LOG.info("batch : [" + "pro] Batch SQL statements when executing them");
LOG.info("mixed : Allow mixing transactional and non-transactional statements");
LOG.info("encoding : Encoding of SQL migrations");
LOG.info("placeholderReplacement : Whether placeholders should be replaced");
LOG.info("placeholders : Placeholders to replace in sql migrations");
LOG.info("placeholderPrefix : Prefix of every placeholder");
LOG.info("placeholderSuffix : Suffix of every placeholder");
LOG.info("installedBy : Username that will be recorded in the schema history table");
LOG.info("target : Target version up to which Flyway should use migrations");
LOG.info("outOfOrder : Allows migrations to be run \"out of order\"");
LOG.info("callbacks : Comma-separated list of FlywayCallback classes");
LOG.info("skipDefaultCallbacks : Skips default callbacks (sql)");
LOG.info("validateOnMigrate : Validate when running migrate");
LOG.info("ignoreMissingMigrations : Allow missing migrations when validating");
LOG.info("ignoreIgnoredMigrations : Allow ignored migrations when validating");
LOG.info("ignoreFutureMigrations : Allow future migrations when validating");
LOG.info("cleanOnValidationError : Automatically clean on a validation error");
LOG.info("cleanDisabled : Whether to disable clean");
LOG.info("baselineVersion : Version to tag schema with when executing baseline");
LOG.info("baselineDescription : Description to tag schema with when executing baseline");
LOG.info("baselineOnMigrate : Baseline on migrate against uninitialized non-empty schema");
LOG.info("configFiles : Comma-separated list of config files to use");
LOG.info("configFileEncoding : Encoding to use when loading the config files");
LOG.info("jarDirs : Comma-separated list of dirs for Jdbc drivers & Java migrations");
LOG.info("dryRunOutput : [" + "pro] File where to output the SQL statements of a migration dry run");
LOG.info("errorHandlers : [" + "pro] Comma-separated list of handlers for errors and warnings");
LOG.info("oracle.sqlplus : [" + "pro] Oracle SQL*Plus command support");
LOG.info("");
LOG.info("Flags");
LOG.info("-----");
LOG.info("-X : Print debug output");
LOG.info("-q : Suppress all output, except for errors and warnings");
LOG.info("-n : Suppress prompting for a user and password");
LOG.info("-v : Print the Flyway version and exit");
LOG.info("-? : Print this usage info and exit");
LOG.info("");
LOG.info("Example");
LOG.info("-------");
LOG.info("flyway -user=myuser -password=s3cr3t -url=jdbc:h2:mem -placeholders.abc=def migrate");
LOG.info("");
LOG.info("More info at https://flywaydb.org/documentation/commandline");
}
/**
* Loads all the driver jars contained in the drivers folder. (For Jdbc drivers)
*
* @param classLoader The current ClassLoader.
* @return The new ClassLoader containing the additional driver jars.
* @throws IOException When the jars could not be loaded.
*/
private static ClassLoader loadJdbcDrivers(ClassLoader classLoader) throws IOException {
File driversDir = new File(getInstallationDir(), "drivers");
File[] files = driversDir.listFiles(new FilenameFilter() {
public boolean accept(File dir, String name) {
return name.endsWith(".jar");
}
});
// see javadoc of listFiles(): null if given path is not a real directory
if (files == null) {
LOG.debug("Directory for Jdbc Drivers not found: " + driversDir.getAbsolutePath());
return classLoader;
}
for (File file : files) {
classLoader = ClassUtils.addJarOrDirectoryToClasspath(classLoader, file.getPath());
}
return classLoader;
}
/**
* Loads all the jars contained in the jars folder. (For Java Migrations)
*
* @param classLoader The current ClassLoader.
* @param properties The configured properties.
* @return The new ClassLoader containing the additional jars.
* @throws IOException When the jars could not be loaded.
*/
private static ClassLoader loadJavaMigrationsFromJarDirs(ClassLoader classLoader, Properties properties) throws IOException {
String jarDirs = properties.getProperty(ConfigUtils.JAR_DIRS);
if (!StringUtils.hasLength(jarDirs)) {
return classLoader;
}
jarDirs = jarDirs.replace(File.pathSeparator, ",");
String[] dirs = StringUtils.tokenizeToStringArray(jarDirs, ",");
for (String dirName : dirs) {
File dir = new File(dirName);
File[] files = dir.listFiles(new FilenameFilter() {
public boolean accept(File dir, String name) {
return name.endsWith(".jar");
}
});
// see javadoc of listFiles(): null if given path is not a real directory
if (files == null) {
LOG.error("Directory for Java Migrations not found: " + dirName);
System.exit(1);
}
for (File file : files) {
classLoader = ClassUtils.addJarOrDirectoryToClasspath(classLoader, file.getPath());
}
}
return classLoader;
}
/**
* Loads the configuration from the various possible locations.
*
* @param properties The properties object to load to configuration into.
* @param args The command-line arguments passed in.
* @param envVars The environment variables, converted into properties.
*/
/* private -> for testing */
static void loadConfigurationFromConfigFiles(Properties properties, String[] args, Map envVars) {
String encoding = determineConfigurationFileEncoding(args, envVars);
properties.putAll(ConfigUtils.loadConfigurationFile(new File(getInstallationDir() + "/conf/" + ConfigUtils.CONFIG_FILE_NAME), encoding, false));
properties.putAll(ConfigUtils.loadConfigurationFile(new File(System.getProperty("user.home") + "/" + ConfigUtils.CONFIG_FILE_NAME), encoding, false));
properties.putAll(ConfigUtils.loadConfigurationFile(new File(ConfigUtils.CONFIG_FILE_NAME), encoding, false));
for (File configFile : determineConfigFilesFromArgs(args, envVars)) {
properties.putAll(ConfigUtils.loadConfigurationFile(configFile, encoding, true));
}
}
/**
* If no user or password has been provided, prompt for it. If you want to avoid the prompt,
* pass in an empty user or password.
*
* @param properties The properties object to load to configuration into.
*/
private static void promptForCredentialsIfMissing(Properties properties) {
Console console = System.console();
if (console == null) {
// We are running in an automated build. Prompting is not possible.
return;
}
if (!properties.containsKey(ConfigUtils.URL)) {
// URL is not set. We are doomed for failure anyway.
return;
}
if (!properties.containsKey(ConfigUtils.USER)) {
properties.put(ConfigUtils.USER, console.readLine("Database user: "));
}
if (!properties.containsKey(ConfigUtils.PASSWORD)) {
char[] password = console.readPassword("Database password: ");
properties.put(ConfigUtils.PASSWORD, password == null ? "" : String.valueOf(password));
}
}
/**
* Dumps the configuration to the console when debug output is activated.
*
* @param properties The configured properties.
*/
private static void dumpConfiguration(Properties properties) {
LOG.debug("Using configuration:");
for (Map.Entry
© 2015 - 2025 Weber Informatics LLC | Privacy Policy