MRI version bumpup for Aluminium
[netvirt.git] / vpnmanager / api / src / main / java / org / opendaylight / netvirt / vpnmanager / api / VpnExtraRouteHelper.java
1 /*
2  * Copyright (c) 2016 Ericsson India Global Services Pvt Ltd. 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.netvirt.vpnmanager.api;
9
10 import static java.util.stream.Collectors.toList;
11
12 import java.util.ArrayList;
13 import java.util.Collections;
14 import java.util.List;
15 import java.util.Map;
16 import java.util.Optional;
17 import java.util.concurrent.ExecutionException;
18 import java.util.concurrent.Future;
19 import org.eclipse.jdt.annotation.Nullable;
20 import org.opendaylight.genius.datastoreutils.SingleTransactionDataBroker;
21 import org.opendaylight.genius.infra.Datastore.Configuration;
22 import org.opendaylight.genius.infra.Datastore.Operational;
23 import org.opendaylight.genius.infra.TypedReadTransaction;
24 import org.opendaylight.genius.mdsalutil.MDSALUtil;
25 import org.opendaylight.mdsal.binding.api.DataBroker;
26 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
27 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.rev160406.TunnelTypeBase;
28 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.itm.rpcs.rev160406.GetTunnelTypeInputBuilder;
29 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.itm.rpcs.rev160406.GetTunnelTypeOutput;
30 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.itm.rpcs.rev160406.ItmRpcService;
31 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.ExtrarouteRdsMap;
32 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.extraroute.rds.map.ExtrarouteRds;
33 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.extraroute.rds.map.ExtrarouteRdsKey;
34 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.extraroute.rds.map.extraroute.rds.DestPrefixes;
35 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.extraroute.rds.map.extraroute.rds.DestPrefixesKey;
36 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.extraroute.rds.map.extraroute.rds.dest.prefixes.AllocatedRds;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fibmanager.rev150330.extraroute.rds.map.extraroute.rds.dest.prefixes.AllocatedRdsKey;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.VpnToExtraroutes;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.to.extraroutes.Vpn;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.to.extraroutes.VpnKey;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.to.extraroutes.vpn.ExtraRoutes;
42 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.to.extraroutes.vpn.ExtraRoutesKey;
43 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.to.extraroutes.vpn.extra.routes.Routes;
44 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.to.extraroutes.vpn.extra.routes.RoutesKey;
45 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
46 import org.opendaylight.yangtools.yang.common.RpcResult;
47 import org.opendaylight.yangtools.yang.common.Uint32;
48 import org.slf4j.Logger;
49 import org.slf4j.LoggerFactory;
50
51 public final class VpnExtraRouteHelper {
52     private static final Logger LOG = LoggerFactory.getLogger(VpnExtraRouteHelper.class);
53
54     private VpnExtraRouteHelper() {
55
56     }
57
58     public static Optional<Routes> getVpnExtraroutes(DataBroker broker, String vpnName,
59                                                      String vpnRd, String destPrefix) {
60         InstanceIdentifier<Routes> vpnExtraRoutesId = getVpnToExtrarouteVrfIdIdentifier(vpnName, vpnRd, destPrefix);
61         Optional<Routes> extraRouteOptional = Optional.empty();
62         try {
63             extraRouteOptional = SingleTransactionDataBroker.syncReadOptional(broker, LogicalDatastoreType.OPERATIONAL,
64                     vpnExtraRoutesId);
65         } catch (ExecutionException | InterruptedException e) {
66             LOG.error("getVpnExtraroutes: failed to read VpnToExtraRoutes for vpn {} rd {} destprefix {} due "
67                     + "to exception", vpnName, vpnRd, destPrefix, e);
68         }
69         return extraRouteOptional;
70     }
71
72     public static Optional<Routes> getVpnExtraroutes(TypedReadTransaction<Operational> operTx, String vpnName,
73             String vpnRd, String destPrefix) throws ExecutionException, InterruptedException {
74         return operTx.read(getVpnToExtrarouteVrfIdIdentifier(vpnName, vpnRd, destPrefix)).get();
75     }
76
77     public static  InstanceIdentifier<Routes> getVpnToExtrarouteVrfIdIdentifier(String vpnName, String vrfId,
78             String ipPrefix) {
79         return InstanceIdentifier.builder(VpnToExtraroutes.class)
80                 .child(Vpn.class, new VpnKey(vpnName)).child(ExtraRoutes.class,
81                         new ExtraRoutesKey(vrfId)).child(Routes.class, new RoutesKey(ipPrefix)).build();
82     }
83
84     public static  InstanceIdentifier<Vpn> getVpnToExtrarouteVpnIdentifier(String vpnName) {
85         return InstanceIdentifier.builder(VpnToExtraroutes.class)
86                 .child(Vpn.class, new VpnKey(vpnName)).build();
87     }
88
89     public static List<Routes> getAllVpnExtraRoutes(DataBroker dataBroker, String vpnName,
90             List<String> usedRds, String destPrefix) {
91         List<Routes> routes = new ArrayList<>();
92         for (String rd : usedRds) {
93             try {
94                 Optional<Routes> extraRouteInfo = MDSALUtil.read(dataBroker, LogicalDatastoreType.OPERATIONAL,
95                                 getVpnToExtrarouteVrfIdIdentifier(vpnName, rd, destPrefix));
96                 extraRouteInfo.ifPresent(routes::add);
97             } catch (ExecutionException | InterruptedException e) {
98                 LOG.error("getAllVpnExtraRoutes: failed to read VpnToExtraRouteVrf for vpn {} rd {} destprefix {} due "
99                        + "to exception", vpnName, rd, destPrefix, e);
100             }
101         }
102         return routes;
103     }
104
105     public static  List<String> getUsedRds(DataBroker broker, Uint32 vpnId, String destPrefix) {
106         InstanceIdentifier<DestPrefixes> usedRdsId = getUsedRdsIdentifier(vpnId, destPrefix);
107         Optional<DestPrefixes> usedRds = Optional.empty();
108         try {
109             usedRds = MDSALUtil.read(broker, LogicalDatastoreType.CONFIGURATION, usedRdsId);
110         } catch (ExecutionException | InterruptedException e) {
111             LOG.error("getUsedRds: failed to read Used Rds for vpn {} destprefix {} due to exception", vpnId,
112                     destPrefix, e);
113         }
114         return usedRds.isPresent() && usedRds.get().getAllocatedRds() != null
115                 ? usedRds.get().getAllocatedRds().values().stream()
116                 .map(AllocatedRds::getRd).distinct().collect(toList()) : new ArrayList<>();
117     }
118
119     public static  List<String> getUsedRds(TypedReadTransaction<Configuration> confTx, Uint32 vpnId, String destPrefix)
120             throws ExecutionException, InterruptedException {
121         Optional<DestPrefixes> usedRds = confTx.read(getUsedRdsIdentifier(vpnId, destPrefix)).get();
122         return usedRds.isPresent() && usedRds.get().getAllocatedRds() != null
123                 ? usedRds.get().getAllocatedRds().values().stream()
124             .map(AllocatedRds::getRd).distinct().collect(toList()) : new ArrayList<>();
125     }
126
127     public static  InstanceIdentifier<ExtrarouteRds> getUsedRdsIdentifier(Uint32 vpnId) {
128         return InstanceIdentifier.builder(ExtrarouteRdsMap.class)
129                 .child(ExtrarouteRds.class, new ExtrarouteRdsKey(vpnId)).build();
130     }
131
132     public static  InstanceIdentifier<DestPrefixes> getUsedRdsIdentifier(Uint32 vpnId, String destPrefix) {
133         return InstanceIdentifier.builder(ExtrarouteRdsMap.class)
134                 .child(ExtrarouteRds.class, new ExtrarouteRdsKey(vpnId))
135                 .child(DestPrefixes.class, new DestPrefixesKey(destPrefix)).build();
136     }
137
138     public static  InstanceIdentifier<AllocatedRds> getUsedRdsIdentifier(Uint32 vpnId, String destPrefix, String nh) {
139         return InstanceIdentifier.builder(ExtrarouteRdsMap.class)
140                 .child(ExtrarouteRds.class, new ExtrarouteRdsKey(vpnId))
141                 .child(DestPrefixes.class, new DestPrefixesKey(destPrefix))
142                 .child(AllocatedRds.class, new AllocatedRdsKey(nh)).build();
143     }
144
145     @Nullable
146     public static Class<? extends TunnelTypeBase> getTunnelType(ItmRpcService itmRpcService, String ifName) {
147         try {
148             Future<RpcResult<GetTunnelTypeOutput>> result =
149                     itmRpcService.getTunnelType(new GetTunnelTypeInputBuilder().setIntfName(ifName).build());
150             RpcResult<GetTunnelTypeOutput> rpcResult = result.get();
151             if (!rpcResult.isSuccessful()) {
152                 LOG.warn("RPC Call to getTunnelInterfaceId returned with Errors {}", rpcResult.getErrors());
153             } else {
154                 return rpcResult.getResult().getTunnelType();
155             }
156         } catch (InterruptedException | ExecutionException e) {
157             LOG.warn("Exception when getting tunnel interface Id for tunnel type", e);
158         }
159         return null;
160     }
161
162     public static java.util.Optional<String> getRdAllocatedForExtraRoute(DataBroker broker,
163             Uint32 vpnId, String destPrefix, String nextHop) {
164         InstanceIdentifier<AllocatedRds> usedRdsId = getUsedRdsIdentifier(vpnId, destPrefix, nextHop);
165         try {
166             return MDSALUtil.read(broker, LogicalDatastoreType.CONFIGURATION, usedRdsId)
167                     .map(AllocatedRds::getRd);
168         } catch (ExecutionException | InterruptedException e) {
169             LOG.error("getRdAllocatedForExtraRoute: failed to read Used Rds for vpn {} destprefix {} nexthop {} due "
170                     + "to exception", vpnId, destPrefix, nextHop, e);
171         }
172         return Optional.empty();
173     }
174
175     public static List<DestPrefixes> getExtraRouteDestPrefixes(DataBroker broker, Uint32 vpnId) {
176         try {
177             Optional<ExtrarouteRds> optionalExtraRoutes = MDSALUtil.read(broker, LogicalDatastoreType.CONFIGURATION,
178                     getUsedRdsIdentifier(vpnId));
179             Map<DestPrefixesKey, DestPrefixes> prefixesMap
180                     = optionalExtraRoutes.map(ExtrarouteRds::getDestPrefixes).orElse(null);
181             return prefixesMap == null ? Collections.emptyList() : new ArrayList<DestPrefixes>(prefixesMap.values());
182         } catch (ExecutionException | InterruptedException e) {
183             LOG.error("getExtraRouteDestPrefixes: failed to read ExRoutesRdsMap for vpn {} due to exception", vpnId, e);
184         }
185         return new ArrayList<>();
186     }
187 }