39912e883418286046f0a20623f47ffd68233e0a
[netvirt.git] / vpnservice / elanmanager / elanmanager-impl / src / main / java / org / opendaylight / netvirt / elan / internal / ElanServiceProvider.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
9 package org.opendaylight.netvirt.elan.internal;
10
11 import com.google.common.base.Optional;
12 import java.math.BigInteger;
13 import java.util.ArrayList;
14 import java.util.Collection;
15 import java.util.Collections;
16 import java.util.HashSet;
17 import java.util.List;
18 import java.util.Map;
19 import java.util.Set;
20 import java.util.concurrent.Future;
21 import java.util.function.BiFunction;
22 import org.opendaylight.controller.md.sal.binding.api.DataBroker;
23 import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
24 import org.opendaylight.genius.interfacemanager.exceptions.InterfaceAlreadyExistsException;
25 import org.opendaylight.genius.interfacemanager.globals.IfmConstants;
26 import org.opendaylight.genius.interfacemanager.interfaces.IInterfaceManager;
27 import org.opendaylight.genius.mdsalutil.MDSALUtil;
28 import org.opendaylight.netvirt.elan.statusanddiag.ElanStatusMonitor;
29 import org.opendaylight.netvirt.elan.utils.ElanConstants;
30 import org.opendaylight.netvirt.elan.utils.ElanUtils;
31 import org.opendaylight.netvirt.elanmanager.api.IElanService;
32 import org.opendaylight.netvirt.elanmanager.exceptions.MacNotFoundException;
33 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.Interfaces;
34 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.Interface;
35 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.InterfaceKey;
36 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.yang.types.rev130715.PhysAddress;
37 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.idmanager.rev160406.CreateIdPoolInput;
38 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.idmanager.rev160406.CreateIdPoolInputBuilder;
39 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.idmanager.rev160406.IdManagerService;
40 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.rev160406.IfL2vlan;
41 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInstance;
42 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInstanceBuilder;
43 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInterface;
44 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInterface.EtreeInterfaceType;
45 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.etree.rev160614.EtreeInterfaceBuilder;
46 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.ElanInstances;
47 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.ElanInterfaces;
48 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan._interface.forwarding.entries.ElanInterfaceMac;
49 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.dpn.interfaces.elan.dpn.interfaces.list.DpnInterfaces;
50 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.instances.ElanInstance;
51 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.instances.ElanInstanceBuilder;
52 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.instances.ElanInstanceKey;
53 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.interfaces.ElanInterface;
54 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.interfaces.ElanInterfaceBuilder;
55 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.interfaces.ElanInterfaceKey;
56 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.elan.state.Elan;
57 import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.elan.rev150602.forwarding.entries.MacEntry;
58 import org.opendaylight.yang.gen.v1.urn.tbd.params.xml.ns.yang.network.topology.rev131021.network.topology.topology.Node;
59 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
60 import org.opendaylight.yangtools.yang.common.RpcResult;
61 import org.slf4j.Logger;
62 import org.slf4j.LoggerFactory;
63
64 public class ElanServiceProvider implements IElanService {
65
66     private static final Logger LOG = LoggerFactory.getLogger(ElanServiceProvider.class);
67
68     private final IdManagerService idManager;
69     private final IInterfaceManager interfaceManager;
70     private final ElanInstanceManager elanInstanceManager;
71     private final ElanBridgeManager bridgeMgr;
72     private final DataBroker broker;
73     private final ElanStatusMonitor elanStatusMonitor;
74     private static ElanUtils elanUtils;
75
76     private boolean generateIntBridgeMac = true;
77
78     public ElanServiceProvider(IdManagerService idManager, IInterfaceManager interfaceManager,
79                                ElanInstanceManager elanInstanceManager, ElanBridgeManager bridgeMgr,
80                                DataBroker dataBroker,
81                                ElanInterfaceManager elanInterfaceManager,
82                                ElanStatusMonitor elanStatusMonitor, ElanUtils elanUtils) {
83         this.idManager = idManager;
84         this.interfaceManager = interfaceManager;
85         this.elanInstanceManager = elanInstanceManager;
86         this.bridgeMgr = bridgeMgr;
87         this.broker = dataBroker;
88         this.elanStatusMonitor = elanStatusMonitor;
89         this.elanUtils = elanUtils;
90         elanInterfaceManager.setElanUtils(elanUtils);
91     }
92
93     public void init() {
94         LOG.info("Starting ElnaServiceProvider");
95         elanStatusMonitor.reportStatus("STARTING");
96         try {
97             createIdPool();
98
99             elanStatusMonitor.reportStatus("OPERATIONAL");
100         } catch (Exception e) {
101             LOG.error("Error initializing services", e);
102             elanStatusMonitor.reportStatus("ERROR");
103         }
104     }
105
106     private void createIdPool() {
107         CreateIdPoolInput createPool = new CreateIdPoolInputBuilder().setPoolName(ElanConstants.ELAN_ID_POOL_NAME)
108                 .setLow(ElanConstants.ELAN_ID_LOW_VALUE).setHigh(ElanConstants.ELAN_ID_HIGH_VALUE).build();
109         try {
110             Future<RpcResult<Void>> result = idManager.createIdPool(createPool);
111             if (result != null && result.get().isSuccessful()) {
112                 LOG.debug("ELAN Id Pool is created successfully");
113             }
114         } catch (Exception e) {
115             LOG.error("Failed to create ELAN Id pool {}", e);
116         }
117     }
118
119     @Override
120     public boolean createElanInstance(String elanInstanceName, long macTimeout, String description) {
121         ElanInstance existingElanInstance = elanInstanceManager.getElanInstanceByName(elanInstanceName);
122         boolean isSuccess = true;
123         if (existingElanInstance != null) {
124             if (compareWithExistingElanInstance(existingElanInstance, macTimeout, description)) {
125                 LOG.debug("Elan Instance is already present in the Operational DS {}", existingElanInstance);
126                 return true;
127             } else {
128                 ElanInstance updateElanInstance = new ElanInstanceBuilder().setElanInstanceName(elanInstanceName)
129                         .setDescription(description).setMacTimeout(macTimeout)
130                         .setKey(new ElanInstanceKey(elanInstanceName)).build();
131                 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
132                         ElanUtils.getElanInstanceConfigurationDataPath(elanInstanceName), updateElanInstance);
133                 LOG.debug("Updating the Elan Instance {} with MAC TIME-OUT %l and Description %s ",
134                         updateElanInstance, macTimeout, description);
135             }
136         } else {
137             ElanInstance elanInstance = new ElanInstanceBuilder().setElanInstanceName(elanInstanceName)
138                     .setMacTimeout(macTimeout).setDescription(description).setKey(new ElanInstanceKey(elanInstanceName))
139                     .build();
140             MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
141                     ElanUtils.getElanInstanceConfigurationDataPath(elanInstanceName), elanInstance);
142             LOG.debug("Creating the new Elan Instance {}", elanInstance);
143         }
144         return isSuccess;
145     }
146
147     @Override
148     public boolean createEtreeInstance(String elanInstanceName, long macTimeout, String description) {
149         ElanInstance existingElanInstance = elanInstanceManager.getElanInstanceByName(elanInstanceName);
150         boolean isSuccess = true;
151         if (existingElanInstance != null) {
152             if (compareWithExistingElanInstance(existingElanInstance, macTimeout, description)) {
153                 LOG.warn("Etree Instance is already present in the Operational DS {}", existingElanInstance);
154                 return true;
155             } else {
156                 EtreeInstance etreeInstance = new EtreeInstanceBuilder().build();
157                 ElanInstance updateElanInstance = new ElanInstanceBuilder().setElanInstanceName(elanInstanceName)
158                         .setDescription(description).setMacTimeout(macTimeout)
159                         .setKey(new ElanInstanceKey(elanInstanceName))
160                         .addAugmentation(EtreeInstance.class, etreeInstance).build();
161                 MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
162                         ElanUtils.getElanInstanceConfigurationDataPath(elanInstanceName), updateElanInstance);
163                 LOG.debug("Updating the Etree Instance {} with MAC TIME-OUT %l and Description %s ",
164                         updateElanInstance, macTimeout, description);
165             }
166         } else {
167             EtreeInstance etreeInstance = new EtreeInstanceBuilder().build();
168             ElanInstance elanInstance = new ElanInstanceBuilder().setElanInstanceName(elanInstanceName)
169                     .setMacTimeout(macTimeout).setDescription(description).setKey(new ElanInstanceKey(elanInstanceName))
170                     .addAugmentation(EtreeInstance.class, etreeInstance).build();
171             MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
172                     ElanUtils.getElanInstanceConfigurationDataPath(elanInstanceName), elanInstance);
173             LOG.debug("Creating the new Etree Instance {}", elanInstance);
174         }
175         return isSuccess;
176     }
177
178     @Override
179     public EtreeInterface getEtreeInterfaceByElanInterfaceName(String elanInterface) {
180         return ElanUtils.getEtreeInterfaceByElanInterfaceName(broker, elanInterface);
181     }
182
183     public static boolean compareWithExistingElanInstance(ElanInstance existingElanInstance, long macTimeOut,
184             String description) {
185         boolean isEqual = false;
186         if (existingElanInstance.getMacTimeout() == macTimeOut
187                 && existingElanInstance.getDescription().equals(description)) {
188             isEqual = true;
189         }
190         return isEqual;
191     }
192
193     @Override
194     public void updateElanInstance(String elanInstanceName, long newMacTimout, String newDescription) {
195         createElanInstance(elanInstanceName, newMacTimout, newDescription);
196     }
197
198     @Override
199     public boolean deleteEtreeInstance(String etreeInstanceName) {
200         return deleteElanInstance(etreeInstanceName);
201     }
202
203     @Override
204     public boolean deleteElanInstance(String elanInstanceName) {
205         boolean isSuccess = false;
206         ElanInstance existingElanInstance = elanInstanceManager.getElanInstanceByName(elanInstanceName);
207         if (existingElanInstance == null) {
208             LOG.debug("Elan Instance is not present {}", existingElanInstance);
209             return isSuccess;
210         }
211         LOG.debug("Deletion of the existing Elan Instance {}", existingElanInstance);
212         ElanUtils.delete(broker, LogicalDatastoreType.CONFIGURATION,
213                 ElanUtils.getElanInstanceConfigurationDataPath(elanInstanceName));
214         isSuccess = true;
215         return isSuccess;
216     }
217
218     @Override
219     public void addEtreeInterface(String etreeInstanceName, String interfaceName, EtreeInterfaceType interfaceType,
220             List<String> staticMacAddresses, String description) {
221         ElanInstance existingElanInstance = elanInstanceManager.getElanInstanceByName(etreeInstanceName);
222         if (existingElanInstance != null && existingElanInstance.getAugmentation(EtreeInstance.class) != null) {
223             EtreeInterface etreeInterface = new EtreeInterfaceBuilder().setEtreeInterfaceType(interfaceType).build();
224             ElanInterface elanInterface;
225             if (staticMacAddresses == null) {
226                 elanInterface = new ElanInterfaceBuilder().setElanInstanceName(etreeInstanceName)
227                         .setDescription(description).setName(interfaceName).setKey(new ElanInterfaceKey(interfaceName))
228                         .addAugmentation(EtreeInterface.class, etreeInterface).build();
229             } else {
230                 elanInterface = new ElanInterfaceBuilder().setElanInstanceName(etreeInstanceName)
231                         .setDescription(description).setName(interfaceName)
232                         .setStaticMacEntries(getPhysAddress(staticMacAddresses))
233                         .setKey(new ElanInterfaceKey(interfaceName))
234                         .addAugmentation(EtreeInterface.class, etreeInterface).build();
235             }
236             MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
237                     ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName), elanInterface);
238             LOG.debug("Creating the new Etree Interface {}", elanInterface);
239         }
240     }
241
242     @Override
243     public void addElanInterface(String elanInstanceName, String interfaceName, List<String> staticMacAddresses,
244             String description) {
245         ElanInstance existingElanInstance = elanInstanceManager.getElanInstanceByName(elanInstanceName);
246         if (existingElanInstance != null) {
247             ElanInterface elanInterface;
248             if (staticMacAddresses == null) {
249                 elanInterface = new ElanInterfaceBuilder().setElanInstanceName(elanInstanceName)
250                         .setDescription(description).setName(interfaceName).setKey(new ElanInterfaceKey(interfaceName))
251                         .build();
252             } else {
253                 elanInterface = new ElanInterfaceBuilder().setElanInstanceName(elanInstanceName)
254                         .setDescription(description).setName(interfaceName)
255                         .setStaticMacEntries(getPhysAddress(staticMacAddresses))
256                         .setKey(new ElanInterfaceKey(interfaceName)).build();
257             }
258             MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
259                     ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName), elanInterface);
260             LOG.debug("Creating the new ELan Interface {}", elanInterface);
261         }
262     }
263
264     @Override
265     public void updateElanInterface(String elanInstanceName, String interfaceName,
266             List<String> updatedStaticMacAddresses, String newDescription) {
267         ElanInterface existingElanInterface = ElanUtils.getElanInterfaceByElanInterfaceName(broker, interfaceName);
268         if (existingElanInterface == null) {
269             return;
270         }
271         List<PhysAddress> existingMacAddress = existingElanInterface.getStaticMacEntries();
272         List<PhysAddress> updatedMacAddresses = getPhysAddress(updatedStaticMacAddresses);
273         List<PhysAddress> updatedPhysAddress = getUpdatedPhyAddress(existingMacAddress, updatedMacAddresses);
274         if (updatedPhysAddress.size() > 0) {
275             LOG.debug("updating the ElanInterface with new Mac Entries {}", updatedStaticMacAddresses);
276             ElanInterface elanInterface = new ElanInterfaceBuilder().setElanInstanceName(elanInstanceName)
277                     .setName(interfaceName).setDescription(newDescription).setStaticMacEntries(updatedPhysAddress)
278                     .setKey(new ElanInterfaceKey(interfaceName)).build();
279             MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
280                     ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName), elanInterface);
281         }
282     }
283
284     @Override
285     public void deleteEtreeInterface(String elanInstanceName, String interfaceName) {
286         deleteElanInterface(elanInstanceName, interfaceName);
287         LOG.debug("deleting the Etree Interface {}", interfaceName);
288     }
289
290     @Override
291     public void deleteElanInterface(String elanInstanceName, String interfaceName) {
292         ElanInterface existingElanInterface = ElanUtils.getElanInterfaceByElanInterfaceName(broker, interfaceName);
293         if (existingElanInterface != null) {
294             ElanUtils.delete(broker, LogicalDatastoreType.CONFIGURATION,
295                     ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName));
296             LOG.debug("deleting the Elan Interface {}", existingElanInterface);
297         }
298     }
299
300     @Override
301     public void addStaticMacAddress(String elanInstanceName, String interfaceName, String macAddress) {
302         ElanInterface existingElanInterface = ElanUtils.getElanInterfaceByElanInterfaceName(broker, interfaceName);
303         PhysAddress updateStaticMacAddress = new PhysAddress(macAddress);
304         if (existingElanInterface != null) {
305             List<PhysAddress> existingMacAddress = existingElanInterface.getStaticMacEntries();
306             if (existingMacAddress.contains(updateStaticMacAddress)) {
307                 return;
308             }
309             existingMacAddress.add(updateStaticMacAddress);
310             ElanInterface elanInterface = new ElanInterfaceBuilder().setElanInstanceName(elanInstanceName)
311                     .setName(interfaceName).setStaticMacEntries(existingMacAddress)
312                     .setDescription(existingElanInterface.getDescription()).setKey(new ElanInterfaceKey(interfaceName))
313                     .build();
314             MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
315                     ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName), elanInterface);
316         }
317     }
318
319     @Override
320     public void deleteStaticMacAddress(String elanInstanceName, String interfaceName, String macAddress)
321             throws MacNotFoundException {
322         ElanInterface existingElanInterface = ElanUtils.getElanInterfaceByElanInterfaceName(broker, interfaceName);
323         PhysAddress physAddress = new PhysAddress(macAddress);
324         if (existingElanInterface == null) {
325             return;
326         }
327         List<PhysAddress> existingMacAddress = existingElanInterface.getStaticMacEntries();
328         if (existingMacAddress.contains(physAddress)) {
329             existingMacAddress.remove(physAddress);
330             ElanInterface elanInterface = new ElanInterfaceBuilder().setElanInstanceName(elanInstanceName)
331                     .setName(interfaceName).setStaticMacEntries(existingMacAddress)
332                     .setDescription(existingElanInterface.getDescription()).setKey(new ElanInterfaceKey(interfaceName))
333                     .build();
334             MDSALUtil.syncWrite(broker, LogicalDatastoreType.CONFIGURATION,
335                     ElanUtils.getElanInterfaceConfigurationDataPathId(interfaceName), elanInterface);
336         } else {
337             throw new MacNotFoundException("deleteStaticMacAddress did not find MAC: " + macAddress);
338         }
339     }
340
341     @Override
342     public Collection<MacEntry> getElanMacTable(String elanInstanceName) {
343         Elan elanInfo = ElanUtils.getElanByName(broker, elanInstanceName);
344         List<MacEntry> macAddress = new ArrayList<>();
345         if (elanInfo == null) {
346             return macAddress;
347         }
348         List<String> elanInterfaces = elanInfo.getElanInterfaces();
349         if (elanInterfaces != null && elanInterfaces.size() > 0) {
350             for (String elanInterface : elanInterfaces) {
351                 ElanInterfaceMac elanInterfaceMac = elanUtils.getElanInterfaceMacByInterfaceName(elanInterface);
352                 if (elanInterfaceMac != null && elanInterfaceMac.getMacEntry() != null
353                         && elanInterfaceMac.getMacEntry().size() > 0) {
354                     macAddress.addAll(elanInterfaceMac.getMacEntry());
355                 }
356             }
357         }
358         return macAddress;
359     }
360
361     @Override
362     public void flushMACTable(String elanInstanceName) {
363         Elan elanInfo = ElanUtils.getElanByName(broker, elanInstanceName);
364         if (elanInfo == null) {
365             return;
366         }
367         List<String> elanInterfaces = elanInfo.getElanInterfaces();
368         if (elanInterfaces == null || elanInterfaces.isEmpty()) {
369             return;
370         }
371         for (String elanInterface : elanInterfaces) {
372             ElanInterfaceMac elanInterfaceMac = elanUtils.getElanInterfaceMacByInterfaceName(elanInterface);
373             if (elanInterfaceMac.getMacEntry() != null && elanInterfaceMac.getMacEntry().size() > 0) {
374                 List<MacEntry> macEntries = elanInterfaceMac.getMacEntry();
375                 for (MacEntry macEntry : macEntries) {
376                     try {
377                         deleteStaticMacAddress(elanInstanceName, elanInterface, macEntry.getMacAddress().getValue());
378                     } catch (MacNotFoundException e) {
379                         LOG.error("Mac Not Found Exception {}", e);
380                     }
381                 }
382             }
383         }
384
385     }
386
387     public static List<PhysAddress> getPhysAddress(List<String> macAddress) {
388         List<PhysAddress> physAddresses = new ArrayList<>();
389         for (String mac : macAddress) {
390             physAddresses.add(new PhysAddress(mac));
391         }
392         return physAddresses;
393     }
394
395     public List<PhysAddress> getUpdatedPhyAddress(List<PhysAddress> originalAddresses,
396             List<PhysAddress> updatePhyAddresses) {
397         if (updatePhyAddresses != null && !updatePhyAddresses.isEmpty()) {
398             List<PhysAddress> existingClonedPhyAddress = new ArrayList<>();
399             if (originalAddresses != null && !originalAddresses.isEmpty()) {
400                 existingClonedPhyAddress.addAll(0, originalAddresses);
401                 originalAddresses.removeAll(updatePhyAddresses);
402                 updatePhyAddresses.removeAll(existingClonedPhyAddress);
403             }
404         }
405         return updatePhyAddresses;
406     }
407
408     @Override
409     public ElanInstance getElanInstance(String elanName) {
410         return ElanUtils.getElanInstanceByName(broker, elanName);
411     }
412
413     @Override
414     public List<ElanInstance> getElanInstances() {
415         List<ElanInstance> elanList = new ArrayList<>();
416         InstanceIdentifier<ElanInstances> elanInstancesIdentifier = InstanceIdentifier.builder(ElanInstances.class)
417                 .build();
418         Optional<ElanInstances> elansOptional = elanUtils.read(broker, LogicalDatastoreType.CONFIGURATION,
419                 elanInstancesIdentifier);
420         if (elansOptional.isPresent()) {
421             elanList.addAll(elansOptional.get().getElanInstance());
422         }
423         return elanList;
424     }
425
426     @Override
427     public List<String> getElanInterfaces(String elanInstanceName) {
428         List<String> elanInterfaces = new ArrayList<>();
429         InstanceIdentifier<ElanInterfaces> elanInterfacesIdentifier = InstanceIdentifier.builder(ElanInterfaces.class)
430                 .build();
431         Optional<ElanInterfaces> elanInterfacesOptional = elanUtils.read(broker, LogicalDatastoreType.CONFIGURATION,
432                 elanInterfacesIdentifier);
433         if (!elanInterfacesOptional.isPresent()) {
434             return elanInterfaces;
435         }
436         List<ElanInterface> elanInterfaceList = elanInterfacesOptional.get().getElanInterface();
437         for (ElanInterface elanInterface : elanInterfaceList) {
438             if (elanInterface.getElanInstanceName().equals(elanInstanceName)) {
439                 elanInterfaces.add(elanInterface.getName());
440             }
441         }
442         return elanInterfaces;
443     }
444
445     public boolean getGenerateIntBridgeMac() {
446         return generateIntBridgeMac;
447     }
448
449     public void setGenerateIntBridgeMac(boolean generateIntBridgeMac) {
450         this.generateIntBridgeMac = generateIntBridgeMac;
451     }
452
453     @Override
454     public void createExternalElanNetworks(Node node) {
455         handleExternalElanNetworks(node, (elanInstance, interfaceName) -> {
456             createExternalElanNetwork(elanInstance, interfaceName);
457             return null;
458         });
459     }
460
461     @Override
462     public void createExternalElanNetwork(ElanInstance elanInstance) {
463         handleExternalElanNetwork(elanInstance, (elanInstance1, interfaceName) -> {
464             createExternalElanNetwork(elanInstance1, interfaceName);
465             return null;
466         });
467     }
468
469     private void createExternalElanNetwork(ElanInstance elanInstance, String interfaceName) {
470         if (interfaceName == null) {
471             LOG.trace("No physial interface is attached to {}", elanInstance.getPhysicalNetworkName());
472             return;
473         }
474
475         String elanInterfaceName = createIetfInterfaces(elanInstance, interfaceName);
476         addElanInterface(elanInstance.getElanInstanceName(), elanInterfaceName, null, null);
477     }
478
479     @Override
480     public void deleteExternalElanNetworks(Node node) {
481         handleExternalElanNetworks(node, (elanInstance, interfaceName) -> {
482             deleteExternalElanNetwork(elanInstance, interfaceName);
483             return null;
484         });
485     }
486
487     @Override
488     public void deleteExternalElanNetwork(ElanInstance elanInstance) {
489         handleExternalElanNetwork(elanInstance, (elanInstance1, interfaceName) -> {
490             deleteExternalElanNetwork(elanInstance1, interfaceName);
491             return null;
492         });
493     }
494
495     private void deleteExternalElanNetwork(ElanInstance elanInstance, String interfaceName) {
496         if (interfaceName == null) {
497             LOG.trace("No physial interface is attached to {}", elanInstance.getPhysicalNetworkName());
498             return;
499         }
500
501         String elanInstanceName = elanInstance.getElanInstanceName();
502         for (String elanInterface : getExternalElanInterfaces(elanInstanceName)) {
503             if (elanInterface.startsWith(interfaceName)) {
504                 deleteIetfInterface(elanInterface);
505                 deleteElanInterface(elanInstanceName, elanInterface);
506             }
507         }
508     }
509
510     @Override
511     public void updateExternalElanNetworks(Node origNode, Node updatedNode) {
512         if (!bridgeMgr.isIntegrationBridge(updatedNode)) {
513             return;
514         }
515
516         List<ElanInstance> elanInstances = getElanInstances();
517         if (elanInstances == null || elanInstances.isEmpty()) {
518             LOG.trace("No ELAN instances found");
519             return;
520         }
521
522         Optional<Map<String, String>> origProviderMapOpt = bridgeMgr.getOpenvswitchOtherConfigMap(origNode,
523                 ElanBridgeManager.PROVIDER_MAPPINGS_KEY);
524         Optional<Map<String, String>> updatedProviderMapOpt = bridgeMgr.getOpenvswitchOtherConfigMap(updatedNode,
525                 ElanBridgeManager.PROVIDER_MAPPINGS_KEY);
526         Map<String, String> origProviderMappping = origProviderMapOpt.or(Collections.emptyMap());
527         Map<String, String> updatedProviderMappping = updatedProviderMapOpt.or(Collections.emptyMap());
528
529         for (ElanInstance elanInstance : elanInstances) {
530             String physicalNetworkName = elanInstance.getPhysicalNetworkName();
531             if (physicalNetworkName != null) {
532                 String origPortName = origProviderMappping.get(physicalNetworkName);
533                 String updatedPortName = updatedProviderMappping.get(physicalNetworkName);
534                 if (origPortName != null && !origPortName.equals(updatedPortName)) {
535                     deleteExternalElanNetwork(elanInstance, getExtInterfaceName(origNode, physicalNetworkName));
536                 }
537                 if (updatedPortName != null && !updatedPortName.equals(origPortName)) {
538                     createExternalElanNetwork(elanInstance, getExtInterfaceName(updatedNode, updatedPortName));
539                 }
540             }
541         }
542     }
543
544     @Override
545     public String getExternalElanInterface(String elanInstanceName, BigInteger dpnId) {
546         DpnInterfaces dpnInterfaces = elanUtils.getElanInterfaceInfoByElanDpn(elanInstanceName, dpnId);
547         if (dpnInterfaces == null || dpnInterfaces.getInterfaces() == null) {
548             LOG.trace("Elan {} does not have interfaces in DPN {}", elanInstanceName, dpnId);
549             return null;
550         }
551
552         for (String dpnInterface : dpnInterfaces.getInterfaces()) {
553             if (elanUtils.isExternal(dpnInterface)) {
554                 return dpnInterface;
555             }
556         }
557
558         LOG.trace("Elan {} does not have any external interace attached to DPN {}", elanInstanceName, dpnId);
559         return null;
560     }
561
562     @Override
563     public Collection<String> getExternalElanInterfaces(String elanInstanceName) {
564         List<String> elanInterfaces = getElanInterfaces(elanInstanceName);
565         if (elanInterfaces == null || elanInterfaces.isEmpty()) {
566             LOG.trace("No ELAN interfaces defined for {}", elanInstanceName);
567             return Collections.emptySet();
568         }
569
570         Set<String> externalElanInterfaces = new HashSet<>();
571         for (String elanInterface : elanInterfaces) {
572             if (elanUtils.isExternal(elanInterface)) {
573                 externalElanInterfaces.add(elanInterface);
574             }
575         }
576
577         return externalElanInterfaces;
578     }
579
580     @Override
581     public boolean isExternalInterface(String interfaceName) {
582         return elanUtils.isExternal(interfaceName);
583     }
584
585     @Override
586     public ElanInterface getElanInterfaceByElanInterfaceName(String interfaceName) {
587         return ElanUtils.getElanInterfaceByElanInterfaceName(broker, interfaceName);
588     }
589
590     /**
591      * Create ietf-interfaces based on the ELAN segment type.<br>
592      * For segment type flat - create transparent interface pointing to the
593      * patch-port attached to the physnet port.<br>
594      * For segment type vlan - create trunk interface pointing to the patch-port
595      * attached to the physnet port + trunk-member interface pointing to the
596      * trunk interface.
597      *
598      * @param elanInstance
599      *            ELAN instance
600      * @param parentRef
601      *            parent interface name
602      * @return the name of the interface to be added to the ELAN instance i.e.
603      *         trunk-member name for vlan network and transparent for flat
604      *         network or null otherwise
605      */
606     private String createIetfInterfaces(ElanInstance elanInstance, String parentRef) {
607         String interfaceName = null;
608
609         try {
610             if (ElanUtils.isFlat(elanInstance)) {
611                 interfaceName = parentRef + IfmConstants.OF_URI_SEPARATOR + "flat";
612                 interfaceManager.createVLANInterface(interfaceName, parentRef, null, null, null,
613                         IfL2vlan.L2vlanMode.Transparent, true);
614             } else if (ElanUtils.isVlan(elanInstance)) {
615                 Long segmentationId = elanInstance.getSegmentationId();
616                 interfaceName = parentRef + IfmConstants.OF_URI_SEPARATOR + segmentationId;
617                 String trunkName = parentRef + IfmConstants.OF_URI_SEPARATOR + "trunk";
618                 // trunk interface may have been created by other vlan network
619                 Interface trunkInterface = ElanUtils.getInterfaceFromConfigDS(trunkName, broker);
620                 if (trunkInterface == null) {
621                     interfaceManager.createVLANInterface(trunkName, parentRef, null, null, null,
622                             IfL2vlan.L2vlanMode.Trunk, true);
623                 }
624                 interfaceManager.createVLANInterface(interfaceName, trunkName, null, segmentationId.intValue(), null,
625                         IfL2vlan.L2vlanMode.TrunkMember, true);
626             }
627         } catch (InterfaceAlreadyExistsException e) {
628             LOG.trace("Interface {} was already created", interfaceName);
629         }
630
631         return interfaceName;
632     }
633
634     private void deleteIetfInterface(String interfaceName) {
635         InterfaceKey interfaceKey = new InterfaceKey(interfaceName);
636         InstanceIdentifier<Interface> interfaceInstanceIdentifier = InstanceIdentifier.builder(Interfaces.class)
637                 .child(Interface.class, interfaceKey).build();
638         MDSALUtil.syncDelete(broker, LogicalDatastoreType.CONFIGURATION, interfaceInstanceIdentifier);
639         LOG.debug("Deleting IETF interface {}", interfaceName);
640     }
641
642     private String getExtInterfaceName(Node node, String physicalNetworkName) {
643         if (physicalNetworkName == null) {
644             return null;
645         }
646
647         String providerMappingValue = bridgeMgr.getProviderMappingValue(node, physicalNetworkName);
648         if (providerMappingValue == null) {
649             LOG.trace("No provider mapping found for physicalNetworkName {} node {}", physicalNetworkName,
650                     node.getNodeId().getValue());
651             return null;
652         }
653
654         return bridgeMgr.southboundUtils.getDataPathId(node) + IfmConstants.OF_URI_SEPARATOR
655                 + bridgeMgr.getIntBridgePortNameFor(node, providerMappingValue);
656     }
657
658     private void handleExternalElanNetworks(Node node, BiFunction<ElanInstance, String, Void> function) {
659         if (!bridgeMgr.isIntegrationBridge(node)) {
660             return;
661         }
662
663         List<ElanInstance> elanInstances = getElanInstances();
664         if (elanInstances == null || elanInstances.isEmpty()) {
665             LOG.trace("No ELAN instances found");
666             return;
667         }
668
669         for (ElanInstance elanInstance : elanInstances) {
670             String interfaceName = getExtInterfaceName(node, elanInstance.getPhysicalNetworkName());
671             if (interfaceName != null) {
672                 function.apply(elanInstance, interfaceName);
673             }
674         }
675     }
676
677     private void handleExternalElanNetwork(ElanInstance elanInstance, BiFunction<ElanInstance, String, Void> function) {
678         String elanInstanceName = elanInstance.getElanInstanceName();
679         if (elanInstance.getPhysicalNetworkName() == null) {
680             LOG.trace("No physical network attached to {}", elanInstanceName);
681             return;
682         }
683
684         List<Node> nodes = bridgeMgr.southboundUtils.getOvsdbNodes();
685         if (nodes == null || nodes.isEmpty()) {
686             LOG.trace("No OVS nodes found while creating external network for ELAN {}",
687                     elanInstance.getElanInstanceName());
688             return;
689         }
690
691         for (Node node : nodes) {
692             if (bridgeMgr.isIntegrationBridge(node)) {
693                 String interfaceName = getExtInterfaceName(node, elanInstance.getPhysicalNetworkName());
694                 function.apply(elanInstance, interfaceName);
695             }
696         }
697     }
698
699 }