또 뭐하지

[Dreamhack] 40 Birthdays 본문

Write-up/Crypto

[Dreamhack] 40 Birthdays

mameul 2024. 11. 8. 19:32
728x90

풀이

import hashlib

def birthday_hash(msg):
	return hashlib.sha256(msg).digest()[12:17]

msg1 = bytes.fromhex(input("Input message 1 in hex: "))
msg2 = bytes.fromhex(input("Input message 2 in hex: "))

if msg1 == msg2:
	print("Those two messages are the same! >:(")

elif birthday_hash(msg1) != birthday_hash(msg2):
	print("Those two messages don't have the same birthday! T.T")

else:
	print("Finally! They have the same birthday ^o^")
	print(open("flag.txt").read())

제공된 코드를 살펴보면 문제에서 다이제스트의 12부터 16까지 5 바이트를 떼어서 반환하는 birthday_hash를 사용한다. 여기서 나온 값이 동일한 충돌쌍을 찾아 입력하면 flag를 얻을 수 있는 문제이다.

 

문제에서 생일역설(Birthday paradox)가 나오는 것을 보고 이에 관해 알아보았다. 

생일 역설에 따르면, N이 가능한 값의 수일 때, 서로 다른 입력값 중 약 √N개 정도를 시도하면 두 값이 같은 해시 출력을 가질 확률이 높아진다고 한다.

여기서 5바이트 (40비트)에 대하여 충돌쌍을 찾기 때문에 N = 2**40 인데, 생일 역설에 따라 √N =2**20 정도를 시도하면 충돌쌍을 찾을 수 있을 것이다.

import hashlib

def birthday_hash(msg):
    return hashlib.sha256(msg).digest()[12:17]

l = [] 
s = set() # 빠른 검색을 위해 추가

for i in range(2**20): # sqrt(N) = 2**20까지 전수조사
    msg = str(i).encode()
    tmp = birthday_hash(msg)

    if tmp in s: # 해시값이 이전에 나온적 있는지 확인
        msg1 = msg
        msg2 = str(l.index(tmp)).encode() # 이전에 저장돼 있던 충돌값 불러오기
        break

    l.append(tmp)
    s.add(tmp) # 해시값을 리스트와 세트에 저장

if birthday_hash(msg1) == birthday_hash(msg2) :
    print("ok") # 충돌쌍이 맞는 확인
    
print("msg1:",bytes.hex(msg1).encode())
print("msg2:",bytes.hex(msg2).encode())

충돌하는 값 찾는 과정을 빠르게 진행하기 위해서 set을 이용하였다. 위 코드를 실행해보면 아래와 같은 결과를 확인할 수 있다. 

위의 값을 입력해주면 flag를 얻을 수 있다.

'Write-up > Crypto' 카테고리의 다른 글

[Dreamhack] X-Time Pad  (0) 2024.11.18
[Dreamhack] uncommon e  (0) 2024.11.15
[Dreamhack] No sub please!  (1) 2024.11.08
[Dreamhack] Insecure Seed  (0) 2024.10.31
[Dreamhack] Easy Linguistics  (0) 2024.10.24