io.github.jhipster.config.liquibase.AsyncSpringLiquibase Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jhipster-framework Show documentation
Show all versions of jhipster-framework Show documentation
Server-side library used by applications created with the JHipster generator, see https://www.jhipster.tech/ for more information on JHipster
/*
* Copyright 2016-2018 the original author or authors from the JHipster project.
*
* This file is part of the JHipster project, see https://www.jhipster.tech/
* for more information.
*
* 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 io.github.jhipster.config.liquibase;
import static io.github.jhipster.config.JHipsterConstants.SPRING_PROFILE_DEVELOPMENT;
import static io.github.jhipster.config.JHipsterConstants.SPRING_PROFILE_HEROKU;
import static io.github.jhipster.config.JHipsterConstants.SPRING_PROFILE_NO_LIQUIBASE;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.core.env.Environment;
import org.springframework.core.task.TaskExecutor;
import org.springframework.util.StopWatch;
import liquibase.exception.LiquibaseException;
import liquibase.integration.spring.SpringLiquibase;
import java.sql.Connection;
import java.sql.SQLException;
/**
* Specific liquibase.integration.spring.SpringLiquibase that will update the database asynchronously. By default,
* this asynchronous version only works when using the "dev" profile.
The standard
* liquibase.integration.spring.SpringLiquibase starts Liquibase in the current thread:
- This is needed if you
* want to do some database requests at startup
- This ensure that the database is ready when the application
* starts
But as this is a rather slow process, we use this asynchronous version to speed up our start-up
* time: - On a recent MacBook Pro, start-up time is down from 14 seconds to 8 seconds
- In production,
* this can help your application run on platforms like Heroku, where it must start/restart very quickly
*/
public class AsyncSpringLiquibase extends SpringLiquibase {
public static final String DISABLED_MESSAGE = "Liquibase is disabled";
public static final String STARTING_ASYNC_MESSAGE =
"Starting Liquibase asynchronously, your database might not be ready at startup!";
public static final String STARTING_SYNC_MESSAGE = "Starting Liquibase synchronously";
public static final String STARTED_MESSAGE = "Liquibase has updated your database in {} ms";
public static final String EXCEPTION_MESSAGE = "Liquibase could not start correctly, your database is NOT ready: " +
"{}";
public static final long SLOWNESS_THRESHOLD = 5; // seconds
public static final String SLOWNESS_MESSAGE = "Warning, Liquibase took more than {} seconds to start up!";
// named "logger" because there is already a field called "log" in "SpringLiquibase"
private final Logger logger = LoggerFactory.getLogger(AsyncSpringLiquibase.class);
private final TaskExecutor taskExecutor;
private final Environment env;
public AsyncSpringLiquibase(@Qualifier("taskExecutor") TaskExecutor taskExecutor, Environment env) {
this.taskExecutor = taskExecutor;
this.env = env;
}
@Override
public void afterPropertiesSet() throws LiquibaseException {
if (!env.acceptsProfiles(SPRING_PROFILE_NO_LIQUIBASE)) {
if (env.acceptsProfiles(SPRING_PROFILE_DEVELOPMENT, SPRING_PROFILE_HEROKU)) {
// Prevent Thread Lock with spring-cloud-context GenericScope
// https://github.com/spring-cloud/spring-cloud-commons/commit/aaa7288bae3bb4d6fdbef1041691223238d77b7b#diff-afa0715eafc2b0154475fe672dab70e4R328
try (Connection connection = getDataSource().getConnection()) {
taskExecutor.execute(() -> {
try {
logger.warn(STARTING_ASYNC_MESSAGE);
initDb();
} catch (LiquibaseException e) {
logger.error(EXCEPTION_MESSAGE, e.getMessage(), e);
}
});
} catch (SQLException e) {
logger.error(EXCEPTION_MESSAGE, e.getMessage(), e);
}
} else {
logger.debug(STARTING_SYNC_MESSAGE);
initDb();
}
} else {
logger.debug(DISABLED_MESSAGE);
}
}
protected void initDb() throws LiquibaseException {
StopWatch watch = new StopWatch();
watch.start();
super.afterPropertiesSet();
watch.stop();
logger.debug(STARTED_MESSAGE, watch.getTotalTimeMillis());
if (watch.getTotalTimeMillis() > SLOWNESS_THRESHOLD * 1000L) {
logger.warn(SLOWNESS_MESSAGE, SLOWNESS_THRESHOLD);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy