相关推荐recommended
2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】
作者:mmseoamin日期:2024-01-19

一、题目如下:

2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第1张


二、代码解读:

这段代码是一个简单的PHP脚本,它接受通过GET请求传递的两个参数:‘pass’和’func’:

① $password = trim($_GET['pass'] ?? '');:从GET请求中获取名为’pass’的参数,然后使用trim函数去除首尾空格,并将结果赋给变量 $password。如果’pass’参数不存在,则使用空字符串。

② $func = trim($_GET['func'] ?? 'hint');:从GET请求中获取名为’func’的参数,然后使用trim函数去除首尾空格,并将结果赋给变量$func。如果’func’参数不存在,则默认使用字符串’hint’。

③ function hint() {show_source(__FILE__);} :定义了一个名为hint的函数,它的作用是显示当前文件的源代码,使用了show_source函数。

④ if (md5(\$password)==='21c6008facc283f8839d3b9fed640c15') {:检查通过GET请求传递的’pass’参数的MD5哈希值是否等于指定的值。如果匹配,进入条件块。

⑤ function youwin() {echo file_get_contents("/flag");}:在条件成立时,定义了一个名为youwin的函数,它的作用是输出名为’/flag’的文件的内容,使用了file_get_contents函数。

⑥ $func();:调用根据’func’参数确定的函数。默认情况下,如果’func’参数不存在,将调用hint函数。


二、解题思路:

常规思路就是绕过 md5 效验。但是这里是强比较,显然无法绕过。尝试参数爆破,看有没有除pass、func 额外的请求参数, bp 跑参数字典:

2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第2张

果然发现了一个 file 请求参数,使用 readfile() 函数读取文件,尝试任意文件读取:

2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第3张尝试读取 flag, 发现存在拦截:

2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第4张

readfile() 可以直接读取 php 文件,且php代码不会被解析

读取当前文件源代码:

2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第5张尝试绕过 preg_match 正则,传递数组参数:

2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第6张成功绕过,但是 readfile() 不接受数组作为参数, 网上没找到 readfile() 相关漏洞的文献资料。故放弃…


想起上上周看到 p 牛10月份写的一篇文章,提到了 PHP 临时函数名特性 , 且给的题目案例跟这题很像。 ==> 2023年10月PHP函数小挑战

PHP在编译“函数定义”的时候,会使用 zend_compile_func_decl 函数。这个函数有个关键的参数叫 toplevel,这个参数表示当前的函数定义是否在顶层作用域。

顶层作用域: 顶层作用域通常是指在全局范围内声明的变量、函数或类,而不是在任何函数或控制结构内声明的。


    echo 'func1';
}
if (true) {
	// 在if条件判断内部定义的函数,不是顶层作用域
    function func2() {
        echo 'func2';
    }
}

当toplevel为true的时候,就是直接将当前函数名 lcname 加入函数表;当 toplevel为false的时候,使用 zend_build_runtime_definition_key 函数生成一个 key,将 key 作为函数名加入函数表。也就是说,根据函数所在的位置的不同(是否是顶级作用域),PHP编译时生成的函数名也会不同。

2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第7张

key 是按照如下算法生成:

'
  • name 函数名
  • ' + name + filename + ':' + start_lineno + '$' + rtd_key_counter

    除了第一个 \0 字符,后面四部分的含义如下:

    • filename PHP文件绝对路径
    • start_lineno 函数起始定义行号(以1为第一行)
    •  ./php -d vld.execute=0 -d vld.active=1  E:\phpstudy_pro\WWW\php-learn5ee283f4c848ad99b0a7961ed89199.php
      
    • rtd_key_counter 一个全局访问计数,每次执行会自增1,从0开始

      最后保存在函数表中的函数名,就是上面这个以 \0 开头的字符串。


      以上是看 p 牛的文章简要梳理一下核心知识点。但是我发现一个问题:

      2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第8张图中说只要构造:'\0' + name + filename + ':' + start_lineno + '$' + rtd_key_counter 调用就可以了,但是后面图中 payload 又是:

      '%00' + name + filename + ':' + start_lineno + '$' + rtd_key_counter 这种方式:

      2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第9张2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第10张

      为什么又变成了 %00 而不是 \0 呢?小小的探究一下:

      前提:windows,php 7.4 (要限制7.4版本,稍后会说), vld.dll 内存监控插件 (要下载和php版本相对应的)

      本地编写和比赛环境一样的代码,文件名为:965ee283f4c848ad99b0a7961ed89199.php,使用php7.4 配合vld插件解释运行:

      php 底层是由C语言编写的,在C语言中,
      %00youwin/var/www/html/965ee283f4c848ad99b0a7961ed89199.php:7$rtd_key_counter
      
      和%00实际上有相似的作用,都表示空字符(null character)。
      \%00youwin/var/www/html/965ee283f4c848ad99b0a7961ed89199.php:7$rtd_key_counter
      
      是C语言中用于表示空字符的一种方式,通常用在字符串中作为字符串的结束标志。%00通常是在格式化输入输出函数中使用的格式说明符,表示空字符。

      2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第11张2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第12张可以看到实际的临时函数名确实是以 %00 开头的。seo积分系统 我们请求参数被php接收并解释运行,也是一个输入的过程,所以要使用 %00 为开头构建。(个人见解)


      为什么要限制 php7.4 这个版本去复现呢?

      我尝试使用 php7.3 版本配合vld插件去解释运行相同的代码,得到如下:

      2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第13张2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第14张不同版本有差异性,7.4 版本结尾更加具有规律性,能爆破。


      按照上面的思路,按照 zend_build_runtime_definition_key 的算法计算出 key 作为函数名:(rtd_key_counter 从 0 开始爆破)

      网站设计公司排名

      比赛环境没了,本地搭了一个。

      发现爆破不出flag:

      2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第15张

      trim 函数在接收参数的时候会去除掉字符串首尾的空白字符。可以加反斜杠绕过:\%00

      故要改写成如下:

      上海做网站公司

      bp 开始爆破 rtd_key_counter 得 flag:

      2023年第四届 “赣网杯” 网络安全大赛 gwb-web3 Write UP【PHP 临时函数名特性 + 绕过trim函数】,在这里插入图片描述,第16张