Reduce backend requests for SLO If-Match / HEAD requests

... by storing SLO Etag and size in sysmeta.

Previously, we had to GET the manifest for every HEAD or conditional
request to an SLO. Worse, since SLO PUTs require that we HEAD every
segment, we'd GET all included sub-SLO manifests. This was necessary so
we could recompute the large object's Etag and content-length.

Since we already know both of those during PUT, we'll now store it in
object sysmeta. This allows us to:

 * satisfy HEAD requests based purely off of the manifest's HEAD
   response, and
 * perform the If-(None-)Match comparison on the object server, without
   any additional subrequests.

Note that the large object content-length can't just be parsed from
content-type -- with fast-POST enabled, the content-type coming out of
the object-server won't necessarily include swift_bytes.

Also note that we must still fall back to GETting the manifest if the
sysmeta headers were not found. Otherwise, we'd break existing large
objects.

Change-Id: Ia6ad32354105515560b005cea750aa64a88c96f9
This commit is contained in:
Tim Burke 2016-07-25 15:06:23 -07:00
parent 5d7a3a4172
commit 2d25fe6ad3
3 changed files with 208 additions and 85 deletions

View File

@ -200,15 +200,12 @@ the manifest and the segments it's referring to) in the container and account
metadata which can be used for stats purposes. metadata which can be used for stats purposes.
""" """
from six.moves import range
from collections import defaultdict from collections import defaultdict
from datetime import datetime from datetime import datetime
import json import json
import mimetypes import mimetypes
import re import re
import six import six
from six import BytesIO
from hashlib import md5 from hashlib import md5
from swift.common.exceptions import ListingIterError, SegmentError from swift.common.exceptions import ListingIterError, SegmentError
from swift.common.swob import Request, HTTPBadRequest, HTTPServerError, \ from swift.common.swob import Request, HTTPBadRequest, HTTPServerError, \
@ -219,7 +216,8 @@ from swift.common.utils import get_logger, config_true_value, \
get_valid_utf8_str, override_bytes_from_content_type, split_path, \ get_valid_utf8_str, override_bytes_from_content_type, split_path, \
register_swift_info, RateLimitedIterator, quote, close_if_possible, \ register_swift_info, RateLimitedIterator, quote, close_if_possible, \
closing_if_possible, LRUCache, StreamingPile closing_if_possible, LRUCache, StreamingPile
from swift.common.request_helpers import SegmentedIterable from swift.common.request_helpers import SegmentedIterable, \
get_sys_meta_prefix, update_etag_is_at_header
from swift.common.constraints import check_utf8, MAX_BUFFERED_SLO_SEGMENTS from swift.common.constraints import check_utf8, MAX_BUFFERED_SLO_SEGMENTS
from swift.common.http import HTTP_NOT_FOUND, HTTP_UNAUTHORIZED, is_success from swift.common.http import HTTP_NOT_FOUND, HTTP_UNAUTHORIZED, is_success
from swift.common.wsgi import WSGIContext, make_subrequest from swift.common.wsgi import WSGIContext, make_subrequest
@ -236,6 +234,9 @@ REQUIRED_SLO_KEYS = set(['path', 'etag', 'size_bytes'])
OPTIONAL_SLO_KEYS = set(['range']) OPTIONAL_SLO_KEYS = set(['range'])
ALLOWED_SLO_KEYS = REQUIRED_SLO_KEYS | OPTIONAL_SLO_KEYS ALLOWED_SLO_KEYS = REQUIRED_SLO_KEYS | OPTIONAL_SLO_KEYS
SYSMETA_SLO_ETAG = get_sys_meta_prefix('object') + 'slo-etag'
SYSMETA_SLO_SIZE = get_sys_meta_prefix('object') + 'slo-size'
def parse_and_validate_input(req_body, req_path): def parse_and_validate_input(req_body, req_path):
""" """
@ -361,25 +362,6 @@ def parse_and_validate_input(req_body, req_path):
return parsed_data return parsed_data
class SloPutContext(WSGIContext):
def __init__(self, slo, slo_etag):
super(SloPutContext, self).__init__(slo.app)
self.slo_etag = '"' + slo_etag.hexdigest() + '"'
def handle_slo_put(self, req, start_response):
app_resp = self._app_call(req.environ)
for i in range(len(self._response_headers)):
if self._response_headers[i][0].lower() == 'etag':
self._response_headers[i] = ('Etag', self.slo_etag)
break
start_response(self._response_status,
self._response_headers,
self._response_exc_info)
return app_resp
class SloGetContext(WSGIContext): class SloGetContext(WSGIContext):
max_slo_recursion_depth = 10 max_slo_recursion_depth = 10
@ -539,6 +521,9 @@ class SloGetContext(WSGIContext):
Note: this assumes that X-Static-Large-Object has already been found. Note: this assumes that X-Static-Large-Object has already been found.
""" """
if req.method == 'HEAD': if req.method == 'HEAD':
# We've already looked for SYSMETA_SLO_ETAG/SIZE in the response
# and didn't find them. We have to fetch the whole manifest and
# recompute.
return True return True
response_status = int(self._response_status[:3]) response_status = int(self._response_status[:3])
@ -581,14 +566,31 @@ class SloGetContext(WSGIContext):
what may be a static large object manifest (or may not). what may be a static large object manifest (or may not).
:param start_response: WSGI start_response callable :param start_response: WSGI start_response callable
""" """
if req.params.get('multipart-manifest') != 'get':
# If this object is an SLO manifest, we may have saved off the
# large object etag during the original PUT. Send an
# X-Backend-Etag-Is-At header so that, if the SLO etag *was*
# saved, we can trust the object-server to respond appropriately
# to If-Match/If-None-Match requests.
update_etag_is_at_header(req, SYSMETA_SLO_ETAG)
resp_iter = self._app_call(req.environ) resp_iter = self._app_call(req.environ)
# make sure this response is for a static large object manifest # make sure this response is for a static large object manifest
slo_marker = slo_etag = slo_size = None
for header, value in self._response_headers: for header, value in self._response_headers:
if (header.lower() == 'x-static-large-object' and header = header.lower()
config_true_value(value)): if header == SYSMETA_SLO_ETAG:
slo_etag = value
elif header == SYSMETA_SLO_SIZE:
slo_size = value
elif (header == 'x-static-large-object' and
config_true_value(value)):
slo_marker = value
if slo_marker and slo_etag and slo_size:
break break
else:
if not slo_marker:
# Not a static large object manifest. Just pass it through. # Not a static large object manifest. Just pass it through.
start_response(self._response_status, start_response(self._response_status,
self._response_headers, self._response_headers,
@ -614,6 +616,22 @@ class SloGetContext(WSGIContext):
self._response_exc_info) self._response_exc_info)
return resp_iter return resp_iter
is_conditional = self._response_status.startswith(('304', '412')) and (
req.if_match or req.if_none_match)
if slo_etag and slo_size and (
req.method == 'HEAD' or is_conditional):
# Since we have length and etag, we can respond immediately
for i, (header, _value) in enumerate(self._response_headers):
lheader = header.lower()
if lheader == 'etag':
self._response_headers[i] = (header, '"%s"' % slo_etag)
elif lheader == 'content-length' and not is_conditional:
self._response_headers[i] = (header, slo_size)
start_response(self._response_status,
self._response_headers,
self._response_exc_info)
return resp_iter
if self._need_to_refetch_manifest(req): if self._need_to_refetch_manifest(req):
req.environ['swift.non_client_disconnect'] = True req.environ['swift.non_client_disconnect'] = True
close_if_possible(resp_iter) close_if_possible(resp_iter)
@ -659,8 +677,7 @@ class SloGetContext(WSGIContext):
new_headers = [] new_headers = []
for header, value in resp_headers: for header, value in resp_headers:
if header.lower() == 'content-length': if header.lower() == 'content-length':
new_headers.append(('Content-Length', new_headers.append(('Content-Length', len(json_data)))
len(json_data)))
else: else:
new_headers.append((header, value)) new_headers.append((header, value))
self._response_headers = new_headers self._response_headers = new_headers
@ -680,23 +697,36 @@ class SloGetContext(WSGIContext):
def get_or_head_response(self, req, resp_headers, resp_iter): def get_or_head_response(self, req, resp_headers, resp_iter):
segments = self._get_manifest_read(resp_iter) segments = self._get_manifest_read(resp_iter)
etag = md5() slo_etag = None
content_length = 0 content_length = None
for seg_dict in segments: response_headers = []
if seg_dict.get('range'): for header, value in resp_headers:
etag.update('%s:%s;' % (seg_dict['hash'], seg_dict['range'])) lheader = header.lower()
else: if lheader == SYSMETA_SLO_ETAG:
etag.update(seg_dict['hash']) slo_etag = value
elif lheader == SYSMETA_SLO_SIZE:
content_length = value
elif lheader not in ('etag', 'content-length'):
response_headers.append((header, value))
if config_true_value(seg_dict.get('sub_slo')): if slo_etag is None or content_length is None:
override_bytes_from_content_type( etag = md5()
seg_dict, logger=self.slo.logger) content_length = 0
content_length += self._segment_length(seg_dict) for seg_dict in segments:
if seg_dict.get('range'):
etag.update('%s:%s;' % (seg_dict['hash'],
seg_dict['range']))
else:
etag.update(seg_dict['hash'])
if config_true_value(seg_dict.get('sub_slo')):
override_bytes_from_content_type(
seg_dict, logger=self.slo.logger)
content_length += self._segment_length(seg_dict)
slo_etag = etag.hexdigest()
response_headers = [(h, v) for h, v in resp_headers
if h.lower() not in ('etag', 'content-length')]
response_headers.append(('Content-Length', str(content_length))) response_headers.append(('Content-Length', str(content_length)))
response_headers.append(('Etag', '"%s"' % etag.hexdigest())) response_headers.append(('Etag', '"%s"' % slo_etag))
if req.method == 'HEAD': if req.method == 'HEAD':
return self._manifest_head_response(req, response_headers) return self._manifest_head_response(req, response_headers)
@ -942,7 +972,6 @@ class StaticLargeObject(object):
resp_body = get_response_body( resp_body = get_response_body(
out_content_type, {}, problem_segments) out_content_type, {}, problem_segments)
raise HTTPBadRequest(resp_body, content_type=out_content_type) raise HTTPBadRequest(resp_body, content_type=out_content_type)
env = req.environ
slo_etag = md5() slo_etag = md5()
for seg_data in data_for_storage: for seg_data in data_for_storage:
@ -952,20 +981,33 @@ class StaticLargeObject(object):
else: else:
slo_etag.update(seg_data['hash']) slo_etag.update(seg_data['hash'])
slo_etag = slo_etag.hexdigest()
req.headers.update({
SYSMETA_SLO_ETAG: slo_etag,
SYSMETA_SLO_SIZE: total_size,
'X-Static-Large-Object': 'True',
})
json_data = json.dumps(data_for_storage)
if six.PY3:
json_data = json_data.encode('utf-8')
req.body = json_data
env = req.environ
if not env.get('CONTENT_TYPE'): if not env.get('CONTENT_TYPE'):
guessed_type, _junk = mimetypes.guess_type(req.path_info) guessed_type, _junk = mimetypes.guess_type(req.path_info)
env['CONTENT_TYPE'] = guessed_type or 'application/octet-stream' env['CONTENT_TYPE'] = guessed_type or 'application/octet-stream'
env['swift.content_type_overridden'] = True env['swift.content_type_overridden'] = True
env['CONTENT_TYPE'] += ";swift_bytes=%d" % total_size env['CONTENT_TYPE'] += ";swift_bytes=%d" % total_size
env['HTTP_X_STATIC_LARGE_OBJECT'] = 'True'
json_data = json.dumps(data_for_storage)
if six.PY3:
json_data = json_data.encode('utf-8')
env['CONTENT_LENGTH'] = str(len(json_data))
env['wsgi.input'] = BytesIO(json_data)
slo_put_context = SloPutContext(self, slo_etag) def start_response_wrapper(status, headers, exc_info=None):
return slo_put_context.handle_slo_put(req, start_response) for i, (header, _value) in enumerate(headers):
if header.lower() == 'etag':
headers[i] = ('Etag', '"%s"' % slo_etag)
break
return start_response(status, headers, exc_info)
return self.app(env, start_response_wrapper)
def get_segments_to_delete_iter(self, req): def get_segments_to_delete_iter(self, req):
""" """

