/* * Copyright (c) 2016, 2018 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.netvirt.aclservice.listeners; import static org.opendaylight.genius.infra.Datastore.CONFIGURATION; import java.util.Collections; import javax.annotation.PreDestroy; import javax.inject.Inject; import javax.inject.Singleton; import org.eclipse.jdt.annotation.Nullable; import org.opendaylight.genius.infra.ManagedNewTransactionRunner; import org.opendaylight.genius.infra.ManagedNewTransactionRunnerImpl; import org.opendaylight.genius.mdsalutil.MDSALUtil; import org.opendaylight.genius.mdsalutil.interfaces.IMdsalApiManager; import org.opendaylight.infrautils.jobcoordinator.JobCoordinator; import org.opendaylight.infrautils.utils.concurrent.Executors; import org.opendaylight.mdsal.binding.api.DataBroker; import org.opendaylight.mdsal.common.api.LogicalDatastoreType; import org.opendaylight.netvirt.aclservice.utils.AclConstants; import org.opendaylight.netvirt.aclservice.utils.AclNodeDefaultFlowsTxBuilder; import org.opendaylight.netvirt.aclservice.utils.AclServiceUtils; import org.opendaylight.serviceutils.srm.RecoverableListener; import org.opendaylight.serviceutils.srm.ServiceRecoveryRegistry; import org.opendaylight.serviceutils.tools.listener.AbstractAsyncDataTreeChangeListener; import org.opendaylight.yang.gen.v1.urn.opendaylight.flow.inventory.rev130819.FlowCapableNode; import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.Nodes; import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.nodes.Node; import org.opendaylight.yang.gen.v1.urn.opendaylight.inventory.rev130819.nodes.NodeKey; import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.aclservice.config.rev160806.AclserviceConfig; import org.opendaylight.yang.gen.v1.urn.opendaylight.netvirt.aclservice.config.rev160806.AclserviceConfig.SecurityGroupMode; import org.opendaylight.yangtools.yang.binding.InstanceIdentifier; import org.opendaylight.yangtools.yang.common.Uint64; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * Listener to handle flow capable node updates. Configures default ACL flows * during when node is discovered. */ @Singleton public class AclNodeListener extends AbstractAsyncDataTreeChangeListener implements RecoverableListener { private static final Logger LOG = LoggerFactory.getLogger(AclNodeListener.class); private final IMdsalApiManager mdsalManager; private final AclserviceConfig config; private final DataBroker dataBroker; private final ManagedNewTransactionRunner txRunner; private final AclServiceUtils aclServiceUtils; private final JobCoordinator jobCoordinator; @Nullable private SecurityGroupMode securityGroupMode = null; @Inject public AclNodeListener(final IMdsalApiManager mdsalManager, DataBroker dataBroker, AclserviceConfig config, AclServiceUtils aclServiceUtils, JobCoordinator jobCoordinator, ServiceRecoveryRegistry serviceRecoveryRegistry) { super(dataBroker, LogicalDatastoreType.OPERATIONAL, InstanceIdentifier.create(Nodes.class).child(Node.class) .augmentation(FlowCapableNode.class), Executors.newListeningSingleThreadExecutor("AclNodeListener", LOG)); this.mdsalManager = mdsalManager; this.dataBroker = dataBroker; this.txRunner = new ManagedNewTransactionRunnerImpl(dataBroker); this.config = config; this.aclServiceUtils = aclServiceUtils; this.jobCoordinator = jobCoordinator; serviceRecoveryRegistry.addRecoverableListener(AclServiceUtils.getRecoverServiceRegistryKey(), this); init(); } public void init() { LOG.info("{} start", getClass().getSimpleName()); if (config != null) { this.securityGroupMode = config.getSecurityGroupMode(); } LOG.info("AclserviceConfig: {}", this.config); } @Override public void registerListener() { super.register(); } @Override public void deregisterListener() { super.close(); } @Override public void remove(InstanceIdentifier key, FlowCapableNode dataObjectModification) { // do nothing } @Override public void update(InstanceIdentifier key, FlowCapableNode dataObjectModificationBefore, FlowCapableNode dataObjectModificationAfter) { // do nothing } @Override public void add(InstanceIdentifier key, FlowCapableNode dataObjectModification) { NodeKey nodeKey = key.firstKeyOf(Node.class); Uint64 dpId = MDSALUtil.getDpnIdFromNodeName(nodeKey.getId()); LOG.info("Received ACL node [{}] add event", dpId); if (securityGroupMode != null && securityGroupMode != SecurityGroupMode.Stateful) { LOG.error("Invalid security group mode ({}) obtained from AclserviceConfig. dpId={}", securityGroupMode, dpId); return; } jobCoordinator.enqueueJob(dpId.toString(), () -> Collections.singletonList(txRunner.callWithNewWriteOnlyTransactionAndSubmit(CONFIGURATION, tx -> { new AclNodeDefaultFlowsTxBuilder(dpId, mdsalManager, config, tx).build(); LOG.info("Adding default ACL flows for dpId={}", dpId); })), AclConstants.JOB_MAX_RETRIES); LOG.trace("FlowCapableNode (dpid: {}) add event is processed.", dpId); } @Override @PreDestroy public void close() { super.close(); Executors.shutdownAndAwaitTermination(getExecutorService()); } }