blob: 26118cf620cfbc903ad5c2e61587bc9822d0989a [file] [log] [blame]
Aaron Kruglikov92511f22015-10-12 14:39:04 -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 */
16
17package org.onosproject.persistence.impl;
18
19import com.google.common.collect.Iterators;
20import org.mapdb.DB;
21import org.mapdb.Hasher;
22import org.mapdb.Serializer;
23
24import java.util.Collection;
25import java.util.Iterator;
26import java.util.Set;
27
28import static com.google.common.base.Preconditions.checkNotNull;
29
30/**
31 * A set implementation that gets and receives all data from a serialized internal set.
32 */
33//TODO add locking for reads and writes
34public class PersistentSet<E> implements Set<E> {
35
36 private final org.onosproject.store.service.Serializer serializer;
37
38 private final org.mapdb.DB database;
39
40 private final Set<byte[]> items;
41
42 private final String name;
43
44 public PersistentSet(org.onosproject.store.service.Serializer serializer, DB database, String name) {
45 this.serializer = checkNotNull(serializer);
46 this.database = checkNotNull(database);
47 this.name = checkNotNull(name);
48
49 items = database
50 .createHashSet(name)
51 .serializer(Serializer.BYTE_ARRAY)
52 .hasher(Hasher.BYTE_ARRAY)
53 .makeOrGet();
54 }
55
56 public void readInto(Set<E> items) {
57 this.items.forEach(item -> items.add(serializer.decode(item)));
58 }
59
60 @Override
61 public int size() {
62 return items.size();
63 }
64
65 @Override
66 public boolean isEmpty() {
67 return items.isEmpty();
68 }
69
70 @Override
71 public boolean contains(Object o) {
72 checkNotNull(o, "The argument cannot be null");
73 return items.contains(serializer.encode(o));
74 }
75
76 @Override
77 public Iterator<E> iterator() {
78 return Iterators.transform(items.iterator(), serializer::decode);
79 }
80
81 @Override
82 public Object[] toArray() {
83 Object[] retArray = new Object[items.size()];
84 int index = 0;
85 Iterator<byte[]> iterator = items.iterator();
86 while (iterator.hasNext()) {
87 retArray[index] = serializer.decode(iterator.next());
88 index++;
89 }
90 return retArray;
91 }
92
93 @Override
94 public <T> T[] toArray(T[] a) {
95 checkNotNull(a, "The passed in array cannot be null.");
96 int index = 0;
97 Iterator<byte[]> iterator = items.iterator();
98 T[] retArray;
99 if (a.length >= items.size()) {
100 retArray = a;
101 } else {
102 retArray = (T[]) new Object[items.size()];
103 }
104 while (iterator.hasNext()) {
105 retArray[index++] = serializer.decode(iterator.next());
106 }
107 if (retArray.length > items.size()) {
108 retArray[index] = null;
109 }
110 return retArray;
111 }
112
113 @Override
114 public boolean add(E item) {
115 checkNotNull("Item to be added cannot be null.");
116 return items.add(serializer.encode(item));
117 }
118
119 @Override
120 public boolean remove(Object o) {
121 checkNotNull(o, "Item to be removed cannot be null.");
122 return items.remove(serializer.encode(o));
123 }
124
125 @Override
126 public boolean containsAll(Collection<?> c) {
127 checkNotNull(c, "Collection cannot be internal.");
128 for (Object item : c) {
129 if (!items.contains(serializer.encode(item))) {
130 return false;
131 }
132 }
133 return true;
134 }
135
136 @Override
137 public boolean addAll(Collection<? extends E> c) {
138 checkNotNull(c, "The collection to be added cannot be null.");
139 boolean changed = false;
140 for (Object item : c) {
141 changed = items.add(serializer.encode(item)) || changed;
142 }
143 return changed;
144 }
145
146 @Override
147 public boolean retainAll(Collection<?> c) {
148 boolean changed = false;
149 for (byte[] item : items) {
150 E deserialized = serializer.decode(item);
151 if (!c.contains(deserialized)) {
152 changed = items.remove(item) || changed;
153 }
154 }
155 return changed;
156 }
157
158 @Override
159 public boolean removeAll(Collection<?> c) {
160 boolean changed = false;
161 for (Object item : c) {
162 changed = items.remove(serializer.encode(item)) || changed;
163 }
164 return changed;
165 }
166
167 @Override
168 public void clear() {
169 items.clear();
170 }
171
172 @Override
173 public boolean equals(Object set) {
174 //This is not threadsafe and on larger sets incurs a significant processing cost
175 if (!(set instanceof Set)) {
176 return false;
177 }
178 Set asSet = (Set) set;
179 if (asSet.size() != this.size()) {
180 return false;
181 }
182 for (Object item : this) {
183 if (!asSet.contains(item)) {
184 return false;
185 }
186 }
187 return true;
188 }
189
190 @Override
191 public int hashCode() {
192 return super.hashCode();
193 }
194}