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 ↩︎
相关推荐
LinXunFeng8 天前
Obsidian - 使用 Share Note 分享笔记并自部署
前端·笔记·github
闪闪发亮的小星星12 天前
高斯光以及高斯光公式解释
笔记
cqbzcsq12 天前
CellFlow虚拟细胞论文阅读
论文阅读·人工智能·笔记·学习·生物信息
阿米亚波12 天前
【Windows】QEMU 启动 openEuler aarch64/arm64 架构系统 + 离线软件源
linux·windows·经验分享·笔记·架构·arm
自传.12 天前
尚硅谷 Vibe Coding|第三章(1) Claude Code深度使用与进阶技巧 学习笔记
笔记·学习·尚硅谷·vibecoding
.千余12 天前
【C++】模板进阶全解:非类型参数|全特化|偏特化|分离编译完全指南
开发语言·c++·笔记·学习·其他
自传.12 天前
尚硅谷 Vibe Coding|第二章 AI编程工具生态 学习笔记
笔记·学习·ai编程·尚硅谷·vibe coding
秋波。未央12 天前
Java Agent 开发 · Day 1 学习笔记(含作业完整标准答案)
java·笔记·学习
中屹指纹浏览器12 天前
2026指纹浏览器字体指纹、字体渲染偏差检测与全维度虚拟字体池搭建方案
经验分享·笔记