97ecc2cf4313477939f64e2f80c3573869983b74
[netconf.git] / apps / netconf-topology / src / main / java / org / opendaylight / netconf / topology / spi / NetconfTopologyRPCProvider.java
1 /*
2  * Copyright (c) 2017 Brocade Communication Systems 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.topology.spi;
9
10 import static java.util.Objects.requireNonNull;
11
12 import com.google.common.annotations.VisibleForTesting;
13 import com.google.common.collect.ImmutableClassToInstanceMap;
14 import com.google.common.util.concurrent.FutureCallback;
15 import com.google.common.util.concurrent.ListenableFuture;
16 import com.google.common.util.concurrent.MoreExecutors;
17 import com.google.common.util.concurrent.SettableFuture;
18 import java.util.Base64;
19 import org.eclipse.jdt.annotation.NonNull;
20 import org.opendaylight.aaa.encrypt.AAAEncryptionService;
21 import org.opendaylight.mdsal.binding.api.DataBroker;
22 import org.opendaylight.mdsal.binding.api.RpcProviderService;
23 import org.opendaylight.mdsal.binding.api.WriteTransaction;
24 import org.opendaylight.mdsal.common.api.CommitInfo;
25 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
26 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.device.rev240104.credentials.Credentials;
27 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.device.rev240104.credentials.credentials.KeyAuthBuilder;
28 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.device.rev240104.credentials.credentials.LoginPwBuilder;
29 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.device.rev240104.credentials.credentials.LoginPwUnencryptedBuilder;
30 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.device.rev240104.credentials.credentials.key.auth.KeyBasedBuilder;
31 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.device.rev240104.credentials.credentials.login.pw.LoginPasswordBuilder;
32 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.device.rev240104.credentials.credentials.login.pw.unencrypted.LoginPasswordUnencryptedBuilder;
33 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.optional.rev221225.NetconfNodeAugmentedOptionalBuilder;
34 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.CreateDevice;
35 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.CreateDeviceInput;
36 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.CreateDeviceOutput;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.CreateDeviceOutputBuilder;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.DeleteDevice;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.DeleteDeviceInput;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.DeleteDeviceOutput;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.DeleteDeviceOutputBuilder;
42 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.NetconfNode;
43 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.NetconfNodeBuilder;
44 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.rpc.credentials.RpcCredentials;
45 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.rpc.credentials.rpc.credentials.KeyAuth;
46 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.rpc.credentials.rpc.credentials.LoginPw;
47 import org.opendaylight.yang.gen.v1.urn.opendaylight.netconf.node.topology.rev231121.rpc.credentials.rpc.credentials.LoginPwUnencrypted;
48 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.NetworkTopology;
49 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.NodeId;
50 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.TopologyId;
51 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.Topology;
52 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.TopologyKey;
53 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.Node;
54 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.NodeBuilder;
55 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.NodeKey;
56 import org.opendaylight.yangtools.concepts.Registration;
57 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
58 import org.opendaylight.yangtools.yang.binding.Rpc;
59 import org.opendaylight.yangtools.yang.common.RpcResult;
60 import org.opendaylight.yangtools.yang.common.RpcResultBuilder;
61 import org.slf4j.Logger;
62 import org.slf4j.LoggerFactory;
63
64 @Deprecated
65 public final class NetconfTopologyRPCProvider implements AutoCloseable {
66     private static final Logger LOG = LoggerFactory.getLogger(NetconfTopologyRPCProvider.class);
67
68     private final @NonNull InstanceIdentifier<Topology> topologyPath;
69     private final @NonNull AAAEncryptionService encryptionService;
70     private final @NonNull DataBroker dataBroker;
71
72     private final Registration reg;
73
74     public NetconfTopologyRPCProvider(final RpcProviderService rpcProviderService, final DataBroker dataBroker,
75             final AAAEncryptionService encryptionService, final String topologyId) {
76         this.dataBroker = requireNonNull(dataBroker);
77         this.encryptionService = requireNonNull(encryptionService);
78         topologyPath = InstanceIdentifier.builder(NetworkTopology.class)
79             .child(Topology.class, new TopologyKey(new TopologyId(topologyId)))
80             .build();
81         reg = rpcProviderService.registerRpcImplementations(ImmutableClassToInstanceMap.<Rpc<?, ?>>builder()
82             .put(CreateDevice.class, this::createDevice)
83             .put(DeleteDevice.class, this::deleteDevice)
84             .build());
85     }
86
87     protected @NonNull InstanceIdentifier<Topology> topologyPath() {
88         return topologyPath;
89     }
90
91     @Override
92     public void close() {
93         reg.close();
94     }
95
96     private ListenableFuture<RpcResult<CreateDeviceOutput>> createDevice(final CreateDeviceInput input) {
97         final var netconfNode = encryptPassword(input);
98         final var nodeId = new NodeId(input.getNodeId());
99         final var nodeBuilder = new NodeBuilder()
100             .setNodeId(nodeId)
101             .addAugmentation(netconfNode);
102         if (input.getIgnoreMissingSchemaSources() != null) {
103             final var netconfNodeOptionalBuilder = new NetconfNodeAugmentedOptionalBuilder(input);
104             nodeBuilder.addAugmentation(netconfNodeOptionalBuilder.build());
105         }
106         final SettableFuture<RpcResult<CreateDeviceOutput>> futureResult = SettableFuture.create();
107         writeToConfigDS(nodeBuilder.build(), nodeId, futureResult);
108         return futureResult;
109     }
110
111     private ListenableFuture<RpcResult<DeleteDeviceOutput>> deleteDevice(final DeleteDeviceInput input) {
112         final NodeId nodeId = new NodeId(input.getNodeId());
113
114         final InstanceIdentifier<Node> niid = topologyPath.child(Node.class, new NodeKey(nodeId));
115
116         final WriteTransaction wtx = dataBroker.newWriteOnlyTransaction();
117         wtx.delete(LogicalDatastoreType.CONFIGURATION, niid);
118
119         final SettableFuture<RpcResult<DeleteDeviceOutput>> rpcFuture = SettableFuture.create();
120
121         wtx.commit().addCallback(new FutureCallback<CommitInfo>() {
122
123             @Override
124             public void onSuccess(final CommitInfo result) {
125                 LOG.info("delete-device RPC: Removed netconf node successfully.");
126                 rpcFuture.set(RpcResultBuilder.success(new DeleteDeviceOutputBuilder().build()).build());
127             }
128
129             @Override
130             public void onFailure(final Throwable exception) {
131                 LOG.error("delete-device RPC: Unable to remove netconf node.", exception);
132                 rpcFuture.setException(exception);
133             }
134         }, MoreExecutors.directExecutor());
135
136         return rpcFuture;
137     }
138
139     @VisibleForTesting
140     NetconfNode encryptPassword(final CreateDeviceInput input) {
141         final NetconfNodeBuilder builder = new NetconfNodeBuilder();
142         builder.fieldsFrom(input);
143         return builder.setCredentials(translate(input.getRpcCredentials()))
144             .build();
145     }
146
147     private Credentials translate(final RpcCredentials credentialsRpc) {
148         if (credentialsRpc instanceof LoginPw loginPw) {
149             final var loginPassword = loginPw.getLoginPassword();
150             return new LoginPwBuilder()
151                 .setLoginPassword(new LoginPasswordBuilder()
152                     .setUsername(loginPassword.getUsername())
153                     .setPassword(Base64.getDecoder().decode(encryptionService.encrypt(loginPassword.getPassword())))
154                     .build())
155                 .build();
156         } else if (credentialsRpc instanceof LoginPwUnencrypted loginPwUnencrypted) {
157             final var loginPassword = loginPwUnencrypted.getLoginPasswordUnencrypted();
158             return new LoginPwUnencryptedBuilder()
159                 .setLoginPasswordUnencrypted(new LoginPasswordUnencryptedBuilder()
160                     .setUsername(loginPassword.getUsername())
161                     .setPassword(loginPassword.getPassword())
162                     .build())
163                 .build();
164         } else if (credentialsRpc instanceof KeyAuth keyAuth) {
165             final var loginPassword = keyAuth.getKeyBased();
166             return new KeyAuthBuilder()
167                 .setKeyBased(new KeyBasedBuilder()
168                     .setUsername(loginPassword.getUsername())
169                     .setKeyId(loginPassword.getKeyId())
170                     .build())
171                 .build();
172         } else {
173             throw new IllegalArgumentException("Unsupported credential type: " + credentialsRpc.getClass());
174         }
175     }
176
177     private void writeToConfigDS(final Node node, final NodeId nodeId,
178             final SettableFuture<RpcResult<CreateDeviceOutput>> futureResult) {
179
180         final WriteTransaction writeTransaction = dataBroker.newWriteOnlyTransaction();
181         final InstanceIdentifier<Node> niid = topologyPath.child(Node.class, new NodeKey(nodeId));
182         writeTransaction.mergeParentStructureMerge(LogicalDatastoreType.CONFIGURATION, niid, node);
183         writeTransaction.commit().addCallback(new FutureCallback<CommitInfo>() {
184
185             @Override
186             public void onSuccess(final CommitInfo result) {
187                 LOG.info("add-netconf-node RPC: Added netconf node successfully.");
188                 futureResult.set(RpcResultBuilder.success(new CreateDeviceOutputBuilder().build()).build());
189             }
190
191             @Override
192             public void onFailure(final Throwable exception) {
193                 LOG.error("add-netconf-node RPC: Unable to add netconf node.", exception);
194                 futureResult.setException(exception);
195             }
196         }, MoreExecutors.directExecutor());
197     }
198 }