| 
									
										
										
										
											2014-11-22 10:19:56 -05:00
										 |  |  | # -*- coding: utf-8 -*- | 
					
						
							| 
									
										
										
										
											2014-08-27 11:17:06 -04:00
										 |  |  | from __future__ import unicode_literals | 
					
						
							| 
									
										
										
										
											2017-05-19 15:59:25 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | import datetime | 
					
						
							| 
									
										
										
										
											2014-08-26 13:25:50 -04:00
										 |  |  | from six.moves.urllib.request import urlopen | 
					
						
							|  |  |  | from six.moves.urllib.error import HTTPError | 
					
						
							| 
									
										
										
										
											2015-02-27 18:42:37 -05:00
										 |  |  | from functools import wraps | 
					
						
							| 
									
										
										
										
											2017-06-03 19:06:49 -04:00
										 |  |  | from gzip import GzipFile | 
					
						
							| 
									
										
										
										
											2013-03-26 14:52:33 +00:00
										 |  |  | from io import BytesIO | 
					
						
							| 
									
										
										
										
											2017-06-03 19:06:49 -04:00
										 |  |  | import zlib | 
					
						
							| 
									
										
										
										
											2018-12-20 11:15:15 -08:00
										 |  |  | import pickle | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-23 17:33:52 -04:00
										 |  |  | import json | 
					
						
							| 
									
										
										
										
											2013-02-18 16:09:40 -05:00
										 |  |  | import boto | 
					
						
							| 
									
										
										
										
											2015-08-02 09:54:23 -04:00
										 |  |  | import boto3 | 
					
						
							| 
									
										
										
										
											2015-08-13 21:16:55 +00:00
										 |  |  | from botocore.client import ClientError | 
					
						
							| 
									
										
										
										
											2017-05-19 15:59:25 -07:00
										 |  |  | import botocore.exceptions | 
					
						
							| 
									
										
										
										
											2014-03-16 21:25:14 -04:00
										 |  |  | from boto.exception import S3CreateError, S3ResponseError | 
					
						
							| 
									
										
										
										
											2017-09-16 15:48:20 -07:00
										 |  |  | from botocore.handlers import disable_signing | 
					
						
							| 
									
										
										
										
											2014-07-08 20:35:48 -04:00
										 |  |  | from boto.s3.connection import S3Connection | 
					
						
							| 
									
										
										
										
											2013-02-18 16:09:40 -05:00
										 |  |  | from boto.s3.key import Key | 
					
						
							| 
									
										
										
										
											2013-03-29 17:45:33 -04:00
										 |  |  | from freezegun import freeze_time | 
					
						
							| 
									
										
										
										
											2017-04-30 08:03:46 +03:00
										 |  |  | import six | 
					
						
							| 
									
										
										
										
											2013-02-25 23:21:49 -05:00
										 |  |  | import requests | 
					
						
							| 
									
										
										
										
											2014-11-19 21:06:23 -05:00
										 |  |  | import tests.backport_assert_raises  # noqa | 
					
						
							| 
									
										
										
										
											2014-08-26 13:25:50 -04:00
										 |  |  | from nose.tools import assert_raises | 
					
						
							| 
									
										
										
										
											2013-02-18 16:09:40 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-03 17:21:25 -04:00
										 |  |  | import sure  # noqa | 
					
						
							| 
									
										
										
										
											2013-02-18 17:17:19 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 19:43:48 -05:00
										 |  |  | from moto import settings, mock_s3, mock_s3_deprecated | 
					
						
							|  |  |  | import moto.s3.models as s3model | 
					
						
							| 
									
										
										
										
											2013-02-18 16:09:40 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 19:43:48 -05:00
										 |  |  | if settings.TEST_SERVER_MODE: | 
					
						
							|  |  |  |     REDUCED_PART_SIZE = s3model.UPLOAD_PART_MIN_SIZE | 
					
						
							|  |  |  |     EXPECTED_ETAG = '"140f92a6df9f9e415f74a1463bcee9bb-2"' | 
					
						
							|  |  |  | else: | 
					
						
							|  |  |  |     REDUCED_PART_SIZE = 256 | 
					
						
							|  |  |  |     EXPECTED_ETAG = '"66d1a1a2ed08fd05c137f316af4ff255-2"' | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def reduced_min_part_size(f): | 
					
						
							|  |  |  |     """ speed up tests by temporarily making the multipart minimum part size
 | 
					
						
							|  |  |  |         small | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     orig_size = s3model.UPLOAD_PART_MIN_SIZE | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-27 18:42:37 -05:00
										 |  |  |     @wraps(f) | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  |     def wrapped(*args, **kwargs): | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             s3model.UPLOAD_PART_MIN_SIZE = REDUCED_PART_SIZE | 
					
						
							|  |  |  |             return f(*args, **kwargs) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             s3model.UPLOAD_PART_MIN_SIZE = orig_size | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  |     return wrapped | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-18 16:09:40 -05:00
										 |  |  | class MyModel(object): | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-18 16:09:40 -05:00
										 |  |  |     def __init__(self, name, value): | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.value = value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def save(self): | 
					
						
							| 
									
										
										
										
											2017-02-23 19:43:48 -05:00
										 |  |  |         s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |         s3.put_object(Bucket='mybucket', Key=self.name, Body=self.value) | 
					
						
							| 
									
										
										
										
											2013-02-18 16:09:40 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-20 11:15:15 -08:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_keys_are_pickleable(): | 
					
						
							|  |  |  |     """Keys must be pickleable due to boto3 implementation details.""" | 
					
						
							|  |  |  |     key = s3model.FakeKey('name', b'data!') | 
					
						
							| 
									
										
										
										
											2018-12-20 11:40:13 -08:00
										 |  |  |     assert key.value == b'data!' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-20 11:15:15 -08:00
										 |  |  |     pickled = pickle.dumps(key) | 
					
						
							|  |  |  |     loaded = pickle.loads(pickled) | 
					
						
							|  |  |  |     assert loaded.value == key.value | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-20 11:40:13 -08:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_append_to_value__basic(): | 
					
						
							|  |  |  |     key = s3model.FakeKey('name', b'data!') | 
					
						
							|  |  |  |     assert key.value == b'data!' | 
					
						
							|  |  |  |     assert key.size == 5 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key.append_to_value(b' And even more data') | 
					
						
							|  |  |  |     assert key.value == b'data! And even more data' | 
					
						
							|  |  |  |     assert key.size == 24 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_append_to_value__nothing_added(): | 
					
						
							|  |  |  |     key = s3model.FakeKey('name', b'data!') | 
					
						
							|  |  |  |     assert key.value == b'data!' | 
					
						
							|  |  |  |     assert key.size == 5 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key.append_to_value(b'') | 
					
						
							|  |  |  |     assert key.value == b'data!' | 
					
						
							|  |  |  |     assert key.size == 5 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_append_to_value__empty_key(): | 
					
						
							|  |  |  |     key = s3model.FakeKey('name', b'') | 
					
						
							|  |  |  |     assert key.value == b'' | 
					
						
							|  |  |  |     assert key.size == 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key.append_to_value(b'stuff') | 
					
						
							|  |  |  |     assert key.value == b'stuff' | 
					
						
							|  |  |  |     assert key.size == 5 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 19:43:48 -05:00
										 |  |  | @mock_s3 | 
					
						
							| 
									
										
										
										
											2013-02-18 16:09:40 -05:00
										 |  |  | def test_my_model_save(): | 
					
						
							|  |  |  |     # Create Bucket so that test can run | 
					
						
							| 
									
										
										
										
											2017-02-23 19:43:48 -05:00
										 |  |  |     conn = boto3.resource('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     conn.create_bucket(Bucket='mybucket') | 
					
						
							| 
									
										
										
										
											2013-02-18 16:09:40 -05:00
										 |  |  |     #################################### | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     model_instance = MyModel('steve', 'is awesome') | 
					
						
							|  |  |  |     model_instance.save() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-26 22:57:55 -07:00
										 |  |  |     body = conn.Object('mybucket', 'steve').get()['Body'].read().decode() | 
					
						
							| 
									
										
										
										
											2017-02-23 19:43:48 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 22:28:09 -05:00
										 |  |  |     assert body == 'is awesome' | 
					
						
							| 
									
										
										
										
											2013-02-18 17:17:19 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-18 21:22:03 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 19:50:34 -05:00
										 |  |  | @mock_s3 | 
					
						
							| 
									
										
										
										
											2014-04-02 14:40:04 +03:00
										 |  |  | def test_key_etag(): | 
					
						
							| 
									
										
										
										
											2017-02-23 19:50:34 -05:00
										 |  |  |     conn = boto3.resource('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     conn.create_bucket(Bucket='mybucket') | 
					
						
							| 
									
										
										
										
											2014-04-02 14:40:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     model_instance = MyModel('steve', 'is awesome') | 
					
						
							|  |  |  |     model_instance.save() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 19:50:34 -05:00
										 |  |  |     conn.Bucket('mybucket').Object('steve').e_tag.should.equal( | 
					
						
							| 
									
										
										
										
											2014-04-02 14:40:04 +03:00
										 |  |  |         '"d32bda93738f7e03adb22e66c90fbc04"') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-11-15 11:59:30 +02:00
										 |  |  | def test_multipart_upload_too_small(): | 
					
						
							| 
									
										
										
										
											2013-03-26 14:52:33 +00:00
										 |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     multipart = bucket.initiate_multipart_upload("the-key") | 
					
						
							| 
									
										
										
										
											2014-08-27 11:54:23 -04:00
										 |  |  |     multipart.upload_part_from_file(BytesIO(b'hello'), 1) | 
					
						
							|  |  |  |     multipart.upload_part_from_file(BytesIO(b'world'), 2) | 
					
						
							| 
									
										
										
										
											2013-03-26 14:52:33 +00:00
										 |  |  |     # Multipart with total size under 5MB is refused | 
					
						
							| 
									
										
										
										
											2013-11-07 17:07:56 -05:00
										 |  |  |     multipart.complete_upload.should.throw(S3ResponseError) | 
					
						
							| 
									
										
										
										
											2013-03-26 14:52:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-15 11:59:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  | @reduced_min_part_size | 
					
						
							| 
									
										
										
										
											2013-11-15 11:59:30 +02:00
										 |  |  | def test_multipart_upload(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-26 14:52:33 +00:00
										 |  |  |     multipart = bucket.initiate_multipart_upload("the-key") | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  |     part1 = b'0' * REDUCED_PART_SIZE | 
					
						
							| 
									
										
										
										
											2013-11-15 11:59:30 +02:00
										 |  |  |     multipart.upload_part_from_file(BytesIO(part1), 1) | 
					
						
							|  |  |  |     # last part, can be less than 5 MB | 
					
						
							| 
									
										
										
										
											2014-08-27 11:54:23 -04:00
										 |  |  |     part2 = b'1' | 
					
						
							| 
									
										
										
										
											2013-11-15 11:59:30 +02:00
										 |  |  |     multipart.upload_part_from_file(BytesIO(part2), 2) | 
					
						
							| 
									
										
										
										
											2013-03-26 14:52:33 +00:00
										 |  |  |     multipart.complete_upload() | 
					
						
							| 
									
										
										
										
											2013-11-15 11:59:30 +02:00
										 |  |  |     # we should get both parts as the key contents | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "the-key").get_contents_as_string().should.equal(part1 + part2) | 
					
						
							| 
									
										
										
										
											2013-03-26 14:52:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  | @reduced_min_part_size | 
					
						
							| 
									
										
										
										
											2015-02-10 13:05:31 +02:00
										 |  |  | def test_multipart_upload_out_of_order(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     multipart = bucket.initiate_multipart_upload("the-key") | 
					
						
							|  |  |  |     # last part, can be less than 5 MB | 
					
						
							|  |  |  |     part2 = b'1' | 
					
						
							|  |  |  |     multipart.upload_part_from_file(BytesIO(part2), 4) | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  |     part1 = b'0' * REDUCED_PART_SIZE | 
					
						
							| 
									
										
										
										
											2015-02-10 13:05:31 +02:00
										 |  |  |     multipart.upload_part_from_file(BytesIO(part1), 2) | 
					
						
							|  |  |  |     multipart.complete_upload() | 
					
						
							|  |  |  |     # we should get both parts as the key contents | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "the-key").get_contents_as_string().should.equal(part1 + part2) | 
					
						
							| 
									
										
										
										
											2015-02-10 13:05:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  | @reduced_min_part_size | 
					
						
							| 
									
										
										
										
											2014-12-07 12:43:14 -05:00
										 |  |  | def test_multipart_upload_with_headers(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     multipart = bucket.initiate_multipart_upload( | 
					
						
							|  |  |  |         "the-key", metadata={"foo": "bar"}) | 
					
						
							| 
									
										
										
										
											2014-12-07 12:43:14 -05:00
										 |  |  |     part1 = b'0' * 10 | 
					
						
							|  |  |  |     multipart.upload_part_from_file(BytesIO(part1), 1) | 
					
						
							|  |  |  |     multipart.complete_upload() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = bucket.get_key("the-key") | 
					
						
							|  |  |  |     key.metadata.should.equal({"foo": "bar"}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  | @reduced_min_part_size | 
					
						
							| 
									
										
										
										
											2014-03-30 11:50:36 -04:00
										 |  |  | def test_multipart_upload_with_copy_key(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "original-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("key_value") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     multipart = bucket.initiate_multipart_upload("the-key") | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  |     part1 = b'0' * REDUCED_PART_SIZE | 
					
						
							| 
									
										
										
										
											2014-03-30 11:50:36 -04:00
										 |  |  |     multipart.upload_part_from_file(BytesIO(part1), 1) | 
					
						
							| 
									
										
										
										
											2016-05-05 22:52:12 -04:00
										 |  |  |     multipart.copy_part_from_key("foobar", "original-key", 2, 0, 3) | 
					
						
							| 
									
										
										
										
											2014-03-30 11:50:36 -04:00
										 |  |  |     multipart.complete_upload() | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "the-key").get_contents_as_string().should.equal(part1 + b"key_") | 
					
						
							| 
									
										
										
										
											2014-03-30 11:50:36 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  | @reduced_min_part_size | 
					
						
							| 
									
										
										
										
											2014-03-30 11:50:36 -04:00
										 |  |  | def test_multipart_upload_cancel(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     multipart = bucket.initiate_multipart_upload("the-key") | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  |     part1 = b'0' * REDUCED_PART_SIZE | 
					
						
							| 
									
										
										
										
											2014-03-30 11:50:36 -04:00
										 |  |  |     multipart.upload_part_from_file(BytesIO(part1), 1) | 
					
						
							|  |  |  |     multipart.cancel_upload() | 
					
						
							|  |  |  |     # TODO we really need some sort of assertion here, but we don't currently | 
					
						
							|  |  |  |     # have the ability to list mulipart uploads for a bucket. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  | @reduced_min_part_size | 
					
						
							| 
									
										
										
										
											2014-04-02 14:40:04 +03:00
										 |  |  | def test_multipart_etag(): | 
					
						
							|  |  |  |     # Create Bucket so that test can run | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('mybucket') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     multipart = bucket.initiate_multipart_upload("the-key") | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  |     part1 = b'0' * REDUCED_PART_SIZE | 
					
						
							| 
									
										
										
										
											2014-04-02 14:40:04 +03:00
										 |  |  |     multipart.upload_part_from_file(BytesIO(part1), 1) | 
					
						
							|  |  |  |     # last part, can be less than 5 MB | 
					
						
							| 
									
										
										
										
											2014-08-27 11:54:23 -04:00
										 |  |  |     part2 = b'1' | 
					
						
							| 
									
										
										
										
											2014-04-02 14:40:04 +03:00
										 |  |  |     multipart.upload_part_from_file(BytesIO(part2), 2) | 
					
						
							|  |  |  |     multipart.complete_upload() | 
					
						
							|  |  |  |     # we should get both parts as the key contents | 
					
						
							| 
									
										
										
										
											2017-02-23 19:43:48 -05:00
										 |  |  |     bucket.get_key("the-key").etag.should.equal(EXPECTED_ETAG) | 
					
						
							| 
									
										
										
										
											2014-04-02 14:40:04 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-02-10 13:19:45 +02:00
										 |  |  | @reduced_min_part_size | 
					
						
							| 
									
										
										
										
											2015-02-10 16:55:32 +02:00
										 |  |  | def test_multipart_invalid_order(): | 
					
						
							|  |  |  |     # Create Bucket so that test can run | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('mybucket') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     multipart = bucket.initiate_multipart_upload("the-key") | 
					
						
							|  |  |  |     part1 = b'0' * 5242880 | 
					
						
							|  |  |  |     etag1 = multipart.upload_part_from_file(BytesIO(part1), 1).etag | 
					
						
							|  |  |  |     # last part, can be less than 5 MB | 
					
						
							|  |  |  |     part2 = b'1' | 
					
						
							|  |  |  |     etag2 = multipart.upload_part_from_file(BytesIO(part2), 2).etag | 
					
						
							| 
									
										
										
										
											2015-02-10 17:31:40 +02:00
										 |  |  |     xml = "<Part><PartNumber>{0}</PartNumber><ETag>{1}</ETag></Part>" | 
					
						
							| 
									
										
										
										
											2015-02-10 16:55:32 +02:00
										 |  |  |     xml = xml.format(2, etag2) + xml.format(1, etag1) | 
					
						
							| 
									
										
										
										
											2015-02-10 17:31:40 +02:00
										 |  |  |     xml = "<CompleteMultipartUpload>{0}</CompleteMultipartUpload>".format(xml) | 
					
						
							| 
									
										
										
										
											2015-02-10 16:55:32 +02:00
										 |  |  |     bucket.complete_multipart_upload.when.called_with( | 
					
						
							|  |  |  |         multipart.key_name, multipart.id, xml).should.throw(S3ResponseError) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-03 15:36:41 -04:00
										 |  |  | @mock_s3_deprecated | 
					
						
							|  |  |  | @reduced_min_part_size | 
					
						
							|  |  |  | def test_multipart_etag_quotes_stripped(): | 
					
						
							|  |  |  |     # Create Bucket so that test can run | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('mybucket') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     multipart = bucket.initiate_multipart_upload("the-key") | 
					
						
							|  |  |  |     part1 = b'0' * REDUCED_PART_SIZE | 
					
						
							|  |  |  |     etag1 = multipart.upload_part_from_file(BytesIO(part1), 1).etag | 
					
						
							|  |  |  |     # last part, can be less than 5 MB | 
					
						
							|  |  |  |     part2 = b'1' | 
					
						
							|  |  |  |     etag2 = multipart.upload_part_from_file(BytesIO(part2), 2).etag | 
					
						
							|  |  |  |     # Strip quotes from etags | 
					
						
							|  |  |  |     etag1 = etag1.replace('"','') | 
					
						
							|  |  |  |     etag2 = etag2.replace('"','') | 
					
						
							|  |  |  |     xml = "<Part><PartNumber>{0}</PartNumber><ETag>{1}</ETag></Part>" | 
					
						
							|  |  |  |     xml = xml.format(1, etag1) + xml.format(2, etag2) | 
					
						
							|  |  |  |     xml = "<CompleteMultipartUpload>{0}</CompleteMultipartUpload>".format(xml) | 
					
						
							|  |  |  |     bucket.complete_multipart_upload.when.called_with( | 
					
						
							|  |  |  |         multipart.key_name, multipart.id, xml).should_not.throw(S3ResponseError) | 
					
						
							|  |  |  |     # we should get both parts as the key contents | 
					
						
							|  |  |  |     bucket.get_key("the-key").etag.should.equal(EXPECTED_ETAG) | 
					
						
							| 
									
										
										
										
											2015-06-27 17:56:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-02-27 18:48:51 -05:00
										 |  |  | @reduced_min_part_size | 
					
						
							|  |  |  | def test_multipart_duplicate_upload(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							| 
									
										
										
										
											2015-02-10 16:55:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-27 18:48:51 -05:00
										 |  |  |     multipart = bucket.initiate_multipart_upload("the-key") | 
					
						
							|  |  |  |     part1 = b'0' * REDUCED_PART_SIZE | 
					
						
							|  |  |  |     multipart.upload_part_from_file(BytesIO(part1), 1) | 
					
						
							|  |  |  |     # same part again | 
					
						
							|  |  |  |     multipart.upload_part_from_file(BytesIO(part1), 1) | 
					
						
							|  |  |  |     part2 = b'1' * 1024 | 
					
						
							|  |  |  |     multipart.upload_part_from_file(BytesIO(part2), 2) | 
					
						
							|  |  |  |     multipart.complete_upload() | 
					
						
							|  |  |  |     # We should get only one copy of part 1. | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "the-key").get_contents_as_string().should.equal(part1 + part2) | 
					
						
							| 
									
										
										
										
											2015-06-27 17:56:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-04-02 19:03:40 +03:00
										 |  |  | def test_list_multiparts(): | 
					
						
							|  |  |  |     # Create Bucket so that test can run | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('mybucket') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     multipart1 = bucket.initiate_multipart_upload("one-key") | 
					
						
							|  |  |  |     multipart2 = bucket.initiate_multipart_upload("two-key") | 
					
						
							|  |  |  |     uploads = bucket.get_all_multipart_uploads() | 
					
						
							|  |  |  |     uploads.should.have.length_of(2) | 
					
						
							|  |  |  |     dict([(u.key_name, u.id) for u in uploads]).should.equal( | 
					
						
							|  |  |  |         {'one-key': multipart1.id, 'two-key': multipart2.id}) | 
					
						
							|  |  |  |     multipart2.cancel_upload() | 
					
						
							|  |  |  |     uploads = bucket.get_all_multipart_uploads() | 
					
						
							|  |  |  |     uploads.should.have.length_of(1) | 
					
						
							|  |  |  |     uploads[0].key_name.should.equal("one-key") | 
					
						
							|  |  |  |     multipart1.cancel_upload() | 
					
						
							|  |  |  |     uploads = bucket.get_all_multipart_uploads() | 
					
						
							|  |  |  |     uploads.should.be.empty | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-07-08 21:20:29 -04:00
										 |  |  | def test_key_save_to_missing_bucket(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.get_bucket('mybucket', validate=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     key.set_contents_from_string.when.called_with( | 
					
						
							|  |  |  |         "foobar").should.throw(S3ResponseError) | 
					
						
							| 
									
										
										
										
											2014-07-08 21:20:29 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-02-18 21:22:03 -05:00
										 |  |  | def test_missing_key(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     bucket.get_key("the-key").should.equal(None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  | def test_missing_key_urllib2(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     conn.create_bucket("foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     urlopen.when.called_with( | 
					
						
							|  |  |  |         "http://foobar.s3.amazonaws.com/the-key").should.throw(HTTPError) | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-03-26 09:02:04 -04:00
										 |  |  | def test_empty_key(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-19 21:06:23 -05:00
										 |  |  |     key = bucket.get_key("the-key") | 
					
						
							|  |  |  |     key.size.should.equal(0) | 
					
						
							|  |  |  |     key.get_contents_as_string().should.equal(b'') | 
					
						
							| 
									
										
										
										
											2013-03-26 09:02:04 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-03-26 10:11:16 -04:00
										 |  |  | def test_empty_key_set_on_existing_key(): | 
					
						
							| 
									
										
										
										
											2013-02-19 23:29:46 -05:00
										 |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							| 
									
										
										
										
											2013-03-26 10:11:16 -04:00
										 |  |  |     key.set_contents_from_string("foobar") | 
					
						
							| 
									
										
										
										
											2013-02-19 23:29:46 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-19 21:06:23 -05:00
										 |  |  |     key = bucket.get_key("the-key") | 
					
						
							|  |  |  |     key.size.should.equal(6) | 
					
						
							|  |  |  |     key.get_contents_as_string().should.equal(b'foobar') | 
					
						
							| 
									
										
										
										
											2013-02-19 23:29:46 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-26 10:11:16 -04:00
										 |  |  |     key.set_contents_from_string("") | 
					
						
							| 
									
										
										
										
											2014-08-26 13:25:50 -04:00
										 |  |  |     bucket.get_key("the-key").get_contents_as_string().should.equal(b'') | 
					
						
							| 
									
										
										
										
											2013-03-26 10:11:16 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-05-06 23:33:59 -04:00
										 |  |  | def test_large_key_save(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("foobar" * 100000) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "the-key").get_contents_as_string().should.equal(b'foobar' * 100000) | 
					
						
							| 
									
										
										
										
											2013-02-19 23:29:46 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-02-19 23:29:46 -05:00
										 |  |  | def test_copy_key(): | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("some value") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-19 23:29:46 -05:00
										 |  |  |     bucket.copy_key('new-key', 'foobar', 'the-key') | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "the-key").get_contents_as_string().should.equal(b"some value") | 
					
						
							|  |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "new-key").get_contents_as_string().should.equal(b"some value") | 
					
						
							| 
									
										
										
										
											2016-04-27 17:08:44 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-16 16:42:00 +01:00
										 |  |  | @mock_s3_deprecated | 
					
						
							|  |  |  | def test_copy_key_with_unicode(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-unicode-💩-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("some value") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.copy_key('new-key', 'foobar', 'the-unicode-💩-key') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "the-unicode-💩-key").get_contents_as_string().should.equal(b"some value") | 
					
						
							|  |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "new-key").get_contents_as_string().should.equal(b"some value") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2016-04-27 17:08:44 -07:00
										 |  |  | def test_copy_key_with_version(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     bucket.configure_versioning(versioning=True) | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("some value") | 
					
						
							|  |  |  |     key.set_contents_from_string("another value") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     key = [ | 
					
						
							|  |  |  |         key.version_id | 
					
						
							|  |  |  |         for key in bucket.get_all_versions() | 
					
						
							|  |  |  |         if not key.is_latest | 
					
						
							|  |  |  |     ][0] | 
					
						
							|  |  |  |     bucket.copy_key('new-key', 'foobar', 'the-key', src_version_id=key) | 
					
						
							| 
									
										
										
										
											2016-04-27 17:08:44 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "the-key").get_contents_as_string().should.equal(b"another value") | 
					
						
							|  |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "new-key").get_contents_as_string().should.equal(b"some value") | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-05-17 11:43:09 +02:00
										 |  |  | def test_set_metadata(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = 'the-key' | 
					
						
							|  |  |  |     key.set_metadata('md', 'Metadatastring') | 
					
						
							|  |  |  |     key.set_contents_from_string("Testval") | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 11:43:09 +02:00
										 |  |  |     bucket.get_key('the-key').get_metadata('md').should.equal('Metadatastring') | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-03-26 16:02:14 +02:00
										 |  |  | def test_copy_key_replace_metadata(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_metadata('md', 'Metadatastring') | 
					
						
							|  |  |  |     key.set_contents_from_string("some value") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.copy_key('new-key', 'foobar', 'the-key', | 
					
						
							|  |  |  |                     metadata={'momd': 'Mometadatastring'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.get_key("new-key").get_metadata('md').should.be.none | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "new-key").get_metadata('momd').should.equal('Mometadatastring') | 
					
						
							| 
									
										
										
										
											2014-03-26 16:02:14 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-29 17:45:33 -04:00
										 |  |  | @freeze_time("2012-01-01 12:00:00") | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-03-29 17:45:33 -04:00
										 |  |  | def test_last_modified(): | 
					
						
							|  |  |  |     # See https://github.com/boto/boto/issues/466 | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("some value") | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-29 17:45:33 -04:00
										 |  |  |     rs = bucket.get_all_keys() | 
					
						
							| 
									
										
										
										
											2014-11-29 23:34:40 -05:00
										 |  |  |     rs[0].last_modified.should.equal('2012-01-01T12:00:00.000Z') | 
					
						
							| 
									
										
										
										
											2013-03-29 17:45:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.get_key( | 
					
						
							|  |  |  |         "the-key").last_modified.should.equal('Sun, 01 Jan 2012 12:00:00 GMT') | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-02-18 17:17:19 -05:00
										 |  |  | def test_missing_bucket(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     conn.get_bucket.when.called_with('mybucket').should.throw(S3ResponseError) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-03-23 09:57:20 -04:00
										 |  |  | def test_bucket_with_dash(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     conn.get_bucket.when.called_with( | 
					
						
							|  |  |  |         'mybucket-test').should.throw(S3ResponseError) | 
					
						
							| 
									
										
										
										
											2013-03-23 09:57:20 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-03-16 21:25:14 -04:00
										 |  |  | def test_create_existing_bucket(): | 
					
						
							|  |  |  |     "Trying to create a bucket that already exists should raise an Error" | 
					
						
							| 
									
										
										
										
											2014-12-10 20:20:43 -05:00
										 |  |  |     conn = boto.s3.connect_to_region("us-west-2") | 
					
						
							| 
									
										
										
										
											2014-03-16 21:25:14 -04:00
										 |  |  |     conn.create_bucket("foobar") | 
					
						
							| 
									
										
										
										
											2014-08-26 13:25:50 -04:00
										 |  |  |     with assert_raises(S3CreateError): | 
					
						
							|  |  |  |         conn.create_bucket('foobar') | 
					
						
							| 
									
										
										
										
											2014-03-16 21:25:14 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-12-10 20:20:43 -05:00
										 |  |  | def test_create_existing_bucket_in_us_east_1(): | 
					
						
							|  |  |  |     "Trying to create a bucket that already exists in us-east-1 returns the bucket" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """"
 | 
					
						
							|  |  |  |     http://docs.aws.amazon.com/AmazonS3/latest/API/ErrorResponses.html | 
					
						
							|  |  |  |     Your previous request to create the named bucket succeeded and you already | 
					
						
							|  |  |  |     own it. You get this error in all AWS regions except US Standard, | 
					
						
							|  |  |  |     us-east-1. In us-east-1 region, you will get 200 OK, but it is no-op (if | 
					
						
							|  |  |  |     bucket exists it Amazon S3 will not do anything). | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     conn = boto.s3.connect_to_region("us-east-1") | 
					
						
							|  |  |  |     conn.create_bucket("foobar") | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     bucket.name.should.equal("foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-07-08 20:35:48 -04:00
										 |  |  | def test_other_region(): | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     conn = S3Connection( | 
					
						
							|  |  |  |         'key', 'secret', host='s3-website-ap-southeast-2.amazonaws.com') | 
					
						
							| 
									
										
										
										
											2014-07-08 20:35:48 -04:00
										 |  |  |     conn.create_bucket("foobar") | 
					
						
							|  |  |  |     list(conn.get_bucket("foobar").get_all_keys()).should.equal([]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-02-18 17:17:19 -05:00
										 |  |  | def test_bucket_deletion(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("some value") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-18 21:22:03 -05:00
										 |  |  |     # Try to delete a bucket that still has keys | 
					
						
							| 
									
										
										
										
											2013-02-18 17:17:19 -05:00
										 |  |  |     conn.delete_bucket.when.called_with("foobar").should.throw(S3ResponseError) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.delete_key("the-key") | 
					
						
							|  |  |  |     conn.delete_bucket("foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-18 21:22:03 -05:00
										 |  |  |     # Get non-existing bucket | 
					
						
							| 
									
										
										
										
											2013-02-18 17:17:19 -05:00
										 |  |  |     conn.get_bucket.when.called_with("foobar").should.throw(S3ResponseError) | 
					
						
							| 
									
										
										
										
											2013-02-18 17:31:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-18 21:22:03 -05:00
										 |  |  |     # Delete non-existant bucket | 
					
						
							|  |  |  |     conn.delete_bucket.when.called_with("foobar").should.throw(S3ResponseError) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-18 17:31:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-02-18 17:31:15 -05:00
										 |  |  | def test_get_all_buckets(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							| 
									
										
										
										
											2013-02-27 01:12:11 -05:00
										 |  |  |     conn.create_bucket("foobar") | 
					
						
							|  |  |  |     conn.create_bucket("foobar2") | 
					
						
							| 
									
										
										
										
											2013-02-18 17:31:15 -05:00
										 |  |  |     buckets = conn.get_all_buckets() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     buckets.should.have.length_of(2) | 
					
						
							| 
									
										
										
										
											2013-02-25 23:21:49 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 11:43:09 +02:00
										 |  |  | @mock_s3 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-05-17 11:43:09 +02:00
										 |  |  | def test_post_to_bucket(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							| 
									
										
										
										
											2013-05-17 19:41:39 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 11:43:09 +02:00
										 |  |  |     requests.post("https://foobar.s3.amazonaws.com/", { | 
					
						
							|  |  |  |         'key': 'the-key', | 
					
						
							|  |  |  |         'file': 'nothing' | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-26 13:25:50 -04:00
										 |  |  |     bucket.get_key('the-key').get_contents_as_string().should.equal(b'nothing') | 
					
						
							| 
									
										
										
										
											2013-05-17 19:41:39 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 11:43:09 +02:00
										 |  |  | @mock_s3 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-05-17 11:43:09 +02:00
										 |  |  | def test_post_with_metadata_to_bucket(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							| 
									
										
										
										
											2013-05-17 19:41:39 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 11:43:09 +02:00
										 |  |  |     requests.post("https://foobar.s3.amazonaws.com/", { | 
					
						
							|  |  |  |         'key': 'the-key', | 
					
						
							|  |  |  |         'file': 'nothing', | 
					
						
							|  |  |  |         'x-amz-meta-test': 'metadata' | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.get_key('the-key').get_metadata('test').should.equal('metadata') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-08 20:35:48 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-06-27 17:56:37 -04:00
										 |  |  | def test_delete_missing_key(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-08 17:14:29 -04:00
										 |  |  |     deleted_key = bucket.delete_key("foobar") | 
					
						
							|  |  |  |     deleted_key.key.should.equal("foobar") | 
					
						
							| 
									
										
										
										
											2015-06-27 17:56:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-05-06 17:21:33 -04:00
										 |  |  | def test_delete_keys(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Key(bucket=bucket, name='file1').set_contents_from_string('abc') | 
					
						
							|  |  |  |     Key(bucket=bucket, name='file2').set_contents_from_string('abc') | 
					
						
							|  |  |  |     Key(bucket=bucket, name='file3').set_contents_from_string('abc') | 
					
						
							|  |  |  |     Key(bucket=bucket, name='file4').set_contents_from_string('abc') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result = bucket.delete_keys(['file2', 'file3']) | 
					
						
							|  |  |  |     result.deleted.should.have.length_of(2) | 
					
						
							|  |  |  |     result.errors.should.have.length_of(0) | 
					
						
							|  |  |  |     keys = bucket.get_all_keys() | 
					
						
							|  |  |  |     keys.should.have.length_of(2) | 
					
						
							|  |  |  |     keys[0].name.should.equal('file1') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-08 20:35:48 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-05-06 17:21:33 -04:00
										 |  |  | def test_delete_keys_with_invalid(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Key(bucket=bucket, name='file1').set_contents_from_string('abc') | 
					
						
							|  |  |  |     Key(bucket=bucket, name='file2').set_contents_from_string('abc') | 
					
						
							|  |  |  |     Key(bucket=bucket, name='file3').set_contents_from_string('abc') | 
					
						
							|  |  |  |     Key(bucket=bucket, name='file4').set_contents_from_string('abc') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result = bucket.delete_keys(['abc', 'file3']) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result.deleted.should.have.length_of(1) | 
					
						
							|  |  |  |     result.errors.should.have.length_of(1) | 
					
						
							|  |  |  |     keys = bucket.get_all_keys() | 
					
						
							|  |  |  |     keys.should.have.length_of(3) | 
					
						
							|  |  |  |     keys[0].name.should.equal('file1') | 
					
						
							| 
									
										
										
										
											2013-05-17 19:41:39 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-07-08 20:35:48 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-04-13 17:28:00 -04:00
										 |  |  | def test_bucket_name_with_dot(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('firstname.lastname') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     k = Key(bucket, 'somekey') | 
					
						
							|  |  |  |     k.set_contents_from_string('somedata') | 
					
						
							| 
									
										
										
										
											2013-04-13 19:00:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-04-13 19:00:37 -04:00
										 |  |  | def test_key_with_special_characters(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('test_bucket_name') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = Key(bucket, 'test_list_keys_2/x?y') | 
					
						
							|  |  |  |     key.set_contents_from_string('value1') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key_list = bucket.list('test_list_keys_2/', '/') | 
					
						
							|  |  |  |     keys = [x for x in key_list] | 
					
						
							|  |  |  |     keys[0].name.should.equal("test_list_keys_2/x?y") | 
					
						
							| 
									
										
										
										
											2013-04-13 19:23:32 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2016-05-01 22:03:52 -04:00
										 |  |  | def test_unicode_key_with_slash(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "/the-key-unîcode/test" | 
					
						
							|  |  |  |     key.set_contents_from_string("value") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = bucket.get_key("/the-key-unîcode/test") | 
					
						
							|  |  |  |     key.get_contents_as_string().should.equal(b'value') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2013-04-13 19:23:32 -04:00
										 |  |  | def test_bucket_key_listing_order(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('test_bucket') | 
					
						
							|  |  |  |     prefix = 'toplevel/' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def store(name): | 
					
						
							|  |  |  |         k = Key(bucket, prefix + name) | 
					
						
							|  |  |  |         k.set_contents_from_string('somedata') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     names = ['x/key', 'y.key1', 'y.key2', 'y.key3', 'x/y/key', 'x/y/z/key'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for name in names: | 
					
						
							|  |  |  |         store(name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     delimiter = None | 
					
						
							|  |  |  |     keys = [x.name for x in bucket.list(prefix, delimiter)] | 
					
						
							|  |  |  |     keys.should.equal([ | 
					
						
							|  |  |  |         'toplevel/x/key', 'toplevel/x/y/key', 'toplevel/x/y/z/key', | 
					
						
							|  |  |  |         'toplevel/y.key1', 'toplevel/y.key2', 'toplevel/y.key3' | 
					
						
							|  |  |  |     ]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     delimiter = '/' | 
					
						
							|  |  |  |     keys = [x.name for x in bucket.list(prefix, delimiter)] | 
					
						
							|  |  |  |     keys.should.equal([ | 
					
						
							|  |  |  |         'toplevel/y.key1', 'toplevel/y.key2', 'toplevel/y.key3', 'toplevel/x/' | 
					
						
							|  |  |  |     ]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Test delimiter with no prefix | 
					
						
							|  |  |  |     delimiter = '/' | 
					
						
							|  |  |  |     keys = [x.name for x in bucket.list(prefix=None, delimiter=delimiter)] | 
					
						
							| 
									
										
										
										
											2015-01-29 09:20:40 -08:00
										 |  |  |     keys.should.equal(['toplevel/']) | 
					
						
							| 
									
										
										
										
											2013-05-07 00:03:05 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     delimiter = None | 
					
						
							|  |  |  |     keys = [x.name for x in bucket.list(prefix + 'x', delimiter)] | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     keys.should.equal( | 
					
						
							|  |  |  |         [u'toplevel/x/key', u'toplevel/x/y/key', u'toplevel/x/y/z/key']) | 
					
						
							| 
									
										
										
										
											2013-05-07 00:03:05 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     delimiter = '/' | 
					
						
							|  |  |  |     keys = [x.name for x in bucket.list(prefix + 'x', delimiter)] | 
					
						
							|  |  |  |     keys.should.equal([u'toplevel/x/']) | 
					
						
							| 
									
										
										
										
											2014-03-26 17:52:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-03-26 17:52:31 +02:00
										 |  |  | def test_key_with_reduced_redundancy(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('test_bucket_name') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = Key(bucket, 'test_rr_key') | 
					
						
							|  |  |  |     key.set_contents_from_string('value1', reduced_redundancy=True) | 
					
						
							|  |  |  |     # we use the bucket iterator because of: | 
					
						
							|  |  |  |     # https:/github.com/boto/boto/issues/1173 | 
					
						
							|  |  |  |     list(bucket)[0].storage_class.should.equal('REDUCED_REDUNDANCY') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-03-26 17:52:31 +02:00
										 |  |  | def test_copy_key_reduced_redundancy(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("some value") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     bucket.copy_key('new-key', 'foobar', 'the-key', | 
					
						
							|  |  |  |                     storage_class='REDUCED_REDUNDANCY') | 
					
						
							| 
									
										
										
										
											2014-03-26 17:52:31 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # we use the bucket iterator because of: | 
					
						
							|  |  |  |     # https:/github.com/boto/boto/issues/1173 | 
					
						
							|  |  |  |     keys = dict([(k.name, k) for k in bucket]) | 
					
						
							|  |  |  |     keys['new-key'].storage_class.should.equal("REDUCED_REDUNDANCY") | 
					
						
							|  |  |  |     keys['the-key'].storage_class.should.equal("STANDARD") | 
					
						
							| 
									
										
										
										
											2014-03-26 19:15:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @freeze_time("2012-01-01 12:00:00") | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-03-26 19:15:08 +02:00
										 |  |  | def test_restore_key(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("some value") | 
					
						
							|  |  |  |     list(bucket)[0].ongoing_restore.should.be.none | 
					
						
							|  |  |  |     key.restore(1) | 
					
						
							|  |  |  |     key = bucket.get_key('the-key') | 
					
						
							|  |  |  |     key.ongoing_restore.should_not.be.none | 
					
						
							|  |  |  |     key.ongoing_restore.should.be.false | 
					
						
							|  |  |  |     key.expiry_date.should.equal("Mon, 02 Jan 2012 12:00:00 GMT") | 
					
						
							|  |  |  |     key.restore(2) | 
					
						
							|  |  |  |     key = bucket.get_key('the-key') | 
					
						
							|  |  |  |     key.ongoing_restore.should_not.be.none | 
					
						
							|  |  |  |     key.ongoing_restore.should.be.false | 
					
						
							|  |  |  |     key.expiry_date.should.equal("Tue, 03 Jan 2012 12:00:00 GMT") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @freeze_time("2012-01-01 12:00:00") | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-03-26 19:15:08 +02:00
										 |  |  | def test_restore_key_headers(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket("foobar") | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = "the-key" | 
					
						
							|  |  |  |     key.set_contents_from_string("some value") | 
					
						
							|  |  |  |     key.restore(1, headers={'foo': 'bar'}) | 
					
						
							|  |  |  |     key = bucket.get_key('the-key') | 
					
						
							|  |  |  |     key.ongoing_restore.should_not.be.none | 
					
						
							|  |  |  |     key.ongoing_restore.should.be.false | 
					
						
							|  |  |  |     key.expiry_date.should.equal("Mon, 02 Jan 2012 12:00:00 GMT") | 
					
						
							| 
									
										
										
										
											2014-06-27 11:34:00 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-06-27 11:34:00 -06:00
										 |  |  | def test_get_versioning_status(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  |     d = bucket.get_versioning_status() | 
					
						
							|  |  |  |     d.should.be.empty | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.configure_versioning(versioning=True) | 
					
						
							|  |  |  |     d = bucket.get_versioning_status() | 
					
						
							|  |  |  |     d.shouldnt.be.empty | 
					
						
							|  |  |  |     d.should.have.key('Versioning').being.equal('Enabled') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.configure_versioning(versioning=False) | 
					
						
							|  |  |  |     d = bucket.get_versioning_status() | 
					
						
							|  |  |  |     d.should.have.key('Versioning').being.equal('Suspended') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-06-27 11:34:00 -06:00
										 |  |  | def test_key_version(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  |     bucket.configure_versioning(versioning=True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     versions = [] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 11:34:00 -06:00
										 |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = 'the-key' | 
					
						
							|  |  |  |     key.version_id.should.be.none | 
					
						
							|  |  |  |     key.set_contents_from_string('some string') | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     versions.append(key.version_id) | 
					
						
							| 
									
										
										
										
											2014-06-27 11:34:00 -06:00
										 |  |  |     key.set_contents_from_string('some string') | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     versions.append(key.version_id) | 
					
						
							|  |  |  |     set(versions).should.have.length_of(2) | 
					
						
							| 
									
										
										
										
											2014-06-27 11:34:00 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     key = bucket.get_key('the-key') | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     key.version_id.should.equal(versions[-1]) | 
					
						
							| 
									
										
										
										
											2014-06-27 16:21:32 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-06-27 16:21:32 -06:00
										 |  |  | def test_list_versions(): | 
					
						
							|  |  |  |     conn = boto.connect_s3('the_key', 'the_secret') | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  |     bucket.configure_versioning(versioning=True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     key_versions = [] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-27 16:21:32 -06:00
										 |  |  |     key = Key(bucket, 'the-key') | 
					
						
							|  |  |  |     key.version_id.should.be.none | 
					
						
							|  |  |  |     key.set_contents_from_string("Version 1") | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     key_versions.append(key.version_id) | 
					
						
							| 
									
										
										
										
											2014-06-27 16:21:32 -06:00
										 |  |  |     key.set_contents_from_string("Version 2") | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     key_versions.append(key.version_id) | 
					
						
							|  |  |  |     key_versions.should.have.length_of(2) | 
					
						
							| 
									
										
										
										
											2014-06-27 16:21:32 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     versions = list(bucket.list_versions()) | 
					
						
							|  |  |  |     versions.should.have.length_of(2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     versions[0].name.should.equal('the-key') | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     versions[0].version_id.should.equal(key_versions[0]) | 
					
						
							| 
									
										
										
										
											2014-08-26 13:25:50 -04:00
										 |  |  |     versions[0].get_contents_as_string().should.equal(b"Version 1") | 
					
						
							| 
									
										
										
										
											2014-06-27 16:21:32 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |     versions[1].name.should.equal('the-key') | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     versions[1].version_id.should.equal(key_versions[1]) | 
					
						
							| 
									
										
										
										
											2014-08-26 13:25:50 -04:00
										 |  |  |     versions[1].get_contents_as_string().should.equal(b"Version 2") | 
					
						
							| 
									
										
										
										
											2014-08-13 22:13:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-10 17:14:31 -07:00
										 |  |  |     key = Key(bucket, 'the2-key') | 
					
						
							|  |  |  |     key.set_contents_from_string("Version 1") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     keys = list(bucket.list()) | 
					
						
							|  |  |  |     keys.should.have.length_of(2) | 
					
						
							|  |  |  |     versions = list(bucket.list_versions(prefix='the2-')) | 
					
						
							|  |  |  |     versions.should.have.length_of(1) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-13 22:13:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-10-07 00:04:22 -07:00
										 |  |  | def test_acl_setting(): | 
					
						
							| 
									
										
										
										
											2014-08-13 22:13:17 -04:00
										 |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							| 
									
										
										
										
											2014-08-26 13:25:50 -04:00
										 |  |  |     content = b'imafile' | 
					
						
							| 
									
										
										
										
											2014-08-13 22:13:17 -04:00
										 |  |  |     keyname = 'test.txt' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = Key(bucket, name=keyname) | 
					
						
							|  |  |  |     key.content_type = 'text/plain' | 
					
						
							|  |  |  |     key.set_contents_from_string(content) | 
					
						
							|  |  |  |     key.make_public() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = bucket.get_key(keyname) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert key.get_contents_as_string() == content | 
					
						
							| 
									
										
										
										
											2014-11-22 10:19:56 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-07 00:04:22 -07:00
										 |  |  |     grants = key.get_acl().acl.grants | 
					
						
							|  |  |  |     assert any(g.uri == 'http://acs.amazonaws.com/groups/global/AllUsers' and | 
					
						
							|  |  |  |                g.permission == 'READ' for g in grants), grants | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-10-07 00:04:22 -07:00
										 |  |  | def test_acl_setting_via_headers(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  |     content = b'imafile' | 
					
						
							|  |  |  |     keyname = 'test.txt' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = Key(bucket, name=keyname) | 
					
						
							|  |  |  |     key.content_type = 'text/plain' | 
					
						
							|  |  |  |     key.set_contents_from_string(content, headers={ | 
					
						
							|  |  |  |         'x-amz-grant-full-control': 'uri="http://acs.amazonaws.com/groups/global/AllUsers"' | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = bucket.get_key(keyname) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert key.get_contents_as_string() == content | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     grants = key.get_acl().acl.grants | 
					
						
							|  |  |  |     assert any(g.uri == 'http://acs.amazonaws.com/groups/global/AllUsers' and | 
					
						
							|  |  |  |                g.permission == 'FULL_CONTROL' for g in grants), grants | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-10-07 00:04:22 -07:00
										 |  |  | def test_acl_switching(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  |     content = b'imafile' | 
					
						
							|  |  |  |     keyname = 'test.txt' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = Key(bucket, name=keyname) | 
					
						
							|  |  |  |     key.content_type = 'text/plain' | 
					
						
							|  |  |  |     key.set_contents_from_string(content, policy='public-read') | 
					
						
							|  |  |  |     key.set_acl('private') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     grants = key.get_acl().acl.grants | 
					
						
							|  |  |  |     assert not any(g.uri == 'http://acs.amazonaws.com/groups/global/AllUsers' and | 
					
						
							|  |  |  |                    g.permission == 'READ' for g in grants), grants | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-22 10:19:56 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-11-11 20:26:29 -05:00
										 |  |  | def test_bucket_acl_setting(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.make_public() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     grants = bucket.get_acl().acl.grants | 
					
						
							|  |  |  |     assert any(g.uri == 'http://acs.amazonaws.com/groups/global/AllUsers' and | 
					
						
							|  |  |  |                g.permission == 'READ' for g in grants), grants | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-11-11 20:26:29 -05:00
										 |  |  | def test_bucket_acl_switching(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('foobar') | 
					
						
							|  |  |  |     bucket.make_public() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.set_acl('private') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     grants = bucket.get_acl().acl.grants | 
					
						
							|  |  |  |     assert not any(g.uri == 'http://acs.amazonaws.com/groups/global/AllUsers' and | 
					
						
							|  |  |  |                    g.permission == 'READ' for g in grants), grants | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-16 06:08:27 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_s3_object_in_public_bucket(): | 
					
						
							|  |  |  |     s3 = boto3.resource('s3') | 
					
						
							|  |  |  |     bucket = s3.Bucket('test-bucket') | 
					
						
							|  |  |  |     bucket.create(ACL='public-read') | 
					
						
							| 
									
										
										
										
											2017-09-20 19:04:23 -07:00
										 |  |  |     bucket.put_object(Body=b'ABCD', Key='file.txt') | 
					
						
							| 
									
										
										
										
											2017-09-16 15:48:20 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     s3_anonymous = boto3.resource('s3') | 
					
						
							|  |  |  |     s3_anonymous.meta.client.meta.events.register('choose-signer.s3.*', disable_signing) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-22 10:57:06 -07:00
										 |  |  |     contents = s3_anonymous.Object(key='file.txt', bucket_name='test-bucket').get()['Body'].read() | 
					
						
							|  |  |  |     contents.should.equal(b'ABCD') | 
					
						
							| 
									
										
										
										
											2017-09-16 15:48:20 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-16 06:08:27 -07:00
										 |  |  |     bucket.put_object(ACL='private', Body=b'ABCD', Key='file.txt') | 
					
						
							| 
									
										
										
										
											2017-09-16 15:48:20 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-22 10:57:06 -07:00
										 |  |  |     with assert_raises(ClientError) as exc: | 
					
						
							|  |  |  |         s3_anonymous.Object(key='file.txt', bucket_name='test-bucket').get() | 
					
						
							|  |  |  |     exc.exception.response['Error']['Code'].should.equal('403') | 
					
						
							| 
									
										
										
										
											2017-09-16 06:08:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  |     params = {'Bucket': 'test-bucket', 'Key': 'file.txt'} | 
					
						
							| 
									
										
										
										
											2017-09-22 10:42:13 -07:00
										 |  |  |     presigned_url = boto3.client('s3').generate_presigned_url('get_object', params, ExpiresIn=900) | 
					
						
							|  |  |  |     response = requests.get(presigned_url) | 
					
						
							|  |  |  |     assert response.status_code == 200 | 
					
						
							| 
									
										
										
										
											2017-09-16 06:08:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-16 06:08:27 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_s3_object_in_private_bucket(): | 
					
						
							|  |  |  |     s3 = boto3.resource('s3') | 
					
						
							|  |  |  |     bucket = s3.Bucket('test-bucket') | 
					
						
							|  |  |  |     bucket.create(ACL='private') | 
					
						
							|  |  |  |     bucket.put_object(ACL='private', Body=b'ABCD', Key='file.txt') | 
					
						
							| 
									
										
										
										
											2017-09-16 15:48:20 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     s3_anonymous = boto3.resource('s3') | 
					
						
							|  |  |  |     s3_anonymous.meta.client.meta.events.register('choose-signer.s3.*', disable_signing) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(ClientError) as exc: | 
					
						
							|  |  |  |         s3_anonymous.Object(key='file.txt', bucket_name='test-bucket').get() | 
					
						
							|  |  |  |     exc.exception.response['Error']['Code'].should.equal('403') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-16 06:08:27 -07:00
										 |  |  |     bucket.put_object(ACL='public-read', Body=b'ABCD', Key='file.txt') | 
					
						
							| 
									
										
										
										
											2017-09-16 15:48:20 -07:00
										 |  |  |     contents = s3_anonymous.Object(key='file.txt', bucket_name='test-bucket').get()['Body'].read() | 
					
						
							|  |  |  |     contents.should.equal(b'ABCD') | 
					
						
							| 
									
										
										
										
											2017-09-16 06:08:27 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-11-22 10:19:56 -05:00
										 |  |  | def test_unicode_key(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('mybucket') | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = u'こんにちは.jpg' | 
					
						
							|  |  |  |     key.set_contents_from_string('Hello world!') | 
					
						
							| 
									
										
										
										
											2016-09-04 04:31:03 +06:00
										 |  |  |     assert [listed_key.key for listed_key in bucket.list()] == [key.key] | 
					
						
							|  |  |  |     fetched_key = bucket.get_key(key.key) | 
					
						
							|  |  |  |     assert fetched_key.key == key.key | 
					
						
							|  |  |  |     assert fetched_key.get_contents_as_string().decode("utf-8") == 'Hello world!' | 
					
						
							| 
									
										
										
										
											2014-11-22 10:19:56 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-11-22 10:19:56 -05:00
										 |  |  | def test_unicode_value(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('mybucket') | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = 'some_key' | 
					
						
							|  |  |  |     key.set_contents_from_string(u'こんにちは.jpg') | 
					
						
							|  |  |  |     list(bucket.list()) | 
					
						
							|  |  |  |     key = bucket.get_key(key.key) | 
					
						
							| 
									
										
										
										
											2014-11-22 10:42:43 -05:00
										 |  |  |     assert key.get_contents_as_string().decode("utf-8") == u'こんにちは.jpg' | 
					
						
							| 
									
										
										
										
											2014-11-27 10:43:10 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-11-27 10:43:10 -05:00
										 |  |  | def test_setting_content_encoding(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('mybucket') | 
					
						
							|  |  |  |     key = bucket.new_key("keyname") | 
					
						
							|  |  |  |     key.set_metadata("Content-Encoding", "gzip") | 
					
						
							|  |  |  |     compressed_data = "abcdef" | 
					
						
							|  |  |  |     key.set_contents_from_string(compressed_data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = bucket.get_key("keyname") | 
					
						
							|  |  |  |     key.content_encoding.should.equal("gzip") | 
					
						
							| 
									
										
										
										
											2014-12-10 20:44:00 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2014-12-10 20:44:00 -05:00
										 |  |  | def test_bucket_location(): | 
					
						
							|  |  |  |     conn = boto.s3.connect_to_region("us-west-2") | 
					
						
							|  |  |  |     bucket = conn.create_bucket('mybucket') | 
					
						
							|  |  |  |     bucket.get_location().should.equal("us-west-2") | 
					
						
							| 
									
										
										
										
											2015-02-10 17:28:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-29 07:07:29 -05:00
										 |  |  | @mock_s3 | 
					
						
							| 
									
										
										
										
											2018-11-20 19:42:51 +01:00
										 |  |  | def test_bucket_location_us_east_1(): | 
					
						
							|  |  |  |     cli = boto3.client('s3') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     # No LocationConstraint ==> us-east-1 | 
					
						
							|  |  |  |     cli.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     cli.get_bucket_location(Bucket=bucket_name)['LocationConstraint'].should.equal(None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-02-10 17:28:18 +02:00
										 |  |  | def test_ranged_get(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('mybucket') | 
					
						
							|  |  |  |     key = Key(bucket) | 
					
						
							|  |  |  |     key.key = 'bigkey' | 
					
						
							| 
									
										
										
										
											2015-02-11 11:08:00 +02:00
										 |  |  |     rep = b"0123456789" | 
					
						
							| 
									
										
										
										
											2015-02-10 19:43:24 +02:00
										 |  |  |     key.set_contents_from_string(rep * 10) | 
					
						
							| 
									
										
										
										
											2015-06-30 02:04:58 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Implicitly bounded range requests. | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=0-'}).should.equal(rep * 10) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=50-'}).should.equal(rep * 5) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=99-'}).should.equal(b'9') | 
					
						
							| 
									
										
										
										
											2015-06-30 02:04:58 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Explicitly bounded range requests starting from the first byte. | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=0-0'}).should.equal(b'0') | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=0-49'}).should.equal(rep * 5) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=0-99'}).should.equal(rep * 10) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=0-100'}).should.equal(rep * 10) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=0-700'}).should.equal(rep * 10) | 
					
						
							| 
									
										
										
										
											2015-06-30 02:04:58 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Explicitly bounded range requests starting from the / a middle byte. | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=50-54'}).should.equal(rep[:5]) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=50-99'}).should.equal(rep * 5) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=50-100'}).should.equal(rep * 5) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=50-700'}).should.equal(rep * 5) | 
					
						
							| 
									
										
										
										
											2015-06-30 02:04:58 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Explicitly bounded range requests starting from the last byte. | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=99-99'}).should.equal(b'9') | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=99-100'}).should.equal(b'9') | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=99-700'}).should.equal(b'9') | 
					
						
							| 
									
										
										
										
											2015-06-30 02:04:58 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Suffix range requests. | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=-1'}).should.equal(b'9') | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=-60'}).should.equal(rep * 6) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=-100'}).should.equal(rep * 10) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=-101'}).should.equal(rep * 10) | 
					
						
							|  |  |  |     key.get_contents_as_string( | 
					
						
							|  |  |  |         headers={'Range': 'bytes=-700'}).should.equal(rep * 10) | 
					
						
							| 
									
										
										
										
											2015-06-30 02:04:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-10 19:15:10 +02:00
										 |  |  |     key.size.should.equal(100) | 
					
						
							| 
									
										
										
										
											2015-07-23 17:33:52 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							| 
									
										
										
										
											2015-07-23 17:33:52 -04:00
										 |  |  | def test_policy(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     bucket = conn.create_bucket(bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     policy = json.dumps({ | 
					
						
							|  |  |  |         "Version": "2012-10-17", | 
					
						
							|  |  |  |         "Id": "PutObjPolicy", | 
					
						
							|  |  |  |         "Statement": [ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "Sid": "DenyUnEncryptedObjectUploads", | 
					
						
							|  |  |  |                 "Effect": "Deny", | 
					
						
							|  |  |  |                 "Principal": "*", | 
					
						
							|  |  |  |                 "Action": "s3:PutObject", | 
					
						
							|  |  |  |                 "Resource": "arn:aws:s3:::{bucket_name}/*".format(bucket_name=bucket_name), | 
					
						
							|  |  |  |                 "Condition": { | 
					
						
							|  |  |  |                     "StringNotEquals": { | 
					
						
							|  |  |  |                         "s3:x-amz-server-side-encryption": "aws:kms" | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(S3ResponseError) as err: | 
					
						
							|  |  |  |         bucket.get_policy() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ex = err.exception | 
					
						
							|  |  |  |     ex.box_usage.should.be.none | 
					
						
							|  |  |  |     ex.error_code.should.equal('NoSuchBucketPolicy') | 
					
						
							|  |  |  |     ex.message.should.equal('The bucket policy does not exist') | 
					
						
							|  |  |  |     ex.reason.should.equal('Not Found') | 
					
						
							|  |  |  |     ex.resource.should.be.none | 
					
						
							|  |  |  |     ex.status.should.equal(404) | 
					
						
							|  |  |  |     ex.body.should.contain(bucket_name) | 
					
						
							|  |  |  |     ex.request_id.should_not.be.none | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.set_policy(policy).should.be.true | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket = conn.get_bucket(bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bucket.get_policy().decode('utf-8').should.equal(policy) | 
					
						
							| 
									
										
										
										
											2015-08-02 09:54:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-07 22:43:24 -04:00
										 |  |  |     bucket.delete_policy() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(S3ResponseError) as err: | 
					
						
							|  |  |  |         bucket.get_policy() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-02 09:54:23 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 22:35:45 -05:00
										 |  |  | @mock_s3_deprecated | 
					
						
							|  |  |  | def test_website_configuration_xml(): | 
					
						
							|  |  |  |     conn = boto.connect_s3() | 
					
						
							|  |  |  |     bucket = conn.create_bucket('test-bucket') | 
					
						
							|  |  |  |     bucket.set_website_configuration_xml(TEST_XML) | 
					
						
							|  |  |  |     bucket.get_website_configuration_xml().should.equal(TEST_XML) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3_deprecated | 
					
						
							|  |  |  | def test_key_with_trailing_slash_in_ordinary_calling_format(): | 
					
						
							|  |  |  |     conn = boto.connect_s3( | 
					
						
							|  |  |  |         'access_key', | 
					
						
							|  |  |  |         'secret_key', | 
					
						
							|  |  |  |         calling_format=boto.s3.connection.OrdinaryCallingFormat() | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     bucket = conn.create_bucket('test_bucket_name') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key_name = 'key_with_slash/' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = Key(bucket, key_name) | 
					
						
							|  |  |  |     key.set_contents_from_string('some value') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [k.name for k in bucket.get_all_keys()].should.contain(key_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-02 09:54:23 -04:00
										 |  |  | """
 | 
					
						
							|  |  |  | boto3 | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-20 15:42:21 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_key_etag(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket='mybucket') | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='steve', Body=b'is awesome') | 
					
						
							|  |  |  |     resp = s3.get_object(Bucket='mybucket', Key='steve') | 
					
						
							|  |  |  |     resp['ETag'].should.equal('"d32bda93738f7e03adb22e66c90fbc04"') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-14 13:32:52 +02:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_website_redirect_location(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket='mybucket') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='steve', Body=b'is awesome') | 
					
						
							|  |  |  |     resp = s3.get_object(Bucket='mybucket', Key='steve') | 
					
						
							|  |  |  |     resp.get('WebsiteRedirectLocation').should.be.none | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     url = 'https://github.com/spulec/moto' | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='steve', Body=b'is awesome', WebsiteRedirectLocation=url) | 
					
						
							|  |  |  |     resp = s3.get_object(Bucket='mybucket', Key='steve') | 
					
						
							|  |  |  |     resp['WebsiteRedirectLocation'].should.equal(url) | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 23:13:24 +00:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_list_keys_xml_escaped(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket='mybucket') | 
					
						
							|  |  |  |     key_name = 'Q&A.txt' | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key=key_name, Body=b'is awesome') | 
					
						
							| 
									
										
										
										
											2017-02-09 03:21:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 23:13:24 +00:00
										 |  |  |     resp = s3.list_objects_v2(Bucket='mybucket', Prefix=key_name) | 
					
						
							| 
									
										
										
										
											2017-02-09 03:21:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-03 23:13:24 +00:00
										 |  |  |     assert resp['Contents'][0]['Key'] == key_name | 
					
						
							| 
									
										
										
										
											2017-02-09 03:21:43 +01:00
										 |  |  |     assert resp['KeyCount'] == 1 | 
					
						
							|  |  |  |     assert resp['MaxKeys'] == 1000 | 
					
						
							|  |  |  |     assert resp['Prefix'] == key_name | 
					
						
							|  |  |  |     assert resp['IsTruncated'] == False | 
					
						
							|  |  |  |     assert 'Delimiter' not in resp | 
					
						
							|  |  |  |     assert 'StartAfter' not in resp | 
					
						
							|  |  |  |     assert 'NextContinuationToken' not in resp | 
					
						
							|  |  |  |     assert 'Owner' not in resp['Contents'][0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 16:17:28 -08:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_list_objects_v2_common_prefix_pagination(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket='mybucket') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     max_keys = 1 | 
					
						
							|  |  |  |     keys = ['test/{i}/{i}'.format(i=i) for i in range(3)] | 
					
						
							|  |  |  |     for key in keys: | 
					
						
							|  |  |  |         s3.put_object(Bucket='mybucket', Key=key, Body=b'v') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     prefixes = [] | 
					
						
							|  |  |  |     args = {"Bucket": 'mybucket', "Delimiter": "/", "Prefix": "test/", "MaxKeys": max_keys} | 
					
						
							|  |  |  |     resp = {"IsTruncated": True} | 
					
						
							|  |  |  |     while resp.get("IsTruncated", False): | 
					
						
							|  |  |  |         if "NextContinuationToken" in resp: | 
					
						
							|  |  |  |             args["ContinuationToken"] = resp["NextContinuationToken"] | 
					
						
							|  |  |  |         resp = s3.list_objects_v2(**args) | 
					
						
							|  |  |  |         if "CommonPrefixes" in resp: | 
					
						
							|  |  |  |             assert len(resp["CommonPrefixes"]) == max_keys | 
					
						
							|  |  |  |             prefixes.extend(i["Prefix"] for i in resp["CommonPrefixes"]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert prefixes == [k[:k.rindex('/') + 1] for k in keys] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-09 03:21:43 +01:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_list_objects_v2_truncated_response(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket='mybucket') | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='one', Body=b'1') | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='two', Body=b'22') | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='three', Body=b'333') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # First list | 
					
						
							|  |  |  |     resp = s3.list_objects_v2(Bucket='mybucket', MaxKeys=1) | 
					
						
							|  |  |  |     listed_object = resp['Contents'][0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert listed_object['Key'] == 'one' | 
					
						
							|  |  |  |     assert resp['MaxKeys'] == 1 | 
					
						
							|  |  |  |     assert resp['Prefix'] == '' | 
					
						
							|  |  |  |     assert resp['KeyCount'] == 1 | 
					
						
							|  |  |  |     assert resp['IsTruncated'] == True | 
					
						
							|  |  |  |     assert 'Delimiter' not in resp | 
					
						
							|  |  |  |     assert 'StartAfter' not in resp | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     assert 'Owner' not in listed_object  # owner info was not requested | 
					
						
							| 
									
										
										
										
											2017-02-09 03:21:43 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     next_token = resp['NextContinuationToken'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Second list | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     resp = s3.list_objects_v2( | 
					
						
							|  |  |  |         Bucket='mybucket', MaxKeys=1, ContinuationToken=next_token) | 
					
						
							| 
									
										
										
										
											2017-02-09 03:21:43 +01:00
										 |  |  |     listed_object = resp['Contents'][0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert listed_object['Key'] == 'three' | 
					
						
							|  |  |  |     assert resp['MaxKeys'] == 1 | 
					
						
							|  |  |  |     assert resp['Prefix'] == '' | 
					
						
							|  |  |  |     assert resp['KeyCount'] == 1 | 
					
						
							|  |  |  |     assert resp['IsTruncated'] == True | 
					
						
							|  |  |  |     assert 'Delimiter' not in resp | 
					
						
							|  |  |  |     assert 'StartAfter' not in resp | 
					
						
							|  |  |  |     assert 'Owner' not in listed_object | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     next_token = resp['NextContinuationToken'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Third list | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     resp = s3.list_objects_v2( | 
					
						
							|  |  |  |         Bucket='mybucket', MaxKeys=1, ContinuationToken=next_token) | 
					
						
							| 
									
										
										
										
											2017-02-09 03:21:43 +01:00
										 |  |  |     listed_object = resp['Contents'][0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert listed_object['Key'] == 'two' | 
					
						
							|  |  |  |     assert resp['MaxKeys'] == 1 | 
					
						
							|  |  |  |     assert resp['Prefix'] == '' | 
					
						
							|  |  |  |     assert resp['KeyCount'] == 1 | 
					
						
							|  |  |  |     assert resp['IsTruncated'] == False | 
					
						
							|  |  |  |     assert 'Delimiter' not in resp | 
					
						
							|  |  |  |     assert 'Owner' not in listed_object | 
					
						
							|  |  |  |     assert 'StartAfter' not in resp | 
					
						
							|  |  |  |     assert 'NextContinuationToken' not in resp | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_list_objects_v2_truncated_response_start_after(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket='mybucket') | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='one', Body=b'1') | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='two', Body=b'22') | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='three', Body=b'333') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # First list | 
					
						
							|  |  |  |     resp = s3.list_objects_v2(Bucket='mybucket', MaxKeys=1, StartAfter='one') | 
					
						
							|  |  |  |     listed_object = resp['Contents'][0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert listed_object['Key'] == 'three' | 
					
						
							|  |  |  |     assert resp['MaxKeys'] == 1 | 
					
						
							|  |  |  |     assert resp['Prefix'] == '' | 
					
						
							|  |  |  |     assert resp['KeyCount'] == 1 | 
					
						
							|  |  |  |     assert resp['IsTruncated'] == True | 
					
						
							|  |  |  |     assert resp['StartAfter'] == 'one' | 
					
						
							|  |  |  |     assert 'Delimiter' not in resp | 
					
						
							|  |  |  |     assert 'Owner' not in listed_object | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     next_token = resp['NextContinuationToken'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Second list | 
					
						
							|  |  |  |     # The ContinuationToken must take precedence over StartAfter. | 
					
						
							|  |  |  |     resp = s3.list_objects_v2(Bucket='mybucket', MaxKeys=1, StartAfter='one', | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |                               ContinuationToken=next_token) | 
					
						
							| 
									
										
										
										
											2017-02-09 03:21:43 +01:00
										 |  |  |     listed_object = resp['Contents'][0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert listed_object['Key'] == 'two' | 
					
						
							|  |  |  |     assert resp['MaxKeys'] == 1 | 
					
						
							|  |  |  |     assert resp['Prefix'] == '' | 
					
						
							|  |  |  |     assert resp['KeyCount'] == 1 | 
					
						
							|  |  |  |     assert resp['IsTruncated'] == False | 
					
						
							|  |  |  |     # When ContinuationToken is given, StartAfter is ignored. This also means | 
					
						
							|  |  |  |     # AWS does not return it in the response. | 
					
						
							|  |  |  |     assert 'StartAfter' not in resp | 
					
						
							|  |  |  |     assert 'Delimiter' not in resp | 
					
						
							|  |  |  |     assert 'Owner' not in listed_object | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_list_objects_v2_fetch_owner(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket='mybucket') | 
					
						
							|  |  |  |     s3.put_object(Bucket='mybucket', Key='one', Body=b'11') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.list_objects_v2(Bucket='mybucket', FetchOwner=True) | 
					
						
							|  |  |  |     owner = resp['Contents'][0]['Owner'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert 'ID' in owner | 
					
						
							|  |  |  |     assert 'DisplayName' in owner | 
					
						
							|  |  |  |     assert len(owner.keys()) == 2 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-20 15:42:21 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-02 09:54:23 -04:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_bucket_create(): | 
					
						
							|  |  |  |     s3 = boto3.resource('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket="blah") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.Object('blah', 'hello.txt').put(Body="some text") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     s3.Object('blah', 'hello.txt').get()['Body'].read().decode( | 
					
						
							|  |  |  |         "utf-8").should.equal("some text") | 
					
						
							| 
									
										
										
										
											2015-08-13 21:16:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 19:29:59 -04:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_bucket_create_duplicate(): | 
					
						
							|  |  |  |     s3 = boto3.resource('s3', region_name='us-west-2') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket="blah", CreateBucketConfiguration={ | 
					
						
							|  |  |  |         'LocationConstraint': 'us-west-2', | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     with assert_raises(ClientError) as exc: | 
					
						
							|  |  |  |         s3.create_bucket( | 
					
						
							|  |  |  |             Bucket="blah", | 
					
						
							|  |  |  |             CreateBucketConfiguration={ | 
					
						
							|  |  |  |                 'LocationConstraint': 'us-west-2', | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     exc.exception.response['Error']['Code'].should.equal('BucketAlreadyExists') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-20 18:39:31 +01:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_bucket_create_force_us_east_1(): | 
					
						
							|  |  |  |     s3 = boto3.resource('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     with assert_raises(ClientError) as exc: | 
					
						
							|  |  |  |         s3.create_bucket(Bucket="blah", CreateBucketConfiguration={ | 
					
						
							|  |  |  |             'LocationConstraint': 'us-east-1', | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     exc.exception.response['Error']['Code'].should.equal('InvalidLocationConstraint') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-27 14:43:03 -05:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_bucket_create_eu_central(): | 
					
						
							|  |  |  |     s3 = boto3.resource('s3', region_name='eu-central-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket="blah") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.Object('blah', 'hello.txt').put(Body="some text") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     s3.Object('blah', 'hello.txt').get()['Body'].read().decode( | 
					
						
							|  |  |  |         "utf-8").should.equal("some text") | 
					
						
							| 
									
										
										
										
											2015-11-27 14:43:03 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-13 21:16:55 +00:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_head_object(): | 
					
						
							|  |  |  |     s3 = boto3.resource('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket="blah") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.Object('blah', 'hello.txt').put(Body="some text") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     s3.Object('blah', 'hello.txt').meta.client.head_object( | 
					
						
							|  |  |  |         Bucket='blah', Key='hello.txt') | 
					
						
							| 
									
										
										
										
											2015-08-13 21:16:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-17 13:03:33 -07:00
										 |  |  |     with assert_raises(ClientError) as e: | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |         s3.Object('blah', 'hello2.txt').meta.client.head_object( | 
					
						
							|  |  |  |             Bucket='blah', Key='hello_bad.txt') | 
					
						
							| 
									
										
										
										
											2017-05-17 13:03:33 -07:00
										 |  |  |     e.exception.response['Error']['Code'].should.equal('404') | 
					
						
							| 
									
										
										
										
											2015-11-04 18:55:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-19 17:18:31 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_bucket_deletion(): | 
					
						
							|  |  |  |     cli = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     cli.create_bucket(Bucket="foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cli.put_object(Bucket="foobar", Key="the-key", Body="some value") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Try to delete a bucket that still has keys | 
					
						
							|  |  |  |     cli.delete_bucket.when.called_with(Bucket="foobar").should.throw( | 
					
						
							|  |  |  |         cli.exceptions.ClientError, | 
					
						
							|  |  |  |         ('An error occurred (BucketNotEmpty) when calling the DeleteBucket operation: ' | 
					
						
							|  |  |  |          'The bucket you tried to delete is not empty')) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cli.delete_object(Bucket="foobar", Key="the-key") | 
					
						
							|  |  |  |     cli.delete_bucket(Bucket="foobar") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Get non-existing bucket | 
					
						
							|  |  |  |     cli.head_bucket.when.called_with(Bucket="foobar").should.throw( | 
					
						
							|  |  |  |         cli.exceptions.ClientError, | 
					
						
							|  |  |  |         "An error occurred (404) when calling the HeadBucket operation: Not Found") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Delete non-existing bucket | 
					
						
							|  |  |  |     cli.delete_bucket.when.called_with(Bucket="foobar").should.throw(cli.exceptions.NoSuchBucket) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-08 19:32:34 +00:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_get_object(): | 
					
						
							|  |  |  |     s3 = boto3.resource('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket="blah") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.Object('blah', 'hello.txt').put(Body="some text") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-23 21:37:43 -05:00
										 |  |  |     s3.Object('blah', 'hello.txt').meta.client.head_object( | 
					
						
							|  |  |  |         Bucket='blah', Key='hello.txt') | 
					
						
							| 
									
										
										
										
											2016-07-08 19:32:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(ClientError) as e: | 
					
						
							|  |  |  |         s3.Object('blah', 'hello2.txt').get() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     e.exception.response['Error']['Code'].should.equal('NoSuchKey') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-28 13:17:33 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_head_object_with_versioning(): | 
					
						
							|  |  |  |     s3 = boto3.resource('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket = s3.create_bucket(Bucket='blah') | 
					
						
							|  |  |  |     bucket.Versioning().enable() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     old_content = 'some text' | 
					
						
							|  |  |  |     new_content = 'some new text' | 
					
						
							|  |  |  |     s3.Object('blah', 'hello.txt').put(Body=old_content) | 
					
						
							|  |  |  |     s3.Object('blah', 'hello.txt').put(Body=new_content) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     versions = list(s3.Bucket('blah').object_versions.all()) | 
					
						
							|  |  |  |     latest = list(filter(lambda item: item.is_latest, versions))[0] | 
					
						
							|  |  |  |     oldest = list(filter(lambda item: not item.is_latest, versions))[0] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-28 13:17:33 -07:00
										 |  |  |     head_object = s3.Object('blah', 'hello.txt').meta.client.head_object( | 
					
						
							|  |  |  |         Bucket='blah', Key='hello.txt') | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     head_object['VersionId'].should.equal(latest.id) | 
					
						
							| 
									
										
										
										
											2016-04-28 13:17:33 -07:00
										 |  |  |     head_object['ContentLength'].should.equal(len(new_content)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     old_head_object = s3.Object('blah', 'hello.txt').meta.client.head_object( | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |         Bucket='blah', Key='hello.txt', VersionId=oldest.id) | 
					
						
							|  |  |  |     old_head_object['VersionId'].should.equal(oldest.id) | 
					
						
							| 
									
										
										
										
											2016-04-28 13:17:33 -07:00
										 |  |  |     old_head_object['ContentLength'].should.equal(len(old_content)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     old_head_object['VersionId'].should_not.equal(head_object['VersionId']) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-28 13:17:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-19 00:22:35 +01:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_copy_object_with_versioning(): | 
					
						
							|  |  |  |     client = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client.create_bucket(Bucket='blah', CreateBucketConfiguration={'LocationConstraint': 'eu-west-1'}) | 
					
						
							|  |  |  |     client.put_bucket_versioning(Bucket='blah', VersioningConfiguration={'Status': 'Enabled'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client.put_object(Bucket='blah', Key='test1', Body=b'test1') | 
					
						
							|  |  |  |     client.put_object(Bucket='blah', Key='test2', Body=b'test2') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     obj1_version = client.get_object(Bucket='blah', Key='test1')['VersionId'] | 
					
						
							|  |  |  |     obj2_version = client.get_object(Bucket='blah', Key='test2')['VersionId'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client.copy_object(CopySource={'Bucket': 'blah', 'Key': 'test1'}, Bucket='blah', Key='test2') | 
					
						
							|  |  |  |     obj2_version_new = client.get_object(Bucket='blah', Key='test2')['VersionId'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Version should be different to previous version | 
					
						
							|  |  |  |     obj2_version_new.should_not.equal(obj2_version) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-25 12:19:33 -07:00
										 |  |  |     client.copy_object(CopySource={'Bucket': 'blah', 'Key': 'test2', 'VersionId': obj2_version}, Bucket='blah', Key='test3') | 
					
						
							|  |  |  |     obj3_version_new = client.get_object(Bucket='blah', Key='test3')['VersionId'] | 
					
						
							|  |  |  |     obj3_version_new.should_not.equal(obj2_version_new) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Copy file that doesn't exist | 
					
						
							|  |  |  |     with assert_raises(ClientError) as e: | 
					
						
							|  |  |  |         client.copy_object(CopySource={'Bucket': 'blah', 'Key': 'test4', 'VersionId': obj2_version}, Bucket='blah', Key='test5') | 
					
						
							|  |  |  |     e.exception.response['Error']['Code'].should.equal('404') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     response = client.create_multipart_upload(Bucket='blah', Key='test4') | 
					
						
							|  |  |  |     upload_id = response['UploadId'] | 
					
						
							|  |  |  |     response = client.upload_part_copy(Bucket='blah', Key='test4', CopySource={'Bucket': 'blah', 'Key': 'test3', 'VersionId': obj3_version_new}, | 
					
						
							|  |  |  |                                        UploadId=upload_id, PartNumber=1) | 
					
						
							|  |  |  |     etag = response["CopyPartResult"]["ETag"] | 
					
						
							|  |  |  |     client.complete_multipart_upload( | 
					
						
							|  |  |  |         Bucket='blah', Key='test4', UploadId=upload_id, | 
					
						
							|  |  |  |         MultipartUpload={'Parts': [{'ETag': etag, 'PartNumber': 1}]}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     response = client.get_object(Bucket='blah', Key='test4') | 
					
						
							|  |  |  |     data = response["Body"].read() | 
					
						
							|  |  |  |     data.should.equal(b'test2') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-19 00:22:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-02 15:30:01 +02:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_copy_object_from_unversioned_to_versioned_bucket(): | 
					
						
							|  |  |  |     client = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client.create_bucket(Bucket='src', CreateBucketConfiguration={'LocationConstraint': 'eu-west-1'}) | 
					
						
							|  |  |  |     client.create_bucket(Bucket='dest', CreateBucketConfiguration={'LocationConstraint': 'eu-west-1'}) | 
					
						
							|  |  |  |     client.put_bucket_versioning(Bucket='dest', VersioningConfiguration={'Status': 'Enabled'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client.put_object(Bucket='src', Key='test', Body=b'content') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     obj2_version_new = client.copy_object(CopySource={'Bucket': 'src', 'Key': 'test'}, Bucket='dest', Key='test') \ | 
					
						
							|  |  |  |         .get('VersionId') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # VersionId should be present in the response | 
					
						
							|  |  |  |     obj2_version_new.should_not.equal(None) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-06 01:50:22 -08:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_deleted_versionings_list(): | 
					
						
							|  |  |  |     client = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client.create_bucket(Bucket='blah') | 
					
						
							|  |  |  |     client.put_bucket_versioning(Bucket='blah', VersioningConfiguration={'Status': 'Enabled'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client.put_object(Bucket='blah', Key='test1', Body=b'test1') | 
					
						
							|  |  |  |     client.put_object(Bucket='blah', Key='test2', Body=b'test2') | 
					
						
							|  |  |  |     client.delete_objects(Bucket='blah', Delete={'Objects': [{'Key': 'test1'}]}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     listed = client.list_objects_v2(Bucket='blah') | 
					
						
							|  |  |  |     assert len(listed['Contents']) == 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-03 02:30:29 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_delete_versioned_bucket(): | 
					
						
							|  |  |  |     client = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client.create_bucket(Bucket='blah') | 
					
						
							|  |  |  |     client.put_bucket_versioning(Bucket='blah', VersioningConfiguration={'Status': 'Enabled'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = client.put_object(Bucket='blah', Key='test1', Body=b'test1') | 
					
						
							|  |  |  |     client.delete_object(Bucket='blah', Key='test1', VersionId=resp["VersionId"]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     client.delete_bucket(Bucket='blah') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-19 15:59:25 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_head_object_if_modified_since(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = "blah" | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     key = 'hello.txt' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-23 11:29:01 -07:00
										 |  |  |     s3.put_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Body='test' | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2017-05-19 15:59:25 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(botocore.exceptions.ClientError) as err: | 
					
						
							|  |  |  |         s3.head_object( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             Key=key, | 
					
						
							| 
									
										
										
										
											2017-05-23 11:29:01 -07:00
										 |  |  |             IfModifiedSince=datetime.datetime.utcnow() + datetime.timedelta(hours=1) | 
					
						
							| 
									
										
										
										
											2017-05-19 15:59:25 -07:00
										 |  |  |         ) | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response['Error'].should.equal({'Code': '304', 'Message': 'Not Modified'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-20 15:42:21 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | @reduced_min_part_size | 
					
						
							|  |  |  | def test_boto3_multipart_etag(): | 
					
						
							|  |  |  |     # Create Bucket so that test can run | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket='mybucket') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     upload_id = s3.create_multipart_upload( | 
					
						
							|  |  |  |         Bucket='mybucket', Key='the-key')['UploadId'] | 
					
						
							|  |  |  |     part1 = b'0' * REDUCED_PART_SIZE | 
					
						
							|  |  |  |     etags = [] | 
					
						
							|  |  |  |     etags.append( | 
					
						
							|  |  |  |         s3.upload_part(Bucket='mybucket', Key='the-key', PartNumber=1, | 
					
						
							|  |  |  |                        UploadId=upload_id, Body=part1)['ETag']) | 
					
						
							|  |  |  |     # last part, can be less than 5 MB | 
					
						
							|  |  |  |     part2 = b'1' | 
					
						
							|  |  |  |     etags.append( | 
					
						
							|  |  |  |         s3.upload_part(Bucket='mybucket', Key='the-key', PartNumber=2, | 
					
						
							|  |  |  |                        UploadId=upload_id, Body=part2)['ETag']) | 
					
						
							|  |  |  |     s3.complete_multipart_upload( | 
					
						
							|  |  |  |         Bucket='mybucket', Key='the-key', UploadId=upload_id, | 
					
						
							|  |  |  |         MultipartUpload={'Parts': [{'ETag': etag, 'PartNumber': i} | 
					
						
							|  |  |  |                                    for i, etag in enumerate(etags, 1)]}) | 
					
						
							|  |  |  |     # we should get both parts as the key contents | 
					
						
							|  |  |  |     resp = s3.get_object(Bucket='mybucket', Key='the-key') | 
					
						
							| 
									
										
										
										
											2017-02-23 19:43:48 -05:00
										 |  |  |     resp['ETag'].should.equal(EXPECTED_ETAG) | 
					
						
							| 
									
										
										
										
											2016-09-20 15:42:21 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-15 22:36:12 -04:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_put_object_with_tagging(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     key = 'key-with-tags' | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.put_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Body='test', | 
					
						
							|  |  |  |         Tagging='foo=bar', | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.get_object_tagging(Bucket=bucket_name, Key=key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp['TagSet'].should.contain({'Key': 'foo', 'Value': 'bar'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-07 11:30:05 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_put_bucket_tagging(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     bucket_name = "mybucket" | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-22 13:35:00 -07:00
										 |  |  |     # With 1 tag: | 
					
						
							|  |  |  |     resp = s3.put_bucket_tagging(Bucket=bucket_name, | 
					
						
							|  |  |  |                                  Tagging={ | 
					
						
							|  |  |  |                                      "TagSet": [ | 
					
						
							|  |  |  |                                          { | 
					
						
							|  |  |  |                                              "Key": "TagOne", | 
					
						
							|  |  |  |                                              "Value": "ValueOne" | 
					
						
							|  |  |  |                                          } | 
					
						
							|  |  |  |                                      ] | 
					
						
							|  |  |  |                                  }) | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(200) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # With multiple tags: | 
					
						
							| 
									
										
										
										
											2017-09-07 11:30:05 -07:00
										 |  |  |     resp = s3.put_bucket_tagging(Bucket=bucket_name, | 
					
						
							|  |  |  |                                  Tagging={ | 
					
						
							|  |  |  |                                      "TagSet": [ | 
					
						
							|  |  |  |                                          { | 
					
						
							|  |  |  |                                              "Key": "TagOne", | 
					
						
							|  |  |  |                                              "Value": "ValueOne" | 
					
						
							|  |  |  |                                          }, | 
					
						
							|  |  |  |                                          { | 
					
						
							|  |  |  |                                              "Key": "TagTwo", | 
					
						
							|  |  |  |                                              "Value": "ValueTwo" | 
					
						
							|  |  |  |                                          } | 
					
						
							|  |  |  |                                      ] | 
					
						
							|  |  |  |                                  }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(200) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # No tags is also OK: | 
					
						
							|  |  |  |     resp = s3.put_bucket_tagging(Bucket=bucket_name, Tagging={ | 
					
						
							|  |  |  |         "TagSet": [] | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(200) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-26 15:56:46 -08:00
										 |  |  |     # With duplicate tag keys: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         resp = s3.put_bucket_tagging(Bucket=bucket_name, | 
					
						
							|  |  |  |                                      Tagging={ | 
					
						
							|  |  |  |                                          "TagSet": [ | 
					
						
							|  |  |  |                                              { | 
					
						
							|  |  |  |                                                  "Key": "TagOne", | 
					
						
							|  |  |  |                                                  "Value": "ValueOne" | 
					
						
							|  |  |  |                                              }, | 
					
						
							|  |  |  |                                              { | 
					
						
							|  |  |  |                                                  "Key": "TagOne", | 
					
						
							|  |  |  |                                                  "Value": "ValueOneAgain" | 
					
						
							|  |  |  |                                              } | 
					
						
							|  |  |  |                                          ] | 
					
						
							|  |  |  |                                      }) | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response["Error"]["Code"].should.equal("InvalidTag") | 
					
						
							|  |  |  |     e.response["Error"]["Message"].should.equal("Cannot provide multiple Tags with the same key") | 
					
						
							| 
									
										
										
										
											2017-09-07 11:30:05 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_get_bucket_tagging(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     bucket_name = "mybucket" | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     s3.put_bucket_tagging(Bucket=bucket_name, | 
					
						
							|  |  |  |                           Tagging={ | 
					
						
							|  |  |  |                               "TagSet": [ | 
					
						
							|  |  |  |                                   { | 
					
						
							|  |  |  |                                       "Key": "TagOne", | 
					
						
							|  |  |  |                                       "Value": "ValueOne" | 
					
						
							|  |  |  |                                   }, | 
					
						
							|  |  |  |                                   { | 
					
						
							|  |  |  |                                       "Key": "TagTwo", | 
					
						
							|  |  |  |                                       "Value": "ValueTwo" | 
					
						
							|  |  |  |                                   } | 
					
						
							|  |  |  |                               ] | 
					
						
							|  |  |  |                           }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Get the tags for the bucket: | 
					
						
							|  |  |  |     resp = s3.get_bucket_tagging(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(200) | 
					
						
							|  |  |  |     len(resp["TagSet"]).should.equal(2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # With no tags: | 
					
						
							|  |  |  |     s3.put_bucket_tagging(Bucket=bucket_name, Tagging={ | 
					
						
							|  |  |  |         "TagSet": [] | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.get_bucket_tagging(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response["Error"]["Code"].should.equal("NoSuchTagSet") | 
					
						
							|  |  |  |     e.response["Error"]["Message"].should.equal("The TagSet does not exist") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_delete_bucket_tagging(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     bucket_name = "mybucket" | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.put_bucket_tagging(Bucket=bucket_name, | 
					
						
							|  |  |  |                           Tagging={ | 
					
						
							|  |  |  |                               "TagSet": [ | 
					
						
							|  |  |  |                                   { | 
					
						
							|  |  |  |                                       "Key": "TagOne", | 
					
						
							|  |  |  |                                       "Value": "ValueOne" | 
					
						
							|  |  |  |                                   }, | 
					
						
							|  |  |  |                                   { | 
					
						
							|  |  |  |                                       "Key": "TagTwo", | 
					
						
							|  |  |  |                                       "Value": "ValueTwo" | 
					
						
							|  |  |  |                                   } | 
					
						
							|  |  |  |                               ] | 
					
						
							|  |  |  |                           }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.delete_bucket_tagging(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(204) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.get_bucket_tagging(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response["Error"]["Code"].should.equal("NoSuchTagSet") | 
					
						
							|  |  |  |     e.response["Error"]["Message"].should.equal("The TagSet does not exist") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_put_bucket_cors(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     bucket_name = "mybucket" | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.put_bucket_cors(Bucket=bucket_name, CORSConfiguration={ | 
					
						
							|  |  |  |         "CORSRules": [ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "AllowedOrigins": [ | 
					
						
							|  |  |  |                     "*" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "AllowedMethods": [ | 
					
						
							|  |  |  |                     "GET", | 
					
						
							|  |  |  |                     "POST" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "AllowedHeaders": [ | 
					
						
							|  |  |  |                     "Authorization" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "ExposeHeaders": [ | 
					
						
							|  |  |  |                     "x-amz-request-id" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "MaxAgeSeconds": 123 | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "AllowedOrigins": [ | 
					
						
							|  |  |  |                     "*" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "AllowedMethods": [ | 
					
						
							|  |  |  |                     "PUT" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "AllowedHeaders": [ | 
					
						
							|  |  |  |                     "Authorization" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "ExposeHeaders": [ | 
					
						
							|  |  |  |                     "x-amz-request-id" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "MaxAgeSeconds": 123 | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(200) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_cors(Bucket=bucket_name, CORSConfiguration={ | 
					
						
							|  |  |  |             "CORSRules": [ | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "AllowedOrigins": [ | 
					
						
							|  |  |  |                         "*" | 
					
						
							|  |  |  |                     ], | 
					
						
							|  |  |  |                     "AllowedMethods": [ | 
					
						
							|  |  |  |                         "NOTREAL", | 
					
						
							|  |  |  |                         "POST" | 
					
						
							|  |  |  |                     ] | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response["Error"]["Code"].should.equal("InvalidRequest") | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  |     e.response["Error"]["Message"].should.equal("Found unsupported HTTP method in CORS config. " | 
					
						
							| 
									
										
										
										
											2017-09-07 11:30:05 -07:00
										 |  |  |                                                 "Unsupported method is NOTREAL") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_cors(Bucket=bucket_name, CORSConfiguration={ | 
					
						
							|  |  |  |             "CORSRules": [] | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response["Error"]["Code"].should.equal("MalformedXML") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # And 101: | 
					
						
							|  |  |  |     many_rules = [{"AllowedOrigins": ["*"], "AllowedMethods": ["GET"]}] * 101 | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_cors(Bucket=bucket_name, CORSConfiguration={ | 
					
						
							|  |  |  |             "CORSRules": many_rules | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response["Error"]["Code"].should.equal("MalformedXML") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_get_bucket_cors(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     bucket_name = "mybucket" | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Without CORS: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.get_bucket_cors(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response["Error"]["Code"].should.equal("NoSuchCORSConfiguration") | 
					
						
							|  |  |  |     e.response["Error"]["Message"].should.equal("The CORS configuration does not exist") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.put_bucket_cors(Bucket=bucket_name, CORSConfiguration={ | 
					
						
							|  |  |  |         "CORSRules": [ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "AllowedOrigins": [ | 
					
						
							|  |  |  |                     "*" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "AllowedMethods": [ | 
					
						
							|  |  |  |                     "GET", | 
					
						
							|  |  |  |                     "POST" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "AllowedHeaders": [ | 
					
						
							|  |  |  |                     "Authorization" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "ExposeHeaders": [ | 
					
						
							|  |  |  |                     "x-amz-request-id" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "MaxAgeSeconds": 123 | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "AllowedOrigins": [ | 
					
						
							|  |  |  |                     "*" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "AllowedMethods": [ | 
					
						
							|  |  |  |                     "PUT" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "AllowedHeaders": [ | 
					
						
							|  |  |  |                     "Authorization" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "ExposeHeaders": [ | 
					
						
							|  |  |  |                     "x-amz-request-id" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "MaxAgeSeconds": 123 | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.get_bucket_cors(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(200) | 
					
						
							|  |  |  |     len(resp["CORSRules"]).should.equal(2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_delete_bucket_cors(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     bucket_name = "mybucket" | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     s3.put_bucket_cors(Bucket=bucket_name, CORSConfiguration={ | 
					
						
							|  |  |  |         "CORSRules": [ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "AllowedOrigins": [ | 
					
						
							|  |  |  |                     "*" | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 "AllowedMethods": [ | 
					
						
							|  |  |  |                     "GET" | 
					
						
							|  |  |  |                 ] | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ] | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.delete_bucket_cors(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(204) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Verify deletion: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.get_bucket_cors(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response["Error"]["Code"].should.equal("NoSuchCORSConfiguration") | 
					
						
							|  |  |  |     e.response["Error"]["Message"].should.equal("The CORS configuration does not exist") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_put_bucket_acl_body(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     s3.create_bucket(Bucket="bucket") | 
					
						
							|  |  |  |     bucket_owner = s3.get_bucket_acl(Bucket="bucket")["Owner"] | 
					
						
							|  |  |  |     s3.put_bucket_acl(Bucket="bucket", AccessControlPolicy={ | 
					
						
							|  |  |  |         "Grants": [ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "Grantee": { | 
					
						
							|  |  |  |                     "URI": "http://acs.amazonaws.com/groups/s3/LogDelivery", | 
					
						
							|  |  |  |                     "Type": "Group" | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 "Permission": "WRITE" | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "Grantee": { | 
					
						
							|  |  |  |                     "URI": "http://acs.amazonaws.com/groups/s3/LogDelivery", | 
					
						
							|  |  |  |                     "Type": "Group" | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 "Permission": "READ_ACP" | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ], | 
					
						
							|  |  |  |         "Owner": bucket_owner | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result = s3.get_bucket_acl(Bucket="bucket") | 
					
						
							|  |  |  |     assert len(result["Grants"]) == 2 | 
					
						
							|  |  |  |     for g in result["Grants"]: | 
					
						
							|  |  |  |         assert g["Grantee"]["URI"] == "http://acs.amazonaws.com/groups/s3/LogDelivery" | 
					
						
							|  |  |  |         assert g["Grantee"]["Type"] == "Group" | 
					
						
							|  |  |  |         assert g["Permission"] in ["WRITE", "READ_ACP"] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # With one: | 
					
						
							|  |  |  |     s3.put_bucket_acl(Bucket="bucket", AccessControlPolicy={ | 
					
						
							|  |  |  |         "Grants": [ | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 "Grantee": { | 
					
						
							|  |  |  |                     "URI": "http://acs.amazonaws.com/groups/s3/LogDelivery", | 
					
						
							|  |  |  |                     "Type": "Group" | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 "Permission": "WRITE" | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ], | 
					
						
							|  |  |  |         "Owner": bucket_owner | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     result = s3.get_bucket_acl(Bucket="bucket") | 
					
						
							|  |  |  |     assert len(result["Grants"]) == 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # With no owner: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_acl(Bucket="bucket", AccessControlPolicy={ | 
					
						
							|  |  |  |             "Grants": [ | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Grantee": { | 
					
						
							|  |  |  |                         "URI": "http://acs.amazonaws.com/groups/s3/LogDelivery", | 
					
						
							|  |  |  |                         "Type": "Group" | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                     "Permission": "WRITE" | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Code"] == "MalformedACLError" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # With incorrect permission: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_acl(Bucket="bucket", AccessControlPolicy={ | 
					
						
							|  |  |  |             "Grants": [ | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Grantee": { | 
					
						
							|  |  |  |                         "URI": "http://acs.amazonaws.com/groups/s3/LogDelivery", | 
					
						
							|  |  |  |                         "Type": "Group" | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                     "Permission": "lskjflkasdjflkdsjfalisdjflkdsjf" | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ], | 
					
						
							|  |  |  |             "Owner": bucket_owner | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Code"] == "MalformedACLError" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Clear the ACLs: | 
					
						
							|  |  |  |     result = s3.put_bucket_acl(Bucket="bucket", AccessControlPolicy={"Grants": [], "Owner": bucket_owner}) | 
					
						
							|  |  |  |     assert not result.get("Grants") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-21 09:11:24 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_put_bucket_notification(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     s3.create_bucket(Bucket="bucket") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # With no configuration: | 
					
						
							|  |  |  |     result = s3.get_bucket_notification(Bucket="bucket") | 
					
						
							|  |  |  |     assert not result.get("TopicConfigurations") | 
					
						
							|  |  |  |     assert not result.get("QueueConfigurations") | 
					
						
							|  |  |  |     assert not result.get("LambdaFunctionConfigurations") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Place proper topic configuration: | 
					
						
							|  |  |  |     s3.put_bucket_notification_configuration(Bucket="bucket", | 
					
						
							|  |  |  |                                              NotificationConfiguration={ | 
					
						
							|  |  |  |                                                  "TopicConfigurations": [ | 
					
						
							|  |  |  |                                                      { | 
					
						
							|  |  |  |                                                          "TopicArn": "arn:aws:sns:us-east-1:012345678910:mytopic", | 
					
						
							|  |  |  |                                                          "Events": [ | 
					
						
							|  |  |  |                                                              "s3:ObjectCreated:*", | 
					
						
							|  |  |  |                                                              "s3:ObjectRemoved:*" | 
					
						
							|  |  |  |                                                          ] | 
					
						
							|  |  |  |                                                      }, | 
					
						
							|  |  |  |                                                      { | 
					
						
							|  |  |  |                                                          "TopicArn": "arn:aws:sns:us-east-1:012345678910:myothertopic", | 
					
						
							|  |  |  |                                                          "Events": [ | 
					
						
							|  |  |  |                                                              "s3:ObjectCreated:*" | 
					
						
							|  |  |  |                                                          ], | 
					
						
							|  |  |  |                                                          "Filter": { | 
					
						
							|  |  |  |                                                              "Key": { | 
					
						
							|  |  |  |                                                                  "FilterRules": [ | 
					
						
							|  |  |  |                                                                      { | 
					
						
							|  |  |  |                                                                          "Name": "prefix", | 
					
						
							|  |  |  |                                                                          "Value": "images/" | 
					
						
							|  |  |  |                                                                      }, | 
					
						
							|  |  |  |                                                                      { | 
					
						
							|  |  |  |                                                                          "Name": "suffix", | 
					
						
							|  |  |  |                                                                          "Value": "png" | 
					
						
							|  |  |  |                                                                      } | 
					
						
							|  |  |  |                                                                  ] | 
					
						
							|  |  |  |                                                              } | 
					
						
							|  |  |  |                                                          } | 
					
						
							|  |  |  |                                                      } | 
					
						
							|  |  |  |                                                  ] | 
					
						
							|  |  |  |                                              }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Verify to completion: | 
					
						
							|  |  |  |     result = s3.get_bucket_notification_configuration(Bucket="bucket") | 
					
						
							|  |  |  |     assert len(result["TopicConfigurations"]) == 2 | 
					
						
							|  |  |  |     assert not result.get("QueueConfigurations") | 
					
						
							|  |  |  |     assert not result.get("LambdaFunctionConfigurations") | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][0]["TopicArn"] == "arn:aws:sns:us-east-1:012345678910:mytopic" | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][1]["TopicArn"] == "arn:aws:sns:us-east-1:012345678910:myothertopic" | 
					
						
							|  |  |  |     assert len(result["TopicConfigurations"][0]["Events"]) == 2 | 
					
						
							|  |  |  |     assert len(result["TopicConfigurations"][1]["Events"]) == 1 | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][0]["Events"][0] == "s3:ObjectCreated:*" | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][0]["Events"][1] == "s3:ObjectRemoved:*" | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][1]["Events"][0] == "s3:ObjectCreated:*" | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][0]["Id"] | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][1]["Id"] | 
					
						
							|  |  |  |     assert not result["TopicConfigurations"][0].get("Filter") | 
					
						
							|  |  |  |     assert len(result["TopicConfigurations"][1]["Filter"]["Key"]["FilterRules"]) == 2 | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][1]["Filter"]["Key"]["FilterRules"][0]["Name"] == "prefix" | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][1]["Filter"]["Key"]["FilterRules"][0]["Value"] == "images/" | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][1]["Filter"]["Key"]["FilterRules"][1]["Name"] == "suffix" | 
					
						
							|  |  |  |     assert result["TopicConfigurations"][1]["Filter"]["Key"]["FilterRules"][1]["Value"] == "png" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Place proper queue configuration: | 
					
						
							|  |  |  |     s3.put_bucket_notification_configuration(Bucket="bucket", | 
					
						
							|  |  |  |                                              NotificationConfiguration={ | 
					
						
							|  |  |  |                                                  "QueueConfigurations": [ | 
					
						
							|  |  |  |                                                      { | 
					
						
							|  |  |  |                                                          "Id": "SomeID", | 
					
						
							|  |  |  |                                                          "QueueArn": "arn:aws:sqs:us-east-1:012345678910:myQueue", | 
					
						
							|  |  |  |                                                          "Events": ["s3:ObjectCreated:*"], | 
					
						
							|  |  |  |                                                          "Filter": { | 
					
						
							|  |  |  |                                                              "Key": { | 
					
						
							|  |  |  |                                                                  "FilterRules": [ | 
					
						
							|  |  |  |                                                                      { | 
					
						
							|  |  |  |                                                                          "Name": "prefix", | 
					
						
							|  |  |  |                                                                          "Value": "images/" | 
					
						
							|  |  |  |                                                                      } | 
					
						
							|  |  |  |                                                                  ] | 
					
						
							|  |  |  |                                                              } | 
					
						
							|  |  |  |                                                          } | 
					
						
							|  |  |  |                                                      } | 
					
						
							|  |  |  |                                                  ] | 
					
						
							|  |  |  |                                              }) | 
					
						
							|  |  |  |     result = s3.get_bucket_notification_configuration(Bucket="bucket") | 
					
						
							|  |  |  |     assert len(result["QueueConfigurations"]) == 1 | 
					
						
							|  |  |  |     assert not result.get("TopicConfigurations") | 
					
						
							|  |  |  |     assert not result.get("LambdaFunctionConfigurations") | 
					
						
							|  |  |  |     assert result["QueueConfigurations"][0]["Id"] == "SomeID" | 
					
						
							|  |  |  |     assert result["QueueConfigurations"][0]["QueueArn"] == "arn:aws:sqs:us-east-1:012345678910:myQueue" | 
					
						
							|  |  |  |     assert result["QueueConfigurations"][0]["Events"][0] == "s3:ObjectCreated:*" | 
					
						
							|  |  |  |     assert len(result["QueueConfigurations"][0]["Events"]) == 1 | 
					
						
							|  |  |  |     assert len(result["QueueConfigurations"][0]["Filter"]["Key"]["FilterRules"]) == 1 | 
					
						
							|  |  |  |     assert result["QueueConfigurations"][0]["Filter"]["Key"]["FilterRules"][0]["Name"] == "prefix" | 
					
						
							|  |  |  |     assert result["QueueConfigurations"][0]["Filter"]["Key"]["FilterRules"][0]["Value"] == "images/" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Place proper Lambda configuration: | 
					
						
							|  |  |  |     s3.put_bucket_notification_configuration(Bucket="bucket", | 
					
						
							|  |  |  |                                              NotificationConfiguration={ | 
					
						
							|  |  |  |                                                  "LambdaFunctionConfigurations": [ | 
					
						
							|  |  |  |                                                      { | 
					
						
							|  |  |  |                                                          "LambdaFunctionArn": | 
					
						
							|  |  |  |                                                              "arn:aws:lambda:us-east-1:012345678910:function:lambda", | 
					
						
							|  |  |  |                                                          "Events": ["s3:ObjectCreated:*"], | 
					
						
							|  |  |  |                                                          "Filter": { | 
					
						
							|  |  |  |                                                              "Key": { | 
					
						
							|  |  |  |                                                                  "FilterRules": [ | 
					
						
							|  |  |  |                                                                      { | 
					
						
							|  |  |  |                                                                          "Name": "prefix", | 
					
						
							|  |  |  |                                                                          "Value": "images/" | 
					
						
							|  |  |  |                                                                      } | 
					
						
							|  |  |  |                                                                  ] | 
					
						
							|  |  |  |                                                              } | 
					
						
							|  |  |  |                                                          } | 
					
						
							|  |  |  |                                                      } | 
					
						
							|  |  |  |                                                  ] | 
					
						
							|  |  |  |                                              }) | 
					
						
							|  |  |  |     result = s3.get_bucket_notification_configuration(Bucket="bucket") | 
					
						
							|  |  |  |     assert len(result["LambdaFunctionConfigurations"]) == 1 | 
					
						
							|  |  |  |     assert not result.get("TopicConfigurations") | 
					
						
							|  |  |  |     assert not result.get("QueueConfigurations") | 
					
						
							|  |  |  |     assert result["LambdaFunctionConfigurations"][0]["Id"] | 
					
						
							|  |  |  |     assert result["LambdaFunctionConfigurations"][0]["LambdaFunctionArn"] == \ | 
					
						
							|  |  |  |         "arn:aws:lambda:us-east-1:012345678910:function:lambda" | 
					
						
							|  |  |  |     assert result["LambdaFunctionConfigurations"][0]["Events"][0] == "s3:ObjectCreated:*" | 
					
						
							|  |  |  |     assert len(result["LambdaFunctionConfigurations"][0]["Events"]) == 1 | 
					
						
							|  |  |  |     assert len(result["LambdaFunctionConfigurations"][0]["Filter"]["Key"]["FilterRules"]) == 1 | 
					
						
							|  |  |  |     assert result["LambdaFunctionConfigurations"][0]["Filter"]["Key"]["FilterRules"][0]["Name"] == "prefix" | 
					
						
							|  |  |  |     assert result["LambdaFunctionConfigurations"][0]["Filter"]["Key"]["FilterRules"][0]["Value"] == "images/" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # And with all 3 set: | 
					
						
							|  |  |  |     s3.put_bucket_notification_configuration(Bucket="bucket", | 
					
						
							|  |  |  |                                              NotificationConfiguration={ | 
					
						
							|  |  |  |                                                  "TopicConfigurations": [ | 
					
						
							|  |  |  |                                                      { | 
					
						
							|  |  |  |                                                          "TopicArn": "arn:aws:sns:us-east-1:012345678910:mytopic", | 
					
						
							|  |  |  |                                                          "Events": [ | 
					
						
							|  |  |  |                                                              "s3:ObjectCreated:*", | 
					
						
							|  |  |  |                                                              "s3:ObjectRemoved:*" | 
					
						
							|  |  |  |                                                          ] | 
					
						
							|  |  |  |                                                      } | 
					
						
							|  |  |  |                                                  ], | 
					
						
							|  |  |  |                                                  "LambdaFunctionConfigurations": [ | 
					
						
							|  |  |  |                                                      { | 
					
						
							|  |  |  |                                                          "LambdaFunctionArn": | 
					
						
							|  |  |  |                                                              "arn:aws:lambda:us-east-1:012345678910:function:lambda", | 
					
						
							|  |  |  |                                                          "Events": ["s3:ObjectCreated:*"] | 
					
						
							|  |  |  |                                                      } | 
					
						
							|  |  |  |                                                  ], | 
					
						
							|  |  |  |                                                  "QueueConfigurations": [ | 
					
						
							|  |  |  |                                                      { | 
					
						
							|  |  |  |                                                          "QueueArn": "arn:aws:sqs:us-east-1:012345678910:myQueue", | 
					
						
							|  |  |  |                                                          "Events": ["s3:ObjectCreated:*"] | 
					
						
							|  |  |  |                                                      } | 
					
						
							|  |  |  |                                                  ] | 
					
						
							|  |  |  |                                              }) | 
					
						
							|  |  |  |     result = s3.get_bucket_notification_configuration(Bucket="bucket") | 
					
						
							|  |  |  |     assert len(result["LambdaFunctionConfigurations"]) == 1 | 
					
						
							|  |  |  |     assert len(result["TopicConfigurations"]) == 1 | 
					
						
							|  |  |  |     assert len(result["QueueConfigurations"]) == 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # And clear it out: | 
					
						
							|  |  |  |     s3.put_bucket_notification_configuration(Bucket="bucket", NotificationConfiguration={}) | 
					
						
							|  |  |  |     result = s3.get_bucket_notification_configuration(Bucket="bucket") | 
					
						
							|  |  |  |     assert not result.get("TopicConfigurations") | 
					
						
							|  |  |  |     assert not result.get("QueueConfigurations") | 
					
						
							|  |  |  |     assert not result.get("LambdaFunctionConfigurations") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_put_bucket_notification_errors(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     s3.create_bucket(Bucket="bucket") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # With incorrect ARNs: | 
					
						
							|  |  |  |     for tech, arn in [("Queue", "sqs"), ("Topic", "sns"), ("LambdaFunction", "lambda")]: | 
					
						
							|  |  |  |         with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |             s3.put_bucket_notification_configuration(Bucket="bucket", | 
					
						
							|  |  |  |                                                      NotificationConfiguration={ | 
					
						
							|  |  |  |                                                          "{}Configurations".format(tech): [ | 
					
						
							|  |  |  |                                                              { | 
					
						
							|  |  |  |                                                                  "{}Arn".format(tech): | 
					
						
							|  |  |  |                                                                      "arn:aws:{}:us-east-1:012345678910:lksajdfkldskfj", | 
					
						
							|  |  |  |                                                                  "Events": ["s3:ObjectCreated:*"] | 
					
						
							|  |  |  |                                                              } | 
					
						
							|  |  |  |                                                          ] | 
					
						
							|  |  |  |                                                      }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         assert err.exception.response["Error"]["Code"] == "InvalidArgument" | 
					
						
							|  |  |  |         assert err.exception.response["Error"]["Message"] == "The ARN is not well formed" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Region not the same as the bucket: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_notification_configuration(Bucket="bucket", | 
					
						
							|  |  |  |                                                  NotificationConfiguration={ | 
					
						
							|  |  |  |                                                      "QueueConfigurations": [ | 
					
						
							|  |  |  |                                                          { | 
					
						
							|  |  |  |                                                              "QueueArn": | 
					
						
							|  |  |  |                                                                  "arn:aws:sqs:us-west-2:012345678910:lksajdfkldskfj", | 
					
						
							|  |  |  |                                                              "Events": ["s3:ObjectCreated:*"] | 
					
						
							|  |  |  |                                                          } | 
					
						
							|  |  |  |                                                      ] | 
					
						
							|  |  |  |                                                  }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Code"] == "InvalidArgument" | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Message"] == \ | 
					
						
							|  |  |  |         "The notification destination service region is not valid for the bucket location constraint" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Invalid event name: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_notification_configuration(Bucket="bucket", | 
					
						
							|  |  |  |                                                  NotificationConfiguration={ | 
					
						
							|  |  |  |                                                      "QueueConfigurations": [ | 
					
						
							|  |  |  |                                                          { | 
					
						
							|  |  |  |                                                              "QueueArn": | 
					
						
							|  |  |  |                                                                  "arn:aws:sqs:us-east-1:012345678910:lksajdfkldskfj", | 
					
						
							|  |  |  |                                                              "Events": ["notarealeventname"] | 
					
						
							|  |  |  |                                                          } | 
					
						
							|  |  |  |                                                      ] | 
					
						
							|  |  |  |                                                  }) | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Code"] == "InvalidArgument" | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Message"] == "The event is not supported for notifications" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_put_bucket_logging(): | 
					
						
							|  |  |  |     s3 = boto3.client("s3", region_name="us-east-1") | 
					
						
							|  |  |  |     bucket_name = "mybucket" | 
					
						
							|  |  |  |     log_bucket = "logbucket" | 
					
						
							|  |  |  |     wrong_region_bucket = "wrongregionlogbucket" | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=log_bucket)  # Adding the ACL for log-delivery later... | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=wrong_region_bucket, CreateBucketConfiguration={"LocationConstraint": "us-west-2"}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # No logging config: | 
					
						
							|  |  |  |     result = s3.get_bucket_logging(Bucket=bucket_name) | 
					
						
							|  |  |  |     assert not result.get("LoggingEnabled") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # A log-bucket that doesn't exist: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_logging(Bucket=bucket_name, BucketLoggingStatus={ | 
					
						
							|  |  |  |             "LoggingEnabled": { | 
					
						
							|  |  |  |                 "TargetBucket": "IAMNOTREAL", | 
					
						
							|  |  |  |                 "TargetPrefix": "" | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Code"] == "InvalidTargetBucketForLogging" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # A log-bucket that's missing the proper ACLs for LogDelivery: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_logging(Bucket=bucket_name, BucketLoggingStatus={ | 
					
						
							|  |  |  |             "LoggingEnabled": { | 
					
						
							|  |  |  |                 "TargetBucket": log_bucket, | 
					
						
							|  |  |  |                 "TargetPrefix": "" | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Code"] == "InvalidTargetBucketForLogging" | 
					
						
							|  |  |  |     assert "log-delivery" in err.exception.response["Error"]["Message"] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Add the proper "log-delivery" ACL to the log buckets: | 
					
						
							|  |  |  |     bucket_owner = s3.get_bucket_acl(Bucket=log_bucket)["Owner"] | 
					
						
							|  |  |  |     for bucket in [log_bucket, wrong_region_bucket]: | 
					
						
							|  |  |  |         s3.put_bucket_acl(Bucket=bucket, AccessControlPolicy={ | 
					
						
							|  |  |  |             "Grants": [ | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Grantee": { | 
					
						
							|  |  |  |                         "URI": "http://acs.amazonaws.com/groups/s3/LogDelivery", | 
					
						
							|  |  |  |                         "Type": "Group" | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                     "Permission": "WRITE" | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Grantee": { | 
					
						
							|  |  |  |                         "URI": "http://acs.amazonaws.com/groups/s3/LogDelivery", | 
					
						
							|  |  |  |                         "Type": "Group" | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                     "Permission": "READ_ACP" | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Grantee": { | 
					
						
							|  |  |  |                         "Type": "CanonicalUser", | 
					
						
							|  |  |  |                         "ID": bucket_owner["ID"] | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                     "Permission": "FULL_CONTROL" | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ], | 
					
						
							|  |  |  |             "Owner": bucket_owner | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # A log-bucket that's in the wrong region: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_logging(Bucket=bucket_name, BucketLoggingStatus={ | 
					
						
							|  |  |  |             "LoggingEnabled": { | 
					
						
							|  |  |  |                 "TargetBucket": wrong_region_bucket, | 
					
						
							|  |  |  |                 "TargetPrefix": "" | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Code"] == "CrossLocationLoggingProhibitted" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Correct logging: | 
					
						
							|  |  |  |     s3.put_bucket_logging(Bucket=bucket_name, BucketLoggingStatus={ | 
					
						
							|  |  |  |         "LoggingEnabled": { | 
					
						
							|  |  |  |             "TargetBucket": log_bucket, | 
					
						
							|  |  |  |             "TargetPrefix": "{}/".format(bucket_name) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     result = s3.get_bucket_logging(Bucket=bucket_name) | 
					
						
							|  |  |  |     assert result["LoggingEnabled"]["TargetBucket"] == log_bucket | 
					
						
							|  |  |  |     assert result["LoggingEnabled"]["TargetPrefix"] == "{}/".format(bucket_name) | 
					
						
							|  |  |  |     assert not result["LoggingEnabled"].get("TargetGrants") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # And disabling: | 
					
						
							|  |  |  |     s3.put_bucket_logging(Bucket=bucket_name, BucketLoggingStatus={}) | 
					
						
							|  |  |  |     assert not s3.get_bucket_logging(Bucket=bucket_name).get("LoggingEnabled") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # And enabling with multiple target grants: | 
					
						
							|  |  |  |     s3.put_bucket_logging(Bucket=bucket_name, BucketLoggingStatus={ | 
					
						
							|  |  |  |         "LoggingEnabled": { | 
					
						
							|  |  |  |             "TargetBucket": log_bucket, | 
					
						
							|  |  |  |             "TargetPrefix": "{}/".format(bucket_name), | 
					
						
							|  |  |  |             "TargetGrants": [ | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Grantee": { | 
					
						
							|  |  |  |                         "ID": "SOMEIDSTRINGHERE9238748923734823917498237489237409123840983274", | 
					
						
							|  |  |  |                         "Type": "CanonicalUser" | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                     "Permission": "READ" | 
					
						
							|  |  |  |                 }, | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Grantee": { | 
					
						
							|  |  |  |                         "ID": "SOMEIDSTRINGHERE9238748923734823917498237489237409123840983274", | 
					
						
							|  |  |  |                         "Type": "CanonicalUser" | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                     "Permission": "WRITE" | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result = s3.get_bucket_logging(Bucket=bucket_name) | 
					
						
							|  |  |  |     assert len(result["LoggingEnabled"]["TargetGrants"]) == 2 | 
					
						
							|  |  |  |     assert result["LoggingEnabled"]["TargetGrants"][0]["Grantee"]["ID"] == \ | 
					
						
							| 
									
										
										
										
											2018-03-21 09:11:24 -07:00
										 |  |  |            "SOMEIDSTRINGHERE9238748923734823917498237489237409123840983274" | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Test with just 1 grant: | 
					
						
							|  |  |  |     s3.put_bucket_logging(Bucket=bucket_name, BucketLoggingStatus={ | 
					
						
							|  |  |  |         "LoggingEnabled": { | 
					
						
							|  |  |  |             "TargetBucket": log_bucket, | 
					
						
							|  |  |  |             "TargetPrefix": "{}/".format(bucket_name), | 
					
						
							|  |  |  |             "TargetGrants": [ | 
					
						
							|  |  |  |                 { | 
					
						
							|  |  |  |                     "Grantee": { | 
					
						
							|  |  |  |                         "ID": "SOMEIDSTRINGHERE9238748923734823917498237489237409123840983274", | 
					
						
							|  |  |  |                         "Type": "CanonicalUser" | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                     "Permission": "READ" | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             ] | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     result = s3.get_bucket_logging(Bucket=bucket_name) | 
					
						
							|  |  |  |     assert len(result["LoggingEnabled"]["TargetGrants"]) == 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # With an invalid grant: | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_bucket_logging(Bucket=bucket_name, BucketLoggingStatus={ | 
					
						
							|  |  |  |             "LoggingEnabled": { | 
					
						
							|  |  |  |                 "TargetBucket": log_bucket, | 
					
						
							|  |  |  |                 "TargetPrefix": "{}/".format(bucket_name), | 
					
						
							|  |  |  |                 "TargetGrants": [ | 
					
						
							|  |  |  |                     { | 
					
						
							|  |  |  |                         "Grantee": { | 
					
						
							|  |  |  |                             "ID": "SOMEIDSTRINGHERE9238748923734823917498237489237409123840983274", | 
					
						
							|  |  |  |                             "Type": "CanonicalUser" | 
					
						
							|  |  |  |                         }, | 
					
						
							|  |  |  |                         "Permission": "NOTAREALPERM" | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 ] | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     assert err.exception.response["Error"]["Code"] == "MalformedXML" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-15 22:36:12 -04:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_put_object_tagging(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     key = 'key-with-tags' | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(ClientError) as err: | 
					
						
							|  |  |  |         s3.put_object_tagging( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             Key=key, | 
					
						
							|  |  |  |             Tagging={'TagSet': [ | 
					
						
							|  |  |  |                 {'Key': 'item1', 'Value': 'foo'}, | 
					
						
							|  |  |  |                 {'Key': 'item2', 'Value': 'bar'}, | 
					
						
							|  |  |  |             ]} | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     e = err.exception | 
					
						
							|  |  |  |     e.response['Error'].should.equal({ | 
					
						
							|  |  |  |         'Code': 'NoSuchKey', | 
					
						
							|  |  |  |         'Message': 'The specified key does not exist.', | 
					
						
							|  |  |  |         'RequestID': '7a62c49f-347e-4fc4-9331-6e8eEXAMPLE', | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.put_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Body='test' | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.put_object_tagging( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Tagging={'TagSet': [ | 
					
						
							|  |  |  |             {'Key': 'item1', 'Value': 'foo'}, | 
					
						
							|  |  |  |             {'Key': 'item2', 'Value': 'bar'}, | 
					
						
							|  |  |  |         ]} | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(200) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-02 23:03:54 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_put_object_tagging_with_single_tag(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     key = 'key-with-tags' | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.put_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Body='test' | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.put_object_tagging( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Tagging={'TagSet': [ | 
					
						
							|  |  |  |             {'Key': 'item1', 'Value': 'foo'} | 
					
						
							|  |  |  |         ]} | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp['ResponseMetadata']['HTTPStatusCode'].should.equal(200) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-15 22:36:12 -04:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_get_object_tagging(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     key = 'key-with-tags' | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3.put_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Body='test' | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.get_object_tagging(Bucket=bucket_name, Key=key) | 
					
						
							|  |  |  |     resp['TagSet'].should.have.length_of(0) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp = s3.put_object_tagging( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Tagging={'TagSet': [ | 
					
						
							|  |  |  |             {'Key': 'item1', 'Value': 'foo'}, | 
					
						
							|  |  |  |             {'Key': 'item2', 'Value': 'bar'}, | 
					
						
							|  |  |  |         ]} | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     resp = s3.get_object_tagging(Bucket=bucket_name, Key=key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resp['TagSet'].should.have.length_of(2) | 
					
						
							|  |  |  |     resp['TagSet'].should.contain({'Key': 'item1', 'Value': 'foo'}) | 
					
						
							|  |  |  |     resp['TagSet'].should.contain({'Key': 'item2', 'Value': 'bar'}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-29 21:56:48 +03:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_list_object_versions(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     key = 'key-with-versions' | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							| 
									
										
										
										
											2017-05-14 20:00:26 +03:00
										 |  |  |     s3.put_bucket_versioning( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         VersioningConfiguration={ | 
					
						
							|  |  |  |             'Status': 'Enabled' | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2017-04-30 08:03:46 +03:00
										 |  |  |     items = (six.b('v1'), six.b('v2')) | 
					
						
							|  |  |  |     for body in items: | 
					
						
							| 
									
										
										
										
											2017-04-29 21:56:48 +03:00
										 |  |  |         s3.put_object( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             Key=key, | 
					
						
							|  |  |  |             Body=body | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     response = s3.list_object_versions( | 
					
						
							|  |  |  |         Bucket=bucket_name | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     # Two object versions should be returned | 
					
						
							|  |  |  |     len(response['Versions']).should.equal(2) | 
					
						
							|  |  |  |     keys = set([item['Key'] for item in response['Versions']]) | 
					
						
							|  |  |  |     keys.should.equal({key}) | 
					
						
							|  |  |  |     # Test latest object version is returned | 
					
						
							|  |  |  |     response = s3.get_object(Bucket=bucket_name, Key=key) | 
					
						
							| 
									
										
										
										
											2017-04-30 08:03:46 +03:00
										 |  |  |     response['Body'].read().should.equal(items[-1]) | 
					
						
							| 
									
										
										
										
											2017-04-29 21:56:48 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_list_object_versions_with_versioning_disabled(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     key = 'key-with-versions' | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     items = (six.b('v1'), six.b('v2')) | 
					
						
							|  |  |  |     for body in items: | 
					
						
							|  |  |  |         s3.put_object( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             Key=key, | 
					
						
							|  |  |  |             Body=body | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     response = s3.list_object_versions( | 
					
						
							|  |  |  |         Bucket=bucket_name | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # One object version should be returned | 
					
						
							|  |  |  |     len(response['Versions']).should.equal(1) | 
					
						
							|  |  |  |     response['Versions'][0]['Key'].should.equal(key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # The version id should be the string null | 
					
						
							|  |  |  |     response['Versions'][0]['VersionId'].should.equal('null') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Test latest object version is returned | 
					
						
							|  |  |  |     response = s3.get_object(Bucket=bucket_name, Key=key) | 
					
						
							|  |  |  |     response['Body'].read().should.equal(items[-1]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_list_object_versions_with_versioning_enabled_late(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     key = 'key-with-versions' | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     items = (six.b('v1'), six.b('v2')) | 
					
						
							|  |  |  |     s3.put_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Body=six.b('v1') | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     s3.put_bucket_versioning( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         VersioningConfiguration={ | 
					
						
							|  |  |  |             'Status': 'Enabled' | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     s3.put_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							|  |  |  |         Body=six.b('v2') | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     response = s3.list_object_versions( | 
					
						
							|  |  |  |         Bucket=bucket_name | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Two object versions should be returned | 
					
						
							|  |  |  |     len(response['Versions']).should.equal(2) | 
					
						
							|  |  |  |     keys = set([item['Key'] for item in response['Versions']]) | 
					
						
							|  |  |  |     keys.should.equal({key}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # There should still be a null version id. | 
					
						
							|  |  |  |     versionsId = set([item['VersionId'] for item in response['Versions']]) | 
					
						
							|  |  |  |     versionsId.should.contain('null') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Test latest object version is returned | 
					
						
							|  |  |  |     response = s3.get_object(Bucket=bucket_name, Key=key) | 
					
						
							|  |  |  |     response['Body'].read().should.equal(items[-1]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-22 11:59:01 -07:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_bad_prefix_list_object_versions(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     key = 'key-with-versions' | 
					
						
							|  |  |  |     bad_prefix = 'key-that-does-not-exist' | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     s3.put_bucket_versioning( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         VersioningConfiguration={ | 
					
						
							|  |  |  |             'Status': 'Enabled' | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     items = (six.b('v1'), six.b('v2')) | 
					
						
							|  |  |  |     for body in items: | 
					
						
							|  |  |  |         s3.put_object( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             Key=key, | 
					
						
							|  |  |  |             Body=body | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     response = s3.list_object_versions( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Prefix=bad_prefix, | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     response['ResponseMetadata']['HTTPStatusCode'].should.equal(200) | 
					
						
							|  |  |  |     response.should_not.contain('Versions') | 
					
						
							|  |  |  |     response.should_not.contain('DeleteMarkers') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 20:00:26 +03:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_delete_markers(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							| 
									
										
										
										
											2017-11-06 21:39:08 +00:00
										 |  |  |     key = u'key-with-versions-and-unicode-ó' | 
					
						
							| 
									
										
										
										
											2017-05-14 20:00:26 +03:00
										 |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     s3.put_bucket_versioning( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         VersioningConfiguration={ | 
					
						
							|  |  |  |             'Status': 'Enabled' | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     items = (six.b('v1'), six.b('v2')) | 
					
						
							|  |  |  |     for body in items: | 
					
						
							|  |  |  |         s3.put_object( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             Key=key, | 
					
						
							|  |  |  |             Body=body | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2017-11-06 21:39:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     s3.delete_objects(Bucket=bucket_name, Delete={'Objects': [{'Key': key}]}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 20:00:26 +03:00
										 |  |  |     with assert_raises(ClientError) as e: | 
					
						
							|  |  |  |         s3.get_object( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             Key=key | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     e.exception.response['Error']['Code'].should.equal('NoSuchKey') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     response = s3.list_object_versions( | 
					
						
							|  |  |  |         Bucket=bucket_name | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     response['Versions'].should.have.length_of(2) | 
					
						
							|  |  |  |     response['DeleteMarkers'].should.have.length_of(1) | 
					
						
							| 
									
										
										
										
											2017-05-14 20:00:26 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     s3.delete_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |         VersionId=response['DeleteMarkers'][0]['VersionId'] | 
					
						
							| 
									
										
										
										
											2017-05-14 20:00:26 +03:00
										 |  |  |     ) | 
					
						
							|  |  |  |     response = s3.get_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     response['Body'].read().should.equal(items[-1]) | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-14 20:00:26 +03:00
										 |  |  |     response = s3.list_object_versions( | 
					
						
							|  |  |  |         Bucket=bucket_name | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     response['Versions'].should.have.length_of(2) | 
					
						
							| 
									
										
										
										
											2017-11-06 21:39:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # We've asserted there is only 2 records so one is newest, one is oldest | 
					
						
							|  |  |  |     latest = list(filter(lambda item: item['IsLatest'], response['Versions']))[0] | 
					
						
							|  |  |  |     oldest = list(filter(lambda item: not item['IsLatest'], response['Versions']))[0] | 
					
						
							|  |  |  |     # Double check ordering of version ID's | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     latest['VersionId'].should_not.equal(oldest['VersionId']) | 
					
						
							| 
									
										
										
										
											2017-11-06 21:39:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Double check the name is still unicode | 
					
						
							|  |  |  |     latest['Key'].should.equal('key-with-versions-and-unicode-ó') | 
					
						
							|  |  |  |     oldest['Key'].should.equal('key-with-versions-and-unicode-ó') | 
					
						
							| 
									
										
										
										
											2017-05-14 20:00:26 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-05 15:12:45 -05:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_multiple_delete_markers(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     bucket_name = 'mybucket' | 
					
						
							|  |  |  |     key = u'key-with-versions-and-unicode-ó' | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     s3.put_bucket_versioning( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         VersioningConfiguration={ | 
					
						
							|  |  |  |             'Status': 'Enabled' | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     items = (six.b('v1'), six.b('v2')) | 
					
						
							|  |  |  |     for body in items: | 
					
						
							|  |  |  |         s3.put_object( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             Key=key, | 
					
						
							|  |  |  |             Body=body | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Delete the object twice to add multiple delete markers | 
					
						
							|  |  |  |     s3.delete_object(Bucket=bucket_name, Key=key) | 
					
						
							|  |  |  |     s3.delete_object(Bucket=bucket_name, Key=key) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     response = s3.list_object_versions(Bucket=bucket_name) | 
					
						
							|  |  |  |     response['DeleteMarkers'].should.have.length_of(2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with assert_raises(ClientError) as e: | 
					
						
							|  |  |  |         s3.get_object( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             Key=key | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         e.response['Error']['Code'].should.equal('404') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Remove both delete markers to restore the object | 
					
						
							|  |  |  |     s3.delete_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |         VersionId=response['DeleteMarkers'][0]['VersionId'] | 
					
						
							| 
									
										
										
										
											2018-01-05 15:12:45 -05:00
										 |  |  |     ) | 
					
						
							|  |  |  |     s3.delete_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key, | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |         VersionId=response['DeleteMarkers'][1]['VersionId'] | 
					
						
							| 
									
										
										
										
											2018-01-05 15:12:45 -05:00
										 |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     response = s3.get_object( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         Key=key | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     response['Body'].read().should.equal(items[-1]) | 
					
						
							|  |  |  |     response = s3.list_object_versions(Bucket=bucket_name) | 
					
						
							|  |  |  |     response['Versions'].should.have.length_of(2) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # We've asserted there is only 2 records so one is newest, one is oldest | 
					
						
							|  |  |  |     latest = list(filter(lambda item: item['IsLatest'], response['Versions']))[0] | 
					
						
							|  |  |  |     oldest = list(filter(lambda item: not item['IsLatest'], response['Versions']))[0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Double check ordering of version ID's | 
					
						
							| 
									
										
										
										
											2019-03-08 22:01:27 +01:00
										 |  |  |     latest['VersionId'].should_not.equal(oldest['VersionId']) | 
					
						
							| 
									
										
										
										
											2018-01-05 15:12:45 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Double check the name is still unicode | 
					
						
							|  |  |  |     latest['Key'].should.equal('key-with-versions-and-unicode-ó') | 
					
						
							|  |  |  |     oldest['Key'].should.equal('key-with-versions-and-unicode-ó') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-25 12:19:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-03 19:06:49 -04:00
										 |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_get_stream_gzipped(): | 
					
						
							| 
									
										
										
										
											2017-06-03 19:35:23 -04:00
										 |  |  |     payload = b"this is some stuff here" | 
					
						
							| 
									
										
										
										
											2017-06-03 19:06:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     s3_client = boto3.client("s3", region_name='us-east-1') | 
					
						
							|  |  |  |     s3_client.create_bucket(Bucket='moto-tests') | 
					
						
							|  |  |  |     buffer_ = BytesIO() | 
					
						
							|  |  |  |     with GzipFile(fileobj=buffer_, mode='w') as f: | 
					
						
							|  |  |  |         f.write(payload) | 
					
						
							|  |  |  |     payload_gz = buffer_.getvalue() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     s3_client.put_object( | 
					
						
							|  |  |  |         Bucket='moto-tests', | 
					
						
							|  |  |  |         Key='keyname', | 
					
						
							|  |  |  |         Body=payload_gz, | 
					
						
							|  |  |  |         ContentEncoding='gzip', | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     obj = s3_client.get_object( | 
					
						
							|  |  |  |         Bucket='moto-tests', | 
					
						
							|  |  |  |         Key='keyname', | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2018-01-02 23:47:57 -05:00
										 |  |  |     res = zlib.decompress(obj['Body'].read(), 16 + zlib.MAX_WBITS) | 
					
						
							| 
									
										
										
										
											2017-06-03 19:06:49 -04:00
										 |  |  |     assert res == payload | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-04 18:55:41 -05:00
										 |  |  | TEST_XML = """\
 | 
					
						
							|  |  |  | <?xml version="1.0" encoding="UTF-8"?> | 
					
						
							|  |  |  | <ns0:WebsiteConfiguration xmlns:ns0="http://s3.amazonaws.com/doc/2006-03-01/"> | 
					
						
							|  |  |  |     <ns0:IndexDocument> | 
					
						
							|  |  |  |         <ns0:Suffix>index.html</ns0:Suffix> | 
					
						
							|  |  |  |     </ns0:IndexDocument> | 
					
						
							|  |  |  |     <ns0:RoutingRules> | 
					
						
							|  |  |  |         <ns0:RoutingRule> | 
					
						
							|  |  |  |             <ns0:Condition> | 
					
						
							|  |  |  |                 <ns0:KeyPrefixEquals>test/testing</ns0:KeyPrefixEquals> | 
					
						
							|  |  |  |             </ns0:Condition> | 
					
						
							|  |  |  |             <ns0:Redirect> | 
					
						
							|  |  |  |                 <ns0:ReplaceKeyWith>test.txt</ns0:ReplaceKeyWith> | 
					
						
							|  |  |  |             </ns0:Redirect> | 
					
						
							|  |  |  |         </ns0:RoutingRule> | 
					
						
							|  |  |  |     </ns0:RoutingRules> | 
					
						
							|  |  |  | </ns0:WebsiteConfiguration> | 
					
						
							|  |  |  | """
 | 
					
						
							| 
									
										
										
										
											2018-11-22 06:08:03 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_bucket_name_too_long(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     with assert_raises(ClientError) as exc: | 
					
						
							|  |  |  |         s3.create_bucket(Bucket='x'*64) | 
					
						
							|  |  |  |     exc.exception.response['Error']['Code'].should.equal('InvalidBucketName') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_boto3_bucket_name_too_short(): | 
					
						
							|  |  |  |     s3 = boto3.client('s3', region_name='us-east-1') | 
					
						
							|  |  |  |     with assert_raises(ClientError) as exc: | 
					
						
							|  |  |  |         s3.create_bucket(Bucket='x'*2) | 
					
						
							|  |  |  |     exc.exception.response['Error']['Code'].should.equal('InvalidBucketName') | 
					
						
							| 
									
										
										
										
											2019-05-25 12:19:00 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_accelerated_none_when_unspecified(): | 
					
						
							|  |  |  |     bucket_name = 'some_bucket' | 
					
						
							|  |  |  |     s3 = boto3.client('s3') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp = s3.get_bucket_accelerate_configuration(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp.shouldnt.have.key('Status') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_can_enable_bucket_acceleration(): | 
					
						
							|  |  |  |     bucket_name = 'some_bucket' | 
					
						
							|  |  |  |     s3 = boto3.client('s3') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp = s3.put_bucket_accelerate_configuration( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         AccelerateConfiguration={'Status': 'Enabled'}, | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     resp.keys().should.have.length_of(1)    # Response contains nothing (only HTTP headers) | 
					
						
							|  |  |  |     resp = s3.get_bucket_accelerate_configuration(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp.should.have.key('Status') | 
					
						
							|  |  |  |     resp['Status'].should.equal('Enabled') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_can_suspend_bucket_acceleration(): | 
					
						
							|  |  |  |     bucket_name = 'some_bucket' | 
					
						
							|  |  |  |     s3 = boto3.client('s3') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp = s3.put_bucket_accelerate_configuration( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         AccelerateConfiguration={'Status': 'Enabled'}, | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     resp = s3.put_bucket_accelerate_configuration( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         AccelerateConfiguration={'Status': 'Suspended'}, | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     resp.keys().should.have.length_of(1)    # Response contains nothing (only HTTP headers) | 
					
						
							|  |  |  |     resp = s3.get_bucket_accelerate_configuration(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp.should.have.key('Status') | 
					
						
							|  |  |  |     resp['Status'].should.equal('Suspended') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_suspending_acceleration_on_not_configured_bucket_does_nothing(): | 
					
						
							|  |  |  |     bucket_name = 'some_bucket' | 
					
						
							|  |  |  |     s3 = boto3.client('s3') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp = s3.put_bucket_accelerate_configuration( | 
					
						
							|  |  |  |         Bucket=bucket_name, | 
					
						
							|  |  |  |         AccelerateConfiguration={'Status': 'Suspended'}, | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     resp.keys().should.have.length_of(1)    # Response contains nothing (only HTTP headers) | 
					
						
							|  |  |  |     resp = s3.get_bucket_accelerate_configuration(Bucket=bucket_name) | 
					
						
							|  |  |  |     resp.shouldnt.have.key('Status') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_accelerate_configuration_status_validation(): | 
					
						
							|  |  |  |     bucket_name = 'some_bucket' | 
					
						
							|  |  |  |     s3 = boto3.client('s3') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     with assert_raises(ClientError) as exc: | 
					
						
							|  |  |  |         s3.put_bucket_accelerate_configuration( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             AccelerateConfiguration={'Status': 'bad_status'}, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     exc.exception.response['Error']['Code'].should.equal('MalformedXML') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @mock_s3 | 
					
						
							|  |  |  | def test_accelerate_configuration_is_not_supported_when_bucket_name_has_dots(): | 
					
						
							|  |  |  |     bucket_name = 'some.bucket.with.dots' | 
					
						
							|  |  |  |     s3 = boto3.client('s3') | 
					
						
							|  |  |  |     s3.create_bucket(Bucket=bucket_name) | 
					
						
							|  |  |  |     with assert_raises(ClientError) as exc: | 
					
						
							|  |  |  |         s3.put_bucket_accelerate_configuration( | 
					
						
							|  |  |  |             Bucket=bucket_name, | 
					
						
							|  |  |  |             AccelerateConfiguration={'Status': 'Enabled'}, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |     exc.exception.response['Error']['Code'].should.equal('InvalidRequest') |