blob: f58f220836122fa1b7892a1b88ec8ce76c7ef3ca [file] [log] [blame]
Madan Jampani2914e4e2016-09-13 17:48:56 -07001/*
2 * Copyright 2016-present Open Networking Laboratory
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package org.onosproject.store.primitives.impl;
18
19import java.util.Map;
20import java.util.concurrent.CompletableFuture;
21
22import org.onosproject.store.service.AsyncDocumentTree;
23import org.onosproject.store.service.DocumentPath;
24import org.onosproject.store.service.DocumentTreeEvent;
25import org.onosproject.store.service.DocumentTreeListener;
26import org.onosproject.store.service.Serializer;
27import org.onosproject.store.service.Versioned;
28
29import com.google.common.collect.Maps;
30
31/**
32 * Default implementation of {@link AsyncDocumentTree}.
33 * <p>
34 * This implementation delegates execution to a backing tree implemented on top of Atomix framework.
35 *
Madan Jampani2914e4e2016-09-13 17:48:56 -070036 * @param <V> tree node value type.
37 */
38public class DefaultDistributedDocumentTree<V> implements AsyncDocumentTree<V> {
39
40 private final String name;
41 private final AsyncDocumentTree<byte[]> backingTree;
42 private final Serializer serializer;
43 private final Map<DocumentTreeListener<V>, InternalBackingDocumentTreeListener> listeners =
44 Maps.newIdentityHashMap();
45
46 DefaultDistributedDocumentTree(String name, AsyncDocumentTree<byte[]> backingTree, Serializer serializer) {
47 this.name = name;
48 this.backingTree = backingTree;
49 this.serializer = serializer;
50 }
51
52 @Override
53 public String name() {
54 return name;
55 }
56
57 @Override
58 public Type primitiveType() {
59 return backingTree.primitiveType();
60 }
61
62 @Override
63 public DocumentPath root() {
64 return backingTree.root();
65 }
66
67 @Override
68 public CompletableFuture<Map<String, Versioned<V>>> getChildren(DocumentPath path) {
69 return backingTree.getChildren(path)
70 .thenApply(map -> Maps.transformValues(map, v -> v.map(serializer::decode)));
71 }
72
73 @Override
74 public CompletableFuture<Versioned<V>> get(DocumentPath path) {
75 return backingTree.get(path)
76 .thenApply(v -> v == null ? null : v.map(serializer::decode));
77 }
78
79 @Override
80 public CompletableFuture<Versioned<V>> set(DocumentPath path, V value) {
81 return backingTree.set(path, serializer.encode(value))
82 .thenApply(v -> v == null ? null : v.map(serializer::decode));
83 }
84
85 @Override
86 public CompletableFuture<Boolean> create(DocumentPath path, V value) {
87 return backingTree.create(path, serializer.encode(value));
88 }
89
90 @Override
91 public CompletableFuture<Boolean> replace(DocumentPath path, V newValue, long version) {
92 return backingTree.replace(path, serializer.encode(newValue), version);
93 }
94
95 @Override
96 public CompletableFuture<Boolean> replace(DocumentPath path, V newValue, V currentValue) {
97 return backingTree.replace(path, serializer.encode(newValue), serializer.encode(currentValue));
98 }
99
100 @Override
101 public CompletableFuture<Versioned<V>> removeNode(DocumentPath path) {
102 return backingTree.removeNode(path)
103 .thenApply(v -> v == null ? null : v.map(serializer::decode));
104 }
105
106 @Override
107 public CompletableFuture<Void> addListener(DocumentPath path, DocumentTreeListener<V> listener) {
108 synchronized (listeners) {
109 InternalBackingDocumentTreeListener backingListener =
110 listeners.computeIfAbsent(listener, k -> new InternalBackingDocumentTreeListener(listener));
111 return backingTree.addListener(path, backingListener);
112 }
113 }
114
115 @Override
116 public CompletableFuture<Void> removeListener(DocumentTreeListener<V> listener) {
117 synchronized (listeners) {
118 InternalBackingDocumentTreeListener backingListener = listeners.remove(listener);
119 if (backingListener != null) {
120 return backingTree.removeListener(backingListener);
121 } else {
122 return CompletableFuture.completedFuture(null);
123 }
124 }
125 }
126
127 private class InternalBackingDocumentTreeListener implements DocumentTreeListener<byte[]> {
128
129 private final DocumentTreeListener<V> listener;
130
131 InternalBackingDocumentTreeListener(DocumentTreeListener<V> listener) {
132 this.listener = listener;
133 }
134
135 @Override
136 public void event(DocumentTreeEvent<byte[]> event) {
137 listener.event(new DocumentTreeEvent<V>(event.path(),
138 event.type(),
139 event.newValue().map(v -> v.map(serializer::decode)),
140 event.oldValue().map(v -> v.map(serializer::decode))));
141 }
142 }
143}