-
Notifications
You must be signed in to change notification settings - Fork 18
/
shake_wrapper.py
61 lines (45 loc) · 1.66 KB
/
shake_wrapper.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
from hashlib import shake_128, shake_256
"""
hashlib has implemented shake_128, shake_256 but they haven't designed it so you
can read bytes properly... every call generates all bytes without updating
shake_128.digest(1) == shake_128.digest(1)
and we have no shake_128.read() :(
So, here's a wrapper which calls to shake_128.digest and collects a bunch of
bytes which we can then read through.
"""
class Shake:
def __init__(self, algorithm, block_length):
self.algorithm = algorithm
self.block_length = block_length
self.buf = b""
self.len_buf = 0
def absorb(self, input_bytes):
"""
Initialise the XOF with the seed and reset other init.
"""
# Initalize the buffer
self.index = 0
# Set the reading method from hashlib digest
self.xof_read = self.algorithm(input_bytes).digest
# Start by requesting 5 blocks from the XOF
self.buf = self.xof_read(5 * self.block_length)
self.len_buf = 5 * self.block_length
def read(self, n):
"""
Read n bytes from the XOF
"""
# Make sure there are enough bytes to read
while self.index + n > self.len_buf:
# double the size of the buffer
self.len_buf *= 2
self.buf = self.xof_read(self.len_buf)
# Read from the buffer data the bytes requested
send = self.buf[self.index : self.index + n]
# Shift the index along the buffer
self.index += n
return send
def __call__(self, input_bytes):
self.absorb(input_bytes)
return self
shake128 = Shake(shake_128, 168)
shake256 = Shake(shake_256, 136)