PHP开源系统学习之fluxbb_2

谴责下某位同学,转载了我的上一篇文章,也不给个原文地址,希望这次再来转时能加上。

//检查登录,在common.php判断

//cookie串: 2|dc4fab5bb354be5104bae0affe2c1b615c565cf5|1384165106|eb084e693bb241a29e9986b62e69cf5f465f354d

//cookie有效期大于当前时间=》继续判断:用户cookie中保存的用户ID、生存周期、及预设置的cookie密钥,经过特殊处理后与$cookie['cookie_hash']比较

 1 function check_cookie(&$pun_user)

 2 {

 3     global $db, $db_type, $pun_config, $cookie_name, $cookie_seed;

 4 

 5     $now = time();

 6 

 7     // If the cookie is set and it matches the correct pattern, then read the values from it

 8     if (isset($_COOKIE[$cookie_name]) && preg_match('%^(\d+)\|([0-9a-fA-F]+)\|(\d+)\|([0-9a-fA-F]+)$%', $_COOKIE[$cookie_name], $matches))

 9     {

10         $cookie = array(

11             'user_id'            => intval($matches[1]),

12             'password_hash'     => $matches[2],

13             'expiration_time'    => intval($matches[3]),

14             'cookie_hash'        => $matches[4],

15         );

16     }

17 

18     // If it has a non-guest user, and hasn't expired

19     if (isset($cookie) && $cookie['user_id'] > 1 && $cookie['expiration_time'] > $now)

20     {

21         // If the cookie has been tampered with

22         if (forum_hmac($cookie['user_id'].'|'.$cookie['expiration_time'], $cookie_seed.'_cookie_hash') != $cookie['cookie_hash'])

23         {

24             $expire = $now + 31536000; // The cookie expires after a year

25             pun_setcookie(1, pun_hash(uniqid(rand(), true)), $expire);

26             set_default_user();

27 

28             return;

29         }

30 ...............

//从上面得到的$cookie变量中的user_id,去数据库查询取到密码,特殊处理后与$cookie中的password_hash比较

 1 // Check if there's a user with the user ID and password hash from the cookie

 2         $result = $db->query('SELECT u.*, g.*, o.logged, o.idle FROM '.$db->prefix.'users AS u INNER JOIN '.$db->prefix.'groups AS g ON u.group_id=g.g_id LEFT JOIN '.$db->prefix.'online AS o ON o.user_id=u.id WHERE u.id='.intval($cookie['user_id'])) or error('Unable to fetch user information', __FILE__, __LINE__, $db->error());

 3         $pun_user = $db->fetch_assoc($result);

 4 

 5         // If user authorisation failed

 6         if (!isset($pun_user['id']) || forum_hmac($pun_user['password'], $cookie_seed.'_password_hash') !== $cookie['password_hash'])

 7         {

 8             $expire = $now + 31536000; // The cookie expires after a year

 9             pun_setcookie(1, pun_hash(uniqid(rand(), true)), $expire);

10             set_default_user();

11 

12             return;

13         }

//退出 "login.php?action=out&id=2&csrf_token=40ce5abbaf76ba6fe66bb8a833eb9e8d4da5c273" ,此处没有直接删除cookie

//先判断用户是否已登录,链接是否有相应的标识

 1 if ($action == 'out')

 2 {

 3     if ($pun_user['is_guest'] || !isset($_GET['id']) || $_GET['id'] != $pun_user['id'] || !isset($_GET['csrf_token']) || $_GET['csrf_token'] != pun_hash($pun_user['id'].pun_hash(get_remote_address())))

 4     {

 5         header('Location: index.php');

 6         exit;

 7     }

 8 

 9     // Remove user from "users online" list

10     $db->query('DELETE FROM '.$db->prefix.'online WHERE user_id='.$pun_user['id']) or error('Unable to delete from online list', __FILE__, __LINE__, $db->error());

11 

12     // Update last_visit (make sure there's something to update it with)

13     if (isset($pun_user['logged']))

14         $db->query('UPDATE '.$db->prefix.'users SET last_visit='.$pun_user['logged'].' WHERE id='.$pun_user['id']) or error('Unable to update user visit data', __FILE__, __LINE__, $db->error());

15 

16     pun_setcookie(1, pun_hash(uniqid(rand(), true)), time() + 31536000);

17 

18     redirect('index.php', $lang_login['Logout redirect']);

19 }

//get_remote_address()取远程IP,区分了是否用了代理

 1 function get_remote_address()

 2 {

 3     $remote_addr = $_SERVER['REMOTE_ADDR'];

 4 

 5     // If we are behind a reverse proxy try to find the real users IP

 6     if (defined('FORUM_BEHIND_REVERSE_PROXY'))

 7     {

 8         if (isset($_SERVER['HTTP_X_FORWARDED_FOR']))

 9         {

10             // The general format of the field is:

11             // X-Forwarded-For: client1, proxy1, proxy2

12             // where the value is a comma+space separated list of IP addresses, the left-most being the farthest downstream client,

13             // and each successive proxy that passed the request adding the IP address where it received the request from.

14             $forwarded_for = explode(',', $_SERVER['HTTP_X_FORWARDED_FOR']);

15             $forwarded_for = trim($forwarded_for[0]);

16 

17             if (@preg_match('%^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}$%', $forwarded_for) || @preg_match('%^((([0-9A-Fa-f]{1,4}:){7}[0-9A-Fa-f]{1,4})|(([0-9A-Fa-f]{1,4}:){6}:[0-9A-Fa-f]{1,4})|(([0-9A-Fa-f]{1,4}:){5}:([0-9A-Fa-f]{1,4}:)?[0-9A-Fa-f]{1,4})|(([0-9A-Fa-f]{1,4}:){4}:([0-9A-Fa-f]{1,4}:){0,2}[0-9A-Fa-f]{1,4})|(([0-9A-Fa-f]{1,4}:){3}:([0-9A-Fa-f]{1,4}:){0,3}[0-9A-Fa-f]{1,4})|(([0-9A-Fa-f]{1,4}:){2}:([0-9A-Fa-f]{1,4}:){0,4}[0-9A-Fa-f]{1,4})|(([0-9A-Fa-f]{1,4}:){6}((\b((25[0-5])|(1\d{2})|(2[0-4]\d)|(\d{1,2}))\b)\.){3}(\b((25[0-5])|(1\d{2})|(2[0-4]\d)|(\d{1,2}))\b))|(([0-9A-Fa-f]{1,4}:){0,5}:((\b((25[0-5])|(1\d{2})|(2[0-4]\d)|(\d{1,2}))\b)\.){3}(\b((25[0-5])|(1\d{2})|(2[0-4]\d)|(\d{1,2}))\b))|(::([0-9A-Fa-f]{1,4}:){0,5}((\b((25[0-5])|(1\d{2})|(2[0-4]\d)|(\d{1,2}))\b)\.){3}(\b((25[0-5])|(1\d{2})|(2[0-4]\d)|(\d{1,2}))\b))|([0-9A-Fa-f]{1,4}::([0-9A-Fa-f]{1,4}:){0,5}[0-9A-Fa-f]{1,4})|(::([0-9A-Fa-f]{1,4}:){0,6}[0-9A-Fa-f]{1,4})|(([0-9A-Fa-f]{1,4}:){1,7}:))$%', $forwarded_for))

18                 $remote_addr = $forwarded_for;

19         }

20     }

21 

22     return $remote_addr;

23 }

//退出时cookie处理

1 pun_setcookie(1, pun_hash(uniqid(rand(), true)), time() + 31536000);
 1 // Set a cookie, FluxBB style!

 2 // Wrapper for forum_setcookie

 3 //

 4 function pun_setcookie($user_id, $password_hash, $expire)

 5 {

 6     global $cookie_name, $cookie_seed;

 7 

 8     forum_setcookie($cookie_name, $user_id.'|'.forum_hmac($password_hash, $cookie_seed.'_password_hash').'|'.$expire.'|'.forum_hmac($user_id.'|'.$expire, $cookie_seed.'_cookie_hash'), $expire);

 9 }

10 

11 

12 //

13 // Set a cookie, FluxBB style!

14 //

15 function forum_setcookie($name, $value, $expire)

16 {

17     global $cookie_path, $cookie_domain, $cookie_secure, $pun_config;

18 

19     if ($expire - time() - $pun_config['o_timeout_visit'] < 1)

20         $expire = 0;

21 

22     // Enable sending of a P3P header

23     header('P3P: CP="CUR ADM"');

24 

25     if (version_compare(PHP_VERSION, '5.2.0', '>='))

26         setcookie($name, $value, $expire, $cookie_path, $cookie_domain, $cookie_secure, true);

27     else

28         setcookie($name, $value, $expire, $cookie_path.'; HttpOnly', $cookie_domain, $cookie_secure);

29 }

//定义一些通用函数,

如:error():

       error('Unable to fetch user info', __FILE__, __LINE__, $db->error())

message($message, $no_back_link = false, $http_status = null)

utf8_trim( $str, $charlist=false);  //去空格

redirect($destination_url, $message);  //重定向

fluxbb多数页面都写在一个文件中,如login.php包括登录、退出、忘记密码等操作,通过不同的action区分。

引入公用header.php、不同页面分别替换相应的内容($pun_user变量),如title、要加载的css等。

$pun_user变量前面提到的check_cookie()函数中设置的值。 

1 $pun_user = array();

2 check_cookie($pun_user);

function check_cookie(&$pun_user){}    //注意后面的&地址符

check_cookie(函数中,通过cookie值来判断用户信息,如是合法的已登录用户,直接将一些值保存到$pun_user,否则设置一些默认值。通过set_default_user()函数,并记录在线用户。

 1 //

 2 // Fill $pun_user with default values (for guests)

 3 //

 4 function set_default_user()

 5 {

 6     global $db, $db_type, $pun_user, $pun_config;

 7 

 8     $remote_addr = get_remote_address();

 9 

10     // Fetch guest user

11     $result = $db->query('SELECT u.*, g.*, o.logged, o.last_post, o.last_search FROM '.$db->prefix.'users AS u INNER JOIN '.$db->prefix.'groups AS g ON u.group_id=g.g_id LEFT JOIN '.$db->prefix.'online AS o ON o.ident=\''.$db->escape($remote_addr).'\' WHERE u.id=1') or error('Unable to fetch guest information', __FILE__, __LINE__, $db->error());

12     if (!$db->num_rows($result))

13         exit('Unable to fetch guest information. Your database must contain both a guest user and a guest user group.');

14 

15     $pun_user = $db->fetch_assoc($result);

16 

17     // Update online list

18     if (!$pun_user['logged'])

19     {

20         $pun_user['logged'] = time();

21 

22         // With MySQL/MySQLi/SQLite, REPLACE INTO avoids a user having two rows in the online table

23         switch ($db_type)

24         {

25             case 'mysql':

26             case 'mysqli':

27             case 'mysql_innodb':

28             case 'mysqli_innodb':

29             case 'sqlite':

30                 $db->query('REPLACE INTO '.$db->prefix.'online (user_id, ident, logged) VALUES(1, \''.$db->escape($remote_addr).'\', '.$pun_user['logged'].')') or error('Unable to insert into online list', __FILE__, __LINE__, $db->error());

31                 break;

32 

33             default:

34                 $db->query('INSERT INTO '.$db->prefix.'online (user_id, ident, logged) SELECT 1, \''.$db->escape($remote_addr).'\', '.$pun_user['logged'].' WHERE NOT EXISTS (SELECT 1 FROM '.$db->prefix.'online WHERE ident=\''.$db->escape($remote_addr).'\')') or error('Unable to insert into online list', __FILE__, __LINE__, $db->error());

35                 break;

36         }

37     }

38     else

39         $db->query('UPDATE '.$db->prefix.'online SET logged='.time().' WHERE ident=\''.$db->escape($remote_addr).'\'') or error('Unable to update online list', __FILE__, __LINE__, $db->error());

40 

41     $pun_user['disp_topics'] = $pun_config['o_disp_topics_default'];

42     $pun_user['disp_posts'] = $pun_config['o_disp_posts_default'];

43     $pun_user['timezone'] = $pun_config['o_default_timezone'];

44     $pun_user['dst'] = $pun_config['o_default_dst'];

45     $pun_user['language'] = $pun_config['o_default_lang'];

46     $pun_user['style'] = $pun_config['o_default_style'];

47     $pun_user['is_guest'] = true;

48     $pun_user['is_admmod'] = false;

49 }

//footer.php

页面header和footer内容通过$tpl_main变量保存,footer.php最后,exit($tpl_main);直接输出并终止运行。主要用到了输出缓冲的一些函数。

在整个fluxbb系统中,只看到了两个js文件,分别是common.js和minmax.js,而且两个文件都没超过3kb,其中minmax.js是在header.php引入了。想想自己,没有jquery就有点寸步难行了。

 

原文作者:lltong,博客园地址:http://www.cnblogs.com/lltong/

你可能感兴趣的:(PHP)