net.sf.hajdbc.sync.SynchronizationSupportImpl Maven / Gradle / Ivy
/*
* HA-JDBC: High-Availability JDBC
* Copyright (C) 2012 Paul Ferraro
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program. If not, see .
*/
package net.sf.hajdbc.sync;
import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.sql.Types;
import java.text.MessageFormat;
import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
import net.sf.hajdbc.Database;
import net.sf.hajdbc.ExceptionType;
import net.sf.hajdbc.ForeignKeyConstraint;
import net.sf.hajdbc.IdentityColumnSupport;
import net.sf.hajdbc.Messages;
import net.sf.hajdbc.SequenceProperties;
import net.sf.hajdbc.SequenceSupport;
import net.sf.hajdbc.TableProperties;
import net.sf.hajdbc.UniqueConstraint;
import net.sf.hajdbc.dialect.Dialect;
import net.sf.hajdbc.logging.Level;
import net.sf.hajdbc.logging.Logger;
import net.sf.hajdbc.logging.LoggerFactory;
import net.sf.hajdbc.util.Resources;
import net.sf.hajdbc.util.Strings;
/**
* Default {@link SynchronizationSupport} implementation.
* @author Paul Ferraro
*/
public class SynchronizationSupportImpl> implements SynchronizationSupport
{
private final Logger logger = LoggerFactory.getLogger(this.getClass());
private final SynchronizationContext context;
public SynchronizationSupportImpl(SynchronizationContext context)
{
this.context = context;
}
/**
* {@inheritDoc}
* @see net.sf.hajdbc.sync.SynchronizationSupport#dropForeignKeys()
*/
@Override
public void dropForeignKeys() throws SQLException
{
Dialect dialect = this.context.getDialect();
Connection connection = this.context.getConnection(this.context.getTargetDatabase());
boolean autoCommit = connection.getAutoCommit();
try
{
connection.setAutoCommit(true);
Statement statement = connection.createStatement();
try
{
for (TableProperties table: this.context.getTargetDatabaseProperties().getTables())
{
for (ForeignKeyConstraint constraint: table.getForeignKeyConstraints())
{
String sql = dialect.getDropForeignKeyConstraintSQL(constraint);
this.logger.log(Level.DEBUG, sql);
statement.addBatch(sql);
}
}
statement.executeBatch();
}
finally
{
Resources.close(statement);
}
}
finally
{
connection.setAutoCommit(autoCommit);
}
}
/**
* {@inheritDoc}
* @see net.sf.hajdbc.sync.SynchronizationSupport#restoreForeignKeys()
*/
@Override
public void restoreForeignKeys() throws SQLException
{
Dialect dialect = this.context.getDialect();
Connection connection = this.context.getConnection(this.context.getTargetDatabase());
boolean autoCommit = connection.getAutoCommit();
try
{
connection.setAutoCommit(true);
Statement statement = connection.createStatement();
try
{
for (TableProperties table: this.context.getSourceDatabaseProperties().getTables())
{
for (ForeignKeyConstraint constraint: table.getForeignKeyConstraints())
{
String sql = dialect.getCreateForeignKeyConstraintSQL(constraint);
this.logger.log(Level.DEBUG, sql);
statement.addBatch(sql);
}
}
statement.executeBatch();
}
finally
{
Resources.close(statement);
}
}
finally
{
connection.setAutoCommit(autoCommit);
}
}
/**
* {@inheritDoc}
* @see net.sf.hajdbc.sync.SynchronizationSupport#synchronizeSequences()
*/
@Override
public void synchronizeSequences() throws SQLException
{
SequenceSupport support = this.context.getDialect().getSequenceSupport();
if (support != null)
{
Collection sequences = this.context.getSourceDatabaseProperties().getSequences();
if (!sequences.isEmpty())
{
D sourceDatabase = this.context.getSourceDatabase();
Set databases = this.context.getActiveDatabaseSet();
ExecutorService executor = this.context.getExecutor();
Map sequenceMap = new HashMap();
Map> futureMap = new HashMap>();
for (SequenceProperties sequence: sequences)
{
final String sql = support.getNextSequenceValueSQL(sequence);
this.logger.log(Level.DEBUG, sql);
for (final D database: databases)
{
final SynchronizationContext context = this.context;
Callable task = new Callable()
{
@Override
public Long call() throws SQLException
{
Statement statement = context.getConnection(database).createStatement();
try
{
ResultSet resultSet = statement.executeQuery(sql);
resultSet.next();
return resultSet.getLong(1);
}
finally
{
Resources.close(statement);
}
}
};
futureMap.put(database, executor.submit(task));
}
try
{
Long sourceValue = futureMap.get(sourceDatabase).get();
sequenceMap.put(sequence, sourceValue);
for (D database: databases)
{
if (!database.equals(sourceDatabase))
{
Long value = futureMap.get(database).get();
if (!value.equals(sourceValue))
{
throw new SQLException(Messages.SEQUENCE_OUT_OF_SYNC.getMessage(sequence, database, value, sourceDatabase, sourceValue));
}
}
}
}
catch (InterruptedException e)
{
throw new SQLException(e);
}
catch (ExecutionException e)
{
throw ExceptionType.SQL.getExceptionFactory().createException(e.getCause());
}
}
Connection targetConnection = this.context.getConnection(this.context.getTargetDatabase());
Statement targetStatement = targetConnection.createStatement();
try
{
for (SequenceProperties sequence: sequences)
{
String sql = support.getAlterSequenceSQL(sequence, sequenceMap.get(sequence) + 1);
this.logger.log(Level.DEBUG, sql);
targetStatement.addBatch(sql);
}
targetStatement.executeBatch();
}
finally
{
Resources.close(targetStatement);
}
}
}
}
/**
* {@inheritDoc}
* @see net.sf.hajdbc.sync.SynchronizationSupport#synchronizeIdentityColumns()
*/
@Override
public void synchronizeIdentityColumns() throws SQLException
{
IdentityColumnSupport support = this.context.getDialect().getIdentityColumnSupport();
if (support != null)
{
Statement sourceStatement = this.context.getConnection(this.context.getSourceDatabase()).createStatement();
try
{
Statement targetStatement = this.context.getConnection(this.context.getTargetDatabase()).createStatement();
try
{
for (TableProperties table: this.context.getSourceDatabaseProperties().getTables())
{
Collection columns = table.getIdentityColumns();
if (!columns.isEmpty())
{
String selectSQL = MessageFormat.format("SELECT max({0}) FROM {1}", Strings.join(columns, "), max("), table.getName()); //$NON-NLS-1$ //$NON-NLS-2$
this.logger.log(Level.DEBUG, selectSQL);
Map map = new HashMap();
ResultSet resultSet = sourceStatement.executeQuery(selectSQL);
try
{
if (resultSet.next())
{
int i = 0;
for (String column: columns)
{
map.put(column, resultSet.getLong(++i));
}
}
}
finally
{
Resources.close(resultSet);
}
if (!map.isEmpty())
{
for (Map.Entry mapEntry: map.entrySet())
{
String alterSQL = support.getAlterIdentityColumnSQL(table, table.getColumnProperties(mapEntry.getKey()), mapEntry.getValue() + 1);
if (alterSQL != null)
{
this.logger.log(Level.DEBUG, alterSQL);
targetStatement.addBatch(alterSQL);
}
}
targetStatement.executeBatch();
}
}
}
}
finally
{
Resources.close(targetStatement);
}
}
finally
{
Resources.close(sourceStatement);
}
}
}
/**
* {@inheritDoc}
* @see net.sf.hajdbc.sync.SynchronizationSupport#dropUniqueConstraints()
*/
@Override
public void dropUniqueConstraints() throws SQLException
{
Dialect dialect = this.context.getDialect();
Connection connection = this.context.getConnection(this.context.getTargetDatabase());
boolean autoCommit = connection.getAutoCommit();
try
{
connection.setAutoCommit(true);
Statement statement = connection.createStatement();
try
{
for (TableProperties table: this.context.getTargetDatabaseProperties().getTables())
{
for (UniqueConstraint constraint: table.getUniqueConstraints())
{
String sql = dialect.getDropUniqueConstraintSQL(constraint);
this.logger.log(Level.DEBUG, sql);
statement.addBatch(sql);
}
}
statement.executeBatch();
}
finally
{
Resources.close(statement);
}
}
finally
{
connection.setAutoCommit(autoCommit);
}
}
/**
* {@inheritDoc}
* @see net.sf.hajdbc.sync.SynchronizationSupport#restoreUniqueConstraints()
*/
@Override
public void restoreUniqueConstraints() throws SQLException
{
Dialect dialect = this.context.getDialect();
Connection connection = this.context.getConnection(this.context.getTargetDatabase());
boolean autoCommit = connection.getAutoCommit();
try
{
connection.setAutoCommit(true);
Statement statement = connection.createStatement();
try
{
for (TableProperties table: this.context.getSourceDatabaseProperties().getTables())
{
// Drop unique constraints on the current table
for (UniqueConstraint constraint: table.getUniqueConstraints())
{
String sql = dialect.getCreateUniqueConstraintSQL(constraint);
this.logger.log(Level.DEBUG, sql);
statement.addBatch(sql);
}
}
statement.executeBatch();
}
finally
{
Resources.close(statement);
}
}
finally
{
connection.setAutoCommit(autoCommit);
}
}
/**
* {@inheritDoc}
* @see net.sf.hajdbc.sync.SynchronizationSupport#rollback(java.sql.Connection)
*/
@Override
public void rollback(Connection connection)
{
try
{
connection.rollback();
}
catch (SQLException e)
{
this.logger.log(Level.WARN, e);
}
}
/**
* {@inheritDoc}
* @see net.sf.hajdbc.sync.SynchronizationSupport#getObject(java.sql.ResultSet, int, int)
*/
@Override
public Object getObject(ResultSet resultSet, int index, int type) throws SQLException
{
switch (type)
{
case Types.BLOB:
{
return resultSet.getBlob(index);
}
case Types.CLOB:
{
return resultSet.getClob(index);
}
default:
{
return resultSet.getObject(index);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy