All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.springframework.boot.autoconfigure.flyway.FlywayAutoConfiguration Maven / Gradle / Ivy

There is a newer version: 3.3.3
Show newest version
/*
 * Copyright 2012-2018 the original author or authors.
 *
 * 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.springframework.boot.autoconfigure.flyway;

import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.function.Supplier;
import java.util.stream.Collectors;

import javax.persistence.EntityManagerFactory;
import javax.sql.DataSource;

import org.flywaydb.core.Flyway;
import org.flywaydb.core.api.MigrationVersion;
import org.flywaydb.core.api.callback.Callback;
import org.flywaydb.core.api.callback.FlywayCallback;
import org.flywaydb.core.api.configuration.FluentConfiguration;

import org.springframework.beans.factory.ObjectProvider;
import org.springframework.boot.autoconfigure.AutoConfigureAfter;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.boot.autoconfigure.condition.ConditionalOnBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.boot.autoconfigure.data.jpa.EntityManagerFactoryDependsOnPostProcessor;
import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
import org.springframework.boot.autoconfigure.jdbc.DataSourceProperties;
import org.springframework.boot.autoconfigure.jdbc.JdbcOperationsDependsOnPostProcessor;
import org.springframework.boot.autoconfigure.jdbc.JdbcTemplateAutoConfiguration;
import org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration;
import org.springframework.boot.context.properties.ConfigurationPropertiesBinding;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.boot.context.properties.PropertyMapper;
import org.springframework.boot.jdbc.DatabaseDriver;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.convert.TypeDescriptor;
import org.springframework.core.convert.converter.GenericConverter;
import org.springframework.core.io.ResourceLoader;
import org.springframework.jdbc.core.JdbcOperations;
import org.springframework.jdbc.support.JdbcUtils;
import org.springframework.jdbc.support.MetaDataAccessException;
import org.springframework.orm.jpa.AbstractEntityManagerFactoryBean;
import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean;
import org.springframework.util.Assert;
import org.springframework.util.CollectionUtils;
import org.springframework.util.ObjectUtils;
import org.springframework.util.StringUtils;

/**
 * {@link EnableAutoConfiguration Auto-configuration} for Flyway database migrations.
 *
 * @author Dave Syer
 * @author Phillip Webb
 * @author Vedran Pavic
 * @author Stephane Nicoll
 * @author Jacques-Etienne Beaudet
 * @author Eddú Meléndez
 * @author Dominic Gunn
 * @since 1.1.0
 */
@SuppressWarnings("deprecation")
@Configuration
@ConditionalOnClass(Flyway.class)
@ConditionalOnBean(DataSource.class)
@ConditionalOnProperty(prefix = "spring.flyway", name = "enabled", matchIfMissing = true)
@AutoConfigureAfter({ DataSourceAutoConfiguration.class,
		JdbcTemplateAutoConfiguration.class, HibernateJpaAutoConfiguration.class })
public class FlywayAutoConfiguration {

	@Bean
	@ConfigurationPropertiesBinding
	public StringOrNumberToMigrationVersionConverter stringOrNumberMigrationVersionConverter() {
		return new StringOrNumberToMigrationVersionConverter();
	}

	@Bean
	public FlywaySchemaManagementProvider flywayDefaultDdlModeProvider(
			ObjectProvider flyways) {
		return new FlywaySchemaManagementProvider(flyways);
	}

	@Configuration
	@ConditionalOnMissingBean(Flyway.class)
	@EnableConfigurationProperties({ DataSourceProperties.class, FlywayProperties.class })
	public static class FlywayConfiguration {

		private final FlywayProperties properties;

		private final DataSourceProperties dataSourceProperties;

		private final ResourceLoader resourceLoader;

		private final DataSource dataSource;

		private final DataSource flywayDataSource;

		private final FlywayMigrationStrategy migrationStrategy;

		private final List configurationCustomizers;

		private final List callbacks;

		private final List flywayCallbacks;

