2 * Copyright (c) 2016, 2017 Ericsson India Global Services Pvt Ltd. and others. All rights reserved.
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
8 package org.opendaylight.genius.interfacemanager.listeners;
10 import com.google.common.util.concurrent.ListenableFuture;
12 import java.math.BigInteger;
13 import java.util.ArrayList;
14 import java.util.List;
15 import java.util.Objects;
16 import java.util.concurrent.Callable;
17 import javax.inject.Inject;
18 import javax.inject.Singleton;
19 import org.opendaylight.controller.md.sal.binding.api.DataBroker;
20 import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
21 import org.opendaylight.genius.datastoreutils.AsyncClusteredDataTreeChangeListenerBase;
22 import org.opendaylight.genius.datastoreutils.DataStoreJobCoordinator;
23 import org.opendaylight.genius.interfacemanager.IfmConstants;
24 import org.opendaylight.genius.interfacemanager.IfmUtil;
25 import org.opendaylight.genius.interfacemanager.commons.InterfaceManagerCommonUtils;
26 import org.opendaylight.genius.interfacemanager.commons.InterfaceMetaUtils;
27 import org.opendaylight.genius.interfacemanager.renderer.ovs.statehelpers.OvsInterfaceStateAddHelper;
28 import org.opendaylight.genius.interfacemanager.renderer.ovs.statehelpers.OvsInterfaceStateRemoveHelper;
29 import org.opendaylight.genius.interfacemanager.renderer.ovs.statehelpers.OvsInterfaceStateUpdateHelper;
30 import org.opendaylight.genius.interfacemanager.renderer.ovs.utilities.IfmClusterUtils;
31 import org.opendaylight.genius.mdsalutil.interfaces.IMdsalApiManager;
32 import org.opendaylight.yang.gen.v1.urn.opendaylight.flow.inventory.rev130819.FlowCapableNodeConnector;
33 import org.opendaylight.yang.gen.v1.urn.opendaylight.flow.types.port.rev130925.PortReason;
34 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.alivenessmonitor.rev160411.AlivenessMonitorService;
35 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.idmanager.rev160406.IdManagerService;
36 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.meta.rev160406._interface.child.info.InterfaceParentEntry;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.meta.rev160406._interface.child.info._interface.parent.entry.InterfaceChildEntry;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.NodeConnectorId;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.Nodes;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.node.NodeConnector;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.nodes.Node;
42 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
43 import org.slf4j.Logger;
44 import org.slf4j.LoggerFactory;
47 * This Class is a Data Change Listener for FlowCapableNodeConnector updates.
48 * This creates an entry in the interface-state OperDS for every node-connector
52 * NOTE: This class just creates an ifstate entry whose interface-name will be
53 * the same as the node-connector portname. If PortName is not unique across
54 * DPNs, this implementation can have problems.
57 public class InterfaceInventoryStateListener
58 extends AsyncClusteredDataTreeChangeListenerBase<FlowCapableNodeConnector, InterfaceInventoryStateListener> {
59 private static final Logger LOG = LoggerFactory.getLogger(InterfaceInventoryStateListener.class);
60 private final DataBroker dataBroker;
61 private final IdManagerService idManager;
62 private final IMdsalApiManager mdsalApiManager;
63 private final AlivenessMonitorService alivenessMonitorService;
66 public InterfaceInventoryStateListener(final DataBroker dataBroker, final IdManagerService idManagerService,
67 final IMdsalApiManager mdsalApiManager, final AlivenessMonitorService alivenessMonitorService) {
68 super(FlowCapableNodeConnector.class, InterfaceInventoryStateListener.class);
69 this.dataBroker = dataBroker;
70 this.idManager = idManagerService;
71 this.mdsalApiManager = mdsalApiManager;
72 this.alivenessMonitorService = alivenessMonitorService;
73 this.registerListener(LogicalDatastoreType.OPERATIONAL, dataBroker);
77 protected InstanceIdentifier<FlowCapableNodeConnector> getWildCardPath() {
78 return InstanceIdentifier.create(Nodes.class).child(Node.class).child(NodeConnector.class)
79 .augmentation(FlowCapableNodeConnector.class);
83 protected InterfaceInventoryStateListener getDataTreeChangeListener() {
84 return InterfaceInventoryStateListener.this;
88 protected void remove(InstanceIdentifier<FlowCapableNodeConnector> key,
89 FlowCapableNodeConnector flowCapableNodeConnectorOld) {
90 if (!IfmClusterUtils.isEntityOwner(IfmClusterUtils.INTERFACE_CONFIG_ENTITY)) {
94 LOG.debug("Received NodeConnector Remove Event: {}, {}", key, flowCapableNodeConnectorOld);
95 NodeConnectorId nodeConnectorId = InstanceIdentifier.keyOf(key.firstIdentifierOf(NodeConnector.class))
97 String portName = InterfaceManagerCommonUtils.getPortNameForInterface(nodeConnectorId,
98 flowCapableNodeConnectorOld.getName());
100 remove(nodeConnectorId, null, flowCapableNodeConnectorOld, portName, true);
103 private void remove(NodeConnectorId nodeConnectorIdNew, NodeConnectorId nodeConnectorIdOld,
104 FlowCapableNodeConnector fcNodeConnectorNew, String portName, boolean isNetworkEvent) {
105 boolean isNodePresent = InterfaceManagerCommonUtils.isNodePresent(dataBroker, nodeConnectorIdNew);
106 DataStoreJobCoordinator coordinator = DataStoreJobCoordinator.getInstance();
107 InterfaceStateRemoveWorker portStateRemoveWorker = new InterfaceStateRemoveWorker(idManager, nodeConnectorIdNew,
108 nodeConnectorIdOld, fcNodeConnectorNew, portName, portName, isNodePresent, isNetworkEvent, true);
109 coordinator.enqueueJob(portName, portStateRemoveWorker, IfmConstants.JOB_MAX_RETRIES);
113 protected void update(InstanceIdentifier<FlowCapableNodeConnector> key, FlowCapableNodeConnector fcNodeConnectorOld,
114 FlowCapableNodeConnector fcNodeConnectorNew) {
115 if (!IfmClusterUtils.isEntityOwner(IfmClusterUtils.INTERFACE_CONFIG_ENTITY)) {
119 LOG.debug("Received NodeConnector Update Event: {}, {}, {}", key, fcNodeConnectorOld, fcNodeConnectorNew);
120 NodeConnectorId nodeConnectorId = InstanceIdentifier.keyOf(key.firstIdentifierOf(NodeConnector.class))
122 String portName = InterfaceManagerCommonUtils.getPortNameForInterface(nodeConnectorId,
123 fcNodeConnectorNew.getName());
124 DataStoreJobCoordinator coordinator = DataStoreJobCoordinator.getInstance();
126 InterfaceStateUpdateWorker portStateUpdateWorker = new InterfaceStateUpdateWorker(key, fcNodeConnectorOld,
127 fcNodeConnectorNew, portName);
128 coordinator.enqueueJob(portName, portStateUpdateWorker, IfmConstants.JOB_MAX_RETRIES);
132 protected void add(InstanceIdentifier<FlowCapableNodeConnector> key, FlowCapableNodeConnector fcNodeConnectorNew) {
133 if (!IfmClusterUtils.isEntityOwner(IfmClusterUtils.INTERFACE_CONFIG_ENTITY)) {
137 LOG.debug("Received NodeConnector Add Event: {}, {}", key, fcNodeConnectorNew);
138 NodeConnectorId nodeConnectorId = InstanceIdentifier.keyOf(key.firstIdentifierOf(NodeConnector.class))
140 String portName = InterfaceManagerCommonUtils.getPortNameForInterface(nodeConnectorId,
141 fcNodeConnectorNew.getName());
143 if (InterfaceManagerCommonUtils.isNovaPort(portName)) {
144 NodeConnectorId nodeConnectorIdOld = IfmUtil.getNodeConnectorIdFromInterface(portName, dataBroker);
145 if (nodeConnectorIdOld != null && !nodeConnectorId.equals(nodeConnectorIdOld)) {
146 BigInteger dpnIdOld = IfmUtil.getDpnFromNodeConnectorId(nodeConnectorIdOld);
147 BigInteger dpnIdNew = IfmUtil.getDpnFromNodeConnectorId(nodeConnectorId);
148 if (!Objects.equals(dpnIdOld, dpnIdNew)) {
149 if (fcNodeConnectorNew.getReason() != PortReason.Add) {
150 LOG.error("Dropping Port update event for {}, as DPN id is changed from {} to {}",
151 fcNodeConnectorNew.getName(), dpnIdOld, dpnIdNew);
155 LOG.warn("Port number update detected for {}", fcNodeConnectorNew.getName());
157 //VM Migration or Port Number Update: Delete existing interface entry for older DPN
158 LOG.debug("Triggering NodeConnector Remove Event for the interface: {}, {}, {}", portName,
159 nodeConnectorId, nodeConnectorIdOld);
160 remove(nodeConnectorId, nodeConnectorIdOld, fcNodeConnectorNew, portName, false);
161 // Adding a delay of 10sec for VM migration, so applications will have sufficient time
162 // for processing remove before add
164 Thread.sleep(IfmConstants.DELAY_TIME_IN_MILLISECOND);
165 } catch (InterruptedException e) {
166 LOG.error("Error while waiting for the vm migration remove events to get processed");
170 DataStoreJobCoordinator coordinator = DataStoreJobCoordinator.getInstance();
171 InterfaceStateAddWorker ifStateAddWorker = new InterfaceStateAddWorker(idManager, nodeConnectorId,
172 fcNodeConnectorNew, portName);
173 coordinator.enqueueJob(portName, ifStateAddWorker, IfmConstants.JOB_MAX_RETRIES);
177 private class InterfaceStateAddWorker implements Callable {
178 private final NodeConnectorId nodeConnectorId;
179 private final FlowCapableNodeConnector fcNodeConnectorNew;
180 private final String interfaceName;
181 private final IdManagerService idManager;
183 InterfaceStateAddWorker(IdManagerService idManager, NodeConnectorId nodeConnectorId,
184 FlowCapableNodeConnector fcNodeConnectorNew, String portName) {
185 this.nodeConnectorId = nodeConnectorId;
186 this.fcNodeConnectorNew = fcNodeConnectorNew;
187 this.interfaceName = portName;
188 this.idManager = idManager;
192 public Object call() {
193 List<ListenableFuture<Void>> futures = OvsInterfaceStateAddHelper.addState(dataBroker, idManager,
194 mdsalApiManager, alivenessMonitorService, nodeConnectorId, interfaceName, fcNodeConnectorNew);
195 List<InterfaceChildEntry> interfaceChildEntries = getInterfaceChildEntries(dataBroker, interfaceName);
196 for (InterfaceChildEntry interfaceChildEntry : interfaceChildEntries) {
197 InterfaceStateAddWorker interfaceStateAddWorker = new InterfaceStateAddWorker(idManager,
198 nodeConnectorId, fcNodeConnectorNew, interfaceChildEntry.getChildInterface());
199 DataStoreJobCoordinator.getInstance().enqueueJob(interfaceName, interfaceStateAddWorker);
205 public String toString() {
206 return "InterfaceStateAddWorker{" + "nodeConnectorId=" + nodeConnectorId + ", fcNodeConnectorNew="
207 + fcNodeConnectorNew + ", interfaceName='" + interfaceName + '\'' + '}';
211 private class InterfaceStateUpdateWorker implements Callable {
212 private final InstanceIdentifier<FlowCapableNodeConnector> key;
213 private final FlowCapableNodeConnector fcNodeConnectorOld;
214 private final FlowCapableNodeConnector fcNodeConnectorNew;
215 private final String interfaceName;
217 InterfaceStateUpdateWorker(InstanceIdentifier<FlowCapableNodeConnector> key,
218 FlowCapableNodeConnector fcNodeConnectorOld, FlowCapableNodeConnector fcNodeConnectorNew,
221 this.fcNodeConnectorOld = fcNodeConnectorOld;
222 this.fcNodeConnectorNew = fcNodeConnectorNew;
223 this.interfaceName = portName;
227 public Object call() {
228 List<ListenableFuture<Void>> futures = OvsInterfaceStateUpdateHelper.updateState(key,
229 alivenessMonitorService, dataBroker, interfaceName, fcNodeConnectorNew, fcNodeConnectorOld);
230 List<InterfaceChildEntry> interfaceChildEntries = getInterfaceChildEntries(dataBroker, interfaceName);
231 for (InterfaceChildEntry interfaceChildEntry : interfaceChildEntries) {
232 InterfaceStateUpdateWorker interfaceStateUpdateWorker = new InterfaceStateUpdateWorker(key,
233 fcNodeConnectorOld, fcNodeConnectorNew, interfaceChildEntry.getChildInterface());
234 DataStoreJobCoordinator.getInstance().enqueueJob(interfaceName, interfaceStateUpdateWorker);
240 public String toString() {
241 return "InterfaceStateUpdateWorker{" + "key=" + key + ", fcNodeConnectorOld=" + fcNodeConnectorOld
242 + ", fcNodeConnectorNew=" + fcNodeConnectorNew + ", interfaceName='" + interfaceName + '\'' + '}';
246 private class InterfaceStateRemoveWorker implements Callable {
247 private final NodeConnectorId nodeConnectorIdNew;
248 private NodeConnectorId nodeConnectorIdOld;
249 FlowCapableNodeConnector fcNodeConnectorOld;
250 private final String interfaceName;
251 private final String parentInterface;
252 private final IdManagerService idManager;
253 private final boolean isNodePresent;
254 private final boolean isNetworkEvent;
255 private final boolean isParentInterface;
257 InterfaceStateRemoveWorker(IdManagerService idManager, NodeConnectorId nodeConnectorIdNew,
258 NodeConnectorId nodeConnectorIdOld, FlowCapableNodeConnector fcNodeConnectorOld, String interfaceName,
259 String parentInterface, boolean isNodePresent, boolean isNetworkEvent, boolean isParentInterface) {
260 this.nodeConnectorIdNew = nodeConnectorIdNew;
261 this.nodeConnectorIdOld = nodeConnectorIdOld;
262 this.fcNodeConnectorOld = fcNodeConnectorOld;
263 this.interfaceName = interfaceName;
264 this.parentInterface = parentInterface;
265 this.idManager = idManager;
266 this.isNodePresent = isNodePresent;
267 this.isNetworkEvent = isNetworkEvent;
268 this.isParentInterface = isParentInterface;
272 public Object call() {
273 List<ListenableFuture<Void>> futures = null;
274 // VM Migration: Skip OFPPR_DELETE event received after OFPPR_ADD
275 // for same interface from Older DPN
276 if (isParentInterface && isNetworkEvent) {
277 nodeConnectorIdOld = IfmUtil.getNodeConnectorIdFromInterface(interfaceName, dataBroker);
278 if (nodeConnectorIdOld != null && !nodeConnectorIdNew.equals(nodeConnectorIdOld)) {
279 LOG.debug("Dropping the NodeConnector Remove Event for the interface: {}, {}, {}", interfaceName,
280 nodeConnectorIdNew, nodeConnectorIdOld);
285 futures = OvsInterfaceStateRemoveHelper.removeInterfaceStateConfiguration(idManager, mdsalApiManager,
286 alivenessMonitorService, nodeConnectorIdNew, nodeConnectorIdOld, dataBroker, interfaceName,
287 fcNodeConnectorOld, isNodePresent, parentInterface);
289 List<InterfaceChildEntry> interfaceChildEntries = getInterfaceChildEntries(dataBroker, interfaceName);
290 for (InterfaceChildEntry interfaceChildEntry : interfaceChildEntries) {
291 // Fetch all interfaces on this port and trigger remove worker
293 InterfaceStateRemoveWorker interfaceStateRemoveWorker = new InterfaceStateRemoveWorker(idManager,
294 nodeConnectorIdNew, nodeConnectorIdOld, fcNodeConnectorOld,
295 interfaceChildEntry.getChildInterface(), interfaceName, isNodePresent, isNetworkEvent, false);
296 DataStoreJobCoordinator.getInstance().enqueueJob(interfaceName, interfaceStateRemoveWorker);
302 public String toString() {
303 return "InterfaceStateRemoveWorker{" + "nodeConnectorIdNew=" + nodeConnectorIdNew + ", nodeConnectorIdOld="
304 + nodeConnectorIdOld + ", fcNodeConnectorOld=" + fcNodeConnectorOld + ", interfaceName='"
305 + interfaceName + '\'' + '}';
309 public static List<InterfaceChildEntry> getInterfaceChildEntries(DataBroker dataBroker, String interfaceName) {
310 InterfaceParentEntry interfaceParentEntry = InterfaceMetaUtils
311 .getInterfaceParentEntryFromConfigDS(interfaceName, dataBroker);
312 if (interfaceParentEntry != null && interfaceParentEntry.getInterfaceChildEntry() != null) {
313 return interfaceParentEntry.getInterfaceChildEntry();
315 return new ArrayList<>();