All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.couchbase.client.java.AsyncScope 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.CoreKeyspace;
import com.couchbase.client.core.annotation.SinceCouchbase;
import com.couchbase.client.core.annotation.Stability;
import com.couchbase.client.core.api.CoreCouchbaseOps;
import com.couchbase.client.core.api.manager.CoreBucketAndScope;
import com.couchbase.client.core.api.query.CoreQueryContext;
import com.couchbase.client.core.api.query.CoreQueryOps;
import com.couchbase.client.core.api.search.CoreSearchOps;
import com.couchbase.client.core.api.search.CoreSearchQuery;
import com.couchbase.client.core.api.search.queries.CoreSearchRequest;
import com.couchbase.client.core.cnc.RequestSpan;
import com.couchbase.client.core.cnc.TracingIdentifiers;
import com.couchbase.client.core.error.CouchbaseException;
import com.couchbase.client.core.error.TimeoutException;
import com.couchbase.client.core.error.context.ReducedAnalyticsErrorContext;
import com.couchbase.client.core.error.context.ReducedQueryErrorContext;
import com.couchbase.client.core.error.context.ReducedSearchErrorContext;
import com.couchbase.client.core.io.CollectionIdentifier;
import com.couchbase.client.core.msg.analytics.AnalyticsRequest;
import com.couchbase.client.core.retry.RetryStrategy;
import com.couchbase.client.core.util.PreventsGarbageCollection;
import com.couchbase.client.java.analytics.AnalyticsAccessor;
import com.couchbase.client.java.analytics.AnalyticsOptions;
import com.couchbase.client.java.analytics.AnalyticsResult;
import com.couchbase.client.java.codec.JsonSerializer;
import com.couchbase.client.java.env.ClusterEnvironment;
import com.couchbase.client.java.json.JsonObject;
import com.couchbase.client.java.manager.eventing.AsyncScopeEventingFunctionManager;
import com.couchbase.client.java.manager.search.AsyncScopeSearchIndexManager;
import com.couchbase.client.java.query.QueryAccessor;
import com.couchbase.client.java.query.QueryOptions;
import com.couchbase.client.java.query.QueryResult;
import com.couchbase.client.java.search.SearchOptions;
import com.couchbase.client.java.search.SearchQuery;
import com.couchbase.client.java.search.SearchRequest;
import com.couchbase.client.java.search.result.SearchResult;
import com.couchbase.client.java.search.vector.VectorSearch;

import java.nio.charset.StandardCharsets;
import java.time.Duration;
import java.util.Map;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;

import static com.couchbase.client.core.util.Golang.encodeDurationToMs;
import static com.couchbase.client.core.util.Validators.notNull;
import static com.couchbase.client.core.util.Validators.notNullOrEmpty;
import static com.couchbase.client.java.ReactiveCluster.DEFAULT_ANALYTICS_OPTIONS;
import static com.couchbase.client.java.ReactiveCluster.DEFAULT_QUERY_OPTIONS;
import static com.couchbase.client.java.ReactiveCluster.DEFAULT_SEARCH_OPTIONS;
import static java.util.Objects.requireNonNull;

