| /* |
| * Copyright (c) 1998, 2021 Oracle and/or its affiliates. All rights reserved. |
| * |
| * This program and the accompanying materials are made available under the |
| * terms of the Eclipse Public License v. 2.0 which is available at |
| * http://www.eclipse.org/legal/epl-2.0, |
| * or the Eclipse Distribution License v. 1.0 which is available at |
| * http://www.eclipse.org/org/documents/edl-v10.php. |
| * |
| * SPDX-License-Identifier: EPL-2.0 OR BSD-3-Clause |
| */ |
| |
| // Contributors: |
| // Oracle - initial API and implementation from Oracle TopLink |
| package org.eclipse.persistence.testing.tests.jpa.advanced.concurrency; |
| |
| import jakarta.persistence.EntityManager; |
| import jakarta.persistence.EntityManagerFactory; |
| |
| import junit.framework.*; |
| |
| import org.eclipse.persistence.internal.jpa.EntityManagerImpl; |
| import org.eclipse.persistence.internal.sessions.UnitOfWorkImpl; |
| import org.eclipse.persistence.testing.framework.junit.JUnitTestCase; |
| |
| import org.eclipse.persistence.testing.models.jpa.advanced.ConcurrencyA; |
| import org.eclipse.persistence.testing.models.jpa.advanced.ConcurrencyB; |
| import org.eclipse.persistence.testing.models.jpa.advanced.ConcurrencyC; |
| import org.eclipse.persistence.testing.models.jpa.advanced.Department; |
| import org.eclipse.persistence.testing.models.jpa.advanced.Equipment; |
| |
| public class ConcurrencyTest extends JUnitTestCase { |
| public ConcurrencyTest() { |
| super(); |
| } |
| |
| public ConcurrencyTest(String name) { |
| super(name); |
| } |
| |
| public static Test suite() { |
| TestSuite suite = new TestSuite("ConcurrencyTestSuite"); |
| |
| suite.addTest(new ConcurrencyTest("testDeadLockOnReadLock")); |
| suite.addTest(new ConcurrencyTest("testTransitionToDeferedFailure")); |
| return suite; |
| } |
| |
| /** |
| * The setup is done as a test, both to record its failure, and to allow |
| * execution in the server. |
| */ |
| public void testDeadLockOnReadLock() { |
| if (isOnServer()) { |
| return; |
| } |
| EntityManagerFactory emf = getEntityManagerFactory(); |
| EntityManager em = emf.createEntityManager(); |
| em.getTransaction().begin(); |
| Department dept = new Department(); |
| em.persist(dept); |
| Equipment equip = new Equipment(); |
| em.persist(equip); |
| em.getTransaction().commit(); |
| em.close(); |
| try { |
| Integer i = 5; |
| Thread thread1 = new Thread(new Runner1(i, dept.getId(), equip.getId(), emf)); |
| thread1.setName("Runner1"); |
| Thread thread2 = new Thread(new Runner2(i, dept.getId(), equip.getId(), emf)); |
| thread2.setName("Runner2"); |
| thread2.start(); |
| thread1.start(); |
| try { |
| Thread.sleep(8000); |
| } catch (InterruptedException e) { |
| e.printStackTrace(); |
| } |
| synchronized (i) { |
| i.notifyAll(); |
| } |
| try { |
| thread2.join(30000); |
| thread1.join(30000); |
| } catch (InterruptedException e) { |
| e.printStackTrace(); |
| } |
| if (thread2.isAlive() || thread1.isAlive()) { |
| thread2.interrupt(); |
| thread1.interrupt(); |
| fail("Dead-lock occurred"); |
| } |
| } finally { |
| em = emf.createEntityManager(); |
| em.getTransaction().begin(); |
| em.remove(em.find(Department.class, dept.getId())); |
| em.remove(em.find(Equipment.class, equip.getId())); |
| em.getTransaction().commit(); |
| } |
| } |
| |
| /** |
| * The setup is done as a test, both to record its failure, and to allow |
| * execution in the server. |
| */ |
| public void testTransitionToDeferedFailure() { |
| if (isOnServer()) { |
| return; |
| } |
| Integer toWaitOn = 4; |
| Thread thread1 = null; |
| EntityManagerFactory emf = getEntityManagerFactory(); |
| EntityManager em = emf.createEntityManager(); |
| em.getTransaction().begin(); |
| |
| ConcurrencyA a = new ConcurrencyA(); |
| ConcurrencyA a2 = new ConcurrencyA(); |
| ConcurrencyB b = new ConcurrencyB(); |
| ConcurrencyC c = new ConcurrencyC(); |
| em.persist(a); |
| em.persist(a2); |
| em.persist(c); |
| em.persist(b); |
| em.getTransaction().commit(); |
| em.close(); |
| try { |
| thread1 = new Thread(new TransitionRunner1(toWaitOn, b, c, emf)); |
| thread1.start(); |
| |
| em = emf.createEntityManager(); |
| a = em.find(ConcurrencyA.class, a.getId()); |
| a2 = em.find(ConcurrencyA.class, a2.getId()); |
| b = em.find(ConcurrencyB.class, b.getId()); |
| c = em.find(ConcurrencyC.class, c.getId()); |
| |
| a2.setName(System.currentTimeMillis() + "_A"); |
| |
| a.setConcurrencyB(b); |
| a.setConcurrencyC(c); |
| |
| UnitOfWorkImpl uow = ((EntityManagerImpl) em).getActivePersistenceContext(null); |
| try { |
| Thread.sleep(20000); |
| synchronized (toWaitOn) { |
| toWaitOn.notifyAll(); |
| toWaitOn.wait(); |
| } |
| } catch (InterruptedException e) { |
| } |
| uow.issueSQLbeforeCompletion(true); |
| uow.mergeClonesAfterCompletion(); |
| |
| synchronized (toWaitOn) { |
| toWaitOn.notifyAll(); |
| } |
| try { |
| thread1.join(30000); |
| } catch (InterruptedException e) { |
| e.printStackTrace(); |
| } |
| } finally { |
| if (thread1.isAlive()){ |
| thread1.interrupt(); |
| } |
| em = emf.createEntityManager(); |
| em.getTransaction().begin(); |
| em.remove(em.find(ConcurrencyA.class, a.getId())); |
| em.remove(em.find(ConcurrencyA.class, a2.getId())); |
| em.remove(em.find(ConcurrencyC.class, c.getId())); |
| em.remove(em.find(ConcurrencyB.class, b.getId())); |
| em.getTransaction().commit(); |
| } |
| } |
| |
| } |