		public FlywayConfiguration(FlywayProperties properties,
				DataSourceProperties dataSourceProperties, ResourceLoader resourceLoader,
				ObjectProvider dataSource,
				@FlywayDataSource ObjectProvider flywayDataSource,
				ObjectProvider migrationStrategy,
				ObjectProvider fluentConfigurationCustomizers,
				ObjectProvider callbacks,
				ObjectProvider flywayCallbacks) {
			this.properties = properties;
			this.dataSourceProperties = dataSourceProperties;
			this.resourceLoader = resourceLoader;
			this.dataSource = dataSource.getIfUnique();
			this.flywayDataSource = flywayDataSource.getIfAvailable();
			this.migrationStrategy = migrationStrategy.getIfAvailable();
			this.configurationCustomizers = fluentConfigurationCustomizers.orderedStream()
					.collect(Collectors.toList());
			this.callbacks = callbacks.orderedStream().collect(Collectors.toList());
			this.flywayCallbacks = flywayCallbacks.orderedStream()
					.collect(Collectors.toList());
		}

		@Bean
		public Flyway flyway() {
			FluentConfiguration configuration = new FluentConfiguration();
			DataSource dataSource = configureDataSource(configuration);
			checkLocationExists(dataSource);
			configureProperties(configuration);
			configureCallbacks(configuration);
			this.configurationCustomizers
					.forEach((customizer) -> customizer.customize(configuration));
			Flyway flyway = configuration.load();
			configureFlywayCallbacks(flyway);
			return flyway;
		}

		private DataSource configureDataSource(FluentConfiguration configuration) {
			if (this.properties.isCreateDataSource()) {
				String url = getProperty(this.properties::getUrl,
						this.dataSourceProperties::getUrl);
				String user = getProperty(this.properties::getUser,
						this.dataSourceProperties::getUsername);
				String password = getProperty(this.properties::getPassword,
						this.dataSourceProperties::getPassword);
				configuration.dataSource(url, user, password);
				if (!CollectionUtils.isEmpty(this.properties.getInitSqls())) {
					String initSql = StringUtils.collectionToDelimitedString(
							this.properties.getInitSqls(), "\n");
					configuration.initSql(initSql);
				}
			}
			else if (this.flywayDataSource != null) {
				configuration.dataSource(this.flywayDataSource);
			}
			else {
				configuration.dataSource(this.dataSource);
			}
			return configuration.getDataSource();
		}

		private void checkLocationExists(DataSource dataSource) {
			if (this.properties.isCheckLocation()) {
				String[] locations = new LocationResolver(dataSource)
						.resolveLocations(this.properties.getLocations());
				Assert.state(locations.length != 0,
						"Migration script locations not configured");
				boolean exists = hasAtLeastOneLocation(locations);
				Assert.state(exists, () -> "Cannot find migrations location in: "
						+ Arrays.asList(locations)
						+ " (please add migrations or check your Flyway configuration)");
			}
		}

