Techdebt: Replace string-format with f-strings (for d* dirs) (#5662)
This commit is contained in:
parent
b0b943949d
commit
52892f5481
@ -7,7 +7,7 @@ class DataBrewClientError(JsonRESTError):
|
|||||||
|
|
||||||
class AlreadyExistsException(DataBrewClientError):
|
class AlreadyExistsException(DataBrewClientError):
|
||||||
def __init__(self, typ):
|
def __init__(self, typ):
|
||||||
super().__init__("AlreadyExistsException", "%s already exists." % (typ))
|
super().__init__("AlreadyExistsException", f"{typ} already exists.")
|
||||||
|
|
||||||
|
|
||||||
class ConflictException(DataBrewClientError):
|
class ConflictException(DataBrewClientError):
|
||||||
@ -41,7 +41,7 @@ class ResourceNotFoundException(DataBrewClientError):
|
|||||||
|
|
||||||
class RulesetNotFoundException(EntityNotFoundException):
|
class RulesetNotFoundException(EntityNotFoundException):
|
||||||
def __init__(self, recipe_name):
|
def __init__(self, recipe_name):
|
||||||
super().__init__("Ruleset %s not found." % recipe_name)
|
super().__init__(f"Ruleset {recipe_name} not found.")
|
||||||
|
|
||||||
|
|
||||||
class ServiceQuotaExceededException(JsonRESTError):
|
class ServiceQuotaExceededException(JsonRESTError):
|
||||||
|
@ -488,12 +488,12 @@ class FakeRecipeVersion(BaseModel):
|
|||||||
"Name": self.name,
|
"Name": self.name,
|
||||||
"Steps": self.steps,
|
"Steps": self.steps,
|
||||||
"Description": self.description,
|
"Description": self.description,
|
||||||
"CreateDate": "%.3f" % self.created_time.timestamp(),
|
"CreateDate": f"{self.created_time.timestamp():.3f}",
|
||||||
"Tags": self.tags or dict(),
|
"Tags": self.tags or dict(),
|
||||||
"RecipeVersion": str(self.version),
|
"RecipeVersion": str(self.version),
|
||||||
}
|
}
|
||||||
if self.published_date is not None:
|
if self.published_date is not None:
|
||||||
dict_recipe["PublishedDate"] = "%.3f" % self.published_date.timestamp()
|
dict_recipe["PublishedDate"] = f"{self.published_date.timestamp():.3f}"
|
||||||
|
|
||||||
return dict_recipe
|
return dict_recipe
|
||||||
|
|
||||||
@ -529,7 +529,7 @@ class FakeRuleset(BaseModel):
|
|||||||
"Rules": self.rules,
|
"Rules": self.rules,
|
||||||
"Description": self.description,
|
"Description": self.description,
|
||||||
"TargetArn": self.target_arn,
|
"TargetArn": self.target_arn,
|
||||||
"CreateDate": "%.3f" % self.created_time.timestamp(),
|
"CreateDate": f"{self.created_time.timestamp():.3f}",
|
||||||
"Tags": self.tags or dict(),
|
"Tags": self.tags or dict(),
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -569,7 +569,7 @@ class FakeDataset(BaseModel):
|
|||||||
"FormatOptions": self.format_options,
|
"FormatOptions": self.format_options,
|
||||||
"Input": self.input,
|
"Input": self.input,
|
||||||
"PathOptions": self.path_options,
|
"PathOptions": self.path_options,
|
||||||
"CreateDate": "%.3f" % self.created_time.timestamp(),
|
"CreateDate": f"{self.created_time.timestamp():.3f}",
|
||||||
"Tags": self.tags or dict(),
|
"Tags": self.tags or dict(),
|
||||||
"ResourceArn": self.resource_arn,
|
"ResourceArn": self.resource_arn,
|
||||||
}
|
}
|
||||||
@ -631,7 +631,7 @@ class FakeJob(BaseModel, metaclass=BaseModelABCMeta):
|
|||||||
rtn_dict = {
|
rtn_dict = {
|
||||||
"Name": self.name,
|
"Name": self.name,
|
||||||
"AccountId": self.account_id,
|
"AccountId": self.account_id,
|
||||||
"CreateDate": "%.3f" % self.created_time.timestamp(),
|
"CreateDate": f"{self.created_time.timestamp():.3f}",
|
||||||
"DatasetName": self.dataset_name,
|
"DatasetName": self.dataset_name,
|
||||||
"EncryptionMode": self.encryption_mode,
|
"EncryptionMode": self.encryption_mode,
|
||||||
"Tags": self.tags or dict(),
|
"Tags": self.tags or dict(),
|
||||||
|
@ -3,7 +3,7 @@ from moto.moto_api._internal import mock_random
|
|||||||
|
|
||||||
|
|
||||||
def get_random_pipeline_id():
|
def get_random_pipeline_id():
|
||||||
return "df-{0}".format(mock_random.get_random_hex(length=19))
|
return f"df-{mock_random.get_random_hex(length=19)}"
|
||||||
|
|
||||||
|
|
||||||
def remove_capitalization_of_dict_keys(obj):
|
def remove_capitalization_of_dict_keys(obj):
|
||||||
|
@ -13,9 +13,7 @@ class Location(BaseModel):
|
|||||||
self.metadata = metadata
|
self.metadata = metadata
|
||||||
self.typ = typ
|
self.typ = typ
|
||||||
# Generate ARN
|
# Generate ARN
|
||||||
self.arn = "arn:aws:datasync:{0}:111222333444:location/loc-{1}".format(
|
self.arn = f"arn:aws:datasync:{region_name}:111222333444:location/loc-{str(arn_counter).zfill(17)}"
|
||||||
region_name, str(arn_counter).zfill(17)
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class Task(BaseModel):
|
class Task(BaseModel):
|
||||||
@ -36,9 +34,7 @@ class Task(BaseModel):
|
|||||||
self.status = "AVAILABLE"
|
self.status = "AVAILABLE"
|
||||||
self.current_task_execution_arn = None
|
self.current_task_execution_arn = None
|
||||||
# Generate ARN
|
# Generate ARN
|
||||||
self.arn = "arn:aws:datasync:{0}:111222333444:task/task-{1}".format(
|
self.arn = f"arn:aws:datasync:{region_name}:111222333444:task/task-{str(arn_counter).zfill(17)}"
|
||||||
region_name, str(arn_counter).zfill(17)
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class TaskExecution(BaseModel):
|
class TaskExecution(BaseModel):
|
||||||
@ -63,7 +59,7 @@ class TaskExecution(BaseModel):
|
|||||||
|
|
||||||
def __init__(self, task_arn, arn_counter=0):
|
def __init__(self, task_arn, arn_counter=0):
|
||||||
self.task_arn = task_arn
|
self.task_arn = task_arn
|
||||||
self.arn = "{0}/execution/exec-{1}".format(task_arn, str(arn_counter).zfill(17))
|
self.arn = f"{task_arn}/execution/exec-{str(arn_counter).zfill(17)}"
|
||||||
self.status = self.TASK_EXECUTION_INTERMEDIATE_STATES[0]
|
self.status = self.TASK_EXECUTION_INTERMEDIATE_STATES[0]
|
||||||
|
|
||||||
# Simulate a task execution
|
# Simulate a task execution
|
||||||
@ -80,16 +76,12 @@ class TaskExecution(BaseModel):
|
|||||||
else:
|
else:
|
||||||
self.status = self.TASK_EXECUTION_SUCCESS_STATES[0]
|
self.status = self.TASK_EXECUTION_SUCCESS_STATES[0]
|
||||||
return
|
return
|
||||||
raise Exception(
|
raise Exception(f"TaskExecution.iterate_status: Unknown status={self.status}")
|
||||||
"TaskExecution.iterate_status: Unknown status={0}".format(self.status)
|
|
||||||
)
|
|
||||||
|
|
||||||
def cancel(self):
|
def cancel(self):
|
||||||
if self.status not in self.TASK_EXECUTION_INTERMEDIATE_STATES:
|
if self.status not in self.TASK_EXECUTION_INTERMEDIATE_STATES:
|
||||||
raise InvalidRequestException(
|
raise InvalidRequestException(
|
||||||
"Sync task cannot be cancelled in its current status: {0}".format(
|
f"Sync task cannot be cancelled in its current status: {self.status}"
|
||||||
self.status
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
self.status = "ERROR"
|
self.status = "ERROR"
|
||||||
|
|
||||||
@ -133,14 +125,10 @@ class DataSyncBackend(BaseBackend):
|
|||||||
|
|
||||||
def _get_location(self, location_arn, typ):
|
def _get_location(self, location_arn, typ):
|
||||||
if location_arn not in self.locations:
|
if location_arn not in self.locations:
|
||||||
raise InvalidRequestException(
|
raise InvalidRequestException(f"Location {location_arn} is not found.")
|
||||||
"Location {0} is not found.".format(location_arn)
|
|
||||||
)
|
|
||||||
location = self.locations[location_arn]
|
location = self.locations[location_arn]
|
||||||
if location.typ != typ:
|
if location.typ != typ:
|
||||||
raise InvalidRequestException(
|
raise InvalidRequestException(f"Invalid Location type: {location.typ}")
|
||||||
"Invalid Location type: {0}".format(location.typ)
|
|
||||||
)
|
|
||||||
return location
|
return location
|
||||||
|
|
||||||
def delete_location(self, location_arn):
|
def delete_location(self, location_arn):
|
||||||
@ -153,12 +141,10 @@ class DataSyncBackend(BaseBackend):
|
|||||||
self, source_location_arn, destination_location_arn, name, metadata=None
|
self, source_location_arn, destination_location_arn, name, metadata=None
|
||||||
):
|
):
|
||||||
if source_location_arn not in self.locations:
|
if source_location_arn not in self.locations:
|
||||||
raise InvalidRequestException(
|
raise InvalidRequestException(f"Location {source_location_arn} not found.")
|
||||||
"Location {0} not found.".format(source_location_arn)
|
|
||||||
)
|
|
||||||
if destination_location_arn not in self.locations:
|
if destination_location_arn not in self.locations:
|
||||||
raise InvalidRequestException(
|
raise InvalidRequestException(
|
||||||
"Location {0} not found.".format(destination_location_arn)
|
f"Location {destination_location_arn} not found."
|
||||||
)
|
)
|
||||||
self.arn_counter = self.arn_counter + 1
|
self.arn_counter = self.arn_counter + 1
|
||||||
task = Task(
|
task = Task(
|
||||||
@ -184,9 +170,7 @@ class DataSyncBackend(BaseBackend):
|
|||||||
task.name = name
|
task.name = name
|
||||||
task.metadata = metadata
|
task.metadata = metadata
|
||||||
else:
|
else:
|
||||||
raise InvalidRequestException(
|
raise InvalidRequestException(f"Sync task {task_arn} is not found.")
|
||||||
"Sync task {0} is not found.".format(task_arn)
|
|
||||||
)
|
|
||||||
|
|
||||||
def delete_task(self, task_arn):
|
def delete_task(self, task_arn):
|
||||||
if task_arn in self.tasks:
|
if task_arn in self.tasks:
|
||||||
@ -220,9 +204,7 @@ class DataSyncBackend(BaseBackend):
|
|||||||
self.tasks[task_arn].current_task_execution_arn = None
|
self.tasks[task_arn].current_task_execution_arn = None
|
||||||
self.tasks[task_arn].status = "AVAILABLE"
|
self.tasks[task_arn].status = "AVAILABLE"
|
||||||
return
|
return
|
||||||
raise InvalidRequestException(
|
raise InvalidRequestException(f"Sync task {task_execution_arn} is not found.")
|
||||||
"Sync task {0} is not found.".format(task_execution_arn)
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
datasync_backends = BackendDict(DataSyncBackend, "datasync")
|
datasync_backends = BackendDict(DataSyncBackend, "datasync")
|
||||||
|
@ -83,10 +83,10 @@ class Op(object):
|
|||||||
self.rhs = rhs
|
self.rhs = rhs
|
||||||
|
|
||||||
def expr(self, item):
|
def expr(self, item):
|
||||||
raise NotImplementedError("Expr not defined for {0}".format(type(self)))
|
raise NotImplementedError(f"Expr not defined for {type(self)}")
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "({0} {1} {2})".format(self.lhs, self.OP, self.rhs)
|
return f"({self.lhs} {self.OP} {self.rhs})"
|
||||||
|
|
||||||
|
|
||||||
# TODO add tests for all of these
|
# TODO add tests for all of these
|
||||||
@ -276,11 +276,7 @@ class ConditionExpressionParser:
|
|||||||
),
|
),
|
||||||
(
|
(
|
||||||
self.Nonterminal.OPERAND,
|
self.Nonterminal.OPERAND,
|
||||||
re.compile(
|
re.compile(rf"^{attribute_regex}(\.{attribute_regex}|\[[0-9]\])*"),
|
||||||
r"^{attribute_regex}(\.{attribute_regex}|\[[0-9]\])*".format(
|
|
||||||
attribute_regex=attribute_regex
|
|
||||||
)
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
(self.Nonterminal.COMMA, re.compile(r"^,")),
|
(self.Nonterminal.COMMA, re.compile(r"^,")),
|
||||||
(self.Nonterminal.LEFT_PAREN, re.compile(r"^\(")),
|
(self.Nonterminal.LEFT_PAREN, re.compile(r"^\(")),
|
||||||
@ -294,7 +290,7 @@ class ConditionExpressionParser:
|
|||||||
break
|
break
|
||||||
else: # pragma: no cover
|
else: # pragma: no cover
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"Cannot parse condition starting at:{}".format(remaining_expression)
|
f"Cannot parse condition starting at:{remaining_expression}"
|
||||||
)
|
)
|
||||||
|
|
||||||
node = self.Node(
|
node = self.Node(
|
||||||
@ -327,7 +323,7 @@ class ConditionExpressionParser:
|
|||||||
for child in children:
|
for child in children:
|
||||||
self._assert(
|
self._assert(
|
||||||
child.nonterminal == self.Nonterminal.IDENTIFIER,
|
child.nonterminal == self.Nonterminal.IDENTIFIER,
|
||||||
"Cannot use {} in path".format(child.text),
|
f"Cannot use {child.text} in path",
|
||||||
[node],
|
[node],
|
||||||
)
|
)
|
||||||
output.append(
|
output.append(
|
||||||
@ -401,7 +397,7 @@ class ConditionExpressionParser:
|
|||||||
elif name.startswith("["):
|
elif name.startswith("["):
|
||||||
# e.g. [123]
|
# e.g. [123]
|
||||||
if not name.endswith("]"): # pragma: no cover
|
if not name.endswith("]"): # pragma: no cover
|
||||||
raise ValueError("Bad path element {}".format(name))
|
raise ValueError(f"Bad path element {name}")
|
||||||
return self.Node(
|
return self.Node(
|
||||||
nonterminal=self.Nonterminal.IDENTIFIER,
|
nonterminal=self.Nonterminal.IDENTIFIER,
|
||||||
kind=self.Kind.LITERAL,
|
kind=self.Kind.LITERAL,
|
||||||
@ -640,7 +636,7 @@ class ConditionExpressionParser:
|
|||||||
for i in range(len(expected_kinds)):
|
for i in range(len(expected_kinds)):
|
||||||
self._assert(
|
self._assert(
|
||||||
arguments[i].kind in expected_kinds[i],
|
arguments[i].kind in expected_kinds[i],
|
||||||
"Wrong type for argument %d in" % i,
|
f"Wrong type for argument {i} in",
|
||||||
all_children,
|
all_children,
|
||||||
)
|
)
|
||||||
if function_name.value == "size":
|
if function_name.value == "size":
|
||||||
@ -809,7 +805,7 @@ class ConditionExpressionParser:
|
|||||||
arguments = [self._make_operand(arg) for arg in arguments]
|
arguments = [self._make_operand(arg) for arg in arguments]
|
||||||
return FUNC_CLASS[function_name](*arguments)
|
return FUNC_CLASS[function_name](*arguments)
|
||||||
else: # pragma: no cover
|
else: # pragma: no cover
|
||||||
raise ValueError("Unknown operand: %r" % node)
|
raise ValueError(f"Unknown operand: {node}")
|
||||||
|
|
||||||
def _make_op_condition(self, node):
|
def _make_op_condition(self, node):
|
||||||
if node.kind == self.Kind.OR:
|
if node.kind == self.Kind.OR:
|
||||||
@ -849,7 +845,7 @@ class ConditionExpressionParser:
|
|||||||
self._make_operand(lhs), self._make_operand(rhs)
|
self._make_operand(lhs), self._make_operand(rhs)
|
||||||
)
|
)
|
||||||
else: # pragma: no cover
|
else: # pragma: no cover
|
||||||
raise ValueError("Unknown expression node kind %r" % node.kind)
|
raise ValueError(f"Unknown expression node kind {node.kind}")
|
||||||
|
|
||||||
def _assert(self, condition, message, nodes):
|
def _assert(self, condition, message, nodes):
|
||||||
if not condition:
|
if not condition:
|
||||||
@ -969,7 +965,7 @@ class OpNot(Op):
|
|||||||
return not lhs
|
return not lhs
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "({0} {1})".format(self.OP, self.lhs)
|
return f"({self.OP} {self.lhs})"
|
||||||
|
|
||||||
|
|
||||||
class OpAnd(Op):
|
class OpAnd(Op):
|
||||||
@ -1072,9 +1068,7 @@ class Func(object):
|
|||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "{0}({1})".format(
|
return f"{self.FUNC}({' '.join([repr(arg) for arg in self.arguments])})"
|
||||||
self.FUNC, " ".join([repr(arg) for arg in self.arguments])
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class FuncAttrExists(Func):
|
class FuncAttrExists(Func):
|
||||||
@ -1150,7 +1144,7 @@ class FuncSize(Func):
|
|||||||
|
|
||||||
def expr(self, item):
|
def expr(self, item):
|
||||||
if self.attr.get_type(item) is None:
|
if self.attr.get_type(item) is None:
|
||||||
raise ValueError("Invalid attribute name {0}".format(self.attr))
|
raise ValueError(f"Invalid attribute name {self.attr}")
|
||||||
|
|
||||||
if self.attr.get_type(item) in ("S", "SS", "NS", "B", "BS", "L", "M"):
|
if self.attr.get_type(item) in ("S", "SS", "NS", "B", "BS", "L", "M"):
|
||||||
return len(self.attr.expr(item))
|
return len(self.attr.expr(item))
|
||||||
|
@ -157,18 +157,14 @@ class ItemSizeToUpdateTooLarge(MockValidationException):
|
|||||||
|
|
||||||
class HashKeyTooLong(MockValidationException):
|
class HashKeyTooLong(MockValidationException):
|
||||||
# deliberately no space between of and {lim}
|
# deliberately no space between of and {lim}
|
||||||
key_too_large_msg = "One or more parameter values were invalid: Size of hashkey has exceeded the maximum size limit of{lim} bytes".format(
|
key_too_large_msg = f"One or more parameter values were invalid: Size of hashkey has exceeded the maximum size limit of{HASH_KEY_MAX_LENGTH} bytes"
|
||||||
lim=HASH_KEY_MAX_LENGTH
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
super().__init__(self.key_too_large_msg)
|
super().__init__(self.key_too_large_msg)
|
||||||
|
|
||||||
|
|
||||||
class RangeKeyTooLong(MockValidationException):
|
class RangeKeyTooLong(MockValidationException):
|
||||||
key_too_large_msg = "One or more parameter values were invalid: Aggregated size of all range keys has exceeded the size limit of {lim} bytes".format(
|
key_too_large_msg = f"One or more parameter values were invalid: Aggregated size of all range keys has exceeded the size limit of {RANGE_KEY_MAX_LENGTH} bytes"
|
||||||
lim=RANGE_KEY_MAX_LENGTH
|
|
||||||
)
|
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
super().__init__(self.key_too_large_msg)
|
super().__init__(self.key_too_large_msg)
|
||||||
@ -285,26 +281,25 @@ class ResourceNotFoundException(JsonRESTError):
|
|||||||
class TableNotFoundException(JsonRESTError):
|
class TableNotFoundException(JsonRESTError):
|
||||||
def __init__(self, name):
|
def __init__(self, name):
|
||||||
err = "com.amazonaws.dynamodb.v20111205#TableNotFoundException"
|
err = "com.amazonaws.dynamodb.v20111205#TableNotFoundException"
|
||||||
msg = "Table not found: {}".format(name)
|
super().__init__(err, f"Table not found: {name}")
|
||||||
super().__init__(err, msg)
|
|
||||||
|
|
||||||
|
|
||||||
class SourceTableNotFoundException(JsonRESTError):
|
class SourceTableNotFoundException(JsonRESTError):
|
||||||
def __init__(self, source_table_name):
|
def __init__(self, source_table_name):
|
||||||
er = "com.amazonaws.dynamodb.v20111205#SourceTableNotFoundException"
|
er = "com.amazonaws.dynamodb.v20111205#SourceTableNotFoundException"
|
||||||
super().__init__(er, "Source table not found: %s" % source_table_name)
|
super().__init__(er, f"Source table not found: {source_table_name}")
|
||||||
|
|
||||||
|
|
||||||
class BackupNotFoundException(JsonRESTError):
|
class BackupNotFoundException(JsonRESTError):
|
||||||
def __init__(self, backup_arn):
|
def __init__(self, backup_arn):
|
||||||
er = "com.amazonaws.dynamodb.v20111205#BackupNotFoundException"
|
er = "com.amazonaws.dynamodb.v20111205#BackupNotFoundException"
|
||||||
super().__init__(er, "Backup not found: %s" % backup_arn)
|
super().__init__(er, f"Backup not found: {backup_arn}")
|
||||||
|
|
||||||
|
|
||||||
class TableAlreadyExistsException(JsonRESTError):
|
class TableAlreadyExistsException(JsonRESTError):
|
||||||
def __init__(self, target_table_name):
|
def __init__(self, target_table_name):
|
||||||
er = "com.amazonaws.dynamodb.v20111205#TableAlreadyExistsException"
|
er = "com.amazonaws.dynamodb.v20111205#TableAlreadyExistsException"
|
||||||
super().__init__(er, "Table already exists: %s" % target_table_name)
|
super().__init__(er, f"Table already exists: {target_table_name}")
|
||||||
|
|
||||||
|
|
||||||
class ResourceInUseException(JsonRESTError):
|
class ResourceInUseException(JsonRESTError):
|
||||||
|
@ -97,7 +97,7 @@ class Item(BaseModel):
|
|||||||
)
|
)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "Item: {0}".format(self.to_json())
|
return f"Item: {self.to_json()}"
|
||||||
|
|
||||||
def size(self):
|
def size(self):
|
||||||
return sum(bytesize(key) + value.size() for key, value in self.attrs.items())
|
return sum(bytesize(key) + value.size() for key, value in self.attrs.items())
|
||||||
@ -191,7 +191,7 @@ class Item(BaseModel):
|
|||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(
|
raise NotImplementedError(
|
||||||
"%s action not support for update_with_attribute_updates" % action
|
f"{action} action not support for update_with_attribute_updates"
|
||||||
)
|
)
|
||||||
|
|
||||||
# Filter using projection_expression
|
# Filter using projection_expression
|
||||||
@ -814,9 +814,9 @@ class Table(CloudFormationModel):
|
|||||||
all_indexes = self.all_indexes()
|
all_indexes = self.all_indexes()
|
||||||
indexes_by_name = dict((i.name, i) for i in all_indexes)
|
indexes_by_name = dict((i.name, i) for i in all_indexes)
|
||||||
if index_name not in indexes_by_name:
|
if index_name not in indexes_by_name:
|
||||||
|
all_indexes = ", ".join(indexes_by_name.keys())
|
||||||
raise MockValidationException(
|
raise MockValidationException(
|
||||||
"Invalid index: %s for table: %s. Available indexes are: %s"
|
f"Invalid index: {index_name} for table: {self.name}. Available indexes are: {all_indexes}"
|
||||||
% (index_name, self.name, ", ".join(indexes_by_name.keys()))
|
|
||||||
)
|
)
|
||||||
|
|
||||||
index = indexes_by_name[index_name]
|
index = indexes_by_name[index_name]
|
||||||
@ -826,7 +826,7 @@ class Table(CloudFormationModel):
|
|||||||
][0]
|
][0]
|
||||||
except IndexError:
|
except IndexError:
|
||||||
raise MockValidationException(
|
raise MockValidationException(
|
||||||
"Missing Hash Key. KeySchema: %s" % index.name
|
f"Missing Hash Key. KeySchema: {index.name}"
|
||||||
)
|
)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -945,7 +945,7 @@ class Table(CloudFormationModel):
|
|||||||
indexes_by_name = dict((i.name, i) for i in all_indexes)
|
indexes_by_name = dict((i.name, i) for i in all_indexes)
|
||||||
if error_if_not and index_name not in indexes_by_name:
|
if error_if_not and index_name not in indexes_by_name:
|
||||||
raise InvalidIndexNameError(
|
raise InvalidIndexNameError(
|
||||||
"The table does not have the specified index: %s" % index_name
|
f"The table does not have the specified index: {index_name}"
|
||||||
)
|
)
|
||||||
return indexes_by_name[index_name]
|
return indexes_by_name[index_name]
|
||||||
|
|
||||||
@ -1139,16 +1139,11 @@ class Backup(object):
|
|||||||
timestamp_padded = str("0" + str(timestamp))[-16:16]
|
timestamp_padded = str("0" + str(timestamp))[-16:16]
|
||||||
guid = str(mock_random.uuid4())
|
guid = str(mock_random.uuid4())
|
||||||
guid_shortened = guid[:8]
|
guid_shortened = guid[:8]
|
||||||
return "{}-{}".format(timestamp_padded, guid_shortened)
|
return f"{timestamp_padded}-{guid_shortened}"
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def arn(self):
|
def arn(self):
|
||||||
return "arn:aws:dynamodb:{region}:{account}:table/{table_name}/backup/{identifier}".format(
|
return f"arn:aws:dynamodb:{self.backend.region_name}:{self.backend.account_id}:table/{self.table.name}/backup/{self.identifier}"
|
||||||
region=self.backend.region_name,
|
|
||||||
account=self.backend.account_id,
|
|
||||||
table_name=self.table.name,
|
|
||||||
identifier=self.identifier,
|
|
||||||
)
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def details(self):
|
def details(self):
|
||||||
@ -1227,7 +1222,7 @@ class DynamoDBBackend(BaseBackend):
|
|||||||
def describe_endpoints(self):
|
def describe_endpoints(self):
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
"Address": "dynamodb.{}.amazonaws.com".format(self.region_name),
|
"Address": f"dynamodb.{self.region_name}.amazonaws.com",
|
||||||
"CachePeriodInMinutes": 1440,
|
"CachePeriodInMinutes": 1440,
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
@ -1420,10 +1415,9 @@ class DynamoDBBackend(BaseBackend):
|
|||||||
all_indexes = (table.global_indexes or []) + (table.indexes or [])
|
all_indexes = (table.global_indexes or []) + (table.indexes or [])
|
||||||
indexes_by_name = dict((i.name, i) for i in all_indexes)
|
indexes_by_name = dict((i.name, i) for i in all_indexes)
|
||||||
if index_name not in indexes_by_name:
|
if index_name not in indexes_by_name:
|
||||||
|
all_indexes = ", ".join(indexes_by_name.keys())
|
||||||
raise ResourceNotFoundException(
|
raise ResourceNotFoundException(
|
||||||
"Invalid index: {} for table: {}. Available indexes are: {}".format(
|
f"Invalid index: {index_name} for table: {table_name}. Available indexes are: {all_indexes}"
|
||||||
index_name, table_name, ", ".join(indexes_by_name.keys())
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
return indexes_by_name[index_name].schema
|
return indexes_by_name[index_name].schema
|
||||||
|
@ -100,7 +100,7 @@ class DynamoType(object):
|
|||||||
return self.cast_value >= other.cast_value
|
return self.cast_value >= other.cast_value
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "DynamoType: {0}".format(self.to_json())
|
return f"DynamoType: {self.to_json()}"
|
||||||
|
|
||||||
def __add__(self, other):
|
def __add__(self, other):
|
||||||
if self.type != other.type:
|
if self.type != other.type:
|
||||||
@ -108,9 +108,7 @@ class DynamoType(object):
|
|||||||
if self.is_number():
|
if self.is_number():
|
||||||
self_value = float(self.value) if "." in self.value else int(self.value)
|
self_value = float(self.value) if "." in self.value else int(self.value)
|
||||||
other_value = float(other.value) if "." in other.value else int(other.value)
|
other_value = float(other.value) if "." in other.value else int(other.value)
|
||||||
return DynamoType(
|
return DynamoType({DDBType.NUMBER: f"{self_value + other_value}"})
|
||||||
{DDBType.NUMBER: "{v}".format(v=self_value + other_value)}
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
raise IncorrectDataType()
|
raise IncorrectDataType()
|
||||||
|
|
||||||
@ -120,9 +118,7 @@ class DynamoType(object):
|
|||||||
if self.type == DDBType.NUMBER:
|
if self.type == DDBType.NUMBER:
|
||||||
self_value = float(self.value) if "." in self.value else int(self.value)
|
self_value = float(self.value) if "." in self.value else int(self.value)
|
||||||
other_value = float(other.value) if "." in other.value else int(other.value)
|
other_value = float(other.value) if "." in other.value else int(other.value)
|
||||||
return DynamoType(
|
return DynamoType({DDBType.NUMBER: f"{self_value - other_value}"})
|
||||||
{DDBType.NUMBER: "{v}".format(v=self_value - other_value)}
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
raise TypeError("Sum only supported for Numbers.")
|
raise TypeError("Sum only supported for Numbers.")
|
||||||
|
|
||||||
@ -136,9 +132,7 @@ class DynamoType(object):
|
|||||||
if self.type == DDBType.LIST:
|
if self.type == DDBType.LIST:
|
||||||
return self.value[item]
|
return self.value[item]
|
||||||
raise TypeError(
|
raise TypeError(
|
||||||
"This DynamoType {dt} is not subscriptable by a {it}".format(
|
f"This DynamoType {self.type} is not subscriptable by a {type(item)}"
|
||||||
dt=self.type, it=type(item)
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
def __setitem__(self, key, value):
|
def __setitem__(self, key, value):
|
||||||
@ -153,7 +147,7 @@ class DynamoType(object):
|
|||||||
if self.is_map():
|
if self.is_map():
|
||||||
self.value[key] = value
|
self.value[key] = value
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError("No set_item for {t}".format(t=type(key)))
|
raise NotImplementedError(f"No set_item for {type(key)}")
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def cast_value(self):
|
def cast_value(self):
|
||||||
@ -237,4 +231,4 @@ class DynamoType(object):
|
|||||||
if self.is_map() or self.is_list():
|
if self.is_map() or self.is_list():
|
||||||
self.value.pop(key, *args, **kwargs)
|
self.value.pop(key, *args, **kwargs)
|
||||||
else:
|
else:
|
||||||
raise TypeError("pop not supported for DynamoType {t}".format(t=self.type))
|
raise TypeError(f"pop not supported for DynamoType {self.type}")
|
||||||
|
@ -129,7 +129,7 @@ class SetExecutor(NodeExecutor):
|
|||||||
item_part_to_modify_with_set[attribute_name] = value_to_set
|
item_part_to_modify_with_set[attribute_name] = value_to_set
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(
|
raise NotImplementedError(
|
||||||
"Moto does not support setting {t} yet".format(t=type(element_to_set))
|
f"Moto does not support setting {type(element_to_set)} yet"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -173,7 +173,7 @@ class DeleteExecutor(NodeExecutor):
|
|||||||
attribute_name = element.get_attribute_name()
|
attribute_name = element.get_attribute_name()
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(
|
raise NotImplementedError(
|
||||||
"Moto does not support deleting {t} yet".format(t=type(element))
|
f"Moto does not support deleting {type(element)} yet"
|
||||||
)
|
)
|
||||||
container = self.get_item_before_end_of_path(item)
|
container = self.get_item_before_end_of_path(item)
|
||||||
del container[attribute_name]
|
del container[attribute_name]
|
||||||
@ -203,9 +203,7 @@ class RemoveExecutor(NodeExecutor):
|
|||||||
pass
|
pass
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(
|
raise NotImplementedError(
|
||||||
"Moto does not support setting {t} yet".format(
|
f"Moto does not support setting {type(element_to_remove)} yet"
|
||||||
t=type(element_to_remove)
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -64,19 +64,15 @@ class NestableExpressionParserMixin(object):
|
|||||||
Returns:
|
Returns:
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
pos = self.token_pos
|
||||||
|
fc = factory_class.__class__.__name__
|
||||||
logger.debug(
|
logger.debug(
|
||||||
"Move token pos {pos} to continue parsing with specific factory class {fc}".format(
|
f"Move token pos {pos} to continue parsing with specific factory class {fc}"
|
||||||
pos=self.token_pos, fc=factory_class.__class__.__name__
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
# noinspection PyProtectedMember
|
# noinspection PyProtectedMember
|
||||||
ast, token_pos = factory_class(**self._initializer_args())._parse_with_pos()
|
ast, token_pos = factory_class(**self._initializer_args())._parse_with_pos()
|
||||||
self.target_clauses.append(ast)
|
self.target_clauses.append(ast)
|
||||||
logger.debug(
|
logger.debug(f"Continue where previous parsing ended {token_pos}")
|
||||||
"Continue where previous parsing ended {token_pos}".format(
|
|
||||||
token_pos=token_pos
|
|
||||||
)
|
|
||||||
)
|
|
||||||
self.token_pos = token_pos
|
self.token_pos = token_pos
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
@ -118,9 +114,8 @@ class NestableExpressionParserMixin(object):
|
|||||||
Returns:
|
Returns:
|
||||||
moto.dynamodb.ast_nodes.Node: Node of an AST representing the Expression as produced by the factory.
|
moto.dynamodb.ast_nodes.Node: Node of an AST representing the Expression as produced by the factory.
|
||||||
"""
|
"""
|
||||||
assert len(self.target_clauses) > 0, "No nodes for {cn}".format(
|
cn = self.__class__.__name__
|
||||||
cn=self.__class__.__name__
|
assert len(self.target_clauses) > 0, f"No nodes for {cn}"
|
||||||
)
|
|
||||||
target_node = self._nestable_class()(children=[self.target_clauses.pop()])
|
target_node = self._nestable_class()(children=[self.target_clauses.pop()])
|
||||||
while len(self.target_clauses) > 0:
|
while len(self.target_clauses) > 0:
|
||||||
target_node = self._nestable_class()(
|
target_node = self._nestable_class()(
|
||||||
@ -358,11 +353,7 @@ class NestableBinExpressionParser(ExpressionParser):
|
|||||||
**self._initializer_args()
|
**self._initializer_args()
|
||||||
)._parse_with_pos()
|
)._parse_with_pos()
|
||||||
self.target_nodes.append(ast)
|
self.target_nodes.append(ast)
|
||||||
logger.debug(
|
logger.debug(f"Continue where previous parsing ended {self.token_pos}")
|
||||||
"Continue where previous parsing ended {token_pos}".format(
|
|
||||||
token_pos=self.token_pos
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
def _parse(self):
|
def _parse(self):
|
||||||
self._parse_target_clause(self._operand_factory_class())
|
self._parse_target_clause(self._operand_factory_class())
|
||||||
@ -525,11 +516,8 @@ class UpdateExpressionActionsParser(ExpressionParser, NestableExpressionParserMi
|
|||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _is_possible_start(cls, token):
|
def _is_possible_start(cls, token):
|
||||||
raise RuntimeError(
|
cn = cls._nestable_class().__name__
|
||||||
"{class_name} cannot be identified by the next token.".format(
|
raise RuntimeError(f"{cn} cannot be identified by the next token.")
|
||||||
class_name=cls._nestable_class().__name__
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
@ -562,12 +550,9 @@ class UpdateExpressionActionsParser(ExpressionParser, NestableExpressionParserMi
|
|||||||
break
|
break
|
||||||
|
|
||||||
if len(self.target_clauses) == 0:
|
if len(self.target_clauses) == 0:
|
||||||
logger.debug(
|
nc = self._nestable_class().__name__
|
||||||
"Didn't encounter a single {nc} in {nepc}.".format(
|
nepc = self._nested_expression_parser_class().__name__
|
||||||
nc=self._nestable_class().__name__,
|
logger.debug(f"Didn't encounter a single {nc} in {nepc}.")
|
||||||
nepc=self._nested_expression_parser_class().__name__,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
self.raise_unexpected_token()
|
self.raise_unexpected_token()
|
||||||
|
|
||||||
return self._create_node()
|
return self._create_node()
|
||||||
|
@ -63,11 +63,9 @@ class Token(object):
|
|||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
if isinstance(self.type, int):
|
if isinstance(self.type, int):
|
||||||
return 'Token("{tt}", "{tv}")'.format(
|
return f'Token("{self.PLACEHOLDER_NAMES[self.type]}", "{self.value}")'
|
||||||
tt=self.PLACEHOLDER_NAMES[self.type], tv=self.value
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
return 'Token("{tt}", "{tv}")'.format(tt=self.type, tv=self.value)
|
return f'Token("{self.type}", "{self.value}")'
|
||||||
|
|
||||||
def __eq__(self, other):
|
def __eq__(self, other):
|
||||||
return self.type == other.type and self.value == other.value
|
return self.type == other.type and self.value == other.value
|
||||||
|
@ -112,9 +112,7 @@ class ExpressionPathResolver(object):
|
|||||||
else:
|
else:
|
||||||
raise InvalidUpdateExpressionInvalidDocumentPath
|
raise InvalidUpdateExpressionInvalidDocumentPath
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(
|
raise NotImplementedError(f"Path resolution for {type(child)}")
|
||||||
"Path resolution for {t}".format(t=type(child))
|
|
||||||
)
|
|
||||||
return DDBTypedValue(target)
|
return DDBTypedValue(target)
|
||||||
|
|
||||||
def resolve_expression_path_nodes_to_dynamo_type(
|
def resolve_expression_path_nodes_to_dynamo_type(
|
||||||
@ -216,9 +214,7 @@ class UpdateExpressionFunctionEvaluator(DepthFirstTraverser):
|
|||||||
first_arg.value.append(list_element)
|
first_arg.value.append(list_element)
|
||||||
return DDBTypedValue(first_arg)
|
return DDBTypedValue(first_arg)
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(
|
raise NotImplementedError(f"Unsupported function for moto {function_name}")
|
||||||
"Unsupported function for moto {name}".format(name=function_name)
|
|
||||||
)
|
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def get_list_from_ddb_typed_value(cls, node, function_name):
|
def get_list_from_ddb_typed_value(cls, node, function_name):
|
||||||
@ -270,11 +266,7 @@ class ExecuteOperations(DepthFirstTraverser):
|
|||||||
elif operator == "-":
|
elif operator == "-":
|
||||||
return self.get_subtraction(left_operand, right_operand)
|
return self.get_subtraction(left_operand, right_operand)
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(
|
raise NotImplementedError(f"Moto does not support operator {operator}")
|
||||||
"Moto does not support operator {operator}".format(
|
|
||||||
operator=operator
|
|
||||||
)
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError(
|
raise NotImplementedError(
|
||||||
"UpdateExpressionValue only has implementations for 1 or 3 children."
|
"UpdateExpressionValue only has implementations for 1 or 3 children."
|
||||||
|
@ -28,9 +28,7 @@ def include_consumed_capacity(val=1.0):
|
|||||||
expected_capacity = handler.body.get("ReturnConsumedCapacity", "NONE")
|
expected_capacity = handler.body.get("ReturnConsumedCapacity", "NONE")
|
||||||
if expected_capacity not in ["NONE", "TOTAL", "INDEXES"]:
|
if expected_capacity not in ["NONE", "TOTAL", "INDEXES"]:
|
||||||
type_ = "ValidationException"
|
type_ = "ValidationException"
|
||||||
message = "1 validation error detected: Value '{}' at 'returnConsumedCapacity' failed to satisfy constraint: Member must satisfy enum value set: [INDEXES, TOTAL, NONE]".format(
|
message = f"1 validation error detected: Value '{expected_capacity}' at 'returnConsumedCapacity' failed to satisfy constraint: Member must satisfy enum value set: [INDEXES, TOTAL, NONE]"
|
||||||
expected_capacity
|
|
||||||
)
|
|
||||||
return (
|
return (
|
||||||
400,
|
400,
|
||||||
handler.response_headers,
|
handler.response_headers,
|
||||||
@ -730,7 +728,7 @@ class DynamoHandler(BaseResponse):
|
|||||||
projection_expression,
|
projection_expression,
|
||||||
)
|
)
|
||||||
except ValueError as err:
|
except ValueError as err:
|
||||||
raise MockValidationException("Bad Filter Expression: {0}".format(err))
|
raise MockValidationException(f"Bad Filter Expression: {err}")
|
||||||
|
|
||||||
result = {
|
result = {
|
||||||
"Count": len(items),
|
"Count": len(items),
|
||||||
|
@ -34,7 +34,7 @@ class DynamoType(object):
|
|||||||
return self.type == other.type and self.value == other.value
|
return self.type == other.type and self.value == other.value
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "DynamoType: {0}".format(self.to_json())
|
return f"DynamoType: {self.to_json()}"
|
||||||
|
|
||||||
def add(self, dyn_type):
|
def add(self, dyn_type):
|
||||||
if self.type == "SS":
|
if self.type == "SS":
|
||||||
@ -66,7 +66,7 @@ class Item(BaseModel):
|
|||||||
self.attrs[key] = DynamoType(value)
|
self.attrs[key] = DynamoType(value)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "Item: {0}".format(self.to_json())
|
return f"Item: {self.to_json()}"
|
||||||
|
|
||||||
def to_json(self):
|
def to_json(self):
|
||||||
attributes = {}
|
attributes = {}
|
||||||
|
@ -27,11 +27,7 @@ class ShardIterator(BaseModel):
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def arn(self):
|
def arn(self):
|
||||||
return "{}/stream/{}|1|{}".format(
|
return f"{self.stream_shard.table.table_arn}/stream/{self.stream_shard.table.latest_stream_label}|1|{self.id}"
|
||||||
self.stream_shard.table.table_arn,
|
|
||||||
self.stream_shard.table.latest_stream_label,
|
|
||||||
self.id,
|
|
||||||
)
|
|
||||||
|
|
||||||
def to_json(self):
|
def to_json(self):
|
||||||
return {"ShardIterator": self.arn}
|
return {"ShardIterator": self.arn}
|
||||||
|
Loading…
Reference in New Issue
Block a user