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?

Était-ce utile?

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.

Licencié sous: CC-BY-SA avec attribution
Non affilié à StackOverflow
scroll top