Bump upstream versions
[bgpcep.git] / bgp / benchmark-app / src / main / java / org / opendaylight / protocol / bgp / benchmark / app / AppPeerBenchmark.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.protocol.bgp.benchmark.app;
9
10 import static java.util.Objects.requireNonNull;
11 import static org.opendaylight.protocol.bgp.parser.spi.PathIdUtil.NON_PATH_ID;
12
13 import com.google.common.annotations.VisibleForTesting;
14 import com.google.common.base.Stopwatch;
15 import com.google.common.net.InetAddresses;
16 import com.google.common.util.concurrent.FutureCallback;
17 import com.google.common.util.concurrent.ListenableFuture;
18 import com.google.common.util.concurrent.MoreExecutors;
19 import java.util.Map;
20 import java.util.concurrent.ExecutionException;
21 import java.util.concurrent.TimeUnit;
22 import org.opendaylight.mdsal.binding.api.DataBroker;
23 import org.opendaylight.mdsal.binding.api.RpcProviderService;
24 import org.opendaylight.mdsal.binding.api.Transaction;
25 import org.opendaylight.mdsal.binding.api.TransactionChain;
26 import org.opendaylight.mdsal.binding.api.TransactionChainListener;
27 import org.opendaylight.mdsal.binding.api.WriteTransaction;
28 import org.opendaylight.mdsal.common.api.CommitInfo;
29 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
30 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.Ipv4AddressNoZone;
31 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.Ipv4Prefix;
32 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.inet.rev180329.application.rib.tables.routes.Ipv4RoutesCaseBuilder;
33 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.inet.rev180329.bgp.rib.rib.loc.rib.tables.routes.Ipv4RoutesCase;
34 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.inet.rev180329.ipv4.routes.Ipv4Routes;
35 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.inet.rev180329.ipv4.routes.Ipv4RoutesBuilder;
36 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.inet.rev180329.ipv4.routes.ipv4.routes.Ipv4Route;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.inet.rev180329.ipv4.routes.ipv4.routes.Ipv4RouteBuilder;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.inet.rev180329.ipv4.routes.ipv4.routes.Ipv4RouteKey;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.Attributes;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.AttributesBuilder;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.attributes.AsPath;
42 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.attributes.AsPathBuilder;
43 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.attributes.LocalPref;
44 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.attributes.LocalPrefBuilder;
45 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.attributes.MultiExitDisc;
46 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.attributes.MultiExitDiscBuilder;
47 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.attributes.Origin;
48 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.message.rev200120.path.attributes.attributes.OriginBuilder;
49 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev180329.ApplicationRib;
50 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev180329.ApplicationRibBuilder;
51 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev180329.ApplicationRibId;
52 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev180329.ApplicationRibKey;
53 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev180329.rib.Tables;
54 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev180329.rib.TablesBuilder;
55 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.rib.rev180329.rib.TablesKey;
56 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev200120.BgpOrigin;
57 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev200120.Ipv4AddressFamily;
58 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev200120.UnicastSubsequentAddressFamily;
59 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev200120.next.hop.c.next.hop.Ipv4NextHopCaseBuilder;
60 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.bgp.types.rev200120.next.hop.c.next.hop.ipv4.next.hop._case.Ipv4NextHopBuilder;
61 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.odl.bgp.app.peer.benchmark.rev200120.AddPrefixInput;
62 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.odl.bgp.app.peer.benchmark.rev200120.AddPrefixOutput;
63 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.odl.bgp.app.peer.benchmark.rev200120.AddPrefixOutputBuilder;
64 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.odl.bgp.app.peer.benchmark.rev200120.DeletePrefixInput;
65 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.odl.bgp.app.peer.benchmark.rev200120.DeletePrefixOutput;
66 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.odl.bgp.app.peer.benchmark.rev200120.DeletePrefixOutputBuilder;
67 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.odl.bgp.app.peer.benchmark.rev200120.OdlBgpAppPeerBenchmarkService;
68 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.odl.bgp.app.peer.benchmark.rev200120.output.Result;
69 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.odl.bgp.app.peer.benchmark.rev200120.output.ResultBuilder;
70 import org.opendaylight.yangtools.concepts.ObjectRegistration;
71 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
72 import org.opendaylight.yangtools.yang.binding.KeyedInstanceIdentifier;
73 import org.opendaylight.yangtools.yang.binding.util.BindingMap;
74 import org.opendaylight.yangtools.yang.common.RpcResult;
75 import org.opendaylight.yangtools.yang.common.RpcResultBuilder;
76 import org.opendaylight.yangtools.yang.common.Uint32;
77 import org.slf4j.Logger;
78 import org.slf4j.LoggerFactory;
79
80 public final class AppPeerBenchmark implements OdlBgpAppPeerBenchmarkService, TransactionChainListener, AutoCloseable {
81
82     private static final Logger LOG = LoggerFactory.getLogger(AppPeerBenchmark.class);
83
84     private static final AsPath AS_PATH = new AsPathBuilder().build();
85     private static final Origin ORIGIN = new OriginBuilder().setValue(BgpOrigin.Igp).build();
86     private static final MultiExitDisc MED = new MultiExitDiscBuilder().setMed(Uint32.ZERO).build();
87     private static final LocalPref LOC_PREF = new LocalPrefBuilder().setPref(Uint32.valueOf(100)).build();
88     private static final Map<TablesKey, Tables> EMPTY_TABLES = BindingMap.of(new TablesBuilder()
89         .setAfi(Ipv4AddressFamily.class).setSafi(UnicastSubsequentAddressFamily.class).setRoutes(
90             new Ipv4RoutesCaseBuilder().setIpv4Routes(new Ipv4RoutesBuilder().setIpv4Route(Map.of())
91                 .build()).build()).build());
92
93     private static final String SLASH = "/";
94     private static final String PREFIX = SLASH + "32";
95
96     private final TransactionChain txChain;
97     private final ObjectRegistration<OdlBgpAppPeerBenchmarkService> rpcRegistration;
98     private final InstanceIdentifier<ApplicationRib> appIID;
99     private final InstanceIdentifier<Ipv4Routes> routesIId;
100     private final String appRibId;
101
102     public AppPeerBenchmark(final DataBroker bindingDataBroker, final RpcProviderService rpcProviderRegistry,
103             final String appRibId) {
104         this.appRibId = requireNonNull(appRibId);
105         txChain = bindingDataBroker.createMergingTransactionChain(this);
106
107         appIID = InstanceIdentifier.builder(ApplicationRib.class,
108             new ApplicationRibKey(new ApplicationRibId(appRibId))).build();
109         routesIId = appIID
110             .child(Tables.class, new TablesKey(Ipv4AddressFamily.class, UnicastSubsequentAddressFamily.class))
111             .child(Ipv4RoutesCase.class, Ipv4Routes.class);
112         rpcRegistration = rpcProviderRegistry.registerRpcImplementation(OdlBgpAppPeerBenchmarkService.class, this);
113         LOG.info("BGP Application Peer Benchmark Application started.");
114     }
115
116     public void start() {
117         LOG.debug("Instantiating App Peer Benchmark : {}", appRibId);
118         final ApplicationRib appRib = new ApplicationRibBuilder().setId(new ApplicationRibId(
119             new ApplicationRibId(appRibId))).setTables(EMPTY_TABLES).build();
120
121         final WriteTransaction wTx = txChain.newWriteOnlyTransaction();
122         wTx.put(LogicalDatastoreType.CONFIGURATION, appIID, appRib);
123         wTx.commit().addCallback(new FutureCallback<CommitInfo>() {
124             @Override
125             public void onSuccess(final CommitInfo result) {
126                 LOG.info("Empty Structure created for Application Peer Benchmark {}", appRibId);
127             }
128
129             @Override
130             public void onFailure(final Throwable throwable) {
131                 LOG.error("Failed to create Empty Structure for Application Peer Benchmark {}", appRibId, throwable);
132             }
133         }, MoreExecutors.directExecutor());
134     }
135
136     @Override
137     public void onTransactionChainFailed(final TransactionChain chain, final Transaction transaction,
138             final Throwable cause) {
139         LOG.error("Broken chain {} in DatastoreBaAbstractWrite, transaction {}", chain, transaction.getIdentifier(),
140             cause);
141         close();
142     }
143
144     @Override
145     public void onTransactionChainSuccessful(final TransactionChain chain) {
146         LOG.debug("DatastoreBaAbstractWrite closed successfully, chain {}", chain);
147     }
148
149     @Override
150     public ListenableFuture<RpcResult<AddPrefixOutput>> addPrefix(final AddPrefixInput input) {
151         final long duration = addRoute(input.getPrefix(), input.getNexthop(), input.getCount(), input.getBatchsize());
152         final long rate = countRate(duration, input.getCount());
153
154         return RpcResultBuilder.success(
155             new AddPrefixOutputBuilder().setResult(createResult(input.getCount(), duration, rate)).build())
156             .buildFuture();
157     }
158
159     @Override
160     public ListenableFuture<RpcResult<DeletePrefixOutput>> deletePrefix(final DeletePrefixInput input) {
161         final long duration = deleteRoute(input.getPrefix(), input.getCount(), input.getBatchsize());
162         final long rate = countRate(duration, input.getCount());
163
164         return RpcResultBuilder.success(
165             new DeletePrefixOutputBuilder().setResult(createResult(input.getCount(), duration, rate)).build())
166             .buildFuture();
167     }
168
169     @Override
170     public void close() {
171         rpcRegistration.close();
172         final WriteTransaction dTx = txChain.newWriteOnlyTransaction();
173         dTx.delete(LogicalDatastoreType.CONFIGURATION, appIID);
174         try {
175             dTx.commit().get();
176         } catch (final InterruptedException | ExecutionException e) {
177             LOG.warn("Failed to clean-up BGP Application RIB.", e);
178         }
179         txChain.close();
180         LOG.info("BGP Application Peer Benchmark Application closed.");
181     }
182
183     @VisibleForTesting
184     InstanceIdentifier<Ipv4Routes> getIpv4RoutesIID() {
185         return routesIId;
186     }
187
188     private long addRoute(final Ipv4Prefix ipv4Prefix, final Ipv4AddressNoZone nextHop, final Uint32 count,
189             final Uint32 batch) {
190         return processRoutes(ipv4Prefix, count, batch, new AttributesBuilder()
191             .setCNextHop(new Ipv4NextHopCaseBuilder()
192                 .setIpv4NextHop(new Ipv4NextHopBuilder().setGlobal(nextHop).build())
193                 .build())
194             .setMultiExitDisc(MED)
195             .setLocalPref(LOC_PREF)
196             .setOrigin(ORIGIN)
197             .setAsPath(AS_PATH)
198             .build());
199     }
200
201     private long deleteRoute(final Ipv4Prefix ipv4Prefix, final Uint32 count, final Uint32 batch) {
202         return processRoutes(ipv4Prefix, count, batch, null);
203     }
204
205     private long processRoutes(final Ipv4Prefix ipv4Prefix, final Uint32 count, final Uint32 batch,
206         final Attributes attributes) {
207         WriteTransaction wt = txChain.newWriteOnlyTransaction();
208         String address = getAdddressFromPrefix(ipv4Prefix);
209         final long countLong = count.longValue();
210         final long batchLong = batch.longValue();
211         final Stopwatch stopwatch = Stopwatch.createStarted();
212         for (int i = 1; i <= countLong; i++) {
213             final Ipv4RouteKey routeKey = new Ipv4RouteKey(NON_PATH_ID, createKey(address));
214             final KeyedInstanceIdentifier<Ipv4Route, Ipv4RouteKey> routeIId =
215                 routesIId.child(Ipv4Route.class, routeKey);
216             if (attributes != null) {
217                 wt.put(LogicalDatastoreType.CONFIGURATION, routeIId, new Ipv4RouteBuilder()
218                     .setRouteKey(routeKey.getRouteKey())
219                     .setPrefix(new Ipv4Prefix(routeKey.getRouteKey()))
220                     .withKey(routeKey)
221                     .setAttributes(attributes)
222                     .build());
223             } else {
224                 wt.delete(LogicalDatastoreType.CONFIGURATION, routeIId);
225             }
226             if (i % batchLong == 0) {
227                 wt.commit().addCallback(new FutureCallback<CommitInfo>() {
228                     @Override
229                     public void onSuccess(final CommitInfo result) {
230                         LOG.trace("Successful commit");
231                     }
232
233                     @Override
234                     public void onFailure(final Throwable trw) {
235                         LOG.error("Failed commit", trw);
236                     }
237                 }, MoreExecutors.directExecutor());
238                 wt = txChain.newWriteOnlyTransaction();
239             }
240             address = increasePrefix(address);
241         }
242         wt.commit().addCallback(new FutureCallback<CommitInfo>() {
243             @Override
244             public void onSuccess(final CommitInfo result) {
245                 LOG.trace("Route batch stored.");
246             }
247
248             @Override
249             public void onFailure(final Throwable throwable) {
250                 LOG.error("Failed to store route batch.", throwable);
251             }
252         }, MoreExecutors.directExecutor());
253         return stopwatch.stop().elapsed(TimeUnit.MILLISECONDS);
254     }
255
256     private static long countRate(final long durationMillis, final Uint32 count) {
257         final long durationSec = TimeUnit.MILLISECONDS.toSeconds(durationMillis);
258         if (durationSec != 0) {
259             return count.toJava() / durationSec;
260         }
261         return count.toJava();
262     }
263
264     private static String increasePrefix(final String prefix) {
265         return InetAddresses.increment(InetAddresses.forString(prefix)).getHostAddress();
266     }
267
268     private static Result createResult(final Uint32 count, final long duration, final long rate) {
269         return new ResultBuilder().setCount(count).setDuration(Uint32.valueOf(duration)).setRate(Uint32.valueOf(rate))
270                 .build();
271     }
272
273     private static String getAdddressFromPrefix(final Ipv4Prefix prefix) {
274         return prefix.getValue().split(SLASH)[0];
275     }
276
277     private static String createKey(final String address) {
278         return address + PREFIX;
279     }
280 }