ThinkPHP令牌驗(yàn)證實(shí)例
ThinkPHP內(nèi)置了表單令牌驗(yàn)證功能,可以有效防止表單的遠(yuǎn)程提交等安全防護(hù)。
表單令牌驗(yàn)證相關(guān)的配置參數(shù)有:
'TOKEN_ON'=>true, // 是否開啟令牌驗(yàn)證 'TOKEN_NAME'=>'__hash__', // 令牌驗(yàn)證的表單隱藏字段名稱 'TOKEN_TYPE'=>'md5', //令牌哈希驗(yàn)證規(guī)則 默認(rèn)為MD5
如果開啟表單令牌驗(yàn)證功能,系統(tǒng)會(huì)自動(dòng)在帶有表單的模板文件里面自動(dòng)生成以TOKEN_NAME為名稱的隱藏域,其值則是TOKEN_TYPE方式生成的哈希字符串,用于實(shí)現(xiàn)表單的自動(dòng)令牌驗(yàn)證。
自動(dòng)生成的隱藏域位于表單Form結(jié)束標(biāo)志之前,如果希望自己控制隱藏域的位置,可以手動(dòng)在表單頁面添加__TOKEN__ 標(biāo)識(shí),系統(tǒng)會(huì)在輸出模板的時(shí)候自動(dòng)替換。如果在開啟表單令牌驗(yàn)證的情況下,個(gè)別表單不需要使用令牌驗(yàn)證功能,可以在表單頁面添加__NOTOKEN__,則系統(tǒng)會(huì)忽略當(dāng)前表單的令牌驗(yàn)證。
如果頁面中存在多個(gè)表單,建議添加__TOKEN__標(biāo)識(shí),并確保只有一個(gè)表單需要令牌驗(yàn)證。
模型類在創(chuàng)建數(shù)據(jù)對(duì)象的同時(shí)會(huì)自動(dòng)進(jìn)行表單令牌驗(yàn)證操作,如果你沒有使用create方法創(chuàng)建數(shù)據(jù)對(duì)象的話,則需要手動(dòng)調(diào)用模型的autoCheckToken方法進(jìn)行表單令牌驗(yàn)證。如果返回false,則表示表單令牌驗(yàn)證錯(cuò)誤。例如:
$User = M("User"); // 實(shí)例化User對(duì)象
// 手動(dòng)進(jìn)行令牌驗(yàn)證
if (!$User->autoCheckToken($_POST)){
// 令牌驗(yàn)證錯(cuò)誤
}
在ThinkPHP框架的View.class.php里定義了一個(gè)公共的模板替換函數(shù)
protected function templateContentReplace($content) {
// 系統(tǒng)默認(rèn)的特殊變量替換
$replace = array(
'../Public' => APP_PUBLIC_PATH,// 項(xiàng)目公共目錄
'__PUBLIC__' => WEB_PUBLIC_PATH,// 站點(diǎn)公共目錄
'__TMPL__' => APP_TMPL_PATH, // 項(xiàng)目模板目錄
'__ROOT__' => __ROOT__, // 當(dāng)前網(wǎng)站地址
'__APP__' => __APP__, // 當(dāng)前項(xià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)前項(xiàng)目地址
}
if(C('TOKEN_ON')) {
if(strpos($content,'{__TOKEN__}')) {
// 指定表單令牌隱藏域位置
$replace['{__TOKEN__}'] = $this->buildFormToken();
}elseif(strpos($content,'{__NOTOKEN__}')){
// 標(biāo)記為不需要令牌驗(yàn)證
$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'))是對(duì)令牌驗(yàn)證的開啟狀態(tài)進(jìn)行判斷,若開啟則調(diào)用buildFormToken()方法,$_SESSION[$tokenName] = $tokenValue; 其實(shí)就是給$_SESSION['__hash__']賦值。如果不想進(jìn)行令牌驗(yàn)證,只要在頁面的</form>之前加入{__NOTOKEN__}就行了,它會(huì)被函數(shù)替換成空。
在ThinkPHP的Model.class.php類里定義了令牌的驗(yàn)證函數(shù)
// 表單令牌驗(yàn)證
if(C('TOKEN_ON') && !$this->autoCheckToken($data)) {
$this->error = L('_TOKEN_ERROR_');
return false;
}
// 自動(dòng)表單令牌驗(yàn)證
public function autoCheckToken($data) {
$name = C('TOKEN_NAME');
if(isset($_SESSION[$name])) {
// 當(dāng)前需要令牌驗(yàn)證
if(empty($data[$name]) || $_SESSION[$name] != $data[$name]) {
// 非法提交
return false;
}
// 驗(yàn)證完成銷毀session
unset($_SESSION[$name]);
}
return true;
}
- 基于thinkPHP3.2實(shí)現(xiàn)微信接入及查詢token值的方法
- Thinkphp5 微信公眾號(hào)token驗(yàn)證不成功的原因及解決方法
- thinkphp5框架API token身份驗(yàn)證功能示例
- ThinkPHP5.1表單令牌Token失效問題的解決
- 完美解決thinkphp驗(yàn)證碼出錯(cuò)無法顯示的方法
- ThinkPHP驗(yàn)證碼使用簡明教程
- thinkPHP實(shí)現(xiàn)表單自動(dòng)驗(yàn)證
- ThinkPHP5 驗(yàn)證器的具體使用
- ThinkPHP表單自動(dòng)驗(yàn)證實(shí)例
- 詳解ThinkPHP3.2.3驗(yàn)證碼顯示、刷新、校驗(yàn)
- thinkphp框架使用JWTtoken的方法詳解
相關(guān)文章
php獲取指定日期之間的各個(gè)周和月的起止時(shí)間
本文推薦給大家一個(gè)php類文件,可以根據(jù)指定日期獲取所在周的起始時(shí)間和結(jié)束時(shí)間,并附上使用實(shí)例,有需要的小伙伴參考下吧2014-11-11
淺談PHP中靜態(tài)方法和非靜態(tài)方法的相互調(diào)用
下面小編就為大家?guī)硪黄獪\談PHP中靜態(tài)方法和非靜態(tài)方法的相互調(diào)用。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-10-10
CI框架實(shí)現(xiàn)優(yōu)化文件上傳及多文件上傳的方法
這篇文章主要介紹了CI框架實(shí)現(xiàn)優(yōu)化文件上傳及多文件上傳的方法,結(jié)合實(shí)例形式詳細(xì)分析了CI框架優(yōu)化文件上傳及多文件上傳的實(shí)現(xiàn)思路與具體操作步驟,需要的朋友可以參考下2017-01-01
php使用 readfile() 函數(shù)設(shè)置文件大小大小的方法
本文通過實(shí)例代碼給大家講解了php使用 readfile() 函數(shù)設(shè)置文件大小大小的方法,需要的的朋友參考下吧2017-08-08
destoon切換城市后實(shí)現(xiàn)logo旁邊顯示地區(qū)名稱的方法
這篇文章主要介紹了destoon切換城市后實(shí)現(xiàn)logo旁邊顯示地區(qū)名稱的方法,針對(duì)不同城市建設(shè)分站的時(shí)候很適用,需要的朋友可以參考下2014-08-08

