ctf show web入门99

打开靶场,通过代码审计我们发现这是一道通过合法的参数绕过条件限制,最终利用 file_put_contents 实现任意文件写入的题目,我们的主要目的就是把一个木马写入该网站,然后拿flag

php 复制代码
$allow = array();
for ($i=36; $i < 0x36d; $i++) { 
    array_push($allow, rand(1, $i));
}

这串代码的意思是重复877-36次每次迭代rand(1, i) 生成一个 1 到 i之间的随机整数arraypush(i 之间的随机整数array_push(i之间的随机整数arraypush(allow, ...) 将该随机数追加到 allow 数组的末尾

数组 $allow 的变化

复制代码
假设执行前 $allow 未定义(或为空),PHP 会将其自动创建为数组(但会抛出一个 Notice: Undefined variable)。建议先 $allow = [];。

最终 $allow 是一个索引数组,键从 0 到 840,共 841 个元素。

随机数的范围

复制代码
第 1 轮($i=36):随机范围 [1, 36]

第 2 轮($i=37):范围 [1, 37]

...

最后一轮($i=876):范围 [1, 876]

随着循环进行,随机数的上限逐渐增大,后面生成的数值可能更大(但仍然是均匀分布的整数)

所以1到36包含在这个数组的可能性最大

php 复制代码
if(isset($_GET['n']) && in_array($_GET['n'], $allow)){
    file_put_contents($_GET['n'], $_POST['content']);
}

条件:我们需要传入一个 _GET\['n'\],并且这个值必须存在于 allow 数组中。

结果:如果条件满足,程序会创建/写入一个名为 _GET\['n'\] 的文件,内容为 _POST'content'

这道题的破绽在于 PHP 的 弱类型比较。

在 PHP 中,in_array(value, array) 函数默认是弱类型检查(没有设置第三个参数为 true)。当它拿一个字符串去和一个整数进行比较时,PHP 会尝试将字符串转换为数字。

弱类型转换规则:

如果一个字符串以数字开头,PHP 在进行数学比较或类型转换时,会截取开头的数字部分,忽略后面的非数字字符。

例如:

"1.php" 会被转换为整数 1

因为1到36被包含的概率最大所以我们的文件名为1.php

然后我们就可以开始构造payload

我们先创建一个名为"1.php"的文件,并且在文件内写上一个一句话木马执行后我们在打开一个原url+1.php网站,通过psot方式执行命令

这里通过1=system('ls');命令列出当前目录的文件和子文件夹发现有一个flag36d.php

判断flag应该藏在这个文件下,我们尝试读取该文件内容

通过1=system('tac flag36d.php');命令查看该文件下的内容

得到flag为:ctfshow{024752ba-1087-4064-9c44-36f4887f0aa9}

相关推荐
IT_陈寒40 分钟前
Vue这个坑我跳了两次,原来问题出在这
前端·人工智能·后端
kyriewen43 分钟前
我用 50 行代码重写了 React Router 核心,终于搞懂了前端路由原理
前端·javascript·react.js
WebInfra2 小时前
Rspack 2.1 发布:React Compiler 提速 10 倍!
前端
李明卫杭州2 小时前
CSS 媒体查询详解:一文掌握响应式设计的核心技术
前端
lichenyang4532 小时前
从 H5 按钮到 OpenHarmony 能力调用:我如何理解 ASCF 的运行链路
前端
下家3 小时前
我放弃了 Vue/React,选择自研框架
前端·前端框架
Asize3 小时前
HTML5 Canvas 基础:从按帧动画到 ECharts 数据可视化
前端·javascript·canvas
默_笙3 小时前
🎄 后端给我一堆扁平数据,我 10 行代码把它变成了树
前端·javascript
Mahut3 小时前
我用 Electron + FFmpeg 做了一个本地视频处理工作站 ClipForge
前端·ffmpeg·electron
前端Hardy4 小时前
又一个 AI 神器火了!
前端·javascript·后端