本文最后更新于:1 年前
SSTI
sources:
https://docs.jinkan.org/docs/jinja2/templates.html#builtin-filters
https://jinja.palletsprojects.com/en/3.1.x/templates/
https://dormousehole.readthedocs.io/en/latest/
https://xz.aliyun.com/t/9584
2023FCTF热身赛filechecker_mini
打开题目,让我们上传一个文件:
data:image/s3,"s3://crabby-images/43634/43634e2950db4899430560edf63fa5c78500836f" alt=""
桌面上随便丢了个php文件进去提交看看会有啥情况:
data:image/s3,"s3://crabby-images/567c8/567c89ccde1a345a9bfc1d97ff37abf641374ff2" alt=""
判断文件类型,(MIME绕过预定)
附件下载下来先看源码:
index.html:
data:image/s3,"s3://crabby-images/72912/7291211133315ae0e43e20bf4f80fcf6a80fc977" alt=""
可以看出该网页使用模块渲染将result值渲染进index对应位置,那么就看下后端代码app.py:
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 29 30 31 32 33 34 35 36 37 38 39
| from flask import Flask, request, render_template, render_template_string from waitress import serve import os import subprocess
app_dir = os.path.split(os.path.realpath(__file__))[0] app = Flask(__name__) app.config['UPLOAD_FOLDER'] = f'{app_dir}/upload/'
@app.route('/', methods=['GET','POST']) def index(): try: if request.method == 'GET': return render_template('index.html',result="ヽ(=^・ω・^=)丿 ヽ(=^・ω・^=)丿 ヽ(=^・ω・^=)丿")
elif request.method == 'POST': f = request.files['file-upload'] filepath = os.path.join(app.config['UPLOAD_FOLDER'], f.filename)
if os.path.exists(filepath) and ".." in filepath: return render_template('index.html', result="Don't (^=◕ᴥ◕=^) (^=◕ᴥ◕=^) (^=◕ᴥ◕=^)") else: f.save(filepath) file_check_res = subprocess.check_output( ["/bin/file", "-b", filepath], shell=False, encoding='utf-8', timeout=1 ) os.remove(filepath) if "empty" in file_check_res or "cannot open" in file_check_res: file_check_res="wafxixi ฅ•ω•ฅ ฅ•ω•ฅ ฅ•ω•ฅ" return render_template_string(file_check_res)
except: return render_template('index.html', result='Error ฅ(๑*д*๑)ฅ ฅ(๑*д*๑)ฅ ฅ(๑*д*๑)ฅ')
if __name__ == '__main__': serve(app, host="0.0.0.0", port=3000, threads=1000, cleanup_interval=30)
|
data:image/s3,"s3://crabby-images/1993b/1993b9d451a959de80f34d3995e7de4350f23984" alt=""
上面大家都用render_template()
就你爱用render_template_string()
是吧(指指点点,一眼模板注入,那么我们就希望file_check_res
里有我们能够执行的rce代码。file_check_res
哪来的?倒退往上看。
1 2 3 4 5 6 7 8
| f.save(filepath) file_check_res = subprocess.check_output( ["/bin/file", "-b", filepath], shell=False, encoding='utf-8', imeout=1 ) os.remove(filepath)
|
先保存filepath这样一个文件,subprocess.check_output(command)
返回Linux命令行输出,然后再把filepath文件删除,那么这里的file_check_res
就是file -b {filepath}
的结果。往上看filepath其实就是将上传文件目录的绝对路径和该文件的文件名拼接起来来标定用户上传的这个文件在容器中的绝对位置。逻辑搞明白了,现在的重点就在于如何对一个文件使用file -b
命令后返回值中能回显我们所期望的值。动手操作下flie命令,发现其不会输出文件的内容只会输出其类型,
data:image/s3,"s3://crabby-images/88dd4/88dd441b412b9f06eb6b6061acc496b99047c669" alt=""
代码中的-b参数作用:
所以在文件名上动手脚的想法也破灭了(悲。
卡住了,向大佬博客寻求帮助,去guthub查找file命令源码。第一个仓库点开。data:image/s3,"s3://crabby-images/f31db/f31db79867f2d44a35e02ef566d198d82585998a" alt=""
点开tests里面是各种针对该file
命令的测试结果
data:image/s3,"s3://crabby-images/1dfa6/1dfa61f8e7f9a56107bdbe07cf053bfd79778867" alt=""
这8个分别分别是在文本中写入bash脚本的4种情况和对应的用file
命令执行的输出结果,可以看出如果文本内容为#!/usr/bin
开头的那么输出结果中会显示文本中的其他内容。
data:image/s3,"s3://crabby-images/b2d19/b2d19b3a17a89f2838f56890639133ac7c82004a" alt=""
本地做测试:创建一个文本文件修改内容如下
data:image/s3,"s3://crabby-images/e0deb/e0debd59d344f6aad531ae08ba8b66463efd3e98" alt=""
测试结果如下:
data:image/s3,"s3://crabby-images/a32ac/a32ac1d47eb7ea7a6e46b8a4e31f8460e795b262" alt=""
显而易见,输出可控,可以进行模板渲染。新建一个文本文件内容如下,上传文件
data:image/s3,"s3://crabby-images/c8480/c8480c05deb0f95f12702f619154e1b3f3fb8736" alt=""
data:image/s3,"s3://crabby-images/5db98/5db98a5f4ac1e1e055b1a42a4938d92cd94b11f8" alt=""
存在ssti漏洞,开始利用,调用os模块
data:image/s3,"s3://crabby-images/dd479/dd479374f9ac1513ab5cfc54bc2fd753b08b3a36" alt=""
data:image/s3,"s3://crabby-images/e4251/e425116db0ef93df201ee11f222ba7a6ef426937" alt=""
调用popen()方法。
data:image/s3,"s3://crabby-images/f90cb/f90cbe5446ca0add9cc50becbe472b0f490c7f5f" alt=""
上传文件,获取flag。