Create segment container w/ same policy as primary

When users upload an MPU object, s3api will automatically
create a segment container if one doesn't already exist.

Currently, s3api will create the segment bucket using the
cluster's default storage policy. This patch changes that
behavior to use the same storage policy as the primary bucket.

Change-Id: Ib64a06868bd3670a1d4a1860ac29122e1ede7c39
Closes-Bug: 1832390
This commit is contained in:
Thiago da Silva 2019-10-09 16:24:12 +02:00
parent 1c0661e920
commit d667affb6f
2 changed files with 51 additions and 9 deletions

View File

@ -60,6 +60,7 @@ Static Large Object when the multipart upload is completed.
""" """
import binascii import binascii
import copy
from hashlib import md5 from hashlib import md5
import os import os
import re import re
@ -86,6 +87,7 @@ from swift.common.middleware.s3api.utils import unique_id, \
MULTIUPLOAD_SUFFIX, S3Timestamp, sysmeta_header MULTIUPLOAD_SUFFIX, S3Timestamp, sysmeta_header
from swift.common.middleware.s3api.etree import Element, SubElement, \ from swift.common.middleware.s3api.etree import Element, SubElement, \
fromstring, tostring, XMLSyntaxError, DocumentInvalid fromstring, tostring, XMLSyntaxError, DocumentInvalid
from swift.common.storage_policy import POLICIES
DEFAULT_MAX_PARTS_LISTING = 1000 DEFAULT_MAX_PARTS_LISTING = 1000
DEFAULT_MAX_UPLOADS = 1000 DEFAULT_MAX_UPLOADS = 1000
@ -364,8 +366,7 @@ class UploadsController(Controller):
# Create a unique S3 upload id from UUID to avoid duplicates. # Create a unique S3 upload id from UUID to avoid duplicates.
upload_id = unique_id() upload_id = unique_id()
orig_container = req.container_name seg_container = req.container_name + MULTIUPLOAD_SUFFIX
seg_container = orig_container + MULTIUPLOAD_SUFFIX
content_type = req.headers.get('Content-Type') content_type = req.headers.get('Content-Type')
if content_type: if content_type:
req.headers[sysmeta_header('object', 'has-content-type')] = 'yes' req.headers[sysmeta_header('object', 'has-content-type')] = 'yes'
@ -376,15 +377,21 @@ class UploadsController(Controller):
req.headers['Content-Type'] = 'application/directory' req.headers['Content-Type'] = 'application/directory'
try: try:
req.container_name = seg_container seg_req = copy.copy(req)
req.get_container_info(self.app) seg_req.environ = copy.copy(req.environ)
seg_req.container_name = seg_container
seg_req.get_container_info(self.app)
except NoSuchBucket: except NoSuchBucket:
try: try:
req.get_response(self.app, 'PUT', seg_container, '') # multi-upload bucket doesn't exist, create one with
# same storage policy as the primary bucket
info = req.get_container_info(self.app)
policy_name = POLICIES[info['storage_policy']].name
hdrs = {'X-Storage-Policy': policy_name}
seg_req.get_response(self.app, 'PUT', seg_container, '',
headers=hdrs)
except (BucketAlreadyExists, BucketAlreadyOwnedByYou): except (BucketAlreadyExists, BucketAlreadyOwnedByYou):
pass pass
finally:
req.container_name = orig_container
obj = '%s/%s' % (req.object_name, upload_id) obj = '%s/%s' % (req.object_name, upload_id)

View File

