org.gridgain.grid.cache.query.GridCacheQuery Maven / Gradle / Ivy
/*
Copyright (C) GridGain Systems. All Rights Reserved.
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 org.gridgain.grid.cache.query;
import org.gridgain.grid.*;
import org.gridgain.grid.cache.*;
import org.gridgain.grid.cache.affinity.*;
import org.gridgain.grid.lang.*;
import org.gridgain.grid.spi.indexing.h2.*;
import org.jetbrains.annotations.*;
/**
* Main API for configuring and executing cache queries.
*
* Cache queries are created from {@link GridCacheQueries} API via any of the available
* {@code createXXXQuery(...)} methods.
*
SQL Queries
* {@code SQL} query allows to execute distributed cache
* queries using standard SQL syntax. All values participating in where clauses
* or joins must be annotated with {@link GridCacheQuerySqlField} annotation. Query can be created
* with {@link GridCacheQueries#createSqlQuery(Class, String)} method.
* Field Queries
* By default {@code select} clause is ignored as query result contains full objects.
* If it is needed to select individual fields, use {@link GridCacheQueries#createSqlFieldsQuery(String)} method.
* This type of query replaces full objects with individual fields. Note that selected fields
* must be annotated with {@link GridCacheQuerySqlField} annotation.
* Cross-Cache Queries
* You are allowed to query data from several caches. Cache that this query was created on is
* treated as default schema in this case. Other caches can be referenced by their names.
*
* Note that cache name is case sensitive and has to always be specified in double quotes.
* Here is an example of cross cache query (note that 'replicated' and 'partitioned' are
* cache names for replicated and partitioned caches accordingly):
*
* GridCacheQuery<Map.Entry<Integer, FactPurchase>> storePurchases = cache.queries().createSqlQuery(
* Purchase.class,
* "from \"replicated\".Store, \"partitioned\".Purchase where Store.id=Purchase.storeId and Store.id=?");
*
* Custom functions in SQL queries.
* It is possible to write custom Java methods and call then form SQL queries. These methods must be public static
* and annotated with {@link GridCacheQuerySqlFunction}. Classes containing these methods must be registered in
* {@link GridH2IndexingSpi#setIndexCustomFunctionClasses(Class[])}.
* Full Text Queries
* GridGain supports full text queries based on Apache Lucene engine. This queries are created by
* {@link GridCacheQueries#createFullTextQuery(Class, String)} method. Note that all fields that
* are expected to show up in text query results must be annotated with {@link GridCacheQueryTextField}
* annotation.
* Scan Queries
* Sometimes when it is known in advance that SQL query will cause a full data scan, or whenever data set
* is relatively small, the full scan query may be used. This query will iterate over all cache
* entries, skipping over entries that don't pass the optionally provided key-value filter
* (see {@link GridCacheQueries#createScanQuery(GridBiPredicate)} method).
* Limitations
* Data in GridGain cache is usually distributed across several nodes,
* so some queries may not work as expected. Keep in mind following limitations
* (not applied if data is queried from one node only):
*
* -
* {@code Group by} and {@code sort by} statements are applied separately
* on each node, so result set will likely be incorrectly grouped or sorted
* after results from multiple remote nodes are grouped together.
*
* -
* Aggregation functions like {@code sum}, {@code max}, {@code avg}, etc.
* are also applied on each node. Therefore you will get several results
* containing aggregated values, one for each node.
*
* -
* Joins will work correctly only if joined objects are stored in
* collocated mode or at least one side of the join is stored in
* {@link GridCacheMode#REPLICATED} cache. Refer to
* {@link GridCacheAffinityKey} javadoc for more information about colocation.
*
*
* Query usage
* As an example, suppose we have data model consisting of {@code 'Employee'} and {@code 'Organization'}
* classes defined as follows:
*
* public class Organization {
* @GridCacheQuerySqlField(unique = true)
* private long id;
*
* @GridCacheQuerySqlField
* private String name;
* ...
* }
*
* public class Person {
* // Unique index.
* @GridCacheQuerySqlField(unique=true)
* private long id;
*
* @GridCacheQuerySqlField
* private long orgId; // Organization ID.
*
* // Not indexed.
* private String name;
*
* // Non-unique index.
* @GridCacheQuerySqlField
* private double salary;
*
* // Index for text search.
* @GridCacheQueryTextField
* private String resume;
* ...
* }
*
* Then you can create and execute queries that check various salary ranges like so:
*
* GridCache<Long, Person> cache = G.grid().cache();
* ...
* // Create query which selects salaries based on range for all employees
* // that work for a certain company.
* GridCacheQuery<Map.Entry<Long, Person>> qry = cache.queries().createSqlQuery(Person.class,
* "from Person, Organization where Person.orgId = Organization.id " +
* "and Organization.name = ? and Person.salary > ? and Person.salary <= ?");
*
* // Query all nodes to find all cached GridGain employees
* // with salaries less than 1000.
* qry.execute("GridGain", 0, 1000);
*
* // Query only remote nodes to find all remotely cached GridGain employees
* // with salaries greater than 1000 and less than 2000.
* qry.projection(grid.remoteProjection()).execute("GridGain", 1000, 2000);
*
* Here is a possible query that will use Lucene text search to scan all resumes to
* check if employees have {@code Master} degree:
*
* GridCacheQuery<Map.Entry<Long, Person>> mastersQry =
* cache.queries().createFullTextQuery(Person.class, "Master");
*
* // Query all cache nodes.
* mastersQry.execute();
*
*/
public interface GridCacheQuery {
/** Default query page size. */
public static final int DFLT_PAGE_SIZE = 1024;
/**
* Sets result page size. If not provided, {@link #DFLT_PAGE_SIZE} will be used.
* Results are returned from queried nodes one page at a tme.
*
* @param pageSize Page size.
* @return {@code this} query instance for chaining.
*/
public GridCacheQuery pageSize(int pageSize);
/**
* Sets query timeout. {@code 0} means there is no timeout (this
* is a default value).
*
* @param timeout Query timeout.
* @return {@code this} query instance for chaining.
*/
public GridCacheQuery timeout(long timeout);
/**
* Sets whether or not to keep all query results local. If not - only the current page
* is kept locally. Default value is {@code true}.
*
* @param keepAll Keep results or not.
* @return {@code this} query instance for chaining.
*/
public GridCacheQuery keepAll(boolean keepAll);
/**
* Sets whether or not to include backup entries into query result. This flag
* is {@code false} by default.
*
* @param incBackups Query {@code includeBackups} flag.
* @return {@code this} query instance for chaining.
*/
public GridCacheQuery includeBackups(boolean incBackups);
/**
* Sets whether or not to deduplicate query result set. If this flag is {@code true}
* then query result will not contain some key more than once even if several nodes
* returned entries with the same keys. Default value is {@code false}.
*
* @param dedup Query {@code enableDedup} flag.
* @return {@code this} query instance for chaining.
*/
public GridCacheQuery enableDedup(boolean dedup);
/**
* Sets optional grid projection to execute this query on.
*
* @param prj Projection.
* @return {@code this} query instance for chaining.
*/
public GridCacheQuery projection(GridProjection prj);
/**
* Executes the query and returns the query future. Caller may decide to iterate
* over the returned future directly in which case the iterator may block until
* the next value will become available, or wait for the whole query to finish
* by calling any of the {@code 'get(..)'} methods on the returned future. If
* {@link #keepAll(boolean)} flag is set to {@code false}, then {@code 'get(..)'}
* methods will only return the last page received, otherwise all pages will be
* accumulated and returned to user as a collection.
*
* Note that if the passed in grid projection is a local node, then query
* will be executed locally without distribution to other nodes.
*
* Also note that query state cannot be changed (clause, timeout etc.), except
* arguments, if this method was called at least once.
*
* @param args Optional arguments.
* @return Future for the query result.
*/
public GridCacheQueryFuture execute(@Nullable Object... args);
/**
* Executes the query the same way as {@link #execute(Object...)} method but reduces result remotely.
*
* @param rmtReducer Remote reducer.
* @param args Optional arguments.
* @return Future for the query result.
*/
public GridCacheQueryFuture execute(GridReducer rmtReducer, @Nullable Object... args);
/**
* Executes the query the same way as {@link #execute(Object...)} method but transforms result remotely.
*
* @param rmtTransform Remote transformer.
* @param args Optional arguments.
* @return Future for the query result.
*/
public GridCacheQueryFuture execute(GridClosure rmtTransform, @Nullable Object... args);
/**
* Gets metrics for this query.
*
* @return Query metrics.
*/
public GridCacheQueryMetrics metrics();
/**
* Resets metrics for this query.
*/
public void resetMetrics();
}