2d48dab4b3748eb9d77584b73306810fbe0f5e8c
[netvirt.git] / fibmanager / impl / src / main / java / org / opendaylight / netvirt / fibmanager / FibRpcServiceImpl.java
1 /*
2  * Copyright © 2016, 2017 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.fibmanager;
9
10 import com.google.common.util.concurrent.ListenableFuture;
11 import java.net.InetAddress;
12 import java.net.UnknownHostException;
13 import java.util.ArrayList;
14 import java.util.List;
15 import java.util.concurrent.ExecutionException;
16 import javax.inject.Inject;
17 import javax.inject.Singleton;
18 import org.apache.commons.lang3.tuple.ImmutablePair;
19 import org.eclipse.jdt.annotation.Nullable;
20 import org.opendaylight.genius.datastoreutils.SingleTransactionDataBroker;
21 import org.opendaylight.genius.mdsalutil.MDSALUtil;
22 import org.opendaylight.genius.mdsalutil.MatchInfo;
23 import org.opendaylight.genius.mdsalutil.MetaDataUtil;
24 import org.opendaylight.genius.mdsalutil.NwConstants;
25 import org.opendaylight.genius.mdsalutil.interfaces.IMdsalApiManager;
26 import org.opendaylight.genius.mdsalutil.matches.MatchEthernetType;
27 import org.opendaylight.genius.mdsalutil.matches.MatchIpv4Destination;
28 import org.opendaylight.genius.mdsalutil.matches.MatchMetadata;
29 import org.opendaylight.mdsal.binding.api.DataBroker;
30 import org.opendaylight.mdsal.common.api.LogicalDatastoreType;
31 import org.opendaylight.netvirt.fibmanager.api.IFibManager;
32 import org.opendaylight.netvirt.vpnmanager.api.IVpnFootprintService;
33 import org.opendaylight.yang.gen.v1.urn.opendaylight.flow.inventory.rev130819.tables.table.Flow;
34 import org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types.rev131026.instruction.list.Instruction;
35 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.CleanupDpnForVpnInput;
36 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.CleanupDpnForVpnOutput;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.CleanupDpnForVpnOutputBuilder;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.CreateFibEntryInput;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.CreateFibEntryOutput;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.CreateFibEntryOutputBuilder;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.FibRpcService;
42 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.PopulateFibOnDpnInput;
43 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.PopulateFibOnDpnOutput;
44 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.PopulateFibOnDpnOutputBuilder;
45 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.RemoveFibEntryInput;
46 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.RemoveFibEntryOutput;
47 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.fib.rpc.rev160121.RemoveFibEntryOutputBuilder;
48 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.VpnInstanceToVpnId;
49 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.instance.op.data.vpn.instance.op.data.entry.vpn.to.dpn.list.IpAddresses;
50 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.instance.to.vpn.id.VpnInstance;
51 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.l3vpn.rev130911.vpn.instance.to.vpn.id.VpnInstanceKey;
52 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
53 import org.opendaylight.yangtools.yang.common.RpcResult;
54 import org.opendaylight.yangtools.yang.common.RpcResultBuilder;
55 import org.opendaylight.yangtools.yang.common.Uint32;
56 import org.opendaylight.yangtools.yang.common.Uint64;
57 import org.slf4j.Logger;
58 import org.slf4j.LoggerFactory;
59
60 @Singleton
61 public class FibRpcServiceImpl implements FibRpcService {
62     private static final Logger LOG = LoggerFactory.getLogger(FibRpcServiceImpl.class);
63     private final DataBroker dataBroker;
64     private final IMdsalApiManager mdsalManager;
65     private final IFibManager fibManager;
66     private final IVpnFootprintService vpnFootprintService;
67
68     @Inject
69     public FibRpcServiceImpl(final DataBroker dataBroker, final IMdsalApiManager mdsalManager,
70                              final IFibManager fibManager, final IVpnFootprintService vpnFootprintService) {
71         this.dataBroker = dataBroker;
72         this.mdsalManager = mdsalManager;
73         this.fibManager = fibManager;
74         this.vpnFootprintService = vpnFootprintService;
75     }
76
77     /**
78      * To install FIB routes on specified dpn with given instructions.
79      */
80     @Override
81     public ListenableFuture<RpcResult<CreateFibEntryOutput>> createFibEntry(CreateFibEntryInput input) {
82
83         Uint64 dpnId = input.getSourceDpid();
84         String vpnName = input.getVpnName();
85         Uint32 vpnId = getVpnId(dataBroker, vpnName);
86         String vpnRd = getVpnRd(dataBroker, vpnName);
87         String ipAddress = input.getIpAddress();
88         LOG.info("Create custom FIB entry - {} on dpn {} for VPN {} ", ipAddress, dpnId, vpnName);
89         List<Instruction> instructions = input.getInstruction();
90         LOG.info("ADD: Adding Custom Fib Entry rd {} prefix {} label {}", vpnRd, ipAddress, input.getServiceId());
91         makeLocalFibEntry(vpnId, dpnId, ipAddress, instructions);
92         IpAddresses.IpAddressSource ipAddressSource = IpAddresses.IpAddressSource
93                 .forValue(input.getIpAddressSource().getIntValue());
94         vpnFootprintService.updateVpnToDpnMapping(dpnId, vpnName, vpnRd, null /* interfaceName*/,
95                 new ImmutablePair<>(ipAddressSource, ipAddress), true /*add*/);
96         LOG.info("ADD: Added Custom Fib Entry rd {} prefix {} label {}", vpnRd, ipAddress, input.getServiceId());
97         return RpcResultBuilder.success(new CreateFibEntryOutputBuilder().build()).buildFuture();
98     }
99
100     /**
101      * To remove FIB/LFIB/TST routes from specified dpn.
102      */
103     @Override
104     public ListenableFuture<RpcResult<RemoveFibEntryOutput>> removeFibEntry(RemoveFibEntryInput input) {
105         Uint64 dpnId = input.getSourceDpid();
106         String vpnName = input.getVpnName();
107         Uint32 vpnId = getVpnId(dataBroker, vpnName);
108         String vpnRd = getVpnRd(dataBroker, vpnName);
109
110         String ipAddress = input.getIpAddress();
111
112         LOG.info("Delete custom FIB entry - {} on dpn {} for VPN {} ", ipAddress, dpnId, vpnName);
113         LOG.info("REMOVE: Removing Custom Fib Entry rd {} prefix {} label {}", vpnRd, ipAddress, input.getServiceId());
114         removeLocalFibEntry(dpnId, vpnId, ipAddress);
115         IpAddresses.IpAddressSource ipAddressSource = IpAddresses.IpAddressSource
116                 .forValue(input.getIpAddressSource().getIntValue());
117         vpnFootprintService.updateVpnToDpnMapping(dpnId, vpnName, vpnRd, null /* interfaceName*/,
118                 new ImmutablePair<>(ipAddressSource, ipAddress), false /*add*/);
119         LOG.info("REMOVE: Removed Custom Fib Entry rd {} prefix {} label {}", vpnRd, ipAddress, input.getServiceId());
120         return RpcResultBuilder.success(new RemoveFibEntryOutputBuilder().build()).buildFuture();
121     }
122
123
124     @Override
125     public ListenableFuture<RpcResult<PopulateFibOnDpnOutput>> populateFibOnDpn(PopulateFibOnDpnInput input) {
126         fibManager.populateFibOnNewDpn(input.getDpid(), input.getVpnId(), input.getRd(), null);
127         return RpcResultBuilder.success(new PopulateFibOnDpnOutputBuilder().build()).buildFuture();
128     }
129
130     @Override
131     public ListenableFuture<RpcResult<CleanupDpnForVpnOutput>> cleanupDpnForVpn(CleanupDpnForVpnInput input) {
132         fibManager.cleanUpDpnForVpn(input.getDpid(), input.getVpnId(), input.getRd(), null);
133         return RpcResultBuilder.success(new CleanupDpnForVpnOutputBuilder().build()).buildFuture();
134     }
135
136     private void removeLocalFibEntry(Uint64 dpnId, Uint32 vpnId, String ipPrefix) {
137         String[] values = ipPrefix.split("/");
138         String ipAddress = values[0];
139         int prefixLength = values.length == 1 ? 0 : Integer.parseInt(values[1]);
140         LOG.debug("Removing route from DPN. ip {} masklen {}", ipAddress, prefixLength);
141         InetAddress destPrefix = null;
142         try {
143             destPrefix = InetAddress.getByName(ipAddress);
144         } catch (UnknownHostException e) {
145             LOG.error("UnknowHostException in removeRoute. Failed  to remove Route for ipPrefix {} DPN {} Vpn {}",
146                     ipAddress, dpnId, vpnId, e);
147             return;
148         }
149         List<MatchInfo> matches = new ArrayList<>();
150
151         matches.add(new MatchMetadata(MetaDataUtil.getVpnIdMetadata(vpnId.longValue()),
152             MetaDataUtil.METADATA_MASK_VRFID));
153
154         matches.add(MatchEthernetType.IPV4);
155
156         if (prefixLength != 0) {
157             matches.add(new MatchIpv4Destination(destPrefix.getHostAddress(), Integer.toString(prefixLength)));
158         }
159
160         String flowRef = getFlowRef(dpnId, NwConstants.L3_FIB_TABLE, vpnId, ipAddress);
161
162
163         int priority = FibConstants.DEFAULT_FIB_FLOW_PRIORITY + prefixLength;
164         Flow flowEntity = MDSALUtil.buildFlowNew(NwConstants.L3_FIB_TABLE, flowRef,
165             priority, flowRef, 0, 0,
166             NwConstants.COOKIE_VM_FIB_TABLE, matches, null);
167
168         mdsalManager.removeFlow(dpnId, flowEntity);
169
170         LOG.info("FIB entry for prefix {} on dpn {} vpn {} removed successfully", ipAddress, dpnId,  vpnId);
171     }
172
173     private void makeLocalFibEntry(Uint32 vpnId, Uint64 dpnId, String ipPrefix,
174                                    List<Instruction> customInstructions) {
175         String[] values = ipPrefix.split("/");
176         String ipAddress = values[0];
177         int prefixLength = values.length == 1 ? 0 : Integer.parseInt(values[1]);
178         LOG.debug("Adding route to DPN. ip {} masklen {}", ipAddress, prefixLength);
179         InetAddress destPrefix = null;
180         try {
181             destPrefix = InetAddress.getByName(ipAddress);
182         } catch (UnknownHostException e) {
183             LOG.error("UnknowHostException in addRoute. Failed  to add Route for ipPrefix {} VpnId {} DPN{}",
184                     ipAddress, vpnId, dpnId, e);
185             return;
186         }
187         List<MatchInfo> matches = new ArrayList<>();
188
189         matches.add(new MatchMetadata(MetaDataUtil.getVpnIdMetadata(vpnId.longValue()),
190             MetaDataUtil.METADATA_MASK_VRFID));
191
192         matches.add(MatchEthernetType.IPV4);
193
194         if (prefixLength != 0) {
195             matches.add(new MatchIpv4Destination(destPrefix.getHostAddress(), Integer.toString(prefixLength)));
196         }
197
198         String flowRef = getFlowRef(dpnId, NwConstants.L3_FIB_TABLE, vpnId, ipAddress);
199
200
201         int priority = FibConstants.DEFAULT_FIB_FLOW_PRIORITY + prefixLength;
202         Flow flowEntity = MDSALUtil.buildFlowNew(NwConstants.L3_FIB_TABLE, flowRef,
203             priority, flowRef, 0, 0,
204             NwConstants.COOKIE_VM_FIB_TABLE, matches, customInstructions);
205         mdsalManager.installFlow(dpnId, flowEntity);
206
207         LOG.debug("FIB entry for route {} on dpn {} installed successfully - flow {}", ipAddress, dpnId, flowEntity);
208     }
209
210     private String getFlowRef(Uint64 dpnId, short tableId, Uint32 id, String ipAddress) {
211         return FibConstants.FLOWID_PREFIX + dpnId + NwConstants.FLOWID_SEPARATOR + tableId
212                 + NwConstants.FLOWID_SEPARATOR + id + NwConstants.FLOWID_SEPARATOR + ipAddress;
213     }
214
215     //TODO: Below Util methods to be removed once VpnUtil methods are exposed in api bundle
216     @Nullable
217     public static String getVpnRd(DataBroker broker, String vpnName) {
218         InstanceIdentifier<VpnInstance> id = getVpnInstanceToVpnIdIdentifier(vpnName);
219         try {
220             return SingleTransactionDataBroker.syncReadOptional(broker, LogicalDatastoreType.CONFIGURATION, id)
221                     .map(VpnInstance::getVrfId).orElse(null);
222         } catch (ExecutionException | InterruptedException e) {
223             LOG.error("getVpnRd: Exception while reading VpnInstance DS for the vpn {}", vpnName, e);
224         }
225         return null;
226     }
227
228     static InstanceIdentifier<VpnInstance> getVpnInstanceToVpnIdIdentifier(String vpnName) {
229         return InstanceIdentifier.builder(VpnInstanceToVpnId.class)
230             .child(VpnInstance.class, new VpnInstanceKey(vpnName)).build();
231     }
232
233
234     static Uint32 getVpnId(DataBroker broker, String vpnName) {
235         InstanceIdentifier<VpnInstance> id = getVpnInstanceToVpnIdIdentifier(vpnName);
236         try {
237             return SingleTransactionDataBroker.syncReadOptional(broker, LogicalDatastoreType.CONFIGURATION, id)
238                     .map(VpnInstance::getVpnId).orElse(Uint32.ZERO);
239         } catch (ExecutionException | InterruptedException e) {
240             LOG.error("getVpnId: Exception while reading VpnInstance DS for the vpn {}", vpnName, e);
241         }
242         return Uint32.ZERO;
243     }
244 }