PHP 反序列化漏洞:__PHP_Incomplete_Class 与 serialize(unserialize($x)) !== $x;
文章目录
- 参考
- 环境
- 声明
- __PHP_Incomplete_Class
- 灵显
- 为什么需要 __PHP_Incomplete_Class?
- 不可访问的属性
- serialize(unserialize($x)) === $x;
- serialize(unserialize($x)) !== $x;
- 雾现
- __PHP_Incomplete_Class 对象与其序列化文本的差异
- 试构造 __PHP__Incomplete_Class 对象的序列化文本
- serialize() 函数在处理 __PHP_Incomplete_Class 对象时所进行的特殊操作
- 雾散
- 下限
参考
项目 | 描述 |
---|---|
搜索引擎 | Bing、Google |
AI 大模型 | 文心一言、通义千问、讯飞星火认知大模型、ChatGPT |
PHP 手册 | PHP Manual |
Eki | PHP序列化冷知识 |
环境
项目 | 描述 |
---|---|
PHP | 8.0.0 |
PHP 编辑器 | PhpStorm 2023.1.1(专业版) |
声明
实际上 __PHP_Incomplete_Class
是一个类,暂且使用 __PHP_Incomplete_Class 对象
来指代 __PHP_Incomplete_Class
类的实例对象。
__PHP_Incomplete_Class
灵显
在 PHP 中,当你尝试将序列化文本进行反序列化操作以获得一个 对象
时,若 与序列化文本相关联的类还没有在当前 PHP 上下文中被定义或包含时
,PHP 就会使用 __PHP_Incomplete_Class
对象来代替这个对象。对此,请参考如下示例:
<?php$result = unserialize('O:7:"MyClass":2:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}');
var_dump($result);
执行效果
由于在使用 unserialize()
函数将序列化文本反序列化为对象时,相关的类并尚未被定义或被包含
,于是 PHP 使用 __PHP_Incomplete_Class
对象作为反序列化操作的结果。
object(__PHP_Incomplete_Class)#1 (3) {["__PHP_Incomplete_Class_Name"]=>string(7) "MyClass"["name"]=>string(8) "RedHeart"["nation"]=>string(5) "China"
}
__PHP_Incomplete_Class
对象中包含了 序列化文本尝试创建的对象的信息
,包括了该对象 所属类的名称
以及 该对象的属性及其值
。
为什么需要 __PHP_Incomplete_Class?
PHP 通过这种方式来 防止因错误导致程序的崩溃,提高程序的可靠性与可用性
。如果 PHP 试图反序列化一个不存在的类,而没有任何 后备机制
,那么这可能会导致 致命错误或者不可预期的行为
。通过使用 __PHP_Incomplete_Class
对象,PHP 可以告诉您在反序列过程中出现的问题并继续运行。
不可访问的属性
__PHP_Incomplete_Class
是一个特殊的对象,当你试图访问这个对象的属性时,PHP 将抛出 Warning
异常。当你尝试访问这个 不完整
的对象的属性时,PHP 会 认为这是不安全的(所属类的定义并未在当前上下文中给出。记住,不要随便接收陌生人给的东西🤐
),并给出警告。对此,请参考如下示例:
<?php$result = unserialize('O:7:"MyClass":2:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}');var_dump($result);# 尝试访问 __PHP_Incomplete_Class 对象的属性
var_dump($result -> name);
执行效果
PHP Warning: main(): The script tried to access a property on an incomplete object. Please ensure that the class definition "MyClass" of the object you are trying to operate on was loaded _before_ unserialize() gets called or provide an autoloader to load the class definition in C:\test.php on line 9
object(__PHP_Incomplete_Class)#1 (3) {["__PHP_Incomplete_Class_Name"]=>string(7) "MyClass"["name"]=>string(8) "RedHeart"["nation"]=>string(5) "China"
}Warning: main(): The script tried to access a property on an incomplete object. Please ensure that the class definition "MyClass" of the object you are trying to operate on was loaded _before_ unserialize() gets called or provide an autoloader to load the class definition in C:\test.php on line 9
NULL
serialize(unserialize($x)) === $x;
在 serialize(unserialize($x)) === $x;
中,$x
为一个序列化文本,将一个序列化文本通过 unserialize()
进行反序列化操作后将得到该序列化文本所描述的数据格式,再将这数据格式用序列化文本进行描述将得到原内容即 $x
。
这一操作就像执行了 1 + 1
后再执行了 1 - 1
,两个操作相互抵消。在对数据进行处理前与数据进行处理后,数据相等。所以毫无悬念的 serialize(unserialize($x) ) === $x;
。对此,请参考如下示例:
<?php$serialize_text = 'O:7:"MyClass":2:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}';var_dump(serialize(unserialize($serialize_text)) === $serialize_text);
执行效果
bool(true)
serialize(unserialize($x)) !== $x;
雾现
__PHP_Incomplete_Class
的出现使 serialize(unserialize($x)) !== $x;
也成为了可能。对此,请先参考如下示例,稍后我将对其进行解释。
<?php$serialize_text = 'O:22:"__PHP_Incomplete_Class":2:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}';var_dump(serialize(unserialize($serialize_text)) !== $serialize_text);
执行效果
bool(true)
__PHP_Incomplete_Class 对象与其序列化文本的差异
序列化文本
O:7:"MyClass":2:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}
对象 __PHP_Incomplete_Class
object(__PHP_Incomplete_Class)#1 (3) {["__PHP_Incomplete_Class_Name"]=>string(7) "MyClass"["name"]=>string(8) "RedHeart"["nation"]=>string(5) "China"
}
序列化文本
与其对应的 __PHP_Incomplete_Class
对象存在如下差异:
- 所属类名称
序列化文本在反序列化为__PHP_Incomplete_Class
对象后,对象所属类的名称由MyClass
变为了其__PHP__Incomplete_Class
。 - 对象的属性个数
序列化文本所描述的属性个数要比__PHP_Incomplete_Class
对象的属性个数少1
。 - __PHP_Incomplete_Class_Name 属性
__PHP_Incomplete_Class
对象中包含了__PHP_Incomplete_Class_Name
属性,而其序列化文本中则没有与该属性相关的描述。
试构造 __PHP__Incomplete_Class 对象的序列化文本
__PHP__Incomplete_Class
与其序列化文本存在差异的原因是 PHP 发现了这个对象是一个 __PHP_Incomplete_Class
对象。而 PHP 是通过检查被序列化对象所属类的名称发现的。如果我们尝试描述一个所属类为 __PHP_Incomplete_Class
的对象的序列化文本,这会发生什么有趣的事情呢?为此,请参考如下示例:
<?phpvar_dump(unserialize('O:22:"__PHP_InComplete_Class":2:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}'));
执行效果
由于 PHP 上下文中已经包含了 __PHP_Incomplete_Class
的类定义,所以当我们将序列化文本进行反序列化后并没有产生一个描述 __PHP_Incomplete_Class
对象的另一个 __PHP_Incomplete_Class
对象。
object(__PHP_Incomplete_Class)#2 (2) {["name"]=>string(8) "RedHeart"["nation"]=>string(5) "China"
}
如果我们尝试将这个人为构造的 __PHP_Incomplete_Class
对象进行序列化操作将会发生什么呢?对此,请参考如下示例:
<?phpvar_dump(serialize(unserialize('O:22:"__PHP_Incomplete_Class":2:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}')));
执行效果
string(85) "O:22:"__PHP_Incomplete_Class":1:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}"
神奇的事情发生了。在将人为构造的 __PHP_Incomplete_Class
对象进行序列化后,序列化文本中描述对象属性个数的数值由原先的 2
变为了 1
。
serialize() 函数在处理 __PHP_Incomplete_Class 对象时所进行的特殊操作
unserialize()
在发现当前 PHP 上下文中没有包含相关类的类定义时将创建一个 __PHP_Incomplete_Class
对象。而 serialize()
在发现需要进行序列化的对象是 __PHP_Incomplete_Class
后,将对其进行 特殊处理
以得到描述实际对象而非 __PHP_Incomplete_Class
对象的序列化文本,而这里就包含了 将属性的描述值减一
这一步。
那么对象所属类的名称是否会发生替换,序列化文本中的 __PHP_Incomplete_Class_Name
是否会被自动删除以使得序列化文本中的属性个数描述值与实际相符呢?对此,请参考如下示例:
<?phpvar_dump(serialize(unserialize('O:22:"__PHP_Incomplete_Class":3:{s:27:"__PHP_Incomplete_Class_Name";s:7:"MyClass";s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}')));
执行效果
string(69) "O:7:"MyClass":2:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}"
结合前面观察到的种种现象,我们可以总结出 serialize()
函数对 __PHP_Incomplete_Class
对象执行了如下 特殊操作(操作描述顺序并非 serialize 函数的实际操作顺序)
:
- 将
__PHP_Incomplete_Class
对象中的属性个数减一
并将其作为序列化文本中对实际对象属性个数的描述值
。 - 将
__PHP_Incomplete_Class
对象的__PHP_Incomplete_Class_Name
作为序列化文本中对象所属类的描述
值。若未从__PHP_Incomplete_Class
对象 中检查到__PHP_Incomplete_Class_Name
属性,则跳过此步。 - 将
__PHP_Incomplete_Class
对象的序列化文本中对__PHP_Incomplete_Class_Name
属性的描述删去。若没有发现相关描述,则跳过此步。
雾散
回到 serialize(unserialize($x)) !== $x;
。让我们再尝试对如下示例进行分析:
<?php$serialize_text = 'O:22:"__PHP_Incomplete_Class":2:{s:4:"name";s:8:"RedHeart";s:6:"nation";s:5:"China";}';var_dump(serialize(unserialize($serialize_text)) !== $serialize_text);
执行效果
bool(true)
由于 serialize
函数将对 __PHP_Incomplete_Class
进行特殊操作,故反序列化后得到的 __PHP_Incomplete_Class
对象再进行序列化操作后,序列化文本中对属性个数的描述值将为 1
而不是 2
。于是导致了 serialize(unserialize($x)) !== $x;
。
下限
在 serialize()
函数对 __PHP_Incomplete_Class
对象执行特殊操作的过程中,若 __PHP_Incomplete_Class
对象中的属性个数为零,则 __PHP_Incomplete_Class
的序列化结果中的属性个数描述值也将为零,两者不会存在 1
的差距。对此,请参考如下示例:
<?phpvar_dump(serialize(unserialize('O:22:"__PHP_Incomplete_Class":0:{}')));
执行效果
string(34) "O:22:"__PHP_Incomplete_Class":0:{}"
当 对象(由反序列化操作得到的对象)
所属类为 __PHP_Incomplete_Class
且该对象的属性个数 不为零
时,serialize(unserialize($x)) !== $x;
恒成立,仅当该对象的属性个数 为零
时, serialize(unserialize($x)) === $x;
成立。
相关文章:

