PHP數(shù)據(jù)過(guò)濾的方法
在指南的開(kāi)始,我們說(shuō)過(guò)數(shù)據(jù)過(guò)濾在任何語(yǔ)言、任何平臺(tái)上都是WEB應(yīng)用安全的基石。這包含檢驗(yàn)輸入到應(yīng)用的數(shù)據(jù)以及從應(yīng)用輸出的數(shù)據(jù),而一個(gè)好的軟件設(shè)計(jì)可以幫助開(kāi)發(fā)人員做到:
確保數(shù)據(jù)過(guò)濾無(wú)法被繞過(guò),
確保不合法的信息不會(huì)影響合法的信息,并且
識(shí)別數(shù)據(jù)的來(lái)源。
關(guān)于如何確保數(shù)據(jù)過(guò)濾無(wú)法被繞過(guò)有各種各樣的觀點(diǎn),而其中的兩種觀點(diǎn)比其他更加通用并可提供更高級(jí)別的保障。
調(diào)度方法
這種方法是用一個(gè)單一的 php 腳本調(diào)度(通過(guò) URL)。其他任何操作在必要的時(shí)候使用include或require包含進(jìn)來(lái)。這種方法一般需要每個(gè) URL 都傳遞一個(gè)單獨(dú)的GET變量用于調(diào)度。這個(gè)GET變量可以被認(rèn)為是用來(lái)替代腳本名稱的更加簡(jiǎn)化的設(shè)計(jì)。例如:
http://a.org/dispatch.php?task=PRint_formdispatch.php是唯一的根文件(Document root)。它可以讓開(kāi)發(fā)者做兩件非常重要的事情:
在dispatch.php最開(kāi)始實(shí)現(xiàn)一些全局的安全處理,并且確保這些處理不可以被繞過(guò)。
容易確定在必要的地方進(jìn)行數(shù)據(jù)過(guò)濾,特別是一些特殊目的的控制流操作中。
看下面的例子以便進(jìn)一步討論dispatch.php腳本:
<?php/* 全局安全處理 */switch ($_GET['task']){case 'print_form':include '/inc/presentation/form.inc';break;case 'process_form':$form_valid = false;include '/inc/logic/process.inc';if ($form_valid){include '/inc/presentation/end.inc';}else{include '/inc/presentation/form.inc';}break;default:include '/inc/presentation/index.inc';break;}?>如果這是唯一的可公開(kāi)訪問(wèn)到的 PHP 腳本,則可以確信的一點(diǎn)是這個(gè)程序的設(shè)計(jì)可以確保在最開(kāi)始的全局安全處理無(wú)法被繞過(guò)。同時(shí)也讓開(kāi)發(fā)者容易看到特定任務(wù)的控制流程。例如,不需要瀏覽整個(gè)代碼就可以容易的知道:當(dāng)$form_valid為true時(shí),end.inc是唯一顯示給用戶的;由于它在process.inc被包含之前,并剛剛初始化為false,可以確定的是process.inc的內(nèi)部邏輯會(huì)將設(shè)置它為true;否則表單將再次顯示(可能會(huì)顯示相關(guān)的錯(cuò)誤信息)。
注意
如果你使用目錄定向文件,如index.php(代替dispatch.php),你可以像這樣使用 URL 地址:http://a.org/?task=print_form。
你還可以使用 ApacheForceType重定向或者mod_rewrite來(lái)調(diào)整 URL 地址:http://a.org/app/print-form。
包含方法
另外一種方式是使用單獨(dú)一個(gè)模塊,這個(gè)模塊負(fù)責(zé)所有的安全處理。這個(gè)模塊被包含在所有公開(kāi)的 PHP 腳本的最前端(或者非??壳暗牟糠郑?。參考下面的腳本security.inc
<?phpswitch ($_POST['form']){case 'login':$allowed = array();$allowed[] = 'form';$allowed[] = 'username';$allowed[] = 'passWord';$sent = array_keys($_POST);if ($allowed == $sent){include '/inc/logic/process.inc';}break;}?>
在本例中,每個(gè)提交過(guò)來(lái)的表單都認(rèn)為應(yīng)當(dāng)含有form這個(gè)唯一驗(yàn)證值,并且security.inc獨(dú)立處理表單中0需要過(guò)濾的數(shù)據(jù)。實(shí)現(xiàn)這個(gè)要求的 HTML 表單如下所示:
<form action="/receive.php" method="POST"><input type="hidden" name="form" value="login" /><p>Username:<input type="text" name="username" /></p><p>Password:<input type="password" name="password" /></p><input type="submit" /></form>
叫做$allowed的數(shù)組用來(lái)檢驗(yàn)?zāi)膫€(gè)表單變量是允許的, 這個(gè)列表在表單被處理前應(yīng)當(dāng)是一致的。流程控制決定要執(zhí)行什么,而process.inc是真正過(guò)濾后的數(shù)據(jù)到達(dá)的地方。
注意
確保security.inc總是被包含在每個(gè)腳本的最開(kāi)始的位置比較好的方法是使用auto_prepend_file設(shè)置。
過(guò)濾的例子
建立白名單對(duì)于數(shù)據(jù)過(guò)濾是非常重要的。由于不可能對(duì)每一種可能遇到的表單數(shù)據(jù)都給出例子,部分例子可以幫助你對(duì)此有一個(gè)大體的了解。
下面的代碼對(duì)郵件地址進(jìn)行了驗(yàn)證:
<?php$clean = array();$email_pattern = '/^[^@\s<&>]+@([-a-z0-9]+\.)+[a-z]{2,}$/i';if (preg_match($email_pattern, $_POST['email'])){$clean['email'] = $_POST['email'];}?>
下面的代碼確保了$_POST['color']的內(nèi)容是red,green,或者blue:
[/co<?php$clean = array();switch ($_POST['color']){case 'red':case 'green':case 'blue':$clean['color'] = $_POST['color'];break;}?>de]
下面的代碼確保$_POST['num']是一個(gè)整數(shù)(integer):
[code]
<?php$clean = array();if ($_POST['num'] == strval(intval($_POST['num']))){$clean['num'] = $_POST['num'];}?>
下面的代碼確保$_POST['num']是一個(gè)浮點(diǎn)數(shù)(float):
<?php$clean = array();if ($_POST['num'] == strval(floatval($_POST['num']))){$clean['num'] = $_POST['num'];}?>
名字轉(zhuǎn)換
之前每個(gè)例子都使用了數(shù)組$clean。對(duì)于開(kāi)發(fā)人員判斷數(shù)據(jù)是否有潛在的威脅這是一個(gè)很好的習(xí)慣。 永遠(yuǎn)不要在對(duì)數(shù)據(jù)驗(yàn)證后還將其保存在$_POST或者$_GET中,作為開(kāi)發(fā)人員對(duì)超級(jí)全局?jǐn)?shù)組中保存的數(shù)據(jù)總是應(yīng)當(dāng)保持充分的懷疑。
需要補(bǔ)充的是,使用$clean可以幫助思考還有什么沒(méi)有被過(guò)濾,這更類似一個(gè)白名單的作用??梢蕴嵘踩牡燃?jí)。
如果僅僅將驗(yàn)證過(guò)的數(shù)據(jù)保存在$clean,在數(shù)據(jù)驗(yàn)證上僅存的風(fēng)險(xiǎn)是你所引用的數(shù)組元素不存在,而不是未過(guò)濾的危險(xiǎn)數(shù)據(jù)。
時(shí)機(jī)
一旦 PHP 腳本開(kāi)始執(zhí)行,則意味著 HTTP 請(qǐng)求已經(jīng)全部結(jié)束。此時(shí),用戶便沒(méi)有機(jī)會(huì)向腳本發(fā)送數(shù)據(jù)。因此,沒(méi)有數(shù)據(jù)可以被輸入到腳本中(甚至register_globals被開(kāi)啟的情況下)。這就是為什么初始化變量是非常好的習(xí)慣。
- php中使用Curl、socket、file_get_contents三種方法POST提交數(shù)據(jù)
- php采用ajax數(shù)據(jù)提交post與post常見(jiàn)方法總結(jié)
- php curl模擬post提交數(shù)據(jù)示例
- 解析PHP 使用curl提交json格式數(shù)據(jù)
- PHP防止post重復(fù)提交數(shù)據(jù)的簡(jiǎn)單例子
- PHP中如何判斷AJAX提交的數(shù)據(jù)
- php jq jquery getJSON跨域提交數(shù)據(jù)完整版
- php防止偽造的數(shù)據(jù)從URL提交方法
- php判斷文件上傳類型及過(guò)濾不安全數(shù)據(jù)的方法
- php過(guò)濾所有惡意字符(批量過(guò)濾post,get敏感數(shù)據(jù))
- php 字符過(guò)濾類,用于過(guò)濾各類用戶輸入的數(shù)據(jù)
- PHP數(shù)據(jù)的提交與過(guò)濾基本操作實(shí)例詳解
相關(guān)文章
php關(guān)閉warning問(wèn)題的解決方法
下面小編就為大家?guī)?lái)一篇php關(guān)閉warning問(wèn)題的解決方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-05-05
PHP中創(chuàng)建空文件的代碼[file_put_contents vs touch]
php中用file_put_contents比touch快,大約兩倍左右。2012-01-01
關(guān)于PHP結(jié)束標(biāo)簽的使用細(xì)節(jié)探討及聯(lián)想
PHP解析文件時(shí)會(huì)尋找開(kāi)始?php和結(jié)束標(biāo)記?,標(biāo)記告訴PHP開(kāi)始和停止解釋其中的代碼,接下來(lái)將詳細(xì)介紹下PHP結(jié)束標(biāo)簽的使用細(xì)節(jié)感興趣的你可以參考下本文或許可以幫助到你2013-03-03
PHP實(shí)現(xiàn)對(duì)站點(diǎn)內(nèi)容外部鏈接的過(guò)濾方法
這篇文章主要介紹了PHP實(shí)現(xiàn)對(duì)站點(diǎn)內(nèi)容外部鏈接的過(guò)濾方法,可實(shí)現(xiàn)針對(duì)外部鏈接增加rel="nofollow"的功能,有助于網(wǎng)站SEO建設(shè)以及避免不必要的損失,是非常實(shí)用的技巧,需要的朋友可以參考下2014-09-09
PHP編程計(jì)算文件或數(shù)組中單詞出現(xiàn)頻率的方法
這篇文章主要介紹了PHP編程計(jì)算文件或數(shù)組中單詞出現(xiàn)頻率的方法,給出了2個(gè)統(tǒng)計(jì)單詞頻率的示例,涉及php正則、數(shù)組操作及字符串遍歷等相關(guān)技巧,需要的朋友可以參考下2017-05-05
詳解WordPress中創(chuàng)建和添加過(guò)濾器的相關(guān)PHP函數(shù)
這篇文章主要介紹了WordPress中創(chuàng)建和添加過(guò)濾器的相關(guān)PHP函數(shù),分別是apply_filters()函數(shù)和add_filter()函數(shù)的使用,需要的朋友可以參考下2015-12-12

