blob: 983acfaf97b1038df7ff425999621e2011b865bb [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.cassandra.cql3;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import com.google.common.collect.ImmutableList;
import org.junit.BeforeClass;
import org.junit.Test;
import org.apache.cassandra.cql3.statements.BatchStatement;
import org.apache.cassandra.cql3.statements.ModificationStatement;
import org.apache.cassandra.db.ConsistencyLevel;
import org.apache.cassandra.service.ClientState;
import org.apache.cassandra.service.QueryState;
import org.apache.cassandra.transport.ProtocolVersion;
import org.apache.cassandra.transport.messages.ResultMessage;
import static java.lang.String.format;
import static org.junit.Assert.assertEquals;
public class CustomNowInSecondsTest extends CQLTester
{
@BeforeClass
public static void setUpClass()
{
prepareServer();
requireNetwork();
}
@Test
public void testSelectQuery()
{
testSelectQuery(false);
testSelectQuery(true);
}
private void testSelectQuery(boolean prepared)
{
int day = 86400;
String ks = createKeyspace("CREATE KEYSPACE %s WITH replication={ 'class' : 'SimpleStrategy', 'replication_factor' : 1 }");
String tbl = createTable(ks, "CREATE TABLE %s (id int primary key, val int)");
// insert a row with TTL = 1 day.
executeModify(format("INSERT INTO %s.%s (id, val) VALUES (0, 0) USING TTL %d", ks, tbl, day), Integer.MIN_VALUE, prepared);
int now = (int) (System.currentTimeMillis() / 1000);
// execute a SELECT query without overriding nowInSeconds - make sure we observe one row.
assertEquals(1, executeSelect(format("SELECT * FROM %s.%s", ks, tbl), Integer.MIN_VALUE, prepared).size());
// execute a SELECT query with nowInSeconds set to [now + 1 day + 1], when the row should have expired.
assertEquals(0, executeSelect(format("SELECT * FROM %s.%s", ks, tbl), now + day + 1, prepared).size());
}
@Test
public void testModifyQuery()
{
testModifyQuery(false);
testModifyQuery(true);
}
private void testModifyQuery(boolean prepared)
{
int now = (int) (System.currentTimeMillis() / 1000);
int day = 86400;
String ks = createKeyspace("CREATE KEYSPACE %s WITH replication={ 'class' : 'SimpleStrategy', 'replication_factor' : 1 }");
String tbl = createTable(ks, "CREATE TABLE %s (id int primary key, val int)");
// execute an INSERT query with now set to [now + 1 day], with ttl = 1, making its effective ttl = 1 day + 1.
executeModify(format("INSERT INTO %s.%s (id, val) VALUES (0, 0) USING TTL %d", ks, tbl, 1), now + day, prepared);
// verify that despite TTL having passed (if not for nowInSeconds override) the row is still there.
assertEquals(1, executeSelect(format("SELECT * FROM %s.%s", ks, tbl), now + 1, prepared).size());
// jump in time by one day, make sure the row expired
assertEquals(0, executeSelect(format("SELECT * FROM %s.%s", ks, tbl), now + day + 1, prepared).size());
}
@Test
public void testBatchQuery()
{
testBatchQuery(false);
testBatchQuery(true);
}
private void testBatchQuery(boolean prepared)
{
int now = (int) (System.currentTimeMillis() / 1000);
int day = 86400;
String ks = createKeyspace("CREATE KEYSPACE %s WITH replication={ 'class' : 'SimpleStrategy', 'replication_factor' : 1 }");
String tbl = createTable(ks, "CREATE TABLE %s (id int primary key, val int)");
// execute an BATCH query with now set to [now + 1 day], with ttl = 1, making its effective ttl = 1 day + 1.
String batch = format("BEGIN BATCH " +
"INSERT INTO %s.%s (id, val) VALUES (0, 0) USING TTL %d; " +
"INSERT INTO %s.%s (id, val) VALUES (1, 1) USING TTL %d; " +
"APPLY BATCH;",
ks, tbl, 1,
ks, tbl, 1);
executeModify(batch, now + day, prepared);
// verify that despite TTL having passed at now + 1 the rows are still there.
assertEquals(2, executeSelect(format("SELECT * FROM %s.%s", ks, tbl), now + 1, prepared).size());
// jump in time by one day, make sure the row expired.
assertEquals(0, executeSelect(format("SELECT * FROM %s.%s", ks, tbl), now + day + 1, prepared).size());
}
@Test
public void testBatchMessage()
{
// test BatchMessage path
int now = (int) (System.currentTimeMillis() / 1000);
int day = 86400;
String ks = createKeyspace("CREATE KEYSPACE %s WITH replication={ 'class' : 'SimpleStrategy', 'replication_factor' : 1 }");
String tbl = createTable(ks, "CREATE TABLE %s (id int primary key, val int)");
List<String> queries = ImmutableList.of(
format("INSERT INTO %s.%s (id, val) VALUES (0, 0) USING TTL %d;", ks, tbl, 1),
format("INSERT INTO %s.%s (id, val) VALUES (1, 1) USING TTL %d;", ks, tbl, 1)
);
ClientState cs = ClientState.forInternalCalls();
QueryState qs = new QueryState(cs);
List<ModificationStatement> statements = new ArrayList<>(queries.size());
for (String query : queries)
statements.add((ModificationStatement) QueryProcessor.parseStatement(query, cs));
BatchStatement batch =
new BatchStatement(BatchStatement.Type.UNLOGGED, VariableSpecifications.empty(), statements, Attributes.none());
// execute an BATCH message with now set to [now + 1 day], with ttl = 1, making its effective ttl = 1 day + 1.
QueryProcessor.instance.processBatch(batch, qs, batchQueryOptions(now + day), Collections.emptyMap(), System.nanoTime());
// verify that despite TTL having passed at now + 1 the rows are still there.
assertEquals(2, executeSelect(format("SELECT * FROM %s.%s", ks, tbl), now + 1, false).size());
// jump in time by one day, make sure the row expired.
assertEquals(0, executeSelect(format("SELECT * FROM %s.%s", ks, tbl), now + day + 1, false).size());
}
private static ResultSet executeSelect(String query, int nowInSeconds, boolean prepared)
{
ResultMessage message = execute(query, nowInSeconds, prepared);
return ((ResultMessage.Rows) message).result;
}
private static void executeModify(String query, int nowInSeconds, boolean prepared)
{
execute(query, nowInSeconds, prepared);
}
// prepared = false tests QueryMessage path, prepared = true tests ExecuteMessage path
private static ResultMessage execute(String query, int nowInSeconds, boolean prepared)
{
ClientState cs = ClientState.forInternalCalls();
QueryState qs = new QueryState(cs);
if (prepared)
{
CQLStatement statement = QueryProcessor.parseStatement(query, cs);
return QueryProcessor.instance.processPrepared(statement, qs, queryOptions(nowInSeconds), Collections.emptyMap(), System.nanoTime());
}
else
{
CQLStatement statement = QueryProcessor.instance.parse(query, qs, queryOptions(nowInSeconds));
return QueryProcessor.instance.process(statement, qs, queryOptions(nowInSeconds), Collections.emptyMap(), System.nanoTime());
}
}
private static QueryOptions queryOptions(int nowInSeconds)
{
return QueryOptions.create(ConsistencyLevel.ONE,
Collections.emptyList(),
false,
Integer.MAX_VALUE,
null,
null,
ProtocolVersion.CURRENT,
null,
Long.MIN_VALUE,
nowInSeconds);
}
private static BatchQueryOptions batchQueryOptions(int nowInSeconds)
{
return BatchQueryOptions.withoutPerStatementVariables(queryOptions(nowInSeconds));
}
}