PHP 反序列化漏洞:__PHP_Incomplete_Class 与 serialize(unserialize($x)) !== $x;
文章目录 参考环境声明__PHP_Incomplete_Class灵显为什么需要 __PHP_Incomplete_Class?不可访问的属性 serialize(unserialize($x)) $x;serialize(unserialize($x)) ! $x;雾现__PHP_Incomplete_Class 对象与其序列化文本的差异试构造 __PHP__Incomplete_Class 对象…...

TempleteMethod
TempleteMethod 动机 在软件构建过程中,对于某一项任务,它常常有稳定的整体操作结构,但各个子步骤却有很多改变的需求,或者由于固有的原因 (比如框架与应用之间的关系)而无法和任务的整体结构同时实现。如…...

1558. 得到目标数组的最少函数调用次数
1558. 得到目标数组的最少函数调用次数 原题链接:完成情况:解题思路:参考代码: 原题链接: 1558. 得到目标数组的最少函数调用次数 https://leetcode.cn/problems/minimum-numbers-of-function-calls-to-make-target…...
子域名扫描, 后台扫描
子域名和后台扫描 一, 子域名扫描 在渗透测试的早期阶段,子域名扫描是一个非常重要的步骤,它有助于识别目标组织的网络结构和在线资源。 子域名扫描应该在获得适当的权限和授权的情况下进行,以确保所有活动都是合法和合规的。 1. 原因与目…...

