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
8 package org.opendaylight.restconf.nb.rfc8040.rests.utils;
10 import static org.junit.Assert.assertThrows;
11 import static org.mockito.Mockito.doNothing;
12 import static org.mockito.Mockito.doReturn;
13 import static org.mockito.Mockito.mock;
14 import static org.mockito.Mockito.verify;
15 import static org.opendaylight.yangtools.util.concurrent.FluentFutures.immediateFalseFluentFuture;
16 import static org.opendaylight.yangtools.util.concurrent.FluentFutures.immediateFluentFuture;
18 import com.google.common.util.concurrent.Futures;
19 import java.util.Optional;
20 import org.junit.Before;
21 import org.junit.Test;
22 import org.junit.runner.RunWith;
23 import org.mockito.Mock;
24 import org.mockito.junit.MockitoJUnitRunner;
25 import org.opendaylight.mdsal.common.api.CommitInfo;
26 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
27 import org.opendaylight.mdsal.dom.api.DOMDataBroker;
28 import org.opendaylight.mdsal.dom.api.DOMDataTreeReadTransaction;
29 import org.opendaylight.mdsal.dom.api.DOMDataTreeReadWriteTransaction;
30 import org.opendaylight.mdsal.dom.api.DOMDataTreeWriteTransaction;
31 import org.opendaylight.mdsal.dom.spi.DefaultDOMRpcResult;
32 import org.opendaylight.netconf.dom.api.NetconfDataTreeService;
33 import org.opendaylight.restconf.common.context.InstanceIdentifierContext;
34 import org.opendaylight.restconf.common.errors.RestconfDocumentedException;
35 import org.opendaylight.restconf.nb.rfc8040.TestRestconfUtils;
36 import org.opendaylight.restconf.nb.rfc8040.legacy.NormalizedNodePayload;
37 import org.opendaylight.restconf.nb.rfc8040.rests.services.impl.RestconfDataServiceImpl;
38 import org.opendaylight.restconf.nb.rfc8040.rests.transactions.MdsalRestconfStrategy;
39 import org.opendaylight.restconf.nb.rfc8040.rests.transactions.NetconfRestconfStrategy;
40 import org.opendaylight.yangtools.yang.common.QName;
41 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
42 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier;
43 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifierWithPredicates;
44 import org.opendaylight.yangtools.yang.data.api.schema.ContainerNode;
45 import org.opendaylight.yangtools.yang.data.api.schema.LeafNode;
46 import org.opendaylight.yangtools.yang.data.api.schema.MapEntryNode;
47 import org.opendaylight.yangtools.yang.data.api.schema.MapNode;
48 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
49 import org.opendaylight.yangtools.yang.data.impl.schema.Builders;
50 import org.opendaylight.yangtools.yang.data.util.DataSchemaContextTree;
51 import org.opendaylight.yangtools.yang.model.api.DataSchemaNode;
52 import org.opendaylight.yangtools.yang.model.api.EffectiveModelContext;
53 import org.opendaylight.yangtools.yang.test.util.YangParserTestUtils;
55 @RunWith(MockitoJUnitRunner.StrictStubs.class)
56 public class PutDataTransactionUtilTest {
57 private static final String PATH_FOR_NEW_SCHEMA_CONTEXT = "/jukebox";
60 private DOMDataTreeReadWriteTransaction readWrite;
62 private DOMDataTreeReadTransaction read;
64 private DOMDataTreeWriteTransaction write;
66 private DOMDataBroker mockDataBroker;
68 private NetconfDataTreeService netconfService;
70 private LeafNode<?> buildLeaf;
71 private ContainerNode buildBaseCont;
72 private ContainerNode buildBaseContWithList;
73 private MapEntryNode buildListEntry;
74 private EffectiveModelContext schema;
75 private DataSchemaNode schemaNode;
76 private YangInstanceIdentifier iid;
77 private DataSchemaNode schemaNode2;
78 private YangInstanceIdentifier iid2;
79 private DataSchemaNode schemaNode3;
80 private YangInstanceIdentifier iid3;
83 public void setUp() throws Exception {
85 YangParserTestUtils.parseYangFiles(TestRestconfUtils.loadFiles(PATH_FOR_NEW_SCHEMA_CONTEXT));
87 final QName baseQName = QName.create("http://example.com/ns/example-jukebox", "2015-04-04", "jukebox");
88 final QName containerQname = QName.create(baseQName, "player");
89 final QName leafQname = QName.create(baseQName, "gap");
90 final QName listQname = QName.create(baseQName, "playlist");
91 final QName listKeyQname = QName.create(baseQName, "name");
93 final NodeIdentifierWithPredicates nodeWithKey =
94 NodeIdentifierWithPredicates.of(listQname, listKeyQname, "name of band");
95 final NodeIdentifierWithPredicates nodeWithKey2 =
96 NodeIdentifierWithPredicates.of(listQname, listKeyQname, "name of band 2");
98 final DataSchemaContextTree tree = DataSchemaContextTree.from(schema);
99 iid = YangInstanceIdentifier.builder()
101 .node(containerQname)
104 schemaNode = tree.findChild(iid).orElseThrow().getDataSchemaNode();
106 iid2 = YangInstanceIdentifier.builder()
109 schemaNode2 = tree.findChild(iid2).orElseThrow().getDataSchemaNode();
111 iid3 = YangInstanceIdentifier.builder()
116 schemaNode3 = tree.findChild(iid3).orElseThrow().getDataSchemaNode();
118 buildLeaf = Builders.leafBuilder()
119 .withNodeIdentifier(new NodeIdentifier(leafQname))
122 final ContainerNode buildPlayerCont = Builders.containerBuilder()
123 .withNodeIdentifier(new NodeIdentifier(containerQname))
124 .withChild(buildLeaf)
126 buildBaseCont = Builders.containerBuilder()
127 .withNodeIdentifier(new NodeIdentifier(baseQName))
128 .withChild(buildPlayerCont)
130 final LeafNode<Object> content = Builders.leafBuilder()
131 .withNodeIdentifier(new NodeIdentifier(QName.create(baseQName, "name")))
132 .withValue("name of band")
134 final LeafNode<Object> content2 = Builders.leafBuilder()
135 .withNodeIdentifier(new NodeIdentifier(QName.create(baseQName, "description")))
136 .withValue("band description")
138 buildListEntry = Builders.mapEntryBuilder()
139 .withNodeIdentifier(nodeWithKey)
143 final LeafNode<Object> content3 = Builders.leafBuilder()
144 .withNodeIdentifier(new NodeIdentifier(QName.create(baseQName, "name")))
145 .withValue("name of band 2")
147 final LeafNode<Object> content4 = Builders.leafBuilder()
148 .withNodeIdentifier(new NodeIdentifier(QName.create(baseQName, "description")))
149 .withValue("band description 2")
151 final MapEntryNode buildListEntry2 = Builders.mapEntryBuilder()
152 .withNodeIdentifier(nodeWithKey2)
156 final MapNode buildList = Builders.mapBuilder()
157 .withNodeIdentifier(new NodeIdentifier(listQname))
158 .withChild(buildListEntry)
159 .withChild(buildListEntry2)
161 buildBaseContWithList = Builders.containerBuilder()
162 .withNodeIdentifier(new NodeIdentifier(baseQName))
163 .withChild(buildList)
166 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService).lock();
167 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService).unlock();
171 public void testValidInputData() {
172 final InstanceIdentifierContext<DataSchemaNode> iidContext =
173 new InstanceIdentifierContext<>(iid, schemaNode, null, schema);
174 RestconfDataServiceImpl.validInputData(iidContext.getSchemaNode(),
175 NormalizedNodePayload.of(iidContext, buildLeaf));
179 public void testValidTopLevelNodeName() {
180 InstanceIdentifierContext<DataSchemaNode> iidContext =
181 new InstanceIdentifierContext<>(iid, schemaNode, null, schema);
182 NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildLeaf);
183 RestconfDataServiceImpl.validTopLevelNodeName(iidContext.getInstanceIdentifier(), payload);
185 iidContext = new InstanceIdentifierContext<>(iid2, schemaNode2, null, schema);
186 payload = NormalizedNodePayload.of(iidContext, buildBaseCont);
187 RestconfDataServiceImpl.validTopLevelNodeName(iidContext.getInstanceIdentifier(), payload);
191 public void testValidTopLevelNodeNamePathEmpty() {
192 final InstanceIdentifierContext<DataSchemaNode> iidContext =
193 new InstanceIdentifierContext<>(iid, schemaNode, null, schema);
194 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildLeaf);
196 // FIXME: more asserts
197 assertThrows(RestconfDocumentedException.class,
198 () -> RestconfDataServiceImpl.validTopLevelNodeName(YangInstanceIdentifier.empty(), payload));
202 public void testValidTopLevelNodeNameWrongTopIdentifier() {
203 final InstanceIdentifierContext<DataSchemaNode> iidContext =
204 new InstanceIdentifierContext<>(iid, schemaNode, null, schema);
205 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildLeaf);
207 // FIXME: more asserts
208 assertThrows(RestconfDocumentedException.class,
209 () -> RestconfDataServiceImpl.validTopLevelNodeName(iid.getAncestor(1), payload));
213 public void testValidateListKeysEqualityInPayloadAndUri() {
214 final InstanceIdentifierContext<DataSchemaNode> iidContext =
215 new InstanceIdentifierContext<>(iid3, schemaNode3, null, schema);
216 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildListEntry);
217 RestconfDataServiceImpl.validateListKeysEqualityInPayloadAndUri(payload);
221 public void testPutContainerData() {
222 final InstanceIdentifierContext<DataSchemaNode> iidContext =
223 new InstanceIdentifierContext<>(iid2, schemaNode2, null, schema);
224 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildBaseCont);
226 doReturn(readWrite).when(mockDataBroker).newReadWriteTransaction();
227 doReturn(read).when(mockDataBroker).newReadOnlyTransaction();
228 doReturn(immediateFalseFluentFuture())
229 .when(read).exists(LogicalDatastoreType.CONFIGURATION, iid2);
230 doNothing().when(readWrite).put(LogicalDatastoreType.CONFIGURATION,
231 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData());
232 doReturn(CommitInfo.emptyFluentFuture()).when(readWrite).commit();
234 PutDataTransactionUtil.putData(payload, schema, new MdsalRestconfStrategy(mockDataBroker), null, null);
235 verify(read).exists(LogicalDatastoreType.CONFIGURATION,
236 payload.getInstanceIdentifierContext().getInstanceIdentifier());
237 verify(readWrite).put(LogicalDatastoreType.CONFIGURATION,
238 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData());
242 public void testPutCreateContainerData() {
243 final InstanceIdentifierContext<DataSchemaNode> iidContext =
244 new InstanceIdentifierContext<>(iid2, schemaNode2, null, schema);
245 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildBaseCont);
247 doReturn(immediateFluentFuture(Optional.empty())).when(netconfService).getConfig(iid2);
248 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService).commit();
249 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService)
250 .replace(LogicalDatastoreType.CONFIGURATION, payload.getInstanceIdentifierContext().getInstanceIdentifier(),
251 payload.getData(), Optional.empty());
253 PutDataTransactionUtil.putData(payload, schema, new NetconfRestconfStrategy(netconfService),
255 verify(netconfService).lock();
256 verify(netconfService).getConfig(payload.getInstanceIdentifierContext().getInstanceIdentifier());
257 verify(netconfService).replace(LogicalDatastoreType.CONFIGURATION,
258 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData(), Optional.empty());
262 public void testPutReplaceContainerData() {
263 final InstanceIdentifierContext<DataSchemaNode> iidContext =
264 new InstanceIdentifierContext<>(iid2, schemaNode2, null, schema);
265 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildBaseCont);
267 doReturn(immediateFluentFuture(Optional.of(mock(NormalizedNode.class))))
268 .when(netconfService).getConfig(iid2);
269 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService).commit();
270 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService)
271 .replace(LogicalDatastoreType.CONFIGURATION,
272 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData(), Optional.empty());
274 PutDataTransactionUtil.putData(payload, schema, new NetconfRestconfStrategy(netconfService),
276 verify(netconfService).getConfig(payload.getInstanceIdentifierContext().getInstanceIdentifier());
277 verify(netconfService).replace(LogicalDatastoreType.CONFIGURATION,
278 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData(), Optional.empty());
282 public void testPutLeafData() {
283 final InstanceIdentifierContext<DataSchemaNode> iidContext =
284 new InstanceIdentifierContext<>(iid, schemaNode, null, schema);
285 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildLeaf);
287 doReturn(readWrite).when(mockDataBroker).newReadWriteTransaction();
288 doReturn(read).when(mockDataBroker).newReadOnlyTransaction();
289 doReturn(immediateFalseFluentFuture())
290 .when(read).exists(LogicalDatastoreType.CONFIGURATION, iid);
291 doNothing().when(readWrite).put(LogicalDatastoreType.CONFIGURATION,
292 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData());
293 doReturn(CommitInfo.emptyFluentFuture()).when(readWrite).commit();
295 PutDataTransactionUtil.putData(payload, schema, new MdsalRestconfStrategy(mockDataBroker), null, null);
296 verify(read).exists(LogicalDatastoreType.CONFIGURATION,
297 payload.getInstanceIdentifierContext().getInstanceIdentifier());
298 verify(readWrite).put(LogicalDatastoreType.CONFIGURATION,
299 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData());
303 public void testPutCreateLeafData() {
304 final InstanceIdentifierContext<DataSchemaNode> iidContext =
305 new InstanceIdentifierContext<>(iid, schemaNode, null, schema);
306 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildLeaf);
308 doReturn(immediateFluentFuture(Optional.empty())).when(netconfService).getConfig(iid);
309 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService).commit();
310 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService)
311 .replace(LogicalDatastoreType.CONFIGURATION,
312 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData(), Optional.empty());
314 PutDataTransactionUtil.putData(payload, schema, new NetconfRestconfStrategy(netconfService),
316 verify(netconfService).getConfig(payload.getInstanceIdentifierContext().getInstanceIdentifier());
317 verify(netconfService).replace(LogicalDatastoreType.CONFIGURATION,
318 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData(), Optional.empty());
322 public void testPutReplaceLeafData() {
323 final InstanceIdentifierContext<DataSchemaNode> iidContext =
324 new InstanceIdentifierContext<>(iid, schemaNode, null, schema);
325 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildLeaf);
327 doReturn(immediateFluentFuture(Optional.of(mock(NormalizedNode.class))))
328 .when(netconfService).getConfig(iid);
329 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService).commit();
330 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService)
331 .replace(LogicalDatastoreType.CONFIGURATION,
332 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData(), Optional.empty());
334 PutDataTransactionUtil.putData(payload, schema,
335 new NetconfRestconfStrategy(netconfService), null, null);
336 verify(netconfService).getConfig(payload.getInstanceIdentifierContext().getInstanceIdentifier());
337 verify(netconfService).replace(LogicalDatastoreType.CONFIGURATION,
338 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData(), Optional.empty());
342 public void testPutListData() {
343 final InstanceIdentifierContext<DataSchemaNode> iidContext =
344 new InstanceIdentifierContext<>(iid2, schemaNode2, null, schema);
345 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildBaseContWithList);
347 doReturn(readWrite).when(mockDataBroker).newReadWriteTransaction();
348 doReturn(read).when(mockDataBroker).newReadOnlyTransaction();
349 doReturn(immediateFalseFluentFuture())
350 .when(read).exists(LogicalDatastoreType.CONFIGURATION, iid2);
351 doNothing().when(readWrite).put(LogicalDatastoreType.CONFIGURATION,
352 payload.getInstanceIdentifierContext().getInstanceIdentifier(), payload.getData());
353 doReturn(CommitInfo.emptyFluentFuture()).when(readWrite).commit();
354 PutDataTransactionUtil.putData(payload, schema, new MdsalRestconfStrategy(mockDataBroker), null, null);
355 verify(read).exists(LogicalDatastoreType.CONFIGURATION, iid2);
356 verify(readWrite).put(LogicalDatastoreType.CONFIGURATION, iid2, payload.getData());
360 public void testPutCreateListData() {
361 final InstanceIdentifierContext<DataSchemaNode> iidContext =
362 new InstanceIdentifierContext<>(iid2, schemaNode2, null, schema);
363 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildBaseContWithList);
365 doReturn(immediateFluentFuture(Optional.empty())).when(netconfService)
367 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService).commit();
368 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService)
369 .replace(LogicalDatastoreType.CONFIGURATION, iid2, payload.getData(), Optional.empty());
371 PutDataTransactionUtil.putData(payload, schema, new NetconfRestconfStrategy(netconfService),
373 verify(netconfService).getConfig(iid2);
374 verify(netconfService).replace(LogicalDatastoreType.CONFIGURATION, iid2,
375 payload.getData(), Optional.empty());
379 public void testPutReplaceListData() {
380 final InstanceIdentifierContext<DataSchemaNode> iidContext =
381 new InstanceIdentifierContext<>(iid2, schemaNode2, null, schema);
382 final NormalizedNodePayload payload = NormalizedNodePayload.of(iidContext, buildBaseContWithList);
384 doReturn(immediateFluentFuture(Optional.of(mock(NormalizedNode.class)))).when(netconfService)
386 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService).commit();
387 doReturn(Futures.immediateFuture(new DefaultDOMRpcResult())).when(netconfService)
388 .replace(LogicalDatastoreType.CONFIGURATION,
389 iid2, payload.getData(), Optional.empty());
391 PutDataTransactionUtil.putData(payload, schema,
392 new NetconfRestconfStrategy(netconfService), null, null);
393 verify(netconfService).getConfig(iid2);
394 verify(netconfService).replace(LogicalDatastoreType.CONFIGURATION, iid2,
395 payload.getData(), Optional.empty());