View File

@ -154,15 +154,23 @@ class FakeSwift(object):
self._calls.append( self._calls.append(
FakeSwiftCall(method, path, HeaderKeyDict(req.headers))) FakeSwiftCall(method, path, HeaderKeyDict(req.headers)))
backend_etag_header = req.headers.get('X-Backend-Etag-Is-At')
conditional_etag = None
if backend_etag_header and backend_etag_header in headers:
# Apply conditional etag overrides
conditional_etag = headers[backend_etag_header]
# range requests ought to work, hence conditional_response=True # range requests ought to work, hence conditional_response=True
if isinstance(body, list): if isinstance(body, list):
resp = resp_class( resp = resp_class(
req=req, headers=headers, app_iter=body, req=req, headers=headers, app_iter=body,
conditional_response=req.method in ('GET', 'HEAD')) conditional_response=req.method in ('GET', 'HEAD'),
conditional_etag=conditional_etag)
else: else:
resp = resp_class( resp = resp_class(
req=req, headers=headers, body=body, req=req, headers=headers, body=body,
conditional_response=req.method in ('GET', 'HEAD')) conditional_response=req.method in ('GET', 'HEAD'),
conditional_etag=conditional_etag)
wsgi_iter = resp(env, start_response) wsgi_iter = resp(env, start_response)
self.mark_opened(path) self.mark_opened(path)
return LeakTrackingIter(wsgi_iter, self, path) return LeakTrackingIter(wsgi_iter, self, path)

