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

org.hibernate.testing.cleaner.DB2DatabaseCleaner Maven / Gradle / Ivy

/*
 * SPDX-License-Identifier: LGPL-2.1-or-later
 * 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.logging.Level;
import java.util.logging.Logger;

/**
 * @author Christian Beikov
 */
public class DB2DatabaseCleaner implements DatabaseCleaner {

	private static final Logger LOG = Logger.getLogger( DB2DatabaseCleaner.class.getName() );
	private static final String SYSTEM_SCHEMAS = "'SYSCAT',"
			+ "'SYSIBM',"
			+ "'SYSIBMADM',"
			+ "'SYSPUBLIC',"
			+ "'SYSSTAT',"
			+ "'DB2GSE',"
			+ "'SYSTOOLS'";

	private final List ignoredTables = new ArrayList<>();
	private final Map>> cachedForeignKeysPerSchema = new HashMap<>();

	@Override
	public boolean isApplicable(Connection connection) {
		try {
			return connection.getMetaData().getDatabaseProductName().startsWith( "DB2" );
		}
		catch (SQLException e) {
			throw new RuntimeException( "Could not resolve the database metadata!", e );
		}
	}

	@Override
	public void addIgnoredTable(String tableName) {
		ignoredTables.add( tableName );
	}

	@Override
	public void clearAllSchemas(Connection c) {
		cachedForeignKeysPerSchema.clear();
		try (Statement s = c.createStatement()) {
			ResultSet rs;
			List sqls = new ArrayList<>();

			// Collect schema objects
			LOG.log( Level.FINEST, "Collect schema objects: START" );
			rs = s.executeQuery(
					"SELECT 'DROP INDEX \"' || TRIM(INDSCHEMA) || '\".\"' || TRIM(INDNAME) || '\"' " +
							"FROM SYSCAT.INDEXES " +
							"WHERE UNIQUERULE = 'D' " +
							"AND INDSCHEMA NOT IN (" + SYSTEM_SCHEMAS + ")"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'ALTER TABLE \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\" DROP FOREIGN KEY \"' || TRIM(CONSTNAME) || '\"' " +
							"FROM SYSCAT.TABCONST " +
							"WHERE TYPE = 'F' " +
							"AND TABSCHEMA NOT IN (" + SYSTEM_SCHEMAS + ")"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'ALTER TABLE \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\" DROP UNIQUE \"' || TRIM(INDNAME) || '\"' " +
							"FROM SYSCAT.INDEXES " +
							"WHERE UNIQUERULE = 'U' " +
							"AND INDSCHEMA NOT IN (" + SYSTEM_SCHEMAS + ")"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'ALTER TABLE \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\" DROP PRIMARY KEY' " +
							"FROM SYSCAT.INDEXES " +
							"WHERE UNIQUERULE = 'P' " +
							"AND INDSCHEMA NOT IN (" + SYSTEM_SCHEMAS + ")"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'DROP VIEW \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\"' " +
							"FROM SYSCAT.TABLES " +
							"WHERE TYPE = 'V' " +
							"AND TABSCHEMA  NOT IN (" + SYSTEM_SCHEMAS + ")"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'DROP TABLE \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\"' " +
							"FROM SYSCAT.TABLES " +
							"WHERE TYPE = 'T' " +
							"AND TABSCHEMA NOT IN (" + SYSTEM_SCHEMAS + ")"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'DROP SEQUENCE \"' || TRIM(SEQSCHEMA) || '\".\"' || TRIM(SEQNAME) || '\"' " +
							"FROM SYSCAT.SEQUENCES " +
							"WHERE SEQSCHEMA NOT IN (" + SYSTEM_SCHEMAS + ")"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}
			LOG.log( Level.FINEST, "Collect schema objects: END" );

			LOG.log( Level.FINEST, "Dropping schema objects: START" );
			for ( String sql : sqls ) {
				try {
					s.execute( sql );
				}
				catch (SQLException e) {
					if ( -204 == e.getErrorCode() ) {
						// Apparently we deleted this along with a dependent object since it doesn't exist anymore
					}
					else {
						throw e;
					}
				}
			}
			LOG.log( Level.FINEST, "Dropping schema objects: 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 clearSchema(Connection c, String schemaName) {
		cachedForeignKeysPerSchema.remove( schemaName );
		schemaName = schemaName.toUpperCase();
		try (Statement s = c.createStatement()) {
			ResultSet rs;
			List sqls = new ArrayList<>();

			// Collect schema objects
			LOG.log( Level.FINEST, "Collect schema objects: START" );
			rs = s.executeQuery(
					"SELECT 'DROP INDEX \"' || TRIM(INDSCHEMA) || '\".\"' || TRIM(INDNAME) || '\"' " +
							"FROM SYSCAT.INDEXES " +
							"WHERE UNIQUERULE = 'D' " +
							"AND INDSCHEMA = '" + schemaName + "'"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'ALTER TABLE \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\" DROP FOREIGN KEY \"' || TRIM(CONSTNAME) || '\"' " +
							"FROM SYSCAT.TABCONST " +
							"WHERE TYPE = 'F' " +
							"AND TABSCHEMA = '" + schemaName + "'"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'ALTER TABLE \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\" DROP UNIQUE \"' || TRIM(INDNAME) || '\"' " +
							"FROM SYSCAT.INDEXES " +
							"WHERE UNIQUERULE = 'U' " +
							"AND INDSCHEMA = '" + schemaName + "'"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'ALTER TABLE \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\" DROP PRIMARY KEY' " +
							"FROM SYSCAT.INDEXES " +
							"WHERE UNIQUERULE = 'P' " +
							"AND INDSCHEMA = '" + schemaName + "'"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'DROP VIEW \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\"' " +
							"FROM SYSCAT.TABLES " +
							"WHERE TYPE = 'V' " +
							"AND TABSCHEMA  = '" + schemaName + "'"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'DROP TABLE \"' || TRIM(TABSCHEMA) || '\".\"' || TRIM(TABNAME) || '\"' " +
							"FROM SYSCAT.TABLES " +
							"WHERE TYPE = 'T' " +
							"AND TABSCHEMA = '" + schemaName + "'"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}

			rs = s.executeQuery(
					"SELECT 'DROP SEQUENCE \"' || TRIM(SEQSCHEMA) || '\".\"' || TRIM(SEQNAME) || '\"' " +
							"FROM SYSCAT.SEQUENCES " +
							"WHERE SEQSCHEMA = '" + schemaName + "'"
			);
			while ( rs.next() ) {
				sqls.add( rs.getString( 1 ) );
			}
			LOG.log( Level.FINEST, "Collect schema objects: END" );

			LOG.log( Level.FINEST, "Dropping schema objects: START" );
			for ( String sql : sqls ) {
				try {
					s.execute( sql );
				}
				catch (SQLException e) {
					if ( -204 == e.getErrorCode() ) {
						// Apparently we deleted this along with a dependent object since it doesn't exist anymore
					}
					else {
						throw e;
					}
				}
			}
			LOG.log( Level.FINEST, "Dropping schema objects: 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) {
		Map> cachedForeignKeys = cachedForeignKeysPerSchema.get( null );
		if ( cachedForeignKeys == null ) {
			cachedForeignKeys = collectAllForeignKeys( connection );
			cachedForeignKeysPerSchema.put( null, cachedForeignKeys );
		}
		deleteAllData( connection, cachedForeignKeys );
	}

	@Override
	public void clearData(Connection connection, String schemaName) {
		Map> cachedForeignKeys = cachedForeignKeysPerSchema.get( schemaName );
		if ( cachedForeignKeys == null ) {
			cachedForeignKeys = collectForeignKeys( connection, schemaName.toUpperCase() );
			cachedForeignKeysPerSchema.put( schemaName, cachedForeignKeys );
		}
		deleteAllData( connection, cachedForeignKeys );
	}

	private Map> collectAllForeignKeys(Connection c) {
		try (Statement s = c.createStatement()) {
			// Collect table names for schemas
			LOG.log( Level.FINEST, "Collect table names: START" );
			ResultSet rs = s.executeQuery(
					"SELECT TABLE_SCHEMA || '.' || TABLE_NAME FROM SYSIBM.TABLES WHERE TABLE_SCHEMA NOT IN (" + SYSTEM_SCHEMAS + ")"
			);
			Map> foreignKeys = new HashMap<>();
			while ( rs.next() ) {
				foreignKeys.put( rs.getString( 1 ), new ArrayList() );
			}
			LOG.log( Level.FINEST, "Collect table names: END" );

			// Collect foreign keys for tables
			LOG.log( Level.FINEST, "Collect foreign keys: START" );
			ResultSet rs2 = s.executeQuery(
					"SELECT FKTABLE_SCHEM || '.' || FKTABLE_NAME, FK_NAME FROM SYSIBM.SQLFOREIGNKEYS WHERE FKTABLE_SCHEM NOT IN (" + SYSTEM_SCHEMAS + ")"
			);
			while ( rs2.next() ) {
				foreignKeys.get( rs2.getString( 1 ) ).add( rs2.getString( 2 ) );
			}
			LOG.log( Level.FINEST, "Collect foreign keys: END" );

			return foreignKeys;
		}
		catch (SQLException e) {
			try {
				c.rollback();
			}
			catch (SQLException e1) {
				e.addSuppressed( e1 );
			}

			throw new RuntimeException( e );
		}
	}

	private Map> collectForeignKeys(Connection c, String schemaName) {
		try (Statement s = c.createStatement()) {
			// Collect table names for schemas
			LOG.log( Level.FINEST, "Collect table names: START" );
			ResultSet rs = s.executeQuery(
					"SELECT TRIM(TABLE_SCHEMA) || '.' || TABLE_NAME FROM SYSIBM.TABLES WHERE TABLE_SCHEMA = '" + schemaName + "'"
			);
			Map> foreignKeys = new HashMap<>();
			while ( rs.next() ) {
				foreignKeys.put( rs.getString( 1 ), new ArrayList() );
			}
			LOG.log( Level.FINEST, "Collect table names: END" );

			// Collect foreign keys for tables
			LOG.log( Level.FINEST, "Collect foreign keys: START" );
			ResultSet rs2 = s.executeQuery(
					"SELECT FKTABLE_SCHEM || '.' || FKTABLE_NAME, FK_NAME FROM SYSIBM.SQLFOREIGNKEYS WHERE FKTABLE_SCHEM = '" + schemaName + "'"
			);
			while ( rs2.next() ) {
				foreignKeys.get( rs2.getString( 1 ) ).add( rs2.getString( 2 ) );
			}
			LOG.log( Level.FINEST, "Collect foreign keys: END" );

			return foreignKeys;
		}
		catch (SQLException e) {
			try {
				c.rollback();
			}
			catch (SQLException e1) {
				e.addSuppressed( e1 );
			}

			throw new RuntimeException( e );
		}
	}

	private void deleteAllData(Connection c, Map> foreignKeys) {
		try (Statement s = c.createStatement()) {
			// Disable foreign keys
			LOG.log( Level.FINEST, "Disable foreign keys: START" );
			for ( Map.Entry> entry : foreignKeys.entrySet() ) {
				for ( String fk : entry.getValue() ) {
					s.execute( "ALTER TABLE " + entry.getKey() + " ALTER FOREIGN KEY " + fk + " NOT ENFORCED" );
				}
			}
			c.commit();
			LOG.log( Level.FINEST, "Disable foreign keys: END" );

			// Delete data
			LOG.log( Level.FINEST, "Deleting data: START" );
			for ( String table : foreignKeys.keySet() ) {
				if ( !ignoredTables.contains( table ) ) {
					s.execute( "TRUNCATE TABLE " + table + " IMMEDIATE" );
					// DB2 needs a commit after every truncate statement
					c.commit();
				}
			}
			LOG.log( Level.FINEST, "Deleting data: END" );

			// Enable foreign keys
			LOG.log( Level.FINEST, "Enabling foreign keys: START" );
			for ( Map.Entry> entry : foreignKeys.entrySet() ) {
				for ( String fk : entry.getValue() ) {
					s.execute( "ALTER TABLE " + entry.getKey() + " ALTER FOREIGN KEY " + fk + " ENFORCED" );
				}
			}
			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 );
		}
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy