py3: Fix title-casing in HeaderKeyDict
Change-Id: I1152c47c52f6482ec877142c96845b00bf6dcc5b Related-Change: I130ba5014b7eff458d87ab29eb42fe45607c9a12
This commit is contained in:
parent
d6e790d1b7
commit
60c27d3aef
@ -16,10 +16,20 @@
|
|||||||
import six
|
import six
|
||||||
|
|
||||||
|
|
||||||
|
def _title(s):
|
||||||
|
if six.PY2:
|
||||||
|
return s.title()
|
||||||
|
else:
|
||||||
|
return s.encode('latin1').title().decode('latin1')
|
||||||
|
|
||||||
|
|
||||||
class HeaderKeyDict(dict):
|
class HeaderKeyDict(dict):
|
||||||
"""
|
"""
|
||||||
A dict that title-cases all keys on the way in, so as to be
|
A dict that title-cases all keys on the way in, so as to be
|
||||||
case-insensitive.
|
case-insensitive.
|
||||||
|
|
||||||
|
Note that all keys and values are expected to be wsgi strings,
|
||||||
|
though some allowances are made when setting values.
|
||||||
"""
|
"""
|
||||||
def __init__(self, base_headers=None, **kwargs):
|
def __init__(self, base_headers=None, **kwargs):
|
||||||
if base_headers:
|
if base_headers:
|
||||||
@ -29,32 +39,32 @@ class HeaderKeyDict(dict):
|
|||||||
def update(self, other):
|
def update(self, other):
|
||||||
if hasattr(other, 'keys'):
|
if hasattr(other, 'keys'):
|
||||||
for key in other.keys():
|
for key in other.keys():
|
||||||
self[key.title()] = other[key]
|
self[_title(key)] = other[key]
|
||||||
else:
|
else:
|
||||||
for key, value in other:
|
for key, value in other:
|
||||||
self[key.title()] = value
|
self[_title(key)] = value
|
||||||
|
|
||||||
def __getitem__(self, key):
|
def __getitem__(self, key):
|
||||||
return dict.get(self, key.title())
|
return dict.get(self, _title(key))
|
||||||
|
|
||||||
def __setitem__(self, key, value):
|
def __setitem__(self, key, value):
|
||||||
if value is None:
|
if value is None:
|
||||||
self.pop(key.title(), None)
|
self.pop(_title(key), None)
|
||||||
elif six.PY2 and isinstance(value, six.text_type):
|
elif six.PY2 and isinstance(value, six.text_type):
|
||||||
return dict.__setitem__(self, key.title(), value.encode('utf-8'))
|
return dict.__setitem__(self, _title(key), value.encode('utf-8'))
|
||||||
elif six.PY3 and isinstance(value, six.binary_type):
|
elif six.PY3 and isinstance(value, six.binary_type):
|
||||||
return dict.__setitem__(self, key.title(), value.decode('latin-1'))
|
return dict.__setitem__(self, _title(key), value.decode('latin-1'))
|
||||||
else:
|
else:
|
||||||
return dict.__setitem__(self, key.title(), str(value))
|
return dict.__setitem__(self, _title(key), str(value))
|
||||||
|
|
||||||
def __contains__(self, key):
|
def __contains__(self, key):
|
||||||
return dict.__contains__(self, key.title())
|
return dict.__contains__(self, _title(key))
|
||||||
|
|
||||||
def __delitem__(self, key):
|
def __delitem__(self, key):
|
||||||
return dict.__delitem__(self, key.title())
|
return dict.__delitem__(self, _title(key))
|
||||||
|
|
||||||
def get(self, key, default=None):
|
def get(self, key, default=None):
|
||||||
return dict.get(self, key.title(), default)
|
return dict.get(self, _title(key), default)
|
||||||
|
|
||||||
def setdefault(self, key, value=None):
|
def setdefault(self, key, value=None):
|
||||||
if key not in self:
|
if key not in self:
|
||||||
@ -62,4 +72,4 @@ class HeaderKeyDict(dict):
|
|||||||
return self[key]
|
return self[key]
|
||||||
|
|
||||||
def pop(self, key, default=None):
|
def pop(self, key, default=None):
|
||||||
return dict.pop(self, key.title(), default)
|
return dict.pop(self, _title(key), default)
|
||||||
|
@ -133,7 +133,7 @@ class ObjectController(Controller):
|
|||||||
|
|
||||||
# delete object metadata from response
|
# delete object metadata from response
|
||||||
for key in list(resp.headers.keys()):
|
for key in list(resp.headers.keys()):
|
||||||
if key.startswith('x-amz-meta-'):
|
if key.lower().startswith('x-amz-meta-'):
|
||||||
del resp.headers[key]
|
del resp.headers[key]
|
||||||
|
|
||||||
resp.status = HTTP_OK
|
resp.status = HTTP_OK
|
||||||
|
@ -582,9 +582,9 @@ class TestS3ApiObj(S3ApiTestCase):
|
|||||||
self.assertEqual('200 ', status[:4], body)
|
self.assertEqual('200 ', status[:4], body)
|
||||||
# Check that s3api does not return an etag header,
|
# Check that s3api does not return an etag header,
|
||||||
# specified copy source.
|
# specified copy source.
|
||||||
self.assertTrue(headers.get('etag') is None)
|
self.assertNotIn('etag', headers)
|
||||||
# Check that s3api does not return custom metadata in response
|
# Check that s3api does not return custom metadata in response
|
||||||
self.assertTrue(headers.get('x-amz-meta-something') is None)
|
self.assertNotIn('x-amz-meta-something', headers)
|
||||||
|
|
||||||
_, _, headers = self.swift.calls_with_headers[-1]
|
_, _, headers = self.swift.calls_with_headers[-1]
|
||||||
# Check that s3api converts a Content-MD5 header into an etag.
|
# Check that s3api converts a Content-MD5 header into an etag.
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
|
|
||||||
import unittest
|
import unittest
|
||||||
from swift.common.header_key_dict import HeaderKeyDict
|
from swift.common.header_key_dict import HeaderKeyDict
|
||||||
|
from swift.common.swob import bytes_to_wsgi
|
||||||
|
|
||||||
|
|
||||||
class TestHeaderKeyDict(unittest.TestCase):
|
class TestHeaderKeyDict(unittest.TestCase):
|
||||||
@ -27,6 +28,20 @@ class TestHeaderKeyDict(unittest.TestCase):
|
|||||||
self.assertEqual(headers['content-length'], '20')
|
self.assertEqual(headers['content-length'], '20')
|
||||||
self.assertEqual(headers['CONTENT-LENGTH'], '20')
|
self.assertEqual(headers['CONTENT-LENGTH'], '20')
|
||||||
|
|
||||||
|
def test_unicode(self):
|
||||||
|
def mkstr(prefix):
|
||||||
|
return bytes_to_wsgi((prefix + u'\U0001f44d').encode('utf8'))
|
||||||
|
|
||||||
|
headers = HeaderKeyDict()
|
||||||
|
headers[mkstr('x-object-meta-')] = 'ok'
|
||||||
|
self.assertIn(mkstr('x-object-meta-'), headers)
|
||||||
|
self.assertIn(mkstr('X-Object-Meta-'), headers)
|
||||||
|
self.assertIn(mkstr('X-OBJECT-META-'), headers)
|
||||||
|
keys = list(headers)
|
||||||
|
self.assertNotIn(mkstr('x-object-meta-'), keys)
|
||||||
|
self.assertIn(mkstr('X-Object-Meta-'), keys)
|
||||||
|
self.assertNotIn(mkstr('X-OBJECT-META-'), keys)
|
||||||
|
|
||||||
def test_setdefault(self):
|
def test_setdefault(self):
|
||||||
headers = HeaderKeyDict()
|
headers = HeaderKeyDict()
|
||||||
|
|
||||||
|
@ -38,6 +38,7 @@ from swift.obj.diskfile import (
|
|||||||
from swift.common.ring import RingData
|
from swift.common.ring import RingData
|
||||||
from swift.common import utils
|
from swift.common import utils
|
||||||
from swift.common.header_key_dict import HeaderKeyDict
|
from swift.common.header_key_dict import HeaderKeyDict
|
||||||
|
from swift.common.swob import bytes_to_wsgi
|
||||||
from swift.common.utils import (
|
from swift.common.utils import (
|
||||||
hash_path, normalize_timestamp, mkdirs, write_pickle)
|
hash_path, normalize_timestamp, mkdirs, write_pickle)
|
||||||
from swift.common.storage_policy import StoragePolicy, POLICIES
|
from swift.common.storage_policy import StoragePolicy, POLICIES
|
||||||
@ -504,13 +505,13 @@ class TestObjectUpdater(unittest.TestCase):
|
|||||||
self.assertEqual(inc.readline(),
|
self.assertEqual(inc.readline(),
|
||||||
b'PUT /sda1/0/a/c/o HTTP/1.1\r\n')
|
b'PUT /sda1/0/a/c/o HTTP/1.1\r\n')
|
||||||
headers = HeaderKeyDict()
|
headers = HeaderKeyDict()
|
||||||
line = inc.readline()
|
line = bytes_to_wsgi(inc.readline())
|
||||||
while line and line != b'\r\n':
|
while line and line != '\r\n':
|
||||||
headers[line.split(b':')[0]] = \
|
headers[line.split(':')[0]] = \
|
||||||
line.split(b':')[1].strip()
|
line.split(':')[1].strip()
|
||||||
line = inc.readline()
|
line = bytes_to_wsgi(inc.readline())
|
||||||
self.assertIn(b'x-container-timestamp', headers)
|
self.assertIn('x-container-timestamp', headers)
|
||||||
self.assertIn(b'X-Backend-Storage-Policy-Index',
|
self.assertIn('X-Backend-Storage-Policy-Index',
|
||||||
headers)
|
headers)
|
||||||
except BaseException as err:
|
except BaseException as err:
|
||||||
return err
|
return err
|
||||||
|
Loading…
x
Reference in New Issue
Block a user