BUG-5685: Register BGP Peer Cluster Singleton Service
[bgpcep.git] / bgp / rib-impl / src / test / java / org / opendaylight / protocol / bgp / rib / impl / ParserToSalTest.java
1 /*
2  * Copyright (c) 2013 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.protocol.bgp.rib.impl;
9
10 import static org.junit.Assert.assertFalse;
11 import static org.junit.Assert.assertTrue;
12 import static org.mockito.Matchers.any;
13 import static org.mockito.Mockito.doReturn;
14
15 import com.google.common.base.Optional;
16 import com.google.common.base.Throwables;
17 import com.google.common.collect.Collections2;
18 import com.google.common.collect.ImmutableList;
19 import com.google.common.collect.Lists;
20 import com.google.common.eventbus.EventBus;
21 import io.netty.util.concurrent.GlobalEventExecutor;
22 import java.io.IOException;
23 import java.net.InetSocketAddress;
24 import java.util.Collection;
25 import java.util.Collections;
26 import java.util.List;
27 import java.util.concurrent.ExecutionException;
28 import org.junit.After;
29 import org.junit.Before;
30 import org.junit.Test;
31 import org.mockito.Mock;
32 import org.mockito.Mockito;
33 import org.mockito.MockitoAnnotations;
34 import org.opendaylight.controller.md.sal.binding.api.DataBroker;
35 import org.opendaylight.controller.md.sal.binding.test.AbstractDataBrokerTest;
36 import org.opendaylight.controller.md.sal.binding.test.DataBrokerTestCustomizer;
37 import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
38 import org.opendaylight.controller.sal.core.api.model.SchemaService;
39 import org.opendaylight.mdsal.singleton.common.api.ClusterSingletonService;
40 import org.opendaylight.mdsal.singleton.common.api.ClusterSingletonServiceProvider;
41 import org.opendaylight.mdsal.singleton.common.api.ClusterSingletonServiceRegistration;
42 import org.opendaylight.mdsal.binding.dom.codec.api.BindingCodecTreeFactory;
43 import org.opendaylight.protocol.bgp.inet.RIBActivator;
44 import org.opendaylight.protocol.bgp.mode.impl.base.BasePathSelectionModeFactory;
45 import org.opendaylight.protocol.bgp.parser.BgpTableTypeImpl;
46 import org.opendaylight.protocol.bgp.parser.spi.pojo.ServiceLoaderBGPExtensionProviderContext;
47 import org.opendaylight.protocol.bgp.rib.impl.spi.BGPDispatcher;
48 import org.opendaylight.protocol.bgp.rib.impl.spi.BGPPeerRegistry;
49 import org.opendaylight.protocol.bgp.rib.mock.BGPMock;
50 import org.opendaylight.protocol.bgp.rib.spi.AbstractRIBExtensionProviderActivator;
51 import org.opendaylight.protocol.bgp.rib.spi.RIBExtensionProviderContext;
52 import org.opendaylight.protocol.bgp.rib.spi.SimpleRIBExtensionProviderContext;
53 import org.opendaylight.protocol.bgp.util.HexDumpBGPFileParser;
54 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.AsNumber;
55 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.inet.rev150305.ipv4.routes.ipv4.routes.Ipv4Route;
56 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.inet.rev150305.ipv6.routes.ipv6.routes.Ipv6Route;
57 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.linkstate.rev150210.LinkstateAddressFamily;
58 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.linkstate.rev150210.LinkstateSubsequentAddressFamily;
59 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.linkstate.rev150210.linkstate.routes.linkstate.routes.LinkstateRoute;
60 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.multiprotocol.rev130919.BgpTableType;
61 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.BgpRib;
62 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.PeerRole;
63 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.RibId;
64 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.bgp.rib.Rib;
65 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.bgp.rib.RibKey;
66 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.bgp.rib.rib.LocRib;
67 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.rib.Tables;
68 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev130925.rib.TablesKey;
69 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev130919.BgpId;
70 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev130919.Ipv4AddressFamily;
71 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev130919.UnicastSubsequentAddressFamily;
72 import org.opendaylight.yangtools.concepts.ListenerRegistration;
73 import org.opendaylight.yangtools.sal.binding.generator.impl.GeneratedClassLoadingStrategy;
74 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
75 import org.opendaylight.yangtools.yang.binding.util.BindingReflections;
76
77 public class ParserToSalTest extends AbstractDataBrokerTest {
78
79     private static final String TEST_RIB_ID = "testRib";
80     private BGPMock mock;
81     private AbstractRIBExtensionProviderActivator baseact, lsact;
82     private RIBExtensionProviderContext ext1, ext2;
83     private static final TablesKey TABLE_KEY = new TablesKey(LinkstateAddressFamily.class, LinkstateSubsequentAddressFamily.class);
84     @Mock
85     BGPDispatcher dispatcher;
86     @Mock
87     private ClusterSingletonServiceProvider clusterSingletonServiceProvider;
88     private BindingCodecTreeFactory codecFactory;
89
90     private SchemaService schemaService;
91
92     @Before
93     public void setUp() throws Exception {
94         super.setup();
95         doReturn(Mockito.mock(ClusterSingletonServiceRegistration.class)).when(this.clusterSingletonServiceProvider).
96             registerClusterSingletonService(any(ClusterSingletonService.class));
97     }
98
99     @Override
100     protected java.lang.Iterable<org.opendaylight.yangtools.yang.binding.YangModuleInfo> getModuleInfos() throws Exception {
101         return ImmutableList.of(BindingReflections.getModuleInfo(Ipv4Route.class), BindingReflections.getModuleInfo(Ipv6Route.class), BindingReflections.getModuleInfo(LinkstateRoute.class));
102     }
103
104     @Override
105     protected DataBrokerTestCustomizer createDataBrokerTestCustomizer() {
106         final DataBrokerTestCustomizer customizer = super.createDataBrokerTestCustomizer();
107         this.codecFactory = customizer.getBindingToNormalized();
108         this.schemaService = customizer.getSchemaService();
109         return customizer;
110     }
111
112     @Override
113     protected void setupWithDataBroker(final DataBroker dataBroker) {
114         MockitoAnnotations.initMocks(this);
115         final List<byte[]> bgpMessages;
116         try {
117             final String hexMessages = "/bgp_hex.txt";
118             bgpMessages = HexDumpBGPFileParser.parseMessages(ParserToSalTest.class.getResourceAsStream(hexMessages));
119         } catch (final IOException e) {
120             throw Throwables.propagate(e);
121         }
122         this.mock = new BGPMock(new EventBus("test"), ServiceLoaderBGPExtensionProviderContext.getSingletonInstance().getMessageRegistry(), Lists.newArrayList(fixMessages(bgpMessages)));
123
124         Mockito.doReturn(GlobalEventExecutor.INSTANCE.newSucceededFuture(null)).when(this.dispatcher).createReconnectingClient(
125                 Mockito.any(InetSocketAddress.class), Mockito.any(BGPPeerRegistry.class), Mockito.anyInt(), Mockito.any(Optional.class));
126
127         this.ext1 = new SimpleRIBExtensionProviderContext();
128         this.ext2 = new SimpleRIBExtensionProviderContext();
129         this.baseact = new RIBActivator();
130         this.lsact = new org.opendaylight.protocol.bgp.linkstate.impl.RIBActivator();
131
132         this.baseact.startRIBExtensionProvider(this.ext1);
133         this.lsact.startRIBExtensionProvider(this.ext2);
134     }
135
136     @After
137     public void tearDown() {
138         this.lsact.close();
139         this.baseact.close();
140     }
141
142     @Test
143     public void testWithLinkstate() throws InterruptedException, ExecutionException {
144         final List<BgpTableType> tables = ImmutableList.of(
145                 new BgpTableTypeImpl(LinkstateAddressFamily.class, LinkstateSubsequentAddressFamily.class));
146         final RIBImpl rib = new RIBImpl(this.clusterSingletonServiceProvider, new RibId(TEST_RIB_ID), new AsNumber(72L), new BgpId("127.0.0.1"),
147             null, this.ext2, this.dispatcher, this.codecFactory, getDomBroker(), tables, Collections.singletonMap(TABLE_KEY,
148             BasePathSelectionModeFactory.createBestPathSelectionStrategy()), GeneratedClassLoadingStrategy.getTCCLClassLoadingStrategy(), null);
149         rib.instantiateServiceInstance();
150         assertTablesExists(tables, true);
151         rib.onGlobalContextUpdated(this.schemaService.getGlobalContext());
152         final BGPPeer peer = new BGPPeer("peer-" + this.mock.toString(), rib, PeerRole.Ibgp, null);
153         peer.instantiateServiceInstance();
154         final ListenerRegistration<?> reg = this.mock.registerUpdateListener(peer);
155         reg.close();
156     }
157
158     @Test
159     public void testWithoutLinkstate() throws InterruptedException, ExecutionException {
160         final List<BgpTableType> tables = ImmutableList.of(new BgpTableTypeImpl(Ipv4AddressFamily.class, UnicastSubsequentAddressFamily.class));
161         final RIBImpl rib = new RIBImpl(this.clusterSingletonServiceProvider, new RibId(TEST_RIB_ID), new AsNumber(72L), new BgpId("127.0.0.1"), null,
162             this.ext1, this.dispatcher, this.codecFactory, getDomBroker(), tables, Collections.singletonMap(TABLE_KEY,
163             BasePathSelectionModeFactory.createBestPathSelectionStrategy()), GeneratedClassLoadingStrategy.getTCCLClassLoadingStrategy(), null);
164         rib.instantiateServiceInstance();
165         rib.onGlobalContextUpdated(this.schemaService.getGlobalContext());
166         assertTablesExists(tables, true);
167         final BGPPeer peer = new BGPPeer("peer-" + this.mock.toString(), rib, PeerRole.Ibgp, null);
168         peer.instantiateServiceInstance();
169         final ListenerRegistration<?> reg = this.mock.registerUpdateListener(peer);
170         reg.close();
171     }
172
173     private static Collection<byte[]> fixMessages(final Collection<byte[]> bgpMessages) {
174         return Collections2.transform(bgpMessages, input -> {
175             final byte[] ret = new byte[input.length + 1];
176             // ff
177             ret[0] = -1;
178             System.arraycopy(input, 0, ret, 1, input.length);
179             return ret;
180         });
181     }
182
183     private void assertTablesExists(final List<BgpTableType> expectedTables, final boolean uptodate) throws InterruptedException, ExecutionException {
184         final Optional<LocRib> lockRib = getLocRibTable();
185         assertTrue(lockRib.isPresent());
186         final List<Tables> tables = lockRib.get().getTables();
187         assertFalse(tables.isEmpty());
188         for (final BgpTableType tableType : expectedTables) {
189             boolean found = false;
190             for (final Tables table : tables) {
191                 if(table.getAfi().equals(tableType.getAfi()) && table.getSafi().equals(tableType.getSafi())) {
192                     found = true;
193                     assertTrue(Boolean.valueOf(uptodate).equals(table.getAttributes().isUptodate()));
194                 }
195             }
196             assertTrue(found);
197         }
198     }
199
200     private Optional<LocRib> getLocRibTable() throws InterruptedException, ExecutionException {
201         return getDataBroker().newReadOnlyTransaction().read(LogicalDatastoreType.OPERATIONAL, InstanceIdentifier.builder(BgpRib.class).child(Rib.class, new RibKey(new RibId(TEST_RIB_ID))).child(LocRib.class).build()).get();
202     }
203 }