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

com.couchbase.client.java.query.QueryOptions 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.query;

import com.couchbase.client.core.annotation.SinceCouchbase;
import com.couchbase.client.core.annotation.Stability;
import com.couchbase.client.core.annotation.UsedBy;
import com.couchbase.client.core.api.query.CoreQueryOptions;
import com.couchbase.client.core.api.query.CoreQueryProfile;
import com.couchbase.client.core.api.query.CoreQueryScanConsistency;
import com.couchbase.client.core.api.shared.CoreMutationState;
import com.couchbase.client.core.classic.query.ClassicCoreQueryOps;
import com.couchbase.client.core.deps.com.fasterxml.jackson.core.JsonProcessingException;
import com.couchbase.client.core.deps.com.fasterxml.jackson.databind.JsonNode;
import com.couchbase.client.core.deps.com.fasterxml.jackson.databind.node.ArrayNode;
import com.couchbase.client.core.deps.com.fasterxml.jackson.databind.node.ObjectNode;
import com.couchbase.client.core.endpoint.http.CoreCommonOptions;
import com.couchbase.client.core.error.InvalidArgumentException;
import com.couchbase.client.core.json.Mapper;
import com.couchbase.client.core.msg.kv.MutationToken;
import com.couchbase.client.core.transaction.config.CoreSingleQueryTransactionOptions;
import com.couchbase.client.java.Cluster;
import com.couchbase.client.java.CommonOptions;
import com.couchbase.client.java.codec.JsonSerializer;
import com.couchbase.client.java.env.ClusterEnvironment;
import com.couchbase.client.java.json.JsonArray;
import com.couchbase.client.java.json.JsonObject;
import com.couchbase.client.java.kv.MutationResult;
import com.couchbase.client.java.kv.MutationState;
import com.couchbase.client.java.transactions.config.SingleQueryTransactionOptions;
import reactor.util.annotation.Nullable;

import java.time.Duration;
import java.util.HashMap;
import java.util.Map;

import static com.couchbase.client.core.annotation.UsedBy.Project.SPRING_DATA_COUCHBASE;
import static com.couchbase.client.core.util.Validators.notNull;
import static com.couchbase.client.core.util.Validators.notNullOrEmpty;

