DynamoDB: query(): LastEvaluatedKey does not have to exist (#7484)

This commit is contained in:
Bert Blommers 2024-03-17 20:29:57 +00:00 committed by GitHub
parent ca5d514c61
commit 1940888296
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 150 additions and 20 deletions

View File

@ -744,9 +744,10 @@ class Table(CloudFormationModel):
# Cycle through the previous page of results # Cycle through the previous page of results
# When we encounter our start key, we know we've reached the end of the previous page # When we encounter our start key, we know we've reached the end of the previous page
if processing_previous_page: if processing_previous_page:
if self._item_equals_dct(result, exclusive_start_key): if self._item_smaller_than_dct(result, exclusive_start_key):
processing_previous_page = False
continue continue
else:
processing_previous_page = False
# Check wether we've reached the limit of our result set # Check wether we've reached the limit of our result set
# That can be either in number, or in size # That can be either in number, or in size
@ -868,9 +869,10 @@ class Table(CloudFormationModel):
# Cycle through the previous page of results # Cycle through the previous page of results
# When we encounter our start key, we know we've reached the end of the previous page # When we encounter our start key, we know we've reached the end of the previous page
if processing_previous_page: if processing_previous_page:
if self._item_equals_dct(item, exclusive_start_key): if self._item_smaller_than_dct(item, exclusive_start_key):
processing_previous_page = False
continue continue
else:
processing_previous_page = False
# Check wether we've reached the limit of our result set # Check wether we've reached the limit of our result set
# That can be either in number, or in size # That can be either in number, or in size
@ -921,12 +923,13 @@ class Table(CloudFormationModel):
return results, scanned_count, last_evaluated_key return results, scanned_count, last_evaluated_key
def _item_equals_dct(self, item: Item, dct: Dict[str, Any]) -> bool: def _item_smaller_than_dct(self, item: Item, dct: Dict[str, Any]) -> bool:
hash_key = DynamoType(dct.get(self.hash_key_attr)) # type: ignore[arg-type] hash_key = DynamoType(dct.get(self.hash_key_attr)) # type: ignore[arg-type]
range_key = dct.get(self.range_key_attr) if self.range_key_attr else None range_key = dct.get(self.range_key_attr) if self.range_key_attr else None
if range_key is not None: if range_key is not None:
range_key = DynamoType(range_key) range_key = DynamoType(range_key)
return item.hash_key == hash_key and item.range_key == range_key return item.hash_key <= hash_key and item.range_key <= range_key
return item.hash_key <= hash_key
def _get_last_evaluated_key( def _get_last_evaluated_key(
self, last_result: Item, index_name: Optional[str] self, last_result: Item, index_name: Optional[str]

View File

@ -7,7 +7,7 @@ import boto3
from moto import mock_aws from moto import mock_aws
def dynamodb_aws_verified(create_table: bool = True): def dynamodb_aws_verified(create_table: bool = True, add_range: bool = False):
""" """
Function that is verified to work against AWS. Function that is verified to work against AWS.
Can be run against AWS at any time by setting: Can be run against AWS at any time by setting:
@ -46,10 +46,15 @@ def dynamodb_aws_verified(create_table: bool = True):
def create_table_and_test(table_name): def create_table_and_test(table_name):
client = boto3.client("dynamodb", region_name="us-east-1") client = boto3.client("dynamodb", region_name="us-east-1")
schema = [{"AttributeName": "pk", "KeyType": "HASH"}]
defs = [{"AttributeName": "pk", "AttributeType": "S"}]
if add_range:
schema.append({"AttributeName": "sk", "KeyType": "RANGE"})
defs.append({"AttributeName": "sk", "AttributeType": "S"})
client.create_table( client.create_table(
TableName=table_name, TableName=table_name,
KeySchema=[{"AttributeName": "pk", "KeyType": "HASH"}], KeySchema=schema,
AttributeDefinitions=[{"AttributeName": "pk", "AttributeType": "S"}], AttributeDefinitions=defs,
ProvisionedThroughput={"ReadCapacityUnits": 1, "WriteCapacityUnits": 5}, ProvisionedThroughput={"ReadCapacityUnits": 1, "WriteCapacityUnits": 5},
Tags=[{"Key": "environment", "Value": "moto_tests"}], Tags=[{"Key": "environment", "Value": "moto_tests"}],
) )

View File

@ -1,3 +1,4 @@
import copy
import re import re
import uuid import uuid
from datetime import datetime from datetime import datetime
@ -5957,3 +5958,125 @@ def test_update_item_with_global_secondary_index():
"One or more parameter values were invalid: Type mismatch" "One or more parameter values were invalid: Type mismatch"
in err["Message"] in err["Message"]
) )
@pytest.mark.aws_verified
@dynamodb_aws_verified(add_range=True)
def test_query_with_unknown_last_evaluated_key(table_name=None):
client = boto3.client("dynamodb", region_name="us-east-1")
for i in range(10):
client.put_item(
TableName=table_name,
Item={
"pk": {"S": "hash_value"},
"sk": {"S": f"range_value{i}"},
},
)
p1 = client.query(
TableName=table_name,
KeyConditionExpression="#h = :h",
ExpressionAttributeNames={"#h": "pk"},
ExpressionAttributeValues={":h": {"S": "hash_value"}},
Limit=1,
)
assert p1["Items"] == [{"pk": {"S": "hash_value"}, "sk": {"S": "range_value0"}}]
# Using the Exact ExclusiveStartKey provided
p2 = client.query(
TableName=table_name,
KeyConditionExpression="#h = :h",
ExpressionAttributeNames={"#h": "pk"},
ExpressionAttributeValues={":h": {"S": "hash_value"}},
Limit=1,
ExclusiveStartKey=p1["LastEvaluatedKey"],
)
assert p2["Items"] == [{"pk": {"S": "hash_value"}, "sk": {"S": "range_value1"}}]
# We can change ExclusiveStartKey
# It doesn't need to match - it just needs to be >= page1, but < page1
different_key = copy.copy(p1["LastEvaluatedKey"])
different_key["sk"]["S"] = different_key["sk"]["S"] + "0"
p3 = client.query(
TableName=table_name,
KeyConditionExpression="#h = :h",
ExpressionAttributeNames={"#h": "pk"},
ExpressionAttributeValues={":h": {"S": "hash_value"}},
Limit=1,
ExclusiveStartKey=different_key,
)
assert p3["Items"] == [{"pk": {"S": "hash_value"}, "sk": {"S": "range_value1"}}]
# Sanity check - increasing the sk to something much greater will result in a different outcome
different_key["sk"]["S"] = "range_value500"
p4 = client.query(
TableName=table_name,
KeyConditionExpression="#h = :h",
ExpressionAttributeNames={"#h": "pk"},
ExpressionAttributeValues={":h": {"S": "hash_value"}},
Limit=1,
ExclusiveStartKey=different_key,
)
assert p4["Items"] == [{"pk": {"S": "hash_value"}, "sk": {"S": "range_value6"}}]
@pytest.mark.aws_verified
@dynamodb_aws_verified(add_range=True)
def test_scan_with_unknown_last_evaluated_key(table_name=None):
client = boto3.client("dynamodb", region_name="us-east-1")
for i in range(10):
client.put_item(
TableName=table_name,
Item={
"pk": {"S": "hash_value"},
"sk": {"S": f"range_value{i}"},
},
)
p1 = client.scan(
TableName=table_name,
FilterExpression="#h = :h",
ExpressionAttributeNames={"#h": "pk"},
ExpressionAttributeValues={":h": {"S": "hash_value"}},
Limit=1,
)
assert p1["Items"] == [{"pk": {"S": "hash_value"}, "sk": {"S": "range_value0"}}]
# Using the Exact ExclusiveStartKey provided
p2 = client.scan(
TableName=table_name,
FilterExpression="#h = :h",
ExpressionAttributeNames={"#h": "pk"},
ExpressionAttributeValues={":h": {"S": "hash_value"}},
Limit=1,
ExclusiveStartKey=p1["LastEvaluatedKey"],
)
assert p2["Items"] == [{"pk": {"S": "hash_value"}, "sk": {"S": "range_value1"}}]
# We can change ExclusiveStartKey
# It doesn't need to match - it just needs to be >= page1, but < page1
different_key = copy.copy(p1["LastEvaluatedKey"])
different_key["sk"]["S"] = different_key["sk"]["S"] + "0"
p3 = client.scan(
TableName=table_name,
FilterExpression="#h = :h",
ExpressionAttributeNames={"#h": "pk"},
ExpressionAttributeValues={":h": {"S": "hash_value"}},
Limit=1,
ExclusiveStartKey=different_key,
)
assert p3["Items"] == [{"pk": {"S": "hash_value"}, "sk": {"S": "range_value1"}}]
# Sanity check - increasing the sk to something much greater will result in a different outcome
different_key["sk"]["S"] = "range_value500"
p4 = client.scan(
TableName=table_name,
FilterExpression="#h = :h",
ExpressionAttributeNames={"#h": "pk"},
ExpressionAttributeValues={":h": {"S": "hash_value"}},
Limit=1,
ExclusiveStartKey=different_key,
)
assert p4["Items"] == [{"pk": {"S": "hash_value"}, "sk": {"S": "range_value6"}}]

View File

@ -8,6 +8,8 @@ from botocore.exceptions import ClientError
from moto import mock_aws from moto import mock_aws
from moto.core import DEFAULT_ACCOUNT_ID as ACCOUNT_ID from moto.core import DEFAULT_ACCOUNT_ID as ACCOUNT_ID
from . import dynamodb_aws_verified
@mock_aws @mock_aws
def test_create_table(): def test_create_table():
@ -512,29 +514,26 @@ def test_update_settype_item_with_conditions():
assert returned_item["Item"]["foo"] == set(["baz"]) assert returned_item["Item"]["foo"] == set(["baz"])
@mock_aws @pytest.mark.aws_verified
def test_scan_pagination(): @dynamodb_aws_verified()
table = _create_user_table() def test_scan_pagination(table_name=None):
table = boto3.resource("dynamodb", "us-east-1").Table(table_name)
expected_usernames = [f"user{i}" for i in range(10)] expected_usernames = [f"user{i}" for i in range(10)]
for u in expected_usernames: for u in expected_usernames:
table.put_item(Item={"username": u}) table.put_item(Item={"pk": u})
page1 = table.scan(Limit=6) page1 = table.scan(Limit=6)
assert page1["Count"] == 6 assert page1["Count"] == 6
assert len(page1["Items"]) == 6 assert len(page1["Items"]) == 6
page1_results = set([r["username"] for r in page1["Items"]]) page1_results = [r["pk"] for r in page1["Items"]]
assert page1_results == {"user0", "user3", "user1", "user2", "user5", "user4"}
page2 = table.scan(Limit=6, ExclusiveStartKey=page1["LastEvaluatedKey"]) page2 = table.scan(Limit=6, ExclusiveStartKey=page1["LastEvaluatedKey"])
assert page2["Count"] == 4 assert page2["Count"] == 4
assert len(page2["Items"]) == 4 assert len(page2["Items"]) == 4
assert "LastEvaluatedKey" not in page2 assert "LastEvaluatedKey" not in page2
page2_results = set([r["username"] for r in page2["Items"]]) page2_results = [r["pk"] for r in page2["Items"]]
assert page2_results == {"user6", "user7", "user8", "user9"}
results = page1["Items"] + page2["Items"] usernames = set(page1_results + page2_results)
usernames = set([r["username"] for r in results])
assert usernames == set(expected_usernames) assert usernames == set(expected_usernames)