X-Git-Url: https://git.opendaylight.org/gerrit/gitweb?p=controller.git;a=blobdiff_plain;f=opendaylight%2Fmd-sal%2Fsal-distributed-datastore%2Fsrc%2Fmain%2Fjava%2Forg%2Fopendaylight%2Fcontroller%2Fcluster%2Fdatastore%2FDataChangeListenerSupport.java;fp=opendaylight%2Fmd-sal%2Fsal-distributed-datastore%2Fsrc%2Fmain%2Fjava%2Forg%2Fopendaylight%2Fcontroller%2Fcluster%2Fdatastore%2FDataChangeListenerSupport.java;h=7a033cf21fd2d1037c56e6125ae4637048deef71;hp=0000000000000000000000000000000000000000;hb=189586eeeeeeb8a9b6ed7398450d198f1864c307;hpb=66d39ecc3effd52c96c7a772a46612008e34fbc9 diff --git a/opendaylight/md-sal/sal-distributed-datastore/src/main/java/org/opendaylight/controller/cluster/datastore/DataChangeListenerSupport.java b/opendaylight/md-sal/sal-distributed-datastore/src/main/java/org/opendaylight/controller/cluster/datastore/DataChangeListenerSupport.java new file mode 100644 index 0000000000..7a033cf21f --- /dev/null +++ b/opendaylight/md-sal/sal-distributed-datastore/src/main/java/org/opendaylight/controller/cluster/datastore/DataChangeListenerSupport.java @@ -0,0 +1,101 @@ +/* + * Copyright (c) 2015 Cisco Systems, Inc. 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.controller.cluster.datastore; + +import com.google.common.base.Preconditions; +import java.util.ArrayList; +import java.util.List; +import akka.actor.ActorRef; +import akka.actor.ActorSelection; +import org.opendaylight.controller.cluster.datastore.messages.EnableNotification; +import org.opendaylight.controller.cluster.datastore.messages.RegisterChangeListener; +import org.opendaylight.controller.cluster.datastore.messages.RegisterChangeListenerReply; +import org.opendaylight.controller.md.sal.common.api.data.AsyncDataChangeListener; +import org.opendaylight.yangtools.concepts.ListenerRegistration; +import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier; +import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +final class DataChangeListenerSupport extends LeaderLocalDelegateFactory>>> { + private static final Logger LOG = LoggerFactory.getLogger(DataChangeListenerSupport.class); + private final List delayedListenerRegistrations = new ArrayList<>(); + private final List dataChangeListeners = new ArrayList<>(); + private final Shard shard; + + DataChangeListenerSupport(final Shard shard) { + this.shard = Preconditions.checkNotNull(shard); + } + + @Override + void onLeadershipChange(final boolean isLeader) { + for (ActorSelection dataChangeListener : dataChangeListeners) { + dataChangeListener.tell(new EnableNotification(isLeader), shard.getSelf()); + } + + if (isLeader) { + for (DelayedListenerRegistration reg: delayedListenerRegistrations) { + if(!reg.isClosed()) { + reg.setDelegate(createDelegate(reg.getRegisterChangeListener())); + } + } + + delayedListenerRegistrations.clear(); + } + } + + @Override + void onMessage(final RegisterChangeListener message, final boolean isLeader) { + + LOG.debug("{}: registerDataChangeListener for {}, leader: {}", shard.persistenceId(), message.getPath(), isLeader); + + ListenerRegistration>> registration; + if (isLeader) { + registration = createDelegate(message); + } else { + LOG.debug("{}: Shard is not the leader - delaying registration", shard.persistenceId()); + + DelayedListenerRegistration delayedReg = new DelayedListenerRegistration(message); + delayedListenerRegistrations.add(delayedReg); + registration = delayedReg; + } + + ActorRef listenerRegistration = shard.getContext().actorOf( + DataChangeListenerRegistration.props(registration)); + + LOG.debug("{}: registerDataChangeListener sending reply, listenerRegistrationPath = {} ", + shard.persistenceId(), listenerRegistration.path()); + + shard.getSender().tell(new RegisterChangeListenerReply(listenerRegistration), shard.getSelf()); + } + + @Override + ListenerRegistration>> createDelegate( + final RegisterChangeListener message) { + ActorSelection dataChangeListenerPath = shard.getContext().system().actorSelection( + message.getDataChangeListenerPath()); + + // Notify the listener if notifications should be enabled or not + // If this shard is the leader then it will enable notifications else + // it will not + dataChangeListenerPath.tell(new EnableNotification(true), shard.getSelf()); + + // Now store a reference to the data change listener so it can be notified + // at a later point if notifications should be enabled or disabled + dataChangeListeners.add(dataChangeListenerPath); + + AsyncDataChangeListener> listener = + new DataChangeListenerProxy(dataChangeListenerPath); + + LOG.debug("{}: Registering for path {}", shard.persistenceId(), message.getPath()); + + return shard.getDataStore().registerChangeListener(message.getPath(), listener, + message.getScope()); + } +}