2 * Copyright © 2019 Orange, 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.transportpce.tapi.topology;
10 import com.google.common.util.concurrent.FluentFuture;
11 import com.google.common.util.concurrent.ListenableFuture;
12 import java.nio.charset.Charset;
13 import java.util.ArrayList;
14 import java.util.Collection;
15 import java.util.Collections;
16 import java.util.HashMap;
17 import java.util.Iterator;
18 import java.util.List;
20 import java.util.Map.Entry;
21 import java.util.Optional;
22 import java.util.UUID;
23 import java.util.concurrent.ExecutionException;
24 import java.util.stream.Collectors;
25 import org.eclipse.jdt.annotation.NonNull;
26 import org.eclipse.jdt.annotation.Nullable;
27 import org.opendaylight.mdsal.binding.api.DataBroker;
28 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
29 import org.opendaylight.transportpce.common.InstanceIdentifiers;
30 import org.opendaylight.transportpce.common.NetworkUtils;
31 import org.opendaylight.yang.gen.v1.http.org.opendaylight.transportpce.portmapping.rev200827.network.Nodes;
32 import org.opendaylight.yang.gen.v1.http.org.opendaylight.transportpce.portmapping.rev200827.network.NodesKey;
33 import org.opendaylight.yang.gen.v1.http.org.opendaylight.transportpce.portmapping.rev200827.network.nodes.Mapping;
34 import org.opendaylight.yang.gen.v1.http.org.opendaylight.transportpce.portmapping.rev200827.network.nodes.MappingKey;
35 import org.opendaylight.yang.gen.v1.http.org.openroadm.common.network.rev181130.Link1;
36 import org.opendaylight.yang.gen.v1.http.org.openroadm.common.network.rev181130.TerminationPoint1;
37 import org.opendaylight.yang.gen.v1.http.org.openroadm.network.types.rev181130.OpenroadmLinkType;
38 import org.opendaylight.yang.gen.v1.http.org.openroadm.network.types.rev181130.OpenroadmNodeType;
39 import org.opendaylight.yang.gen.v1.http.org.openroadm.network.types.rev181130.OpenroadmTpType;
40 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.network.rev180226.NodeId;
41 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.network.rev180226.networks.Network;
42 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.network.rev180226.networks.network.Node;
43 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.network.topology.rev180226.Network1;
44 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.network.topology.rev180226.Node1;
45 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.network.topology.rev180226.networks.network.Link;
46 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.network.topology.rev180226.networks.network.node.TerminationPoint;
47 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.common.rev181210.AdministrativeState;
48 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.common.rev181210.LayerProtocolName;
49 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.common.rev181210.LifecycleState;
50 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.common.rev181210.OperationalState;
51 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.common.rev181210.Uuid;
52 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.common.rev181210.global._class.Name;
53 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.common.rev181210.global._class.NameBuilder;
54 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.common.rev181210.global._class.NameKey;
55 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.ForwardingRule;
56 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetLinkDetailsInput;
57 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetLinkDetailsOutput;
58 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetNodeDetailsInput;
59 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetNodeDetailsOutput;
60 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetNodeEdgePointDetailsInput;
61 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetNodeEdgePointDetailsOutput;
62 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetTopologyDetailsInput;
63 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetTopologyDetailsOutput;
64 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetTopologyDetailsOutputBuilder;
65 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetTopologyListInput;
66 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.GetTopologyListOutput;
67 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.RuleType;
68 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.TapiTopologyService;
69 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.get.topology.details.output.Topology;
70 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.get.topology.details.output.TopologyBuilder;
71 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.NodeRuleGroup;
72 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.NodeRuleGroupBuilder;
73 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.NodeRuleGroupKey;
74 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.OwnedNodeEdgePoint;
75 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.OwnedNodeEdgePointKey;
76 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.rule.group.Rule;
77 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.rule.group.RuleBuilder;
78 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.rule.group.RuleKey;
79 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.topology.LinkKey;
80 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.topology.NodeBuilder;
81 import org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.topology.NodeKey;
82 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
83 import org.opendaylight.yangtools.yang.binding.KeyedInstanceIdentifier;
84 import org.opendaylight.yangtools.yang.common.RpcResult;
85 import org.opendaylight.yangtools.yang.common.RpcResultBuilder;
86 import org.slf4j.Logger;
87 import org.slf4j.LoggerFactory;
89 public class TapiTopologyImpl implements TapiTopologyService {
91 private static final Logger LOG = LoggerFactory.getLogger(TapiTopologyImpl.class);
92 private final DataBroker dataBroker;
94 public TapiTopologyImpl(DataBroker dataBroker) {
95 this.dataBroker = dataBroker;
99 public ListenableFuture<RpcResult<GetNodeDetailsOutput>> getNodeDetails(GetNodeDetailsInput input) {
100 // TODO Auto-generated method stub
105 public ListenableFuture<RpcResult<GetTopologyDetailsOutput>> getTopologyDetails(GetTopologyDetailsInput input) {
106 if (!TopologyUtils.T0_MULTILAYER.equals(input.getTopologyIdOrName())
107 && !TopologyUtils.TPDR_100G.equals(input.getTopologyIdOrName())) {
108 LOG.error("Invalid TAPI topology name");
109 return RpcResultBuilder.success(new GetTopologyDetailsOutputBuilder().build()).buildFuture();
112 LOG.info("Building TAPI Topology abstraction for {}", input.getTopologyIdOrName());
113 Topology topology = createAbstractedOtnTopology();
114 if (TopologyUtils.TPDR_100G.equals(input.getTopologyIdOrName())) {
115 topology = createAbstracted100GTpdrTopology(topology);
117 return RpcResultBuilder.success(new GetTopologyDetailsOutputBuilder().setTopology(topology).build())
119 } catch (TapiTopologyException e) {
120 LOG.error("error building TAPI topology");
121 return RpcResultBuilder.success(new GetTopologyDetailsOutputBuilder().build()).buildFuture();
125 private Topology createAbstracted100GTpdrTopology(Topology topology) {
126 List<org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.topology.Node> dsrNodes
127 = topology.nonnullNode().values().stream()
128 .filter(node -> node.getLayerProtocolName().contains(LayerProtocolName.DSR))
129 .collect(Collectors.toList());
130 List<OwnedNodeEdgePoint> nep100GTpdrList = new ArrayList<>();
131 for (org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.topology.Node node2 : dsrNodes) {
132 List<OwnedNodeEdgePoint> nepList = node2.getOwnedNodeEdgePoint().values().stream()
133 .filter(nep -> nep.getName().containsKey(new NameKey("100G-tpdr"))).collect(Collectors.toList());
134 nep100GTpdrList.addAll(nepList);
136 Name topoName = new NameBuilder().setValue(TopologyUtils.TPDR_100G).setValueName("TAPI Topology Name").build();
137 Uuid topoUuid = new Uuid(UUID.nameUUIDFromBytes(TopologyUtils.TPDR_100G.getBytes(Charset.forName("UTF-8")))
139 org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.topology.Node node
140 = createTapiNode(nep100GTpdrList, topoUuid);
141 return new TopologyBuilder()
142 .setName(Map.of(topoName.key(), topoName))
144 .setNode(Map.of(node.key(), node))
148 private Network readTopology(InstanceIdentifier<Network> networkIID)
149 throws TapiTopologyException {
150 Network topology = null;
151 FluentFuture<Optional<Network>> topologyFuture = dataBroker.newReadOnlyTransaction()
152 .read(LogicalDatastoreType.CONFIGURATION, networkIID);
154 topology = topologyFuture.get().get();
155 } catch (InterruptedException e) {
156 Thread.currentThread().interrupt();
157 throw new TapiTopologyException("Unable to get from mdsal topology: " + networkIID
158 .firstKeyOf(Network.class).getNetworkId().getValue(), e);
159 } catch (ExecutionException e) {
160 throw new TapiTopologyException("Unable to get from mdsal topology: " + networkIID
161 .firstKeyOf(Network.class).getNetworkId().getValue(), e);
166 private Topology createAbstractedOtnTopology() throws TapiTopologyException {
167 // read openroadm-topology
168 Network openroadmTopo = readTopology(InstanceIdentifiers.OVERLAY_NETWORK_II);
169 List<Link> linkList = new ArrayList<>();
170 if (openroadmTopo.augmentation(Network1.class) != null) {
171 linkList.addAll(openroadmTopo.augmentation(Network1.class).getLink().values());
173 List<Link> xponderOutLinkList = linkList.stream()
174 .filter(lk -> lk.augmentation(Link1.class).getLinkType().equals(OpenroadmLinkType.XPONDEROUTPUT))
175 .collect(Collectors.toList());
176 List<Link> xponderInLinkList = linkList.stream()
177 .filter(lk -> lk.augmentation(Link1.class).getLinkType().equals(OpenroadmLinkType.XPONDERINPUT))
178 .collect(Collectors.toList());
180 Network otnTopo = readTopology(InstanceIdentifiers.OTN_NETWORK_II);
181 Map<NodeId, Node> otnNodeMap = otnTopo.nonnullNode().values().stream()
182 .collect(Collectors.toMap(Node::getNodeId, node -> node));
184 Map<String, List<String>> networkPortMap = new HashMap<>();
185 Iterator<Entry<NodeId, Node>> itOtnNodeMap = otnNodeMap.entrySet().iterator();
186 while (itOtnNodeMap.hasNext()) {
187 Entry<NodeId, Node> entry = itOtnNodeMap.next();
188 String portMappingNodeId = entry.getValue().getSupportingNode().values().stream()
189 .filter(sn -> sn.getNetworkRef().getValue().equals(NetworkUtils.UNDERLAY_NETWORK_ID))
191 .get().getNodeRef().getValue();
192 List<String> networkPortList = new ArrayList<>();
193 for (TerminationPoint tp : entry.getValue().augmentation(Node1.class).getTerminationPoint().values()) {
194 if (tp.augmentation(TerminationPoint1.class).getTpType().equals(OpenroadmTpType.XPONDERNETWORK)
196 checkTp(entry.getKey().getValue(), portMappingNodeId, tp, xponderOutLinkList, xponderInLinkList)) {
197 networkPortList.add(tp.getTpId().getValue());
200 if (!networkPortList.isEmpty()) {
201 networkPortMap.put(entry.getKey().getValue(), networkPortList);
204 Map<NodeKey, org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.topology.Node>
205 tapiNodeList = new HashMap<>();
206 Map<LinkKey, org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.topology.Link>
207 tapiLinkList = new HashMap<>();
208 Uuid topoUuid = new Uuid(UUID.nameUUIDFromBytes(TopologyUtils.T0_MULTILAYER.getBytes(Charset.forName("UTF-8")))
210 ConvertORTopoToTapiTopo tapiFactory = new ConvertORTopoToTapiTopo(topoUuid);
211 Iterator<Entry<String, List<String>>> it = networkPortMap.entrySet().iterator();
212 while (it.hasNext()) {
213 String nodeId = it.next().getKey();
214 tapiFactory.convertNode(otnNodeMap.get(new NodeId(nodeId)), networkPortMap.get(nodeId));
215 tapiNodeList.putAll(tapiFactory.getTapiNodes());
216 tapiLinkList.putAll(tapiFactory.getTapiLinks());
218 if (openroadmTopo.nonnullNode().values().stream().filter(nt ->
219 nt.augmentation(org.opendaylight.yang.gen.v1.http.org.openroadm.common.network.rev181130.Node1.class)
220 .getNodeType().equals(OpenroadmNodeType.SRG)).count() > 0) {
221 tapiFactory.convertRoadmInfrastructure();
222 tapiNodeList.putAll(tapiFactory.getTapiNodes());
223 tapiLinkList.putAll(tapiFactory.getTapiLinks());
225 LOG.warn("Unable to abstract an ROADM infrasctructure from openroadm-topology");
227 if (otnTopo.augmentation(Network1.class) != null) {
228 List<Link> otnLinkList = new ArrayList<>(otnTopo.augmentation(Network1.class).getLink().values());
229 Collections.sort(otnLinkList, (l1, l2) -> l1.getLinkId().getValue()
230 .compareTo(l2.getLinkId().getValue()));
231 tapiFactory.convertLinks(otnLinkList);
232 tapiLinkList.putAll(tapiFactory.getTapiLinks());
234 Name name = new NameBuilder().setValue(TopologyUtils.T0_MULTILAYER).setValueName("TAPI Topology Name").build();
235 return new TopologyBuilder()
236 .setName(Map.of(name.key(), name))
238 .setNode(tapiNodeList)
239 .setLink(tapiLinkList).build();
243 public ListenableFuture<RpcResult<GetNodeEdgePointDetailsOutput>> getNodeEdgePointDetails(
244 GetNodeEdgePointDetailsInput input) {
245 // TODO Auto-generated method stub
250 public ListenableFuture<RpcResult<GetLinkDetailsOutput>> getLinkDetails(GetLinkDetailsInput input) {
251 // TODO Auto-generated method stub
256 public ListenableFuture<RpcResult<GetTopologyListOutput>> getTopologyList(GetTopologyListInput input) {
257 // TODO Auto-generated method stub
261 private org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.topology.Node
262 createTapiNode(List<OwnedNodeEdgePoint> nepList, Uuid topoUuid) {
263 Name name = new NameBuilder().setValueName("Tpdr100g node name").setValue("Tpdr100g over WDM node").build();
264 Map<OwnedNodeEdgePointKey, OwnedNodeEdgePoint> onepMap = new HashMap<>();
265 for (OwnedNodeEdgePoint ownedNodeEdgePoint : nepList) {
266 onepMap.put(ownedNodeEdgePoint.key(), ownedNodeEdgePoint);
268 Uuid nodeUuid = new Uuid(UUID.nameUUIDFromBytes(name.getValue().getBytes(Charset.forName("UTF-8"))).toString());
269 return new NodeBuilder()
271 .setName(Map.of(name.key(), name))
272 .setLayerProtocolName(List.of(LayerProtocolName.ETH))
273 .setAdministrativeState(AdministrativeState.UNLOCKED)
274 .setOperationalState(OperationalState.ENABLED)
275 .setLifecycleState(LifecycleState.INSTALLED)
276 .setOwnedNodeEdgePoint(onepMap)
277 .setNodeRuleGroup(createNodeRuleGroupFor100gTpdrNode(topoUuid, nodeUuid, nepList))
281 private boolean checkTp(String nodeIdTopo, String nodeIdPortMap, TerminationPoint tp, List<Link> xpdOut, List<
284 if (tp.augmentation(TerminationPoint1.class).getTpType().equals(OpenroadmTpType.XPONDERCLIENT)) {
285 networkLcp = tp.augmentation(
286 org.opendaylight.yang.gen.v1.http.transportpce.topology.rev200129.TerminationPoint1.class)
287 .getAssociatedConnectionMapPort();
289 networkLcp = tp.getTpId().getValue();
292 KeyedInstanceIdentifier<Mapping, MappingKey> pmIID = InstanceIdentifier.create(
293 org.opendaylight.yang.gen.v1.http.org.opendaylight.transportpce.portmapping.rev200827.Network.class)
294 .child(Nodes.class, new NodesKey(nodeIdPortMap)).child(Mapping.class, new MappingKey(networkLcp));
296 FluentFuture<Optional<Mapping>> mappingOpt = dataBroker.newReadOnlyTransaction().read(
297 LogicalDatastoreType.CONFIGURATION, pmIID);
298 Mapping mapping = null;
299 if (mappingOpt.isDone()) {
301 mapping = mappingOpt.get().get();
302 } catch (InterruptedException | ExecutionException e) {
303 LOG.error("Error getting mapping for {}", networkLcp,e);
307 LOG.error("Impossible to get mapping of associated network port {} of tp {}", networkLcp, tp.getTpId()
311 String networkPortDirection = mapping.getPortDirection();
313 switch (networkPortDirection) {
314 case "bidirectional":
315 count += xpdOut.stream().filter(lk -> lk.getSource().getSourceNode().getValue().equals(nodeIdTopo) && lk
316 .getSource().getSourceTp().equals(networkLcp)).count();
317 count += xpdIn.stream().filter(lk -> lk.getDestination().getDestNode().getValue().equals(nodeIdTopo)
318 && lk.getDestination().getDestTp().equals(networkLcp)).count();
323 String partnerLcp = mapping.getPartnerLcp();
324 if (mapping.getPortQual().equals("tx")) {
325 count += xpdOut.stream().filter(lk -> lk.getSource().getSourceNode().getValue().equals(nodeIdTopo)
326 && lk.getSource().getSourceTp().equals(networkLcp)).count();
327 count += xpdIn.stream().filter(lk -> lk.getDestination().getDestNode().getValue().equals(nodeIdTopo)
328 && lk.getDestination().getDestTp().equals(partnerLcp)).count();
330 if (mapping.getPortQual().equals("rx")) {
331 count += xpdIn.stream().filter(lk -> lk.getDestination().getDestNode().getValue().equals(nodeIdTopo)
332 && lk.getDestination().getDestTp().equals(networkLcp)).count();
333 count += xpdOut.stream().filter(lk -> lk.getSource().getSourceNode().getValue().equals(nodeIdTopo)
334 && lk.getSource().getSourceTp().equals(partnerLcp)).count();
338 LOG.error("Invalid port direction for {}", networkLcp);
343 private Map<NodeRuleGroupKey, NodeRuleGroup> createNodeRuleGroupFor100gTpdrNode(Uuid topoUuid, Uuid nodeUuid,
344 Collection<OwnedNodeEdgePoint> onepl) {
345 Map<org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.rule.group.NodeEdgePointKey,
346 org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.rule.group.NodeEdgePoint>
347 nepMap = new HashMap<>();
348 for (OwnedNodeEdgePoint onep : onepl) {
349 org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.rule.group.NodeEdgePoint
350 nep = new org.opendaylight.yang.gen.v1.urn.onf.otcc.yang.tapi.topology.rev181210.node.rule.group
351 .NodeEdgePointBuilder()
352 .setTopologyUuid(topoUuid)
353 .setNodeUuid(nodeUuid)
354 .setNodeEdgePointUuid(onep.key().getUuid())
356 nepMap.put(nep.key(), nep);
358 Map<NodeRuleGroupKey, NodeRuleGroup> nodeRuleGroupMap = new HashMap<>();
359 Map<RuleKey, Rule> ruleList = new HashMap<>();
360 Rule rule = new RuleBuilder()
361 .setLocalId("forward")
362 .setForwardingRule(ForwardingRule.MAYFORWARDACROSSGROUP)
363 .setRuleType(RuleType.FORWARDING)
365 ruleList.put(rule.key(), rule);
366 NodeRuleGroup nodeRuleGroup = new NodeRuleGroupBuilder()
367 .setUuid(new Uuid(UUID.nameUUIDFromBytes(("rdm infra node rule group").getBytes(Charset.forName("UTF-8")))
370 .setNodeEdgePoint(nepMap)
372 nodeRuleGroupMap.put(nodeRuleGroup.key(), nodeRuleGroup);
373 return nodeRuleGroupMap;