2 * Copyright (c) 2014 Cisco Systems, Inc. and others. All rights reserved.
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
8 package org.opendaylight.mdsal.binding.dom.adapter;
10 import org.opendaylight.mdsal.dom.api.ClusteredDOMDataChangeListener;
11 import org.opendaylight.mdsal.dom.api.DOMDataBroker;
12 import org.opendaylight.mdsal.dom.api.DOMDataChangeListener;
14 import org.opendaylight.mdsal.binding.api.ClusteredDataChangeListener;
15 import org.opendaylight.mdsal.binding.api.DataChangeListener;
16 import com.google.common.base.MoreObjects;
17 import com.google.common.base.Optional;
18 import java.util.Collections;
19 import java.util.HashMap;
20 import java.util.HashSet;
22 import java.util.Map.Entry;
24 import org.opendaylight.controller.md.sal.common.api.data.AsyncDataBroker.DataChangeScope;
25 import org.opendaylight.controller.md.sal.common.api.data.AsyncDataChangeEvent;
26 import org.opendaylight.controller.md.sal.common.api.data.LogicalDatastoreType;
27 import org.opendaylight.yangtools.concepts.AbstractListenerRegistration;
28 import org.opendaylight.yangtools.concepts.Delegator;
29 import org.opendaylight.yangtools.concepts.ListenerRegistration;
30 import org.opendaylight.yangtools.yang.binding.DataObject;
31 import org.opendaylight.yangtools.yang.binding.InstanceIdentifier;
32 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
33 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
34 import org.opendaylight.yangtools.yang.data.impl.codec.DeserializationException;
35 import org.slf4j.Logger;
36 import org.slf4j.LoggerFactory;
38 public abstract class AbstractForwardedDataBroker implements Delegator<DOMDataBroker>, AutoCloseable {
40 private static final Logger LOG = LoggerFactory.getLogger(AbstractForwardedDataBroker.class);
41 // The Broker to whom we do all forwarding
42 private final DOMDataBroker domDataBroker;
44 private final BindingToNormalizedNodeCodec codec;
46 protected AbstractForwardedDataBroker(final DOMDataBroker domDataBroker, final BindingToNormalizedNodeCodec codec) {
47 this.domDataBroker = domDataBroker;
51 protected BindingToNormalizedNodeCodec getCodec() {
56 public DOMDataBroker getDelegate() {
60 public ListenerRegistration<DataChangeListener> registerDataChangeListener(final LogicalDatastoreType store,
61 final InstanceIdentifier<?> path, final DataChangeListener listener, final DataChangeScope triggeringScope) {
62 final DOMDataChangeListener domDataChangeListener;
64 if(listener instanceof ClusteredDataChangeListener) {
65 domDataChangeListener = new TranslatingClusteredDataChangeInvoker(store, path, listener, triggeringScope);
67 domDataChangeListener = new TranslatingDataChangeInvoker(store, path, listener,
70 final YangInstanceIdentifier domPath = codec.toYangInstanceIdentifierBlocking(path);
71 final ListenerRegistration<DOMDataChangeListener> domRegistration = domDataBroker.registerDataChangeListener(store,
72 domPath, domDataChangeListener, triggeringScope);
73 return new ListenerRegistrationImpl(listener, domRegistration);
76 protected Map<InstanceIdentifier<?>, DataObject> toBinding(final InstanceIdentifier<?> path,
77 final Map<YangInstanceIdentifier, ? extends NormalizedNode<?, ?>> normalized) {
78 final Map<InstanceIdentifier<?>, DataObject> newMap = new HashMap<>();
80 for (final Map.Entry<YangInstanceIdentifier, ? extends NormalizedNode<?, ?>> entry : normalized.entrySet()) {
82 final Optional<Entry<InstanceIdentifier<? extends DataObject>, DataObject>> potential = getCodec().toBinding(entry);
83 if (potential.isPresent()) {
84 final Entry<InstanceIdentifier<? extends DataObject>, DataObject> binding = potential.get();
85 newMap.put(binding.getKey(), binding.getValue());
87 } catch (final DeserializationException e) {
88 LOG.warn("Failed to transform {}, omitting it", entry, e);
94 protected Set<InstanceIdentifier<?>> toBinding(final InstanceIdentifier<?> path,
95 final Set<YangInstanceIdentifier> normalized) {
96 final Set<InstanceIdentifier<?>> hashSet = new HashSet<>();
97 for (final YangInstanceIdentifier normalizedPath : normalized) {
99 final Optional<InstanceIdentifier<? extends DataObject>> potential = getCodec().toBinding(normalizedPath);
100 if (potential.isPresent()) {
101 final InstanceIdentifier<? extends DataObject> binding = potential.get();
102 hashSet.add(binding);
103 } else if (normalizedPath.getLastPathArgument() instanceof YangInstanceIdentifier.AugmentationIdentifier) {
106 } catch (final DeserializationException e) {
107 LOG.warn("Failed to transform {}, omitting it", normalizedPath, e);
113 protected Optional<DataObject> toBindingData(final InstanceIdentifier<?> path, final NormalizedNode<?, ?> data) {
114 if (path.isWildcarded()) {
115 return Optional.absent();
117 return (Optional<DataObject>) getCodec().deserializeFunction(path).apply(Optional.<NormalizedNode<?, ?>> of(data));
120 private class TranslatingDataChangeInvoker implements DOMDataChangeListener {
121 private final DataChangeListener bindingDataChangeListener;
122 private final LogicalDatastoreType store;
123 private final InstanceIdentifier<?> path;
124 private final DataChangeScope triggeringScope;
126 public TranslatingDataChangeInvoker(final LogicalDatastoreType store, final InstanceIdentifier<?> path,
127 final DataChangeListener bindingDataChangeListener, final DataChangeScope triggeringScope) {
130 this.bindingDataChangeListener = bindingDataChangeListener;
131 this.triggeringScope = triggeringScope;
135 public void onDataChanged(final AsyncDataChangeEvent<YangInstanceIdentifier, NormalizedNode<?, ?>> change) {
136 bindingDataChangeListener.onDataChanged(new TranslatedDataChangeEvent(change, path));
140 public String toString() {
141 return bindingDataChangeListener.getClass().getName();
146 * Translator for ClusteredDataChangeListener
149 private class TranslatingClusteredDataChangeInvoker extends TranslatingDataChangeInvoker implements
150 ClusteredDOMDataChangeListener {
152 public TranslatingClusteredDataChangeInvoker(LogicalDatastoreType store, InstanceIdentifier<?> path,
153 DataChangeListener bindingDataChangeListener,
154 DataChangeScope triggeringScope) {
155 super(store, path, bindingDataChangeListener, triggeringScope);
159 private class TranslatedDataChangeEvent implements AsyncDataChangeEvent<InstanceIdentifier<?>, DataObject> {
160 private final AsyncDataChangeEvent<YangInstanceIdentifier, NormalizedNode<?, ?>> domEvent;
161 private final InstanceIdentifier<?> path;
163 private Map<InstanceIdentifier<?>, DataObject> createdCache;
164 private Map<InstanceIdentifier<?>, DataObject> updatedCache;
165 private Map<InstanceIdentifier<?>, DataObject> originalCache;
166 private Set<InstanceIdentifier<?>> removedCache;
167 private Optional<DataObject> originalDataCache;
168 private Optional<DataObject> updatedDataCache;
170 public TranslatedDataChangeEvent(
171 final AsyncDataChangeEvent<YangInstanceIdentifier, NormalizedNode<?, ?>> change,
172 final InstanceIdentifier<?> path) {
173 this.domEvent = change;
178 public Map<InstanceIdentifier<?>, DataObject> getCreatedData() {
179 if (createdCache == null) {
180 createdCache = Collections.unmodifiableMap(toBinding(path, domEvent.getCreatedData()));
186 public Map<InstanceIdentifier<?>, DataObject> getUpdatedData() {
187 if (updatedCache == null) {
188 updatedCache = Collections.unmodifiableMap(toBinding(path, domEvent.getUpdatedData()));
195 public Set<InstanceIdentifier<?>> getRemovedPaths() {
196 if (removedCache == null) {
197 removedCache = Collections.unmodifiableSet(toBinding(path, domEvent.getRemovedPaths()));
203 public Map<InstanceIdentifier<?>, DataObject> getOriginalData() {
204 if (originalCache == null) {
205 originalCache = Collections.unmodifiableMap(toBinding(path, domEvent.getOriginalData()));
207 return originalCache;
212 public DataObject getOriginalSubtree() {
213 if (originalDataCache == null) {
214 if (domEvent.getOriginalSubtree() != null) {
215 originalDataCache = toBindingData(path, domEvent.getOriginalSubtree());
217 originalDataCache = Optional.absent();
220 return originalDataCache.orNull();
224 public DataObject getUpdatedSubtree() {
225 if (updatedDataCache == null) {
226 if (domEvent.getUpdatedSubtree() != null) {
227 updatedDataCache = toBindingData(path, domEvent.getUpdatedSubtree());
229 updatedDataCache = Optional.absent();
232 return updatedDataCache.orNull();
236 public String toString() {
237 return MoreObjects.toStringHelper(TranslatedDataChangeEvent.class) //
238 .add("created", getCreatedData()) //
239 .add("updated", getUpdatedData()) //
240 .add("removed", getRemovedPaths()) //
241 .add("dom", domEvent) //
246 private static class ListenerRegistrationImpl extends AbstractListenerRegistration<DataChangeListener> {
247 private final ListenerRegistration<DOMDataChangeListener> registration;
249 public ListenerRegistrationImpl(final DataChangeListener listener,
250 final ListenerRegistration<DOMDataChangeListener> registration) {
252 this.registration = registration;
256 protected void removeRegistration() {
257 registration.close();
262 public void close() {