/* * Copyright (c) 2015, 2016 Ericsson India Global Services Pvt Ltd. and others. All rights reserved. * * This program and the accompanying materials are made available under the * terms of the Eclipse Public License v1.0 which accompanies this distribution, * and is available at http://www.eclipse.org/legal/epl-v10.html */ package org.opendaylight.vpnservice.itm.confighelpers; import java.math.BigInteger; import java.util.ArrayList; import java.util.List; import java.util.concurrent.Callable; import com.google.common.util.concurrent.FutureCallback; import com.google.common.util.concurrent.ListenableFuture; import org.opendaylight.vpnservice.mdsalutil.MDSALUtil; import org.opendaylight.vpnservice.itm.globals.ITMConstants; import org.opendaylight.vpnservice.itm.impl.ItmUtils; import org.opendaylight.controller.md.sal.binding.api.DataBroker; import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.inet.types.rev130715.IpAddress; import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.Interface; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.idmanager.rev150403.IdManagerService; import org.opendaylight.vpnservice.mdsalutil.interfaces.IMdsalApiManager; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.interfacemgr.rev150331.TunnelTypeBase; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.interfacemgr.rev150331.TunnelTypeGre; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.interfacemgr.rev150331.TunnelTypeVxlan; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.DpnEndpoints; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.DpnEndpointsBuilder; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.ExternalTunnelList; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.TunnelList; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.dpn.endpoints.DPNTEPsInfo; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.dpn.endpoints.dpn.teps.info.TunnelEndPoints; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.external.tunnel.list.ExternalTunnel; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.external.tunnel.list.ExternalTunnelBuilder; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.external.tunnel.list.ExternalTunnelKey; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.tunnel.list.InternalTunnel; import org.opendaylight.yang.gen.v1.urn.opendaylight.vpnservice.itm.op.rev150701.tunnel.list.InternalTunnelKey; import org.opendaylight.yangtools.yang.binding.InstanceIdentifier; import org.opendaylight.controller.md.sal.binding.api.WriteTransaction; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class ItmInternalTunnelAddWorker { private static final Logger logger = LoggerFactory.getLogger(ItmInternalTunnelAddWorker.class) ; private static final FutureCallback DEFAULT_CALLBACK = new FutureCallback() { public void onSuccess(Void result) { logger.debug("Success in Datastore operation"); } public void onFailure(Throwable error) { logger.error("Error in Datastore operation", error); }; }; public static List> build_all_tunnels(DataBroker dataBroker, IdManagerService idManagerService,IMdsalApiManager mdsalManager, List cfgdDpnList, List meshedDpnList) { logger.trace( "Building tunnels with DPN List {} " , cfgdDpnList ); List> futures = new ArrayList<>(); WriteTransaction t = dataBroker.newWriteOnlyTransaction(); if( null == cfgdDpnList || cfgdDpnList.isEmpty()) { logger.error(" Build Tunnels was invoked with empty list"); return futures; } for( DPNTEPsInfo dpn : cfgdDpnList) { //#####if dpn is not in meshedDpnList build_tunnel_from(dpn, meshedDpnList, dataBroker, idManagerService, mdsalManager, t, futures); if(null == meshedDpnList) { meshedDpnList = new ArrayList() ; } meshedDpnList.add(dpn) ; // Update the operational datastore -- FIXME -- Error Handling updateOperationalDatastore(dataBroker, dpn, t, futures) ; } futures.add( t.submit()) ; return futures ; } private static void updateOperationalDatastore(DataBroker dataBroker, DPNTEPsInfo dpn, WriteTransaction t, List> futures) { logger.debug("Updating CONFIGURATION datastore with DPN {} ", dpn); InstanceIdentifier dep = InstanceIdentifier.builder( DpnEndpoints.class).build() ; List dpnList = new ArrayList() ; dpnList.add(dpn) ; DpnEndpoints tnlBuilder = new DpnEndpointsBuilder().setDPNTEPsInfo(dpnList).build() ; t.merge(LogicalDatastoreType.CONFIGURATION, dep, tnlBuilder, true); } private static void build_tunnel_from( DPNTEPsInfo srcDpn,List meshedDpnList, DataBroker dataBroker, IdManagerService idManagerService, IMdsalApiManager mdsalManager, WriteTransaction t, List> futures) { logger.trace( "Building tunnels from DPN {} " , srcDpn ); if( null == meshedDpnList || 0 == meshedDpnList.size()) { logger.debug( "No DPN in the mesh "); return ; } for( DPNTEPsInfo dstDpn: meshedDpnList) { if ( ! srcDpn.equals(dstDpn) ) wireUpWithinTransportZone(srcDpn, dstDpn, dataBroker, idManagerService, mdsalManager, t, futures) ; } } private static void wireUpWithinTransportZone( DPNTEPsInfo srcDpn, DPNTEPsInfo dstDpn, DataBroker dataBroker, IdManagerService idManagerService, IMdsalApiManager mdsalManager,WriteTransaction t, List> futures) { logger.trace( "Wiring up within Transport Zone for Dpns {}, {} " , srcDpn, dstDpn ); List srcEndPts = srcDpn.getTunnelEndPoints(); List dstEndPts = dstDpn.getTunnelEndPoints(); for( TunnelEndPoints srcte : srcEndPts) { for( TunnelEndPoints dstte : dstEndPts ) { // Compare the Transport zones if (!srcDpn.getDPNID().equals(dstDpn.getDPNID())) { if( (srcte.getTransportZone().equals(dstte.getTransportZone()))) { // wire them up wireUpBidirectionalTunnel( srcte, dstte, srcDpn.getDPNID(), dstDpn.getDPNID(), dataBroker, idManagerService, mdsalManager, t, futures ); // CHECK THIS -- Assumption -- One end point per Dpn per transport zone break ; } } } } } private static void wireUpBidirectionalTunnel( TunnelEndPoints srcte, TunnelEndPoints dstte, BigInteger srcDpnId, BigInteger dstDpnId, DataBroker dataBroker, IdManagerService idManagerService, IMdsalApiManager mdsalManager, WriteTransaction t, List> futures) { // Setup the flow for LLDP monitoring -- PUNT TO CONTROLLER ItmUtils.setUpOrRemoveTerminatingServiceTable(srcDpnId, mdsalManager, true); ItmUtils.setUpOrRemoveTerminatingServiceTable(dstDpnId, mdsalManager, true); // Create the forward direction tunnel if(!wireUp( srcte, dstte, srcDpnId, dstDpnId, dataBroker, idManagerService, t, futures )) logger.error("Could not build tunnel between end points {}, {} " , srcte, dstte ); // CHECK IF FORWARD IS NOT BUILT , REVERSE CAN BE BUILT // Create the tunnel for the reverse direction if(! wireUp( dstte, srcte, dstDpnId, srcDpnId, dataBroker, idManagerService, t, futures )) logger.error("Could not build tunnel between end points {}, {} " , dstte, srcte); } private static boolean wireUp(TunnelEndPoints srcte, TunnelEndPoints dstte, BigInteger srcDpnId, BigInteger dstDpnId , DataBroker dataBroker, IdManagerService idManagerService, WriteTransaction t, List> futures) { // Wire Up logic logger.trace( "Wiring between source tunnel end points {}, destination tunnel end points {} " , srcte, dstte ); String interfaceName = srcte.getInterfaceName() ; Class tunType = srcte.getTunnelType(); String tunTypeStr = srcte.getTunnelType().getName(); // Form the trunk Interface Name String trunkInterfaceName = ItmUtils.getTrunkInterfaceName( idManagerService, interfaceName, srcte.getIpAddress().getIpv4Address().getValue(), dstte.getIpAddress().getIpv4Address().getValue(), tunTypeStr) ; IpAddress gatewayIpObj = new IpAddress("0.0.0.0".toCharArray()); IpAddress gwyIpAddress = ( srcte.getSubnetMask().equals(dstte.getSubnetMask()) ) ? gatewayIpObj : srcte.getGwIpAddress() ; logger.debug( " Creating Trunk Interface with parameters trunk I/f Name - {}, parent I/f name - {}, source IP - {}, destination IP - {} gateway IP - {}",trunkInterfaceName, interfaceName, srcte.getIpAddress(), dstte.getIpAddress(), gwyIpAddress ) ; Boolean monitorEnabled = ItmUtils.readMonitoringStateFromDS(dataBroker); Integer monitorInterval = ItmUtils.readMonitorIntervalfromDS(dataBroker); if(monitorInterval == null) monitorInterval = ITMConstants.DEFAULT_MONITOR_INTERVAL; Interface iface = ItmUtils.buildTunnelInterface(srcDpnId, trunkInterfaceName, String.format( "%s %s",ItmUtils.convertTunnelTypetoString(srcte.getTunnelType()), "Trunk Interface"), true, tunType, srcte.getIpAddress(), dstte.getIpAddress(), gwyIpAddress, srcte.getVLANID(), true, monitorEnabled, monitorInterval*1000); logger.debug( " Trunk Interface builder - {} ", iface ) ; InstanceIdentifier trunkIdentifier = ItmUtils.buildId(trunkInterfaceName); logger.debug( " Trunk Interface Identifier - {} ", trunkIdentifier ) ; logger.trace( " Writing Trunk Interface to Config DS {}, {} ", trunkIdentifier, iface ) ; t.merge(LogicalDatastoreType.CONFIGURATION, trunkIdentifier, iface, true); ItmUtils.itmCache.addInterface(iface); // also update itm-state ds? InstanceIdentifier path = InstanceIdentifier.create( TunnelList.class) .child(InternalTunnel.class, new InternalTunnelKey( dstDpnId, srcDpnId, tunType)); InternalTunnel tnl = ItmUtils.buildInternalTunnel(srcDpnId, dstDpnId, tunType, trunkInterfaceName); //ItmUtils.asyncUpdate(LogicalDatastoreType.CONFIGURATION, path, tnl, dataBroker, DEFAULT_CALLBACK); t.merge(LogicalDatastoreType.CONFIGURATION,path, tnl, true) ; ItmUtils.itmCache.addInternalTunnel(tnl); return true; } }