		private void configureProperties(FluentConfiguration configuration) {
			PropertyMapper map = PropertyMapper.get().alwaysApplyingWhenNonNull();
			String[] locations = new LocationResolver(configuration.getDataSource())
					.resolveLocations(this.properties.getLocations());
			map.from(locations).to(configuration::locations);
			map.from(this.properties.getEncoding()).to(configuration::encoding);
			map.from(this.properties.getConnectRetries())
					.to(configuration::connectRetries);
			map.from(this.properties.getSchemas()).as(StringUtils::toStringArray)
					.to(configuration::schemas);
			map.from(this.properties.getTable()).to(configuration::table);
			map.from(this.properties.getBaselineDescription())
					.to(configuration::baselineDescription);
			map.from(this.properties.getBaselineVersion())
					.to(configuration::baselineVersion);
			map.from(this.properties.getInstalledBy()).to(configuration::installedBy);
			map.from(this.properties.getPlaceholders()).to(configuration::placeholders);
			map.from(this.properties.getPlaceholderPrefix())
					.to(configuration::placeholderPrefix);
			map.from(this.properties.getPlaceholderSuffix())
					.to(configuration::placeholderSuffix);
			map.from(this.properties.isPlaceholderReplacement())
					.to(configuration::placeholderReplacement);
			map.from(this.properties.getSqlMigrationPrefix())
					.to(configuration::sqlMigrationPrefix);
			map.from(this.properties.getSqlMigrationSuffixes())
					.as(StringUtils::toStringArray)
					.to(configuration::sqlMigrationSuffixes);
			map.from(this.properties.getSqlMigrationSeparator())
					.to(configuration::sqlMigrationSeparator);
			map.from(this.properties.getRepeatableSqlMigrationPrefix())
					.to(configuration::repeatableSqlMigrationPrefix);
			map.from(this.properties.getTarget()).to(configuration::target);
			map.from(this.properties.isBaselineOnMigrate())
					.to(configuration::baselineOnMigrate);
			map.from(this.properties.isCleanDisabled()).to(configuration::cleanDisabled);
			map.from(this.properties.isCleanOnValidationError())
					.to(configuration::cleanOnValidationError);
			map.from(this.properties.isGroup()).to(configuration::group);
			map.from(this.properties.isIgnoreMissingMigrations())
					.to(configuration::ignoreMissingMigrations);
			map.from(this.properties.isIgnoreIgnoredMigrations())
					.to(configuration::ignoreIgnoredMigrations);
			map.from(this.properties.isIgnorePendingMigrations())
					.to(configuration::ignorePendingMigrations);
			map.from(this.properties.isIgnoreFutureMigrations())
					.to(configuration::ignoreFutureMigrations);
			map.from(this.properties.isMixed()).to(configuration::mixed);
			map.from(this.properties.isOutOfOrder()).to(configuration::outOfOrder);
			map.from(this.properties.isSkipDefaultCallbacks())
					.to(configuration::skipDefaultCallbacks);
			map.from(this.properties.isSkipDefaultResolvers())
					.to(configuration::skipDefaultResolvers);
			map.from(this.properties.isValidateOnMigrate())
					.to(configuration::validateOnMigrate);
		}

		private void configureCallbacks(FluentConfiguration configuration) {
			if (!this.callbacks.isEmpty()) {
				configuration.callbacks(this.callbacks.toArray(new Callback[0]));
			}
		}

		private void configureFlywayCallbacks(Flyway flyway) {
			if (!this.flywayCallbacks.isEmpty()) {
				if (!this.callbacks.isEmpty()) {
					throw new IllegalStateException(
							"Found a mixture of Callback and FlywayCallback beans."
									+ " One type must be used exclusively.");
				}
				flyway.setCallbacks(this.flywayCallbacks.toArray(new FlywayCallback[0]));
			}
		}

		private String getProperty(Supplier property,
				Supplier defaultValue) {
			String value = property.get();
			return (value != null) ? value : defaultValue.get();
		}

		private boolean hasAtLeastOneLocation(String... locations) {
			for (String location : locations) {
				if (this.resourceLoader.getResource(normalizePrefix(location)).exists()) {
					return true;
				}
			}
			return false;
		}

		private String normalizePrefix(String location) {
			return location.replace("filesystem:", "file:");
		}

		@Bean
		@ConditionalOnMissingBean
		public FlywayMigrationInitializer flywayInitializer(Flyway flyway) {
			return new FlywayMigrationInitializer(flyway, this.migrationStrategy);
		}

		/**
		 * Additional configuration to ensure that {@link EntityManagerFactory} beans
		 * depend on the {@code flywayInitializer} bean.
		 */
		@Configuration
		@ConditionalOnClass(LocalContainerEntityManagerFactoryBean.class)
		@ConditionalOnBean(AbstractEntityManagerFactoryBean.class)
		protected static class FlywayInitializerJpaDependencyConfiguration
				extends EntityManagerFactoryDependsOnPostProcessor {

			public FlywayInitializerJpaDependencyConfiguration() {
				super("flywayInitializer");
			}

		}

