import copy import gc import warnings from functools import wraps from moto import settings from moto.s3 import models as s3model from moto.s3.responses import S3ResponseInstance from unittest import SkipTest, TestCase def verify_zero_warnings(f): @wraps(f) def wrapped(*args, **kwargs): with warnings.catch_warnings(record=True) as warning_list: warnings.simplefilter("always", ResourceWarning) # add filter resp = f(*args, **kwargs) # Get the TestClass reference, and reset the S3Backend that we've created as part of that class (class_ref,) = args class_ref.__dict__["s3"].reset() # Now collect garbage, which will throw any warnings if there are unclosed FileHandles gc.collect() warning_types = [type(warn.message) for warn in warning_list] warning_types.shouldnt.contain(ResourceWarning) return resp return wrapped class TestS3FileHandleClosures(TestCase): """ Large Uploads are written to disk for performance reasons These tests verifies that the filehandles are properly closed after specific actions """ def setUp(self) -> None: if settings.TEST_SERVER_MODE: raise SkipTest("No point in testing ServerMode, we're not using boto3") self.s3 = s3model.S3Backend("us-west-1", "1234") self.s3.create_bucket("my-bucket", "us-west-1") self.s3.create_bucket("versioned-bucket", "us-west-1") self.s3.put_object("my-bucket", "my-key", "x" * 10_000_000) def tearDown(self) -> None: self.s3.reset() @verify_zero_warnings def test_upload_large_file(self): # Verify that we can create an object, as done in the setUp-method # Do not create any other objects, as it will taint the state that we're testing pass @verify_zero_warnings def test_delete_large_file(self): self.s3.delete_object(bucket_name="my-bucket", key_name="my-key") @verify_zero_warnings def test_overwriting_file(self): self.s3.put_object("my-bucket", "my-key", "b" * 10_000_000) @verify_zero_warnings def test_versioned_file(self): self.s3.put_bucket_versioning("my-bucket", "Enabled") self.s3.put_object("my-bucket", "my-key", "b" * 10_000_000) @verify_zero_warnings def test_copy_object(self): key = self.s3.get_object("my-bucket", "my-key") self.s3.copy_object( src_key=key, dest_bucket_name="my-bucket", dest_key_name="key-2" ) @verify_zero_warnings def test_part_upload(self): multipart_id = self.s3.create_multipart_upload( bucket_name="my-bucket", key_name="mp-key", metadata={}, storage_type="STANDARD", tags=[], acl=None, sse_encryption=None, kms_key_id=None, ) self.s3.upload_part( bucket_name="my-bucket", multipart_id=multipart_id, part_id=1, value="b" * 10_000_000, ) @verify_zero_warnings def test_overwriting_part_upload(self): multipart_id = self.s3.create_multipart_upload( bucket_name="my-bucket", key_name="mp-key", metadata={}, storage_type="STANDARD", tags=[], acl=None, sse_encryption=None, kms_key_id=None, ) self.s3.upload_part( bucket_name="my-bucket", multipart_id=multipart_id, part_id=1, value="b" * 10_000_000, ) self.s3.upload_part( bucket_name="my-bucket", multipart_id=multipart_id, part_id=1, value="c" * 10_000_000, ) @verify_zero_warnings def test_aborting_part_upload(self): multipart_id = self.s3.create_multipart_upload( bucket_name="my-bucket", key_name="mp-key", metadata={}, storage_type="STANDARD", tags=[], acl=None, sse_encryption=None, kms_key_id=None, ) self.s3.upload_part( bucket_name="my-bucket", multipart_id=multipart_id, part_id=1, value="b" * 10_000_000, ) self.s3.abort_multipart_upload( bucket_name="my-bucket", multipart_id=multipart_id ) @verify_zero_warnings def test_completing_part_upload(self): multipart_id = self.s3.create_multipart_upload( bucket_name="my-bucket", key_name="mp-key", metadata={}, storage_type="STANDARD", tags=[], acl=None, sse_encryption=None, kms_key_id=None, ) etag = self.s3.upload_part( bucket_name="my-bucket", multipart_id=multipart_id, part_id=1, value="b" * 10_000_000, ).etag mp_body = f"""{etag}1""" body = S3ResponseInstance._complete_multipart_body(mp_body) self.s3.complete_multipart_upload( bucket_name="my-bucket", multipart_id=multipart_id, body=body ) @verify_zero_warnings def test_single_versioned_upload(self): self.s3.put_object("versioned-bucket", "my-key", "x" * 10_000_000) @verify_zero_warnings def test_overwrite_versioned_upload(self): self.s3.put_object("versioned-bucket", "my-key", "x" * 10_000_000) @verify_zero_warnings def test_multiple_versions_upload(self): self.s3.put_object("versioned-bucket", "my-key", "x" * 10_000_000) self.s3.put_object("versioned-bucket", "my-key", "y" * 10_000_000) self.s3.put_object("versioned-bucket", "my-key", "z" * 10_000_000) @verify_zero_warnings def test_delete_versioned_upload(self): self.s3.put_object("versioned-bucket", "my-key", "x" * 10_000_000) self.s3.put_object("versioned-bucket", "my-key", "x" * 10_000_000) self.s3.delete_object(bucket_name="my-bucket", key_name="my-key") @verify_zero_warnings def test_delete_specific_version(self): self.s3.put_object("versioned-bucket", "my-key", "x" * 10_000_000) key = self.s3.put_object("versioned-bucket", "my-key", "y" * 10_000_000) self.s3.delete_object( bucket_name="my-bucket", key_name="my-key", version_id=key._version_id ) def test_verify_key_can_be_copied_after_disposing(): # https://github.com/getmoto/moto/issues/5588 # Multithreaded bug where: # - User: calls list_object_versions # - Moto creates a list of all keys # - User: deletes a key # - Moto iterates over the previously created list, that contains a now-deleted key, and creates a copy of it # # This test verifies the copy-operation succeeds, it will just not have any data key = s3model.FakeKey(name="test", bucket_name="bucket", value="sth") assert not key._value_buffer.closed key.dispose() assert key._value_buffer.closed copied_key = copy.copy(key) copied_key.value.should.equal(b"")