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
                
             The newest version!
        
        /*
 * SPDX-License-Identifier: Apache-2.0
 * Copyright Red Hat Inc. and Hibernate Authors
 */
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);
}
        © 2015 - 2025 Weber Informatics LLC | Privacy Policy