Crypto

RSA LSB Oracle Attack ( + 키 값 계속 바뀔 시도 가능)

MyriaBreak 2019. 10. 12. 12:00

RSA LSB Oracle Attack에서 키값이 바뀌거나 고정되어있을때 공격할 수 있는 방법이다.

이전과 다른 점은 N값이 계속 바뀐다는 점인데, 이는 한 비트씩 추출하는 방법으로 풀 수 있다.


2의 제곱들의 역원을 이용해서 한비트씩 추출해내는 방법으로 문제를 푸는 것인데,

기존의 LSB Attack 공격에 쓰이는 방법과 비교했을때, (기존에 이 블로그 writeup에 쓰인 방법들은 이진탐색 방법이였다.)

한 bit씩 추출하여 flag를 찾아낸다는 점에서 더 괜찮은 공격방법일 수 있다.


생각해보니 이 방법 자체가 엄청 좋은 것같다. ㅎㅎ...



example server

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
62
63
64
65
66
from Crypto.PublicKey import RSA
from Crypto.Util.number import long_to_bytes, bytes_to_long
 
menu = """
1. Get PublicKey
2. Encrypt
3. Decrypt
4. Print FLAG
"""
 
with open("./flag""r") as f:
    flag = f.read().strip()
    
privkey = RSA.generate(1024)
= privkey.n
= privkey.e
= privkey.d
 
def encrypt(p):
    return pow(p, e, N)
 
def decrypt(c):
    return pow(c, d, N)%2
    
def main():
    global N,e,d
    print(menu.strip())
    choice = int(input("> "))
    if choice == 1:
        print("n is "+str(N))
        print("e is "+str(e))
    elif choice == 2:
        data = raw_input("msg : ")
        print("enc : "+str(encrypt(bytes_to_long(data))))
    elif choice == 3:
        data = input("msg (demical number) : ")
        print("dec : "+str(decrypt(data)))
        privkey = RSA.generate(1024)
        N = privkey.n
        e = privkey.e
        d = privkey.d
    elif choice == 4:
        print(str(encrypt(bytes_to_long(flag))))
    else:
        print("invalid menu")
 
welcome = """
 /$$$$$$$   /$$$$$$   /$$$$$$         /$$$$$$  /$$     /$$ /$$$$$$  /$$$$$$$$ /$$$$$$$$ /$$      /$$
| $$__  $$ /$$__  $$ /$$__  $$       /$$__  $$|  $$   /$$//$$__  $$|__  $$__/| $$_____/| $$$    /$$$
| $$  \ $$| $$  \__/| $$  \ $$      | $$  \__/ \  $$ /$$/| $$  \__/   | $$   | $$      | $$$$  /$$$$
| $$$$$$$/|  $$$$$$ | $$$$$$$$      |  $$$$$$   \  $$$$/ |  $$$$$$    | $$   | $$$$$   | $$ $$/$$ $$
| $$__  $$ \____  $$| $$__  $$       \____  $$   \  $$/   \____  $$   | $$   | $$__/   | $$  $$$| $$
| $$  \ $$ /$$  \ $$| $$  | $$       /$$  \ $$    | $$    /$$  \ $$   | $$   | $$      | $$\  $ | $$
| $$  | $$|  $$$$$$/| $$  | $$      |  $$$$$$/    | $$   |  $$$$$$/   | $$   | $$$$$$$$| $$ \/  | $$
|__/  |__/ \______/ |__/  |__/       \______/     |__/    \______/    |__/   |________/|__/     |__/
version 3.2.7                                                    
"""
 
print(privkey.d)
 
if __name__ == '__main__':
    print(welcome)
    while True:
        try: main()
        except: break
cs






attack code


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
from Crypto.Util.number import *
from Crypto.PublicKey import RSA
from pwn import *
 
conn = process(["python""server.py"])
 
def get_key():
    conn.sendlineafter("4. Print FLAG""1")
    conn.recvuntil("n is ")
    return int(conn.recvline().strip())
    
def get_flag():
    conn.sendlineafter("4. Print FLAG""4")
    conn.recvuntil("> ")
    return int(conn.recvline().strip())
 
def decrypt(ct):
    conn.sendlineafter("4. Print FLAG""3")
    conn.sendline(str(ct))
    conn.recvuntil("dec : ")
    result = int(conn.recvline().strip())
    return result
    
flag_enc = get_flag()
init_N = get_key()
 
# Public exponent
= 65537
 
bits = "1"
flag = ""
= 1
while True:
    enc = get_flag()
    N = get_key()
 
    inv = inverse(2**i, N)
    chosen_ct = (enc*pow(inv, e, N))%N
    output = decrypt(chosen_ct)
    flag_char = (output - ((int(bits, 2)*inv) % N)) % 2
 
    bits = str(flag_char) + bits
    if len(bits) % 8 == 0:
        flag = long_to_bytes(int(bits, 2))
        print(flag)
        if flag_enc == pow(int(bits, 2), e,init_N):
            break
    i+=1
cs