blob: c13a3b85d93cabf0dd0aaf6166fa43601817e621 [file] [log] [blame]
Madan Jampani762246d2015-07-21 15:40:59 -07001/*
2 * Copyright 2015 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 */
16package org.onosproject.store.consistent.impl;
17
18import java.util.Set;
19import java.util.concurrent.CopyOnWriteArraySet;
20
21import org.onosproject.store.service.AtomicValue;
22import org.onosproject.store.service.AtomicValueEvent;
23import org.onosproject.store.service.AtomicValueEventListener;
24import org.onosproject.store.service.ConsistentMap;
25import org.onosproject.store.service.MapEvent;
26import org.onosproject.store.service.MapEventListener;
27import org.onosproject.store.service.Serializer;
28import org.onosproject.store.service.Versioned;
29
30/**
31 * Default implementation of AtomicValue.
32 *
33 * @param <V> value type
34 */
35public class DefaultAtomicValue<V> implements AtomicValue<V> {
36
37 private final Set<AtomicValueEventListener<V>> listeners = new CopyOnWriteArraySet<>();
38 private final ConsistentMap<String, byte[]> valueMap;
39 private final String name;
40 private final Serializer serializer;
41 private final MapEventListener<String, byte[]> mapEventListener = new InternalMapEventListener();
42
43 public DefaultAtomicValue(ConsistentMap<String, byte[]> valueMap,
44 String name,
45 Serializer serializer) {
46 this.valueMap = valueMap;
47 this.name = name;
48 this.serializer = serializer;
49 }
50
51 @Override
52 public boolean compareAndSet(V expect, V update) {
53 if (expect == null) {
54 if (update == null) {
55 return true;
56 }
57 return valueMap.putIfAbsent(name, serializer.encode(update)) == null;
58 } else {
59 if (update == null) {
60 return valueMap.remove(name, serializer.encode(expect));
61 }
62 return valueMap.replace(name, serializer.encode(expect), serializer.encode(update));
63 }
64 }
65
66 @Override
67 public V get() {
68 Versioned<byte[]> rawValue = valueMap.get(name);
69 return rawValue == null ? null : serializer.decode(rawValue.value());
70 }
71
72 @Override
73 public V getAndSet(V value) {
74 Versioned<byte[]> previousValue = value == null ?
75 valueMap.remove(name) : valueMap.put(name, serializer.encode(value));
76 return previousValue == null ? null : serializer.decode(previousValue.value());
77 }
78
79 @Override
80 public void set(V value) {
81 getAndSet(value);
82 }
83
84 @Override
85 public void addListener(AtomicValueEventListener<V> listener) {
86 synchronized (listeners) {
87 if (listeners.add(listener)) {
88 if (listeners.size() == 1) {
89 valueMap.addListener(mapEventListener);
90 }
91 }
92 }
93 }
94
95 @Override
96 public void removeListener(AtomicValueEventListener<V> listener) {
97 synchronized (listeners) {
98 if (listeners.remove(listener)) {
99 if (listeners.size() == 0) {
100 valueMap.removeListener(mapEventListener);
101 }
102 }
103 }
104 }
105
106 private class InternalMapEventListener implements MapEventListener<String, byte[]> {
107
108 @Override
109 public void event(MapEvent<String, byte[]> mapEvent) {
110 V newValue = mapEvent.type() == MapEvent.Type.REMOVE ? null : serializer.decode(mapEvent.value().value());
111 AtomicValueEvent<V> atomicValueEvent = new AtomicValueEvent<>(name, AtomicValueEvent.Type.UPDATE, newValue);
112 listeners.forEach(l -> l.event(atomicValueEvent));
113 }
114 }
115}