@ -26,7 +26,7 @@ from swift.common import swob
from swift.common.swob import Request from swift.common.swob import Request
from swift.common.utils import json from swift.common.utils import json
from test.unit import FakeMemcache from test.unit import FakeMemcache, patch_policies
from test.unit.common.middleware.s3api import S3ApiTestCase from test.unit.common.middleware.s3api import S3ApiTestCase
from test.unit.common.middleware.s3api.helpers import UnreadableInput from test.unit.common.middleware.s3api.helpers import UnreadableInput
from swift.common.middleware.s3api.etree import fromstring, tostring from swift.common.middleware.s3api.etree import fromstring, tostring
@ -36,6 +36,7 @@ from test.unit.common.middleware.s3api.test_s3_acl import s3acl
from swift.common.middleware.s3api.utils import sysmeta_header, mktime, \ from swift.common.middleware.s3api.utils import sysmeta_header, mktime, \
S3Timestamp S3Timestamp
from swift.common.middleware.s3api.s3request import MAX_32BIT_INT from swift.common.middleware.s3api.s3request import MAX_32BIT_INT
from swift.common.storage_policy import StoragePolicy
from swift.proxy.controllers.base import get_cache_key from swift.proxy.controllers.base import get_cache_key
XML = '<CompleteMultipartUpload>' \ XML = '<CompleteMultipartUpload>' \
@ -555,7 +556,8 @@ class TestS3ApiMultiUpload(S3ApiTestCase):
@patch('swift.common.middleware.s3api.controllers.' @patch('swift.common.middleware.s3api.controllers.'
'multi_upload.unique_id', lambda: 'X') 'multi_upload.unique_id', lambda: 'X')
def _test_object_multipart_upload_initiate(self, headers, cache=None, def _test_object_multipart_upload_initiate(self, headers, cache=None,
bucket_exists=True): bucket_exists=True,
expected_policy=None):
headers.update({ headers.update({
'Authorization': 'AWS test:tester:hmac', 'Authorization': 'AWS test:tester:hmac',
'Date': self.get_date_header(), 'Date': self.get_date_header(),
@ -584,6 +586,10 @@ class TestS3ApiMultiUpload(S3ApiTestCase):
('PUT', '/v1/AUTH_test/bucket+segments'), ('PUT', '/v1/AUTH_test/bucket+segments'),
('PUT', '/v1/AUTH_test/bucket+segments/object/X'), ('PUT', '/v1/AUTH_test/bucket+segments/object/X'),
], self.swift.calls) ], self.swift.calls)
if expected_policy:
_, _, req_headers = self.swift.calls_with_headers[-2]
self.assertEqual(req_headers.get('X-Storage-Policy'),
expected_policy)
self.swift.clear_calls() self.swift.clear_calls()
def test_object_multipart_upload_initiate_with_segment_bucket(self): def test_object_multipart_upload_initiate_with_segment_bucket(self):
@ -601,6 +607,8 @@ class TestS3ApiMultiUpload(S3ApiTestCase):
self.swift.register('PUT', '/v1/AUTH_test/bucket+segments', self.swift.register('PUT', '/v1/AUTH_test/bucket+segments',
swob.HTTPCreated, {}, None) swob.HTTPCreated, {}, None)
fake_memcache = FakeMemcache() fake_memcache = FakeMemcache()
fake_memcache.store[get_cache_key(
'AUTH_test', 'bucket')] = {'status': 204}
fake_memcache.store[get_cache_key( fake_memcache.store[get_cache_key(
'AUTH_test', 'bucket+segments')] = {'status': 404} 'AUTH_test', 'bucket+segments')] = {'status': 404}
self._test_object_multipart_upload_initiate({}, fake_memcache, self._test_object_multipart_upload_initiate({}, fake_memcache,
@ -613,6 +621,33 @@ class TestS3ApiMultiUpload(S3ApiTestCase):
fake_memcache, fake_memcache,
bucket_exists=False) bucket_exists=False)
@patch_policies([
StoragePolicy(0, 'gold', is_default=True),
StoragePolicy(1, 'silver')])
def test_object_mpu_initiate_without_segment_bucket_same_container(self):
self.swift.register('PUT', '/v1/AUTH_test/bucket+segments',
swob.HTTPCreated,
{'X-Storage-Policy': 'silver'}, None)
fake_memcache = FakeMemcache()
fake_memcache.store[get_cache_key(
'AUTH_test', 'bucket')] = {'status': 204,
'storage_policy': '1'}
fake_memcache.store[get_cache_key(
'AUTH_test', 'bucket+segments')] = {'status': 404}
self.s3api.conf.derived_container_policy_use_default = False
self._test_object_multipart_upload_initiate({}, fake_memcache,
bucket_exists=False,
expected_policy='silver')
self._test_object_multipart_upload_initiate({'Etag': 'blahblahblah'},
fake_memcache,
bucket_exists=False,
expected_policy='silver')
self._test_object_multipart_upload_initiate(
{'Content-MD5': base64.b64encode(b'blahblahblahblah').strip()},
fake_memcache,
bucket_exists=False,
expected_policy='silver')
@patch('swift.common.middleware.s3api.controllers.multi_upload.' @patch('swift.common.middleware.s3api.controllers.multi_upload.'
'unique_id', lambda: 'X') 'unique_id', lambda: 'X')
def _test_object_multipart_upload_initiate_s3acl( def _test_object_multipart_upload_initiate_s3acl(