Merge "Fix Akka exception: cannot use non-static local Creator to create actors"
[controller.git] / opendaylight / md-sal / sal-distributed-datastore / src / main / java / org / opendaylight / controller / cluster / datastore / DataChangeListener.java
1 /*
2  * Copyright (c) 2014 Cisco Systems, Inc. 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.controller.cluster.datastore;
10
11 import akka.actor.Props;
12 import akka.japi.Creator;
13 import com.google.common.base.Preconditions;
14 import org.opendaylight.controller.cluster.common.actor.AbstractUntypedActor;
15 import org.opendaylight.controller.cluster.datastore.messages.DataChanged;
16 import org.opendaylight.controller.cluster.datastore.messages.DataChangedReply;
17 import org.opendaylight.controller.cluster.datastore.messages.EnableNotification;
18 import org.opendaylight.controller.md.sal.common.api.data.AsyncDataChangeEvent;
19 import org.opendaylight.controller.md.sal.common.api.data.AsyncDataChangeListener;
20 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
21 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
22 import org.slf4j.Logger;
23 import org.slf4j.LoggerFactory;
24
25 public class DataChangeListener extends AbstractUntypedActor {
26     private static final Logger LOG = LoggerFactory.getLogger(DataChangeListener.class);
27
28     private final AsyncDataChangeListener<YangInstanceIdentifier, NormalizedNode<?, ?>> listener;
29     private boolean notificationsEnabled = false;
30
31     public DataChangeListener(AsyncDataChangeListener<YangInstanceIdentifier,
32                                                       NormalizedNode<?, ?>> listener) {
33         this.listener = Preconditions.checkNotNull(listener, "listener should not be null");
34     }
35
36     @Override
37     public void handleReceive(Object message) throws Exception {
38         if(message instanceof DataChanged){
39             dataChanged(message);
40         } else if(message instanceof EnableNotification){
41             enableNotification((EnableNotification) message);
42         }
43     }
44
45     private void enableNotification(EnableNotification message) {
46         notificationsEnabled = message.isEnabled();
47         LOG.debug("{} notifications for listener {}", (notificationsEnabled ? "Enabled" : "Disabled"),
48                 listener);
49     }
50
51     private void dataChanged(Object message) {
52
53         // Do nothing if notifications are not enabled
54         if(!notificationsEnabled) {
55             LOG.debug("Notifications not enabled for listener {} - dropping change notification",
56                     listener);
57             return;
58         }
59
60         DataChanged reply = (DataChanged) message;
61         AsyncDataChangeEvent<YangInstanceIdentifier, NormalizedNode<?, ?>> change = reply.getChange();
62
63         LOG.debug("Sending change notification {} to listener {}", change, listener);
64
65         try {
66             this.listener.onDataChanged(change);
67         } catch (RuntimeException e) {
68             LOG.error( String.format( "Error notifying listener %s", this.listener ), e );
69         }
70
71         // It seems the sender is never null but it doesn't hurt to check. If the caller passes in
72         // a null sender (ActorRef.noSender()), akka translates that to the deadLetters actor.
73         if(getSender() != null && !getContext().system().deadLetters().equals(getSender())) {
74             getSender().tell(new DataChangedReply(), getSelf());
75         }
76     }
77
78     public static Props props(final AsyncDataChangeListener<YangInstanceIdentifier,
79                                                             NormalizedNode<?, ?>> listener) {
80         return Props.create(new DataChangeListenerCreator(listener));
81     }
82
83     private static class DataChangeListenerCreator implements Creator<DataChangeListener> {
84         private static final long serialVersionUID = 1L;
85
86         final AsyncDataChangeListener<YangInstanceIdentifier, NormalizedNode<?, ?>> listener;
87
88         DataChangeListenerCreator(
89                 AsyncDataChangeListener<YangInstanceIdentifier, NormalizedNode<?, ?>> listener) {
90             this.listener = listener;
91         }
92
93         @Override
94         public DataChangeListener create() throws Exception {
95             return new DataChangeListener(listener);
96         }
97     }
98 }