0bca5198301684e3105585434e82965fe87d0f0f
[bgpcep.git] / pcep / tunnel / tunnel-provider / src / test / java / org / opendaylight / bgpcep / pcep / tunnel / provider / NodeChangedListenerTest.java
1 /*
2  * Copyright (c) 2014 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.bgpcep.pcep.tunnel.provider;
9
10 import static org.junit.Assert.assertEquals;
11 import static org.junit.Assert.assertNotNull;
12 import static org.junit.Assert.assertNull;
13 import static org.opendaylight.protocol.util.CheckTestUtil.readDataOperational;
14
15 import com.google.common.collect.Iterables;
16 import java.util.Iterator;
17 import java.util.concurrent.ExecutionException;
18 import org.junit.After;
19 import org.junit.Before;
20 import org.junit.Test;
21 import org.opendaylight.mdsal.binding.api.DataTreeIdentifier;
22 import org.opendaylight.mdsal.binding.api.WriteTransaction;
23 import org.opendaylight.mdsal.binding.dom.adapter.test.AbstractConcurrentDataBrokerTest;
24 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
25 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.IpAddress;
26 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.IpAddressNoZone;
27 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.Ipv4Address;
28 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.Ipv4AddressNoZone;
29 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.network.concepts.rev131125.Bandwidth;
30 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.ietf.stateful.rev200720.Path1Builder;
31 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.ietf.stateful.rev200720.lsp.identifiers.tlv.LspIdentifiersBuilder;
32 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.ietf.stateful.rev200720.lsp.identifiers.tlv.lsp.identifiers.address.family.Ipv4CaseBuilder;
33 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.ietf.stateful.rev200720.lsp.identifiers.tlv.lsp.identifiers.address.family.ipv4._case.Ipv4Builder;
34 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.ietf.stateful.rev200720.lsp.object.LspBuilder;
35 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.ietf.stateful.rev200720.lsp.object.lsp.TlvsBuilder;
36 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.pcep.types.rev181109.bandwidth.object.BandwidthBuilder;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.rsvp.rev150820.Ipv4ExtendedTunnelId;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.rsvp.rev150820.LspId;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.topology.pcep.rev220730.Node1Builder;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.topology.pcep.rev220730.PccSyncState;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.topology.pcep.rev220730.pcep.client.attributes.PathComputationClientBuilder;
42 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.topology.pcep.rev220730.pcep.client.attributes.path.computation.client.ReportedLsp;
43 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.topology.pcep.rev220730.pcep.client.attributes.path.computation.client.ReportedLspBuilder;
44 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.topology.pcep.rev220730.pcep.client.attributes.path.computation.client.ReportedLspKey;
45 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.topology.pcep.rev220730.pcep.client.attributes.path.computation.client.reported.lsp.PathBuilder;
46 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.topology.pcep.rev220730.pcep.client.attributes.path.computation.client.reported.lsp.PathKey;
47 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.NetworkTopology;
48 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.NodeId;
49 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.TopologyId;
50 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.TpId;
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.TopologyBuilder;
53 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.TopologyKey;
54 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.Link;
55 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.Node;
56 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.NodeBuilder;
57 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.NodeKey;
58 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.node.TerminationPoint;
59 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.node.attributes.SupportingNode;
60 import org.opendaylight.yangtools.concepts.ListenerRegistration;
61 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
62 import org.opendaylight.yangtools.yang.binding.util.BindingMap;
63 import org.opendaylight.yangtools.yang.common.Uint32;
64
65 public class NodeChangedListenerTest extends AbstractConcurrentDataBrokerTest {
66
67     private static final TopologyId PCEP_TOPOLOGY_ID = new TopologyId("pcep-topology");
68     private static final TopologyId TUNNEL_TOPOLOGY_ID = new TopologyId("tunnel-topology");
69
70     private static final String NODE1_IPV4 = "39.39.39.39";
71     private static final NodeId NODE1_ID = new NodeId("pcc://" + NODE1_IPV4);
72     private static final String LSP1_NAME = "lsp1";
73     private static final Uint32 LSP1_ID = Uint32.ONE;
74
75     private static final String NODE2_IPV4 = "40.40.40.40";
76     private static final NodeId NODE2_ID = new NodeId("pcc://" + NODE2_IPV4);
77     private static final String LSP2_NAME = "lsp2";
78     private static final Uint32 LSP2_ID = Uint32.TWO;
79
80     private static final InstanceIdentifier<Topology> PCEP_TOPO_IID = InstanceIdentifier.builder(NetworkTopology.class)
81             .child(Topology.class, new TopologyKey(PCEP_TOPOLOGY_ID)).build();
82     private static final InstanceIdentifier<Topology> TUNNEL_TOPO_IID = InstanceIdentifier
83             .builder(NetworkTopology.class).child(Topology.class, new TopologyKey(TUNNEL_TOPOLOGY_ID)).build();
84
85     private ListenerRegistration<NodeChangedListener> listenerRegistration;
86
87     @Before
88     public void setUp() throws InterruptedException, ExecutionException {
89         final WriteTransaction wTx = getDataBroker().newWriteOnlyTransaction();
90         wTx.mergeParentStructurePut(LogicalDatastoreType.OPERATIONAL, PCEP_TOPO_IID, new TopologyBuilder()
91                 .withKey(new TopologyKey(PCEP_TOPOLOGY_ID))
92                 .setTopologyId(PCEP_TOPOLOGY_ID).build());
93         wTx.mergeParentStructurePut(LogicalDatastoreType.OPERATIONAL, TUNNEL_TOPO_IID, new TopologyBuilder()
94                 .withKey(new TopologyKey(TUNNEL_TOPOLOGY_ID)).setTopologyId(TUNNEL_TOPOLOGY_ID).build());
95         wTx.commit().get();
96         final NodeChangedListener nodeListener = new NodeChangedListener(getDataBroker(),
97                 PCEP_TOPOLOGY_ID, TUNNEL_TOPO_IID);
98         this.listenerRegistration = getDataBroker().registerDataTreeChangeListener(DataTreeIdentifier.create(
99                 LogicalDatastoreType.OPERATIONAL, PCEP_TOPO_IID.child(Node.class)), nodeListener);
100     }
101
102     @Test
103     public void testNodeChangedListener() throws InterruptedException, ExecutionException {
104         // add node -> create two nodes with TPs and link
105         createNode(NODE1_ID, NODE1_IPV4, LSP1_NAME, LSP1_ID, NODE2_IPV4);
106         final Topology tunnelTopo = readDataOperational(getDataBroker(), TUNNEL_TOPO_IID, tunnelTopo1 -> {
107             assertNotNull(tunnelTopo1.getNode());
108             assertEquals(2, tunnelTopo1.getNode().size());
109             return tunnelTopo1;
110         });
111
112         final NodeId srcId = new NodeId("ip://" + new IpAddress(new Ipv4Address(NODE1_IPV4)));
113         final NodeId dstId = new NodeId("ip://" + new IpAddress(new Ipv4Address(NODE2_IPV4)));
114
115         final Node dst;
116         final Node src;
117
118         final Iterator<Node> it = tunnelTopo.nonnullNode().values().iterator();
119         final Node tmp = it.next();
120         if (tmp.getNodeId().equals(srcId)) {
121             src = tmp;
122             dst = it.next();
123         } else {
124             src = it.next();
125             dst = tmp;
126         }
127
128         assertEquals(srcId, src.getNodeId());
129         assertEquals(dstId, dst.getNodeId());
130
131         assertEquals(1, dst.getTerminationPoint().size());
132         assertEquals(1, src.getTerminationPoint().size());
133         final TerminationPoint dstTp = dst.nonnullTerminationPoint().values().iterator().next();
134         final TerminationPoint srcTp = src.nonnullTerminationPoint().values().iterator().next();
135         final TpId dstNodeTpId = new TpId(dstId.getValue());
136         final TpId srcNodeTpId = new TpId(srcId.getValue());
137         assertEquals(dstNodeTpId, dstTp.getTpId());
138         assertEquals(srcNodeTpId, srcTp.getTpId());
139
140         assertEquals(1, src.getSupportingNode().size());
141         assertNull(dst.getSupportingNode());
142         final SupportingNode sNode = src.nonnullSupportingNode().values().iterator().next();
143         assertEquals(NODE1_ID, sNode.key().getNodeRef());
144
145         assertEquals(1, tunnelTopo.nonnullLink().size());
146         final Link link = tunnelTopo.nonnullLink().values().iterator().next();
147         assertEquals(srcId, link.getSource().getSourceNode());
148         assertEquals(srcNodeTpId, link.getSource().getSourceTp());
149         assertEquals(dstId, link.getDestination().getDestNode());
150         assertEquals(dstNodeTpId, link.getDestination().getDestTp());
151
152         // update second node -> adds supporting node and second link
153         createNode(NODE2_ID, NODE2_IPV4, LSP2_NAME, LSP2_ID, NODE1_IPV4);
154         readDataOperational(getDataBroker(), TUNNEL_TOPO_IID, updatedNodeTopo -> {
155             assertNotNull(updatedNodeTopo.getNode());
156             assertEquals(2, updatedNodeTopo.getNode().size());
157             final Node updatedNode;
158             if (updatedNodeTopo.nonnullNode().values().iterator().next().getNodeId().equals(srcId)) {
159                 updatedNode = Iterables.get(updatedNodeTopo.nonnullNode().values(), 1);
160             } else {
161                 updatedNode = updatedNodeTopo.nonnullNode().values().iterator().next();
162             }
163
164             assertNotNull(updatedNode.getSupportingNode());
165             assertEquals(1, updatedNode.nonnullSupportingNode().size());
166             final SupportingNode sNode2 = updatedNode.nonnullSupportingNode().values().iterator().next();
167             assertEquals(NODE2_ID, sNode2.getNodeRef());
168             assertEquals(2, updatedNodeTopo.getLink().size());
169             return updatedNodeTopo;
170
171         });
172
173         readDataOperational(getDataBroker(), TUNNEL_TOPO_IID, updatedNodeTopo -> {
174             Link link2;
175             Iterator<Link> it2 = updatedNodeTopo.nonnullLink().values().iterator();
176             link2 = it2.next();
177             if (srcId.equals(link2.getSource().getSourceNode())) {
178                 link2 = it2.next();
179             }
180
181             assertEquals(dstId, link2.getSource().getSourceNode());
182             assertEquals(dstNodeTpId, link2.getSource().getSourceTp());
183             assertEquals(srcId, link2.getDestination().getDestNode());
184             assertEquals(srcNodeTpId, link2.getDestination().getDestTp());
185             return updatedNodeTopo;
186         });
187
188         // remove nodes -> remove link
189         removeNode(NODE1_ID);
190         removeNode(NODE2_ID);
191         readDataOperational(getDataBroker(), TUNNEL_TOPO_IID, removedNodeTopo -> {
192             assertNull(removedNodeTopo.getNode());
193             assertNull(removedNodeTopo.getLink());
194             return removedNodeTopo;
195         });
196     }
197
198     @After
199     public void tearDown() {
200         this.listenerRegistration.close();
201     }
202
203     private void createNode(final NodeId nodeId, final String ipv4Address, final String lspName, final Uint32 lspId,
204             final String dstIpv4Address) throws InterruptedException, ExecutionException {
205         final NodeBuilder nodeBuilder = new NodeBuilder();
206         nodeBuilder.withKey(new NodeKey(nodeId));
207         nodeBuilder.setNodeId(nodeId);
208         final PathBuilder pathBuilder = new PathBuilder();
209         pathBuilder.withKey(new PathKey(new LspId(lspId)));
210         pathBuilder.setBandwidth(new BandwidthBuilder().setBandwidth(
211                 new Bandwidth(new byte[]{0x00, 0x00, (byte) 0xff, (byte) 0xff})).build());
212         pathBuilder.addAugmentation(new Path1Builder().setLsp(new LspBuilder().setTlvs(new TlvsBuilder()
213                 .setLspIdentifiers(new LspIdentifiersBuilder().setAddressFamily(new Ipv4CaseBuilder().setIpv4(
214                         new Ipv4Builder().setIpv4TunnelSenderAddress(new Ipv4AddressNoZone(ipv4Address))
215                                 .setIpv4ExtendedTunnelId(new Ipv4ExtendedTunnelId(ipv4Address))
216                                 .setIpv4TunnelEndpointAddress(new Ipv4AddressNoZone(dstIpv4Address))
217                                 .build()).build()).build()).build()).setAdministrative(true)
218                 .setDelegate(true).build()).build());
219         final ReportedLsp reportedLps = new ReportedLspBuilder().withKey(new ReportedLspKey(lspName)).setPath(
220                 BindingMap.of(pathBuilder.build())).build();
221         final Node1Builder node1Builder = new Node1Builder();
222         node1Builder.setPathComputationClient(new PathComputationClientBuilder()
223                 .setStateSync(PccSyncState.Synchronized)
224                 .setReportedLsp(BindingMap.of(reportedLps))
225                 .setIpAddress(new IpAddressNoZone(new Ipv4AddressNoZone(ipv4Address)))
226                 .build());
227         nodeBuilder.addAugmentation(node1Builder.build());
228         final WriteTransaction wTx = getDataBroker().newWriteOnlyTransaction();
229         wTx.put(LogicalDatastoreType.OPERATIONAL, PCEP_TOPO_IID.builder().child(Node.class,
230                 new NodeKey(nodeId)).build(), nodeBuilder.build());
231         wTx.commit().get();
232     }
233
234     private void removeNode(final NodeId nodeId) throws InterruptedException, ExecutionException {
235         final WriteTransaction wTx = getDataBroker().newWriteOnlyTransaction();
236         wTx.delete(LogicalDatastoreType.OPERATIONAL, PCEP_TOPO_IID.builder()
237                 .child(Node.class, new NodeKey(nodeId)).build());
238         wTx.commit().get();
239     }
240 }