Postgresql、Hibernate、およびJPAを使用しています。データベースに例外があるときはいつでも、DBサーバーで実際に何がうまくいかなかったかを示さないため、あまり役に立ちません。
Caused by: Java.sql.BatchUpdateException: Batch entry 0 update foo set ALERT_FLAG='3' was aborted. Call getNextException to see the cause.
at org.postgresql.jdbc2.AbstractJdbc2Statement$BatchResultHandler.handleError(AbstractJdbc2Statement.Java:2621)
at org.postgresql.core.v3.QueryExecutorImpl.processResults(QueryExecutorImpl.Java:1837)
at org.postgresql.core.v3.QueryExecutorImpl.execute(QueryExecutorImpl.Java:407)
at org.postgresql.jdbc2.AbstractJdbc2Statement.executeBatch(AbstractJdbc2Statement.Java:2754)
at com.mchange.v2.c3p0.impl.NewProxyPreparedStatement.executeBatch(NewProxyPreparedStatement.Java:1723)
at org.hibernate.jdbc.BatchingBatcher.doExecuteBatch(BatchingBatcher.Java:70)
at org.hibernate.jdbc.AbstractBatcher.executeBatch(AbstractBatcher.Java:268)
... 82 more
データベースからの例外メッセージをアプリケーションのログに表示したい。
私は この記事 に遭遇しました。これは、アスペクトを使用して例外チェーンを設定します。
アスペクトやカスタムコードを使用せずにこれを修正する方法はありますか。理想的なソリューションには、構成ファイルの変更のみが含まれます。
これを実現するためにカスタムコードを記述する必要はありません。Hibernateはデフォルトで例外の原因をログに記録します。これが表示されない場合、Hibernateロギングが正しく設定されていてはなりません。 slf4j + log4jを使用し、依存関係管理にMavenを使用する例を次に示します。
src/main/Java/pgextest/PGExceptionTest.Java
public class PGExceptionTest {
public static void main(String[] args) throws Exception {
EntityManagerFactory entityManagerFactory = Persistence.createEntityManagerFactory(
"pgextest");
EntityManager entityManager = entityManagerFactory.createEntityManager();
entityManager.getTransaction().begin();
// here I attempt to persist an object with an ID that is already in use
entityManager.persist(new PGExceptionTestBean(1));
entityManager.getTransaction().commit();
entityManager.close();
}
}
src/main/resources/log4j.properties
log4j.rootLogger=ERROR, stdout
log4j.appender.stdout=org.Apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.Apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%5p [%t] - %m%n
src/main/resources/META-INF/persistence.xml
<persistence xmlns="http://Java.Sun.com/xml/ns/persistence"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://Java.Sun.com/xml/ns/persistence http://Java.Sun.com/xml/ns/persistence/persistence_2_0.xsd"
version="2.0">
<persistence-unit name="pgextest">
<properties>
<property name="javax.persistence.jdbc.driver" value="org.postgresql.Driver"/>
<property name="javax.persistence.jdbc.url" value="jdbc:postgresql://localhost/pgextest"/>
<property name="javax.persistence.jdbc.user" value="postgres"/>
<property name="javax.persistence.jdbc.password" value="postgres"/>
<property name="hibernate.dialect" value="org.hibernate.dialect.PostgreSQLDialect"/>
<property name="hibernate.jdbc.batch_size" value="5"/>
</properties>
</persistence-unit>
</persistence>
pom.xml
<project xmlns="http://maven.Apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.Apache.org/POM/4.0.0 http://maven.Apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>pgextest</groupId>
<artifactId>pgextest</artifactId>
<version>0.0.1-SNAPSHOT</version>
<build>
<plugins>
<plugin>
<artifactId>maven-compiler-plugin</artifactId>
<configuration>
<source>1.6</source>
<target>1.6</target>
</configuration>
</plugin>
</plugins>
</build>
<dependencies>
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-entitymanager</artifactId>
<version>3.6.9.Final</version>
</dependency>
<dependency>
<groupId>postgresql</groupId>
<artifactId>postgresql</artifactId>
<version>9.1-901.jdbc4</version>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-log4j12</artifactId>
<version>1.6.1</version>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>log4j</groupId>
<artifactId>log4j</artifactId>
<version>1.2.15</version>
<scope>runtime</scope>
</dependency>
</dependencies>
</project>
Mainメソッドを実行すると、次のログが記録されます。
ERROR [main] - Batch entry 0 insert into PGExceptionTestBean (label, id) values (NULL, '1') was aborted. Call getNextException to see the cause.
ERROR [main] - ERROR: duplicate key value violates unique constraint "pgexceptiontestbean_pkey"
おそらく、プロパティhibernate.jdbc.batch_size
から0
(おそらく、実稼働環境でこれを行いたくないと言う必要はありません。)
これにより、問題の原因となった例外メッセージ(Hibernate 3.2.5.ga)を取得できました。
catch (JDBCException jdbce) {
jdbce.getSQLException().getNextException().printStackTrace();
}
アスペクトプログラミングは、この種の問題を解決するためのより良いソリューションだと思います。
ただし、そのためのカスタムコードを記述したい場合は、SqlExceptionをキャッチしてループ処理し、各例外をログに記録できます。このような何かが動作するはずです。
try {
// whatever your code is
} catch (SQLException e) {
while(e!= null) {
logger.log(e);
e = e.getNextException();
}
}
私にとって例外はPersistenceExceptionだったので、これをしなければなりませんでした:
try {
//...
} catch (javax.persistence.PersistenceException e) {
log.error(((Java.sql.BatchUpdateException) e.getCause().getCause()).getNextException());
}
try {
// code
} catch (SQLException e) {
for (Throwable throwable : e) {
log.error("{}", throwable);
}
}
JUnitテスト内からこの例外を取得している場合は、TestRule
でJUnit例外を変換できます(これはExpectedException
TestRule
)
public class HibernateBatchUnwindRule implements TestRule {
private boolean handleAssumptionViolatedExceptions = false;
private boolean handleAssertionErrors = false;
private HibernateBatchUnwindRule() {
}
public static HibernateBatchUnwindRule create(){
return new HibernateBatchUnwindRule();
}
public HibernateBatchUnwindRule handleAssertionErrors() {
handleAssertionErrors = true;
return this;
}
public HibernateBatchUnwindRule handleAssumptionViolatedExceptions() {
handleAssumptionViolatedExceptions = true;
return this;
}
public Statement apply(Statement base,
org.junit.runner.Description description) {
return new ExpectedExceptionStatement(base);
}
private class ExpectedExceptionStatement extends Statement {
private final Statement fNext;
public ExpectedExceptionStatement(Statement base) {
fNext = base;
}
@Override
public void evaluate() throws Throwable {
try {
fNext.evaluate();
} catch (AssumptionViolatedException e) {
optionallyHandleException(e, handleAssumptionViolatedExceptions);
} catch (AssertionError e) {
optionallyHandleException(e, handleAssertionErrors);
} catch (Throwable e) {
handleException(e);
}
}
}
private void optionallyHandleException(Throwable e, boolean handleException)
throws Throwable {
if (handleException) {
handleException(e);
} else {
throw e;
}
}
private void handleException(Throwable e) throws Throwable {
Throwable cause = e.getCause();
while (cause != null) {
if (cause instanceof BatchUpdateException) {
BatchUpdateException batchUpdateException = (BatchUpdateException) cause;
throw batchUpdateException.getNextException();
}
cause = cause.getCause();
};
throw e;
}
}
そして、ルールをテストケースに追加します
public class SomeTest {
@Rule
public HibernateBatchUnwindRule batchUnwindRule = HibernateBatchUnwindRule.create();
@Test
public void testSomething(){...}
}
Kafka-Connectからこの例外が発生した場合は、batch.sizeプロパティを0(一時的に)に設定して、シンクワーカーで発生した例外を明らかにすることができます。