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.Mockito.doNothing;
15 import static org.mockito.Mockito.doReturn;
16 import static org.mockito.Mockito.mock;
17 import static org.mockito.Mockito.timeout;
18 import static org.mockito.Mockito.verify;
19 import static org.mockito.MockitoAnnotations.initMocks;
20 import static org.opendaylight.netconf.topology.singleton.impl.utils.NetconfTopologyUtils.DEFAULT_SCHEMA_REPOSITORY;
22 import akka.actor.ActorRef;
23 import akka.actor.ActorSystem;
24 import akka.actor.Props;
25 import akka.pattern.Patterns;
26 import akka.testkit.JavaTestKit;
27 import akka.testkit.TestActorRef;
28 import akka.util.Timeout;
29 import com.google.common.collect.Lists;
30 import com.google.common.util.concurrent.CheckedFuture;
31 import com.google.common.util.concurrent.Futures;
32 import java.net.InetAddress;
33 import java.net.InetSocketAddress;
34 import java.util.List;
35 import java.util.concurrent.TimeUnit;
36 import org.junit.After;
37 import org.junit.Before;
38 import org.junit.Rule;
39 import org.junit.Test;
40 import org.junit.rules.ExpectedException;
41 import org.mockito.Mock;
42 import org.opendaylight.controller.md.sal.binding.api.DataBroker;
43 import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
44 import org.opendaylight.controller.md.sal.common.api.data.TransactionCommitFailedException;
45 import org.opendaylight.controller.md.sal.dom.api.DOMDataBroker;
46 import org.opendaylight.controller.md.sal.dom.api.DOMDataReadOnlyTransaction;
47 import org.opendaylight.controller.md.sal.dom.api.DOMDataWriteTransaction;
48 import org.opendaylight.controller.md.sal.dom.api.DOMMountPointService;
49 import org.opendaylight.controller.md.sal.dom.api.DOMRpcService;
50 import org.opendaylight.netconf.sal.connect.util.RemoteDeviceId;
51 import org.opendaylight.netconf.topology.singleton.impl.ProxyDOMDataBroker;
52 import org.opendaylight.netconf.topology.singleton.impl.actors.NetconfNodeActor;
53 import org.opendaylight.netconf.topology.singleton.impl.utils.NetconfTopologySetup;
54 import org.opendaylight.netconf.topology.singleton.messages.CreateInitialMasterActorData;
55 import org.opendaylight.netconf.topology.singleton.messages.MasterActorDataInitialized;
56 import org.opendaylight.yangtools.yang.common.QName;
57 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
58 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
59 import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNodes;
60 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableContainerNodeBuilder;
61 import org.opendaylight.yangtools.yang.model.repo.api.SourceIdentifier;
62 import scala.concurrent.Await;
63 import scala.concurrent.Future;
64 import scala.concurrent.duration.Duration;
66 public class WriteOnlyTransactionTest {
67 private static final Timeout TIMEOUT = new Timeout(Duration.create(5, "seconds"));
68 private static final int TIMEOUT_SEC = 5;
69 private static ActorSystem system;
72 public final ExpectedException exception = ExpectedException.none();
74 private ActorRef masterRef;
75 private ProxyDOMDataBroker slaveDataBroker;
76 private List<SourceIdentifier> sourceIdentifiers;
78 private DOMDataBroker deviceDataBroker;
80 private DOMDataWriteTransaction writeTx;
82 private DOMRpcService domRpcService;
84 private DOMMountPointService mountPointService;
86 private NormalizedNode<?, ?> testNode;
87 private YangInstanceIdentifier instanceIdentifier;
88 private LogicalDatastoreType storeType;
91 public void setup() throws Exception {
94 system = ActorSystem.create();
96 final RemoteDeviceId remoteDeviceId = new RemoteDeviceId("netconf-topology",
97 new InetSocketAddress(InetAddress.getByName("127.0.0.1"), 9999));
99 final NetconfTopologySetup setup = mock(NetconfTopologySetup.class);
100 doReturn(Duration.apply(0, TimeUnit.SECONDS)).when(setup).getIdleTimeout();
101 final Props props = NetconfNodeActor.props(setup, remoteDeviceId, DEFAULT_SCHEMA_REPOSITORY,
102 DEFAULT_SCHEMA_REPOSITORY, TIMEOUT, mountPointService);
104 masterRef = TestActorRef.create(system, props, "master_read");
106 sourceIdentifiers = Lists.newArrayList();
108 final DOMDataReadOnlyTransaction readTx = mock(DOMDataReadOnlyTransaction.class);
110 doReturn(writeTx).when(deviceDataBroker).newWriteOnlyTransaction();
111 doReturn(readTx).when(deviceDataBroker).newReadOnlyTransaction();
112 doNothing().when(writeTx).put(storeType, instanceIdentifier, testNode);
113 doNothing().when(writeTx).merge(storeType, instanceIdentifier, testNode);
114 doNothing().when(writeTx).delete(storeType, instanceIdentifier);
116 // Create slave data broker for testing proxy
118 new ProxyDOMDataBroker(system, remoteDeviceId, masterRef, Timeout.apply(5, TimeUnit.SECONDS));
119 initializeDataTest();
120 testNode = ImmutableContainerNodeBuilder.create()
121 .withNodeIdentifier(new YangInstanceIdentifier.NodeIdentifier(QName.create("TestQname")))
122 .withChild(ImmutableNodes.leafNode(QName.create("NodeQname"), "foo")).build();
123 instanceIdentifier = YangInstanceIdentifier.EMPTY;
124 storeType = LogicalDatastoreType.CONFIGURATION;
128 public void teardown() {
129 JavaTestKit.shutdownActorSystem(system, null, true);
134 public void testPut() throws Exception {
135 // Test of invoking put on master through slave proxy
136 final DOMDataWriteTransaction wTx = slaveDataBroker.newWriteOnlyTransaction();
137 wTx.put(storeType, instanceIdentifier, testNode);
139 verify(writeTx, timeout(2000)).put(storeType, instanceIdentifier, testNode);
145 public void testMerge() throws Exception {
146 // Test of invoking merge on master through slave proxy
147 final DOMDataWriteTransaction wTx = slaveDataBroker.newWriteOnlyTransaction();
148 wTx.merge(storeType, instanceIdentifier, testNode);
150 verify(writeTx, timeout(2000)).merge(storeType, instanceIdentifier, testNode);
156 public void testDelete() throws Exception {
157 final YangInstanceIdentifier instanceIdentifier = YangInstanceIdentifier.EMPTY;
158 final LogicalDatastoreType storeType = LogicalDatastoreType.CONFIGURATION;
160 // Test of invoking delete on master through slave proxy
161 final DOMDataWriteTransaction wTx = slaveDataBroker.newWriteOnlyTransaction();
162 wTx.delete(storeType, instanceIdentifier);
165 verify(writeTx, timeout(2000)).delete(storeType, instanceIdentifier);
169 public void testSubmit() throws Exception {
170 final CheckedFuture<Void, TransactionCommitFailedException> resultSubmit = Futures.immediateCheckedFuture(null);
171 doReturn(resultSubmit).when(writeTx).submit();
174 final DOMDataWriteTransaction wTx = slaveDataBroker.newWriteOnlyTransaction();
176 final CheckedFuture<Void, TransactionCommitFailedException> resultSubmitResponse = wTx.submit();
178 final Object result = resultSubmitResponse.checkedGet(TIMEOUT_SEC, TimeUnit.SECONDS);
184 public void testSubmitWithOperation() throws Exception {
185 final CheckedFuture<Void, TransactionCommitFailedException> resultSubmitTx = Futures.immediateCheckedFuture(null);
186 doReturn(resultSubmitTx).when(writeTx).submit();
188 final DOMDataWriteTransaction wTx = slaveDataBroker.newWriteOnlyTransaction();
189 wTx.delete(LogicalDatastoreType.CONFIGURATION,
190 YangInstanceIdentifier.EMPTY);
192 final CheckedFuture<Void, TransactionCommitFailedException> resultSubmitTxResponse = wTx.submit();
194 final Object resultTx = resultSubmitTxResponse.checkedGet(TIMEOUT_SEC, TimeUnit.SECONDS);
196 assertNull(resultTx);
200 public void testSubmitFail() throws Exception {
201 final TransactionCommitFailedException throwable = new TransactionCommitFailedException("Fail", null);
202 final CheckedFuture<Void, TransactionCommitFailedException> resultThrowable =
203 Futures.immediateFailedCheckedFuture(throwable);
204 doReturn(resultThrowable).when(writeTx).submit();
206 final DOMDataWriteTransaction wTx = slaveDataBroker.newWriteOnlyTransaction();
207 wTx.delete(LogicalDatastoreType.CONFIGURATION,
208 YangInstanceIdentifier.EMPTY);
209 final CheckedFuture<Void, TransactionCommitFailedException> resultThrowableResponse =
211 exception.expect(TransactionCommitFailedException.class);
212 resultThrowableResponse.checkedGet(TIMEOUT_SEC, TimeUnit.SECONDS);
216 public void testCancel() throws Exception {
217 doReturn(true).when(writeTx).cancel();
220 final DOMDataWriteTransaction wTx = slaveDataBroker.newWriteOnlyTransaction();
221 final Boolean resultFalseNoTx = wTx.cancel();
222 assertEquals(true, resultFalseNoTx);
226 public void testCancelWithOperation() throws Exception {
227 doReturn(true).when(writeTx).cancel();
229 // With Tx, readWriteTx test
230 final DOMDataWriteTransaction wTx = slaveDataBroker.newWriteOnlyTransaction();
231 wTx.delete(LogicalDatastoreType.CONFIGURATION,
232 YangInstanceIdentifier.EMPTY);
234 final Boolean resultTrue = wTx.cancel();
235 assertEquals(true, resultTrue);
237 final Boolean resultFalse = wTx.cancel();
238 assertEquals(false, resultFalse);
241 private void initializeDataTest() throws Exception {
242 final Future<Object> initialDataToActor =
243 Patterns.ask(masterRef, new CreateInitialMasterActorData(deviceDataBroker, sourceIdentifiers,
244 domRpcService), TIMEOUT);
246 final Object success = Await.result(initialDataToActor, TIMEOUT.duration());
248 assertTrue(success instanceof MasterActorDataInitialized);