Я нашел (очень хакерский) обходной путь:
/**
* Invoked when a transaction completes.
*/
public abstract class TransactionListener extends Value
{
private boolean invoked;
/**
* Invoked when the transaction completes.
*/
protected abstract void onCompleted();
@Override
public String getSQL()
{
return null;
}
@Override
public int getType()
{
throw new AssertionError("Unexpected method invocation");
}
@Override
public long getPrecision()
{
throw new AssertionError("Unexpected method invocation");
}
@Override
public int getDisplaySize()
{
throw new AssertionError("Unexpected method invocation");
}
@Override
public String getString()
{
throw new AssertionError("Unexpected method invocation");
}
@Override
public Object getObject()
{
throw new AssertionError("Unexpected method invocation");
}
@Override
public void set(PreparedStatement prep, int parameterIndex) throws SQLException
{
throw new AssertionError("Unexpected method invocation");
}
@Override
protected int compareSecure(Value v, CompareMode mode)
{
throw new AssertionError("Unexpected method invocation");
}
@Override
public int hashCode()
{
throw new AssertionError("Unexpected method invocation");
}
@Override
public boolean equals(Object other)
{
throw new AssertionError("Unexpected method invocation");
}
@Override
public boolean isLinked()
{
return !invoked;
}
@Override
public void close()
{
invoked = true;
onCompleted();
}
}
// -------------TRIGGER BELOW-----------
public void fire(final Connection connection, ResultSet oldRow, ResultSet newRow)
throws SQLException
{
Statement statement = connection.createStatement();
long transactionId;
ResultSet rs = statement.executeQuery("SELECT @TRANSACTION_ID");
try
{
rs.next();
transactionId = rs.getLong(1);
if (transactionId == 0)
{
// Generate a new transaction id
rs.close();
JdbcConnection jdbcConnection = (JdbcConnection) connection;
final Session session = (Session) jdbcConnection.getSession();
session.unlinkAtCommit(new TransactionListener()
{
@Override
protected void onCompleted()
{
boolean oldAutoCommit = session.getAutoCommit();
session.setAutoCommit(false);
try
{
Statement statement = connection.createStatement();
statement.executeQuery("SELECT SET(@TRANSACTION_ID, NULL)");
statement.close();
}
catch (SQLException e)
{
throw new AssertionError(e);
}
finally
{
session.setAutoCommit(oldAutoCommit);
}
}
});
rs = statement.executeQuery("SELECT SET(@TRANSACTION_ID, "
+ "audit_transaction_sequence.NEXTVAL)");
rs.next();
transactionId = rs.getLong(1);
}
}
finally
{
rs.close();
}
assert (transactionId != 0);
// ...
}
Вот как это работает:
Две основные проблемы с этим обходным путем:
- Это очень хрупкий.Если Session.unlinkAtCommit () изменится в будущем, он, вероятно, сломает прослушиватель событий.
- Мы должны повторить много шаблонного кода в верхней части каждого триггера просто для получения идентификатора транзакции.
Было бы намного проще реализовать это как встроенную функцию TRANSACTION_LOCAL_ID ().Эта функция возвращает идентификатор транзакции, специфичной для экземпляра базы данных, аналогично HSQLDB.