View File

@ -414,7 +414,9 @@ class TestSloPutManifest(SloTestCase):
'/v1/AUTH_test/c/man?multipart-manifest=put', '/v1/AUTH_test/c/man?multipart-manifest=put',
environ={'REQUEST_METHOD': 'PUT'}, headers={'Accept': 'test'}, environ={'REQUEST_METHOD': 'PUT'}, headers={'Accept': 'test'},
body=test_json_data) body=test_json_data)
self.assertNotIn('X-Static-Large-Object', req.headers) for h in ('X-Static-Large-Object', 'X-Object-Sysmeta-Slo-Etag',
'X-Object-Sysmeta-Slo-Size'):
self.assertNotIn(h, req.headers)
def my_fake_start_response(*args, **kwargs): def my_fake_start_response(*args, **kwargs):
gen_etag = '"' + md5hex('etagoftheobjectsegment') + '"' gen_etag = '"' + md5hex('etagoftheobjectsegment') + '"'
@ -423,6 +425,11 @@ class TestSloPutManifest(SloTestCase):
self.slo(req.environ, my_fake_start_response) self.slo(req.environ, my_fake_start_response)
self.assertIn('X-Static-Large-Object', req.headers) self.assertIn('X-Static-Large-Object', req.headers)
self.assertEqual(req.headers['X-Static-Large-Object'], 'True') self.assertEqual(req.headers['X-Static-Large-Object'], 'True')
self.assertIn('X-Object-Sysmeta-Slo-Etag', req.headers)
self.assertEqual(req.headers['X-Object-Sysmeta-Slo-Etag'],
md5hex('etagoftheobjectsegment'))
self.assertIn('X-Object-Sysmeta-Slo-Size', req.headers)
self.assertEqual(req.headers['X-Object-Sysmeta-Slo-Size'], '100')
self.assertIn('Content-Type', req.headers) self.assertIn('Content-Type', req.headers)
self.assertTrue( self.assertTrue(
req.headers['Content-Type'].endswith(';swift_bytes=100'), req.headers['Content-Type'].endswith(';swift_bytes=100'),
@ -1078,11 +1085,11 @@ class TestSloDeleteManifest(SloTestCase):
'man-all-there?multipart-manifest=delete'))])) 'man-all-there?multipart-manifest=delete'))]))
class TestSloHeadManifest(SloTestCase): class TestSloHeadOldManifest(SloTestCase):
slo_etag = md5hex("seg01-hashseg02-hash") slo_etag = md5hex("seg01-hashseg02-hash")
def setUp(self): def setUp(self):
super(TestSloHeadManifest, self).setUp() super(TestSloHeadOldManifest, self).setUp()
manifest_json = json.dumps([ manifest_json = json.dumps([
{'name': '/gettest/seg01', {'name': '/gettest/seg01',
'bytes': '100', 'bytes': '100',
@ -1100,6 +1107,8 @@ class TestSloHeadManifest(SloTestCase):
'X-Static-Large-Object': 'true', 'X-Static-Large-Object': 'true',
'Etag': md5hex(manifest_json)} 'Etag': md5hex(manifest_json)}
manifest_headers.update(getattr(self, 'extra_manifest_headers', {})) manifest_headers.update(getattr(self, 'extra_manifest_headers', {}))
self.manifest_has_sysmeta = all(h in manifest_headers for h in (
'X-Object-Sysmeta-Slo-Etag', 'X-Object-Sysmeta-Slo-Size'))
self.app.register( self.app.register(
'GET', '/v1/AUTH_test/headtest/man', 'GET', '/v1/AUTH_test/headtest/man',
swob.HTTPOk, manifest_headers, manifest_json) swob.HTTPOk, manifest_headers, manifest_json)
@ -1116,9 +1125,9 @@ class TestSloHeadManifest(SloTestCase):
self.assertIn(('Content-Type', 'test/data'), headers) self.assertIn(('Content-Type', 'test/data'), headers)
self.assertEqual(body, '') # it's a HEAD request, after all self.assertEqual(body, '') # it's a HEAD request, after all
expected_app_calls = [ expected_app_calls = [('HEAD', '/v1/AUTH_test/headtest/man')]
('HEAD', '/v1/AUTH_test/headtest/man'), if not self.manifest_has_sysmeta:
('GET', '/v1/AUTH_test/headtest/man')] expected_app_calls.append(('GET', '/v1/AUTH_test/headtest/man'))
self.assertEqual(self.app.calls, expected_app_calls) self.assertEqual(self.app.calls, expected_app_calls)
def test_if_none_match_etag_matching(self): def test_if_none_match_etag_matching(self):
@ -1132,9 +1141,9 @@ class TestSloHeadManifest(SloTestCase):
self.assertIn(('Content-Length', '0'), headers) self.assertIn(('Content-Length', '0'), headers)
self.assertIn(('Content-Type', 'test/data'), headers) self.assertIn(('Content-Type', 'test/data'), headers)
expected_app_calls = [ expected_app_calls = [('HEAD', '/v1/AUTH_test/headtest/man')]
('HEAD', '/v1/AUTH_test/headtest/man'), if not self.manifest_has_sysmeta:
('GET', '/v1/AUTH_test/headtest/man')] expected_app_calls.append(('GET', '/v1/AUTH_test/headtest/man'))
self.assertEqual(self.app.calls, expected_app_calls) self.assertEqual(self.app.calls, expected_app_calls)
def test_if_match_etag_not_matching(self): def test_if_match_etag_not_matching(self):
@ -1148,12 +1157,21 @@ class TestSloHeadManifest(SloTestCase):
self.assertIn(('Content-Length', '0'), headers) self.assertIn(('Content-Length', '0'), headers)
self.assertIn(('Content-Type', 'test/data'), headers) self.assertIn(('Content-Type', 'test/data'), headers)
expected_app_calls = [ expected_app_calls = [('HEAD', '/v1/AUTH_test/headtest/man')]
('HEAD', '/v1/AUTH_test/headtest/man'), if not self.manifest_has_sysmeta:
('GET', '/v1/AUTH_test/headtest/man')] expected_app_calls.append(('GET', '/v1/AUTH_test/headtest/man'))
self.assertEqual(self.app.calls, expected_app_calls) self.assertEqual(self.app.calls, expected_app_calls)
class TestSloHeadManifest(TestSloHeadOldManifest):
def setUp(self):
self.extra_manifest_headers = {
'X-Object-Sysmeta-Slo-Etag': self.slo_etag,
'X-Object-Sysmeta-Slo-Size': '300',
}
super(TestSloHeadManifest, self).setUp()
class TestSloGetRawManifest(SloTestCase): class TestSloGetRawManifest(SloTestCase):
def setUp(self): def setUp(self):
@ -2759,7 +2777,7 @@ class TestSloGetManifest(SloTestCase):
'ERROR: An error occurred while retrieving segments')) 'ERROR: An error occurred while retrieving segments'))
class TestSloConditionalGetManifest(SloTestCase): class TestSloConditionalGetOldManifest(SloTestCase):
slo_data = [ slo_data = [
{'name': '/gettest/a_5', 'hash': md5hex("a" * 5), {'name': '/gettest/a_5', 'hash': md5hex("a" * 5),
'content_type': 'text/plain', 'bytes': '5'}, 'content_type': 'text/plain', 'bytes': '5'},
@ -2772,7 +2790,7 @@ class TestSloConditionalGetManifest(SloTestCase):
slo_etag = md5hex(''.join(seg['hash'] for seg in slo_data)) slo_etag = md5hex(''.join(seg['hash'] for seg in slo_data))
def setUp(self): def setUp(self):
super(TestSloConditionalGetManifest, self).setUp() super(TestSloConditionalGetOldManifest, self).setUp()
# some plain old objects # some plain old objects
self.app.register( self.app.register(
@ -2816,6 +2834,8 @@ class TestSloConditionalGetManifest(SloTestCase):
'X-Static-Large-Object': 'true', 'X-Static-Large-Object': 'true',
'Etag': md5hex(_abcd_manifest_json)} 'Etag': md5hex(_abcd_manifest_json)}
manifest_headers.update(getattr(self, 'extra_manifest_headers', {})) manifest_headers.update(getattr(self, 'extra_manifest_headers', {}))
self.manifest_has_sysmeta = all(h in manifest_headers for h in (
'X-Object-Sysmeta-Slo-Etag', 'X-Object-Sysmeta-Slo-Size'))
self.app.register( self.app.register(
'GET', '/v1/AUTH_test/gettest/manifest-abcd', 'GET', '/v1/AUTH_test/gettest/manifest-abcd',
swob.HTTPOk, manifest_headers, swob.HTTPOk, manifest_headers,
@ -2833,13 +2853,16 @@ class TestSloConditionalGetManifest(SloTestCase):
self.assertIn(('Etag', '"%s"' % self.slo_etag), headers) self.assertIn(('Etag', '"%s"' % self.slo_etag), headers)
self.assertEqual(body, '') self.assertEqual(body, '')
expected_app_calls = [ expected_app_calls = [('GET', '/v1/AUTH_test/gettest/manifest-abcd')]
('GET', '/v1/AUTH_test/gettest/manifest-abcd'), if not self.manifest_has_sysmeta:
# Need to verify the first segment # We *still* verify the first segment
('GET', '/v1/AUTH_test/gettest/manifest-bc'), expected_app_calls.extend([
('GET', '/v1/AUTH_test/gettest/a_5?multipart-manifest=get'), ('GET', '/v1/AUTH_test/gettest/manifest-bc'),
] ('GET', '/v1/AUTH_test/gettest/a_5?multipart-manifest=get'),
])
self.assertEqual(self.app.calls, expected_app_calls) self.assertEqual(self.app.calls, expected_app_calls)
self.assertEqual(self.app.headers[0].get('X-Backend-Etag-Is-At'),
'x-object-sysmeta-slo-etag')
def test_if_none_match_does_not_match(self): def test_if_none_match_does_not_match(self):
req = Request.blank( req = Request.blank(
@ -2863,6 +2886,8 @@ class TestSloConditionalGetManifest(SloTestCase):
('GET', '/v1/AUTH_test/gettest/d_20?multipart-manifest=get'), ('GET', '/v1/AUTH_test/gettest/d_20?multipart-manifest=get'),
] ]
self.assertEqual(self.app.calls, expected_app_calls) self.assertEqual(self.app.calls, expected_app_calls)
self.assertEqual(self.app.headers[0].get('X-Backend-Etag-Is-At'),
'x-object-sysmeta-slo-etag')
def test_if_match_matches(self): def test_if_match_matches(self):
req = Request.blank( req = Request.blank(
@ -2877,17 +2902,21 @@ class TestSloConditionalGetManifest(SloTestCase):
self.assertEqual( self.assertEqual(
body, 'aaaaabbbbbbbbbbcccccccccccccccdddddddddddddddddddd') body, 'aaaaabbbbbbbbbbcccccccccccccccdddddddddddddddddddd')
expected_app_calls = [ expected_app_calls = [('GET', '/v1/AUTH_test/gettest/manifest-abcd')]
('GET', '/v1/AUTH_test/gettest/manifest-abcd'), if not self.manifest_has_sysmeta:
# Manifest never matches -> got back a 412; need to re-fetch # Manifest never matches -> got back a 412; need to re-fetch
('GET', '/v1/AUTH_test/gettest/manifest-abcd'), expected_app_calls.append(
('GET', '/v1/AUTH_test/gettest/manifest-abcd'))
expected_app_calls.extend([
('GET', '/v1/AUTH_test/gettest/manifest-bc'), ('GET', '/v1/AUTH_test/gettest/manifest-bc'),
('GET', '/v1/AUTH_test/gettest/a_5?multipart-manifest=get'), ('GET', '/v1/AUTH_test/gettest/a_5?multipart-manifest=get'),
('GET', '/v1/AUTH_test/gettest/b_10?multipart-manifest=get'), ('GET', '/v1/AUTH_test/gettest/b_10?multipart-manifest=get'),
('GET', '/v1/AUTH_test/gettest/c_15?multipart-manifest=get'), ('GET', '/v1/AUTH_test/gettest/c_15?multipart-manifest=get'),
('GET', '/v1/AUTH_test/gettest/d_20?multipart-manifest=get'), ('GET', '/v1/AUTH_test/gettest/d_20?multipart-manifest=get'),
] ])
self.assertEqual(self.app.calls, expected_app_calls) self.assertEqual(self.app.calls, expected_app_calls)
self.assertEqual(self.app.headers[0].get('X-Backend-Etag-Is-At'),
'x-object-sysmeta-slo-etag')
def test_if_match_does_not_match(self): def test_if_match_does_not_match(self):
req = Request.blank( req = Request.blank(
@ -2901,15 +2930,18 @@ class TestSloConditionalGetManifest(SloTestCase):
self.assertIn(('Etag', '"%s"' % self.slo_etag), headers) self.assertIn(('Etag', '"%s"' % self.slo_etag), headers)
self.assertEqual(body, '') self.assertEqual(body, '')
expected_app_calls = [ expected_app_calls = [('GET', '/v1/AUTH_test/gettest/manifest-abcd')]
('GET', '/v1/AUTH_test/gettest/manifest-abcd'), if not self.manifest_has_sysmeta:
# Manifest never matches -> got back a 412; need to re-fetch # We *still* verify the first segment
('GET', '/v1/AUTH_test/gettest/manifest-abcd'), expected_app_calls.extend([
# We need to verify the first segment # Manifest never matches -> got back a 412; need to re-fetch
('GET', '/v1/AUTH_test/gettest/manifest-bc'), ('GET', '/v1/AUTH_test/gettest/manifest-abcd'),
('GET', '/v1/AUTH_test/gettest/a_5?multipart-manifest=get'), ('GET', '/v1/AUTH_test/gettest/manifest-bc'),
] ('GET', '/v1/AUTH_test/gettest/a_5?multipart-manifest=get'),
])
self.assertEqual(self.app.calls, expected_app_calls) self.assertEqual(self.app.calls, expected_app_calls)
self.assertEqual(self.app.headers[0].get('X-Backend-Etag-Is-At'),
'x-object-sysmeta-slo-etag')
def test_if_match_matches_and_range(self): def test_if_match_matches_and_range(self):
req = Request.blank( req = Request.blank(
@ -2935,6 +2967,47 @@ class TestSloConditionalGetManifest(SloTestCase):
('GET', '/v1/AUTH_test/gettest/b_10?multipart-manifest=get'), ('GET', '/v1/AUTH_test/gettest/b_10?multipart-manifest=get'),
] ]
self.assertEqual(self.app.calls, expected_app_calls) self.assertEqual(self.app.calls, expected_app_calls)
self.assertEqual(self.app.headers[0].get('X-Backend-Etag-Is-At'),
'x-object-sysmeta-slo-etag')
def test_if_match_matches_passthrough(self):
# first fetch and stash the manifest etag
req = Request.blank(
'/v1/AUTH_test/gettest/manifest-abcd?multipart-manifest=get',
environ={'REQUEST_METHOD': 'GET'})
status, headers, body = self.call_slo(req)
self.assertEqual(status, '200 OK')
headers = HeaderKeyDict(headers)
self.assertEqual('application/json; charset=utf-8',
headers['Content-Type'])
manifest_etag = headers['Etag']
# now use it as a condition and expect to match
req = Request.blank(
'/v1/AUTH_test/gettest/manifest-abcd?multipart-manifest=get',
environ={'REQUEST_METHOD': 'GET'},
headers={'If-Match': manifest_etag})
status, headers, body = self.call_slo(req)
self.assertEqual(status, '200 OK')
headers = HeaderKeyDict(headers)
self.assertEqual(manifest_etag, headers['Etag'])
expected_app_calls = [
('GET',
'/v1/AUTH_test/gettest/manifest-abcd?multipart-manifest=get')] * 2
self.assertEqual(self.app.calls, expected_app_calls)
self.assertNotIn('X-Backend-Etag-Is-At', self.app.headers[0])
self.assertNotIn('X-Backend-Etag-Is-At', self.app.headers[1])
class TestSloConditionalGetNewManifest(TestSloConditionalGetOldManifest):
def setUp(self):
self.extra_manifest_headers = {
'X-Object-Sysmeta-Slo-Etag': self.slo_etag,
'X-Object-Sysmeta-Slo-Size': '50',
}
super(TestSloConditionalGetNewManifest, self).setUp()
class TestSloBulkLogger(unittest.TestCase): class TestSloBulkLogger(unittest.TestCase):