Enhancing interface-manager logging
[genius.git] / interfacemanager / interfacemanager-impl / src / main / java / org / opendaylight / genius / interfacemanager / servicebindings / flowbased / state / helpers / FlowBasedEgressServicesStateBindHelper.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.genius.interfacemanager.servicebindings.flowbased.state.helpers;
9
10 import com.google.common.util.concurrent.ListenableFuture;
11 import java.math.BigInteger;
12 import java.util.ArrayList;
13 import java.util.Collections;
14 import java.util.List;
15 import org.opendaylight.controller.md.sal.binding.api.DataBroker;
16 import org.opendaylight.controller.md.sal.binding.api.WriteTransaction;
17 import org.opendaylight.genius.interfacemanager.IfmUtil;
18 import org.opendaylight.genius.interfacemanager.InterfacemgrProvider;
19 import org.opendaylight.genius.interfacemanager.commons.InterfaceManagerCommonUtils;
20 import org.opendaylight.genius.interfacemanager.servicebindings.flowbased.state.factory.FlowBasedServicesStateAddable;
21 import org.opendaylight.genius.interfacemanager.servicebindings.flowbased.utilities.FlowBasedServicesUtils;
22 import org.opendaylight.genius.mdsalutil.NwConstants;
23 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.iana._if.type.rev140508.L2vlan;
24 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.iana._if.type.rev140508.Tunnel;
25 import org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.state.Interface;
26 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.servicebinding.rev160406.ServiceModeEgress;
27 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.servicebinding.rev160406.service.bindings.ServicesInfo;
28 import org.opendaylight.yang.gen.v1.urn.opendaylight.genius.interfacemanager.servicebinding.rev160406.service.bindings.services.info.BoundServices;
29 import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.NodeConnectorId;
30 import org.slf4j.Logger;
31 import org.slf4j.LoggerFactory;
32
33 public class FlowBasedEgressServicesStateBindHelper implements FlowBasedServicesStateAddable{
34     private static final Logger LOG = LoggerFactory.getLogger(FlowBasedEgressServicesStateBindHelper.class);
35
36     private InterfacemgrProvider interfaceMgrProvider;
37     private static volatile FlowBasedServicesStateAddable flowBasedServicesStateAddable;
38
39     private FlowBasedEgressServicesStateBindHelper(InterfacemgrProvider interfaceMgrProvider) {
40         this.interfaceMgrProvider = interfaceMgrProvider;
41     }
42
43     public static void intitializeFlowBasedEgressServicesStateBindHelper(InterfacemgrProvider interfaceMgrProvider) {
44         if (flowBasedServicesStateAddable == null) {
45             synchronized (FlowBasedEgressServicesStateBindHelper.class) {
46                 if (flowBasedServicesStateAddable == null) {
47                     flowBasedServicesStateAddable = new FlowBasedEgressServicesStateBindHelper(interfaceMgrProvider);
48                 }
49             }
50         }
51     }
52
53     public static FlowBasedServicesStateAddable getFlowBasedEgressServicesStateBindHelper() {
54         if (flowBasedServicesStateAddable == null) {
55             LOG.error("OvsInterfaceConfigAdd Renderer is not initialized");
56         }
57         return flowBasedServicesStateAddable;
58     }
59
60     public List<ListenableFuture<Void>> bindServicesOnInterface(Interface ifaceState) {
61         List<ListenableFuture<Void>> futures = new ArrayList<>();
62         if(ifaceState.getType() == null) {
63             return futures;
64         }
65         LOG.debug("binding services on interface {}", ifaceState.getName());
66         DataBroker dataBroker = interfaceMgrProvider.getDataBroker();
67         ServicesInfo servicesInfo = FlowBasedServicesUtils.getServicesInfoForInterface(ifaceState.getName(), ServiceModeEgress.class, dataBroker);
68         if (servicesInfo == null) {
69             LOG.trace("service info is null for interface {}", ifaceState.getName());
70             return futures;
71         }
72
73         List<BoundServices> allServices = servicesInfo.getBoundServices();
74         if (allServices == null || allServices.isEmpty()) {
75             LOG.trace("bound services is empty for interface {}", ifaceState.getName());
76             return futures;
77         }
78
79         if (ifaceState.getType().isAssignableFrom(L2vlan.class)) {
80             return bindServiceOnVlan(allServices, ifaceState, dataBroker);
81         } else if (ifaceState.getType().isAssignableFrom(Tunnel.class)){
82             return bindServiceOnTunnel(allServices, ifaceState, dataBroker);
83         }
84         return futures;
85     }
86
87     private static List<ListenableFuture<Void>> bindServiceOnTunnel(
88             List<BoundServices> allServices, Interface ifState, DataBroker dataBroker) {
89         List<ListenableFuture<Void>> futures = new ArrayList<>();
90         // FIXME : not supported yet
91         return futures;
92     }
93
94     private static List<ListenableFuture<Void>> bindServiceOnVlan(
95             List<BoundServices> allServices,
96             Interface ifState, DataBroker dataBroker) {
97         LOG.info("bind all egress services for vlan port: {}", ifState.getName());
98         List<ListenableFuture<Void>> futures = new ArrayList<>();
99         NodeConnectorId nodeConnectorId = FlowBasedServicesUtils.getNodeConnectorIdFromInterface(ifState);
100         BigInteger dpId = IfmUtil.getDpnFromNodeConnectorId(nodeConnectorId);
101         WriteTransaction t = dataBroker.newWriteOnlyTransaction();
102         Collections.sort(allServices,
103                 (serviceInfo1, serviceInfo2) -> serviceInfo1.getServicePriority().compareTo(serviceInfo2.getServicePriority()));
104         BoundServices highestPriority = allServices.remove(0);
105         short nextServiceIndex = (short) (allServices.size() > 0 ? allServices.get(0).getServicePriority() : highestPriority.getServicePriority() + 1);
106         org.opendaylight.yang.gen.v1.urn.ietf.params.xml.ns.yang.ietf.interfaces.rev140508.interfaces.Interface iface = InterfaceManagerCommonUtils.getInterfaceFromConfigDS(ifState.getName(), dataBroker);
107         FlowBasedServicesUtils.installEgressDispatcherFlows(dpId, highestPriority, ifState.getName(), t, ifState.getIfIndex(), NwConstants.DEFAULT_SERVICE_INDEX, nextServiceIndex, iface);
108         BoundServices prev = null;
109         for (BoundServices boundService : allServices) {
110             if (prev!=null) {
111                 FlowBasedServicesUtils.installEgressDispatcherFlows(dpId, prev, ifState.getName(), t, ifState.getIfIndex(), prev.getServicePriority(), boundService.getServicePriority(), iface);
112             }
113             prev = boundService;
114         }
115         if (prev!=null) {
116             FlowBasedServicesUtils.installEgressDispatcherFlows(dpId, prev, ifState.getName(), t, ifState.getIfIndex(), prev.getServicePriority(), (short) (prev.getServicePriority()+1), iface);
117         }
118         futures.add(t.submit());
119         return futures;
120
121     }
122
123 }