com.couchbase.client.java.AsyncBucket Maven / Gradle / Ivy
/*
* Copyright (c) 2018 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;
import com.couchbase.client.core.Core;
import com.couchbase.client.core.annotation.Stability;
import com.couchbase.client.core.api.CoreCouchbaseOps;
import com.couchbase.client.core.cnc.RequestSpan;
import com.couchbase.client.core.cnc.TracingIdentifiers;
import com.couchbase.client.core.diagnostics.ClusterState;
import com.couchbase.client.core.diagnostics.HealthPinger;
import com.couchbase.client.core.diagnostics.PingResult;
import com.couchbase.client.core.error.context.ReducedViewErrorContext;
import com.couchbase.client.core.io.CollectionIdentifier;
import com.couchbase.client.core.msg.view.ViewRequest;
import com.couchbase.client.core.retry.RetryStrategy;
import com.couchbase.client.core.util.PreventsGarbageCollection;
import com.couchbase.client.java.codec.JsonSerializer;
import com.couchbase.client.java.diagnostics.PingOptions;
import com.couchbase.client.java.diagnostics.WaitUntilReadyOptions;
import com.couchbase.client.java.env.ClusterEnvironment;
import com.couchbase.client.java.manager.collection.AsyncCollectionManager;
import com.couchbase.client.java.manager.view.AsyncViewIndexManager;
import com.couchbase.client.java.view.ViewAccessor;
import com.couchbase.client.java.view.ViewOptions;
import com.couchbase.client.java.view.ViewResult;
import java.nio.charset.StandardCharsets;
import java.time.Duration;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import static com.couchbase.client.core.util.Validators.notNull;
import static com.couchbase.client.core.util.Validators.notNullOrEmpty;
import static com.couchbase.client.java.ReactiveBucket.DEFAULT_VIEW_OPTIONS;
import static com.couchbase.client.java.ReactiveCluster.DEFAULT_PING_OPTIONS;
import static com.couchbase.client.java.ReactiveCluster.DEFAULT_WAIT_UNTIL_READY_OPTIONS;
import static java.util.Objects.requireNonNull;
/**
* Provides access to a Couchbase bucket in an async fashion.
*/
public class AsyncBucket {
/**
* The name of the bucket.
*/
private final String name;
/**
* The underlying attached environment.
*/
private final ClusterEnvironment environment;
private final CoreCouchbaseOps couchbaseOps;
@PreventsGarbageCollection
private final AsyncCluster cluster;
/**
* Stores already opened scopes for reuse.
*/
private final Map scopeCache = new ConcurrentHashMap<>();
AsyncBucket(
final String name,
final CoreCouchbaseOps couchbaseOps,
final ClusterEnvironment environment,
final AsyncCluster cluster
) {
this.couchbaseOps = requireNonNull(couchbaseOps);
this.environment = requireNonNull(environment);
this.name = requireNonNull(name);
this.cluster = requireNonNull(cluster);
}
/**
* Returns the name of the {@link AsyncBucket}.
*/
public String name() {
return name;
}
/**
* Returns the attached {@link ClusterEnvironment}.
*/
public ClusterEnvironment environment() {
return environment;
}
/**
* Provides access to the underlying {@link Core}.
*
* This is advanced API, use with care!
*/
@Stability.Volatile
public Core core() {
return couchbaseOps.asCore();
}
public AsyncCollectionManager collections() {
return new AsyncCollectionManager(couchbaseOps.collectionManager(name), cluster);
}
public AsyncViewIndexManager viewIndexes() {
return new AsyncViewIndexManager(core(), name, cluster);
}
/**
* Opens the {@link AsyncScope} with the given name.
*
* @param name the name of the scope.
* @return the {@link AsyncScope} once opened.
*/
public AsyncScope scope(final String name) {
return maybeCreateAsyncScope(name);
}
/**
* Opens the default {@link AsyncScope}.
*
* @return the {@link AsyncScope} once opened.
*/
public AsyncScope defaultScope() {
return maybeCreateAsyncScope(CollectionIdentifier.DEFAULT_SCOPE);
}
/**
* Helper method to create the scope or load it from the cache if present.
*
* @param scopeName the name of the scope.
* @return the created or cached scope.
*/
private AsyncScope maybeCreateAsyncScope(final String scopeName) {
return scopeCache.computeIfAbsent(scopeName, ignored -> new AsyncScope(scopeName, name, couchbaseOps, environment, cluster));
}
/**
* Opens the default collection for this {@link AsyncBucket} using the default scope.
*
* This method does not block and the client will try to establish all needed resources in the background. If you
* need to eagerly await until all resources are established before performing an operation, use the
* {@link #waitUntilReady(Duration)} method on the {@link AsyncBucket}.
*
* @return the opened default {@link AsyncCollection}.
*/
public AsyncCollection defaultCollection() {
return defaultScope().defaultCollection();
}
/**
* Provides access to the collection with the given name for this {@link AsyncBucket} using the default scope.
*
* This method does not block and the client will try to establish all needed resources in the background. If you
* need to eagerly await until all resources are established before performing an operation, use the
* {@link #waitUntilReady(Duration)} method on the {@link AsyncBucket}.
*
* @return the opened named {@link AsyncCollection}.
*/
public AsyncCollection collection(final String collectionName) {
return defaultScope().collection(collectionName);
}
public CompletableFuture viewQuery(final String designDoc, final String viewName) {
return viewQuery(designDoc, viewName, DEFAULT_VIEW_OPTIONS);
}
public CompletableFuture viewQuery(final String designDoc, final String viewName, final ViewOptions options) {
notNull(options, "ViewOptions", () -> new ReducedViewErrorContext(designDoc, viewName, name));
ViewOptions.Built opts = options.build();
JsonSerializer serializer = opts.serializer() == null ? environment.jsonSerializer() : opts.serializer();
return ViewAccessor.viewQueryAsync(core(), viewRequest(designDoc, viewName, opts), serializer);
}
ViewRequest viewRequest(final String designDoc, final String viewName, final ViewOptions.Built opts) {
notNullOrEmpty(designDoc, "DesignDoc", () -> new ReducedViewErrorContext(designDoc, viewName, name));
notNullOrEmpty(viewName, "ViewName", () -> new ReducedViewErrorContext(designDoc, viewName, name));
String query = opts.query();
Optional keysJson = Optional.ofNullable(opts.keys()).map(s -> s.getBytes(StandardCharsets.UTF_8));
boolean development = opts.development();
Duration timeout = opts.timeout().orElse(environment.timeoutConfig().viewTimeout());
RetryStrategy retryStrategy = opts.retryStrategy().orElse(environment.retryStrategy());
final RequestSpan span = environment()
.requestTracer()
.requestSpan(TracingIdentifiers.SPAN_REQUEST_VIEWS, opts.parentSpan().orElse(null));
ViewRequest request = new ViewRequest(timeout, core().context(), retryStrategy, core().context().authenticator(), name, designDoc,
viewName, query, keysJson, development, span);
request.context().clientContext(opts.clientContext());
return request;
}
/**
* Performs application-level ping requests against services in the couchbase cluster.
*
* @return the {@link PingResult} once complete.
*/
public CompletableFuture ping() {
return ping(DEFAULT_PING_OPTIONS);
}
/**
* Performs application-level ping requests with custom options against services in the couchbase cluster.
*
* @return the {@link PingResult} once complete.
*/
public CompletableFuture ping(final PingOptions options) {
notNull(options, "PingOptions");
final PingOptions.Built opts = options.build();
return HealthPinger.ping(
core(),
opts.timeout(),
opts.retryStrategy().orElse(environment.retryStrategy()),
opts.serviceTypes(),
opts.reportId(),
Optional.of(name)
).toFuture();
}
/**
* Waits until the desired {@link ClusterState} is reached.
*
* This method will wait until either the cluster state is "online", or the timeout is reached. Since the SDK is
* bootstrapping lazily, this method allows to eagerly check during bootstrap if all of the services are online
* and usable before moving on.
*
* @param timeout the maximum time to wait until readiness.
* @return a completable future that completes either once ready or timeout.
*/
public CompletableFuture waitUntilReady(final Duration timeout) {
return waitUntilReady(timeout, DEFAULT_WAIT_UNTIL_READY_OPTIONS);
}
/**
* Waits until the desired {@link ClusterState} is reached.
*
* This method will wait until either the cluster state is "online" by default, or the timeout is reached. Since the
* SDK is bootstrapping lazily, this method allows to eagerly check during bootstrap if all of the services are online
* and usable before moving on. You can tune the properties through {@link WaitUntilReadyOptions}.
*
* @param timeout the maximum time to wait until readiness.
* @param options the options to customize the readiness waiting.
* @return a completable future that completes either once ready or timeout.
*/
public CompletableFuture waitUntilReady(final Duration timeout, final WaitUntilReadyOptions options) {
notNull(options, "WaitUntilReadyOptions");
final WaitUntilReadyOptions.Built opts = options.build();
return couchbaseOps.waitUntilReady(opts.serviceTypes(), timeout, opts.desiredState(), name);
}
}