		/**
		 * Additional configuration to ensure that {@link JdbcOperations} beans depend on
		 * the {@code flywayInitializer} bean.
		 */
		@Configuration
		@ConditionalOnClass(JdbcOperations.class)
		@ConditionalOnBean(JdbcOperations.class)
		protected static class FlywayInitializerJdbcOperationsDependencyConfiguration
				extends JdbcOperationsDependsOnPostProcessor {

			public FlywayInitializerJdbcOperationsDependencyConfiguration() {
				super("flywayInitializer");
			}

		}

	}

	/**
	 * Additional configuration to ensure that {@link EntityManagerFactory} beans depend
	 * on the {@code flyway} bean.
	 */
	@Configuration
	@ConditionalOnClass(LocalContainerEntityManagerFactoryBean.class)
	@ConditionalOnBean(AbstractEntityManagerFactoryBean.class)
	protected static class FlywayJpaDependencyConfiguration
			extends EntityManagerFactoryDependsOnPostProcessor {

		public FlywayJpaDependencyConfiguration() {
			super("flyway");
		}

	}

	/**
	 * Additional configuration to ensure that {@link JdbcOperations} beans depend on the
	 * {@code flyway} bean.
	 */
	@Configuration
	@ConditionalOnClass(JdbcOperations.class)
	@ConditionalOnBean(JdbcOperations.class)
	protected static class FlywayJdbcOperationsDependencyConfiguration
			extends JdbcOperationsDependsOnPostProcessor {

		public FlywayJdbcOperationsDependencyConfiguration() {
			super("flyway");
		}

	}

	private static class LocationResolver {

		private static final String VENDOR_PLACEHOLDER = "{vendor}";

		private final DataSource dataSource;

		LocationResolver(DataSource dataSource) {
			this.dataSource = dataSource;
		}

		public String[] resolveLocations(Collection locations) {
			return resolveLocations(StringUtils.toStringArray(locations));
		}

		public String[] resolveLocations(String[] locations) {
			if (usesVendorLocation(locations)) {
				DatabaseDriver databaseDriver = getDatabaseDriver();
				return replaceVendorLocations(locations, databaseDriver);
			}
			return locations;
		}

		private String[] replaceVendorLocations(String[] locations,
				DatabaseDriver databaseDriver) {
			if (databaseDriver == DatabaseDriver.UNKNOWN) {
				return locations;
			}
			String vendor = databaseDriver.getId();
			return Arrays.stream(locations)
					.map((location) -> location.replace(VENDOR_PLACEHOLDER, vendor))
					.toArray(String[]::new);
		}

		private DatabaseDriver getDatabaseDriver() {
			try {
				String url = JdbcUtils.extractDatabaseMetaData(this.dataSource, "getURL");
				return DatabaseDriver.fromJdbcUrl(url);
			}
			catch (MetaDataAccessException ex) {
				throw new IllegalStateException(ex);
			}

		}

		private boolean usesVendorLocation(String... locations) {
			for (String location : locations) {
				if (location.contains(VENDOR_PLACEHOLDER)) {
					return true;
				}
			}
			return false;
		}

	}

	/**
	 * Convert a String or Number to a {@link MigrationVersion}.
	 */
	private static class StringOrNumberToMigrationVersionConverter
			implements GenericConverter {

		private static final Set CONVERTIBLE_TYPES;

		static {
			Set types = new HashSet<>(2);
			types.add(new ConvertiblePair(String.class, MigrationVersion.class));
			types.add(new ConvertiblePair(Number.class, MigrationVersion.class));
			CONVERTIBLE_TYPES = Collections.unmodifiableSet(types);
		}

		@Override
		public Set getConvertibleTypes() {
			return CONVERTIBLE_TYPES;
		}

		@Override
		public Object convert(Object source, TypeDescriptor sourceType,
				TypeDescriptor targetType) {
			String value = ObjectUtils.nullSafeToString(source);
			return MigrationVersion.fromVersion(value);
		}

	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy