com.couchbase.client.java.datastructures.CouchbaseArraySet Maven / Gradle / Ivy
/*
* Copyright (c) 2019 Couchbase, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.couchbase.client.java.datastructures;
import java.util.AbstractSet;
import java.util.ArrayList;
import java.util.Collections;
import java.util.ConcurrentModificationException;
import java.util.Iterator;
import java.util.Set;
import com.couchbase.client.core.annotation.Stability;
import com.couchbase.client.core.error.CouchbaseException;
import com.couchbase.client.core.error.DocumentNotFoundException;
import com.couchbase.client.core.error.context.ReducedKeyValueErrorContext;
import com.couchbase.client.core.error.subdoc.PathExistsException;
import com.couchbase.client.core.error.subdoc.PathNotFoundException;
import com.couchbase.client.core.retry.reactor.RetryExhaustedException;
import com.couchbase.client.java.Collection;
import com.couchbase.client.java.json.JsonArray;
import com.couchbase.client.java.json.JsonObject;
import com.couchbase.client.java.json.JsonValue;
import com.couchbase.client.core.error.CasMismatchException;
import com.couchbase.client.java.kv.ArraySetOptions;
import com.couchbase.client.java.kv.GetOptions;
import com.couchbase.client.java.kv.GetResult;
import com.couchbase.client.java.kv.LookupInOptions;
import com.couchbase.client.java.kv.LookupInResult;
import com.couchbase.client.java.kv.LookupInSpec;
import com.couchbase.client.java.kv.MutateInResult;
import com.couchbase.client.java.kv.MutateInSpec;
import com.couchbase.client.java.kv.StoreSemantics;
import static com.couchbase.client.core.util.Validators.notNull;
import static com.couchbase.client.core.util.Validators.notNullOrEmpty;
/**
* A CouchbaseArraySet is a {@link Set} backed by a {@link Collection Couchbase} document (more
* specifically a {@link JsonArray JSON array}).
*
* Note that a CouchbaseArraySet is restricted to primitive types (the types that a {@link JsonArray JSON array}
* can contain, except {@link JsonObject} and {@link JsonArray}). null entries are supported.
*
* @param the type of values in the set.
*
* @since 2.3.6
*/
@Stability.Committed
public class CouchbaseArraySet extends AbstractSet {
private final String id;
private final Collection collection;
private ArraySetOptions.Built arraySetOptions;
private final GetOptions getOptions;
private final LookupInOptions lookupInOptions;
/**
* Create a new {@link CouchbaseArraySet}, backed by the document identified by id
* in the given Couchbase bucket
. Note that if the document already exists,
* its content will be used as initial content for this collection. Otherwise it is created empty.
*
* @param id the id of the Couchbase document to back the set.
* @param collection the {@link Collection} through which to interact with the document.
* @param entityType a Class describing the type of objects in this Set.
* @param options a {@link ArraySetOptions} to use for all operations on this instance of the list.
*
**/
public CouchbaseArraySet(String id, Collection collection, Class entityType, ArraySetOptions options) {
notNull(collection, "Collection", () -> ReducedKeyValueErrorContext.create(id, null, null, null));
notNullOrEmpty(id, "Id", () -> ReducedKeyValueErrorContext.create(id, collection.bucketName(), collection.scopeName(), collection.name()));
notNull(entityType, "EntityType", () -> ReducedKeyValueErrorContext.create(id, collection.bucketName(), collection.scopeName(), collection.name()));
notNull(options, "ArraySetOptions", () -> ReducedKeyValueErrorContext.create(id, collection.bucketName(), collection.scopeName(), collection.name()));
this.id = id;
this.collection = collection;
// copy the options just in case they are reused later
ArraySetOptions.Built optionsIn = options.build();
ArraySetOptions arrayOpts = ArraySetOptions.arraySetOptions();
optionsIn.copyInto(arrayOpts);
this.arraySetOptions = arrayOpts.build();
this.getOptions = optionsIn.getOptions();
this.lookupInOptions = optionsIn.lookupInOptions();
}
@Override
public int size() {
try {
LookupInResult result = collection.lookupIn(
id,
Collections.singletonList(LookupInSpec.count("")),
lookupInOptions
);
return result.contentAs(0, Integer.class);
} catch (DocumentNotFoundException e) {
return 0;
}
}
@Override
public boolean isEmpty() {
try {
LookupInResult current = collection.lookupIn(
id,
Collections.singletonList(LookupInSpec.exists("[0]")),
lookupInOptions
);
return !current.exists(0);
} catch (DocumentNotFoundException e) {
return true;
}
}
@Override
public boolean contains(Object t) {
//TODO subpar implementation for a Set, use ARRAY_CONTAINS when available
enforcePrimitive(t);
try {
GetResult result = collection.get(id, getOptions);
JsonArray current = result.contentAs(JsonArray.class);
for (Object in : current) {
if (safeEquals(in, t)) {
return true;
}
}
return false;
} catch(DocumentNotFoundException e) {
return false;
}
}
@Override
public Iterator iterator() {
return new CouchbaseArraySetIterator<>();
}
@Override
public boolean add(T t) {
enforcePrimitive(t);
try {
collection.mutateIn(
id,
Collections.singletonList(MutateInSpec.arrayAddUnique("", t)),
arraySetOptions.mutateInOptions().storeSemantics(StoreSemantics.UPSERT)
);
return true;
} catch (PathExistsException ex) {
return false;
}
}
@Override
public boolean remove(Object t) {
enforcePrimitive(t);
for (int i = 0; i < arraySetOptions.casMismatchRetries(); i++) {
try {
GetResult result = collection.get(id);
JsonArray current = result.contentAsArray();
long cas = result.cas();
int index = 0;
boolean found = false;
for (Object next : current) {
if (safeEquals(next, t)) {
found = true;
break;
}
index++;
}
String path = "[" + index + "]";
if (!found) {
return false;
} else {
collection.mutateIn(
id,
Collections.singletonList(MutateInSpec.remove(path)),
arraySetOptions.mutateInOptions().cas(cas)
);
return true;
}
} catch (CasMismatchException e) {
//retry
} catch (DocumentNotFoundException ex) {
return false;
}
}
throw new CouchbaseException("CouchbaseArraySet remove failed",
new RetryExhaustedException("Couldn't perform remove in less than "
+ arraySetOptions.casMismatchRetries()
+ " iterations. It is likely concurrent modifications of this document are the reason")
);
}
@Override
public void clear() {
collection.remove(id);
}
/**
* Verify that the type of object t is compatible with CouchbaseArraySet storage.
*
* @param t the object to check.
* @throws ClassCastException if the object is incompatible.
*/
private void enforcePrimitive(Object t) throws ClassCastException {
if (!JsonValue.checkType(t)
|| t instanceof JsonValue) {
throw new ClassCastException("Only primitive types are supported in CouchbaseArraySet, got a " + t.getClass().getName());
}
}
private boolean safeEquals(Object expected, Object tested) {
if (expected == null) {
return tested == null;
}
return expected.equals(tested);
}
private class CouchbaseArraySetIterator implements Iterator {
private long cas;
private final Iterator delegate;
private int lastVisited;
private int cursor;
@SuppressWarnings("unchecked")
CouchbaseArraySetIterator() {
JsonArray current;
try {
GetResult result = collection.get(id);
current = result.contentAs(JsonArray.class);
// We use a list rather than a set, so the index of the
// removed item matches the index in the actual document in
// the server
this.cas = result.cas();
} catch (DocumentNotFoundException e) {
current = JsonArray.create();
this.cas = 0;
}
ArrayList list = new ArrayList<>(current.size());
for (E value : (Iterable) current) {
list.add(value);
}
this.delegate = list.iterator();
this.lastVisited = -1;
this.cursor = 0;
}
@Override
public boolean hasNext() {
return delegate.hasNext();
}
@Override
public E next() {
E next = delegate.next();
lastVisited = cursor;
cursor++;
return next;
}
@Override
public void remove() {
if (lastVisited < 0) {
throw new IllegalStateException();
}
// we simply want to remove lastVisited from
// the document on the server, and this set, assuming
// nothing has changed.
int index = lastVisited;
String idx = "[" + index + "]";
try {
MutateInResult updated = collection.mutateIn(
id,
Collections.singletonList(MutateInSpec.remove(idx)),
arraySetOptions.mutateInOptions().cas(cas));
//update the cas so that several removes in a row can work
this.cas = updated.cas();
//also correctly reset the state:
delegate.remove();
this.cursor = lastVisited;
this.lastVisited = -1;
} catch (CasMismatchException | DocumentNotFoundException ex) {
throw new ConcurrentModificationException("List was modified since iterator creation: " + ex);
} catch (PathNotFoundException ex) {
throw new ConcurrentModificationException("Element doesn't exist anymore at index: " + index);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy