DbSchemas.java
/*-
* ========================LICENSE_START=================================
* flyway-core
* ========================================================================
* Copyright (C) 2010 - 2025 Red Gate Software Ltd
* ========================================================================
* 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.
* =========================LICENSE_END==================================
*/
package org.flywaydb.core.internal.command;
import lombok.CustomLog;
import org.flywaydb.core.api.FlywayException;
import org.flywaydb.core.api.callback.Event;
import org.flywaydb.core.internal.callback.CallbackExecutor;
import org.flywaydb.core.internal.database.base.Connection;
import org.flywaydb.core.internal.database.base.Database;
import org.flywaydb.core.internal.database.base.Schema;
import org.flywaydb.core.internal.jdbc.ExecutionTemplateFactory;
import org.flywaydb.core.internal.schemahistory.SchemaHistory;
import java.util.ArrayList;
import java.util.List;
/**
* Handles Flyway's automatic schema creation.
*/
@CustomLog
public class DbSchemas {
/**
* The database connection to use for accessing the schema history table.
*/
private final Connection connection;
/**
* The schemas managed by Flyway.
*/
private final Schema[] schemas;
/**
* The schema history table.
*/
private final SchemaHistory schemaHistory;
/**
* The database.
*/
private final Database database;
/**
* The callback executor.
*/
private final CallbackExecutor callbackExecutor;
/**
* Creates a new DbSchemas.
*
* @param database The database to use.
* @param schemas The schemas managed by Flyway.
* @param schemaHistory The schema history table.
*/
public DbSchemas(Database database, Schema[] schemas, SchemaHistory schemaHistory, CallbackExecutor callbackExecutor) {
this.database = database;
this.connection = database.getMainConnection();
this.schemas = schemas;
this.schemaHistory = schemaHistory;
this.callbackExecutor = callbackExecutor;
}
/**
* Creates the schemas.
*
* @param baseline Whether to include the creation of a baseline marker.
*/
public void create(final boolean baseline) {
callbackExecutor.onEvent(Event.CREATE_SCHEMA);
callbackExecutor.onEvent(Event.BEFORE_CREATE_SCHEMA);
int retries = 0;
while (true) {
try {
ExecutionTemplateFactory.createExecutionTemplate(connection.getJdbcConnection(), database).execute(() -> {
List<Schema> createdSchemas = new ArrayList<>();
for (Schema schema : schemas) {
if (!schema.exists()) {
if (schema.getName() == null) {
throw new FlywayException("Unable to determine schema for the schema history table." +
" Set a default schema for the connection or specify one using the defaultSchema property!");
}
LOG.debug("Creating schema: " + schema);
schema.create();
createdSchemas.add(schema);
} else {
LOG.debug("Skipping creation of existing schema: " + schema);
}
}
if (!createdSchemas.isEmpty()) {
schemaHistory.create(baseline);
schemaHistory.addSchemasMarker(createdSchemas.toArray(new Schema[0]));
}
return null;
});
return;
} catch (RuntimeException e) {
if (++retries >= 10) {
throw e;
}
try {
LOG.debug("Schema creation failed. Retrying in 1 sec ...");
Thread.sleep(1000);
} catch (InterruptedException e1) {
// Ignore
}
}
}
}
}