-
Notifications
You must be signed in to change notification settings - Fork 128
/
fms.py
33 lines (27 loc) · 982 Bytes
/
fms.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
from collections import Counter
def _possible_key_bit(key, c):
s = [i for i in range(256)]
j = 0
for i in range(len(key)):
j = (j + s[i] + key[i]) % 256
tmp = s[i]
s[i] = s[j]
s[j] = tmp
return (c[0] - j - s[len(key)]) % 256
def attack(encrypt_oracle, key_len):
"""
Recovers the hidden part of an RC4 key using the Fluhrer-Mantin-Shamir attack.
:param encrypt_oracle: the padding oracle, returns the encryption of a plaintext under a hidden key concatenated with the iv
:param key_len: the length of the hidden part of the key
:return: the hidden part of the key
"""
key = bytearray([3, 255, 0])
for a in range(key_len):
key[0] = a + 3
possible = Counter()
for x in range(256):
key[2] = x
c = encrypt_oracle(key[:3], b"\x00")
possible[_possible_key_bit(key, c)] += 1
key.append(possible.most_common(1)[0][0])
return key[3:]