com.basho.riak.client.bucket.DomainBucket Maven / Gradle / Ivy
/*
* This file is provided to you 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.basho.riak.client.bucket;
import com.basho.riak.client.RiakException;
import com.basho.riak.client.builders.DomainBucketBuilder;
import com.basho.riak.client.cap.ConflictResolver;
import com.basho.riak.client.cap.Mutation;
import com.basho.riak.client.cap.MutationProducer;
import com.basho.riak.client.cap.Retrier;
import com.basho.riak.client.cap.VClock;
import com.basho.riak.client.convert.Converter;
import com.basho.riak.client.convert.KeyUtil;
import com.basho.riak.client.convert.RiakKey;
import com.basho.riak.client.convert.VClockUtil;
import com.basho.riak.client.operations.DeleteObject;
import com.basho.riak.client.operations.FetchObject;
import com.basho.riak.client.operations.MultiFetchObject;
import com.basho.riak.client.operations.StoreObject;
import com.basho.riak.client.query.MultiFetchFuture;
import com.basho.riak.client.raw.DeleteMeta;
import com.basho.riak.client.raw.FetchMeta;
import com.basho.riak.client.raw.StoreMeta;
import java.util.Arrays;
import java.util.List;
/**
* A domain bucket is a wrapper around a {@link Bucket} that is strongly typed and uses
* a preset {@link ConflictResolver}, {@link MutationProducer}, {@link Converter}, r, w, dw, rw, {@link Retrier},
* returnBody etc
*
*
* If you are working with one specific type of data only it can be simpler to
* create a {@link DomainBucket} around your bucket. It reduces the amount of
* code since the {@link Converter}, {@link Mutation}, {@link Retrier} and
* {@link ConflictResolver} are likely to be the same for each operation.
*
*
* Example:
*
* final Bucket b = client.createBucket(bucketName).allowSiblings(true).nVal(3).execute();
*
* final DomainBucket carts = DomainBucket.builder(b, ShoppingCart.class)
* .withResolver(new MergeCartResolver())
* .returnBody(true)
* .retrier(DefaultRetrier.attempts(3))
* .w(1)
* .dw(1)
* .r(1)
* .rw(1)
* .build();
*
* final ShoppingCart cart = new ShoppingCart(userId);
*
* cart.addItem("coffee");
* cart.addItem("fixie");
* cart.addItem("moleskine");
*
* final ShoppingCart storedCart = carts.store(cart);
* ShoppinCart cart2 = carts.fetch("userX");
* cart.addItem("toaster");
* carts.store(cart2);
* //etc
*
*
*
* @author russell
* @see RiakBucket
* @see DomainBucketBuilder
*
*/
public class DomainBucket {
private final Bucket bucket;
private final ConflictResolver resolver;
private final Converter converter;
private final MutationProducer mutationProducer;
private final StoreMeta storeMeta;
private final FetchMeta fetchMeta;
private final DeleteMeta deleteMeta;
private final Class clazz;
private final Retrier retrier;
private final boolean withoutFetch;
/**
* Create a new {@link DomainBucket} for clazz
Class objects
* wrapped around bucket
*
* It is recommended to use the {@link DomainBucketBuilder} rather than this
* constructor.
*
* @param bucket
* The bucket to wrap.
* @param resolver
* the {@link ConflictResolver}
* @param converter
* the {@link Converter} to use
* @param mutationProducer
* the {@link MutationProducer} to use
* @param storeMeta
* the set of store parameters
* @param fetchMeta
* the set of fetch parameters
* @param deleteMeta
* the set of delete parameters
* @param clazz
* the Class type of the DomainBucket
* @param retrier
* the {@link Retrier} to use for each operation
* @param withoutFetch
* controls whether a store operation should do a fetch first
* @see StoreObject#withoutFetch()
*/
public DomainBucket(Bucket bucket, ConflictResolver resolver, Converter converter,
MutationProducer mutationProducer, StoreMeta storeMeta, FetchMeta fetchMeta, DeleteMeta deleteMeta,
Class clazz, final Retrier retrier, boolean withoutFetch) {
this.bucket = bucket;
this.resolver = resolver;
this.converter = converter;
this.mutationProducer = mutationProducer;
this.storeMeta = storeMeta;
this.fetchMeta = fetchMeta;
this.deleteMeta = deleteMeta;
this.clazz = clazz;
this.retrier = retrier;
this.withoutFetch = withoutFetch;
}
public DomainBucket(Bucket bucket, ConflictResolver resolver, Converter converter,
MutationProducer mutationProducer, StoreMeta storeMeta, FetchMeta fetchMeta, DeleteMeta deleteMeta,
Class clazz, final Retrier retrier) {
this(bucket, resolver, converter, mutationProducer, storeMeta, fetchMeta, deleteMeta, clazz, retrier, false);
}
/**
* Store o
in Riak.
* T
must have a field annotated with {@link RiakKey}.
*
*
* This is equivalent to creating and executing a {@link StoreObject}
* operation with the {@link Converter}, {@link ConflictResolver},
* {@link Retrier}, and a {@link Mutation} (from calling
* {@link MutationProducer#produce(Object)} on o
), r, w, dw
* etc. passed when the DomainBucket was constructed.
*
*
* @param o
* instance of to store.
* @return stored instance of T
* @throws RiakException
*/
public T store(T o) throws RiakException {
final Mutation mutation = mutationProducer.produce(o);
final StoreObject so = bucket.store(o)
.withConverter(converter)
.withMutator(mutation)
.withResolver(resolver)
.withRetrier(retrier);
if (withoutFetch) {
so.withoutFetch();
}
if (fetchMeta.getR() != null) {
so.r(fetchMeta.getR());
}
if (storeMeta.hasW()) {
so.w(storeMeta.getW());
}
if (storeMeta.hasDw()) {
so.dw(storeMeta.getDw());
}
if (fetchMeta.hasPr()) {
so.pr(fetchMeta.getPr());
}
if (storeMeta.hasPw()) {
so.pw(storeMeta.getPw());
}
if (fetchMeta.hasBasicQuorum()) {
so.basicQuorum(fetchMeta.getBasicQuorum());
}
if (fetchMeta.hasNotFoundOk()) {
so.notFoundOK(fetchMeta.getNotFoundOK());
}
if (storeMeta.hasIfNotModified()) {
so.ifNotModified(storeMeta.isIfNotModified());
}
if (storeMeta.hasIfNoneMatch()) {
so.ifNoneMatch(storeMeta.isIfNoneMatch());
}
if (fetchMeta.hasReturnDeletedVClock()) {
so.returnDeletedVClock(fetchMeta.getReturnDeletedVClock());
}
if (storeMeta.hasReturnBody()) {
so.returnBody(storeMeta.getReturnBody());
}
return so.execute();
}
/**
* Fetch data stored at key
in this bucket as an instance of
* T
.
*
*
* This is equivalent to creating and executing a {@link FetchObject}
* configured with the {@link Converter}, {@link ConflictResolver},
* {@link Retrier} and r value specified in the constructor.
*
*
* @param key
* @return
* @throws RiakException
*/
public T fetch(String key) throws RiakException {
final FetchObject fo = bucket.fetch(key, clazz)
.withConverter(converter)
.withResolver(resolver)
.withRetrier(retrier);
if (fetchMeta.hasR()) {
fo.r(fetchMeta.getR());
}
if (fetchMeta.hasPr()) {
fo.pr(fetchMeta.getPr());
}
if (fetchMeta.hasBasicQuorum()) {
fo.basicQuorum(fetchMeta.getBasicQuorum());
}
if (fetchMeta.hasNotFoundOk()) {
fo.notFoundOK(fetchMeta.getNotFoundOK());
}
if (fetchMeta.hasReturnDeletedVClock()) {
fo.returnDeletedVClock(fetchMeta.getReturnDeletedVClock());
}
return fo.execute();
}
/**
* Fetch data stored at the key extracted from o
's
* {@link RiakKey} annotated field as an instance of
* T
.
*
*
* This is equivalent to creating and executing a {@link FetchObject}
* configured with the {@link Converter}, {@link ConflictResolver},
* {@link Retrier} and r value specified in the constructor.
*
*
* @param key
* @return
* @throws RiakException
*/
public T fetch(T o) throws RiakException {
final FetchObject fo = bucket.fetch(o)
.withConverter(converter)
.withResolver(resolver)
.withRetrier(retrier);
if (fetchMeta.hasR()) {
fo.r(fetchMeta.getR());
}
if (fetchMeta.hasPr()) {
fo.pr(fetchMeta.getPr());
}
if (fetchMeta.hasBasicQuorum()) {
fo.basicQuorum(fetchMeta.getBasicQuorum());
}
if (fetchMeta.hasNotFoundOk()) {
fo.notFoundOK(fetchMeta.getNotFoundOK());
}
if (fetchMeta.hasReturnDeletedVClock()) {
fo.returnDeletedVClock(fetchMeta.getReturnDeletedVClock());
}
return fo.execute();
}
/**
* Fetch data stored at keys
in this bucket as a List of
* {@link MultiFetchFuture} objects that will return instances of {@code T}
*
*
* This is equivalent to creating and executing a {@link MultiFetchObject}
* configured with the {@link Converter}, {@link ConflictResolver},
* {@link Retrier} and r value specified in the constructor.
*
*
* @param keys
* @return A list of futures
* @see MultiFetchObject
*/
public List> multiFetch(String[] keys)
{
final MultiFetchObject fo = bucket.multiFetch(Arrays.asList(keys), clazz)
.withConverter(converter)
.withResolver(resolver)
.withRetrier(retrier);
if (fetchMeta.hasR()) {
fo.r(fetchMeta.getR());
}
if (fetchMeta.hasPr()) {
fo.pr(fetchMeta.getPr());
}
if (fetchMeta.hasBasicQuorum()) {
fo.basicQuorum(fetchMeta.getBasicQuorum());
}
if (fetchMeta.hasNotFoundOk()) {
fo.notFoundOK(fetchMeta.getNotFoundOK());
}
if (fetchMeta.hasReturnDeletedVClock()) {
fo.returnDeletedVClock(fetchMeta.getReturnDeletedVClock());
}
return fo.execute();
}
/**
* Fetch data stored at the keys extracted from each obj
's
* {@link RiakKey} annotated field as an instance of
* T
.
*
*
* This is equivalent to creating and executing a {@link MultiFetchObject}
* configured with the {@link Converter}, {@link ConflictResolver},
* {@link Retrier} and r value specified in the constructor.
*
*
* @param keys
* @return a list of futures
* @see MultiFetchObject
*/
public List> multiFetch(List objs)
{
final MultiFetchObject fo = bucket.multiFetch(objs)
.withConverter(converter)
.withResolver(resolver)
.withRetrier(retrier);
if (fetchMeta.hasR()) {
fo.r(fetchMeta.getR());
}
if (fetchMeta.hasPr()) {
fo.pr(fetchMeta.getPr());
}
if (fetchMeta.hasBasicQuorum()) {
fo.basicQuorum(fetchMeta.getBasicQuorum());
}
if (fetchMeta.hasNotFoundOk()) {
fo.notFoundOK(fetchMeta.getNotFoundOK());
}
if (fetchMeta.hasReturnDeletedVClock()) {
fo.returnDeletedVClock(fetchMeta.getReturnDeletedVClock());
}
return fo.execute();
}
/**
* Delete the key/value stored at the key extracted from o
's
* {@link RiakKey} annotated field.
*
*
* This is equivalent to creating and executing a {@link DeleteObject}
* configured with the {@link Retrier} and r value specified in the
* constructor.
*
*
* If a field is annotated with {@code @RiakVClock} and not null
* the value is used. Otherwise a fetch is performed to retrieve it.
*
* @param key
* @return
* @throws RiakException
*/
public void delete(T o) throws RiakException {
final String key = KeyUtil.getKey(o);
final VClock vclock = VClockUtil.getVClock(o);
delete(key, vclock);
}
/**
* Delete the key/value stored at the key
*
*
* This is equivalent to creating and executing a {@link DeleteObject}
* configured with the {@link Retrier} and r value specified in the
* constructor. A fetch is performed to get the vclock.
*
*
* @param key the key for the object
* @throws RiakException
* @deprecated - use {@link #delete(java.lang.String, com.basho.riak.client.cap.VClock)}
*/
@Deprecated
public void delete(String key) throws RiakException {
delete(key, null);
}
/**
* Delete the key/value stored at the key
*
*
* This is equivalent to creating and executing a {@link DeleteObject}
* configured with the {@link Retrier} and r value specified in the
* constructor. If vclock is null a fetch is performed to get it.
*
*
* @param key the key for the object
* @param vclock the vclock for the existing object
* @throws RiakException
*/
public void delete(String key, VClock vclock) throws RiakException
{
final DeleteObject delete = bucket.delete(key).withRetrier(retrier);
if (deleteMeta.hasR()) {
delete.r(deleteMeta.getR());
}
if (deleteMeta.hasPr()) {
delete.pr(deleteMeta.getPr());
}
if (deleteMeta.hasW()) {
delete.w(deleteMeta.getW());
}
if (deleteMeta.hasDw()) {
delete.dw(deleteMeta.getDw());
}
if (deleteMeta.hasPw()) {
delete.pw(deleteMeta.getPw());
}
if (null == vclock)
{
delete.fetchBeforeDelete(true);
}
else
{
delete.vclock(vclock);
}
delete.execute();
}
/**
* Factory method to create a new {@link DomainBucketBuilder} for the given {@link Bucket} and Class.
* @param b
* the Bucket to wrap
* @param clazz the type of object to store/fetch with the new {@link DomainBucket}
* @return a DomainBucketBuilder for the wrapped bucket
*/
public static DomainBucketBuilder builder(Bucket b, Class clazz) {
return new DomainBucketBuilder(b, clazz);
}
}