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:
*
* - CREATE INDEX
* - DROP INDEX
* - INSERT
* - MERGE
* - UPDATE
* - UPSERT
* - DELETE
*
*
* @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;
}
}
}