2 * Copyright © 2017 Orange, Inc. and others. All rights reserved.
4 * This program and the accompanying materials are made available under the
5 * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6 * and is available at http://www.eclipse.org/legal/epl-v10.html
9 package org.opendaylight.transportpce.common.device;
11 import static org.mockito.ArgumentMatchers.any;
13 import com.google.common.util.concurrent.FluentFuture;
14 import com.google.common.util.concurrent.ListeningExecutorService;
15 import com.google.common.util.concurrent.MoreExecutors;
16 import java.util.LinkedList;
17 import java.util.List;
18 import java.util.Optional;
19 import java.util.concurrent.ExecutionException;
20 import java.util.concurrent.Executors;
21 import java.util.concurrent.Future;
22 import java.util.concurrent.TimeUnit;
23 import java.util.concurrent.TimeoutException;
24 import org.eclipse.jdt.annotation.NonNull;
25 import org.junit.After;
26 import org.junit.Assert;
27 import org.junit.Before;
28 import org.junit.Ignore;
29 import org.junit.Test;
30 import org.junit.runner.RunWith;
31 import org.mockito.Mock;
32 import org.mockito.Mockito;
33 import org.mockito.junit.MockitoJUnitRunner;
34 import org.opendaylight.mdsal.binding.api.DataBroker;
35 import org.opendaylight.mdsal.binding.api.MountPoint;
36 import org.opendaylight.mdsal.binding.api.MountPointService;
37 import org.opendaylight.mdsal.binding.api.ReadWriteTransaction;
38 import org.opendaylight.mdsal.common.api.CommitInfo;
39 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
40 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.network.rev180226.networks.Network;
41 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.network.rev180226.networks.NetworkBuilder;
42 import org.opendaylight.yangtools.util.concurrent.FluentFutures;
43 import org.opendaylight.yangtools.yang.binding.DataObject;
44 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
47 @RunWith(MockitoJUnitRunner.class)
48 public class DeviceTransactionManagerTest {
50 @Mock private MountPointService mountPointServiceMock;
51 @Mock private MountPoint mountPointMock;
52 @Mock private DataBroker dataBrokerMock;
53 @Mock private ReadWriteTransaction rwTransactionMock;
55 private DeviceTransactionManagerImpl transactionManager;
56 private String defaultDeviceId = "device-id";
57 private LogicalDatastoreType defaultDatastore = LogicalDatastoreType.OPERATIONAL;
58 private InstanceIdentifier<Network> defaultIid = InstanceIdentifier.create(Network.class);
59 private Network defaultData = new NetworkBuilder().build();
60 private long defaultTimeout = 1000;
61 private TimeUnit defaultTimeUnit = TimeUnit.MILLISECONDS;
64 public void before() {
65 Mockito.when(mountPointServiceMock.getMountPoint(any())).thenReturn(Optional.of(mountPointMock));
66 Mockito.when(mountPointMock.getService(any())).thenReturn(Optional.of(dataBrokerMock));
67 Mockito.when(dataBrokerMock.newReadWriteTransaction()).thenReturn(rwTransactionMock);
68 Mockito.when(rwTransactionMock.commit()).thenReturn(FluentFutures.immediateNullFluentFuture());
70 this.transactionManager = new DeviceTransactionManagerImpl(mountPointServiceMock, 3000);
75 transactionManager.preDestroy();
79 public void basicPositiveTransactionTest() {
81 putAndSubmit(transactionManager, defaultDeviceId, defaultDatastore, defaultIid, defaultData);
82 } catch (InterruptedException | ExecutionException e) {
83 Assert.fail("Exception catched! " + e);
87 Mockito.verify(rwTransactionMock, Mockito.times(1)).put(defaultDatastore, defaultIid, defaultData);
88 Mockito.verify(rwTransactionMock, Mockito.times(1)).commit();
93 public void advancedPositiveTransactionTest() {
95 Future<java.util.Optional<DeviceTransaction>> firstDeviceTxFuture =
96 transactionManager.getDeviceTransaction(defaultDeviceId);
97 DeviceTransaction firstDeviceTx = firstDeviceTxFuture.get().get();
99 Future<java.util.Optional<DeviceTransaction>> secondDeviceTxFuture =
100 transactionManager.getDeviceTransaction(defaultDeviceId);
101 Assert.assertFalse(secondDeviceTxFuture.isDone());
103 Future<java.util.Optional<DeviceTransaction>> thirdDeviceTxFuture =
104 transactionManager.getDeviceTransaction(defaultDeviceId);
105 Assert.assertFalse(thirdDeviceTxFuture.isDone());
107 firstDeviceTx.put(defaultDatastore, defaultIid, defaultData);
108 Assert.assertFalse(secondDeviceTxFuture.isDone());
109 Assert.assertFalse(thirdDeviceTxFuture.isDone());
111 Assert.assertFalse(secondDeviceTxFuture.isDone());
112 Assert.assertFalse(thirdDeviceTxFuture.isDone());
114 Future<java.util.Optional<DeviceTransaction>> anotherDeviceTxFuture =
115 transactionManager.getDeviceTransaction("another-id");
116 Assert.assertTrue(anotherDeviceTxFuture.isDone());
117 anotherDeviceTxFuture.get().get().commit(defaultTimeout, defaultTimeUnit);
119 firstDeviceTx.commit(defaultTimeout, defaultTimeUnit);
121 Assert.assertTrue(secondDeviceTxFuture.isDone());
122 Assert.assertFalse(thirdDeviceTxFuture.isDone());
124 DeviceTransaction secondDeviceTx = secondDeviceTxFuture.get().get();
125 secondDeviceTx.put(defaultDatastore, defaultIid, defaultData);
126 Assert.assertFalse(thirdDeviceTxFuture.isDone());
128 secondDeviceTx.commit(defaultTimeout, defaultTimeUnit);
130 Assert.assertTrue(thirdDeviceTxFuture.isDone());
132 DeviceTransaction thirdDeviceTx = thirdDeviceTxFuture.get().get();
133 thirdDeviceTx.put(defaultDatastore, defaultIid, defaultData);
134 thirdDeviceTx.commit(defaultTimeout, defaultTimeUnit);
136 Mockito.verify(rwTransactionMock, Mockito.times(3)).put(defaultDatastore, defaultIid, defaultData);
137 Mockito.verify(rwTransactionMock, Mockito.times(4)).commit();
138 } catch (InterruptedException | ExecutionException e) {
139 Assert.fail("Exception catched! " + e);
144 public void bigAmountOfTransactionsOnSameDeviceTest() {
145 int numberOfTxs = 100;
146 List<Future<java.util.Optional<DeviceTransaction>>> deviceTransactionFutures = new LinkedList<>();
147 List<DeviceTransaction> deviceTransactions = new LinkedList<>();
149 for (int i = 0; i < numberOfTxs; i++) {
150 deviceTransactionFutures.add(transactionManager.getDeviceTransaction(defaultDeviceId));
154 for (Future<java.util.Optional<DeviceTransaction>> futureTx : deviceTransactionFutures) {
155 DeviceTransaction deviceTx = futureTx.get().get();
156 deviceTx.commit(defaultTimeout, defaultTimeUnit);
157 deviceTransactions.add(deviceTx);
159 } catch (InterruptedException | ExecutionException e) {
160 Assert.fail("Exception catched! " + e);
163 for (DeviceTransaction deviceTx : deviceTransactions) {
164 Assert.assertTrue(deviceTx.wasSubmittedOrCancelled().get());
169 public void bigAmountOfTransactionsOnDifferentDevicesTest() {
170 int numberOfTxs = 1000;
171 List<DeviceTransaction> deviceTransactions = new LinkedList<>();
174 for (int i = 0; i < numberOfTxs; i++) {
175 deviceTransactions.add(transactionManager.getDeviceTransaction(defaultDeviceId + " " + i).get().get());
177 } catch (InterruptedException | ExecutionException e) {
178 Assert.fail("Exception catched! " + e);
181 deviceTransactions.parallelStream()
182 .forEach(deviceTransaction -> deviceTransaction.commit(defaultTimeout, defaultTimeUnit));
184 deviceTransactions.parallelStream()
185 .forEach(deviceTransaction -> Assert.assertTrue(deviceTransaction.wasSubmittedOrCancelled().get()));
189 public void bigAmountOfTransactionsOnDifferentDevicesWithoutSubmitTest() {
190 int numberOfTxs = 1000;
191 List<DeviceTransaction> deviceTransactions = new LinkedList<>();
194 for (int i = 0; i < numberOfTxs; i++) {
195 deviceTransactions.add(transactionManager.getDeviceTransaction(defaultDeviceId + " " + i).get().get());
197 } catch (InterruptedException | ExecutionException e) {
198 Assert.fail("Exception catched! " + e);
202 Thread.sleep(transactionManager.getMaxDurationToSubmitTransaction() + 1000);
203 } catch (InterruptedException e) {
204 Assert.fail("Exception catched! " + e);
206 deviceTransactions.parallelStream()
207 .forEach(deviceTransaction -> Assert.assertTrue(deviceTransaction.wasSubmittedOrCancelled().get()));
211 public void notSubmittedTransactionTest() {
212 Future<java.util.Optional<DeviceTransaction>> deviceTxFuture =
213 transactionManager.getDeviceTransaction(defaultDeviceId);
215 deviceTxFuture.get();
216 Thread.sleep(transactionManager.getMaxDurationToSubmitTransaction() + 1000);
217 } catch (InterruptedException | ExecutionException e) {
218 Assert.fail("Exception catched! " + e);
220 Mockito.verify(rwTransactionMock, Mockito.times(1)).cancel();
223 putAndSubmit(transactionManager, defaultDeviceId, defaultDatastore, defaultIid, defaultData);
224 } catch (InterruptedException | ExecutionException e) {
225 Assert.fail("Exception catched! " + e);
229 Mockito.verify(rwTransactionMock, Mockito.times(1)).cancel();
230 Mockito.verify(rwTransactionMock, Mockito.times(1)).put(defaultDatastore, defaultIid, defaultData);
231 Mockito.verify(rwTransactionMock, Mockito.times(1)).commit();
235 public void dataBrokerTimeoutTransactionTest() {
236 Mockito.when(dataBrokerMock.newReadWriteTransaction()).then(invocation -> {
237 Thread.sleep(transactionManager.getMaxDurationToSubmitTransaction() + 1000);
238 return rwTransactionMock;
242 putAndSubmit(transactionManager, defaultDeviceId, defaultDatastore, defaultIid, defaultData);
243 } catch (InterruptedException | ExecutionException e) {
244 Assert.fail("Exception catched! " + e);
247 Mockito.verify(rwTransactionMock, Mockito.times(1)).commit();
249 Mockito.when(dataBrokerMock.newReadWriteTransaction()).thenReturn(rwTransactionMock); // remove sleep
252 putAndSubmit(transactionManager, defaultDeviceId, defaultDatastore, defaultIid, defaultData);
253 } catch (InterruptedException | ExecutionException e) {
254 Assert.fail("Exception catched! " + e);
258 Mockito.verify(rwTransactionMock, Mockito.times(2)).put(defaultDatastore, defaultIid, defaultData);
259 Mockito.verify(rwTransactionMock, Mockito.times(2)).commit();
263 public void getFutureTimeoutTransactionTest() {
264 Mockito.when(dataBrokerMock.newReadWriteTransaction()).then(invocation -> {
266 return rwTransactionMock;
269 Exception throwedException = null;
271 Future<java.util.Optional<DeviceTransaction>> deviceTxFuture =
272 transactionManager.getDeviceTransaction(defaultDeviceId);
274 deviceTxFuture.get(1000, TimeUnit.MILLISECONDS);
275 } catch (InterruptedException | ExecutionException e) {
276 Assert.fail("Exception catched! " + e);
277 } catch (TimeoutException e) {
278 throwedException = e;
281 if (throwedException == null) {
282 Assert.fail("TimeoutException should be thrown!");
286 Mockito.when(dataBrokerMock.newReadWriteTransaction()).thenReturn(rwTransactionMock); // remove sleep
289 putAndSubmit(transactionManager, defaultDeviceId, defaultDatastore, defaultIid, defaultData);
290 } catch (InterruptedException | ExecutionException e) {
291 Assert.fail("Exception catched! " + e);
295 Mockito.verify(rwTransactionMock, Mockito.times(1)).put(defaultDatastore, defaultIid, defaultData);
296 Mockito.verify(rwTransactionMock, Mockito.times(1)).commit();
301 public void submitTxTimeoutTransactionTest() {
302 Future<java.util.Optional<DeviceTransaction>> deviceTxFuture =
303 transactionManager.getDeviceTransaction(defaultDeviceId);
304 DeviceTransaction deviceTx;
306 deviceTx = deviceTxFuture.get().get();
307 } catch (InterruptedException | ExecutionException e) {
308 Assert.fail("Exception catched! " + e);
312 deviceTx.put(defaultDatastore, defaultIid, defaultData);
314 Exception throwedException = null;
316 FluentFuture<? extends @NonNull CommitInfo> submitFuture = deviceTx.commit(200, defaultTimeUnit);
319 } catch (InterruptedException e) {
320 Assert.fail("Exception catched! " + e);
321 } catch (ExecutionException e) {
322 throwedException = e;
325 if (throwedException == null
326 || !throwedException.getMessage().contains(TimeoutException.class.getName())) {
327 Assert.fail("TimeoutException inside of should be thrown!");
332 Mockito.doReturn(FluentFutures.immediateNullFluentFuture()).when(rwTransactionMock.commit());
335 putAndSubmit(transactionManager, defaultDeviceId, defaultDatastore, defaultIid, defaultData);
336 } catch (InterruptedException | ExecutionException e) {
337 Assert.fail("Exception catched! " + e);
341 Mockito.verify(rwTransactionMock, Mockito.times(2)).put(defaultDatastore, defaultIid, defaultData);
342 Mockito.verify(rwTransactionMock, Mockito.times(2)).commit();
343 ListeningExecutorService executor = MoreExecutors
344 .listeningDecorator(Executors.newSingleThreadExecutor());
348 private <T extends DataObject> void putAndSubmit(DeviceTransactionManagerImpl deviceTxManager, String deviceId,
349 LogicalDatastoreType store, InstanceIdentifier<T> path, T data)
350 throws ExecutionException, InterruptedException {
351 Future<java.util.Optional<DeviceTransaction>> deviceTxFuture = deviceTxManager.getDeviceTransaction(deviceId);
352 DeviceTransaction deviceTx = deviceTxFuture.get().get();
353 deviceTx.put(store, path, data);
354 deviceTx.commit(defaultTimeout, defaultTimeUnit);