blob: 22a034509e0713ffc77cf2819f723ba2717a1f9f [file] [log] [blame]
Aaron Kruglikova26f6542016-04-19 13:37:42 -07001/*
Brian O'Connor0a4e6742016-09-15 23:03:10 -07002 * Copyright 2016-present Open Networking Laboratory
Aaron Kruglikova26f6542016-04-19 13:37:42 -07003 *
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.resources.impl;
18
19import com.google.common.collect.Lists;
Aaron Kruglikova26f6542016-04-19 13:37:42 -070020import com.google.common.collect.Multiset;
21import io.atomix.copycat.client.CopycatClient;
22import io.atomix.resource.AbstractResource;
Aaron Kruglikov44a1fef2016-04-27 11:29:23 -070023import io.atomix.resource.ResourceTypeInfo;
Aaron Kruglikova26f6542016-04-19 13:37:42 -070024import org.onosproject.store.service.AsyncConsistentMultimap;
Jonathan Hart46bf89b2017-02-27 15:56:42 -080025import org.onosproject.store.service.MultimapEvent;
26import org.onosproject.store.service.MultimapEventListener;
Aaron Kruglikova26f6542016-04-19 13:37:42 -070027import org.onosproject.store.service.Versioned;
28
29import java.util.Collection;
30import java.util.ConcurrentModificationException;
Jonathan Hart46bf89b2017-02-27 15:56:42 -080031import java.util.List;
Aaron Kruglikova26f6542016-04-19 13:37:42 -070032import java.util.Map;
33import java.util.Properties;
34import java.util.Set;
35import java.util.concurrent.CompletableFuture;
Jonathan Hart46bf89b2017-02-27 15:56:42 -080036import java.util.concurrent.ConcurrentHashMap;
37import java.util.concurrent.Executor;
Aaron Kruglikova26f6542016-04-19 13:37:42 -070038
Aaron Kruglikova1801aa2016-07-12 15:17:30 -070039import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Clear;
40import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.ContainsEntry;
41import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.ContainsKey;
42import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.ContainsValue;
43import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Entries;
44import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Get;
45import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.IsEmpty;
46import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.KeySet;
47import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Keys;
Jonathan Hart46bf89b2017-02-27 15:56:42 -080048import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Listen;
Aaron Kruglikova1801aa2016-07-12 15:17:30 -070049import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.MultiRemove;
50import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Put;
51import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.RemoveAll;
52import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Replace;
53import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Size;
Jonathan Hart46bf89b2017-02-27 15:56:42 -080054import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Unlisten;
Aaron Kruglikova1801aa2016-07-12 15:17:30 -070055import static org.onosproject.store.primitives.resources.impl.AtomixConsistentMultimapCommands.Values;
Aaron Kruglikova26f6542016-04-19 13:37:42 -070056
Jonathan Hart46bf89b2017-02-27 15:56:42 -080057
Aaron Kruglikova26f6542016-04-19 13:37:42 -070058/**
59 * Set based implementation of the {@link AsyncConsistentMultimap}.
60 * <p>
61 * Note: this implementation does not allow null entries or duplicate entries.
62 */
Aaron Kruglikova1801aa2016-07-12 15:17:30 -070063@ResourceTypeInfo(id = -153, factory = AtomixConsistentSetMultimapFactory.class)
64public class AtomixConsistentSetMultimap
65 extends AbstractResource<AtomixConsistentSetMultimap>
Aaron Kruglikova26f6542016-04-19 13:37:42 -070066 implements AsyncConsistentMultimap<String, byte[]> {
67
Jonathan Hart46bf89b2017-02-27 15:56:42 -080068 private final Map<MultimapEventListener<String, byte[]>, Executor> mapEventListeners = new ConcurrentHashMap<>();
69
70 public static final String CHANGE_SUBJECT = "multimapChangeEvents";
71
Aaron Kruglikova1801aa2016-07-12 15:17:30 -070072 public AtomixConsistentSetMultimap(CopycatClient client,
73 Properties properties) {
Aaron Kruglikova26f6542016-04-19 13:37:42 -070074 super(client, properties);
75 }
76
77 @Override
Aaron Kruglikova1801aa2016-07-12 15:17:30 -070078 public CompletableFuture<AtomixConsistentSetMultimap> open() {
Jonathan Hart46bf89b2017-02-27 15:56:42 -080079 return super.open().thenApply(result -> {
80 client.onStateChange(state -> {
81 if (state == CopycatClient.State.CONNECTED && isListening()) {
82 client.submit(new Listen());
83 }
84 });
85 client.onEvent(CHANGE_SUBJECT, this::handleEvent);
86 return result;
87 });
88 }
89
90 private void handleEvent(List<MultimapEvent<String, byte[]>> events) {
91 events.forEach(event ->
92 mapEventListeners.forEach((listener, executor) -> executor.execute(() -> listener.event(event))));
Aaron Kruglikova26f6542016-04-19 13:37:42 -070093 }
94
95 @Override
96 public CompletableFuture<Integer> size() {
Madan Jampani630e7ac2016-05-31 11:34:05 -070097 return client.submit(new Size());
Aaron Kruglikova26f6542016-04-19 13:37:42 -070098 }
99
100 @Override
101 public CompletableFuture<Boolean> isEmpty() {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700102 return client.submit(new IsEmpty());
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700103 }
104
105 @Override
106 public CompletableFuture<Boolean> containsKey(String key) {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700107 return client.submit(new ContainsKey(key));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700108 }
109
110 @Override
111 public CompletableFuture<Boolean> containsValue(byte[] value) {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700112 return client.submit(new ContainsValue(value));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700113 }
114
115 @Override
116 public CompletableFuture<Boolean> containsEntry(String key, byte[] value) {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700117 return client.submit(new ContainsEntry(key, value));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700118 }
119
120 @Override
121 public CompletableFuture<Boolean> put(String key, byte[] value) {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700122 return client.submit(new Put(key, Lists.newArrayList(value), null));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700123 }
124
125 @Override
126 public CompletableFuture<Boolean> remove(String key, byte[] value) {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700127 return client.submit(new MultiRemove(key,
128 Lists.newArrayList(value),
129 null));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700130 }
131
132 @Override
Madan Jampani630e7ac2016-05-31 11:34:05 -0700133 public CompletableFuture<Boolean> removeAll(String key, Collection<? extends byte[]> values) {
134 return client.submit(new MultiRemove(key, (Collection<byte[]>) values, null));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700135 }
136
137 @Override
Madan Jampani630e7ac2016-05-31 11:34:05 -0700138 public CompletableFuture<Versioned<Collection<? extends byte[]>>> removeAll(String key) {
139 return client.submit(new RemoveAll(key, null));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700140 }
141
142 @Override
Aaron Kruglikov44a1fef2016-04-27 11:29:23 -0700143 public CompletableFuture<Boolean> putAll(
144 String key, Collection<? extends byte[]> values) {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700145 return client.submit(new Put(key, values, null));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700146 }
147
148 @Override
Madan Jampani630e7ac2016-05-31 11:34:05 -0700149 public CompletableFuture<Versioned<Collection<? extends byte[]>>> replaceValues(
Aaron Kruglikov44a1fef2016-04-27 11:29:23 -0700150 String key, Collection<byte[]> values) {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700151 return client.submit(new Replace(key, values, null));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700152 }
153
154 @Override
155 public CompletableFuture<Void> clear() {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700156 return client.submit(new Clear());
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700157 }
158
159 @Override
Madan Jampani630e7ac2016-05-31 11:34:05 -0700160 public CompletableFuture<Versioned<Collection<? extends byte[]>>> get(String key) {
161 return client.submit(new Get(key));
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700162 }
163
164 @Override
165 public CompletableFuture<Set<String>> keySet() {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700166 return client.submit(new KeySet());
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700167 }
168
169 @Override
170 public CompletableFuture<Multiset<String>> keys() {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700171 return client.submit(new Keys());
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700172 }
173
Jonathan Hartad0c3022017-02-22 14:06:01 -0800174 @Override
Aaron Kruglikov44a1fef2016-04-27 11:29:23 -0700175 public CompletableFuture<Multiset<byte[]>> values() {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700176 return client.submit(new Values());
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700177 }
178
179 @Override
180 public CompletableFuture<Collection<Map.Entry<String, byte[]>>> entries() {
Madan Jampani630e7ac2016-05-31 11:34:05 -0700181 return client.submit(new Entries());
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700182 }
183
184 @Override
Jonathan Hart46bf89b2017-02-27 15:56:42 -0800185 public CompletableFuture<Void> addListener(MultimapEventListener<String, byte[]> listener, Executor executor) {
186 if (mapEventListeners.isEmpty()) {
187 return client.submit(new Listen()).thenRun(() -> mapEventListeners.put(listener, executor));
188 } else {
189 mapEventListeners.put(listener, executor);
190 return CompletableFuture.completedFuture(null);
191 }
192 }
193
194 @Override
195 public CompletableFuture<Void> removeListener(MultimapEventListener<String, byte[]> listener) {
196 if (mapEventListeners.remove(listener) != null && mapEventListeners.isEmpty()) {
197 return client.submit(new Unlisten()).thenApply(v -> null);
198 }
199 return CompletableFuture.completedFuture(null);
200 }
201
202 @Override
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700203 public CompletableFuture<Map<String, Collection<byte[]>>> asMap() {
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700204 throw new UnsupportedOperationException("Expensive operation.");
205 }
206
207 @Override
208 public String name() {
209 return null;
210 }
211
212 /**
213 * Helper to check if there was a lock based issue.
214 * @param status the status of an update result
215 */
216 private void throwIfLocked(MapEntryUpdateResult.Status status) {
217 if (status == MapEntryUpdateResult.Status.WRITE_LOCK) {
Aaron Kruglikov44a1fef2016-04-27 11:29:23 -0700218 throw new ConcurrentModificationException("Cannot update map: " +
219 "Another transaction " +
220 "in progress");
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700221 }
222 }
Jonathan Hart46bf89b2017-02-27 15:56:42 -0800223
224 private boolean isListening() {
225 return !mapEventListeners.isEmpty();
226 }
Aaron Kruglikova26f6542016-04-19 13:37:42 -0700227}