DynamoDB: query(): LastEvaluatedKey does not have to exist (#7484)
This commit is contained in:
parent
ca5d514c61
commit
1940888296
@ -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]
|
||||||
|
@ -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"}],
|
||||||
)
|
)
|
||||||
|
@ -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"}}]
|
||||||
|
@ -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)
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user