Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.datastax.driver.mapping.Mapper Maven / Gradle / Ivy
/*
* Copyright (C) 2012-2015 DataStax 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.datastax.driver.mapping;
import java.util.*;
import com.google.common.base.Function;
import com.google.common.base.Functions;
import com.google.common.base.Objects;
import com.google.common.base.Preconditions;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import static com.google.common.base.Preconditions.checkArgument;
import com.datastax.driver.core.*;
import com.datastax.driver.core.querybuilder.Delete;
import com.datastax.driver.core.querybuilder.Insert;
import com.datastax.driver.core.querybuilder.QueryBuilder;
import com.datastax.driver.mapping.Mapper.Option.SaveNullFields;
import com.datastax.driver.mapping.annotations.Accessor;
import com.datastax.driver.mapping.annotations.Computed;
import static com.datastax.driver.mapping.Mapper.Option.Type.SAVE_NULL_FIELDS;
/**
* An object handling the mapping of a particular class.
*
* A {@code Mapper} object is obtained from a {@code MappingManager} using the
* {@link MappingManager#mapper} method.
*/
public class Mapper {
private static final Logger logger = LoggerFactory.getLogger(EntityMapper.class);
final MappingManager manager;
final ProtocolVersion protocolVersion;
final Class klass;
final EntityMapper mapper;
final TableMetadata tableMetadata;
// Cache prepared statements for each type of query we use.
private volatile Map preparedQueries = Collections.emptyMap();
private static final Function NOOP = Functions.constant(null);
private volatile EnumMap defaultSaveOptions;
private volatile EnumMap defaultGetOptions;
private volatile EnumMap defaultDeleteOptions;
private static final EnumMap NO_OPTIONS = new EnumMap(Option.Type.class);
final Function mapOneFunction;
final Function mapOneFunctionWithoutAliases;
final Function> mapAllFunctionWithoutAliases;
Mapper(MappingManager manager, Class klass, EntityMapper mapper) {
this.manager = manager;
this.klass = klass;
this.mapper = mapper;
KeyspaceMetadata keyspace = session().getCluster().getMetadata().getKeyspace(mapper.getKeyspace());
this.tableMetadata = keyspace == null ? null : keyspace.getTable(Metadata.quote(mapper.getTable()));
this.protocolVersion = manager.getSession().getCluster().getConfiguration().getProtocolOptions().getProtocolVersion();
this.mapOneFunction = new Function() {
public T apply(ResultSet rs) {
return Mapper.this.mapAliased(rs).one();
}
};
this.mapOneFunctionWithoutAliases = new Function() {
public T apply(ResultSet rs) {
return Mapper.this.map(rs).one();
}
};
this.mapAllFunctionWithoutAliases = new Function>() {
public Result apply(ResultSet rs) {
return Mapper.this.map(rs);
}
};
this.defaultSaveOptions = NO_OPTIONS;
this.defaultGetOptions = NO_OPTIONS;
this.defaultDeleteOptions = NO_OPTIONS;
}
Session session() {
return manager.getSession();
}
PreparedStatement getPreparedQuery(QueryType type, Set> columns, EnumMap options) {
MapperQueryKey pqk = new MapperQueryKey(type, columns, options);
PreparedStatement stmt = preparedQueries.get(pqk);
if (stmt == null) {
synchronized (preparedQueries) {
stmt = preparedQueries.get(pqk);
if (stmt == null) {
String queryString = type.makePreparedQueryString(tableMetadata, mapper, manager, columns, options.values());
logger.debug("Preparing query {}", queryString);
stmt = session().prepare(queryString);
Map newQueries = new HashMap(preparedQueries);
newQueries.put(pqk, stmt);
preparedQueries = newQueries;
}
}
}
return stmt;
}
PreparedStatement getPreparedQuery(QueryType type, EnumMap options) {
return getPreparedQuery(type, Collections.>emptySet(), options);
}
/**
* The {@code TableMetadata} for this mapper.
*
* @return the {@code TableMetadata} for this mapper or {@code null} if keyspace is not set.
*/
public TableMetadata getTableMetadata() {
return tableMetadata;
}
/**
* The {@code MappingManager} managing this mapper.
*
* @return the {@code MappingManager} managing this mapper.
*/
public MappingManager getManager() {
return manager;
}
/**
* Creates a query that can be used to save the provided entity.
*
* This method is useful if you want to setup a number of options (tracing,
* conistency level, ...) of the returned statement before executing it manually
* or need access to the {@code ResultSet} object after execution (to get the
* trace, the execution info, ...), but in other cases, calling {@link #save}
* or {@link #saveAsync} is shorter.
*
* @param entity the entity to save.
* @return a query that saves {@code entity} (based on it's defined mapping).
*/
public Statement saveQuery(T entity) {
return saveQuery(entity, this.defaultSaveOptions);
}
/**
* Creates a query that can be used to save the provided entity.
*
* This method is useful if you want to setup a number of options (tracing,
* conistency level, ...) of the returned statement before executing it manually
* or need access to the {@code ResultSet} object after execution (to get the
* trace, the execution info, ...), but in other cases, calling {@link #save}
* or {@link #saveAsync} is shorter.
* This method allows you to provide a suite of {@link Option} to include in
* the SAVE query. Options currently supported for SAVE are :
*
* Timestamp
* Time-to-live (ttl)
* Consistency level
* Tracing
*
*
* @param entity the entity to save.
* @return a query that saves {@code entity} (based on it's defined mapping).
*/
public Statement saveQuery(T entity, Option... options) {
return saveQuery(entity, toMapWithDefaults(options, this.defaultSaveOptions));
}
private Statement saveQuery(T entity, EnumMap options) {
Map, Object> values = new HashMap, Object>();
boolean saveNullFields = shouldSaveNullFields(options);
for (ColumnMapper cm : mapper.allColumns()) {
Object value = cm.getValue(entity);
if (cm.kind != ColumnMapper.Kind.COMPUTED && (saveNullFields || value != null)) {
values.put(cm, value);
}
}
BoundStatement bs = getPreparedQuery(QueryType.SAVE, values.keySet(), options).bind();
int i = 0;
for (Map.Entry, Object> entry : values.entrySet()) {
ColumnMapper> mapper = entry.getKey();
Object value = entry.getValue();
setObject(bs, i++, value, mapper);
}
if (mapper.writeConsistency != null)
bs.setConsistencyLevel(mapper.writeConsistency);
for (Option opt : options.values()) {
opt.checkValidFor(QueryType.SAVE, manager);
opt.addToPreparedStatement(bs, i++);
}
return bs;
}
private static boolean shouldSaveNullFields(EnumMap options) {
SaveNullFields option = (SaveNullFields)options.get(SAVE_NULL_FIELDS);
return option == null || option.saveNullFields;
}
private static void setObject(BoundStatement bs, int i, Object value, ColumnMapper> mapper) {
TypeCodec customCodec = mapper.getCustomCodec();
if (customCodec != null)
bs.set(i, value, customCodec);
else
bs.set(i, value, mapper.getPivotType());
}
/**
* Save an entity mapped by this mapper.
*
* This method is basically equivalent to: {@code getManager().getSession().execute(saveQuery(entity))}.
*
* @param entity the entity to save.
*/
public void save(T entity) {
session().execute(saveQuery(entity));
}
/**
* Save an entity mapped by this mapper and using special options for save.
* This method allows you to provide a suite of {@link Option} to include in
* the SAVE query. Options currently supported for SAVE are :
*
* Timestamp
* Time-to-live (ttl)
* Consistency level
* Tracing
*
*
* @param entity the entity to save.
* @param options the options object specified defining special options when saving.
*/
public void save(T entity, Option... options) {
session().execute(saveQuery(entity, options));
}
/**
* Save an entity mapped by this mapper asynchronously.
*
* This method is basically equivalent to: {@code getManager().getSession().executeAsync(saveQuery(entity))}.
*
* @param entity the entity to save.
* @return a future on the completion of the save operation.
*/
public ListenableFuture saveAsync(T entity) {
return Futures.transform(session().executeAsync(saveQuery(entity)), NOOP);
}
/**
* Save an entity mapped by this mapper asynchronously using special options for save.
*
* This method is basically equivalent to: {@code getManager().getSession().executeAsync(saveQuery(entity, options))}.
*
* @param entity the entity to save.
* @param options the options object specified defining special options when saving.
* @return a future on the completion of the save operation.
*/
public ListenableFuture saveAsync(T entity, Option... options) {
return Futures.transform(session().executeAsync(saveQuery(entity, options)), NOOP);
}
/**
* Creates a query to fetch entity given its PRIMARY KEY.
*
* The values provided must correspond to the columns composing the PRIMARY
* KEY (in the order of said primary key).
*
* This method is useful if you want to setup a number of options (tracing,
* conistency level, ...) of the returned statement before executing it manually,
* but in other cases, calling {@link #get} or {@link #getAsync} is shorter.
*
* This method allows you to provide a suite of {@link Option} to include in
* the GET query. Options currently supported for GET are :
*
* Consistency level
* Tracing
*
*
* @param objects the primary key of the entity to fetch, or more precisely
* the values for the columns of said primary key in the order of the primary key.
* Can be followed by {@link Option} to include in the DELETE query.
* @return a query that fetch the entity of PRIMARY KEY {@code objects}.
* @throws IllegalArgumentException if the number of value provided differ from
* the number of columns composing the PRIMARY KEY of the mapped class, or if
* at least one of those values is {@code null}.
*/
public Statement getQuery(Object... objects) {
// Order and duplicates matter for primary keys
List pks = new ArrayList();
EnumMap options = new EnumMap(defaultGetOptions);
for (Object o : objects) {
if (o instanceof Option) {
Option option = (Option)o;
options.put(option.type, option);
} else {
pks.add(o);
}
}
return getQuery(pks, options);
}
private Statement getQuery(List primaryKeys, EnumMap options) {
if (primaryKeys.size() != mapper.primaryKeySize())
throw new IllegalArgumentException(String.format("Invalid number of PRIMARY KEY columns provided, %d expected but got %d", mapper.primaryKeySize(), primaryKeys.size()));
BoundStatement bs = getPreparedQuery(QueryType.GET, options).bind();
int i = 0;
for (Object value : primaryKeys) {
ColumnMapper column = mapper.getPrimaryKeyColumn(i);
if (value == null) {
throw new IllegalArgumentException(String.format("Invalid null value for PRIMARY KEY column %s (argument %d)", column.getColumnName(), i));
}
setObject(bs, i++, value, column);
}
if (mapper.readConsistency != null)
bs.setConsistencyLevel(mapper.readConsistency);
for (Option opt : options.values()) {
opt.checkValidFor(QueryType.GET, manager);
opt.addToPreparedStatement(bs, i);
if (opt.isIncludedInQuery())
i++;
}
return bs;
}
/**
* Fetch an entity based on its primary key.
*
* This method is basically equivalent to: {@code map(getManager().getSession().execute(getQuery(objects))).one()}.
*
* @param objects the primary key of the entity to fetch, or more precisely
* the values for the columns of said primary key in the order of the primary key.
* Can be followed by {@link Option} to include in the DELETE query.
* @return the entity fetched or {@code null} if it doesn't exist.
* @throws IllegalArgumentException if the number of value provided differ from
* the number of columns composing the PRIMARY KEY of the mapped class, or if
* at least one of those values is {@code null}.
*/
public T get(Object... objects) {
return mapAliased(session().execute(getQuery(objects))).one();
}
/**
* Fetch an entity based on its primary key asynchronously.
*
* This method is basically equivalent to mapping the result of: {@code getManager().getSession().executeAsync(getQuery(objects))}.
*
* @param objects the primary key of the entity to fetch, or more precisely
* the values for the columns of said primary key in the order of the primary key.
* Can be followed by {@link Option} to include in the DELETE query.
* @return a future on the fetched entity. The return future will yield
* {@code null} if said entity doesn't exist.
* @throws IllegalArgumentException if the number of value provided differ from
* the number of columns composing the PRIMARY KEY of the mapped class, or if
* at least one of those values is {@code null}.
*/
public ListenableFuture getAsync(Object... objects) {
return Futures.transform(session().executeAsync(getQuery(objects)), mapOneFunction);
}
/**
* Creates a query that can be used to delete the provided entity.
*
* This method is a shortcut that extract the PRIMARY KEY from the
* provided entity and call {@link #deleteQuery(Object...)} with it.
* This method allows you to provide a suite of {@link Option} to include in
* the DELETE query. Note : currently, only {@link com.datastax.driver.mapping.Mapper.Option.Timestamp}
* is supported for DELETE queries.
*
* This method is useful if you want to setup a number of options (tracing,
* conistency level, ...) of the returned statement before executing it manually
* or need access to the {@code ResultSet} object after execution (to get the
* trace, the execution info, ...), but in other cases, calling {@link #delete}
* or {@link #deleteAsync} is shorter.
*
* This method allows you to provide a suite of {@link Option} to include in
* the DELETE query. Options currently supported for DELETE are :
*
* Timestamp
* Consistency level
* Tracing
*
*
* @param entity the entity to delete.
* @param options the options to add to the DELETE query.
* @return a query that delete {@code entity} (based on it's defined mapping) with
* provided USING options.
*/
public Statement deleteQuery(T entity, Option... options) {
List pks = new ArrayList();
for (int i = 0; i < mapper.primaryKeySize(); i++) {
pks.add(mapper.getPrimaryKeyColumn(i).getValue(entity));
}
return deleteQuery(pks, toMapWithDefaults(options, defaultDeleteOptions));
}
/**
* Creates a query that can be used to delete the provided entity.
*
* This method is a shortcut that extract the PRIMARY KEY from the
* provided entity and call {@link #deleteQuery(Object...)} with it.
*
* This method is useful if you want to setup a number of options (tracing,
* conistency level, ...) of the returned statement before executing it manually
* or need access to the {@code ResultSet} object after execution (to get the
* trace, the execution info, ...), but in other cases, calling {@link #delete}
* or {@link #deleteAsync} is shorter.
*
* @param entity the entity to delete.
* @return a query that delete {@code entity} (based on it's defined mapping).
*/
public Statement deleteQuery(T entity) {
List pks = new ArrayList();
for (int i = 0; i < mapper.primaryKeySize(); i++) {
pks.add(mapper.getPrimaryKeyColumn(i).getValue(entity));
}
return deleteQuery(pks, defaultDeleteOptions);
}
/**
* Creates a query that can be used to delete an entity given its PRIMARY KEY.
*
* The values provided must correspond to the columns composing the PRIMARY
* KEY (in the order of said primary key). The values can also contain, after
* specifying the primary keys columns, a suite of {@link Option} to include in
* the DELETE query. Note : currently, only {@link com.datastax.driver.mapping.Mapper.Option.Timestamp}
* is supported for DELETE queries.
*
* This method is useful if you want to setup a number of options (tracing,
* conistency level, ...) of the returned statement before executing it manually
* or need access to the {@code ResultSet} object after execution (to get the
* trace, the execution info, ...), but in other cases, calling {@link #delete}
* or {@link #deleteAsync} is shorter.
* This method allows you to provide a suite of {@link Option} to include in
* the DELETE query. Options currently supported for DELETE are :
*
* Timestamp
* Consistency level
* Tracing
*
*
* @param objects the primary key of the entity to delete, or more precisely
* the values for the columns of said primary key in the order of the primary key.
* Can be followed by {@link Option} to include in the DELETE
* query.
* @return a query that delete the entity of PRIMARY KEY {@code primaryKey}.
* @throws IllegalArgumentException if the number of value provided differ from
* the number of columns composing the PRIMARY KEY of the mapped class, or if
* at least one of those values is {@code null}.
*/
public Statement deleteQuery(Object... objects) {
// Order and duplicates matter for primary keys
List pks = new ArrayList();
EnumMap options = new EnumMap(defaultDeleteOptions);
for (Object o : objects) {
if (o instanceof Option) {
Option option = (Option)o;
options.put(option.type, option);
} else {
pks.add(o);
}
}
return deleteQuery(pks, options);
}
private Statement deleteQuery(List primaryKey, EnumMap options) {
if (primaryKey.size() != mapper.primaryKeySize())
throw new IllegalArgumentException(String.format("Invalid number of PRIMARY KEY columns provided, %d expected but got %d", mapper.primaryKeySize(), primaryKey.size()));
BoundStatement bs = getPreparedQuery(QueryType.DEL, options).bind();
if (mapper.writeConsistency != null)
bs.setConsistencyLevel(mapper.writeConsistency);
int i = 0;
for (Option opt : options.values()) {
opt.checkValidFor(QueryType.DEL, manager);
opt.addToPreparedStatement(bs, i);
if (opt.isIncludedInQuery())
i++;
}
int columnNumber = 0;
for (Object value : primaryKey) {
ColumnMapper column = mapper.getPrimaryKeyColumn(columnNumber);
if (value == null) {
throw new IllegalArgumentException(String.format("Invalid null value for PRIMARY KEY column %s (argument %d)", column.getColumnName(), i));
}
setObject(bs, i++, value, column);
columnNumber++;
}
return bs;
}
/**
* Deletes an entity mapped by this mapper.
*
* This method is basically equivalent to: {@code getManager().getSession().execute(deleteQuery(entity))}.
*
* @param entity the entity to delete.
*/
public void delete(T entity) {
session().execute(deleteQuery(entity));
}
/**
* Deletes an entity mapped by this mapper using provided options.
*
* This method is basically equivalent to: {@code getManager().getSession().execute(deleteQuery(entity, options))}.
*
* @param entity the entity to delete.
* @param options the options to add to the DELETE query.
*/
public void delete(T entity, Option... options) {
session().execute(deleteQuery(entity, options));
}
/**
* Deletes an entity mapped by this mapper asynchronously.
*
* This method is basically equivalent to: {@code getManager().getSession().executeAsync(deleteQuery(entity))}.
*
* @param entity the entity to delete.
* @return a future on the completion of the deletion.
*/
public ListenableFuture deleteAsync(T entity) {
return Futures.transform(session().executeAsync(deleteQuery(entity)), NOOP);
}
/**
* Deletes an entity mapped by this mapper asynchronously using provided options.
*
* This method is basically equivalent to: {@code getManager().getSession().executeAsync(deleteQuery(entity, options))}.
*
* @param entity the entity to delete.
* @param options the options to add to the DELETE query.
* @return a future on the completion of the deletion.
*/
public ListenableFuture deleteAsync(T entity, Option... options) {
return Futures.transform(session().executeAsync(deleteQuery(entity, options)), NOOP);
}
/**
* Deletes an entity based on its primary key.
*
* This method is basically equivalent to: {@code getManager().getSession().execute(deleteQuery(objects))}.
*
* @param objects the primary key of the entity to delete, or more precisely
* the values for the columns of said primary key in the order
* of the primary key.Can be followed by {@link Option} to include
* in the DELETE query.
* @throws IllegalArgumentException if the number of value provided differ from
* the number of columns composing the PRIMARY KEY of the mapped class, or if
* at least one of those values is {@code null}.
*/
public void delete(Object... objects) {
session().execute(deleteQuery(objects));
}
/**
* Deletes an entity based on its primary key asynchronously.
*
* This method is basically equivalent to: {@code getManager().getSession().executeAsync(deleteQuery(objects))}.
*
* @param objects the primary key of the entity to delete, or more precisely
* the values for the columns of said primary key in the order
* of the primary key. Can be followed by {@link Option} to include
* in the DELETE query.
* @throws IllegalArgumentException if the number of value provided differ from
* the number of columns composing the PRIMARY KEY of the mapped class, or if
* at least one of those values is {@code null}.
*/
public ListenableFuture deleteAsync(Object... objects) {
return Futures.transform(session().executeAsync(deleteQuery(objects)), NOOP);
}
/**
* Maps the rows from a {@code ResultSet} into the class this is a mapper of.
*
* Use this method to map a result set that was not generated by the mapper (e.g. a result set coming from a
* manual query or an {@link Accessor method}.
* It expects that the result set contains all column mapped in the target class,
* and that they are not aliased. {@link Computed} fields will not be filled in mapped objects.
*
* @param resultSet the {@code ResultSet} to map.
* @return the mapped result set. Note that the returned mapped result set
* will encapsulate {@code resultSet} and so consuming results from this
* returned mapped result set will consume results from {@code resultSet}
* and vice-versa.
*
* @see #mapAliased(ResultSet)
*/
public Result map(ResultSet resultSet) {
return new Result(resultSet, mapper, protocolVersion);
}
/**
* Maps the rows from a {@code ResultSet} into the class this is a mapper of.
*
* Use this method to map a result set coming from the execution of the {@code Statement} returned by
* {@link #getQuery(Object...)}. {@link Computed} fields will be filled in the mapped object.
*
* @param resultSet the {@code ResultSet} to map.
* @return the mapped result set. Note that the returned mapped result set
* will encapsulate {@code resultSet} and so consuming results from this
* returned mapped result set will consume results from {@code resultSet}
* and vice-versa.
*
* @see #map(ResultSet)
*/
public Result mapAliased(ResultSet resultSet) {
return (manager.isCassandraV1)
? map(resultSet) // no aliases
: new Result(resultSet, mapper, protocolVersion, true);
}
/**
* Set the default save {@link Option} for this object mapper, that will be used
* in all save operations unless overridden. Refer to {@link Mapper#save(Object, Option...)})}
* to check available save options.
*
* @param options the options to set. To reset, use {@link Mapper#resetDefaultSaveOptions}.
*/
public void setDefaultSaveOptions(Option... options) {
this.defaultSaveOptions = toMap(options);
}
/**
* Reset the default save options for this object mapper.
*/
public void resetDefaultSaveOptions() {
this.defaultSaveOptions = NO_OPTIONS;
}
/**
* Set the default get {@link Option} for this object mapper, that will be used
* in all get operations unless overridden. Refer to {@link Mapper#get(Object...)} )} to check available
* get options.
*
* @param options the options to set. To reset, use {@link Mapper#resetDefaultGetOptions}.
*/
public void setDefaultGetOptions(Option... options) {
this.defaultGetOptions = toMap(options);
}
/**
* Reset the default save options for this object mapper.
*/
public void resetDefaultGetOptions() {
this.defaultGetOptions = NO_OPTIONS;
}
/**
* Set the default delete {@link Option} for this object mapper, that will be used
* in all delete operations unless overridden. Refer to {@link Mapper#delete(Object...)} )}
* to check available delete options.
*
* @param options the options to set. To reset, use {@link Mapper#resetDefaultDeleteOptions}.
*/
public void setDefaultDeleteOptions(Option... options) {
this.defaultDeleteOptions = toMap(options);
}
/**
* Reset the default delete options for this object mapper.
*/
public void resetDefaultDeleteOptions() {
this.defaultDeleteOptions = NO_OPTIONS;
}
private static EnumMap toMap(Option[] options) {
EnumMap result = new EnumMap(Option.Type.class);
for (Option option : options) {
result.put(option.type, option);
}
return result;
}
private static EnumMap toMapWithDefaults(Option[] options, EnumMap defaults) {
EnumMap result = new EnumMap(defaults);
for (Option option : options) {
result.put(option.type, option);
}
return result;
}
/**
* An option for a mapper operation.
*
* Options can be passed to individual operations:
*
* mapper.save(myObject, Option.ttl(3600));
*
*
* The mapper can also have defaults, that will apply to all operations that do not
* override these particular option:
*
* mapper.setDefaultSaveOptions(Option.ttl(3600));
* mapper.save(myObject);
*
*
*
* See the static methods in this class for available options.
*/
public static abstract class Option {
enum Type {TTL, TIMESTAMP, CL, TRACING, SAVE_NULL_FIELDS}
final Type type;
protected Option(Type type) {
this.type = type;
}
/**
* Creates a new Option object to add time-to-live to a mapper operation. This is
* only valid for save operations.
*
* Note that this option is only available if using {@link ProtocolVersion#V2} or above.
*
* @param ttl the TTL (in seconds).
* @return the option.
*/
public static Option ttl(int ttl) {
return new Ttl(ttl);
}
/**
* Creates a new Option object to add a timestamp to a mapper operation. This is
* only valid for save and delete operations.
*
* Note that this option is only available if using {@link ProtocolVersion#V2} or above.
*
* @param timestamp the timestamp (in microseconds).
* @return the option.
*/
public static Option timestamp(long timestamp) {
return new Timestamp(timestamp);
}
/**
* Creates a new Option object to add a consistency level value to a mapper operation. This
* is valid for save, delete and get operations.
*
* Note that the consistency level can also be defined at the mapper level, as a parameter
* of the {@link com.datastax.driver.mapping.annotations.Table} annotation (this is redundant
* for backward compatibility). This option, whether defined on a specific call or as the
* default, will always take precedence over the annotation.
*
* @param cl the {@link com.datastax.driver.core.ConsistencyLevel} to use for the operation.
* @return the option.
*/
public static Option consistencyLevel(ConsistencyLevel cl) {
return new ConsistencyLevelOption(cl);
}
/**
* Creates a new Option object to enable query tracing for a mapper operation. This
* is valid for save, delete and get operations.
*
* @param enabled whether to enable tracing.
* @return the option.
*/
public static Option tracing(boolean enabled) {
return new Tracing(enabled);
}
/**
* Creates a new Option object to specify whether null entity fields should be included in
* insert queries. This option is valid only for save operations.
*
* If this option is not specified, it defaults to {@code true} (null fields are saved).
*
* @param enabled whether to include null fields in queries.
* @return the option.
*/
public static Option saveNullFields(boolean enabled){
return new SaveNullFields(enabled);
}
public Type getType(){
return this.type;
}
abstract void appendTo(Insert.Options usings);
abstract void appendTo(Delete.Options usings);
abstract void addToPreparedStatement(BoundStatement bs, int i);
abstract void checkValidFor(QueryType qt, MappingManager manager) throws IllegalArgumentException;
abstract boolean isIncludedInQuery();
static class Ttl extends Option {
private int ttlValue;
Ttl(int value) {
super(Type.TTL);
this.ttlValue = value;
}
void appendTo(Insert.Options usings) {
usings.and(QueryBuilder.ttl(QueryBuilder.bindMarker()));
}
@Override
void appendTo(Delete.Options usings) {
throw new UnsupportedOperationException("shouldn't be called");
}
@Override
void addToPreparedStatement(BoundStatement bs, int i) {
bs.setInt(i, this.ttlValue);
}
@Override
void checkValidFor(QueryType qt, MappingManager manager) {
checkArgument(!manager.isCassandraV1, "TTL option requires native protocol v2 or above");
checkArgument(qt == QueryType.SAVE, "TTL option is only allowed in save queries");
}
@Override
boolean isIncludedInQuery() {
return true;
}
}
static class Timestamp extends Option {
private long tsValue;
Timestamp(long value) {
super(Type.TIMESTAMP);
this.tsValue = value;
}
@Override
void appendTo(Insert.Options usings) {
usings.and(QueryBuilder.timestamp(QueryBuilder.bindMarker()));
}
@Override
void appendTo(Delete.Options usings) {
usings.and(QueryBuilder.timestamp(QueryBuilder.bindMarker()));
}
@Override
void checkValidFor(QueryType qt, MappingManager manager) {
checkArgument(!manager.isCassandraV1, "Timestamp option requires native protocol v2 or above");
checkArgument(qt == QueryType.SAVE || qt == QueryType.DEL, "Timestamp option is only allowed in save and delete queries");
}
@Override
void addToPreparedStatement(BoundStatement bs, int i) {
bs.setLong(i, this.tsValue);
}
@Override
boolean isIncludedInQuery() {
return true;
}
}
static class ConsistencyLevelOption extends Option {
private ConsistencyLevel cl;
ConsistencyLevelOption(ConsistencyLevel cl) {
super(Type.CL);
this.cl = cl;
}
@Override
void appendTo(Insert.Options usings) {
throw new UnsupportedOperationException("shouldn't be called");
}
@Override
void appendTo(Delete.Options usings) {
throw new UnsupportedOperationException("shouldn't be called");
}
@Override
void addToPreparedStatement(BoundStatement bs, int i) {
bs.setConsistencyLevel(cl);
}
@Override
void checkValidFor(QueryType qt, MappingManager manager) {
checkArgument(qt == QueryType.SAVE || qt == QueryType.DEL || qt == QueryType.GET,
"Consistency level option is only allowed in save, delete and get queries");
}
@Override
boolean isIncludedInQuery() {
return false;
}
}
static class Tracing extends Option {
private boolean tracing;
Tracing(boolean tracing) {
super(Type.TRACING);
this.tracing = tracing;
}
@Override
void appendTo(Insert.Options usings) {
throw new UnsupportedOperationException("shouldn't be called");
}
@Override
void appendTo(Delete.Options usings) {
throw new UnsupportedOperationException("shouldn't be called");
}
@Override
void addToPreparedStatement(BoundStatement bs, int i) {
if (this.tracing)
bs.enableTracing();
}
@Override
void checkValidFor(QueryType qt, MappingManager manager) {
checkArgument(qt == QueryType.SAVE || qt == QueryType.DEL || qt == QueryType.GET,
"Tracing option is only allowed in save, delete and get queries");
}
@Override
boolean isIncludedInQuery() {
return false;
}
}
static class SaveNullFields extends Option {
private boolean saveNullFields;
SaveNullFields(boolean saveNullFields) {
super(SAVE_NULL_FIELDS);
this.saveNullFields = saveNullFields;
}
@Override
void appendTo(Insert.Options usings) {
throw new UnsupportedOperationException("shouldn't be called");
}
@Override
void appendTo(Delete.Options usings) {
throw new UnsupportedOperationException("shouldn't be called");
}
@Override
void addToPreparedStatement(BoundStatement bs, int i) {
// nothing to do
}
@Override
void checkValidFor(QueryType qt, MappingManager manager) {
checkArgument(qt == QueryType.SAVE, "SaveNullFields option is only allowed in save queries");
}
@Override
boolean isIncludedInQuery() {
return false;
}
}
}
private static class MapperQueryKey {
private final QueryType queryType;
private final EnumSet optionTypes;
private final Set> columns;
MapperQueryKey(QueryType queryType, Set> columnMappers, EnumMap options) {
Preconditions.checkNotNull(queryType);
Preconditions.checkNotNull(options);
Preconditions.checkNotNull(columnMappers);
this.queryType = queryType;
this.columns = columnMappers;
this.optionTypes = EnumSet.noneOf(Option.Type.class);
for (Option opt : options.values()) {
if (opt.isIncludedInQuery())
this.optionTypes.add(opt.type);
}
}
@Override
public boolean equals(Object other) {
if (this == other)
return true;
if (other instanceof MapperQueryKey) {
MapperQueryKey that = (MapperQueryKey)other;
return this.queryType.equals(that.queryType)
&& this.optionTypes.equals(that.optionTypes)
&& this.columns.equals(that.columns);
}
return false;
}
public int hashCode() {
return Objects.hashCode(queryType, optionTypes, columns);
}
}
}