From 66a0bf9dee3438d3fa7f744fa1f244426f3e81aa Mon Sep 17 00:00:00 2001 From: Mark Paluch Date: Mon, 9 Sep 2024 09:43:43 +0200 Subject: [PATCH] Apply configured `CodecRegistry` to `StatementBuilder`. Closes #1114 --- .../data/cassandra/core/StatementFactory.java | 25 +++++--- .../core/cql/util/StatementBuilder.java | 40 +++++++++--- .../core/StatementFactoryUnitTests.java | 61 ++++++++++++++++++- .../cql/util/StatementBuilderUnitTests.java | 54 ++++++++++++++++ 4 files changed, 162 insertions(+), 18 deletions(-) diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/core/StatementFactory.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/core/StatementFactory.java index 9469f911e..8c0144181 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/core/StatementFactory.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/core/StatementFactory.java @@ -238,7 +238,9 @@ public StatementBuilder createSelectAndOrder(List selectors, select = QueryBuilder.selectFrom(getKeyspace(entity, from), from).selectors(mappedSelectors); } - StatementBuilder builder = StatementBuilder.of(select, cassandraConverter.getCodecRegistry()); builder.bind((statement, factory) -> { return statement.where(getRelations(filter, factory)); @@ -758,7 +766,8 @@ private StatementBuilder UpdateStart updateStart = QueryBuilder.update(getKeyspace(entity, table), table); - return StatementBuilder.of((com.datastax.oss.driver.api.querybuilder.update.Update) updateStart) + return StatementBuilder + .of((com.datastax.oss.driver.api.querybuilder.update.Update) updateStart, cassandraConverter.getCodecRegistry()) .bind((statement, factory) -> { com.datastax.oss.driver.api.querybuilder.update.Update statementToUse; @@ -918,7 +927,7 @@ private StatementBuilder delete(List columnNames, Cassand select = select.column(columnName); } - return StatementBuilder.of(select.where()).bind((statement, factory) -> { + return StatementBuilder.of(select.where(), cassandraConverter.getCodecRegistry()).bind((statement, factory) -> { WriteOptions options = optionsOptional.orElse(null); Delete statementToUse; diff --git a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/core/cql/util/StatementBuilder.java b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/core/cql/util/StatementBuilder.java index dee29937f..f2c4640fe 100644 --- a/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/core/cql/util/StatementBuilder.java +++ b/spring-data-cassandra/src/main/java/org/springframework/data/cassandra/core/cql/util/StatementBuilder.java @@ -43,8 +43,8 @@ * Functional builder for Cassandra {@link BuildableQuery statements}. Statements are built by applying * {@link UnaryOperator builder functions} that get applied when {@link #build() building} the actual * {@link SimpleStatement statement}. The {@code StatementBuilder} provides a mutable container for statement creation - * allowing a functional declaration of actions that are necessary to build a statement. This class helps building CQL - * statements as a {@link BuildableQuery} classes are typically immutable and require return value tracking across + * allowing a functional declaration of actions that are necessary to build a statement. This class helps with building + * CQL statements as a {@link BuildableQuery} classes are typically immutable and require return value tracking across * methods that want to apply modifications to a statement. *