/**
 * The scope identifies a group of collections and allows high application
 * density as a result.
 *
 * 

If no scope is explicitly provided, the default scope is used.

* * @since 3.0.0 */ public class AsyncScope { private final CoreCouchbaseOps couchbaseOps; /** * The name of the bucket at which this scope belongs. */ private final String bucketName; /** * The actual name of this scope. */ private final String scopeName; /** * The attached environment to pass on and use. */ private final ClusterEnvironment environment; /** * For executing queries. */ final CoreQueryOps queryOps; final CoreQueryContext queryContext; /** * Stores already opened collections for reuse. */ private final Map collectionCache = new ConcurrentHashMap<>(); /** * Strategy for performing search operations. */ final CoreSearchOps searchOps; @PreventsGarbageCollection private final AsyncCluster cluster; AsyncScope( final String scopeName, final String bucketName, final CoreCouchbaseOps couchbaseOps, final ClusterEnvironment environment, final AsyncCluster cluster ) { this.scopeName = requireNonNull(scopeName); this.bucketName = requireNonNull(bucketName); this.couchbaseOps = requireNonNull(couchbaseOps); this.environment = requireNonNull(environment); this.queryOps = couchbaseOps.queryOps(); queryContext = CoreQueryContext.of(bucketName, scopeName); this.searchOps = couchbaseOps.searchOps(new CoreBucketAndScope(bucketName, name())); this.cluster = requireNonNull(cluster); } /** * The name of the scope. */ public String name() { return scopeName; } /** * The name of the bucket this scope is attached to. */ public String bucketName() { return bucketName; } /** * * Provides access to the underlying {@link Core}. * *

This is advanced API, use with care!

*/ @Stability.Volatile public Core core() { return couchbaseOps.asCore(); } /** * Provides access to the configured {@link ClusterEnvironment} for this scope. */ public ClusterEnvironment environment() { return environment; } /** * Opens the default collection for this scope. * *

Note that this method is package private because it is called from the bucket class only!

* * @return the default collection once opened. */ AsyncCollection defaultCollection() { return maybeCreateAsyncCollection( CollectionIdentifier.DEFAULT_COLLECTION, !scopeName.equals(CollectionIdentifier.DEFAULT_SCOPE) ); } /** * Opens a collection for this scope with an explicit name. * * @param collectionName the collection name. * @return the requested collection if successful. */ public AsyncCollection collection(final String collectionName) { boolean defaultScopeAndCollection = collectionName.equals(CollectionIdentifier.DEFAULT_COLLECTION) && scopeName.equals(CollectionIdentifier.DEFAULT_SCOPE); return maybeCreateAsyncCollection(collectionName, !defaultScopeAndCollection); } /** * Helper method to maybe create a new collection or load it from the cache. * * @param collectionName the name of the collection. * @param refreshMap if the collection map should be refreshed on the config provider. * @return a collection, either from the cache or a freshly populated one. */ private AsyncCollection maybeCreateAsyncCollection(final String collectionName, final boolean refreshMap) { return collectionCache.computeIfAbsent(collectionName, name -> { CoreKeyspace keyspace = new CoreKeyspace(bucketName, scopeName, collectionName); if (refreshMap) { if (couchbaseOps instanceof Core) { ((Core) couchbaseOps) .configurationProvider() .refreshCollectionId(keyspace.toCollectionIdentifier()); } } return new AsyncCollection(keyspace, couchbaseOps, environment, cluster); }); } /** * Performs a N1QL query with default {@link QueryOptions} in a Scope * * @param statement the N1QL query statement. * @return the {@link QueryResult} once the response arrives successfully. * @throws TimeoutException if the operation times out before getting a result. * @throws CouchbaseException for all other error reasons (acts as a base type and catch-all). */ public CompletableFuture query(final String statement) { return query(statement, DEFAULT_QUERY_OPTIONS); } /** * Performs a N1QL query with custom {@link QueryOptions} in a Scope. * * @param statement the N1QL query statement as a raw string. * @param options the custom options for this query. * @return the {@link QueryResult} once the response arrives successfully. */ public CompletableFuture query(final String statement, final QueryOptions options) { notNull(options, "QueryOptions", () -> new ReducedQueryErrorContext(statement)); final QueryOptions.Built opts = options.build(); JsonSerializer serializer = opts.serializer() == null ? environment.jsonSerializer() : opts.serializer(); return queryOps.queryAsync(statement, opts, queryContext, null, QueryAccessor::convertCoreQueryError) .thenApply(r -> new QueryResult(r, serializer)); } /** * Performs an Analytics query with default {@link AnalyticsOptions}. * * @param statement the Analytics query statement as a raw string. * @return the {@link AnalyticsResult} once the response arrives successfully. */ public CompletableFuture analyticsQuery(final String statement) { return analyticsQuery(statement, DEFAULT_ANALYTICS_OPTIONS); } /** * Performs an Analytics query with custom {@link AnalyticsOptions}. * * @param statement the Analytics query statement as a raw string. * @param options the custom options for this analytics query. * @return the {@link AnalyticsResult} once the response arrives successfully. */ public CompletableFuture analyticsQuery(final String statement, final AnalyticsOptions options) { notNull(options, "AnalyticsOptions", () -> new ReducedAnalyticsErrorContext(statement)); AnalyticsOptions.Built opts = options.build(); JsonSerializer serializer = opts.serializer() == null ? environment.jsonSerializer() : opts.serializer(); return AnalyticsAccessor.analyticsQueryAsync(core(), analyticsRequest(statement, opts), serializer); } /** * Helper method to craft an analytics request. * * @param statement the statement to use. * @param opts the built analytics options. * @return the created analytics request. */ AnalyticsRequest analyticsRequest(final String statement, final AnalyticsOptions.Built opts) { notNullOrEmpty(statement, "Statement", () -> new ReducedAnalyticsErrorContext(statement)); Duration timeout = opts.timeout().orElse(environment.timeoutConfig().analyticsTimeout()); RetryStrategy retryStrategy = opts.retryStrategy().orElse(environment.retryStrategy()); JsonObject query = JsonObject.create(); query.put("statement", statement); query.put("timeout", encodeDurationToMs(timeout)); query.put("query_context", AnalyticsRequest.queryContext(bucketName, scopeName)); opts.injectParams(query); final byte[] queryBytes = query.toString().getBytes(StandardCharsets.UTF_8); final String clientContextId = query.getString("client_context_id"); final RequestSpan span = couchbaseOps.coreResources() .requestTracer() .requestSpan(TracingIdentifiers.SPAN_REQUEST_ANALYTICS, opts.parentSpan().orElse(null)); AnalyticsRequest request = new AnalyticsRequest(timeout, core().context(), retryStrategy, core().context().authenticator(), queryBytes, opts.priority(), opts.readonly(), clientContextId, statement, span, bucketName, scopeName ); request.context().clientContext(opts.clientContext()); return request; } /** * Performs a request against the Full Text Search (FTS) service, with default {@link SearchOptions}. *

* This can be used to perform a traditional FTS query, and/or a vector search. *

* This method is for scoped FTS indexes. For global indexes, use {@link AsyncCluster} instead. * * @param searchRequest the request, in the form of a {@link SearchRequest} * @return the {@link SearchResult} once the response arrives successfully, inside a {@link CompletableFuture} * @throws TimeoutException if the operation times out before getting a result. * @throws CouchbaseException for all other error reasons (acts as a base type and catch-all). */ @SinceCouchbase("7.6") public CompletableFuture search(final String indexName, final SearchRequest searchRequest) { return search(indexName, searchRequest, DEFAULT_SEARCH_OPTIONS); } /** * Performs a request against the Full Text Search (FTS) service, with custom {@link SearchOptions}. *

* This can be used to perform a traditional FTS query, and/or a vector search. *

* This method is for scoped FTS indexes. For global indexes, use {@link AsyncCluster} instead. * * @param searchRequest the request, in the form of a {@link SearchRequest} * @return the {@link SearchResult} once the response arrives successfully, inside a {@link CompletableFuture} * @throws TimeoutException if the operation times out before getting a result. * @throws CouchbaseException for all other error reasons (acts as a base type and catch-all). */ @SinceCouchbase("7.6") public CompletableFuture search(final String indexName, final SearchRequest searchRequest, final SearchOptions options) { notNull(searchRequest, "SearchRequest", () -> new ReducedSearchErrorContext(indexName, null)); notNull(options, "SearchOptions", () -> new ReducedSearchErrorContext(indexName, null)); CoreSearchRequest coreRequest = searchRequest.toCore(); SearchOptions.Built opts = options.build(); JsonSerializer serializer = opts.serializer() == null ? environment.jsonSerializer() : opts.serializer(); return searchOps.searchAsync(indexName, coreRequest, opts) .thenApply(r -> new SearchResult(r, serializer)); } /** * Performs a Full Text Search (FTS) query with default {@link SearchOptions}. *

* This method is for scoped FTS indexes. For global indexes, use {@link AsyncCluster} instead. *

* New users should consider the newer {@link #search(String, SearchRequest)} interface instead, which can do both the traditional FTS {@link SearchQuery} that this method performs, * and/or can also be used to perform a {@link VectorSearch}. * * @param query the query, in the form of a {@link SearchQuery} * @return the {@link SearchResult} once the response arrives successfully, inside a {@link CompletableFuture} */ public CompletableFuture searchQuery(final String indexName, final SearchQuery query) { return searchQuery(indexName, query, DEFAULT_SEARCH_OPTIONS); } /** * Performs a Full Text Search (FTS) query with custom {@link SearchOptions}. *

* This method is for scoped FTS indexes. For global indexes, use {@link AsyncCluster} instead. *

* New users should consider the newer {@link #search(String, SearchRequest)} interface instead, which can do both the traditional FTS {@link SearchQuery} that this method performs, * and/or can also be used to perform a {@link VectorSearch}. * * @param query the query, in the form of a {@link SearchQuery} * @param options the custom options for this query. * @return the {@link SearchResult} once the response arrives successfully, inside a {@link CompletableFuture} */ @Stability.Volatile public CompletableFuture searchQuery(final String indexName, final SearchQuery query, final SearchOptions options) { notNull(query, "SearchQuery", () -> new ReducedSearchErrorContext(indexName, null)); CoreSearchQuery coreQuery = query.toCore(); notNull(options, "SearchOptions", () -> new ReducedSearchErrorContext(indexName, coreQuery)); SearchOptions.Built opts = options.build(); JsonSerializer serializer = opts.serializer() == null ? environment.jsonSerializer() : opts.serializer(); return searchOps.searchQueryAsync(indexName, coreQuery, opts) .thenApply(r -> new SearchResult(r, serializer)); } /** * Allows managed scope FTS indexes. */ @SinceCouchbase("7.6") public AsyncScopeSearchIndexManager searchIndexes() { return new AsyncScopeSearchIndexManager(couchbaseOps, this, cluster); } /** * Provides access to the eventing function management services for functions in this scope. */ @Stability.Volatile @SinceCouchbase("7.1") public AsyncScopeEventingFunctionManager eventingFunctions() { return new AsyncScopeEventingFunctionManager(core(), this.cluster, new CoreBucketAndScope(bucketName, name())); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy