org.hibernate.testing.cleaner.AbstractMySQLDatabaseCleaner Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of hibernate-testing Show documentation
Show all versions of hibernate-testing Show documentation
Support for testing Hibernate ORM functionality
/*
* Hibernate, Relational Persistence for Idiomatic Java
*
* License: GNU Lesser General Public License (LGPL), version 2.1 or later.
* See the lgpl.txt file in the root directory or .
*/
package org.hibernate.testing.cleaner;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.function.Function;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
* @author Christian Beikov
*/
public abstract class AbstractMySQLDatabaseCleaner implements DatabaseCleaner {
private static final Logger LOG = Logger.getLogger( AbstractMySQLDatabaseCleaner.class.getName() );
private static final String SYSTEM_SCHEMAS = "'information_schema'," +
"'mysql'," +
"'sys'," +
"'performance_schema'";
private final List ignoredTables = new ArrayList<>();
private final Map> clearingSqlsPerSchema = new HashMap<>();
@Override
public void addIgnoredTable(String tableName) {
ignoredTables.add( tableName );
}
@Override
public void clearAllSchemas(Connection connection) {
try {
clearSchema( connection, connection.getSchema() );
}
catch (SQLException e) {
throw new RuntimeException( e );
}
}
@Override
public void clearSchema(Connection connection, String schemaName) {
clearSchema0( connection, schemaName );
}
private void clearSchema0(Connection c, String schemaName) {
clearingSqlsPerSchema.remove( schemaName );
try (Statement s = c.createStatement()) {
// Collect schema names
LOG.log( Level.FINEST, "Collect table names: START" );
ResultSet rs = s.executeQuery(
"SELECT TABLE_SCHEMA, TABLE_NAME FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA = '" + schemaName + "'" );
StringBuilder sb = new StringBuilder( "DROP TABLE " );
if ( rs.next() ) {
do {
String tableSchema = rs.getString( 1 );
String tableName = rs.getString( 2 );
sb.append( tableSchema );
sb.append( '.' );
sb.append( tableName );
sb.append( ',' );
} while ( rs.next() );
}
else {
// Nothing to clear since there are no tables
return;
}
sb.setCharAt( sb.length() - 1, ' ' );
LOG.log( Level.FINEST, "Collect table names: END" );
// Disable foreign keys
LOG.log( Level.FINEST, "Disable foreign keys: START" );
s.execute( "SET FOREIGN_KEY_CHECKS = 0" );
LOG.log( Level.FINEST, "Disable foreign keys: END" );
LOG.log( Level.FINEST, "Dropping tables: START" );
String sql = sb.toString();
s.execute( sql );
LOG.log( Level.FINEST, "Dropping tables: END" );
// Enable foreign keys
LOG.log( Level.FINEST, "Enabling foreign keys: START" );
s.execute( "SET FOREIGN_KEY_CHECKS = 1" );
LOG.log( Level.FINEST, "Enabling foreign keys: END" );
LOG.log( Level.FINEST, "Committing: START" );
c.commit();
LOG.log( Level.FINEST, "Committing: END" );
}
catch (SQLException e) {
try {
c.rollback();
}
catch (SQLException e1) {
e.addSuppressed( e1 );
}
throw new RuntimeException( e );
}
}
@Override
public void clearAllData(Connection connection) {
clearData0(
connection,
null,
statement -> {
try {
return statement.executeQuery(
"SELECT TABLE_SCHEMA, TABLE_NAME FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA NOT IN (" + SYSTEM_SCHEMAS + ")" );
}
catch (SQLException e) {
throw new RuntimeException( e );
}
}
);
}
@Override
public void clearData(Connection connection, String schemaName) {
clearData0(
connection,
schemaName,
statement -> {
try {
return statement.executeQuery(
"SELECT TABLE_SCHEMA, TABLE_NAME FROM INFORMATION_SCHEMA.TABLES WHERE TABLE_SCHEMA = '" + schemaName + "'" );
}
catch (SQLException e) {
throw new RuntimeException( e );
}
}
);
}
private void clearData0(Connection connection, String schemaName, Function tablesProvider) {
try (Statement s = connection.createStatement()) {
// Disable foreign keys
LOG.log( Level.FINEST, "Disable foreign keys: START" );
s.execute( "SET FOREIGN_KEY_CHECKS = 0" );
LOG.log( Level.FINEST, "Disable foreign keys: END" );
// Delete data
LOG.log( Level.FINEST, "Deleting data: START" );
List clearingSqls = clearingSqlsPerSchema.get( schemaName );
if ( clearingSqls == null ) {
clearingSqls = new ArrayList<>();
ResultSet rs = tablesProvider.apply( s );
while ( rs.next() ) {
String tableSchema = rs.getString( 1 );
String tableName = rs.getString( 2 );
if ( !ignoredTables.contains( tableName ) ) {
clearingSqls.add( createClearingStatementForTable( tableSchema, tableName ) );
}
}
clearingSqlsPerSchema.put( schemaName, clearingSqls );
}
for ( String clearingSql : clearingSqls ) {
s.execute( clearingSql );
}
LOG.log( Level.FINEST, "Deleting data: END" );
// Enable foreign keys
LOG.log( Level.FINEST, "Enabling foreign keys: START" );
s.execute( "SET FOREIGN_KEY_CHECKS = 1" );
LOG.log( Level.FINEST, "Enabling foreign keys: END" );
LOG.log( Level.FINEST, "Committing: START" );
connection.commit();
LOG.log( Level.FINEST, "Committing: END" );
}
catch (SQLException e) {
try {
connection.rollback();
}
catch (SQLException e1) {
e.addSuppressed( e1 );
}
throw new RuntimeException( e );
}
}
protected abstract String createClearingStatementForTable(String tableSchema, String tableName);
}