9b2a94fd09fe03474af20b432ee0bb3a64955c88
[netvirt.git] / vpnservice / neutronvpn / neutronvpn-impl / src / main / java / org / opendaylight / netvirt / neutronvpn / NeutronRouterChangeListener.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.neutronvpn;
9
10 import com.google.common.base.Optional;
11 import java.util.ArrayList;
12 import java.util.HashMap;
13 import java.util.Iterator;
14 import java.util.List;
15 import org.opendaylight.controller.md.sal.binding.api.DataBroker;
16 import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
17 import org.opendaylight.genius.datastoreutils.AsyncDataTreeChangeListenerBase;
18 import org.opendaylight.genius.mdsalutil.NwConstants;
19 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.yang.types.rev130715.Uuid;
20 import org.opendaylight.yang.gen.v1.urn.opendaylight.neutron.l3.rev150712.l3.attributes.Routes;
21 import org.opendaylight.yang.gen.v1.urn.opendaylight.neutron.l3.rev150712.routers.attributes.Routers;
22 import org.opendaylight.yang.gen.v1.urn.opendaylight.neutron.l3.rev150712.routers.attributes.routers.Router;
23 import org.opendaylight.yang.gen.v1.urn.opendaylight.neutron.rev150712.Neutron;
24 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.netvirt.inter.vpn.link.rev160311.inter.vpn.link.states.InterVpnLinkState;
25 import org.opendaylight.yang.gen.v1.urn.opendaylight.params.xml.ns.yang.netvirt.inter.vpn.link.rev160311.inter.vpn.links.InterVpnLink;
26 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
27 import org.slf4j.Logger;
28 import org.slf4j.LoggerFactory;
29
30 public class NeutronRouterChangeListener extends AsyncDataTreeChangeListenerBase<Router, NeutronRouterChangeListener>
31         implements AutoCloseable {
32     private static final Logger LOG = LoggerFactory.getLogger(NeutronRouterChangeListener.class);
33     private final DataBroker dataBroker;
34     private final NeutronvpnManager nvpnManager;
35     private final NeutronvpnNatManager nvpnNatManager;
36     private final NeutronSubnetGwMacResolver gwMacResolver;
37
38     public NeutronRouterChangeListener(final DataBroker dataBroker, final NeutronvpnManager nVpnMgr,
39                                        final NeutronvpnNatManager nVpnNatMgr, NeutronSubnetGwMacResolver gwMacResolver) {
40         super(Router.class, NeutronRouterChangeListener.class);
41         this.dataBroker = dataBroker;
42         nvpnManager = nVpnMgr;
43         nvpnNatManager = nVpnNatMgr;
44         this.gwMacResolver = gwMacResolver;
45     }
46
47     public void start() {
48         LOG.info("{} start", getClass().getSimpleName());
49         registerListener(LogicalDatastoreType.CONFIGURATION, dataBroker);
50     }
51
52     @Override
53     protected InstanceIdentifier<Router> getWildCardPath() {
54         return InstanceIdentifier.create(Neutron.class).child(Routers.class).child(Router.class);
55     }
56
57     @Override
58     protected NeutronRouterChangeListener getDataTreeChangeListener() {
59         return NeutronRouterChangeListener.this;
60     }
61
62
63     @Override
64     protected void add(InstanceIdentifier<Router> identifier, Router input) {
65         LOG.trace("Adding Router : key: {}, value={}", identifier, input);
66         NeutronvpnUtils.addToRouterCache(input);
67         // Create internal VPN
68         nvpnManager.createL3InternalVpn(input.getUuid(), null, null, null, null, null, input.getUuid(), null);
69         nvpnNatManager.handleExternalNetworkForRouter(null, input);
70         gwMacResolver.sendArpRequestsToExtGateways(input);
71     }
72
73     @Override
74     protected void remove(InstanceIdentifier<Router> identifier, Router input) {
75         LOG.trace("Removing router : key: {}, value={}", identifier, input);
76         Uuid routerId = input.getUuid();
77         //NOTE: Pass an empty routerSubnetIds list, as router interfaces
78         //will be removed from VPN by invocations from NeutronPortChangeListener
79         List<Uuid> routerSubnetIds = new ArrayList<>();
80         nvpnManager.handleNeutronRouterDeleted(routerId, routerSubnetIds);
81
82         // Handle router deletion for the NAT service
83         if (input.getExternalGatewayInfo() != null) {
84             Uuid extNetId = input.getExternalGatewayInfo().getExternalNetworkId();
85             nvpnNatManager.removeExternalNetworkFromRouter(extNetId, input);
86         }
87         NeutronvpnUtils.removeFromRouterCache(input);
88     }
89
90     @Override
91     protected void update(InstanceIdentifier<Router> identifier, Router original, Router update) {
92         LOG.trace("Updating Router : key: {}, original value={}, update value={}", identifier, original, update);
93         NeutronvpnUtils.addToRouterCache(update);
94         Uuid routerId = update.getUuid();
95         NeutronvpnUtils.addToRouterCache(update);
96         Uuid vpnId = NeutronvpnUtils.getVpnForRouter(dataBroker, routerId, true);
97         // internal vpn always present in case external vpn not found
98         if (vpnId == null) {
99             vpnId = routerId;
100         }
101         List<Routes> oldRoutes = (original.getRoutes() != null) ? original.getRoutes() : new ArrayList<Routes>();
102         List<Routes> newRoutes = (update.getRoutes() != null) ? update.getRoutes() : new ArrayList<Routes>();
103         if (!oldRoutes.equals(newRoutes)) {
104             Iterator<Routes> iterator = newRoutes.iterator();
105             while (iterator.hasNext()) {
106                 Routes route = iterator.next();
107                 if (oldRoutes.remove(route)) {
108                     iterator.remove();
109                 }
110             }
111
112             handleChangedRoutes(vpnId, newRoutes, NwConstants.ADD_FLOW);
113
114             if (!oldRoutes.isEmpty()) {
115                 handleChangedRoutes(vpnId, oldRoutes, NwConstants.DEL_FLOW);
116             }
117         }
118
119         nvpnNatManager.handleExternalNetworkForRouter(original, update);
120         gwMacResolver.sendArpRequestsToExtGateways(update);
121     }
122
123     private void handleChangedRoutes(Uuid vpnName, List<Routes> routes, int addedOrRemoved) {
124         // Some routes may point to an InterVpnLink's endpoint, lets treat them differently
125         List<Routes> interVpnLinkRoutes = new ArrayList<Routes>();
126         List<Routes> otherRoutes = new ArrayList<Routes>();
127         HashMap<String, InterVpnLink> nexthopsXinterVpnLinks = new HashMap<String, InterVpnLink>();
128         for ( Routes route : routes ) {
129             String nextHop = String.valueOf(route.getNexthop().getValue());
130             // Nexthop is another VPN?
131             Optional<InterVpnLink> interVpnLink = NeutronvpnUtils.getInterVpnLinkByEndpointIp(dataBroker, nextHop);
132             if ( interVpnLink.isPresent() ) {
133                 Optional<InterVpnLinkState> interVpnLinkState =
134                         NeutronvpnUtils.getInterVpnLinkState(dataBroker, interVpnLink.get().getName());
135                 if ( interVpnLinkState.isPresent() && interVpnLinkState.get().getState() == InterVpnLinkState.State.Active) {
136                     interVpnLinkRoutes.add(route);
137                     nexthopsXinterVpnLinks.put(nextHop, interVpnLink.get());
138                 } else {
139                     LOG.warn("Failed installing route to {}. Reason: InterVPNLink {} is not Active",
140                             String.valueOf(route.getDestination().getValue()), interVpnLink.get().getName());
141                 }
142             } else {
143                 otherRoutes.add(route);
144             }
145         }
146
147         if ( addedOrRemoved == NwConstants.ADD_FLOW ) {
148             nvpnManager.addInterVpnRoutes(vpnName, interVpnLinkRoutes, nexthopsXinterVpnLinks);
149             nvpnManager.updateVpnInterfaceWithExtraRouteAdjacency(vpnName, otherRoutes);
150         } else {
151             nvpnManager.removeAdjacencyforExtraRoute(vpnName, otherRoutes);
152             nvpnManager.removeInterVpnRoutes(vpnName, interVpnLinkRoutes, nexthopsXinterVpnLinks);
153         }
154     }
155 }