py3: Fix title-casing in HeaderKeyDict

Change-Id: I1152c47c52f6482ec877142c96845b00bf6dcc5b
Related-Change: I130ba5014b7eff458d87ab29eb42fe45607c9a12
This commit is contained in:
Tim Burke 2019-07-25 10:56:53 -07:00
parent d6e790d1b7
commit 60c27d3aef
5 changed files with 47 additions and 21 deletions

View File

@ -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)

View File

@ -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

View File

@ -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.

View File

@ -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()

View File

@ -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