* Building a statement consists of three phases: @@ -61,7 +61,11 @@ * The builder can be used for structural evolution and value evolution of statements. Values are bound through * {@link BindFunction binding functions} that accept the statement and a {@link TermFactory}. Values can be bound * inline or through bind markers when {@link #build(ParameterHandling, CodecRegistry) building} the statement. All - * functions are applied in the order of their declaration. + * functions remain in the order of their declaration. + *

+ * {@link ParameterHandling#INLINE Inline} rendering of parameters requires a {@link CodecRegistry}. A StatementBuilder + * can be {@link StatementBuilder#of(BuildableQuery, CodecRegistry) created} by providing a custom CodecRegistry. + * Otherwise, the registry falls back to {@link CodecRegistry#DEFAULT}. *

* All methods returning {@link StatementBuilder} point to the same instance. This class is intended for internal use. * @@ -72,6 +76,7 @@ public class StatementBuilder { private final S statement; + private final CodecRegistry registry; private final List> queryActions = new ArrayList<>(); private final List> onBuild = new ArrayList<>(); @@ -79,7 +84,8 @@ public class StatementBuilder { /** * Factory method used to create a new {@link StatementBuilder} with the given {@link BuildableQuery query stub}. The - * stub is used as base for the built query so each query inherits properties of this stub. + * stub is used as base for the built query so each query inherits properties of this stub. This factory method + * initializes StatementBuilder with the default {@link CodecRegistry#DEFAULT CodecRegistry}. * * @param query type. * @param stub the {@link BuildableQuery query stub} to use. @@ -88,10 +94,27 @@ public class StatementBuilder { * @see com.datastax.oss.driver.api.querybuilder.BuildableQuery */ public static StatementBuilder of(S stub) { + return of(stub, CodecRegistry.DEFAULT); + } + + /** + * Factory method used to create a new {@link StatementBuilder} with the given {@link BuildableQuery query stub}. The + * stub is used as base for the built query so each query inherits properties of this stub. + * + * @param query type. + * @param stub the {@link BuildableQuery query stub} to use. + * @param registry the default {@link CodecRegistry} to use for inline parameter rendering. + * @return a {@link StatementBuilder} for the given {@link BuildableQuery query stub}. + * @throws IllegalArgumentException if the {@link BuildableQuery query stub} is {@literal null}. + * @see com.datastax.oss.driver.api.querybuilder.BuildableQuery + * @since 4.4 + */ + public static StatementBuilder of(S stub, CodecRegistry registry) { Assert.notNull(stub, "Query stub must not be null"); + Assert.notNull(registry, "CodecRegistry stub must not be null"); - return new StatementBuilder<>(stub); + return new StatementBuilder<>(stub, registry); } /** @@ -101,8 +124,9 @@ public static StatementBuilder of(S stub) { * {@link com.datastax.oss.driver.api.core.cql.Statement}. * @see com.datastax.oss.driver.api.querybuilder.BuildableQuery */ - private StatementBuilder(S statement) { + private StatementBuilder(S statement, CodecRegistry registry) { this.statement = statement; + this.registry = registry; } /** @@ -177,7 +201,7 @@ public StatementBuilder transform(UnaryOperator mappingFunct * @return the built {@link SimpleStatement}. */ public SimpleStatement build() { - return build(ParameterHandling.BY_INDEX, CodecRegistry.DEFAULT); + return build(ParameterHandling.BY_INDEX, this.registry); } /** @@ -188,7 +212,7 @@ public SimpleStatement build() { * @return the built {@link SimpleStatement}. */ public SimpleStatement build(ParameterHandling parameterHandling) { - return build(parameterHandling, CodecRegistry.DEFAULT); + return build(parameterHandling, this.registry); } /** diff --git a/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/core/StatementFactoryUnitTests.java b/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/core/StatementFactoryUnitTests.java index 0bdc1e8d9..7cd28f9b0 100644 --- a/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/core/StatementFactoryUnitTests.java +++ b/spring-data-cassandra/src/test/java/org/springframework/data/cassandra/core/StatementFactoryUnitTests.java @@ -19,6 +19,7 @@ import static org.springframework.data.cassandra.core.query.Criteria.*; import static org.springframework.data.domain.Sort.Direction.*; +import java.nio.ByteBuffer; import java.time.Duration; import java.util.Arrays; import java.util.Collections; @@ -29,7 +30,6 @@ import org.junit.jupiter.api.Test; import org.springframework.data.annotation.Id; -import org.springframework.data.cassandra.core.convert.CassandraConverter; import org.springframework.data.cassandra.core.convert.MappingCassandraConverter; import org.springframework.data.cassandra.core.convert.UpdateMapper; import org.springframework.data.cassandra.core.cql.QueryOptions; @@ -47,10 +47,16 @@ import com.datastax.oss.driver.api.core.CqlIdentifier; import com.datastax.oss.driver.api.core.DefaultConsistencyLevel; +import com.datastax.oss.driver.api.core.ProtocolVersion; import com.datastax.oss.driver.api.core.cql.SimpleStatement; +import com.datastax.oss.driver.api.core.type.DataType; +import com.datastax.oss.driver.api.core.type.DataTypes; +import com.datastax.oss.driver.api.core.type.codec.TypeCodec; +import com.datastax.oss.driver.api.core.type.reflect.GenericType; import com.datastax.oss.driver.api.querybuilder.delete.Delete; import com.datastax.oss.driver.api.querybuilder.insert.RegularInsert; import com.datastax.oss.driver.api.querybuilder.select.Select; +import com.datastax.oss.driver.internal.core.type.codec.registry.DefaultCodecRegistry; /** * Unit tests for {@link StatementFactory}. @@ -60,7 +66,7 @@ */ class StatementFactoryUnitTests { - private CassandraConverter converter = new MappingCassandraConverter(); + private MappingCassandraConverter converter = new MappingCassandraConverter(); private UpdateMapper updateMapper = new UpdateMapper(converter); @@ -850,6 +856,53 @@ void shouldCreateCountQuery() { .isEqualTo("SELECT count(1) FROM group WHERE foo='bar'"); } + @Test // GH-1114 + void shouldConsiderCodecRegistry() { + + DefaultCodecRegistry cr = new DefaultCodecRegistry("foo"); + cr.register(new TypeCodec() { + @Override + public GenericType getJavaType() { + return GenericType.of(MyString.class); + } + + @Override + public DataType getCqlType() { + return DataTypes.TEXT; + } + + @Override + public ByteBuffer encode(MyString value, ProtocolVersion protocolVersion) { + return null; + } + + @Override + public MyString decode(ByteBuffer bytes, ProtocolVersion protocolVersion) { + return null; + } + + @Override + public String format(MyString value) { + return "'" + value.value() + "'"; + } + + @Override + public MyString parse(String value) { + return new MyString(value); + } + }); + + converter.setCodecRegistry(cr); + + Query query = Query.query(where("foo").is(new MyString("bar"))); + + StatementBuilder