S3: Days must not be provided for select request when restoring object (#7468)

This commit is contained in:
Akira Noda 2024-03-16 04:52:07 +09:00 committed by GitHub
parent a125b3cdce
commit 6e0b4c6519
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 65 additions and 5 deletions

View File

@ -584,3 +584,23 @@ class HeadOnDeleteMarker(Exception):
def __init__(self, marker: "FakeDeleteMarker"):
self.marker = marker
class DaysMustNotProvidedForSelectRequest(S3ClientError):
code = 400
def __init__(self) -> None:
super().__init__(
"DaysMustNotProvidedForSelectRequest",
"`Days` must not be provided for select requests",
)
class DaysMustProvidedExceptForSelectRequest(S3ClientError):
code = 400
def __init__(self) -> None:
super().__init__(
"DaysMustProvidedExceptForSelectRequest",
"`Days` must be provided except for select requests",
)

View File

@ -37,6 +37,8 @@ from moto.s3.exceptions import (
BucketNeedsToBeNew,
CopyObjectMustChangeSomething,
CrossLocationLoggingProhibitted,
DaysMustNotProvidedForSelectRequest,
DaysMustProvidedExceptForSelectRequest,
EntityTooSmall,
HeadOnDeleteMarker,
InvalidBucketName,
@ -2882,13 +2884,23 @@ class S3Backend(BaseBackend, CloudWatchMetricProvider):
for x in query_result
]
def restore_object(self, bucket_name: str, key_name: str, days: str) -> bool:
def restore_object(
self, bucket_name: str, key_name: str, days: Optional[str], type_: Optional[str]
) -> bool:
key = self.get_object(bucket_name, key_name)
if not key:
raise MissingKey
if days is None and type_ is None:
raise DaysMustProvidedExceptForSelectRequest()
if days and type_:
raise DaysMustNotProvidedForSelectRequest()
if key.storage_class not in ARCHIVE_STORAGE_CLASSES:
raise InvalidObjectState(storage_class=key.storage_class)
had_expiry_date = key.expiry_date is not None
if days:
key.restore(int(days))
return had_expiry_date

View File

@ -2272,10 +2272,12 @@ class S3Response(BaseResponse):
)
elif "restore" in query:
es = minidom.parseString(body).getElementsByTagName("Days")
days = es[0].childNodes[0].wholeText
params = xmltodict.parse(body)["RestoreRequest"]
previously_restored = self.backend.restore_object(
bucket_name, key_name, days
bucket_name,
key_name,
params.get("Days", None),
params.get("Type", None),
)
status_code = 200 if previously_restored else 202
return status_code, {}, ""

View File

@ -627,6 +627,32 @@ def test_cannot_restore_standard_class_object():
)
@mock_aws
def test_restore_object_invalid_request_params():
if not settings.TEST_DECORATOR_MODE:
raise SkipTest("Can't set transition directly in ServerMode")
s3_resource = boto3.resource("s3", region_name=DEFAULT_REGION_NAME)
bucket = s3_resource.Bucket("foobar")
bucket.create()
key = bucket.put_object(Key="the-key", Body=b"somedata", StorageClass="GLACIER")
# `Days` must be provided except for select requests
with pytest.raises(ClientError) as exc:
key.restore_object(RestoreRequest={})
err = exc.value.response["Error"]
assert err["Code"] == "DaysMustProvidedExceptForSelectRequest"
assert err["Message"] == "`Days` must be provided except for select requests"
# `Days` must not be provided for select requests
with pytest.raises(ClientError) as exc:
key.restore_object(RestoreRequest={"Days": 1, "Type": "SELECT"})
err = exc.value.response["Error"]
assert err["Code"] == "DaysMustNotProvidedForSelectRequest"
assert err["Message"] == "`Days` must not be provided for select requests"
@mock_aws
def test_get_versioning_status():
s3_resource = boto3.resource("s3", region_name=DEFAULT_REGION_NAME)