Merge "func tests: work with etag-quoter on by default"
This commit is contained in:
commit
ba01f93ad9
@ -322,12 +322,16 @@ def _load_encryption(proxy_conf_file, swift_conf_file, **kwargs):
|
|||||||
pipeline = pipeline.replace(
|
pipeline = pipeline.replace(
|
||||||
"proxy-logging proxy-server",
|
"proxy-logging proxy-server",
|
||||||
"keymaster encryption proxy-logging proxy-server")
|
"keymaster encryption proxy-logging proxy-server")
|
||||||
|
pipeline = pipeline.replace(
|
||||||
|
"cache listing_formats",
|
||||||
|
"cache etag-quoter listing_formats")
|
||||||
conf.set(section, 'pipeline', pipeline)
|
conf.set(section, 'pipeline', pipeline)
|
||||||
root_secret = base64.b64encode(os.urandom(32))
|
root_secret = base64.b64encode(os.urandom(32))
|
||||||
if not six.PY2:
|
if not six.PY2:
|
||||||
root_secret = root_secret.decode('ascii')
|
root_secret = root_secret.decode('ascii')
|
||||||
conf.set('filter:keymaster', 'encryption_root_secret', root_secret)
|
conf.set('filter:keymaster', 'encryption_root_secret', root_secret)
|
||||||
conf.set('filter:versioned_writes', 'allow_object_versioning', 'true')
|
conf.set('filter:versioned_writes', 'allow_object_versioning', 'true')
|
||||||
|
conf.set('filter:etag-quoter', 'enable_by_default', 'true')
|
||||||
except NoSectionError as err:
|
except NoSectionError as err:
|
||||||
msg = 'Error problem with proxy conf file %s: %s' % \
|
msg = 'Error problem with proxy conf file %s: %s' % \
|
||||||
(proxy_conf_file, err)
|
(proxy_conf_file, err)
|
||||||
|
@ -1726,7 +1726,7 @@ class TestObject(unittest.TestCase):
|
|||||||
if 'etag_quoter' not in tf.cluster_info:
|
if 'etag_quoter' not in tf.cluster_info:
|
||||||
raise SkipTest("etag-quoter middleware is not enabled")
|
raise SkipTest("etag-quoter middleware is not enabled")
|
||||||
|
|
||||||
def do_head(expect_quoted=False):
|
def do_head(expect_quoted=None):
|
||||||
def head(url, token, parsed, conn):
|
def head(url, token, parsed, conn):
|
||||||
conn.request('HEAD', '%s/%s/%s' % (
|
conn.request('HEAD', '%s/%s/%s' % (
|
||||||
parsed.path, self.container, self.obj), '',
|
parsed.path, self.container, self.obj), '',
|
||||||
@ -1736,6 +1736,11 @@ class TestObject(unittest.TestCase):
|
|||||||
resp = retry(head)
|
resp = retry(head)
|
||||||
resp.read()
|
resp.read()
|
||||||
self.assertEqual(resp.status, 200)
|
self.assertEqual(resp.status, 200)
|
||||||
|
|
||||||
|
if expect_quoted is None:
|
||||||
|
expect_quoted = tf.cluster_info.get('etag_quoter', {}).get(
|
||||||
|
'enable_by_default', False)
|
||||||
|
|
||||||
expected_etag = hashlib.md5(b'test').hexdigest()
|
expected_etag = hashlib.md5(b'test').hexdigest()
|
||||||
if expect_quoted:
|
if expect_quoted:
|
||||||
expected_etag = '"%s"' % expected_etag
|
expected_etag = '"%s"' % expected_etag
|
||||||
@ -1771,7 +1776,7 @@ class TestObject(unittest.TestCase):
|
|||||||
post_container('')
|
post_container('')
|
||||||
do_head(expect_quoted=True)
|
do_head(expect_quoted=True)
|
||||||
post_container('f')
|
post_container('f')
|
||||||
do_head()
|
do_head(expect_quoted=False)
|
||||||
finally:
|
finally:
|
||||||
# Don't leave a dirty account
|
# Don't leave a dirty account
|
||||||
post_account('')
|
post_account('')
|
||||||
|
@ -26,6 +26,7 @@ from six.moves.urllib.parse import quote, unquote
|
|||||||
|
|
||||||
import test.functional as tf
|
import test.functional as tf
|
||||||
|
|
||||||
|
from swift.common.swob import normalize_etag
|
||||||
from swift.common.utils import MD5_OF_EMPTY_STRING, config_true_value
|
from swift.common.utils import MD5_OF_EMPTY_STRING, config_true_value
|
||||||
from swift.common.middleware.versioned_writes.object_versioning import \
|
from swift.common.middleware.versioned_writes.object_versioning import \
|
||||||
DELETE_MARKER_CONTENT_TYPE
|
DELETE_MARKER_CONTENT_TYPE
|
||||||
@ -337,7 +338,7 @@ class TestObjectVersioning(TestObjectVersioningBase):
|
|||||||
obj = self.env.unversioned_container.file(oname)
|
obj = self.env.unversioned_container.file(oname)
|
||||||
resp = obj.write(body, return_resp=True)
|
resp = obj.write(body, return_resp=True)
|
||||||
etag = resp.getheader('etag')
|
etag = resp.getheader('etag')
|
||||||
self.assertEqual(md5(body).hexdigest(), etag)
|
self.assertEqual(md5(body).hexdigest(), normalize_etag(etag))
|
||||||
|
|
||||||
# un-versioned object is cool with with if-match
|
# un-versioned object is cool with with if-match
|
||||||
self.assertEqual(body, obj.read(hdrs={'if-match': etag}))
|
self.assertEqual(body, obj.read(hdrs={'if-match': etag}))
|
||||||
@ -350,7 +351,11 @@ class TestObjectVersioning(TestObjectVersioningBase):
|
|||||||
self.assertEqual(resp.getheader('etag'), etag)
|
self.assertEqual(resp.getheader('etag'), etag)
|
||||||
|
|
||||||
# versioned object is too with with if-match
|
# versioned object is too with with if-match
|
||||||
self.assertEqual(body, v_obj.read(hdrs={'if-match': etag}))
|
self.assertEqual(body, v_obj.read(hdrs={
|
||||||
|
'if-match': normalize_etag(etag)}))
|
||||||
|
# works quoted, too
|
||||||
|
self.assertEqual(body, v_obj.read(hdrs={
|
||||||
|
'if-match': '"%s"' % normalize_etag(etag)}))
|
||||||
with self.assertRaises(ResponseError) as cm:
|
with self.assertRaises(ResponseError) as cm:
|
||||||
v_obj.read(hdrs={'if-match': 'not-the-etag'})
|
v_obj.read(hdrs={'if-match': 'not-the-etag'})
|
||||||
self.assertEqual(412, cm.exception.status)
|
self.assertEqual(412, cm.exception.status)
|
||||||
@ -989,13 +994,13 @@ class TestObjectVersioning(TestObjectVersioningBase):
|
|||||||
'Content-Type': 'text/jibberish32'
|
'Content-Type': 'text/jibberish32'
|
||||||
}, return_resp=True)
|
}, return_resp=True)
|
||||||
v1_version_id = resp.getheader('x-object-version-id')
|
v1_version_id = resp.getheader('x-object-version-id')
|
||||||
v1_etag = resp.getheader('etag')
|
v1_etag = normalize_etag(resp.getheader('etag'))
|
||||||
|
|
||||||
resp = obj.write(b'version2', hdrs={
|
resp = obj.write(b'version2', hdrs={
|
||||||
'Content-Type': 'text/jibberish33'
|
'Content-Type': 'text/jibberish33'
|
||||||
}, return_resp=True)
|
}, return_resp=True)
|
||||||
v2_version_id = resp.getheader('x-object-version-id')
|
v2_version_id = resp.getheader('x-object-version-id')
|
||||||
v2_etag = resp.getheader('etag')
|
v2_etag = normalize_etag(resp.getheader('etag'))
|
||||||
|
|
||||||
# sanity
|
# sanity
|
||||||
self.assertEqual(b'version2', obj.read())
|
self.assertEqual(b'version2', obj.read())
|
||||||
@ -1062,7 +1067,7 @@ class TestObjectVersioning(TestObjectVersioningBase):
|
|||||||
self.assertEqual(b'version1', obj.read())
|
self.assertEqual(b'version1', obj.read())
|
||||||
obj_info = obj.info()
|
obj_info = obj.info()
|
||||||
self.assertEqual('text/jibberish32', obj_info['content_type'])
|
self.assertEqual('text/jibberish32', obj_info['content_type'])
|
||||||
self.assertEqual(v1_etag, obj_info['etag'])
|
self.assertEqual(v1_etag, normalize_etag(obj_info['etag']))
|
||||||
|
|
||||||
def test_delete_with_version_api_old_object(self):
|
def test_delete_with_version_api_old_object(self):
|
||||||
versioned_obj_name = Utils.create_name()
|
versioned_obj_name = Utils.create_name()
|
||||||
@ -2378,7 +2383,7 @@ class TestSloWithVersioning(TestObjectVersioningBase):
|
|||||||
expected = {
|
expected = {
|
||||||
'bytes': file_info['content_length'],
|
'bytes': file_info['content_length'],
|
||||||
'content_type': 'application/octet-stream',
|
'content_type': 'application/octet-stream',
|
||||||
'hash': manifest_info['etag'],
|
'hash': normalize_etag(manifest_info['etag']),
|
||||||
'name': 'my-slo-manifest',
|
'name': 'my-slo-manifest',
|
||||||
'slo_etag': file_info['etag'],
|
'slo_etag': file_info['etag'],
|
||||||
'version_symlink': True,
|
'version_symlink': True,
|
||||||
@ -2410,7 +2415,7 @@ class TestSloWithVersioning(TestObjectVersioningBase):
|
|||||||
expected = {
|
expected = {
|
||||||
'bytes': file_info['content_length'],
|
'bytes': file_info['content_length'],
|
||||||
'content_type': 'application/octet-stream',
|
'content_type': 'application/octet-stream',
|
||||||
'hash': manifest_info['etag'],
|
'hash': normalize_etag(manifest_info['etag']),
|
||||||
'name': 'my-slo-manifest',
|
'name': 'my-slo-manifest',
|
||||||
'slo_etag': file_info['etag'],
|
'slo_etag': file_info['etag'],
|
||||||
'version_symlink': True,
|
'version_symlink': True,
|
||||||
|
@ -23,6 +23,8 @@ from copy import deepcopy
|
|||||||
|
|
||||||
import six
|
import six
|
||||||
|
|
||||||
|
from swift.common.swob import normalize_etag
|
||||||
|
|
||||||
import test.functional as tf
|
import test.functional as tf
|
||||||
from test.functional import cluster_info, SkipTest
|
from test.functional import cluster_info, SkipTest
|
||||||
from test.functional.tests import Utils, Base, Base2, BaseEnv
|
from test.functional.tests import Utils, Base, Base2, BaseEnv
|
||||||
@ -299,8 +301,14 @@ class TestSlo(Base):
|
|||||||
# a POST.
|
# a POST.
|
||||||
file_item.initialize(parms={'multipart-manifest': 'get'})
|
file_item.initialize(parms={'multipart-manifest': 'get'})
|
||||||
manifest_etag = file_item.etag
|
manifest_etag = file_item.etag
|
||||||
self.assertFalse(manifest_etag.startswith('"'))
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
self.assertFalse(manifest_etag.endswith('"'))
|
self.assertTrue(manifest_etag.startswith('"'))
|
||||||
|
self.assertTrue(manifest_etag.endswith('"'))
|
||||||
|
# ...but in the listing, it'll be stripped
|
||||||
|
manifest_etag = manifest_etag[1:-1]
|
||||||
|
else:
|
||||||
|
self.assertFalse(manifest_etag.startswith('"'))
|
||||||
|
self.assertFalse(manifest_etag.endswith('"'))
|
||||||
|
|
||||||
file_item.initialize()
|
file_item.initialize()
|
||||||
slo_etag = file_item.etag
|
slo_etag = file_item.etag
|
||||||
@ -715,6 +723,8 @@ class TestSlo(Base):
|
|||||||
source_contents = source.read(parms={'multipart-manifest': 'get'})
|
source_contents = source.read(parms={'multipart-manifest': 'get'})
|
||||||
source_json = json.loads(source_contents)
|
source_json = json.loads(source_contents)
|
||||||
manifest_etag = hashlib.md5(source_contents).hexdigest()
|
manifest_etag = hashlib.md5(source_contents).hexdigest()
|
||||||
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
manifest_etag = '"%s"' % manifest_etag
|
||||||
self.assertEqual(manifest_etag, source.etag)
|
self.assertEqual(manifest_etag, source.etag)
|
||||||
|
|
||||||
source.initialize()
|
source.initialize()
|
||||||
@ -752,14 +762,14 @@ class TestSlo(Base):
|
|||||||
actual = names['manifest-abcde']
|
actual = names['manifest-abcde']
|
||||||
self.assertEqual(4 * 1024 * 1024 + 1, actual['bytes'])
|
self.assertEqual(4 * 1024 * 1024 + 1, actual['bytes'])
|
||||||
self.assertEqual('application/octet-stream', actual['content_type'])
|
self.assertEqual('application/octet-stream', actual['content_type'])
|
||||||
self.assertEqual(manifest_etag, actual['hash'])
|
self.assertEqual(normalize_etag(manifest_etag), actual['hash'])
|
||||||
self.assertEqual(slo_etag, actual['slo_etag'])
|
self.assertEqual(slo_etag, actual['slo_etag'])
|
||||||
|
|
||||||
self.assertIn('copied-abcde-manifest-only', names)
|
self.assertIn('copied-abcde-manifest-only', names)
|
||||||
actual = names['copied-abcde-manifest-only']
|
actual = names['copied-abcde-manifest-only']
|
||||||
self.assertEqual(4 * 1024 * 1024 + 1, actual['bytes'])
|
self.assertEqual(4 * 1024 * 1024 + 1, actual['bytes'])
|
||||||
self.assertEqual('application/octet-stream', actual['content_type'])
|
self.assertEqual('application/octet-stream', actual['content_type'])
|
||||||
self.assertEqual(manifest_etag, actual['hash'])
|
self.assertEqual(normalize_etag(manifest_etag), actual['hash'])
|
||||||
self.assertEqual(slo_etag, actual['slo_etag'])
|
self.assertEqual(slo_etag, actual['slo_etag'])
|
||||||
|
|
||||||
# Test copy manifest including data segments
|
# Test copy manifest including data segments
|
||||||
@ -789,6 +799,8 @@ class TestSlo(Base):
|
|||||||
source_contents = source.read(parms={'multipart-manifest': 'get'})
|
source_contents = source.read(parms={'multipart-manifest': 'get'})
|
||||||
source_json = json.loads(source_contents)
|
source_json = json.loads(source_contents)
|
||||||
manifest_etag = hashlib.md5(source_contents).hexdigest()
|
manifest_etag = hashlib.md5(source_contents).hexdigest()
|
||||||
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
manifest_etag = '"%s"' % manifest_etag
|
||||||
self.assertEqual(manifest_etag, source.etag)
|
self.assertEqual(manifest_etag, source.etag)
|
||||||
|
|
||||||
source.initialize()
|
source.initialize()
|
||||||
@ -831,14 +843,14 @@ class TestSlo(Base):
|
|||||||
self.assertEqual(4 * 1024 * 1024 + 1, actual['bytes'])
|
self.assertEqual(4 * 1024 * 1024 + 1, actual['bytes'])
|
||||||
self.assertEqual('application/octet-stream', actual['content_type'])
|
self.assertEqual('application/octet-stream', actual['content_type'])
|
||||||
# the container listing should have the etag of the manifest contents
|
# the container listing should have the etag of the manifest contents
|
||||||
self.assertEqual(manifest_etag, actual['hash'])
|
self.assertEqual(normalize_etag(manifest_etag), actual['hash'])
|
||||||
self.assertEqual(slo_etag, actual['slo_etag'])
|
self.assertEqual(slo_etag, actual['slo_etag'])
|
||||||
|
|
||||||
self.assertIn('copied-abcde-manifest-only', names)
|
self.assertIn('copied-abcde-manifest-only', names)
|
||||||
actual = names['copied-abcde-manifest-only']
|
actual = names['copied-abcde-manifest-only']
|
||||||
self.assertEqual(4 * 1024 * 1024 + 1, actual['bytes'])
|
self.assertEqual(4 * 1024 * 1024 + 1, actual['bytes'])
|
||||||
self.assertEqual('image/jpeg', actual['content_type'])
|
self.assertEqual('image/jpeg', actual['content_type'])
|
||||||
self.assertEqual(manifest_etag, actual['hash'])
|
self.assertEqual(normalize_etag(manifest_etag), actual['hash'])
|
||||||
self.assertEqual(slo_etag, actual['slo_etag'])
|
self.assertEqual(slo_etag, actual['slo_etag'])
|
||||||
|
|
||||||
def test_slo_copy_the_manifest_account(self):
|
def test_slo_copy_the_manifest_account(self):
|
||||||
@ -1098,12 +1110,7 @@ class TestSlo(Base):
|
|||||||
manifest = self.env.container.file("manifest-db")
|
manifest = self.env.container.file("manifest-db")
|
||||||
got_body = manifest.read(parms={'multipart-manifest': 'get',
|
got_body = manifest.read(parms={'multipart-manifest': 'get',
|
||||||
'format': 'raw'})
|
'format': 'raw'})
|
||||||
body_md5 = hashlib.md5(got_body).hexdigest()
|
self.assert_etag(hashlib.md5(got_body).hexdigest())
|
||||||
headers = dict(
|
|
||||||
(h.lower(), v)
|
|
||||||
for h, v in manifest.conn.response.getheaders())
|
|
||||||
self.assertIn('etag', headers)
|
|
||||||
self.assertEqual(headers['etag'], body_md5)
|
|
||||||
|
|
||||||
# raw format should have the actual manifest object content-type
|
# raw format should have the actual manifest object content-type
|
||||||
self.assertEqual('application/octet-stream', manifest.content_type)
|
self.assertEqual('application/octet-stream', manifest.content_type)
|
||||||
|
@ -25,6 +25,7 @@ from six.moves import urllib
|
|||||||
from uuid import uuid4
|
from uuid import uuid4
|
||||||
|
|
||||||
from swift.common.http import is_success
|
from swift.common.http import is_success
|
||||||
|
from swift.common.swob import normalize_etag
|
||||||
from swift.common.utils import json, MD5_OF_EMPTY_STRING
|
from swift.common.utils import json, MD5_OF_EMPTY_STRING
|
||||||
from swift.common.middleware.slo import SloGetContext
|
from swift.common.middleware.slo import SloGetContext
|
||||||
from test.functional import check_response, retry, requires_acls, \
|
from test.functional import check_response, retry, requires_acls, \
|
||||||
@ -1135,7 +1136,7 @@ class TestSymlink(Base):
|
|||||||
etag=self.env.tgt_etag)
|
etag=self.env.tgt_etag)
|
||||||
|
|
||||||
# overwrite tgt object
|
# overwrite tgt object
|
||||||
old_tgt_etag = self.env.tgt_etag
|
old_tgt_etag = normalize_etag(self.env.tgt_etag)
|
||||||
self.env._create_tgt_object(body='updated target body')
|
self.env._create_tgt_object(body='updated target body')
|
||||||
|
|
||||||
# sanity
|
# sanity
|
||||||
@ -1380,7 +1381,7 @@ class TestSymlink(Base):
|
|||||||
object_list[0]['symlink_path'])
|
object_list[0]['symlink_path'])
|
||||||
obj_info = object_list[0]
|
obj_info = object_list[0]
|
||||||
self.assertIn('symlink_etag', obj_info)
|
self.assertIn('symlink_etag', obj_info)
|
||||||
self.assertEqual(self.env.tgt_etag,
|
self.assertEqual(normalize_etag(self.env.tgt_etag),
|
||||||
obj_info['symlink_etag'])
|
obj_info['symlink_etag'])
|
||||||
self.assertEqual(int(self.env.tgt_length),
|
self.assertEqual(int(self.env.tgt_length),
|
||||||
obj_info['symlink_bytes'])
|
obj_info['symlink_bytes'])
|
||||||
@ -1550,7 +1551,7 @@ class TestSymlinkSlo(Base):
|
|||||||
'symlink_path': '/v1/%s/%s/manifest-abcde' % (
|
'symlink_path': '/v1/%s/%s/manifest-abcde' % (
|
||||||
self.account_name, self.env.container2.name),
|
self.account_name, self.env.container2.name),
|
||||||
'symlink_bytes': 4 * 2 ** 20 + 1,
|
'symlink_bytes': 4 * 2 ** 20 + 1,
|
||||||
'symlink_etag': manifest_etag,
|
'symlink_etag': normalize_etag(manifest_etag),
|
||||||
})
|
})
|
||||||
|
|
||||||
def test_static_link_target_slo_manifest_wrong_etag(self):
|
def test_static_link_target_slo_manifest_wrong_etag(self):
|
||||||
@ -1740,7 +1741,11 @@ class TestSymlinkToSloSegments(Base):
|
|||||||
self.assertEqual(1024 * 1024, f_dict['bytes'])
|
self.assertEqual(1024 * 1024, f_dict['bytes'])
|
||||||
self.assertEqual('application/octet-stream',
|
self.assertEqual('application/octet-stream',
|
||||||
f_dict['content_type'])
|
f_dict['content_type'])
|
||||||
self.assertEqual(manifest_etag, f_dict['hash'])
|
if tf.cluster_info.get('etag_quoter', {}).get(
|
||||||
|
'enable_by_default'):
|
||||||
|
self.assertEqual(manifest_etag, '"%s"' % f_dict['hash'])
|
||||||
|
else:
|
||||||
|
self.assertEqual(manifest_etag, f_dict['hash'])
|
||||||
self.assertEqual(slo_etag, f_dict['slo_etag'])
|
self.assertEqual(slo_etag, f_dict['slo_etag'])
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
@ -1759,7 +1764,11 @@ class TestSymlinkToSloSegments(Base):
|
|||||||
self.assertEqual(1024 * 1024, f_dict['bytes'])
|
self.assertEqual(1024 * 1024, f_dict['bytes'])
|
||||||
self.assertEqual(file_item.content_type,
|
self.assertEqual(file_item.content_type,
|
||||||
f_dict['content_type'])
|
f_dict['content_type'])
|
||||||
self.assertEqual(manifest_etag, f_dict['hash'])
|
if tf.cluster_info.get('etag_quoter', {}).get(
|
||||||
|
'enable_by_default'):
|
||||||
|
self.assertEqual(manifest_etag, '"%s"' % f_dict['hash'])
|
||||||
|
else:
|
||||||
|
self.assertEqual(manifest_etag, f_dict['hash'])
|
||||||
self.assertEqual(slo_etag, f_dict['slo_etag'])
|
self.assertEqual(slo_etag, f_dict['slo_etag'])
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
@ -1778,7 +1787,11 @@ class TestSymlinkToSloSegments(Base):
|
|||||||
self.assertEqual(1024 * 1024, f_dict['bytes'])
|
self.assertEqual(1024 * 1024, f_dict['bytes'])
|
||||||
self.assertEqual(file_item.content_type,
|
self.assertEqual(file_item.content_type,
|
||||||
f_dict['content_type'])
|
f_dict['content_type'])
|
||||||
self.assertEqual(manifest_etag, f_dict['hash'])
|
if tf.cluster_info.get('etag_quoter', {}).get(
|
||||||
|
'enable_by_default'):
|
||||||
|
self.assertEqual(manifest_etag, '"%s"' % f_dict['hash'])
|
||||||
|
else:
|
||||||
|
self.assertEqual(manifest_etag, f_dict['hash'])
|
||||||
self.assertEqual(slo_etag, f_dict['slo_etag'])
|
self.assertEqual(slo_etag, f_dict['slo_etag'])
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
@ -1811,6 +1824,8 @@ class TestSymlinkToSloSegments(Base):
|
|||||||
source_contents = source.read(parms={'multipart-manifest': 'get'})
|
source_contents = source.read(parms={'multipart-manifest': 'get'})
|
||||||
source_json = json.loads(source_contents)
|
source_json = json.loads(source_contents)
|
||||||
manifest_etag = hashlib.md5(source_contents).hexdigest()
|
manifest_etag = hashlib.md5(source_contents).hexdigest()
|
||||||
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
manifest_etag = '"%s"' % manifest_etag
|
||||||
|
|
||||||
source.initialize()
|
source.initialize()
|
||||||
slo_etag = source.etag
|
slo_etag = source.etag
|
||||||
@ -1857,14 +1872,20 @@ class TestSymlinkToSloSegments(Base):
|
|||||||
actual = names['manifest-linkto-ab']
|
actual = names['manifest-linkto-ab']
|
||||||
self.assertEqual(2 * 1024 * 1024, actual['bytes'])
|
self.assertEqual(2 * 1024 * 1024, actual['bytes'])
|
||||||
self.assertEqual('application/octet-stream', actual['content_type'])
|
self.assertEqual('application/octet-stream', actual['content_type'])
|
||||||
self.assertEqual(manifest_etag, actual['hash'])
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
self.assertEqual(manifest_etag, '"%s"' % actual['hash'])
|
||||||
|
else:
|
||||||
|
self.assertEqual(manifest_etag, actual['hash'])
|
||||||
self.assertEqual(slo_etag, actual['slo_etag'])
|
self.assertEqual(slo_etag, actual['slo_etag'])
|
||||||
|
|
||||||
self.assertIn('copied-ab-manifest-only', names)
|
self.assertIn('copied-ab-manifest-only', names)
|
||||||
actual = names['copied-ab-manifest-only']
|
actual = names['copied-ab-manifest-only']
|
||||||
self.assertEqual(2 * 1024 * 1024, actual['bytes'])
|
self.assertEqual(2 * 1024 * 1024, actual['bytes'])
|
||||||
self.assertEqual('application/octet-stream', actual['content_type'])
|
self.assertEqual('application/octet-stream', actual['content_type'])
|
||||||
self.assertEqual(manifest_etag, actual['hash'])
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
self.assertEqual(manifest_etag, '"%s"' % actual['hash'])
|
||||||
|
else:
|
||||||
|
self.assertEqual(manifest_etag, actual['hash'])
|
||||||
self.assertEqual(slo_etag, actual['slo_etag'])
|
self.assertEqual(slo_etag, actual['slo_etag'])
|
||||||
|
|
||||||
|
|
||||||
@ -2000,13 +2021,13 @@ class TestSymlinkTargetObjectComparison(Base):
|
|||||||
else:
|
else:
|
||||||
self.assertEqual(b'', body)
|
self.assertEqual(b'', body)
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
hdrs = {'If-Match': 'bogus'}
|
hdrs = {'If-Match': 'bogus'}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
def testIfMatchMultipleEtags(self):
|
def testIfMatchMultipleEtags(self):
|
||||||
for file_item in self.env.files:
|
for file_item in self.env.files:
|
||||||
@ -2022,13 +2043,13 @@ class TestSymlinkTargetObjectComparison(Base):
|
|||||||
else:
|
else:
|
||||||
self.assertEqual(b'', body)
|
self.assertEqual(b'', body)
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
hdrs = {'If-Match': '"bogus1", "bogus2", "bogus3"'}
|
hdrs = {'If-Match': '"bogus1", "bogus2", "bogus3"'}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
def testIfNoneMatch(self):
|
def testIfNoneMatch(self):
|
||||||
for file_item in self.env.files:
|
for file_item in self.env.files:
|
||||||
@ -2044,13 +2065,13 @@ class TestSymlinkTargetObjectComparison(Base):
|
|||||||
else:
|
else:
|
||||||
self.assertEqual(b'', body)
|
self.assertEqual(b'', body)
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
hdrs = {'If-None-Match': md5}
|
hdrs = {'If-None-Match': md5}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
def testIfNoneMatchMultipleEtags(self):
|
def testIfNoneMatchMultipleEtags(self):
|
||||||
@ -2067,14 +2088,14 @@ class TestSymlinkTargetObjectComparison(Base):
|
|||||||
else:
|
else:
|
||||||
self.assertEqual(b'', body)
|
self.assertEqual(b'', body)
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
hdrs = {'If-None-Match':
|
hdrs = {'If-None-Match':
|
||||||
'"bogus1", "bogus2", "%s"' % md5}
|
'"bogus1", "bogus2", "%s"' % md5}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
def testIfModifiedSince(self):
|
def testIfModifiedSince(self):
|
||||||
@ -2091,19 +2112,19 @@ class TestSymlinkTargetObjectComparison(Base):
|
|||||||
else:
|
else:
|
||||||
self.assertEqual(b'', body)
|
self.assertEqual(b'', body)
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
self.assertTrue(file_symlink.info(hdrs=hdrs, parms=self.env.parms))
|
self.assertTrue(file_symlink.info(hdrs=hdrs, parms=self.env.parms))
|
||||||
|
|
||||||
hdrs = {'If-Modified-Since': self.env.time_new}
|
hdrs = {'If-Modified-Since': self.env.time_new}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
self.assertRaises(ResponseError, file_symlink.info, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.info, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
def testIfUnmodifiedSince(self):
|
def testIfUnmodifiedSince(self):
|
||||||
@ -2120,18 +2141,18 @@ class TestSymlinkTargetObjectComparison(Base):
|
|||||||
else:
|
else:
|
||||||
self.assertEqual(b'', body)
|
self.assertEqual(b'', body)
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
self.assertTrue(file_symlink.info(hdrs=hdrs, parms=self.env.parms))
|
self.assertTrue(file_symlink.info(hdrs=hdrs, parms=self.env.parms))
|
||||||
|
|
||||||
hdrs = {'If-Unmodified-Since': self.env.time_old_f2}
|
hdrs = {'If-Unmodified-Since': self.env.time_old_f2}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
self.assertRaises(ResponseError, file_symlink.info, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.info, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
def testIfMatchAndUnmodified(self):
|
def testIfMatchAndUnmodified(self):
|
||||||
for file_item in self.env.files:
|
for file_item in self.env.files:
|
||||||
@ -2148,21 +2169,21 @@ class TestSymlinkTargetObjectComparison(Base):
|
|||||||
else:
|
else:
|
||||||
self.assertEqual(b'', body)
|
self.assertEqual(b'', body)
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
hdrs = {'If-Match': 'bogus',
|
hdrs = {'If-Match': 'bogus',
|
||||||
'If-Unmodified-Since': self.env.time_new}
|
'If-Unmodified-Since': self.env.time_new}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
hdrs = {'If-Match': md5,
|
hdrs = {'If-Match': md5,
|
||||||
'If-Unmodified-Since': self.env.time_old_f3}
|
'If-Unmodified-Since': self.env.time_old_f3}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
def testLastModified(self):
|
def testLastModified(self):
|
||||||
file_item = self.env.container.file(Utils.create_name())
|
file_item = self.env.container.file(Utils.create_name())
|
||||||
@ -2186,7 +2207,7 @@ class TestSymlinkTargetObjectComparison(Base):
|
|||||||
hdrs = {'If-Modified-Since': last_modified}
|
hdrs = {'If-Modified-Since': last_modified}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
hdrs = {'If-Unmodified-Since': last_modified}
|
hdrs = {'If-Unmodified-Since': last_modified}
|
||||||
@ -2227,20 +2248,20 @@ class TestSymlinkComparison(TestSymlinkTargetObjectComparison):
|
|||||||
body = file_symlink.read(hdrs=hdrs, parms=self.env.parms)
|
body = file_symlink.read(hdrs=hdrs, parms=self.env.parms)
|
||||||
self.assertEqual(b'', body)
|
self.assertEqual(b'', body)
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
hdrs = {'If-Modified-Since': last_modified}
|
hdrs = {'If-Modified-Since': last_modified}
|
||||||
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
self.assertRaises(ResponseError, file_symlink.read, hdrs=hdrs,
|
||||||
parms=self.env.parms)
|
parms=self.env.parms)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
hdrs = {'If-Unmodified-Since': last_modified}
|
hdrs = {'If-Unmodified-Since': last_modified}
|
||||||
body = file_symlink.read(hdrs=hdrs, parms=self.env.parms)
|
body = file_symlink.read(hdrs=hdrs, parms=self.env.parms)
|
||||||
self.assertEqual(b'', body)
|
self.assertEqual(b'', body)
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assert_header('etag', md5)
|
self.assert_etag(md5)
|
||||||
|
|
||||||
|
|
||||||
class TestSymlinkAccountTempurl(Base):
|
class TestSymlinkAccountTempurl(Base):
|
||||||
|
@ -684,7 +684,11 @@ class TestObjectVersioning(Base):
|
|||||||
prev_version = versions_container.file(versioned_obj_name)
|
prev_version = versions_container.file(versioned_obj_name)
|
||||||
prev_version_info = prev_version.info(parms={'symlink': 'get'})
|
prev_version_info = prev_version.info(parms={'symlink': 'get'})
|
||||||
self.assertEqual(b"aaaaa", prev_version.read())
|
self.assertEqual(b"aaaaa", prev_version.read())
|
||||||
self.assertEqual(MD5_OF_EMPTY_STRING, prev_version_info['etag'])
|
symlink_etag = prev_version_info['etag']
|
||||||
|
if symlink_etag.startswith('"') and symlink_etag.endswith('"') and \
|
||||||
|
symlink_etag[1:-1]:
|
||||||
|
symlink_etag = symlink_etag[1:-1]
|
||||||
|
self.assertEqual(MD5_OF_EMPTY_STRING, symlink_etag)
|
||||||
self.assertEqual(sym_tgt_header,
|
self.assertEqual(sym_tgt_header,
|
||||||
prev_version_info['x_symlink_target'])
|
prev_version_info['x_symlink_target'])
|
||||||
return symlink, tgt_a
|
return symlink, tgt_a
|
||||||
@ -698,7 +702,10 @@ class TestObjectVersioning(Base):
|
|||||||
symlink.delete()
|
symlink.delete()
|
||||||
sym_info = symlink.info(parms={'symlink': 'get'})
|
sym_info = symlink.info(parms={'symlink': 'get'})
|
||||||
self.assertEqual(b"aaaaa", symlink.read())
|
self.assertEqual(b"aaaaa", symlink.read())
|
||||||
self.assertEqual(MD5_OF_EMPTY_STRING, sym_info['etag'])
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
self.assertEqual('"%s"' % MD5_OF_EMPTY_STRING, sym_info['etag'])
|
||||||
|
else:
|
||||||
|
self.assertEqual(MD5_OF_EMPTY_STRING, sym_info['etag'])
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
quote(unquote('%s/%s' % (self.env.container.name, target.name))),
|
quote(unquote('%s/%s' % (self.env.container.name, target.name))),
|
||||||
sym_info['x_symlink_target'])
|
sym_info['x_symlink_target'])
|
||||||
|
@ -27,6 +27,7 @@ import uuid
|
|||||||
from copy import deepcopy
|
from copy import deepcopy
|
||||||
import eventlet
|
import eventlet
|
||||||
from swift.common.http import is_success, is_client_error
|
from swift.common.http import is_success, is_client_error
|
||||||
|
from swift.common.swob import normalize_etag
|
||||||
from email.utils import parsedate
|
from email.utils import parsedate
|
||||||
|
|
||||||
if six.PY2:
|
if six.PY2:
|
||||||
@ -131,6 +132,13 @@ class Base(unittest.TestCase):
|
|||||||
'Expected header name %r not found in response.' % header_name)
|
'Expected header name %r not found in response.' % header_name)
|
||||||
self.assertEqual(expected_value, actual_value)
|
self.assertEqual(expected_value, actual_value)
|
||||||
|
|
||||||
|
def assert_etag(self, unquoted_value):
|
||||||
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
expected = '"%s"' % unquoted_value
|
||||||
|
else:
|
||||||
|
expected = unquoted_value
|
||||||
|
self.assert_header('etag', expected)
|
||||||
|
|
||||||
|
|
||||||
class Base2(object):
|
class Base2(object):
|
||||||
@classmethod
|
@classmethod
|
||||||
@ -874,7 +882,11 @@ class TestContainer(Base):
|
|||||||
for actual in file_list:
|
for actual in file_list:
|
||||||
name = actual['name']
|
name = actual['name']
|
||||||
self.assertIn(name, expected)
|
self.assertIn(name, expected)
|
||||||
self.assertEqual(expected[name]['etag'], actual['hash'])
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
self.assertEqual(expected[name]['etag'],
|
||||||
|
'"%s"' % actual['hash'])
|
||||||
|
else:
|
||||||
|
self.assertEqual(expected[name]['etag'], actual['hash'])
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
expected[name]['content_type'], actual['content_type'])
|
expected[name]['content_type'], actual['content_type'])
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
@ -1365,6 +1377,8 @@ class TestFile(Base):
|
|||||||
'x-delete-at': mock.ANY,
|
'x-delete-at': mock.ANY,
|
||||||
'x-trans-id': mock.ANY,
|
'x-trans-id': mock.ANY,
|
||||||
'x-openstack-request-id': mock.ANY}
|
'x-openstack-request-id': mock.ANY}
|
||||||
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
expected_headers['etag'] = '"%s"' % expected_headers['etag']
|
||||||
unexpected_headers = ['connection', 'x-delete-after']
|
unexpected_headers = ['connection', 'x-delete-after']
|
||||||
do_test(put_headers, {}, expected_headers, unexpected_headers)
|
do_test(put_headers, {}, expected_headers, unexpected_headers)
|
||||||
|
|
||||||
@ -1420,7 +1434,7 @@ class TestFile(Base):
|
|||||||
self.fail('Failed to find %s in listing' % dest_filename)
|
self.fail('Failed to find %s in listing' % dest_filename)
|
||||||
|
|
||||||
self.assertEqual(file_item.size, obj['bytes'])
|
self.assertEqual(file_item.size, obj['bytes'])
|
||||||
self.assertEqual(file_item.etag, obj['hash'])
|
self.assertEqual(normalize_etag(file_item.etag), obj['hash'])
|
||||||
self.assertEqual(file_item.content_type, obj['content_type'])
|
self.assertEqual(file_item.content_type, obj['content_type'])
|
||||||
|
|
||||||
file_copy = cont.file(dest_filename)
|
file_copy = cont.file(dest_filename)
|
||||||
@ -1470,7 +1484,7 @@ class TestFile(Base):
|
|||||||
self.fail('Failed to find %s in listing' % dest_filename)
|
self.fail('Failed to find %s in listing' % dest_filename)
|
||||||
|
|
||||||
self.assertEqual(file_item.size, obj['bytes'])
|
self.assertEqual(file_item.size, obj['bytes'])
|
||||||
self.assertEqual(file_item.etag, obj['hash'])
|
self.assertEqual(normalize_etag(file_item.etag), obj['hash'])
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
'application/test-changed', obj['content_type'])
|
'application/test-changed', obj['content_type'])
|
||||||
|
|
||||||
@ -1505,7 +1519,7 @@ class TestFile(Base):
|
|||||||
self.fail('Failed to find %s in listing' % dest_filename)
|
self.fail('Failed to find %s in listing' % dest_filename)
|
||||||
|
|
||||||
self.assertEqual(file_item.size, obj['bytes'])
|
self.assertEqual(file_item.size, obj['bytes'])
|
||||||
self.assertEqual(file_item.etag, obj['hash'])
|
self.assertEqual(normalize_etag(file_item.etag), obj['hash'])
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
'application/test-updated', obj['content_type'])
|
'application/test-updated', obj['content_type'])
|
||||||
|
|
||||||
@ -2088,7 +2102,7 @@ class TestFile(Base):
|
|||||||
self.assertEqual(file_item.read(hdrs=hdrs), data[-i:])
|
self.assertEqual(file_item.read(hdrs=hdrs), data[-i:])
|
||||||
self.assert_header('content-range', 'bytes %d-%d/%d' % (
|
self.assert_header('content-range', 'bytes %d-%d/%d' % (
|
||||||
file_length - i, file_length - 1, file_length))
|
file_length - i, file_length - 1, file_length))
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
range_string = 'bytes=%d-' % (i)
|
range_string = 'bytes=%d-' % (i)
|
||||||
@ -2102,7 +2116,7 @@ class TestFile(Base):
|
|||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(416)
|
self.assert_status(416)
|
||||||
self.assert_header('content-range', 'bytes */%d' % file_length)
|
self.assert_header('content-range', 'bytes */%d' % file_length)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
range_string = 'bytes=%d-%d' % (file_length - 1000, file_length + 2000)
|
range_string = 'bytes=%d-%d' % (file_length - 1000, file_length + 2000)
|
||||||
@ -2416,14 +2430,16 @@ class TestFile(Base):
|
|||||||
file_item.content_type = content_type
|
file_item.content_type = content_type
|
||||||
file_item.write_random(self.env.file_size)
|
file_item.write_random(self.env.file_size)
|
||||||
|
|
||||||
md5 = file_item.md5
|
expected_etag = file_item.md5
|
||||||
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
expected_etag = '"%s"' % expected_etag
|
||||||
|
|
||||||
file_item = self.env.container.file(file_name)
|
file_item = self.env.container.file(file_name)
|
||||||
info = file_item.info()
|
info = file_item.info()
|
||||||
|
|
||||||
self.assert_status(200)
|
self.assert_status(200)
|
||||||
self.assertEqual(info['content_length'], self.env.file_size)
|
self.assertEqual(info['content_length'], self.env.file_size)
|
||||||
self.assertEqual(info['etag'], md5)
|
self.assertEqual(info['etag'], expected_etag)
|
||||||
self.assertEqual(info['content_type'], content_type)
|
self.assertEqual(info['content_type'], content_type)
|
||||||
self.assertIn('last_modified', info)
|
self.assertIn('last_modified', info)
|
||||||
|
|
||||||
@ -2612,14 +2628,7 @@ class TestFile(Base):
|
|||||||
file_item = self.env.container.file(Utils.create_name())
|
file_item = self.env.container.file(Utils.create_name())
|
||||||
|
|
||||||
data = io.BytesIO(file_item.write_random(512))
|
data = io.BytesIO(file_item.write_random(512))
|
||||||
etag = File.compute_md5sum(data)
|
self.assert_etag(File.compute_md5sum(data))
|
||||||
|
|
||||||
headers = dict((h.lower(), v)
|
|
||||||
for h, v in self.env.conn.response.getheaders())
|
|
||||||
self.assertIn('etag', headers.keys())
|
|
||||||
|
|
||||||
header_etag = headers['etag'].strip('"')
|
|
||||||
self.assertEqual(etag, header_etag)
|
|
||||||
|
|
||||||
def testChunkedPut(self):
|
def testChunkedPut(self):
|
||||||
if (tf.web_front_end == 'apache2'):
|
if (tf.web_front_end == 'apache2'):
|
||||||
@ -2645,7 +2654,7 @@ class TestFile(Base):
|
|||||||
self.assertEqual(data, file_item.read())
|
self.assertEqual(data, file_item.read())
|
||||||
|
|
||||||
info = file_item.info()
|
info = file_item.info()
|
||||||
self.assertEqual(etag, info['etag'])
|
self.assertEqual(normalize_etag(info['etag']), etag)
|
||||||
|
|
||||||
def test_POST(self):
|
def test_POST(self):
|
||||||
# verify consistency between object and container listing metadata
|
# verify consistency between object and container listing metadata
|
||||||
@ -2670,7 +2679,10 @@ class TestFile(Base):
|
|||||||
self.fail('Failed to find file %r in listing' % file_name)
|
self.fail('Failed to find file %r in listing' % file_name)
|
||||||
self.assertEqual(1024, f_dict['bytes'])
|
self.assertEqual(1024, f_dict['bytes'])
|
||||||
self.assertEqual('text/foobar', f_dict['content_type'])
|
self.assertEqual('text/foobar', f_dict['content_type'])
|
||||||
self.assertEqual(etag, f_dict['hash'])
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
self.assertEqual(etag, '"%s"' % f_dict['hash'])
|
||||||
|
else:
|
||||||
|
self.assertEqual(etag, f_dict['hash'])
|
||||||
put_last_modified = f_dict['last_modified']
|
put_last_modified = f_dict['last_modified']
|
||||||
|
|
||||||
# now POST updated content-type to each file
|
# now POST updated content-type to each file
|
||||||
@ -2697,7 +2709,10 @@ class TestFile(Base):
|
|||||||
self.assertEqual(1024, f_dict['bytes'])
|
self.assertEqual(1024, f_dict['bytes'])
|
||||||
self.assertEqual('image/foobarbaz', f_dict['content_type'])
|
self.assertEqual('image/foobarbaz', f_dict['content_type'])
|
||||||
self.assertLess(put_last_modified, f_dict['last_modified'])
|
self.assertLess(put_last_modified, f_dict['last_modified'])
|
||||||
self.assertEqual(etag, f_dict['hash'])
|
if tf.cluster_info.get('etag_quoter', {}).get('enable_by_default'):
|
||||||
|
self.assertEqual(etag, '"%s"' % f_dict['hash'])
|
||||||
|
else:
|
||||||
|
self.assertEqual(etag, f_dict['hash'])
|
||||||
|
|
||||||
|
|
||||||
class TestFileUTF8(Base2, TestFile):
|
class TestFileUTF8(Base2, TestFile):
|
||||||
@ -2742,7 +2757,7 @@ class TestFileComparison(Base):
|
|||||||
hdrs = {'If-Match': 'bogus'}
|
hdrs = {'If-Match': 'bogus'}
|
||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
|
|
||||||
def testIfMatchMultipleEtags(self):
|
def testIfMatchMultipleEtags(self):
|
||||||
for file_item in self.env.files:
|
for file_item in self.env.files:
|
||||||
@ -2752,7 +2767,7 @@ class TestFileComparison(Base):
|
|||||||
hdrs = {'If-Match': '"bogus1", "bogus2", "bogus3"'}
|
hdrs = {'If-Match': '"bogus1", "bogus2", "bogus3"'}
|
||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
|
|
||||||
def testIfNoneMatch(self):
|
def testIfNoneMatch(self):
|
||||||
for file_item in self.env.files:
|
for file_item in self.env.files:
|
||||||
@ -2762,7 +2777,7 @@ class TestFileComparison(Base):
|
|||||||
hdrs = {'If-None-Match': file_item.md5}
|
hdrs = {'If-None-Match': file_item.md5}
|
||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
def testIfNoneMatchMultipleEtags(self):
|
def testIfNoneMatchMultipleEtags(self):
|
||||||
@ -2774,7 +2789,7 @@ class TestFileComparison(Base):
|
|||||||
'"bogus1", "bogus2", "%s"' % file_item.md5}
|
'"bogus1", "bogus2", "%s"' % file_item.md5}
|
||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
def testIfModifiedSince(self):
|
def testIfModifiedSince(self):
|
||||||
@ -2786,11 +2801,11 @@ class TestFileComparison(Base):
|
|||||||
hdrs = {'If-Modified-Since': self.env.time_new}
|
hdrs = {'If-Modified-Since': self.env.time_new}
|
||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
self.assertRaises(ResponseError, file_item.info, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.info, hdrs=hdrs)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
def testIfUnmodifiedSince(self):
|
def testIfUnmodifiedSince(self):
|
||||||
@ -2802,10 +2817,10 @@ class TestFileComparison(Base):
|
|||||||
hdrs = {'If-Unmodified-Since': self.env.time_old_f2}
|
hdrs = {'If-Unmodified-Since': self.env.time_old_f2}
|
||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
self.assertRaises(ResponseError, file_item.info, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.info, hdrs=hdrs)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
|
|
||||||
def testIfMatchAndUnmodified(self):
|
def testIfMatchAndUnmodified(self):
|
||||||
for file_item in self.env.files:
|
for file_item in self.env.files:
|
||||||
@ -2817,13 +2832,13 @@ class TestFileComparison(Base):
|
|||||||
'If-Unmodified-Since': self.env.time_new}
|
'If-Unmodified-Since': self.env.time_new}
|
||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
|
|
||||||
hdrs = {'If-Match': file_item.md5,
|
hdrs = {'If-Match': file_item.md5,
|
||||||
'If-Unmodified-Since': self.env.time_old_f3}
|
'If-Unmodified-Since': self.env.time_old_f3}
|
||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(412)
|
self.assert_status(412)
|
||||||
self.assert_header('etag', file_item.md5)
|
self.assert_etag(file_item.md5)
|
||||||
|
|
||||||
def testLastModified(self):
|
def testLastModified(self):
|
||||||
file_name = Utils.create_name()
|
file_name = Utils.create_name()
|
||||||
@ -2844,7 +2859,7 @@ class TestFileComparison(Base):
|
|||||||
hdrs = {'If-Modified-Since': last_modified}
|
hdrs = {'If-Modified-Since': last_modified}
|
||||||
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
self.assertRaises(ResponseError, file_item.read, hdrs=hdrs)
|
||||||
self.assert_status(304)
|
self.assert_status(304)
|
||||||
self.assert_header('etag', etag)
|
self.assert_etag(etag)
|
||||||
self.assert_header('accept-ranges', 'bytes')
|
self.assert_header('accept-ranges', 'bytes')
|
||||||
|
|
||||||
hdrs = {'If-Unmodified-Since': last_modified}
|
hdrs = {'If-Unmodified-Since': last_modified}
|
||||||
|
@ -23,6 +23,7 @@ from swiftclient.exceptions import ClientException
|
|||||||
|
|
||||||
from swift.common import direct_client
|
from swift.common import direct_client
|
||||||
from swift.common.manager import Manager
|
from swift.common.manager import Manager
|
||||||
|
from swift.common.swob import normalize_etag
|
||||||
from test.probe.common import kill_nonprimary_server, \
|
from test.probe.common import kill_nonprimary_server, \
|
||||||
kill_server, ReplProbeTest, start_server, ECProbeTest
|
kill_server, ReplProbeTest, start_server, ECProbeTest
|
||||||
|
|
||||||
@ -210,7 +211,7 @@ class TestUpdateOverridesEC(ECProbeTest):
|
|||||||
self.assertEqual(1, len(listing))
|
self.assertEqual(1, len(listing))
|
||||||
self.assertEqual('o1', listing[0]['name'])
|
self.assertEqual('o1', listing[0]['name'])
|
||||||
self.assertEqual(len(content), listing[0]['bytes'])
|
self.assertEqual(len(content), listing[0]['bytes'])
|
||||||
self.assertEqual(meta['etag'], listing[0]['hash'])
|
self.assertEqual(normalize_etag(meta['etag']), listing[0]['hash'])
|
||||||
self.assertEqual('test/ctype', listing[0]['content_type'])
|
self.assertEqual('test/ctype', listing[0]['content_type'])
|
||||||
|
|
||||||
def test_update_during_POST_only(self):
|
def test_update_during_POST_only(self):
|
||||||
@ -261,7 +262,7 @@ class TestUpdateOverridesEC(ECProbeTest):
|
|||||||
self.assertEqual(1, len(listing))
|
self.assertEqual(1, len(listing))
|
||||||
self.assertEqual('o1', listing[0]['name'])
|
self.assertEqual('o1', listing[0]['name'])
|
||||||
self.assertEqual(len(content), listing[0]['bytes'])
|
self.assertEqual(len(content), listing[0]['bytes'])
|
||||||
self.assertEqual(meta['etag'], listing[0]['hash'])
|
self.assertEqual(normalize_etag(meta['etag']), listing[0]['hash'])
|
||||||
self.assertEqual('test/ctype', listing[0]['content_type'])
|
self.assertEqual('test/ctype', listing[0]['content_type'])
|
||||||
|
|
||||||
# Run the object-updaters to send the async pending from the PUT
|
# Run the object-updaters to send the async pending from the PUT
|
||||||
@ -328,7 +329,7 @@ class TestUpdateOverridesEC(ECProbeTest):
|
|||||||
self.assertEqual(1, len(listing))
|
self.assertEqual(1, len(listing))
|
||||||
self.assertEqual('o1', listing[0]['name'])
|
self.assertEqual('o1', listing[0]['name'])
|
||||||
self.assertEqual(len(content), listing[0]['bytes'])
|
self.assertEqual(len(content), listing[0]['bytes'])
|
||||||
self.assertEqual(meta['etag'], listing[0]['hash'])
|
self.assertEqual(normalize_etag(meta['etag']), listing[0]['hash'])
|
||||||
self.assertEqual('test/ctype', listing[0]['content_type'])
|
self.assertEqual('test/ctype', listing[0]['content_type'])
|
||||||
|
|
||||||
|
|
||||||
|
@ -22,6 +22,7 @@ import uuid
|
|||||||
from swift.common.direct_client import direct_get_suffix_hashes
|
from swift.common.direct_client import direct_get_suffix_hashes
|
||||||
from swift.common.exceptions import DiskFileDeleted
|
from swift.common.exceptions import DiskFileDeleted
|
||||||
from swift.common.internal_client import UnexpectedResponse
|
from swift.common.internal_client import UnexpectedResponse
|
||||||
|
from swift.common.swob import normalize_etag
|
||||||
from swift.container.backend import ContainerBroker
|
from swift.container.backend import ContainerBroker
|
||||||
from swift.common import utils
|
from swift.common import utils
|
||||||
from swiftclient import client
|
from swiftclient import client
|
||||||
@ -129,7 +130,7 @@ class Test(ReplProbeTest):
|
|||||||
|
|
||||||
def _assert_object_metadata_matches_listing(self, listing, metadata):
|
def _assert_object_metadata_matches_listing(self, listing, metadata):
|
||||||
self.assertEqual(listing['bytes'], int(metadata['content-length']))
|
self.assertEqual(listing['bytes'], int(metadata['content-length']))
|
||||||
self.assertEqual(listing['hash'], metadata['etag'])
|
self.assertEqual(listing['hash'], normalize_etag(metadata['etag']))
|
||||||
self.assertEqual(listing['content_type'], metadata['content-type'])
|
self.assertEqual(listing['content_type'], metadata['content-type'])
|
||||||
modified = Timestamp(metadata['x-timestamp']).isoformat
|
modified = Timestamp(metadata['x-timestamp']).isoformat
|
||||||
self.assertEqual(listing['last_modified'], modified)
|
self.assertEqual(listing['last_modified'], modified)
|
||||||
|
Loading…
Reference in New Issue
Block a user