com.couchbase.client.java.datastructures.CouchbaseMap 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.AbstractMap;
import java.util.Collections;
import java.util.Iterator;
import java.util.Map;
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.retry.reactor.RetryExhaustedException;
import com.couchbase.client.java.Bucket;
import com.couchbase.client.java.Collection;
import com.couchbase.client.java.json.JsonArray;
import com.couchbase.client.java.json.JsonObject;
import com.couchbase.client.core.error.CasMismatchException;
import com.couchbase.client.core.error.DocumentExistsException;
import com.couchbase.client.core.error.subdoc.PathNotFoundException;
import com.couchbase.client.java.kv.GetOptions;
import com.couchbase.client.java.kv.InsertOptions;
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.MapOptions;
import com.couchbase.client.java.kv.MutateInSpec;
import static com.couchbase.client.core.util.Validators.notNull;
import static com.couchbase.client.core.util.Validators.notNullOrEmpty;
/**
* A CouchbaseMap is a {@link Map} backed by a {@link Bucket Couchbase} document (more specifically a
* {@link JsonObject JSON object}).
*
* Null keys are NOT permitted, and keys are restricted to {@link String}.
*
* Values in a CouchbaseMap are restricted to the types that a {@link JsonObject JSON objects}
* can contain. JSON sub-objects and sub-arrays can be represented as {@link JsonObject} and {@link JsonArray}
* respectively.
*
* @param the type of values in the map (restricted to {@link JsonObject}.
*
* @since 2.3.6
*/
@Stability.Committed
public class CouchbaseMap extends AbstractMap {
private final String id;
private final Collection collection;
private final Class entityTypeClass;
private final MapOptions.Built mapOptions;
private final GetOptions getOptions;
private final LookupInOptions lookupInOptions;
private final InsertOptions insertOptions;
/**
* Create a new {@link CouchbaseMap}, 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 map.
* @param collection the {@link Collection} through which to interact with the document.
* @param entityType a {@link Class} describing the type of objects used as values in this Map.
* @param options a {@link MapOptions} to use for all operations on this instance of the map.
*/
public CouchbaseMap(String id, Collection collection, Class entityType, MapOptions 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, "MapOptions", () -> ReducedKeyValueErrorContext.create(id, collection.bucketName(), collection.scopeName(), collection.name()));
this.id = id;
this.collection = collection;
this.entityTypeClass = entityType;
MapOptions.Built optionsIn = options.build();
MapOptions mapOpts = MapOptions.mapOptions();
optionsIn.copyInto(mapOpts);
this.mapOptions = mapOpts.build();
this.getOptions = optionsIn.getOptions();
this.lookupInOptions = optionsIn.lookupInOptions();
this.insertOptions = optionsIn.insertOptions();
}
@Override
public E put(String key, E value) {
checkKey(key);
for(int i = 0; i < mapOptions.casMismatchRetries(); i++) {
try {
long returnCas = 0;
E result = null;
try {
LookupInResult current = collection.lookupIn(id,
Collections.singletonList(LookupInSpec.get(key)),
lookupInOptions);
returnCas = current.cas();
if (current.exists(0)) {
result = current.contentAs(0, entityTypeClass);
}
} catch (PathNotFoundException e) {
// that's ok, we will just upsert anyways, and return null
} catch (DocumentNotFoundException e) {
// we will create an empty doc and remember the cas
returnCas = createEmpty();
}
collection.mutateIn(id,
Collections.singletonList(MutateInSpec.upsert(key, value)),
mapOptions.mutateInOptions().cas(returnCas));
return result;
} catch (CasMismatchException ex) {
//will need to retry get-and-set
}
}
throw new CouchbaseException("CouchbaseMap put failed",
new RetryExhaustedException("Couldn't perform put in less than "
+ mapOptions.casMismatchRetries()
+ " iterations. It is likely concurrent modifications of this document are the reason")
);
}
@Override
public E get(Object key) {
String idx = checkKey(key);
try {
return collection.lookupIn(id,
Collections.singletonList(LookupInSpec.get(idx)),
lookupInOptions)
.contentAs(0, entityTypeClass);
} catch (PathNotFoundException | DocumentNotFoundException e) {
return null;
}
}
@Override
public E remove(Object key) {
String idx = checkKey(key);
for(int i = 0; i < mapOptions.casMismatchRetries(); i++) {
try {
LookupInResult current = collection.lookupIn(id,
Collections.singletonList(LookupInSpec.get(idx)),
lookupInOptions);
long returnCas = current.cas();
E result = current.contentAs(0, entityTypeClass);
collection.mutateIn(id,
Collections.singletonList(MutateInSpec.remove(idx)),
mapOptions.mutateInOptions().cas(returnCas));
return result;
} catch (DocumentNotFoundException | PathNotFoundException e) {
return null;
} catch (CasMismatchException ex) {
//will have to retry get-and-remove
}
}
throw new CouchbaseException("CouchbaseMap remove failed",
new RetryExhaustedException("Couldn't perform remove in less than "
+ mapOptions.casMismatchRetries()
+ " iterations. It is likely concurrent modifications of this document are the reason")
);
}
@Override
public void clear() {
//optimized version over AbstractMap's (which uses the entry set)
collection.remove(id);
}
@Override
@SuppressWarnings("unchecked")
public Set> entrySet() {
JsonObject obj;
try {
obj = collection.get(id, getOptions).contentAsObject();
} catch (DocumentNotFoundException e) {
obj = JsonObject.create();
}
// don't actually create the doc, yet.
return new CouchbaseEntrySet((Map) obj.toMap());
}
@Override
public boolean containsKey(Object key) {
String idx = checkKey(key);
try {
return collection.lookupIn(id,
Collections.singletonList(LookupInSpec.exists(idx)),
lookupInOptions)
.exists(0);
} catch (DocumentNotFoundException e) {
return false;
}
}
@Override
public boolean containsValue(Object value) {
return super.containsValue(value); //TODO use ARRAY_CONTAINS subdoc operator when available
}
@Override
public int size() {
try {
LookupInResult current = collection.lookupIn(id,
Collections.singletonList(LookupInSpec.count("")),
lookupInOptions);
return current.contentAs(0, Integer.class);
} catch (DocumentNotFoundException e) {
return 0;
}
}
private String checkKey(final Object key) {
if (key == null) {
throw new NullPointerException("Unsupported null key");
}
return String.valueOf(key);
}
private class CouchbaseEntrySet implements Set> {
private final Set> delegate;
private CouchbaseEntrySet(Map data) {
this.delegate = data.entrySet();
}
@Override
public int size() {
return delegate.size();
}
@Override
public boolean isEmpty() {
return delegate.isEmpty();
}
@Override
public boolean contains(Object o) {
return delegate.contains(o);
}
@Override
public Iterator> iterator() {
return new CouchbaseEntrySetIterator(delegate.iterator());
}
@Override
public Object[] toArray() {
return delegate.toArray();
}
@Override
public T[] toArray(T[] a) {
return delegate.toArray(a);
}
@Override
public boolean add(Entry stringVEntry) {
return delegate.add(stringVEntry);
}
@Override
public boolean remove(Object o) {
if (delegate.remove(o)) {
if (o instanceof Map.Entry) {
Entry entry = (Entry) o;
CouchbaseMap.this.remove(entry.getKey());
} else {
throw new IllegalStateException("Expected entrySet remove() to remove an entry");
}
return true;
} else {
return false;
}
}
@Override
public boolean containsAll(java.util.Collection> c) {
return delegate.containsAll(c);
}
@Override
public boolean addAll(java.util.Collection extends Entry> c) {
return delegate.addAll(c);
}
@Override
public boolean retainAll(java.util.Collection> c) {
return delegate.retainAll(c);
}
@Override
public boolean removeAll(java.util.Collection> c) {
return delegate.removeAll(c);
}
@Override
public void clear() {
delegate.clear();
CouchbaseMap.this.clear();
}
@Override
public String toString() {
return delegate.toString();
}
public int hashCode() {
return delegate.hashCode();
}
public boolean equals(Object obj) {
return delegate.equals(obj);
}
}
private class CouchbaseEntrySetIterator implements Iterator> {
private final Iterator> delegateItr;
private Entry lastNext = null;
CouchbaseEntrySetIterator(Iterator> iterator) {
this.delegateItr = iterator;
}
@Override
public boolean hasNext() {
return delegateItr.hasNext();
}
@Override
public Entry next() {
this.lastNext = delegateItr.next();
return lastNext;
}
@Override
public void remove() {
if (lastNext == null)
throw new IllegalStateException("next() hasn't been called before remove()");
delegateItr.remove();
CouchbaseMap.this.remove(lastNext.getKey());
}
}
private long createEmpty() {
try {
return collection.insert(id, JsonObject.create(), insertOptions).cas();
} catch (DocumentExistsException ex) {
// Ignore concurrent creations, keep on moving.
// but we need the cas, so...
return collection.get(id, getOptions).cas();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy