Query idempotence in DataStax drivers

A CQL query is idempotent if it can be applied multiple times without changing the result of the initial application.

Example: Idempotent and non-idempotent queries

For example, the following query is idempotent because no matter how many times it is executed, list_col will always end up with the value [1]:

UPDATE my_table SET list_col = [1] WHERE pk = 1

In contrast, the following query is not idempotent because, if list_col was initially empty, it will contain [1] after the first execution, then [1, 1] after the second execution, and so on.

UPDATE my_table SET list_col = [1] + list_col WHERE pk = 1

Idempotence is important for the following reasons:

With few exceptions, DataStax drivers can’t infer idempotency because drivers don’t parse CQL query strings.

You must explicitly mark statements as idempotent in your application code. Otherwise, the driver treats the statements as non-idempotent.

The Java driver’s query builder is one of the few exceptions where the driver can infer idempotency.

Set idempotence in DataStax drivers

By default, all DataStax drivers consider queries to be non-idempotent.

You must mark queries as idempotent if you want to leverage features such as retry policies and speculative execution.

For more information, see the following driver documentation on marking queries as idempotent:

Non-idempotent queries

To be able to retry CQL statements consistently, avoid non-idempotent queries, such as the following:

Instead, use statements that bind exact values generated by the application to ensure idempotent data structures. For example, you can use maps and sets instead of lists and counters, which are non-idempotent.

Current time values

Functions that produce values based on the current time aren’t idempotent.

For example, the following query is not idempotent because now() produces a value based on the current time on the coordinator responsible for handling the request. Therefore, each invocation of this statement can produce a different value.

UPDATE my_table SET v = now() WHERE pk = 1

Counters

Counter updates aren’t idempotent because each application of the counter update changes the accumulated value of the counter:

UPDATE my_table SET counter_value = counter_value + 1 WHERE pk = 1;

Lists

Prepend, append, and delete operations on lists aren’t idempotent because the element’s position in the list can change depending on the state of the list when the query is invoked.

Prepend and append operations add elements to the list on each invocation. Delete operations remove elements from the list at a position that can change depending on the state of the list when the query is invoked.

UPDATE my_table SET list_col = [1] + list_col WHERE pk = 1

Sets, maps, and tuples

Update, insert, and delete operations on sets, maps, and tuples can be idempotent, but not always. For example:

  • Appending maps with + is non-idempotent.

  • Appending or removing set elements in non-primary key updates within batches are potentially non-idempotent.

  • Individual field updates in tuples are potentially non-idempotent.

DELETE without WHERE

DELETE without WHERE (removing all rows) is non-idempotent because it depends on the number of rows that exist at the time the request is made.

LWTs where linearizability matters

Treat lightweight transactions (LWTs) as non-idempotent if linearizability is a concern.

For example, if the following statement is executed twice, the IF condition will fail on the second execution. The second execution does nothing, and v still has the value 4.

UPDATE my_table SET v = 4 WHERE k = 1 IF v = 1

The problem appears when multiple clients execute the query with retries enabled. For example:

  1. v has the value 1.

  2. Client 1 executes the preceding query, performing a compare and set operation from 1 to 4.

  3. Client 1’s connection drops, but the query completes successfully. v now has the value 4.

  4. Client 2 executes a compare and set operation on v from 4 to 2.

  5. Client 2’s transaction succeeds. v now has the value 2.

  6. Since Client 1 lost its connection, it considers the query as failed, and transparently retries the compare and set operation on v from 1 to 4. Because v now has a value of 2, it receives a not applied response.

Linearizability is an important aspect of lightweight transactions (LWTs). Given a set of concurrent operations on a column from different clients, there must be a way to reorder them to yield a sequential history that is correct. In the above example, from the client’s point of view there were two operations.

  • Client 1 executed a compare and set operation on v from 1 to 4 that was not applied.

  • Client 2 executed a compare and set operation on v from 4 to 2 that was applied.

Overall, the column changed from 1 to 2. There is no ordering of the two operations that can explain the change, and linearizability was broken by the transparent retry at step 6.

Was this helpful?

Give Feedback

How can we improve the documentation?

© 2025 DataStax | Privacy policy | Terms of use | Manage Privacy Choices

Apache, Apache Cassandra, Cassandra, Apache Tomcat, Tomcat, Apache Lucene, Apache Solr, Apache Hadoop, Hadoop, Apache Pulsar, Pulsar, Apache Spark, Spark, Apache TinkerPop, TinkerPop, Apache Kafka and Kafka are either registered trademarks or trademarks of the Apache Software Foundation or its subsidiaries in Canada, the United States and/or other countries. Kubernetes is the registered trademark of the Linux Foundation.

General Inquiries: +1 (650) 389-6000, info@datastax.com