package org.multiverse.stms.gamma.transactionalobjects.txnref; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.junit.runners.Parameterized; import org.multiverse.api.LockMode; import org.multiverse.api.Txn; import org.multiverse.api.TxnFactory; import org.multiverse.api.exceptions.DeadTxnException; import org.multiverse.api.exceptions.PreparedTxnException; import org.multiverse.api.exceptions.ReadWriteConflict; import org.multiverse.api.functions.Functions; import org.multiverse.api.functions.LongFunction; import org.multiverse.stms.gamma.GammaStm; import org.multiverse.stms.gamma.transactionalobjects.GammaTxnRef; import org.multiverse.stms.gamma.transactionalobjects.Tranlocal; import org.multiverse.stms.gamma.transactions.GammaTxn; import org.multiverse.stms.gamma.transactions.GammaTxnFactory; import org.multiverse.stms.gamma.transactions.fat.FatFixedLengthGammaTxnFactory; import org.multiverse.stms.gamma.transactions.fat.FatMonoGammaTxn; import org.multiverse.stms.gamma.transactions.fat.FatMonoGammaTxnFactory; import org.multiverse.stms.gamma.transactions.fat.FatVariableLengthGammaTxnFactory; import java.util.Collection; import static java.util.Arrays.asList; import static org.junit.Assert.*; import static org.junit.Assume.assumeTrue; import static org.mockito.Matchers.anyLong; import static org.mockito.Mockito.*; import static org.multiverse.TestUtils.LOCKMODE_NONE; import static org.multiverse.TestUtils.*; import static org.multiverse.api.TxnThreadLocal.clearThreadLocalTxn; import static org.multiverse.api.TxnThreadLocal.getThreadLocalTxn; import static org.multiverse.stms.gamma.GammaTestUtils.*; @RunWith(Parameterized.class) public class GammaTxnRef_commute2Test { private final GammaTxnFactory transactionFactory; private final GammaStm stm; public GammaTxnRef_commute2Test(GammaTxnFactory transactionFactory) { this.transactionFactory = transactionFactory; this.stm = transactionFactory.getConfig().getStm(); } @Before public void setUp() { clearThreadLocalTxn(); } @Parameterized.Parameters public static Collection<TxnFactory[]> configs() { return asList( new TxnFactory[]{new FatVariableLengthGammaTxnFactory(new GammaStm())}, new TxnFactory[]{new FatFixedLengthGammaTxnFactory(new GammaStm())}, new TxnFactory[]{new FatMonoGammaTxnFactory(new GammaStm())} ); } @Test public void whenCommuteFunctionCausesProblems_thenNoProblemsSinceCommuteFunctionNotEvaluatedImmediately() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); LongFunction function = mock(LongFunction.class); RuntimeException ex = new RuntimeException(); when(function.call(anyLong())).thenThrow(ex); GammaTxn tx = transactionFactory.newTxn(); ref.commute(tx, function); assertHasCommutingFunctions(tx.getRefTranlocal(ref), function); assertIsActive(tx); assertEquals(initialValue, ref.atomicGet()); assertRefHasNoLocks(ref); assertSurplus(ref, 0); assertNull(getThreadLocalTxn()); } @Test public void whenLockedByOther(){ whenLockedByOther(LockMode.Read); whenLockedByOther(LockMode.Write); whenLockedByOther(LockMode.Exclusive); } public void whenLockedByOther(LockMode lockMode) { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); GammaTxn otherTx = transactionFactory.newTxn(); ref.getLock().acquire(otherTx, lockMode); GammaTxn tx = transactionFactory.newTxn(); ref.commute(tx, Functions.incLongFunction()); try { tx.commit(); fail(); } catch (ReadWriteConflict expected) { } assertSurplus(ref, 1); assertIsAborted(tx); assertRefHasLockMode(ref, otherTx, lockMode.asInt()); assertVersionAndValue(ref, initialVersion, initialValue); } @Test public void whenSuccess() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); LongFunction function = Functions.incLongFunction(); GammaTxn tx = transactionFactory.newTxn(); ref.commute(tx, function); Tranlocal commute = tx.getRefTranlocal(ref); assertTrue(commute.isCommuting()); assertEquals(0, commute.long_value); tx.commit(); assertVersionAndValue(ref, initialVersion + 1, initialValue + 1); } @Test public void whenNoChange() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); LongFunction function = Functions.identityLongFunction(); GammaTxn tx = transactionFactory.newTxn(); ref.commute(tx, function); Tranlocal commute = tx.getRefTranlocal(ref); assertTrue(commute.isCommuting()); assertEquals(0, commute.long_value); tx.commit(); assertVersionAndValue(ref, initialVersion, initialValue); } @Test public void whenNormalTransactionUsed() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); LongFunction function = Functions.incLongFunction(1); Txn tx = transactionFactory.newTxn(); ref.commute(tx, function); tx.commit(); assertVersionAndValue(ref, initialVersion + 1, initialValue + 1); } @Test public void whenAlreadyOpenedForRead() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); LongFunction function = Functions.incLongFunction(1); GammaTxn tx = transactionFactory.newTxn(); ref.get(tx); ref.commute(tx, function); Tranlocal commute = tx.getRefTranlocal(ref); assertFalse(commute.isCommuting()); assertEquals(new Long(11), commute.ref_value); tx.commit(); assertVersionAndValue(ref, initialVersion + 1, initialValue + 1); } @Test public void whenAlreadyOpenedForConstruction() { GammaTxn tx = transactionFactory.newTxn(); Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(tx, initialValue); ref.openForConstruction(tx); ref.commute(tx, Functions.incLongFunction()); Tranlocal commute = tx.getRefTranlocal(ref); assertFalse(commute.isCommuting()); assertEquals(new Long(11), commute.ref_value); tx.commit(); assertEquals(new Long(11), ref.atomicGet()); } @Test public void whenAlreadyOpenedForWrite() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); LongFunction function = Functions.incLongFunction(); GammaTxn tx = transactionFactory.newTxn(); ref.set(tx, new Long(11)); ref.commute(tx, function); Tranlocal commute = tx.getRefTranlocal(ref); assertFalse(commute.isCommuting()); assertEquals(new Long(12), commute.ref_value); tx.commit(); assertEquals(new Long(12), ref.atomicGet()); } @Test public void whenAlreadyCommuting() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); LongFunction function1 = Functions.incLongFunction(); LongFunction function2 = Functions.incLongFunction(); GammaTxn tx = transactionFactory.newTxn(); ref.commute(tx, function1); ref.commute(tx, function2); Tranlocal commute = tx.getRefTranlocal(ref); assertTrue(commute.isCommuting()); assertEquals(0, commute.long_value); tx.commit(); assertVersionAndValue(ref, initialVersion + 1, initialValue + 2); } @Test public void whenNullFunction_thenNullPointerException() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); GammaTxn tx = transactionFactory.newTxn(); long orecValue = ref.orec; try { ref.commute(tx, null); fail(); } catch (NullPointerException expected) { } assertIsAborted(tx); assertVersionAndValue(ref, initialVersion, initialValue); assertOrecValue(ref, orecValue); } @Test public void whenNullTransaction_thenNullPointerException() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); LongFunction function = mock(LongFunction.class); long orecValue = ref.orec; try { ref.commute((Txn) null, function); fail(); } catch (NullPointerException expected) { } verifyZeroInteractions(function); assertVersionAndValue(ref, initialVersion, initialValue); assertOrecValue(ref, orecValue); } @Test public void whenTransactionAborted_thenDeadTxnException() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); LongFunction function = mock(LongFunction.class); GammaTxn tx = transactionFactory.newTxn(); tx.abort(); long orecValue = ref.orec; try { ref.commute(tx, function); fail(); } catch (DeadTxnException expected) { } assertIsAborted(tx); verifyZeroInteractions(function); assertVersionAndValue(ref, initialVersion, initialValue); assertOrecValue(ref,orecValue); } @Test public void whenTransactionCommitted_thenDeadTxnException() { Long initialValue = 20L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); LongFunction function = mock(LongFunction.class); GammaTxn tx = transactionFactory.newTxn(); tx.commit(); long orecValue = ref.orec; try { ref.commute(tx, function); fail(); } catch (DeadTxnException expected) { } assertIsCommitted(tx); verifyZeroInteractions(function); assertVersionAndValue(ref, initialVersion, initialValue); assertOrecValue(ref, orecValue); } @Test public void whenTransactionPrepared_thenPreparedTxnException() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); LongFunction function = mock(LongFunction.class); GammaTxn tx = transactionFactory.newTxn(); tx.prepare(); long orecValue = ref.orec; try { ref.commute(tx, function); fail(); } catch (PreparedTxnException expected) { } assertIsAborted(tx); verifyZeroInteractions(function); assertVersionAndValue(ref, initialVersion, initialValue); assertOrecValue(ref, orecValue); } @Test public void fullExample() { assumeTrue(!(transactionFactory.newTxn() instanceof FatMonoGammaTxn)); Long initialValue = 10L; GammaTxnRef<Long> ref1 = new GammaTxnRef<Long>(stm, initialValue); GammaTxnRef<Long> ref2 = new GammaTxnRef<Long>(stm, initialValue); GammaTxn tx1 = transactionFactory.newTxn(); ref1.openForWrite(tx1, LOCKMODE_NONE).ref_value = new Long(11); ref2.commute(tx1, Functions.incLongFunction(1)); GammaTxn tx2 = transactionFactory.newTxn(); ref2.openForWrite(tx2, LOCKMODE_NONE).ref_value = new Long(11); tx2.commit(); tx1.commit(); assertIsCommitted(tx1); assertEquals(new Long(11), ref1.atomicGet()); assertEquals(new Long(12), ref2.atomicGet()); } @Test public void whenListenersAvailable() { Long initialValue = 10L; GammaTxnRef<Long> ref = new GammaTxnRef<Long>(stm, initialValue); long initialVersion = ref.getVersion(); TxnRefAwaitThread thread = new TxnRefAwaitThread(ref, initialValue + 1); thread.start(); sleepMs(500); GammaTxn tx = transactionFactory.newTxn(); ref.commute(tx, Functions.incLongFunction()); tx.commit(); joinAll(thread); assertRefHasNoLocks(ref); assertVersionAndValue(ref, initialVersion + 1, initialValue + 1); } }