schemaContextRef.get());
final DOMMountPoint mountPoint = instanceIdentifier.getMountPoint();
- DOMDataReadWriteTransaction transaction = null;
+ final DOMDataReadWriteTransaction transaction;
if (mountPoint == null) {
transaction = this.transactionChainHandler.get().newReadWriteTransaction();
} else {
import javax.ws.rs.core.Response;
import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
import org.opendaylight.controller.md.sal.common.api.data.TransactionCommitFailedException;
-import org.opendaylight.controller.md.sal.dom.api.DOMDataWriteTransaction;
+import org.opendaylight.controller.md.sal.dom.api.DOMDataReadWriteTransaction;
import org.opendaylight.restconf.restful.transaction.TransactionVarsWrapper;
import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
* - Wrapper for data of transaction
* @return {@link Response}
*/
-
public static Response deleteData(final TransactionVarsWrapper transactionNode) {
final CheckedFuture<Void, TransactionCommitFailedException> future = submitData(
transactionNode.getTransaction(), transactionNode.getInstanceIdentifier().getInstanceIdentifier());
}
/**
- * Delete data via transaction
+ * Delete data via transaction. Return error if data to delete does not exist.
*
- * @param writeTx
- * - write transaction
+ * @param readWriteTx
+ * - read and write transaction
* @param path
* - path of data to delete
* @return {@link CheckedFuture}
*/
private static CheckedFuture<Void, TransactionCommitFailedException> submitData(
- final DOMDataWriteTransaction writeTx, final YangInstanceIdentifier path) {
- writeTx.delete(LogicalDatastoreType.CONFIGURATION, path);
- return writeTx.submit();
+ final DOMDataReadWriteTransaction readWriteTx, final YangInstanceIdentifier path) {
+ TransactionUtil.checkItemExists(readWriteTx, LogicalDatastoreType.CONFIGURATION, path,
+ RestconfDataServiceConstant.DeleteData.DELETE_TX_TYPE);
+ readWriteTx.delete(LogicalDatastoreType.CONFIGURATION, path);
+ return readWriteTx.submit();
}
}
package org.opendaylight.restconf.restful.utils;
import com.google.common.util.concurrent.CheckedFuture;
-import com.google.common.util.concurrent.ListenableFuture;
import java.net.URI;
-import java.util.concurrent.ExecutionException;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.UriBuilder;
import javax.ws.rs.core.UriInfo;
import org.opendaylight.controller.md.sal.common.api.data.TransactionCommitFailedException;
import org.opendaylight.controller.md.sal.dom.api.DOMDataReadWriteTransaction;
import org.opendaylight.netconf.sal.restconf.impl.NormalizedNodeContext;
-import org.opendaylight.netconf.sal.restconf.impl.RestconfDocumentedException;
-import org.opendaylight.netconf.sal.restconf.impl.RestconfError.ErrorTag;
-import org.opendaylight.netconf.sal.restconf.impl.RestconfError.ErrorType;
import org.opendaylight.restconf.common.references.SchemaContextRef;
import org.opendaylight.restconf.restful.transaction.TransactionVarsWrapper;
import org.opendaylight.restconf.utils.parser.ParserIdentifier;
private static void putChild(final NormalizedNode<?, ?> child, final DOMDataReadWriteTransaction readWriteTx,
final YangInstanceIdentifier path) {
final YangInstanceIdentifier childPath = path.node(child.getIdentifier());
- checkItemDesNotExits(childPath, readWriteTx);
+ TransactionUtil.checkItemDoesNotExists(readWriteTx, LogicalDatastoreType.CONFIGURATION, childPath,
+ RestconfDataServiceConstant.PostData.POST_TX_TYPE);
readWriteTx.put(LogicalDatastoreType.CONFIGURATION, childPath, child);
}
- /**
- * Check if data posted to create doesn't exits.
- *
- * @param path
- * - path to data
- * @param readWriteTx
- * - read write transaction
- */
- private static void checkItemDesNotExits(final YangInstanceIdentifier path,
- final DOMDataReadWriteTransaction readWriteTx) {
- final ListenableFuture<Boolean> existData = readWriteTx.exists(LogicalDatastoreType.CONFIGURATION, path);
- try {
- if (existData.get()) {
- readWriteTx.cancel();
- throw new RestconfDocumentedException("Data already exists for path: " + path, ErrorType.PROTOCOL,
- ErrorTag.DATA_EXISTS);
- }
- } catch (InterruptedException | ExecutionException e) {
- LOG.warn("It wasn't possible to get data loaded from datastore at path {}", path, e);
- }
- }
-
/**
* Get location from {@link YangInstanceIdentifier} and {@link UriInfo}
*
return uriBuilder.build();
}
}
-
--- /dev/null
+/*
+ * Copyright (c) 2016 Cisco Systems, Inc. and others. All rights reserved.
+ *
+ * This program and the accompanying materials are made available under the
+ * terms of the Eclipse Public License v1.0 which accompanies this distribution,
+ * and is available at http://www.eclipse.org/legal/epl-v10.html
+ */
+
+package org.opendaylight.restconf.restful.utils;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.fail;
+
+import com.google.common.util.concurrent.Futures;
+import javax.ws.rs.core.Response;
+import javax.ws.rs.core.Response.Status;
+import org.junit.Before;
+import org.junit.Test;
+import org.mockito.Mock;
+import org.mockito.Mockito;
+import org.mockito.MockitoAnnotations;
+import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
+import org.opendaylight.controller.md.sal.dom.api.DOMDataReadWriteTransaction;
+import org.opendaylight.netconf.sal.restconf.impl.InstanceIdentifierContext;
+import org.opendaylight.netconf.sal.restconf.impl.RestconfDocumentedException;
+import org.opendaylight.netconf.sal.restconf.impl.RestconfError.ErrorTag;
+import org.opendaylight.netconf.sal.restconf.impl.RestconfError.ErrorType;
+import org.opendaylight.restconf.restful.transaction.TransactionVarsWrapper;
+import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
+
+public class DeleteDataTransactionUtilTest {
+ @Mock DOMDataReadWriteTransaction transaction;
+ @Mock InstanceIdentifierContext<?> context;
+
+ @Before
+ public void init() throws Exception {
+ MockitoAnnotations.initMocks(this);
+ Mockito.when(this.transaction.submit()).thenReturn(Futures.immediateCheckedFuture(null));
+ Mockito.when(context.getInstanceIdentifier()).thenReturn(YangInstanceIdentifier.EMPTY);
+ }
+
+ /**
+ * Test of successful DELETE operation.
+ */
+ @Test
+ public void deleteData() throws Exception {
+ // assert that data to delete exists
+ Mockito.when(this.transaction.exists(LogicalDatastoreType.CONFIGURATION, YangInstanceIdentifier.EMPTY))
+ .thenReturn(Futures.immediateCheckedFuture(Boolean.TRUE));
+
+ // test
+ final Response response = DeleteDataTransactionUtil.deleteData(
+ new TransactionVarsWrapper(this.context, null, this.transaction));
+
+ // assert success
+ assertEquals("Not expected response received", Status.OK.getStatusCode(), response.getStatus());
+ }
+
+ /**
+ * Negative test for DELETE operation when data to delete does not exist. Error 404 is expected.
+ */
+ @Test
+ public void deleteDataNegativeTest() throws Exception {
+ // assert that data to delete does NOT exist
+ Mockito.when(this.transaction.exists(LogicalDatastoreType.CONFIGURATION, YangInstanceIdentifier.EMPTY))
+ .thenReturn(Futures.immediateCheckedFuture(Boolean.FALSE));
+
+ // test and assert error
+ try {
+ DeleteDataTransactionUtil.deleteData(new TransactionVarsWrapper(this.context, null, this.transaction));
+ fail("Delete operation should fail due to missing data");
+ } catch (final RestconfDocumentedException e) {
+ assertEquals(ErrorType.PROTOCOL, e.getErrors().get(0).getErrorType());
+ assertEquals(ErrorTag.DATA_MISSING, e.getErrors().get(0).getErrorTag());
+ assertEquals(404, e.getErrors().get(0).getErrorTag().getStatusCode());
+ }
+ }
+}
\ No newline at end of file