Like a lot of developers in this market, I’ve been taking freelance assessments and Discord job leads more seriously than I normally would.
One of those assessments turned into a malware investigation.
One day, I saw a post in a Discord server looking for a fullstack dev. I pitched. The reply looked routine at first: they DM’d me a requirements PDF for an assessment. I did not trust it, so I asked them to paste the requirements in the chat instead. They sent screenshots of the PDF, and it looked like a real assessment. Clean structure, clear expectations, nothing immediately screaming scam.
Then they invited me to a GitHub repo called E-commerce-template-12d46f3e. My first thought was that the name looked autogenerated, like they were appending random numbers for each assessment. That is when I started treating it like a security review, not a coding exercise.
What I checked first
The first thing I looked at was package.json. I was expecting the usual red flags: weird postinstall hooks, obfuscated scripts, or packages I had never heard of. There was one outdated dependency, @zeit/next-css, but nothing in package.json looked obviously malicious.
That is what made the repo interesting. The dependency list looked boring; the problem was in the application flow.
The suspicious startup path
Next, I looked at the startup scripts.
The dev script ran server.js, and that file did something that immediately raised my guard: it called startLoggingErrors() during server startup.
At first glance, that looked like a harmless logging helper. But when I opened the related files, I found this chain:
-
server.jsstarts the server. -
lib/serverStartup.jscallseval(...). -
lib/startupLogs.jsreconstructs a hidden payload from files inpublic/flags/.
That eval() was the key. A server startup path that reconstructs data from assets and then evaluates them is a hard stop.
The hidden payload in the SVGs
The weirdest part was the assets.
The repository had a bunch of country flag SVG files under public/flags/. Those looked normal until I checked the HTML comments inside them. Each SVG had a comment that looked like a fragment of base64-encoded text.
The loader in lib/startupLogs.js:
- Walks through
public/flags/ - Reads each
.svg - Extracts the text inside
<!-- ... --> - Sorts and joins the fragments
- Base64-decodes the result
- Returns the decoded JavaScript
-
lib/serverStartup.jsfeeds that decoded string intoeval()
The two lines that made the whole thing click were basically this:
const dir = path.join(process.cwd(), setLogUrl("sxeolf2iodjv"));
eval(log_manager());
That is the bridge between the SVG comments and the executed payload.
So yes, the SVG comments were not decorative. They were a distributed payload. The code was split across many innocent-looking image files so it would not stand out in a quick scan.
I also used Codex as a coding agent to help with the defensive part of the review. Instead of running the suspicious code, I asked it to inspect the repo, trace the startup flow, and deobfuscate the payload safely so I could understand what it did without accidentally executing it. That helped confirm the hidden flow and surfaced additional suspicious paths for a wider audit.
If you want the exact evidence trail, these are the key files:
server.jslib/serverStartup.jslib/startupLogs.jspublic/flags/*.svg
What the decoded code actually does
Once I reconstructed the payload without executing it, with Codex helping me safely decode and audit the hidden code, the intent was obvious.
It is not just telemetry or error logging. It behaves like a stealer/dropper with persistence.
Part of the reconstructed payload after safely decoding the SVG fragments. I redacted active infrastructure details before publishing.
1. It fingerprints the machine
The payload gathers:
- local IPv4 addresses
- public IP via
api.ipify.org - hostname
- OS type and version
- user info
- a machine/user identifier
- whether the machine looks virtualized
That means it profiles the environment before doing anything else.
2. It sends the profile to a remote server
It posts a JSON system profile to a remote endpoint over HTTP.
3. It drops and runs additional files on Windows
On Windows, it downloads executables into AppData and runs them.
It also writes a runjs.vbs file into the Windows Startup folder so the code can persist across reboots.
That is a classic persistence pattern.
4. It hunts for sensitive files
The script recursively scans user paths and drives for files matching patterns like:
.env.pem.key.cer.secret.txt.xlsxreadme.md.ssh.aws.github
That is collection logic, not a developer convenience feature.
5. It targets browser data
It looks for browser profile directories for:
- Chrome
- Brave
- Edge
- LT Browser
Then it checks for files such as:
Login DataWeb DataLocal Extension Settings
That is the kind of place malware checks when it wants tokens, cookies, or saved credentials.
6. It targets Sticky Notes
It also checks the Microsoft Sticky Notes storage path on Windows. That is another common place where people accidentally leave sensitive information.
Indicators
I am intentionally redacting the live infrastructure in the public draft.
- Redacted base URL:
[redacted-host]:[redacted-port] - Observed endpoints:
/system-info/file-manage/download/track.js/download/apps/language_server_x64_x32_windows.exe/download/apps/assist_language_server_x64_x32_windows.exe
- Data sent to
/system-info:- OS type, platform, release
- hostname
- user info
- local IP addresses
- public IP
- machine/user identifier
- VM detection flag
- Data uploaded to
/file-manage:- file contents
- filename
- path
- system identifier
- File types and stores targeted locally:
-
.env,.pem,.key,.cer,.secret,.xlsx - browser profile databases
- Microsoft Sticky Notes data
-
Why this was easy to miss
The repo looked like a regular Next.js storefront at a glance.
The package.json was mostly boring. The app structure looked normal. The assets looked like flags. The malicious code was buried in a runtime path that almost nobody checks unless they are being cautious on purpose.
That is the lesson here: malicious code does not have to live in node_modules, and it does not have to look obviously hostile. Sometimes it hides in the things you think are static content.
What I told myself while reviewing it
I kept repeating one rule: if a codebase wants to run something dynamically at startup, I need to know exactly why.
The moment I saw:
- an obfuscated loader
- base64 fragments hidden across image assets
-
eval()on decoded content - network calls to a hardcoded remote host
I stopped treating it like a normal assessment and started treating it like an incident.
What I would advise anyone else to do
If a Discord recruiter or random client sends you a repo:
- Check
package.jsonfirst, but do not stop there. - Inspect the runtime entrypoint, not just the UI code.
- Search for
eval,new Function,exec,spawn, and startup hooks. - Look inside static assets if the code mentions them.
- If the repo uses comments, weird strings, or base64-looking fragments, assume it may be encoded payload data until proven otherwise.
- Never run the project on your main machine before you understand the startup path.
Flow Diagram
server.js
↓
lib/serverStartup.js
↓
lib/startupLogs.js
↓
public/flags/*.svg
↓
HTML comment fragments
↓
base64 reconstruction
↓
eval(payload)
Closing thought
I went in expecting a take-home assessment.
What I found was a repo that used a clean-looking frontend as cover for a hidden payload loader. The lesson is simple: when something feels off, slow down and inspect the startup path. That is where malware likes to hide.
If you are job hunting right now, be careful. A polished assessment can still be malicious.
Also, currently open to Fullstack/Backend/AI roles.






Top comments (5)
The VM detection flag is the detail that jumped out at me. That's not a hobbyist — filtering out sandboxes before dropping the real payload is operational tradecraft you see in state-sponsored campaigns (Lazarus Group's "Operation Dream Job" used nearly identical interview-bait repos targeting devs). The SVG comment technique is smart because it sits in a blind spot: security tooling focuses on package.json, postinstall hooks, and node_modules, but nobody's parsing HTML comments inside static assets in public/. Even GitHub's secret scanning wouldn't flag base64 fragments split across multiple files since each chunk is meaningless on its own. The real takeaway for anyone taking freelance assessments is to treat
eval()in a startup path the same way you'd treat a postinstall script — it's arbitrary code execution with extra steps.That’s quite a stringent requirement: “Preferably, the kind where the SVGs don’t assemble malware payloads.”
Furthermore, the entire VM detection process appears unnecessary to me. Additionally, you’re likely to lose a substantial number of actual clients who would have otherwise been infected. Any serious security researcher reverse engineering it will probably not care.
Interesting catch — hiding malware in SVGs is a clever trick since they're often overlooked as vectors. I've seen similar obfuscation techniques in GPU-based container escapes where attackers embed malicious code in unexpected assets. Always good to verify third-party repos, especially when they ask for access to your environment.
This is cool. The SVG base64 comment fragmentation is a brilliant vector because it completely bypasses standard static analysis tools (like Dependabot or Socket) that focus almost exclusively on package.json, postinstall scripts, and node_modules. Nobody is running security audits on static flag graphics in /public.
What makes this so dangerous is that it preys on freelance developers who are actively pitching and trying to move fast to close a deal.
While your runtime audit is the ultimate technical line of defense, your write-up also highlights a massive behavioural lesson: the scam always starts in the chat. The pattern of moving a cold pitch immediately to an autogenerated GitHub repo or forcing a local execution environment for a "simple assessment" is a classic social engineering footprint.
Recognizing this behavioural pattern is exactly why we've been building FreelancerGuard.fyi.
We wanted to create a first line of defense that stops you before you even download the malicious code. We built a Red Flag Detector that scans early client DMs, Discord pitches, and project briefs for these exact social engineering behaviours—flagging fishy project pipelines, suspicious link patterns, and high-risk client requests.
Combining behavioural vetting (screening the client before cloning) with technical vetting (checking startup paths and eval hooks after cloning) is the only way to stay safe in this market.
Stay safe out there, and good luck with the job search!
Great initiative man! I wish you all the best!
Some comments may only be visible to logged-in visitors. Sign in to view all comments.