transaction perdue avec JPA contrainte unique?
-
29-09-2019 - |
Question
J'ai un champ avec une contrainte unique:
@Column(unique=true)
private String uriTitle;
Lorsque je tente de sauver deux entités avec la même valeur, je reçois une exception - mais une autre exception que j'exected:
java.lang.IllegalStateException: <|Exception Description: No transaction is currently active
at org.eclipse.persistence.internal.jpa.transaction.EntityTransactionImpl.rollback(EntityTransactionImpl.java:122)
at com.example.persistence.TransactionFilter.doFilter(TransactionFilter.java:35)
Le look de la méthode de filtre douteux comme celui-ci:
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain)
throws IOException, ServletException {
EntityManager entityManager = ThreadLocalEntityManager.get();
EntityTransaction transaction = entityManager.getTransaction();
transaction.begin();
try {
chain.doFilter(request, response);
transaction.commit();
}
catch (Throwable t) {
transaction.rollback(); // line 35, mentioned in the exception
throw new RuntimeException(t);
}
finally {
entityManager.close();
ThreadLocalEntityManager.reset();
}
}
... et le ThreadLocalEntityManager comme ceci:
public class ThreadLocalEntityManager {
private static EntityManagerFactory entityManagerFactory = null;
private static final ThreadLocal<EntityManager> entityManager =
new ThreadLocal<EntityManager>() {
@Override
protected EntityManager initialValue() {
return entityManagerFactory.createEntityManager();
}
};
private ThreadLocalEntityManager() {}
public static synchronized void init(String persistenceUnit) {
requireNotNull(persistenceUnit);
requireState(entityManagerFactory == null, "'init' can be called only "
+ "once.");
entityManagerFactory =
Persistence.createEntityManagerFactory(persistenceUnit);
}
public static EntityManager get() {
requireState(entityManagerFactory != null, "Call 'init' before calling "
+ "'get'");
return entityManager.get();
}
public static void reset() {
entityManager.remove();
}
}
J'enveloppé l'appel à save
avec try ...catch
pour gérer la violation unique contrainte, mais cela ne fonctionne pas:
try {
ThreadLocalEntityManager.get().persist(article); // article is constrained
}
catch(Throwable t) {
t.printStackTrace();
}
Toute idée pourquoi il n'y a pas de transaction? Quelle est la bonne façon de gérer une violation de contrainte unique?
La solution
Votre transaction a été annulée automatiquement en raison de la violation DB de contrainte, donc il n'y a pas de transaction rollback manuellement. Pour gérer ce cas, vous pouvez écrire
catch (Throwable t) {
if (transaction.isActive())
transaction.rollback(); // line 35, mentioned in the exception
throw new RuntimeException(t);
}
Autres conseils
Vous n'avez pas de séparation entre les transactions et l'interface graphique dans votre application. Vous aurez des problèmes avec la gestion des transactions en général.