Hydroadmin
hackthebox
Task: Fix a GraphQL batching vulnerability that allows PIN brute-force bypass of rate limiting. Solution: Use the online code editor API to read index.js, change allowBatchedHttpRequests from true to false in Apollo Server config, save the patched file, restart the server, and verify the fix to receive the flag.
$ ls tags/ techniques/
Hydroadmin - HackTheBox Challenge
Description
"With reservoirs sealed and cities teetering on thirst, our heroes storm the HydroAdmin control room to reopen valves and restore the flow of water."
Target: 83.136.249.164:45091
Analysis
Initial Reconnaissance
Discovered an online code editor called "HTB Editor" at the target with the following API endpoints:
/api/directory- list files/api/file?path=X- read file content/api/create-file- create new file/api/rename- rename files/api/restart- restart the server/api/verify- verify if vulnerability is patched
Source Code Analysis
Key files discovered:
index.js - Main server with Apollo GraphQL server configured with vulnerable batching:
const server = new ApolloServer({ ...armor.protect(), introspection: false, typeDefs, allowBatchedHttpRequests: true, // VULNERABLE! resolvers });
models/ControlPin.js - PIN generation (4-digit, 1000-9999) and verification logic
schema/resolvers.js - GraphQL resolvers including PIN verification
exploit/solver.py - A script showing how to exploit GraphQL batching to brute-force PINs
The Vulnerability
The GraphQL server had allowBatchedHttpRequests: true which allows:
- Sending multiple GraphQL queries in a single HTTP request
- Bypassing rate limiting (10 requests per minute)
- Brute-forcing all 9000 possible PINs (1000-9999) in just a few batch requests
The Twist
This challenge required PATCHING the vulnerability, not exploiting it!
The /api/verify endpoint checked if the vulnerability was fixed.
Solution
Exploitation Steps
- Rename original file: Backup the vulnerable index.js
- Read backup: Get the original content
- Patch content: Change
allowBatchedHttpRequests: truetoallowBatchedHttpRequests: false - Create patched file: Write the fixed version
- Restart server: Apply changes
- Get flag: Verify the patch
Exploit Script
#!/usr/bin/env python3 """ Hydroadmin - GraphQL Batching Patch Challenge HackTheBox Challenge """ import requests BASE_URL = "http://83.136.249.164:45091" # Step 1: Rename original file to backup print("[*] Renaming index.js to index.js.bak...") resp = requests.post(f"{BASE_URL}/api/rename", json={ "oldPath": "index.js", "newPath": "index.js.bak" }) print(f" Response: {resp.json()}") # Step 2: Read the backup file print("[*] Reading index.js.bak...") resp = requests.get(f"{BASE_URL}/api/file?path=index.js.bak") content = resp.json()['content'] print(f" Got {len(content)} bytes") # Step 3: Patch the vulnerability print("[*] Patching allowBatchedHttpRequests...") patched = content.replace( 'allowBatchedHttpRequests: true', 'allowBatchedHttpRequests: false' ) # Step 4: Create patched file print("[*] Creating patched index.js...") resp = requests.post(f"{BASE_URL}/api/create-file", json={ "path": "index.js", "content": patched }) print(f" Response: {resp.json()}") # Step 5: Restart server print("[*] Restarting server...") resp = requests.post(f"{BASE_URL}/api/restart") print(f" Response: {resp.json()}") # Step 6: Wait and verify import time print("[*] Waiting for server restart...") time.sleep(2) print("[*] Verifying patch...") resp = requests.get(f"{BASE_URL}/api/verify") result = resp.json() print(f" Result: {result}") if 'flag' in result: print(f"\n[+] FLAG: {result['flag']}")
Vulnerable vs Fixed Code
Vulnerable (original):
const server = new ApolloServer({ ...armor.protect(), introspection: false, typeDefs, allowBatchedHttpRequests: true, // VULNERABLE resolvers });
Fixed (patched):
const server = new ApolloServer({ ...armor.protect(), introspection: false, typeDefs, allowBatchedHttpRequests: false, // FIXED resolvers });
Key Indicators
Use this technique when you see:
- Apollo GraphQL server with
allowBatchedHttpRequests: true - Rate limiting on GraphQL endpoint
- PIN/OTP/code verification via GraphQL
- Need to brute-force a small keyspace (e.g., 4-digit PIN)
- Online code editor with file manipulation APIs
Lessons Learned
-
GraphQL Batching Bypass: GraphQL batching can bypass rate limiting - always set
allowBatchedHttpRequests: falsein production Apollo servers -
Defense in Depth: Even with
@escape.tech/graphql-armorlibrary, batching was still enabled manually - security libraries don't protect against misconfigurations -
Patch Challenges: Sometimes CTF challenges require fixing vulnerabilities, not just exploiting them - read the challenge description carefully
-
Code Editor Attack Surface: Online code editors with file manipulation APIs (create, rename, restart) can be powerful attack vectors
-
PIN Bruteforce Math:
- 4-digit PIN (1000-9999) = 9000 possibilities
- Rate limit: 10 requests/minute
- Without batching: 900 minutes = 15 hours
- With batching (1000 queries/request): 9 requests = instant
References
$ cat /etc/motd
Liked this one?
Pro unlocks every writeup, every flag, and API access. $9/mo.
$ cat pricing.md$ grep --similar
Similar writeups
- [web][Pro]Lab 228 — GridPulse— hackadvisor
- [web][Pro]InsightFlow — GraphQL Batching OTP Bypass— hackadvisor
- [web][free]AgriWeb— hackthebox
- [web][Pro]SyncGraph— hackadvisor
- [web][free]Blueprint Heist— hackthebox