2 * Copyright (c) 2016 Cisco Systems, 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.netconf.topology.singleton.impl.tx;
11 import static junit.framework.TestCase.assertNull;
12 import static org.junit.Assert.assertEquals;
13 import static org.junit.Assert.assertTrue;
14 import static org.mockito.Matchers.any;
15 import static org.mockito.Mockito.doNothing;
16 import static org.mockito.Mockito.doReturn;
17 import static org.mockito.Mockito.mock;
18 import static org.mockito.Mockito.times;
19 import static org.mockito.Mockito.verify;
20 import static org.mockito.MockitoAnnotations.initMocks;
21 import static org.opendaylight.netconf.topology.singleton.impl.utils.NetconfTopologyUtils.DEFAULT_SCHEMA_REPOSITORY;
23 import akka.actor.ActorRef;
24 import akka.actor.ActorSystem;
25 import akka.actor.Props;
26 import akka.pattern.Patterns;
27 import akka.testkit.JavaTestKit;
28 import akka.testkit.TestActorRef;
29 import akka.util.Timeout;
30 import com.google.common.collect.Lists;
31 import com.google.common.util.concurrent.CheckedFuture;
32 import com.google.common.util.concurrent.Futures;
33 import java.net.InetAddress;
34 import java.net.InetSocketAddress;
35 import java.net.UnknownHostException;
36 import java.util.List;
37 import java.util.concurrent.TimeUnit;
38 import org.junit.After;
39 import org.junit.Before;
40 import org.junit.Rule;
41 import org.junit.Test;
42 import org.junit.rules.ExpectedException;
43 import org.mockito.Mock;
44 import org.mockito.Mockito;
45 import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
46 import org.opendaylight.controller.md.sal.common.api.data.TransactionCommitFailedException;
47 import org.opendaylight.controller.md.sal.dom.api.DOMDataBroker;
48 import org.opendaylight.controller.md.sal.dom.api.DOMDataReadOnlyTransaction;
49 import org.opendaylight.controller.md.sal.dom.api.DOMDataWriteTransaction;
50 import org.opendaylight.controller.md.sal.dom.api.DOMRpcService;
51 import org.opendaylight.netconf.sal.connect.util.RemoteDeviceId;
52 import org.opendaylight.netconf.topology.singleton.api.NetconfDOMTransaction;
53 import org.opendaylight.netconf.topology.singleton.impl.NetconfDOMDataBroker;
54 import org.opendaylight.netconf.topology.singleton.impl.actors.NetconfNodeActor;
55 import org.opendaylight.netconf.topology.singleton.impl.utils.NetconfTopologySetup;
56 import org.opendaylight.netconf.topology.singleton.messages.CreateInitialMasterActorData;
57 import org.opendaylight.netconf.topology.singleton.messages.MasterActorDataInitialized;
58 import org.opendaylight.yangtools.yang.common.QName;
59 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
60 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
61 import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNodes;
62 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableContainerNodeBuilder;
63 import org.opendaylight.yangtools.yang.model.repo.api.SourceIdentifier;
64 import scala.concurrent.Await;
65 import scala.concurrent.Future;
66 import scala.concurrent.duration.Duration;
68 public class WriteOnlyTransactionTest {
69 private static final Timeout TIMEOUT = new Timeout(Duration.create(5, "seconds"));
70 private static final int TIMEOUT_SEC = 5;
71 private static ActorSystem system;
74 public final ExpectedException exception = ExpectedException.none();
76 private ActorRef masterRef;
77 private NetconfDOMDataBroker slaveDataBroker;
78 private DOMDataBroker masterDataBroker;
79 private List<SourceIdentifier> sourceIdentifiers;
82 private DOMDataWriteTransaction writeTx;
85 private DOMRpcService domRpcService;
88 public void setup() throws UnknownHostException {
91 system = ActorSystem.create();
93 final RemoteDeviceId remoteDeviceId = new RemoteDeviceId("netconf-topology",
94 new InetSocketAddress(InetAddress.getByName("127.0.0.1"), 9999));
96 final NetconfTopologySetup setup = mock(NetconfTopologySetup.class);
97 final Props props = NetconfNodeActor.props(setup, remoteDeviceId, DEFAULT_SCHEMA_REPOSITORY,
98 DEFAULT_SCHEMA_REPOSITORY);
100 masterRef = TestActorRef.create(system, props, "master_read");
102 sourceIdentifiers = Lists.newArrayList();
104 // Create master data broker
106 final DOMDataBroker delegateDataBroker = mock(DOMDataBroker.class);
107 writeTx = mock(DOMDataWriteTransaction.class);
108 final DOMDataReadOnlyTransaction readTx = mock(DOMDataReadOnlyTransaction.class);
110 doReturn(writeTx).when(delegateDataBroker).newWriteOnlyTransaction();
111 doReturn(readTx).when(delegateDataBroker).newReadOnlyTransaction();
113 final NetconfDOMTransaction masterDOMTransactions =
114 new NetconfMasterDOMTransaction(remoteDeviceId, delegateDataBroker);
117 new NetconfDOMDataBroker(system, remoteDeviceId, masterDOMTransactions);
119 // Create slave data broker for testing proxy
121 final NetconfDOMTransaction proxyDOMTransactions =
122 new NetconfProxyDOMTransaction(remoteDeviceId, system, masterRef);
124 slaveDataBroker = new NetconfDOMDataBroker(system, remoteDeviceId, proxyDOMTransactions);
130 public void teardown() {
131 JavaTestKit.shutdownActorSystem(system);
136 public void testPutMergeDeleteCalls() throws Exception {
138 /* Initialize data on master */
140 initializeDataTest();
142 final YangInstanceIdentifier instanceIdentifier = YangInstanceIdentifier.EMPTY;
143 final LogicalDatastoreType storeType = LogicalDatastoreType.CONFIGURATION;
144 final NormalizedNode<?, ?> testNode = ImmutableContainerNodeBuilder.create()
145 .withNodeIdentifier(new YangInstanceIdentifier.NodeIdentifier(QName.create("TestQname")))
146 .withChild(ImmutableNodes.leafNode(QName.create("NodeQname"), "foo")).build();
148 // Test of invoking put on master through slave proxy
150 doNothing().when(writeTx).put(storeType, instanceIdentifier, testNode);
151 slaveDataBroker.newWriteOnlyTransaction().put(storeType, instanceIdentifier, testNode);
153 verify(writeTx, times(1)).put(storeType, instanceIdentifier, testNode);
155 // Test of invoking merge on master through slave proxy
157 doNothing().when(writeTx).merge(storeType, instanceIdentifier, testNode);
158 slaveDataBroker.newWriteOnlyTransaction().merge(storeType, instanceIdentifier, testNode);
160 verify(writeTx, times(1)).merge(storeType, instanceIdentifier, testNode);
162 // Test of invoking delete on master through slave proxy
164 doNothing().when(writeTx).delete(storeType, instanceIdentifier);
165 slaveDataBroker.newWriteOnlyTransaction().delete(storeType, instanceIdentifier);
167 verify(writeTx, times(1)).delete(storeType, instanceIdentifier);
172 public void testSubmit() throws Exception {
174 /* Initialize data on master */
176 initializeDataTest();
180 final CheckedFuture<Void,TransactionCommitFailedException> resultSubmit = Futures.immediateCheckedFuture(null);
181 doReturn(resultSubmit).when(writeTx).submit();
183 final CheckedFuture<Void, TransactionCommitFailedException> resultSubmitResponse =
184 slaveDataBroker.newWriteOnlyTransaction().submit();
186 final Object result= resultSubmitResponse.checkedGet(TIMEOUT_SEC, TimeUnit.SECONDS);
192 doNothing().when(writeTx).delete(any(), any());
193 slaveDataBroker.newWriteOnlyTransaction().delete(LogicalDatastoreType.CONFIGURATION,
194 YangInstanceIdentifier.EMPTY);
196 final CheckedFuture<Void,TransactionCommitFailedException> resultSubmitTx = Futures.immediateCheckedFuture(null);
197 doReturn(resultSubmitTx).when(writeTx).submit();
199 final CheckedFuture<Void, TransactionCommitFailedException> resultSubmitTxResponse =
200 slaveDataBroker.newWriteOnlyTransaction().submit();
202 final Object resultTx = resultSubmitTxResponse.checkedGet(TIMEOUT_SEC, TimeUnit.SECONDS);
204 assertNull(resultTx);
206 slaveDataBroker.newWriteOnlyTransaction().delete(LogicalDatastoreType.CONFIGURATION,
207 YangInstanceIdentifier.EMPTY);
209 final TransactionCommitFailedException throwable = new TransactionCommitFailedException("Fail", null);
210 final CheckedFuture<Void,TransactionCommitFailedException> resultThrowable =
211 Futures.immediateFailedCheckedFuture(throwable);
213 doReturn(resultThrowable).when(writeTx).submit();
215 final CheckedFuture<Void, TransactionCommitFailedException> resultThrowableResponse =
216 slaveDataBroker.newWriteOnlyTransaction().submit();
218 exception.expect(TransactionCommitFailedException.class);
219 resultThrowableResponse.checkedGet(TIMEOUT_SEC, TimeUnit.SECONDS);
223 public void testCancel() throws Exception {
225 /* Initialize data on master */
227 initializeDataTest();
231 final Boolean resultFalseNoTx = slaveDataBroker.newWriteOnlyTransaction().cancel();
232 assertEquals(false, resultFalseNoTx);
234 // With Tx, readWriteTx test
236 doNothing().when(writeTx).delete(any(), any());
237 slaveDataBroker.newReadWriteTransaction().delete(LogicalDatastoreType.CONFIGURATION,
238 YangInstanceIdentifier.EMPTY);
240 doReturn(true).when(writeTx).cancel();
242 final Boolean resultTrue = slaveDataBroker.newWriteOnlyTransaction().cancel();
243 assertEquals(true, resultTrue);
245 doReturn(false).when(writeTx).cancel();
247 final Boolean resultFalse = slaveDataBroker.newWriteOnlyTransaction().cancel();
248 assertEquals(false, resultFalse);
252 private void initializeDataTest() throws Exception {
253 final Future<Object> initialDataToActor =
254 Patterns.ask(masterRef, new CreateInitialMasterActorData(masterDataBroker, sourceIdentifiers,
255 domRpcService), TIMEOUT);
257 final Object success = Await.result(initialDataToActor, TIMEOUT.duration());
259 assertTrue(success instanceof MasterActorDataInitialized);