Commit 65aa5730 authored by Christian Heimes's avatar Christian Heimes

Add more tests for hashlib and hash object attributes

parent 4fec4314
...@@ -96,10 +96,14 @@ class HashLibTestCase(unittest.TestCase): ...@@ -96,10 +96,14 @@ class HashLibTestCase(unittest.TestCase):
super(HashLibTestCase, self).__init__(*args, **kwargs) super(HashLibTestCase, self).__init__(*args, **kwargs)
@property
def hash_constructors(self):
constructors = self.constructors_to_test.values()
return itertools.chain.from_iterable(constructors)
def test_hash_array(self): def test_hash_array(self):
a = array.array("b", range(10)) a = array.array("b", range(10))
constructors = self.constructors_to_test.values() for cons in self.hash_constructors:
for cons in itertools.chain.from_iterable(constructors):
c = cons(a) c = cons(a)
c.hexdigest() c.hexdigest()
...@@ -136,39 +140,57 @@ class HashLibTestCase(unittest.TestCase): ...@@ -136,39 +140,57 @@ class HashLibTestCase(unittest.TestCase):
self.assertRaises(TypeError, get_builtin_constructor, 3) self.assertRaises(TypeError, get_builtin_constructor, 3)
def test_hexdigest(self): def test_hexdigest(self):
for name in self.supported_hash_names: for cons in self.hash_constructors:
h = hashlib.new(name) h = cons()
assert isinstance(h.digest(), bytes), name assert isinstance(h.digest(), bytes), name
self.assertEqual(hexstr(h.digest()), h.hexdigest()) self.assertEqual(hexstr(h.digest()), h.hexdigest())
def test_large_update(self): def test_large_update(self):
aas = b'a' * 128 aas = b'a' * 128
bees = b'b' * 127 bees = b'b' * 127
cees = b'c' * 126 cees = b'c' * 126
dees = b'd' * 2048 # HASHLIB_GIL_MINSIZE
for name in self.supported_hash_names: for cons in self.hash_constructors:
m1 = hashlib.new(name) m1 = cons()
m1.update(aas) m1.update(aas)
m1.update(bees) m1.update(bees)
m1.update(cees) m1.update(cees)
m1.update(dees)
m2 = hashlib.new(name) m2 = cons()
m2.update(aas + bees + cees) m2.update(aas + bees + cees + dees)
self.assertEqual(m1.digest(), m2.digest()) self.assertEqual(m1.digest(), m2.digest())
def check(self, name, data, digest): m3 = cons(aas + bees + cees + dees)
self.assertEqual(m1.digest(), m3.digest())
# verify copy() doesn't touch original
m4 = cons(aas + bees + cees)
m4_digest = m4.digest()
m4_copy = m4.copy()
m4_copy.update(dees)
self.assertEqual(m1.digest(), m4_copy.digest())
self.assertEqual(m4.digest(), m4_digest)
def check(self, name, data, hexdigest):
hexdigest = hexdigest.lower()
constructors = self.constructors_to_test[name] constructors = self.constructors_to_test[name]
# 2 is for hashlib.name(...) and hashlib.new(name, ...) # 2 is for hashlib.name(...) and hashlib.new(name, ...)
self.assertGreaterEqual(len(constructors), 2) self.assertGreaterEqual(len(constructors), 2)
for hash_object_constructor in constructors: for hash_object_constructor in constructors:
computed = hash_object_constructor(data).hexdigest() m = hash_object_constructor(data)
computed = m.hexdigest()
self.assertEqual( self.assertEqual(
computed, digest, computed, hexdigest,
"Hash algorithm %s constructed using %s returned hexdigest" "Hash algorithm %s constructed using %s returned hexdigest"
" %r for %d byte input data that should have hashed to %r." " %r for %d byte input data that should have hashed to %r."
% (name, hash_object_constructor, % (name, hash_object_constructor,
computed, len(data), digest)) computed, len(data), hexdigest))
computed = m.digest()
digest = bytes.fromhex(hexdigest)
self.assertEqual(computed, digest)
self.assertEqual(len(digest), m.digest_size)
def check_no_unicode(self, algorithm_name): def check_no_unicode(self, algorithm_name):
# Unicode objects are not allowed as input. # Unicode objects are not allowed as input.
...@@ -184,6 +206,24 @@ class HashLibTestCase(unittest.TestCase): ...@@ -184,6 +206,24 @@ class HashLibTestCase(unittest.TestCase):
self.check_no_unicode('sha384') self.check_no_unicode('sha384')
self.check_no_unicode('sha512') self.check_no_unicode('sha512')
def check_blocksize_name(self, name, block_size=0, digest_size=0):
constructors = self.constructors_to_test[name]
for hash_object_constructor in constructors:
m = hash_object_constructor()
self.assertEqual(m.block_size, block_size)
self.assertEqual(m.digest_size, digest_size)
self.assertEqual(len(m.digest()), digest_size)
self.assertEqual(m.name.lower(), name.lower())
self.assertIn(name.split("_")[0], repr(m).lower())
def test_blocksize_name(self):
self.check_blocksize_name('md5', 64, 16)
self.check_blocksize_name('sha1', 64, 20)
self.check_blocksize_name('sha224', 64, 28)
self.check_blocksize_name('sha256', 64, 32)
self.check_blocksize_name('sha384', 128, 48)
self.check_blocksize_name('sha512', 128, 64)
def test_case_md5_0(self): def test_case_md5_0(self):
self.check('md5', b'', 'd41d8cd98f00b204e9800998ecf8427e') self.check('md5', b'', 'd41d8cd98f00b204e9800998ecf8427e')
...@@ -323,6 +363,15 @@ class HashLibTestCase(unittest.TestCase): ...@@ -323,6 +363,15 @@ class HashLibTestCase(unittest.TestCase):
# for multithreaded operation (which is hardwired to 2048). # for multithreaded operation (which is hardwired to 2048).
gil_minsize = 2048 gil_minsize = 2048
for cons in self.hash_constructors:
m = cons()
m.update(b'1')
m.update(b'#' * gil_minsize)
m.update(b'1')
m = cons(b'x' * gil_minsize)
m.update(b'1')
m = hashlib.md5() m = hashlib.md5()
m.update(b'1') m.update(b'1')
m.update(b'#' * gil_minsize) m.update(b'#' * gil_minsize)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment