2020-11-11 15:54:01 +00:00
|
|
|
import pytest
|
|
|
|
|
2022-03-09 17:57:25 +00:00
|
|
|
from moto.dynamodb.exceptions import IncorrectOperandType, IncorrectDataType
|
|
|
|
from moto.dynamodb.models import Item, DynamoType
|
2022-12-14 11:07:34 +00:00
|
|
|
from moto.dynamodb.parsing.ast_nodes import (
|
|
|
|
UpdateExpression,
|
|
|
|
UpdateExpressionAddClause,
|
|
|
|
UpdateExpressionAddAction,
|
|
|
|
UpdateExpressionRemoveAction,
|
|
|
|
UpdateExpressionSetAction,
|
|
|
|
)
|
2022-03-09 17:57:25 +00:00
|
|
|
from moto.dynamodb.parsing.executors import UpdateExpressionExecutor
|
|
|
|
from moto.dynamodb.parsing.expressions import UpdateExpressionParser
|
|
|
|
from moto.dynamodb.parsing.validators import UpdateExpressionValidator
|
2020-04-26 14:12:33 +00:00
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_if_not_exists_not_existing_value(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "SET a = if_not_exists(b, a)"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "a": {"S": "A"}},
|
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values=None,
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "a": {"S": "A"}},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_if_not_exists_with_existing_attribute_should_return_attribute(
|
|
|
|
table,
|
|
|
|
):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "SET a = if_not_exists(b, a)"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "a": {"S": "A"}, "b": {"S": "B"}},
|
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values=None,
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "a": {"S": "B"}, "b": {"S": "B"}},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_if_not_exists_with_existing_attribute_should_return_value(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "SET a = if_not_exists(b, :val)"
|
|
|
|
update_expression_values = {":val": {"N": "4"}}
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "b": {"N": "3"}},
|
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values=update_expression_values,
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "b": {"N": "3"}, "a": {"N": "3"}},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_if_not_exists_with_non_existing_attribute_should_return_value(
|
|
|
|
table,
|
|
|
|
):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "SET a = if_not_exists(b, :val)"
|
|
|
|
update_expression_values = {":val": {"N": "4"}}
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
2021-11-10 21:42:33 +00:00
|
|
|
hash_key=DynamoType({"S": "id"}), range_key=None, attrs={"id": {"S": "1"}}
|
2020-04-26 14:12:33 +00:00
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values=update_expression_values,
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "a": {"N": "4"}},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_sum_operation(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "SET a = a + b"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "a": {"N": "3"}, "b": {"N": "4"}},
|
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values=None,
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "a": {"N": "7"}, "b": {"N": "4"}},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_remove(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "Remove a"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "a": {"N": "3"}, "b": {"N": "4"}},
|
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values=None,
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "1"}, "b": {"N": "4"}},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_remove_in_map(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "Remove itemmap.itemlist[1].foo11"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={
|
|
|
|
"id": {"S": "foo2"},
|
|
|
|
"itemmap": {
|
|
|
|
"M": {
|
|
|
|
"itemlist": {
|
|
|
|
"L": [
|
|
|
|
{"M": {"foo00": {"S": "bar1"}, "foo01": {"S": "bar2"}}},
|
|
|
|
{"M": {"foo10": {"S": "bar1"}, "foo11": {"S": "bar2"}}},
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values=None,
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={
|
|
|
|
"id": {"S": "foo2"},
|
|
|
|
"itemmap": {
|
|
|
|
"M": {
|
|
|
|
"itemlist": {
|
|
|
|
"L": [
|
|
|
|
{"M": {"foo00": {"S": "bar1"}, "foo01": {"S": "bar2"}}},
|
2020-11-11 15:55:37 +00:00
|
|
|
{"M": {"foo10": {"S": "bar1"}}},
|
2020-04-26 14:12:33 +00:00
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_remove_in_list(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "Remove itemmap.itemlist[1]"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={
|
|
|
|
"id": {"S": "foo2"},
|
|
|
|
"itemmap": {
|
|
|
|
"M": {
|
|
|
|
"itemlist": {
|
|
|
|
"L": [
|
|
|
|
{"M": {"foo00": {"S": "bar1"}, "foo01": {"S": "bar2"}}},
|
|
|
|
{"M": {"foo10": {"S": "bar1"}, "foo11": {"S": "bar2"}}},
|
|
|
|
]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values=None,
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={
|
|
|
|
"id": {"S": "foo2"},
|
|
|
|
"itemmap": {
|
|
|
|
"M": {
|
|
|
|
"itemlist": {
|
2020-11-11 15:55:37 +00:00
|
|
|
"L": [
|
|
|
|
{"M": {"foo00": {"S": "bar1"}, "foo01": {"S": "bar2"}}},
|
|
|
|
]
|
2020-04-26 14:12:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2021-02-19 07:47:51 +00:00
|
|
|
@pytest.mark.parametrize("attr_name", ["s", "#placeholder"])
|
|
|
|
def test_execution_of_delete_element_from_set(table, attr_name):
|
|
|
|
expression_attribute_names = {"#placeholder": "s"}
|
2022-11-17 22:41:08 +00:00
|
|
|
update_expression = f"delete {attr_name} :value"
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
2021-02-19 07:47:51 +00:00
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"SS": ["value1", "value2", "value3"]}},
|
2020-04-26 14:12:33 +00:00
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
2021-02-19 07:47:51 +00:00
|
|
|
expression_attribute_names=expression_attribute_names,
|
2020-04-26 14:12:33 +00:00
|
|
|
expression_attribute_values={":value": {"SS": ["value2", "value5"]}},
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
2021-02-19 07:47:51 +00:00
|
|
|
UpdateExpressionExecutor(validated_ast, item, expression_attribute_names).execute()
|
2020-04-26 14:12:33 +00:00
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
2020-11-11 15:55:37 +00:00
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"SS": ["value1", "value3"]}},
|
2020-04-26 14:12:33 +00:00
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
2021-02-19 07:47:51 +00:00
|
|
|
# delete last elements
|
2022-11-17 22:41:08 +00:00
|
|
|
update_expression = f"delete {attr_name} :value"
|
2021-02-19 07:47:51 +00:00
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=expression_attribute_names,
|
|
|
|
expression_attribute_values={":value": {"SS": ["value1", "value3"]}},
|
|
|
|
item=item,
|
|
|
|
table=table,
|
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, expression_attribute_names).execute()
|
|
|
|
expected_item = Item(
|
2021-11-10 21:42:33 +00:00
|
|
|
hash_key=DynamoType({"S": "id"}), range_key=None, attrs={"id": {"S": "foo2"}}
|
2021-02-19 07:47:51 +00:00
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
2020-04-26 14:12:33 +00:00
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_add_number(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "add s :value"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
2020-11-11 15:55:37 +00:00
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"N": "5"}},
|
2020-04-26 14:12:33 +00:00
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values={":value": {"N": "10"}},
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"N": "15"}},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_add_set_to_a_number(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "add s :value"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
2020-11-11 15:55:37 +00:00
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"N": "5"}},
|
2020-04-26 14:12:33 +00:00
|
|
|
)
|
|
|
|
try:
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values={":value": {"SS": ["s1"]}},
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"N": "15"}},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
assert False
|
|
|
|
except IncorrectDataType:
|
|
|
|
assert True
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_add_to_a_set(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
update_expression = "ADD s :value"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
2020-11-11 15:55:37 +00:00
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"SS": ["value1", "value2", "value3"]}},
|
2020-04-26 14:12:33 +00:00
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values={":value": {"SS": ["value2", "value5"]}},
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
expected_item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={
|
|
|
|
"id": {"S": "foo2"},
|
|
|
|
"s": {"SS": ["value1", "value2", "value3", "value5"]},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
assert expected_item == item
|
|
|
|
|
|
|
|
|
2020-11-11 15:55:37 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"expression_attribute_values,unexpected_data_type",
|
2020-04-26 14:12:33 +00:00
|
|
|
[
|
2020-11-11 15:55:37 +00:00
|
|
|
({":value": {"S": "10"}}, "STRING"),
|
|
|
|
({":value": {"N": "10"}}, "NUMBER"),
|
|
|
|
({":value": {"B": "10"}}, "BINARY"),
|
|
|
|
({":value": {"BOOL": True}}, "BOOLEAN"),
|
|
|
|
({":value": {"NULL": True}}, "NULL"),
|
|
|
|
({":value": {"M": {"el0": {"S": "10"}}}}, "MAP"),
|
|
|
|
({":value": {"L": []}}, "LIST"),
|
|
|
|
],
|
2020-04-26 14:12:33 +00:00
|
|
|
)
|
|
|
|
def test_execution_of__delete_element_from_set_invalid_value(
|
2020-11-17 09:12:39 +00:00
|
|
|
expression_attribute_values, unexpected_data_type, table
|
2020-04-26 14:12:33 +00:00
|
|
|
):
|
|
|
|
"""A delete statement must use a value of type SS in order to delete elements from a set."""
|
|
|
|
update_expression = "delete s :value"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
2020-11-11 15:55:37 +00:00
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"SS": ["value1", "value2", "value3"]}},
|
2020-04-26 14:12:33 +00:00
|
|
|
)
|
|
|
|
try:
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values=expression_attribute_values,
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
assert False, "Must raise exception"
|
|
|
|
except IncorrectOperandType as e:
|
|
|
|
assert e.operator_or_function == "operator: DELETE"
|
|
|
|
assert e.operand_type == unexpected_data_type
|
|
|
|
|
|
|
|
|
2020-11-17 09:12:39 +00:00
|
|
|
def test_execution_of_delete_element_from_a_string_attribute(table):
|
2020-04-26 14:12:33 +00:00
|
|
|
"""A delete statement must use a value of type SS in order to delete elements from a set."""
|
|
|
|
update_expression = "delete s :value"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
2020-11-11 15:55:37 +00:00
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"S": "5"}},
|
2020-04-26 14:12:33 +00:00
|
|
|
)
|
|
|
|
try:
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values={":value": {"SS": ["value2"]}},
|
|
|
|
item=item,
|
2020-11-17 09:12:39 +00:00
|
|
|
table=table,
|
2020-04-26 14:12:33 +00:00
|
|
|
).validate()
|
|
|
|
UpdateExpressionExecutor(validated_ast, item, None).execute()
|
|
|
|
assert False, "Must raise exception"
|
|
|
|
except IncorrectDataType:
|
|
|
|
assert True
|
2022-12-14 11:07:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_normalize_with_one_action(table):
|
|
|
|
update_expression = "ADD s :value"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={"id": {"S": "foo2"}, "s": {"SS": ["value1", "value2", "value3"]}},
|
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values={":value": {"SS": ["value2", "value5"]}},
|
|
|
|
item=item,
|
|
|
|
table=table,
|
|
|
|
).validate()
|
2023-07-13 10:21:47 +00:00
|
|
|
assert len(validated_ast.children) == 1
|
|
|
|
assert isinstance(validated_ast.children[0], UpdateExpressionAddClause)
|
2022-12-14 11:07:34 +00:00
|
|
|
|
|
|
|
validated_ast.normalize()
|
2023-07-13 10:21:47 +00:00
|
|
|
assert len(validated_ast.children) == 1
|
|
|
|
assert isinstance(validated_ast.children[0], UpdateExpressionAddAction)
|
2022-12-14 11:07:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_normalize_with_multiple_actions__order_is_preserved(table):
|
|
|
|
update_expression = "ADD s :value REMOVE a[3], a[1], a[2] SET t=:value"
|
|
|
|
update_expression_ast = UpdateExpressionParser.make(update_expression)
|
|
|
|
item = Item(
|
|
|
|
hash_key=DynamoType({"S": "id"}),
|
|
|
|
range_key=None,
|
|
|
|
attrs={
|
|
|
|
"id": {"S": "foo2"},
|
|
|
|
"a": {"L": [{"S": "val1"}, {"S": "val2"}, {"S": "val3"}, {"S": "val4"}]},
|
|
|
|
"s": {"SS": ["value1", "value2", "value3"]},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
validated_ast = UpdateExpressionValidator(
|
|
|
|
update_expression_ast,
|
|
|
|
expression_attribute_names=None,
|
|
|
|
expression_attribute_values={":value": {"SS": ["value2", "value5"]}},
|
|
|
|
item=item,
|
|
|
|
table=table,
|
|
|
|
).validate()
|
2023-07-13 10:21:47 +00:00
|
|
|
assert len(validated_ast.children) == 2
|
2022-12-14 11:07:34 +00:00
|
|
|
# add clause first
|
2023-07-13 10:21:47 +00:00
|
|
|
assert isinstance(validated_ast.children[0], UpdateExpressionAddClause)
|
2022-12-14 11:07:34 +00:00
|
|
|
# rest of the expression next
|
2023-07-13 10:21:47 +00:00
|
|
|
assert isinstance(validated_ast.children[1], UpdateExpression)
|
2022-12-14 11:07:34 +00:00
|
|
|
|
|
|
|
validated_ast.normalize()
|
2023-07-13 10:21:47 +00:00
|
|
|
assert len(validated_ast.children) == 5
|
2022-12-14 11:07:34 +00:00
|
|
|
# add action first
|
2023-07-13 10:21:47 +00:00
|
|
|
assert isinstance(validated_ast.children[0], UpdateExpressionAddAction)
|
2022-12-14 11:07:34 +00:00
|
|
|
# Removal actions in reverse order
|
2023-07-13 10:21:47 +00:00
|
|
|
assert isinstance(validated_ast.children[1], UpdateExpressionRemoveAction)
|
|
|
|
assert validated_ast.children[1]._get_value() == 3
|
|
|
|
assert isinstance(validated_ast.children[2], UpdateExpressionRemoveAction)
|
|
|
|
assert validated_ast.children[2]._get_value() == 2
|
|
|
|
assert isinstance(validated_ast.children[3], UpdateExpressionRemoveAction)
|
|
|
|
assert validated_ast.children[3]._get_value() == 1
|
2022-12-14 11:07:34 +00:00
|
|
|
# Set action last, as per insertion order
|
2023-07-13 10:21:47 +00:00
|
|
|
assert isinstance(validated_ast.children[4], UpdateExpressionSetAction)
|