Bug 5708: Schemaless netconf mount point
[netconf.git] / netconf / sal-netconf-connector / src / main / java / org / opendaylight / netconf / sal / connect / netconf / schema / mapping / BaseRpcSchemalessTransformer.java
1 /*
2  * Copyright (c) 2016 Cisco Systems, Inc. and others.  All rights reserved.
3  *
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
7  */
8 package org.opendaylight.netconf.sal.connect.netconf.schema.mapping;
9
10 import com.google.common.base.Preconditions;
11 import java.io.IOException;
12 import java.util.Map;
13 import javax.xml.stream.XMLStreamException;
14 import javax.xml.transform.dom.DOMResult;
15 import javax.xml.transform.dom.DOMSource;
16 import org.opendaylight.controller.config.util.xml.XmlElement;
17 import org.opendaylight.controller.config.util.xml.XmlUtil;
18 import org.opendaylight.controller.md.sal.dom.api.DOMNotification;
19 import org.opendaylight.controller.md.sal.dom.api.DOMRpcResult;
20 import org.opendaylight.controller.md.sal.dom.spi.DefaultDOMRpcResult;
21 import org.opendaylight.netconf.api.NetconfMessage;
22 import org.opendaylight.netconf.sal.connect.api.MessageTransformer;
23 import org.opendaylight.netconf.sal.connect.netconf.util.NetconfMessageTransformUtil;
24 import org.opendaylight.netconf.sal.connect.util.MessageCounter;
25 import org.opendaylight.yangtools.yang.common.QName;
26 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
27 import org.opendaylight.yangtools.yang.data.api.schema.AnyXmlNode;
28 import org.opendaylight.yangtools.yang.data.api.schema.ContainerNode;
29 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
30 import org.opendaylight.yangtools.yang.data.impl.schema.Builders;
31 import org.opendaylight.yangtools.yang.model.api.RpcDefinition;
32 import org.opendaylight.yangtools.yang.model.api.SchemaContext;
33 import org.opendaylight.yangtools.yang.model.api.SchemaPath;
34 import org.w3c.dom.Document;
35 import org.w3c.dom.Element;
36
37 /**
38  * Transforms base netconf rpcs
39  */
40 public class BaseRpcSchemalessTransformer implements MessageTransformer<NetconfMessage> {
41
42     private static final Map<QName, RpcDefinition> mappedRpcs = BaseSchema.BASE_NETCONF_CTX.getMappedRpcs();
43     private static final SchemaContext schemaContext = BaseSchema.BASE_NETCONF_CTX.getSchemaContext();
44
45     private final MessageCounter counter;
46
47     public BaseRpcSchemalessTransformer(final MessageCounter counter) {
48         this.counter = counter;
49     }
50
51     @Override
52     public DOMNotification toNotification(final NetconfMessage message) {
53         throw new UnsupportedOperationException("Notifications not supported.");
54     }
55
56     @Override
57     public NetconfMessage toRpcRequest(final SchemaPath rpc, final NormalizedNode<?, ?> payload) {
58         // In case no input for rpc is defined, we can simply construct the payload here
59         final QName rpcQName = rpc.getLastComponent();
60
61         Preconditions.checkNotNull(mappedRpcs.get(rpcQName), "Unknown rpc %s, available rpcs: %s", rpcQName, mappedRpcs.keySet());
62         final DOMResult domResult = NetconfMessageTransformUtil.prepareDomResultForRpcRequest(rpcQName, counter);
63         if(mappedRpcs.get(rpcQName).getInput() == null) {
64             return new NetconfMessage(domResult.getNode().getOwnerDocument());
65         }
66
67         Preconditions.checkNotNull(payload, "Transforming an rpc with input: %s, payload cannot be null", rpcQName);
68         Preconditions.checkArgument(payload instanceof ContainerNode,
69                 "Transforming an rpc with input: %s, payload has to be a container, but was: %s", rpcQName, payload);
70
71         // Set the path to the input of rpc for the payload stream writer
72         final SchemaPath inputPath = rpc.createChild(QName.create(rpcQName, "input").intern());
73         final DOMResult result = domResult;
74
75         try {
76             NetconfMessageTransformUtil.writeNormalizedRpc(((ContainerNode) payload), result, inputPath, schemaContext);
77         } catch (final XMLStreamException | IOException | IllegalStateException e) {
78             throw new IllegalStateException("Unable to serialize " + inputPath, e);
79         }
80
81         final Document node = result.getNode().getOwnerDocument();
82
83         return new NetconfMessage(node);
84     }
85
86     @Override
87     public DOMRpcResult toRpcResult(final NetconfMessage message, final SchemaPath rpc) {
88         final NormalizedNode<?, ?> normalizedNode;
89         final QName rpcQName = rpc.getLastComponent();
90         if (NetconfMessageTransformUtil.isDataRetrievalOperation(rpcQName)) {
91             final Element xmlData = NetconfMessageTransformUtil.getDataSubtree(message.getDocument());
92             final Document data = XmlUtil.newDocument();
93             data.appendChild(data.importNode(xmlData, true));
94             AnyXmlNode xmlDataNode = Builders.anyXmlBuilder()
95                     .withNodeIdentifier(new YangInstanceIdentifier.NodeIdentifier(NetconfMessageTransformUtil.NETCONF_DATA_QNAME))
96                     .withValue(new DOMSource(data))
97                     .build();
98
99             normalizedNode = Builders.containerBuilder()
100                     .withNodeIdentifier(new YangInstanceIdentifier.NodeIdentifier(NetconfMessageTransformUtil.NETCONF_RPC_REPLY_QNAME))
101                     .withChild(xmlDataNode).build();
102         } else {
103             //other base rpcs don't have any output, we can simply construct the payload here
104             Preconditions.checkArgument(isOkPresent(message.getDocument()),
105                     "Unexpected content in response of rpc: %s, %s", rpc.getLastComponent(), message);
106             normalizedNode = null;
107
108         }
109         return new DefaultDOMRpcResult(normalizedNode);
110     }
111
112     // FIXME this should go to some util class
113     static boolean isOkPresent(final Document doc) {
114         return XmlElement.fromDomDocument(doc).getOnlyChildElementWithSameNamespaceOptionally("ok").isPresent();
115     }
116 }