-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathglobal_metadata_fix.py
122 lines (101 loc) · 3.97 KB
/
global_metadata_fix.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
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
import struct
import xxtea
ENCRYTPED_KEY = b"##%$vsw'lytyqlusxul##p\"lvxrsv\"y\"y'xu%tv\"qsy\"l%%#qlux'ul# wylys\"t 'v$us''A"
XOR_KEY = b"\x41"
def xor_decrypt(ciphertext, key):
plaintext = bytearray(len(ciphertext))
for i in range(len(ciphertext)):
plaintext[i] = ciphertext[i] ^ key[i % len(key)]
return bytes(plaintext)
def offset_count_pair(offset, count):
print("Found Offset + Count: ", offset, count)
correct_order.append(offset)
correct_order.append(count)
incorrect_order.remove(offset)
incorrect_order.remove(count)
#function that find the offsets and counts
def find_offset_count_pair(offset: int):
for sizeI in incorrect_order:
next_offset = offset + sizeI
if next_offset in incorrect_order:
if next_offset == offset:
if len(incorrect_order) <= 6:
pass
else:
continue
offset_count_pair(offset, sizeI)
if find_offset_count_pair(next_offset):
return True
closest_offset_value = 999999999
closest_count = None
closest_offset = None
for countK in incorrect_order:
next_offset = offset + countK
for offsetK in incorrect_order:
if offsetK == offset or offsetK < next_offset:
continue
value = offsetK - next_offset
if value < closest_offset_value:
closest_count = countK
closest_offset = offsetK
closest_offset_value = value
if closest_offset_value <= 20:
if len(incorrect_order) == 2:
offset_count_pair(max(incorrect_order), min(incorrect_order))
return True
offset_count_pair(offset, closest_count)
if find_offset_count_pair(closest_offset):
return True
return False
def decrypt_header():
with open('global-metadata.dat', 'rb') as file:
global_metadata = file.read()
key = xor_decrypt(ENCRYTPED_KEY, XOR_KEY)[:16]
header_offset = int.from_bytes(global_metadata[56:60], byteorder='little')
header_size = int.from_bytes(global_metadata[60:64], byteorder='little')
print("Header Offset: " + str(header_offset))
print("Header Size: " + str(header_size))
#We use the header offset to find beggining of encrypted header, and the size to find the end
encrypted_header = global_metadata[header_offset:header_offset+header_size]
decrypted_header = xxtea.decrypt(encrypted_header, key, padding=False)
#weird for some reason the size of the header is maintained, so we remove it
decrypted_header = decrypted_header[:-4]
print("SUCCESS!")
print(decrypted_header)
return decrypted_header
# decrypt the header and convert it to a list of int_32's
binary_data = decrypt_header()
incorrect_order = list(struct.unpack('i' * (len(binary_data) // struct.calcsize('i')), binary_data))
#checking for sanity and version
sanity = None
if -89056337 in incorrect_order:
incorrect_order.remove(-89056337)
sanity = -89056337
else:
print("didnt find sanity, decryption failed change the ENCRYPTED_KEY and XOR_KEY")
exit(1)
version = None
if 29 in incorrect_order:
incorrect_order.remove(29)
version = 29
else:
print("didnt find version, decryption failed change the ENCRYPTED_KEY and XOR_KEY")
exit(1)
string_offset = min([x for x in incorrect_order if x != 0])
correct_order = [sanity, version]
print(string_offset)
success = find_offset_count_pair(string_offset)
if not success:
print("Failed to find correct order")
exit(1)
else:
print("Success!")
with open('global-metadata.dat', 'rb') as file:
file_data = file.read()
with open('fixed-global-metadata.dat', 'wb') as file:
for value in correct_order:
packed_value = struct.pack('i', value)
file.write(packed_value)
packed_value = struct.pack('i', 256)
file.write(packed_value)
file.write(file_data[260:])