Added pytests to hashes/md5.py (#1100)

* Added pytests to sha1.py

* tweaking md5

* Added Pytests to hashes/md5.py
This commit is contained in:
QuantumNovice 2019-08-06 16:16:30 +05:00 committed by Harshil
parent 89acf5d017
commit d21b4cfb48

View File

@ -1,155 +1,176 @@
from __future__ import print_function from __future__ import print_function
import math import math
def rearrange(bitString32):
"""[summary]
Regroups the given binary string.
Arguments:
bitString32 {[string]} -- [32 bit binary]
Raises:
ValueError -- [if the given string not are 32 bit binary string]
Returns:
[string] -- [32 bit binary string]
"""
if len(bitString32) != 32: def rearrange(bitString32):
raise ValueError("Need length 32") """[summary]
newString = "" Regroups the given binary string.
for i in [3,2,1,0]:
newString += bitString32[8*i:8*i+8] Arguments:
return newString bitString32 {[string]} -- [32 bit binary]
Raises:
ValueError -- [if the given string not are 32 bit binary string]
Returns:
[string] -- [32 bit binary string]
>>> rearrange('1234567890abcdfghijklmnopqrstuvw')
'pqrstuvwhijklmno90abcdfg12345678'
"""
if len(bitString32) != 32:
raise ValueError("Need length 32")
newString = ""
for i in [3, 2,1,0]:
newString += bitString32[8*i:8*i+8]
return newString
def reformatHex(i): def reformatHex(i):
"""[summary] """[summary]
Converts the given integer into 8-digit hex number. Converts the given integer into 8-digit hex number.
Arguments: Arguments:
i {[int]} -- [integer] i {[int]} -- [integer]
""" >>> reformatHex(666)
'9a020000'
"""
hexrep = format(i, '08x')
thing = ""
for i in [3, 2,1,0]:
thing += hexrep[2*i:2*i+2]
return thing
hexrep = format(i,'08x')
thing = ""
for i in [3,2,1,0]:
thing += hexrep[2*i:2*i+2]
return thing
def pad(bitString): def pad(bitString):
"""[summary] """[summary]
Fills up the binary string to a 512 bit binary string Fills up the binary string to a 512 bit binary string
Arguments: Arguments:
bitString {[string]} -- [binary string] bitString {[string]} -- [binary string]
Returns: Returns:
[string] -- [binary string] [string] -- [binary string]
""" """
startLength = len(bitString)
bitString += '1'
while len(bitString) % 512 != 448:
bitString += '0'
lastPart = format(startLength, '064b')
bitString += rearrange(lastPart[32:]) + rearrange(lastPart[:32])
return bitString
startLength = len(bitString)
bitString += '1'
while len(bitString) % 512 != 448:
bitString += '0'
lastPart = format(startLength,'064b')
bitString += rearrange(lastPart[32:]) + rearrange(lastPart[:32])
return bitString
def getBlock(bitString): def getBlock(bitString):
"""[summary] """[summary]
Iterator: Iterator:
Returns by each call a list of length 16 with the 32 bit Returns by each call a list of length 16 with the 32 bit
integer blocks. integer blocks.
Arguments: Arguments:
bitString {[string]} -- [binary string >= 512] bitString {[string]} -- [binary string >= 512]
""" """
currPos = 0
while currPos < len(bitString):
currPart = bitString[currPos:currPos+512]
mySplits = []
for i in range(16):
mySplits.append(int(rearrange(currPart[32*i:32*i+32]), 2))
yield mySplits
currPos += 512
currPos = 0
while currPos < len(bitString):
currPart = bitString[currPos:currPos+512]
mySplits = []
for i in range(16):
mySplits.append(int(rearrange(currPart[32*i:32*i+32]),2))
yield mySplits
currPos += 512
def not32(i): def not32(i):
i_str = format(i,'032b') '''
new_str = '' >>> not32(34)
for c in i_str: 4294967261
new_str += '1' if c=='0' else '0' '''
return int(new_str,2) i_str = format(i, '032b')
new_str = ''
for c in i_str:
new_str += '1' if c == '0' else '0'
return int(new_str, 2)
def sum32(a,b): def sum32(a, b):
return (a + b) % 2**32 '''
'''
return (a + b) % 2**32
def leftrot32(i, s):
return (i << s) ^ (i >> (32-s))
def leftrot32(i,s):
return (i << s) ^ (i >> (32-s))
def md5me(testString): def md5me(testString):
"""[summary] """[summary]
Returns a 32-bit hash code of the string 'testString' Returns a 32-bit hash code of the string 'testString'
Arguments: Arguments:
testString {[string]} -- [message] testString {[string]} -- [message]
""" """
bs ='' bs = ''
for i in testString: for i in testString:
bs += format(ord(i),'08b') bs += format(ord(i), '08b')
bs = pad(bs) bs = pad(bs)
tvals = [int(2**32 * abs(math.sin(i+1))) for i in range(64)] tvals = [int(2**32 * abs(math.sin(i+1))) for i in range(64)]
a0 = 0x67452301 a0 = 0x67452301
b0 = 0xefcdab89 b0 = 0xefcdab89
c0 = 0x98badcfe c0 = 0x98badcfe
d0 = 0x10325476 d0 = 0x10325476
s = [7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22, \ s = [7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22, 7, 12, 17, 22,
5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, \ 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, 5, 9, 14, 20, \
4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, \ 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, 4, 11, 16, 23, \
6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21 ] 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21, 6, 10, 15, 21 ]
for m in getBlock(bs): for m in getBlock(bs):
A = a0 A = a0
B = b0 B = b0
C = c0 C = c0
D = d0 D = d0
for i in range(64): for i in range(64):
if i <= 15: if i <= 15:
#f = (B & C) | (not32(B) & D) #f = (B & C) | (not32(B) & D)
f = D ^ (B & (C ^ D)) f = D ^ (B & (C ^ D))
g = i g = i
elif i<= 31: elif i <= 31:
#f = (D & B) | (not32(D) & C) #f = (D & B) | (not32(D) & C)
f = C ^ (D & (B ^ C)) f = C ^ (D & (B ^ C))
g = (5*i+1) % 16 g = (5*i+1) % 16
elif i <= 47: elif i <= 47:
f = B ^ C ^ D f = B ^ C ^ D
g = (3*i+5) % 16 g = (3*i+5) % 16
else: else:
f = C ^ (B | not32(D)) f = C ^ (B | not32(D))
g = (7*i) % 16 g = (7*i) % 16
dtemp = D dtemp = D
D = C D = C
C = B C = B
B = sum32(B,leftrot32((A + f + tvals[i] + m[g]) % 2**32, s[i])) B = sum32(B, leftrot32((A + f + tvals[i] + m[g]) % 2**32, s[i]))
A = dtemp A = dtemp
a0 = sum32(a0, A) a0 = sum32(a0, A)
b0 = sum32(b0, B) b0 = sum32(b0, B)
c0 = sum32(c0, C) c0 = sum32(c0, C)
d0 = sum32(d0, D) d0 = sum32(d0, D)
digest = reformatHex(a0) + reformatHex(b0) + \
reformatHex(c0) + reformatHex(d0)
return digest
digest = reformatHex(a0) + reformatHex(b0) + reformatHex(c0) + reformatHex(d0)
return digest
def test(): def test():
assert md5me("") == "d41d8cd98f00b204e9800998ecf8427e" assert md5me("") == "d41d8cd98f00b204e9800998ecf8427e"
assert md5me("The quick brown fox jumps over the lazy dog") == "9e107d9d372bb6826bd81d3542a419d6" assert md5me(
print("Success.") "The quick brown fox jumps over the lazy dog") == "9e107d9d372bb6826bd81d3542a419d6"
print("Success.")
if __name__ == "__main__": if __name__ == "__main__":
test() test()
import doctest
doctest.testmod()