ThinkPHP令牌驗證實例
ThinkPHP內(nèi)置了表單令牌驗證功能,可以有效防止表單的遠程提交等安全防護。
表單令牌驗證相關(guān)的配置參數(shù)有:
'TOKEN_ON'=>true, // 是否開啟令牌驗證 'TOKEN_NAME'=>'__hash__', // 令牌驗證的表單隱藏字段名稱 'TOKEN_TYPE'=>'md5', //令牌哈希驗證規(guī)則 默認為MD5
如果開啟表單令牌驗證功能,系統(tǒng)會自動在帶有表單的模板文件里面自動生成以TOKEN_NAME為名稱的隱藏域,其值則是TOKEN_TYPE方式生成的哈希字符串,用于實現(xiàn)表單的自動令牌驗證。
自動生成的隱藏域位于表單Form結(jié)束標(biāo)志之前,如果希望自己控制隱藏域的位置,可以手動在表單頁面添加__TOKEN__ 標(biāo)識,系統(tǒng)會在輸出模板的時候自動替換。如果在開啟表單令牌驗證的情況下,個別表單不需要使用令牌驗證功能,可以在表單頁面添加__NOTOKEN__,則系統(tǒng)會忽略當(dāng)前表單的令牌驗證。
如果頁面中存在多個表單,建議添加__TOKEN__標(biāo)識,并確保只有一個表單需要令牌驗證。
模型類在創(chuàng)建數(shù)據(jù)對象的同時會自動進行表單令牌驗證操作,如果你沒有使用create方法創(chuàng)建數(shù)據(jù)對象的話,則需要手動調(diào)用模型的autoCheckToken方法進行表單令牌驗證。如果返回false,則表示表單令牌驗證錯誤。例如:
$User = M("User"); // 實例化User對象 // 手動進行令牌驗證 if (!$User->autoCheckToken($_POST)){ // 令牌驗證錯誤 }
在ThinkPHP框架的View.class.php里定義了一個公共的模板替換函數(shù)
protected function templateContentReplace($content) { // 系統(tǒng)默認的特殊變量替換 $replace = array( '../Public' => APP_PUBLIC_PATH,// 項目公共目錄 '__PUBLIC__' => WEB_PUBLIC_PATH,// 站點公共目錄 '__TMPL__' => APP_TMPL_PATH, // 項目模板目錄 '__ROOT__' => __ROOT__, // 當(dāng)前網(wǎng)站地址 '__APP__' => __APP__, // 當(dāng)前項目地址 '__UPLOAD__' => __ROOT__.'/Uploads', '__ACTION__' => __ACTION__, // 當(dāng)前操作地址 '__SELF__' => __SELF__, // 當(dāng)前頁面地址 '__URL__' => __URL__, '__INFO__' => __INFO__, ); if(defined('GROUP_NAME')) { $replace['__GROUP__'] = __GROUP__;// 當(dāng)前項目地址 } if(C('TOKEN_ON')) { if(strpos($content,'{__TOKEN__}')) { // 指定表單令牌隱藏域位置 $replace['{__TOKEN__}'] = $this->buildFormToken(); }elseif(strpos($content,'{__NOTOKEN__}')){ // 標(biāo)記為不需要令牌驗證 $replace['{__NOTOKEN__}'] = ''; }elseif(preg_match('/<\/form(\s*)>/is',$content,$match)) { // 智能生成表單令牌隱藏域 $replace[$match[0]] = $this->buildFormToken().$match[0]; } } // 允許用戶自定義模板的字符串替換 if(is_array(C('TMPL_PARSE_STRING')) ) $replace = array_merge($replace,C('TMPL_PARSE_STRING')); $content = str_replace(array_keys($replace),array_values($replace),$content); return $content; }
上面的if(C('TOKEN_ON'))是對令牌驗證的開啟狀態(tài)進行判斷,若開啟則調(diào)用buildFormToken()方法,$_SESSION[$tokenName] = $tokenValue; 其實就是給$_SESSION['__hash__']賦值。如果不想進行令牌驗證,只要在頁面的</form>之前加入{__NOTOKEN__}就行了,它會被函數(shù)替換成空。
在ThinkPHP的Model.class.php類里定義了令牌的驗證函數(shù)
// 表單令牌驗證 if(C('TOKEN_ON') && !$this->autoCheckToken($data)) { $this->error = L('_TOKEN_ERROR_'); return false; } // 自動表單令牌驗證 public function autoCheckToken($data) { $name = C('TOKEN_NAME'); if(isset($_SESSION[$name])) { // 當(dāng)前需要令牌驗證 if(empty($data[$name]) || $_SESSION[$name] != $data[$name]) { // 非法提交 return false; } // 驗證完成銷毀session unset($_SESSION[$name]); } return true; }
相關(guān)文章
淺談PHP中靜態(tài)方法和非靜態(tài)方法的相互調(diào)用
下面小編就為大家?guī)硪黄獪\談PHP中靜態(tài)方法和非靜態(tài)方法的相互調(diào)用。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-10-10CI框架實現(xiàn)優(yōu)化文件上傳及多文件上傳的方法
這篇文章主要介紹了CI框架實現(xiàn)優(yōu)化文件上傳及多文件上傳的方法,結(jié)合實例形式詳細分析了CI框架優(yōu)化文件上傳及多文件上傳的實現(xiàn)思路與具體操作步驟,需要的朋友可以參考下2017-01-01php使用 readfile() 函數(shù)設(shè)置文件大小大小的方法
本文通過實例代碼給大家講解了php使用 readfile() 函數(shù)設(shè)置文件大小大小的方法,需要的的朋友參考下吧2017-08-08destoon切換城市后實現(xiàn)logo旁邊顯示地區(qū)名稱的方法
這篇文章主要介紹了destoon切換城市后實現(xiàn)logo旁邊顯示地區(qū)名稱的方法,針對不同城市建設(shè)分站的時候很適用,需要的朋友可以參考下2014-08-08