blob: dd4ac50ea5abf3201d3a7ce544cb4fdb7400cd1c [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 *
36 * @See AtomixDocumentTree
37 *
38 * @param <V> tree node value type.
39 */
40public class DefaultDistributedDocumentTree<V> implements AsyncDocumentTree<V> {
41
42 private final String name;
43 private final AsyncDocumentTree<byte[]> backingTree;
44 private final Serializer serializer;
45 private final Map<DocumentTreeListener<V>, InternalBackingDocumentTreeListener> listeners =
46 Maps.newIdentityHashMap();
47
48 DefaultDistributedDocumentTree(String name, AsyncDocumentTree<byte[]> backingTree, Serializer serializer) {
49 this.name = name;
50 this.backingTree = backingTree;
51 this.serializer = serializer;
52 }
53
54 @Override
55 public String name() {
56 return name;
57 }
58
59 @Override
60 public Type primitiveType() {
61 return backingTree.primitiveType();
62 }
63
64 @Override
65 public DocumentPath root() {
66 return backingTree.root();
67 }
68
69 @Override
70 public CompletableFuture<Map<String, Versioned<V>>> getChildren(DocumentPath path) {
71 return backingTree.getChildren(path)
72 .thenApply(map -> Maps.transformValues(map, v -> v.map(serializer::decode)));
73 }
74
75 @Override
76 public CompletableFuture<Versioned<V>> get(DocumentPath path) {
77 return backingTree.get(path)
78 .thenApply(v -> v == null ? null : v.map(serializer::decode));
79 }
80
81 @Override
82 public CompletableFuture<Versioned<V>> set(DocumentPath path, V value) {
83 return backingTree.set(path, serializer.encode(value))
84 .thenApply(v -> v == null ? null : v.map(serializer::decode));
85 }
86
87 @Override
88 public CompletableFuture<Boolean> create(DocumentPath path, V value) {
89 return backingTree.create(path, serializer.encode(value));
90 }
91
92 @Override
93 public CompletableFuture<Boolean> replace(DocumentPath path, V newValue, long version) {
94 return backingTree.replace(path, serializer.encode(newValue), version);
95 }
96
97 @Override
98 public CompletableFuture<Boolean> replace(DocumentPath path, V newValue, V currentValue) {
99 return backingTree.replace(path, serializer.encode(newValue), serializer.encode(currentValue));
100 }
101
102 @Override
103 public CompletableFuture<Versioned<V>> removeNode(DocumentPath path) {
104 return backingTree.removeNode(path)
105 .thenApply(v -> v == null ? null : v.map(serializer::decode));
106 }
107
108 @Override
109 public CompletableFuture<Void> addListener(DocumentPath path, DocumentTreeListener<V> listener) {
110 synchronized (listeners) {
111 InternalBackingDocumentTreeListener backingListener =
112 listeners.computeIfAbsent(listener, k -> new InternalBackingDocumentTreeListener(listener));
113 return backingTree.addListener(path, backingListener);
114 }
115 }
116
117 @Override
118 public CompletableFuture<Void> removeListener(DocumentTreeListener<V> listener) {
119 synchronized (listeners) {
120 InternalBackingDocumentTreeListener backingListener = listeners.remove(listener);
121 if (backingListener != null) {
122 return backingTree.removeListener(backingListener);
123 } else {
124 return CompletableFuture.completedFuture(null);
125 }
126 }
127 }
128
129 private class InternalBackingDocumentTreeListener implements DocumentTreeListener<byte[]> {
130
131 private final DocumentTreeListener<V> listener;
132
133 InternalBackingDocumentTreeListener(DocumentTreeListener<V> listener) {
134 this.listener = listener;
135 }
136
137 @Override
138 public void event(DocumentTreeEvent<byte[]> event) {
139 listener.event(new DocumentTreeEvent<V>(event.path(),
140 event.type(),
141 event.newValue().map(v -> v.map(serializer::decode)),
142 event.oldValue().map(v -> v.map(serializer::decode))));
143 }
144 }
145}