ctf show web入门108

这是一道非常经典的 PHP 代码审计/CTF 题目。我们需要绕过两个关键的条件判断来获取 $flag。

题目主要有两层过滤和判断:

php 复制代码
if (ereg ("^[a-zA-Z]+$", $_GET['c'])===FALSE) {
    die('error');
}

含义:使用 ereg 正则表达式匹配输入参数 c。它要求整个字符串必须完全由大小写英文字母组成([1](#1)+$)。如果不匹配(返回 FALSE),程序就会终止并输出 error

php 复制代码
//只有36d的人才能看到flag
if(intval(strrev($_GET['c']))==0x36d){
    echo  $flag;
}

strrev($_GET'c'):将参数 c 的字符串反转。

intval(...):将反转后的字符串转换为整数。

== 0x36d:判断转换后的整数是否等于十六进制的 0x36d。

计算:十六进制的 0x36d 转换为十进制是 877

逆向推导:要让 intval 的结果为 877,那么反转前的字符串开头(或者反转后的字符串)必须代表数字 877。既然 strrev('c') 要变成以 877 开头的数字,那么原本的参数 c 就必须以 778 结尾(例如 778 反转后变成 877)

条件一要求:参数 c 只能包含纯字母。

条件二要求:参数 c 必须以数字 778 结尾。

绕过第一关:ereg() 的 %00 截断漏洞

ereg() 函数存在一个著名的漏洞:字符串截断漏洞

在 ereg 处理字符串时,如果遇到了空字符 %00(即 \x00),它会认为字符串到这里就已经结束了

构造手法:如果我们让 c 的开头全是字母,紧接着一个 %00,后面再放数字

对于 ereg 来说,它只检测 %00 前面的纯字母部分,因此会成功匹配,返回 TRUE(不触发 die)。

例如输入:a%00778

满足第二关:strrev() 与 intval() 的处理

当我们传入 c = a%00778 时,看看第二关会发生什么:

strrev("a\x00778"):反转字符串,结果变成 "877\x00a"

intval("877\x00a"):PHP 的 intval() 函数在弱类型转换时,会从字符串的开头开始读取数字,直到遇到非数字字符为止。在这里,它读取到 "877",遇到 \x00 停止,因此成功转换成了十进制整数 877

弱类型比较:877 == 0x36d(即 877 == 877),条件成立

所以最终payload为:

?c=a%00778

flag为:ctfshow{33571543-8c21-4834-9080-7e870e5ed0dd}


  1. a-zA-Z ↩︎
相关推荐
逸模10 小时前
告别熬夜手工整理台账,逸模智能归集实现项目数据自动化存档
大数据·运维·人工智能·笔记·其他·信息可视化·自动化
Frostnova丶10 小时前
【算法笔记】数学知识
笔记·算法
xqqxqxxq10 小时前
哈希表(HashMap)技术学习笔记
笔记·学习·散列表
AOwhisky10 小时前
MySQL 学习笔记(第四期):SQL 语言之多表查询
linux·运维·网络·数据库·笔记·学习·mysql
xian_wwq11 小时前
【学习笔记】「大模型安全:攻击面演化史」第 07 篇-安全左移
人工智能·笔记·学习
nnsix13 小时前
Unity 贴图压缩格式 笔记
笔记·unity·贴图
xian_wwq14 小时前
【学习笔记】「大模型安全:攻击面演化史」第 03 篇-数据投毒
笔记·学习·ai安全
sheeta199814 小时前
LeetCode 每日一题笔记 日期:2026.06.06 题目:2196. 根据描述创建二叉树
笔记·算法·leetcode
.千余15 小时前
【C++】手写双向链表:list容器模拟实现
开发语言·c++·笔记·学习·其他