毛玻璃带有光影效果的卡片
效果展示 页面结构组成 从效果展示可以看到,页面的主要元素是卡片,卡片的内容呈现上都是比较常规的布局,只是卡片上带有光影效果。 CSS / JavaScript 知识点 transformVanillaTilt.js 使用 页面基础结构实现 <div class"contain…...
【Java】面向过程和面向对象思想||对象和类
1.面向过程和面向对象思想 两者都贯穿于软件分析、设计和开发的各个阶段,对应面向对象就分别称为面向对象的分析(OOA)、面向对象的设计(OOD)和面向对象的编程(OOP)。C语言是一种典型的面向过程语…...

孤举者难起,众行者易趋,openGauss 5.1.0版本正式发布!
📢📢📢📣📣📣 哈喽!大家好,我是【IT邦德】,江湖人称jeames007,10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】!😜&am…...

软考——软件设计师中级2023年11月备考(1.计算机组成原理)
一、计算机组成原理 1.数据的表示 1.1 十进制转R进制 方法:对十进制数除R取余,最后对余数取倒序 如: 1.2 原码反码补码 1.3 浮点数 1.4 校验码 —— 海明码 (非重点,了解即可) 海明码的构成方法&…...

前端JavaScript入门到精通,javascript核心进阶ES6语法、API、js高级等基础知识和实战 —— Web APIs(四)
思维导图 一、日期对象 1.1 实例化 实例化,默认得到当前时间,也可以指定时间 1.2 日期对象方法 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible&q…...
【前端】HTML5 Audio 预加载 按照队列顺序播放音频, 可以陆续往队列中加内容
【前端】Audio 按照队列顺序播放音频, 可以陆续往队列中加内容 var 音频库 {} var 当前音频集合 [] /*** 将文本添加到队列中* 持续去播放* 播放过的音频会自动从队列中删除* * 已规划* 要保障同时进行加载的数据不能超过5个(线程池 5)* * param 文本*/播放音频队列(文本){i…...

【单片机】13-实时时钟DS1302
1.RTC的简介 1.什么是实时时钟(RTC) (rtc for real time clock) (1)时间点和时间段的概念区分 (2)单片机为什么需要时间点【一定的时间点干什么事情】 (3)RTC如何存在于…...
springboot和vue:十三、VueX简介与安装与推荐视频+前端数据模拟MockJS
VueX简介与安装与推荐视频 VueX用于管理分散在vue各个组件中的数据。每一个VueX的核心都是一个store,当store中的状态发生变化时,与之绑定的视图也将重新渲染。store中的状态不允许被直接修改,只能显示提交mutationVueX中有五个重要的概念&a…...

[React] Zustand状态管理库
文章目录 1.Zustand介绍2.创建一个store3.使用方法3.1 获取状态3.2 更新状态3.3 访问存储状态3.4 处理异步数据3.5 在状态中访问和存储数组3.6 持续状态 4.总结 1.Zustand介绍 状态管理一直是现代程序应用中的重要组成部分, Zustand使用 hooks 来管理状态无需样板代码。 更少…...
【ChatGPT】ChatGPT发展历史
更多优质文章请看底部:ChatGPT与日本首相交流核废水事件-精准Prompt... hello,我是小索奇,在AI日益庞大的环境下,接下来将为大家不断的ChatGPT学习 ChatGPT使用了 Transformer 结构,建立在 OpenAI的 GPT-3.5 大型语言模…...

分布式文件存储系统Minio实战
分布式文件系统应用场景 互联网海量非结构化数据的存储需求电商网站:海量商品图片视频网站:海量视频文件网盘 : 海量文件社交网站:海量图片 1. Minio介绍 MinIO 是一个基于Apache License v2.0开源协议的对象存储服务。它兼容亚马逊S3云存…...

【MySQL】MySQL 官方安装包形式
MySQL 官方提供3种包: 1. 源码包 mysql-5.7.42.tar.gz mysql-5.7.42-aarch64.tar.gz http://dev.mysql.com/get/Downloads/MySQL-5.6/mysql-5.6.34.tar.gz http://dev.mysql.com/get/Downloads/MySQL-5.7/mysql-5.7.42.tar.gz需要用户根据自己的CPU架构选择对应的…...

使用sqlmap获取数据步骤
文章目录 1.使用sqlmap获取所有数据库2.使用sqlmap获取当前连接数据库3.使用sqlmap获取当前数据库下所有表名4.使用sqlmap获取当前数据库下某个表下所有列名5.使用sqlmap获取当前数据库下某个表下指定字段的数据6.测试当前用户是否是管理员7.使用burpsqlmap批量检测8.脱库命令9…...
[论文笔记]GLM
引言 今天带来论文GLM: General Language Model Pretraining with Autoregressive Blank Infilling的笔记。论文中文标题为 通用语言模型预训练与自回归填空。 有很多不同类型的预训练架构,包括自编码模型(BERT、RoBERTa、ALBERT)、自回归模型(GPT系列)以及编码器-解码器模型…...

漏洞扫描环境:win10系统用VMware Workstation打开虚拟机若干问题
win10系统用VMware Workstation打开虚拟机若干问题 一 .VMware打开虚拟机就蓝屏重启怎么解决?一. VMware打开虚拟机就蓝屏重启怎么解决?方法一:1、同时按下CTRLSHIFTESC打开任务管理器功能,之后依次点击-详细信息-性能后出现下列界…...

OpenCV实现模板匹配和霍夫线检测,霍夫圆检测
一,模板匹配 1.1代码实现 import cv2 as cv import numpy as np import matplotlib.pyplot as plt from pylab import mplmpl.rcParams[font.sans-serif] [SimHei]#图像和模板的读取 img cv.imread("cat.png") template cv.imread(r"E:\All_in\o…...

wordpress后台更新后 前端没变化的解决方法
使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…...
利用ngx_stream_return_module构建简易 TCP/UDP 响应网关
一、模块概述 ngx_stream_return_module 提供了一个极简的指令: return <value>;在收到客户端连接后,立即将 <value> 写回并关闭连接。<value> 支持内嵌文本和内置变量(如 $time_iso8601、$remote_addr 等)&a…...

《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》
在注意力分散、内容高度同质化的时代,情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现,消费者对内容的“有感”程度,正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中࿰…...

智能在线客服平台:数字化时代企业连接用户的 AI 中枢
随着互联网技术的飞速发展,消费者期望能够随时随地与企业进行交流。在线客服平台作为连接企业与客户的重要桥梁,不仅优化了客户体验,还提升了企业的服务效率和市场竞争力。本文将探讨在线客服平台的重要性、技术进展、实际应用,并…...
多模态商品数据接口:融合图像、语音与文字的下一代商品详情体验
一、多模态商品数据接口的技术架构 (一)多模态数据融合引擎 跨模态语义对齐 通过Transformer架构实现图像、语音、文字的语义关联。例如,当用户上传一张“蓝色连衣裙”的图片时,接口可自动提取图像中的颜色(RGB值&…...
Qwen3-Embedding-0.6B深度解析:多语言语义检索的轻量级利器
第一章 引言:语义表示的新时代挑战与Qwen3的破局之路 1.1 文本嵌入的核心价值与技术演进 在人工智能领域,文本嵌入技术如同连接自然语言与机器理解的“神经突触”——它将人类语言转化为计算机可计算的语义向量,支撑着搜索引擎、推荐系统、…...
生成 Git SSH 证书
🔑 1. 生成 SSH 密钥对 在终端(Windows 使用 Git Bash,Mac/Linux 使用 Terminal)执行命令: ssh-keygen -t rsa -b 4096 -C "your_emailexample.com" 参数说明: -t rsa&#x…...

第一篇:Agent2Agent (A2A) 协议——协作式人工智能的黎明
AI 领域的快速发展正在催生一个新时代,智能代理(agents)不再是孤立的个体,而是能够像一个数字团队一样协作。然而,当前 AI 生态系统的碎片化阻碍了这一愿景的实现,导致了“AI 巴别塔问题”——不同代理之间…...

Module Federation 和 Native Federation 的比较
前言 Module Federation 是 Webpack 5 引入的微前端架构方案,允许不同独立构建的应用在运行时动态共享模块。 Native Federation 是 Angular 官方基于 Module Federation 理念实现的专为 Angular 优化的微前端方案。 概念解析 Module Federation (模块联邦) Modul…...

【JVM面试篇】高频八股汇总——类加载和类加载器
目录 1. 讲一下类加载过程? 2. Java创建对象的过程? 3. 对象的生命周期? 4. 类加载器有哪些? 5. 双亲委派模型的作用(好处)? 6. 讲一下类的加载和双亲委派原则? 7. 双亲委派模…...