-
Notifications
You must be signed in to change notification settings - Fork 43
/
Copy pathtpconf_bin_xml.py
executable file
·346 lines (312 loc) · 12.5 KB
/
tpconf_bin_xml.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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
#!/usr/bin/env python3
# Copyright 2018 Alain Ducharme
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
# Description:
# Command line utility to convert TP-Link router backup config files:
# - conf.bin => decrypt, md5hash and uncompress => conf.xml
# - conf.xml => compress, md5hash and encrypt => conf.bin
import argparse
from hashlib import md5
from os import path
import re
from struct import pack, pack_into, unpack_from
from Cryptodome.Cipher import DES # apt install python3-pycryptodome (OR: pip install pycryptodomex)
__version__ = '0.2.12'
KEYS = {
'default': b'\x47\x8D\xA5\x0B\xF9\xE3\xD2\xCF',
'xz005-g6-un-v1': b'\x45\xEE\x92\x32\xCF\x5B\x1D\xFE',
}
def compress(src, skiphits=False):
'''Compress buffer'''
# Make sure last byte is NULL
if src[-1]:
src += b'\0'
size = len(src)
buffer_countdown = size
hash_table = [0] * 0x2000
dst = bytearray(0x8000) # max compressed buffer size
block16_countdown = 0x10 # 16 byte blocks
block16_dict_bits = 0 # bits for dictionnary bytes
def put_bit(bit):
nonlocal block16_countdown, block16_dict_bits, d_p, d_pb
if block16_countdown:
block16_countdown -= 1
else:
pack_into('H', dst, d_pb, block16_dict_bits)
d_pb = d_p
d_p += 2
block16_countdown = 0xF
block16_dict_bits = (bit + (block16_dict_bits << 1)) & 0xFFFF
def put_dict_ld(bits):
ldb = bits >> 1
while True:
lb = (ldb - 1) & ldb
if not lb:
break
ldb = lb
put_bit(int(ldb & bits > 0))
ldb = ldb >> 1
while ldb:
put_bit(1)
put_bit(int(ldb & bits > 0))
ldb = ldb >> 1
put_bit(0)
def hash_key(offset):
b4 = src[offset:offset+4]
hk = 0
for b in b4[:3]:
hk = (hk + b) * 0x13d
return ((hk + b4[3]) & 0x1FFF)
pack_into(packint, dst, 0, size) # Store original size
dst[4] = src[0] # Copy first byte
buffer_countdown -= 1
s_p = 1
s_ph = 0
d_pb = 5
d_p = 7
while buffer_countdown > 4:
while s_ph < s_p:
hash_table[hash_key(s_ph)] = s_ph
s_ph += 1
hit = hash_table[hash_key(s_p)]
count = 0
if hit:
while True:
if src[hit + count] != src[s_p + count]:
break
count += 1
if count == buffer_countdown:
break
if count >= 4 or count == buffer_countdown:
hit = s_p - hit - 1
put_bit(1)
put_dict_ld(count - 2)
put_dict_ld((hit >> 8) + 2)
dst[d_p] = hit & 0xFF
d_p += 1
buffer_countdown -= count
s_p += count
if skiphits:
hash_table[hash_key(s_ph)] = s_ph
s_ph += count
continue
put_bit(0)
dst[d_p] = src[s_p]
s_p += 1
d_p += 1
buffer_countdown -= 1
while buffer_countdown:
put_bit(0)
dst[d_p] = src[s_p]
s_p += 1
d_p += 1
buffer_countdown -= 1
pack_into('H', dst, d_pb, (block16_dict_bits << block16_countdown) & 0xFFFF)
return d_p, dst[:d_p] # size, compressed buffer
def uncompress(src):
'''Uncompress buffer'''
block16_countdown = 0 # 16 byte blocks
block16_dict_bits = 0 # bits for dictionnary bytes
def get_bit():
nonlocal block16_countdown, block16_dict_bits, s_p
if block16_countdown:
block16_countdown -= 1
else:
block16_dict_bits = unpack_from('H', src, s_p)[0]
s_p += 2
block16_countdown = 0xF
block16_dict_bits = block16_dict_bits << 1
return 1 if block16_dict_bits & 0x10000 else 0 # went past bit
def get_dict_ld():
bits = 1
while True:
bits = (bits << 1) + get_bit()
if not get_bit():
break
return bits
size = unpack_from(packint, src, 0)[0]
dst = bytearray(size)
s_p = 4
d_p = 0
dst[d_p] = src[s_p]
s_p += 1
d_p += 1
while d_p < size:
if get_bit():
num_chars = get_dict_ld() + 2
msB = (get_dict_ld() - 2) << 8
lsB = src[s_p]
s_p += 1
offset = d_p - (lsB + 1 + msB)
for i in range(num_chars):
# 1 by 1 ∵ sometimes copying previously copied byte
dst[d_p] = dst[offset]
d_p += 1
offset += 1
else:
dst[d_p] = src[s_p]
s_p += 1
d_p += 1
return dst
def verify(src):
# Try md5 hash excluding up to last 8 (padding) bytes
if not any(src[:16] == md5(src[16:len(src)-i]).digest() for i in range(8)):
print('ERROR: Bad file or could not decrypt file - MD5 hash check failed!')
exit()
def verify_ac1350(src):
length = unpack_from(packint, src, 16)[0]
payload = src[20:][:length]
if src[:16] != md5(payload).digest():
print('ERROR: Bad file or could not decrypt file - MD5 hash check failed!')
exit()
return payload
def check_size_endianness(src):
global packint
if unpack_from(packint, src)[0] > 0x20000:
packint = '<I' if packint == '>I' else '>I'
if unpack_from(packint, src)[0] > 0x20000:
print('ERROR: compressed size too large for a TP-Link config file!')
exit()
print('WARNING: wrong endianness, automatically switching. (see -h)')
endianness = 'little' if packint == '<I' else 'big'
print(f'OK: appears your device uses {endianness}-endian.')
if __name__ == '__main__':
parser = argparse.ArgumentParser(description='TP-Link router config file processor.')
parser.add_argument('infile', help='input file (e.g. conf.bin or conf.xml)')
parser.add_argument('outfile', help='output file (e.g. conf.bin or conf.xml)')
parser.add_argument('-l', '--littleendian', action='store_true',
help='Use little-endian (default: big-endian)')
parser.add_argument('-n', '--newline', action='store_true',
help='Replace EOF NULL with newline (after uncompress)')
parser.add_argument('-s', '--skiphits', action='store_true',
help='"skiphits" (less) compression (xml to bin) (some Archer models)')
parser.add_argument('-o', '--overwrite', action='store_true',
help='Overwrite output file')
args = parser.parse_args()
if path.getsize(args.infile) > 0x20000:
print('ERROR: Input file too large for a TP-Link config file!')
exit()
if not args.overwrite and path.exists(args.outfile):
print('ERROR: Output file exists, use -o to overwrite')
exit()
packint = '<I' if args.littleendian else '>I'
with open(args.infile, 'rb') as f:
src = f.read()
if src.startswith(b'<?xml'):
key = KEYS['default']
if b'1350 v' in src or b'EC230' in src: # AC1350 (Archer C60) and ISP variants
print('OK: AC1350 XML file - compressing, hashing and encrypting…')
size, dst = compress(src, True)
md5hash = md5(dst[:size]).digest()
dst = md5hash + pack(packint, size) + bytes(dst)
elif b'W9980' in src or b'W8980' in src:
print('OK: W9980/W8980 XML file - hashing, compressing and encrypting…')
md5hash = md5(src).digest()
size, dst = compress(md5hash + src)
elif b'W8970' in src:
print('OK: W8970 XML file - hashing and encrypting…')
# Make sure last byte is NULL
if src[-1]:
src += b'\0'
md5hash = md5(src).digest()
dst = md5hash + src
elif b'WR841N v14' in src: # lock to v14, seems varied between versions otherwise
print('OK: WR841N v14 XML file - compressing, hashing and encrypting…')
if packint == '>I':
print('WARNING: wrong endianness, automatically setting little. (see -h)')
packint = '<I'
size, dst = compress(src, False)
# seems like the router wants compessed data multiple of 8
if len(dst) & 7:
dst += b'\0' * (8 - (len(dst) & 7))
md5hash = md5(dst).digest()
dst = md5hash + bytes(dst)
elif b'XZ005-G6 v1.0' in src and b'(EU)' in src: # XZ005-G6(UN) V1.0 (advertised as "UN" but has "EU" in XML)
print('OK: XZ005-G6(UN) V1.0 XML file - compressing, hashing and encrypting…')
key = KEYS['xz005-g6-un-v1']
if packint == '<I':
print('WARNING: wrong endianness, automatically setting big. (see -h)')
packint = '>I'
size, dst = compress(src, False)
md5hash = md5(dst).digest()
dst = md5hash + bytes(dst)
else:
skiphits = args.skiphits
if b'Archer' in src:
if packint == '>I': # Archer models can be little or big-endian!
print('WARNING: make sure you are using correct endianness. (see -h)')
# Older Archer C2 & C20 v1 skiphits, newer v4 & v5 don't
if re.search(b'Archer C2[0-9]?[A-z]? v1', src):
skiphits = True
print('OK: XML file - compressing, hashing and encrypting…')
size, dst = compress(src, skiphits)
md5hash = md5(dst[:size]).digest()
dst = md5hash + bytes(dst)
# data length for encryption must be multiple of 8
if len(dst) & 7:
dst += b'\0' * (8 - (len(dst) & 7))
crypto = DES.new(key, DES.MODE_ECB)
output = crypto.encrypt(bytes(dst))
else:
xml = None
# Assuming encrypted config file
if len(src) & 7: # Encrypted file length must be multiple of 8
print('ERROR: Wrong input file type!')
exit()
for key in KEYS.values():
crypto = DES.new(key, DES.MODE_ECB)
decrypted = crypto.decrypt(src)
if decrypted[16:21] == b'<?xml': # XML (not compressed?)
verify(decrypted)
print('OK: BIN file decrypted, MD5 hash verified…')
xml = decrypted[16:]
elif decrypted[20:27] == b'<\0\0?xml': # compressed XML (W9970/XZ005-G6(UN) V1.0)
verify(decrypted)
decrypted = decrypted[16:]
check_size_endianness(decrypted)
print('OK: BIN file decrypted, MD5 hash verified, uncompressing…')
xml = uncompress(decrypted)
elif decrypted[22:29] == b'<\0\0?xml': # compressed XML (W9980/W8980)
check_size_endianness(decrypted)
print('OK: BIN file decrypted, uncompressing…')
dst = uncompress(decrypted)
verify(dst)
print('OK: MD5 hash verified')
xml = dst[16:]
elif decrypted[24:31] == b'<\0\0?xml': # compressed XML (AC1350)
'''
payload md5 (16b) | payload size (4b) | payload
'''
check_size_endianness(decrypted[16:])
decrypted = verify_ac1350(decrypted)
print('OK: BIN file decrypted, MD5 hash verified, uncompressing…')
xml = uncompress(decrypted)
else:
print('WARNING: Unrecognized file type when using this key! Attempting next one.')
crypto = None
# XML data found, decryption/decompression succeeded for this key.
if xml is not None:
break
else:
print('ERROR: Unrecognized file type and no known keys left!')
exit()
if args.newline:
if xml[-1] == 0: # NULL
xml[-1] = 0xa # LF
output = xml
with open(args.outfile, 'wb') as f:
f.write(output)
print('Done.')