/**
 * Allows customizing various N1QL query options.
 * 

* This object is NOT thread safe and must be constructed on a single thread and then passed to the * {@link Cluster#query(String, QueryOptions)} method. * * @since 3.0.0 */ public class QueryOptions extends CommonOptions { private boolean adhoc = true; private String clientContextId; private MutationState consistentWith; private Integer maxParallelism; private boolean metrics = false; private JsonObject namedParameters; private Integer pipelineBatch; private Integer pipelineCap; private JsonArray positionalParameters; private CoreQueryProfile profile; private Map raw; private boolean readonly = false; private Duration scanWait; private Integer scanCap; private CoreQueryScanConsistency scanConsistency; private JsonSerializer serializer; private boolean flexIndex = false; private Boolean preserveExpiry = null; private boolean asTransaction = false; private CoreSingleQueryTransactionOptions asTransactionOptions; private Boolean useReplica = null; /** * The options should only be instantiated through the {@link #queryOptions()} static method. */ protected QueryOptions() {} /** * Creates new {@link QueryOptions} with all default params set. * * @return constructed {@link QueryOptions} with its default values. */ public static QueryOptions queryOptions() { return new QueryOptions(); } /** * Allows providing custom JSON key/value pairs for advanced usage. *

* If possible, it is recommended to use other methods on this object to customize the query. The "raw" method should * only be used if no such setter can be found (i.e. if an undocumented property should be set or you are using * an older client and a new server-configuration property has been added to the cluster). *

* Note that the value will be passed through a JSON encoder, so do not provide already encoded JSON as the value. * The value must be one of the following types: *

    *
  • String *
  • Integer *
  • Long *
  • Double *
  • Boolean *
  • BigInteger *
  • BigDecimal *
  • JsonObject *
  • JsonArray *
  • Map (with String keys, and values from this list) *
  • List (with element types from this list) *
* * @param key the parameter name (key of the JSON property) or empty. * @param value the parameter value (value of the JSON property). * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions raw(final String key, final Object value) { notNullOrEmpty(key, "Key"); if (raw == null) { raw = new HashMap<>(); } raw.put(key, value); return this; } /** * Allows turning this request into a prepared statement query. *

* If set to

false
, the SDK will transparently perform "prepare and execute" logic the first time this * query is seen and then subsequently reuse the prepared statement name when sending it to the server. If a query * is executed frequently, this is a good way to speed it up since it will save the server the task of re-parsing * and analyzing the query. *

* If you are using prepared statements, make sure that if certain parts of the query string change you are using * {@link #parameters(JsonObject) named} or {@link #parameters(JsonArray) positional} parameters. If the statement * string itself changes it cannot be cached. * * @param adhoc if set to false this query will be turned into a prepared statement query. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions adhoc(final boolean adhoc) { this.adhoc = adhoc; return this; } /** * Customizes the consistency guarantees for this query. *

* Tuning the scan consistency allows to trade data "freshness" for latency and vice versa. By default * {@link QueryScanConsistency#NOT_BOUNDED} is used, which means that the server returns the data it has in the * index right away. This is fast, but might not include the most recent mutations. If you want to include all * the mutations up to the point of the query, use {@link QueryScanConsistency#REQUEST_PLUS}. *

* Note that you cannot use this method and {@link #consistentWith(MutationState)} at the same time, since they * are mutually exclusive. As a rule of thumb, if you only care to be consistent with the mutation you just wrote * on the same thread/app, use {@link #consistentWith(MutationState)}. If you need "global" scan consistency, use * {@link QueryScanConsistency#REQUEST_PLUS} on this method. * * @param scanConsistency the index scan consistency to be used for this query. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions scanConsistency(final QueryScanConsistency scanConsistency) { notNull(scanConsistency, "QueryScanConsistency"); switch (scanConsistency) { case NOT_BOUNDED: this.scanConsistency = CoreQueryScanConsistency.NOT_BOUNDED; break; case REQUEST_PLUS: this.scanConsistency = CoreQueryScanConsistency.REQUEST_PLUS; break; default: throw new InvalidArgumentException("Unknown scan consistency type " + scanConsistency, null, null); } consistentWith = null; return this; } /** * Provides a custom {@link JsonSerializer} to be used for decoding the rows as they return from the server. *

* If no serializer is provided, the default one from the {@link ClusterEnvironment} will be used (which is * sufficient for most tasks at hand). * * @param serializer a custom serializer for this request. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions serializer(final JsonSerializer serializer) { notNull(serializer, "JsonSerializer"); this.serializer = serializer; return this; } /** * Customizes the server profiling level for this query. *

* Note that you only want to tune this if you want to gather profiling/performance metrics for debugging. Turning * this on in production (depending on the level) will likely have performance impact on the server query engine * as a whole and on this query in particular! *

* This is an Enterprise Edition feature. On Community Edition the parameter will be accepted, but no profiling * information returned. * * @param profile the custom query profile level for this query. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions profile(final QueryProfile profile) { notNull(profile, "QueryProfile"); switch (profile) { case OFF: this.profile = CoreQueryProfile.OFF; break; case PHASES: this.profile = CoreQueryProfile.PHASES; break; case TIMINGS: this.profile = CoreQueryProfile.TIMINGS; break; default: throw new InvalidArgumentException("Unknown profile type " + profile, null, null); } return this; } /** * Supports providing a custom client context ID for this query. *

* If no client context ID is provided by the user, a UUID is generated and sent automatically so by default it is * always possible to identify a query when debugging. If you do not want to send one, pass either null or an empty * string to this method. * * @param clientContextId the client context ID - if null or empty it will not be sent. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions clientContextId(final String clientContextId) { if (clientContextId == null || clientContextId.isEmpty()) { this.clientContextId = null; } else { this.clientContextId = clientContextId; } return this; } /** * Enables per-request metrics in the trailing section of the query. *

* If this method is set to true, the server will send metrics back to the client which are available through the * {@link QueryMetaData#metrics()} section. As opposed to {@link #profile(QueryProfile)}, returning metrics is rather * cheap and can also be enabled in production if needed. * * @param metrics set to true if the server should return simple query metrics. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions metrics(final boolean metrics) { this.metrics = metrics; return this; } /** * Allows customizing how long the query engine is willing to wait until the index catches up to whatever scan * consistency is asked for in this query. *

* Note that if {@link QueryScanConsistency#NOT_BOUNDED} is used, this method doesn't do anything * at all. If no value is provided to this method, the server default is used. * * @param wait the maximum duration the query engine is willing to wait before failing. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions scanWait(final Duration wait) { notNull(wait, "Wait Duration"); if (this.scanConsistency != null && this.scanConsistency == CoreQueryScanConsistency.NOT_BOUNDED) { this.scanWait = null; } else { this.scanWait = wait; } return this; } /** * Allows overriding the default maximum parallelism for the query execution on the server side. *

* If 0 or a negative value is set, the parallelism is disabled. If not provided, the server default will * be used. * * @param maxParallelism the maximum parallelism for this query, 0 or negative values disable it. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions maxParallelism(final int maxParallelism) { this.maxParallelism = maxParallelism; return this; } /** * Allows explicitly marking a query as being readonly and not mutating and documents on the server side. *

* In addition to providing some security in that you are not accidentally modifying data, setting this flag to true * also helps the client to more proactively retry and re-dispatch a query since then it can be sure it is idempotent. * As a result, if your query is readonly then it is a good idea to set this flag. *

* If set to true, then (at least) the following statements are not allowed: *

    *
  1. CREATE INDEX
  2. *
  3. DROP INDEX
  4. *
  5. INSERT
  6. *
  7. MERGE
  8. *
  9. UPDATE
  10. *
  11. UPSERT
  12. *
  13. DELETE
  14. *
* * @param readonly true if readonly should be set, false is the default and will use the server side default. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions readonly(final boolean readonly) { this.readonly = readonly; return this; } /** * Supports customizing the maximum buffered channel size between the indexer and the query service. *

* This is an advanced API and should only be tuned with care. Use 0 or a negative number to disable. * * @param scanCap the scan cap size, use 0 or negative number to disable. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions scanCap(final int scanCap) { this.scanCap = scanCap; return this; } /** * Supports customizing the number of items execution operators can batch for fetch from the KV layer on the server. *

* This is an advanced API and should only be tuned with care. * * @param pipelineBatch the pipeline batch size. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions pipelineBatch(final int pipelineBatch) { this.pipelineBatch = pipelineBatch; return this; } /** * Allows customizing the maximum number of items each execution operator can buffer between various operators on the * server. *

* This is an advanced API and should only be tuned with care. * * @param pipelineCap the pipeline cap size. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions pipelineCap(final int pipelineCap) { this.pipelineCap = pipelineCap; return this; } /** * Sets named parameters for this query. *

* Note that named and positional parameters cannot be used at the same time. If one is set, the other one is "nulled" * out and not being sent to the server. * * @param named {@link JsonObject} the named parameters. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions parameters(final JsonObject named) { notNull(named, "Named Parameters"); this.namedParameters = named; positionalParameters = null; return this; } /** * Sets positional parameters for this query. *

* Note that named and positional parameters cannot be used at the same time. If one is set, the other one is "nulled" * out and not being sent to the server. * * @param positional {@link JsonArray} the positional parameters. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions parameters(final JsonArray positional) { notNull(positional, "Positional Parameters"); this.positionalParameters = positional; namedParameters = null; return this; } /** * Sets the {@link MutationToken}s this query should be consistent with. *

* These mutation tokens are returned from mutations (i.e. as part of a {@link MutationResult}) and if you want your * N1QL query to include those you need to pass the mutation tokens into a {@link MutationState}. *

* Note that you cannot use this method and {@link #scanConsistency(QueryScanConsistency)} at the same time, since * they are mutually exclusive. As a rule of thumb, if you only care to be consistent with the mutation you just wrote * on the same thread/app, use this method. If you need "global" scan consistency, use * {@link QueryScanConsistency#REQUEST_PLUS} on {@link #scanConsistency(QueryScanConsistency)}. * * @param mutationState the mutation state containing the mutation tokens. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions consistentWith(final MutationState mutationState) { notNull(mutationState, "MutationState"); this.consistentWith = mutationState; scanConsistency = null; return this; } /** * Tells the query engine to use a flex index (utilizing the search service). * * @param flexIndex if a flex index should be used, false is the default. * @return the same {@link QueryOptions} for chaining purposes. */ public QueryOptions flexIndex(final boolean flexIndex) { this.flexIndex = flexIndex; return this; } /** * Tells the query engine to preserve expiration values set on any documents modified by this query. *

* This feature works from Couchbase Server 7.1.0 onwards. * * @param preserveExpiry if expiration values should be preserved, the default is false. * @return the same {@link QueryOptions} for chaining purposes. */ @SinceCouchbase("7.1") public QueryOptions preserveExpiry(final boolean preserveExpiry) { this.preserveExpiry = preserveExpiry; return this; } @Stability.Internal public Built build() { return new Built(); } /** * Performs this as a single query transaction, with default options. *

* @return the same {@link QueryOptions} for chaining purposes. */ @SinceCouchbase("7.0") public QueryOptions asTransaction() { asTransaction = true; return this; } /** * Performs this as a single query transaction. *

* @param options configuration options to use. * @return the same {@link QueryOptions} for chaining purposes. */ @SinceCouchbase("7.0") public QueryOptions asTransaction(final SingleQueryTransactionOptions options) { notNull(options, "asTransaction"); asTransaction = true; asTransactionOptions = options.build(); return this; } /** * Allows returning results from replicas. *

* If true, the server is allowed to use data from replicas (which may be stale) when executing the query. * If false, the server must use up-to-date data from primaries. * If null (the default), honor the server-side configuration for this setting. * * @param useReplica if true this query will possible return results from replicas; if false, from primary. * if not set, then according to the server. * @return the same {@link QueryOptions} for chaining purposes. */ @SinceCouchbase("7.6") public QueryOptions useReplica(@Nullable final Boolean useReplica) { this.useReplica = useReplica; return this; } @Stability.Internal public class Built extends BuiltCommonOptions implements CoreQueryOptions { private final CoreCommonOptions common; Built() { common = CoreCommonOptions.ofOptional(timeout(), retryStrategy(), parentSpan()); } public boolean readonly() { return readonly; } @Override public Duration scanWait() { return scanWait; } @Override public Integer scanCap() { return scanCap; } @Override public CoreQueryScanConsistency scanConsistency() { return scanConsistency; } @Override public boolean flexIndex() { return flexIndex; } @Override public Boolean preserveExpiry() { return preserveExpiry; } public JsonSerializer serializer() { return serializer; } @Override public boolean adhoc() { return adhoc; } public String clientContextId() { return clientContextId; } @Override public CoreMutationState consistentWith() { if (consistentWith == null) { return null; } return new CoreMutationState(consistentWith); } @Override public Integer maxParallelism() { return maxParallelism; } @Override public boolean metrics() { return metrics; } @Override public ObjectNode namedParameters() { if (namedParameters == null) { return null; } try { return (ObjectNode) Mapper.reader().readTree(namedParameters.toString()); } catch (JsonProcessingException e) { throw new InvalidArgumentException("Unable to convert named parameters", e, null); } } @Override public Integer pipelineBatch() { return pipelineBatch; } @Override public Integer pipelineCap() { return pipelineCap; } @Override public ArrayNode positionalParameters() { if (positionalParameters == null) { return null; } try { return (ArrayNode) Mapper.reader().readTree(positionalParameters.toString()); } catch (JsonProcessingException e) { throw new InvalidArgumentException("Unable to convert named parameters", e, null); } } @Override public CoreQueryProfile profile() { return profile; } @Override public JsonNode raw() { if (raw == null) { return null; } return QueryOptionsUtil.convert(raw); } @UsedBy(SPRING_DATA_COUCHBASE) // SDC 4 uses this @Stability.Internal public void injectParams(final JsonObject queryJson) { ObjectNode params = ClassicCoreQueryOps.convertOptions(this); @SuppressWarnings("unchecked") Map map = Mapper.convertValue(params, Map.class); map.forEach(queryJson::put); } public boolean asTransaction() { return asTransaction; } public CoreSingleQueryTransactionOptions asTransactionOptions() { return asTransactionOptions; } @Override public CoreCommonOptions commonOptions() { return common; } @Override public Boolean useReplica() { return useReplica; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy