Set Time to Live [TTL] on DynamoDB Items using Java
In this post, we’ll demonstrate how expiring items in DynamoDB works in java, using the AWS SDK 2.0+, which has full reactive support.
We will leverage work done in a previous post, which setup an embedded DynamoDB instance for integration testing, and the source code is available on Github for this and previous posts related to this topic.
Background - How it Works
To start with, the things we’ll need to understand to get TTL working are:
- You must first specify, at the table level, which attribute is the source of truth for when an item expires
- The attribute must specify the unix epoch time, in seconds, that the item should expire.
- Expiring an item, like many things in Dynamo, is a bit “fuzzy”–it will expire around the time it is supposed to.
Further points and nuances can be found in the documentation for DynamoDB TTL, referenced at the start of this post.
Table Setup
For this example, we’ll start by setting up our table in the same way that we have in previous posts:
@Test
public void testTTL() throws Exception {
String currentTableName = "PhoneTTLTest";
createTableAndWaitForComplete(currentTableName);
}
This just leverages code we’ve already written and I won’t rehash that here.
After our test table is created, we will need to specify that TTL is enabled, as well as what attribute dynamo should be looking at to make the decision about when to expire an individual item. Note that in real environments [e.g. production] something like this should really be done with terraform, but this is just integration testing code so all is good:
String EXPIRE_TIME = "ExpireTime";
dynamoDbAsyncClient.updateTimeToLive(
UpdateTimeToLiveRequest.builder()
.tableName(currentTableName)
.timeToLiveSpecification(
TimeToLiveSpecification.builder()
.enabled(true)
.attributeName(EXPIRE_TIME)
.build()
)
.build()
).get();
StepVerifier.create(Mono.fromFuture(dynamoDbAsyncClient.describeTimeToLive(
DescribeTimeToLiveRequest.builder().tableName(currentTableName).build()))
)
.expectNextMatches(describeTimeToLiveResponse ->
describeTimeToLiveResponse
.timeToLiveDescription()
.timeToLiveStatus().equals(TimeToLiveStatus.ENABLED)
)
.verifyComplete();
This chunk of code just sets the TTL specification, enabling TTL on this table and saying that the attribute of “ExpireTime” should be the source of truth for when an attribute should be expired. We then make a follow up call to verify that the settings we have specified on this table have actually taken effect.
Now let’s put an item into this table, specify that it should expire soon, and see dynamo clear it out:
String partitionKey = "Google";
String rangeKey = "Pixel 1";
Map<String, AttributeValue> pixel1ItemAttributes = getMapWith(partitionKey, rangeKey);
pixel1ItemAttributes.put(COLOR, AttributeValue.builder().s("Blue").build());
pixel1ItemAttributes.put(YEAR, AttributeValue.builder().n("2012").build());
// expire about 3 seconds from now
String expireTime = Long.toString((System.currentTimeMillis() / 1000L) + 3);
pixel1ItemAttributes.put(
EXPIRE_TIME,
AttributeValue.builder()
.n(expireTime)
.build()
);
PutItemRequest populateDataItemRequest = PutItemRequest.builder()
.tableName(currentTableName)
.item(pixel1ItemAttributes)
.build();
// put item with TTL into dynamo
StepVerifier.create(Mono.fromFuture(dynamoDbAsyncClient.putItem(populateDataItemRequest)))
.expectNextCount(1)
.verifyComplete();
Map<String, AttributeValue> currentItemKey = Map.of(
COMPANY, AttributeValue.builder().s(partitionKey).build(),
MODEL, AttributeValue.builder().s(rangeKey).build()
);
// get immediately, should exist
StepVerifier.create(Mono.fromFuture(dynamoDbAsyncClient.getItem(
GetItemRequest.builder().tableName(currentTableName).key(currentItemKey).build()))
)
.expectNextMatches(getItemResponse -> getItemResponse.hasItem()
&& getItemResponse.item().get(COLOR).s().equals("Blue"))
.verifyComplete();
// local dynamo seems to need like 10 seconds to actually clear this out
Thread.sleep(13000);
StepVerifier.create(Mono.fromFuture(dynamoDbAsyncClient.getItem(
GetItemRequest.builder()
.key(currentItemKey)
.tableName(currentTableName)
.build())
)
)
.expectNextMatches(getItemResponse -> !getItemResponse.hasItem())
.verifyComplete();
Here, we set the expire time to be about 3 seconds from now on a created item, then immediately grab it from the table to verify that it exists. After a 13 second sleep [necessary in this case basically because of the behavior of embedded/local dynamo], we then verify that trying to get the same item out of the table returns an empty response.
Remember to check out the source code for this one on Github.