亚洲视频二区_亚洲欧洲日本天天堂在线观看_日韩一区二区在线观看_中文字幕不卡一区

公告:魔扣目錄網(wǎng)為廣大站長(zhǎng)提供免費(fèi)收錄網(wǎng)站服務(wù),提交前請(qǐng)做好本站友鏈:【 網(wǎng)站目錄:http://www.430618.com 】, 免友鏈快審服務(wù)(50元/站),

點(diǎn)擊這里在線咨詢客服
新站提交
  • 網(wǎng)站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會(huì)員:747

1. SQL 注入

我賭一包辣條,你肯定會(huì)看到這里。 SQL 注入是對(duì)您網(wǎng)站最大的威脅之一,如果您的數(shù)據(jù)庫(kù)受到別人的 SQL 注入的攻擊的話,別人可以轉(zhuǎn)出你的數(shù)據(jù)庫(kù),也許還會(huì)產(chǎn)生更嚴(yán)重的后果。

網(wǎng)站要從數(shù)據(jù)庫(kù)中獲取動(dòng)態(tài)數(shù)據(jù),就必須執(zhí)行 SQL 語(yǔ)句,舉例如下:

<?php

$username = $_GET['username'];

$query = "SELECT * FROM users WHERE username = '$username'";

攻擊者控制通過(guò) GET 和 POST 發(fā)送的查詢(或者例如 UA 的一些其他查詢)。一般情況下,你希望查詢戶名為「 peter 」的用戶產(chǎn)生的 SQL 語(yǔ)句如下:

SELECT * FROM users WHERE username = 'peter'

但是,攻擊者發(fā)送了特定的用戶名參數(shù),例如:' OR '1'='1

這就會(huì)導(dǎo)致 SQL 語(yǔ)句變成這樣:

SELECT * FROM users WHERE username = 'peter' OR '1' = '1'

這樣,他就能在不需要密碼的情況下導(dǎo)出你的整個(gè)用戶表的數(shù)據(jù)了。

那么,我們?nèi)绾畏乐惯@類事故的發(fā)生呢?主流的解決方法有兩種。轉(zhuǎn)義用戶輸入的數(shù)據(jù)或者使用封裝好的語(yǔ)句。轉(zhuǎn)義的方法是封裝好一個(gè)函數(shù),用來(lái)對(duì)用戶提交的數(shù)據(jù)進(jìn)行過(guò)濾,去掉有害的標(biāo)簽。但是,我不太推薦使用這個(gè)方法,因?yàn)楸容^容易忘記在每個(gè)地方都做此處理。

下面,我來(lái)介紹如何使用 PDO 執(zhí)行封裝好的語(yǔ)句( mysqi 也一樣):

$username = $_GET['username'];

$query = $pdo->prepare('SELECT * FROM users WHERE username = :username');

$query->execute(['username' => $username]);

$data = $query->fetch();

動(dòng)態(tài)數(shù)據(jù)的每個(gè)部分都以:做前綴。然后將所有參數(shù)作為數(shù)組傳遞給執(zhí)行函數(shù),看起來(lái)就像 PDO 為你轉(zhuǎn)義了有害數(shù)據(jù)一樣。

幾乎所有的數(shù)據(jù)庫(kù)驅(qū)動(dòng)程序都支持封裝好的語(yǔ)句,沒(méi)有理由不使用它們!養(yǎng)成使用他們的習(xí)慣,以后就不會(huì)忘記了。

2. XSS

XSS 又叫 css (Cross Site Script) ,跨站腳本攻擊。它指的是惡意攻擊者往 Web 頁(yè)面里插入惡意 html 代碼,當(dāng)用戶瀏覽該頁(yè)之時(shí),嵌入其中 Web 里面的 html 代碼會(huì)被執(zhí)行,從而達(dá)到惡意攻擊用戶的特殊目的。

下面以一個(gè)搜索頁(yè)面為例子:

<body>

<?php

$searchQuery = $_GET['q'];

/* some search magic here */

?>

<h1>You searched for: <?php echo $searchQuery; ?></h1>

<p>We found: Absolutely nothing because this is a demo</p>

</body>

因?yàn)槲覀儼延脩舻膬?nèi)容直接打印出來(lái),不經(jīng)過(guò)任何過(guò)濾,非法用戶可以拼接 URL:

search.php?q=%3Cscript%3Ealert(1)%3B%3C%2Fscript%3E

PHP 渲染出來(lái)的內(nèi)容如下,可以看到 JAVAscript 代碼會(huì)被直接執(zhí)行:

<body>

<h1>You searched for: <script>alert(1);</script></h1>

<p>We found: Absolutely nothing because this is a demo</p>

</body>

問(wèn):JS 代碼被執(zhí)行有什么大不了的?

JavaScript 可以:

偷走你用戶瀏覽器里的 Cookie;

通過(guò)瀏覽器的記住密碼功能獲取到你的站點(diǎn)登錄賬號(hào)和密碼;

盜取用戶的機(jī)密信息;

你的用戶在站點(diǎn)上能做到的事情,有了 JS 權(quán)限執(zhí)行權(quán)限就都能做,也就是說(shuō) A 用戶可以模擬成為任何用戶;

在你的網(wǎng)頁(yè)中嵌入惡意代碼;

...

問(wèn):如何防范此問(wèn)題呢?

好消息是比較先進(jìn)的瀏覽器現(xiàn)在已經(jīng)具備了一些基礎(chǔ)的 XSS 防范功能,不過(guò)請(qǐng)不要依賴與此。

正確的做法是堅(jiān)決不要相信用戶的任何輸入,并過(guò)濾掉輸入中的所有特殊字符。這樣就能消滅絕大部分的 XSS 攻擊:

<?php

$searchQuery = htmlentities($searchQuery, ENT_QUOTES);

或者你可以使用模板引擎 Twig ,一般的模板引擎都會(huì)默認(rèn)為輸出加上 htmlentities 防范。

如果你保持了用戶的輸入內(nèi)容,在輸出時(shí)也要特別注意,在以下的例子中,我們?cè)试S用戶填寫自己的博客鏈接:

<body>

<a href="<?php echo $homepageUrl; ?>">Visit Users homepage</a>

</body>

以上代碼可能第一眼看不出來(lái)有問(wèn)題,但是假設(shè)用戶填入以下內(nèi)容:

#" onclick="alert(1)

會(huì)被渲染為:

<body>

<a href="#" onclick="alert(1)">Visit Users homepage</a>

</body>

永遠(yuǎn)永遠(yuǎn)不要相信用戶輸入的數(shù)據(jù),或者,永遠(yuǎn)都假設(shè)用戶的內(nèi)容是有攻擊性的,態(tài)度端正了,然后小心地處理好每一次的用戶輸入和輸出。

另外設(shè)置 Cookie 時(shí),如果無(wú)需 JS 讀取的話,請(qǐng)必須設(shè)置為 "HTTP ONLY"。這個(gè)設(shè)置可以令 JavaScript 無(wú)法讀取 PHP 端種的 Cookie。

3. XSRF/CSRF

CSRF 是跨站請(qǐng)求偽造的縮寫,它是攻擊者通過(guò)一些技術(shù)手段欺騙用戶去訪問(wèn)曾經(jīng)認(rèn)證過(guò)的網(wǎng)站并運(yùn)行一些操作。

雖然此處展示的例子是 GET 請(qǐng)求,但只是相較于 POST 更容易理解,并非防護(hù)手段,兩者都不是私密的 Cookies 或者多步表單。

假如你有一個(gè)允許用戶刪除賬戶的頁(yè)面,如下所示:

<?php

//delete-account.php

$confirm = $_GET['confirm'];

if($confirm === 'yes') {

//goodbye

}

攻擊者可以在他的站點(diǎn)上構(gòu)建一個(gè)觸發(fā)這個(gè) URL 的表單(同樣適用于 POST 的表單),或者將 URL 加載為圖片誘惑用戶點(diǎn)擊:

<img src="https://example.com/delete-account.php?confirm=yes" />

用戶一旦觸發(fā),就會(huì)執(zhí)行刪除賬戶的指令,眨眼你的賬戶就消失了。

防御這樣的攻擊比防御 XSS 與 SQL 注入更復(fù)雜一些。

最常用的防御方法是生成一個(gè) CSRF 令牌加密安全字符串,一般稱其為 Token,并將 Token 存儲(chǔ)于 Cookie 或者 Session 中。

每次你在網(wǎng)頁(yè)構(gòu)造表單時(shí),將 Token 令牌放在表單中的隱藏字段,表單請(qǐng)求服務(wù)器以后會(huì)根據(jù)用戶的 Cookie 或者 Session 里的 Token 令牌比對(duì),校驗(yàn)成功才給予通過(guò)。

由于攻擊者無(wú)法知道 Token 令牌的內(nèi)容(每個(gè)表單的 Token 令牌都是隨機(jī)的),因此無(wú)法冒充用戶。

<?php /* 你嵌入表單的頁(yè)面 */ ?>

<form action="/delete-account.php" method="post">

<input type="hidden" name="csrf" value="<?php echo $_SESSION['csrf']; ?>">

<input type="hidden" name="confirm" value="yes" />

<input type="submit" value="Delete my account" />

</form>

##

<?php

//delete-account.php

$confirm = $_POST['confirm'];

$csrf = $_POST['csrf'];

$knownGoodToken = $_SESSION['csrf'];

if($csrf !== $knownGoodToken) {

die('Invalid request');

}

if($confirm === 'yes') {

//goodbye

}

請(qǐng)注意,這是個(gè)非常簡(jiǎn)單的示例,你可以加入更多的代碼。如果你使用的是像 Symfony 這樣的 PHP 框架,那么自帶了 CSRF 令牌的功能。

4. LFI

LFI (本地文件包含) 是一個(gè)用戶未經(jīng)驗(yàn)證從磁盤讀取文件的漏洞。

我經(jīng)常遇到編程不規(guī)范的路由代碼示例,它們不驗(yàn)證過(guò)濾用戶的輸入。我們用以下文件為例,將它要渲染的模板文件用 GET 請(qǐng)求加載。

<body>

<?php

$page = $_GET['page'];

if(!$page) {

$page = 'main.php';

}

include($page);

?>

</body>

由于 Include 可以加載任何文件,不僅僅是 PHP,攻擊者可以將系統(tǒng)上的任何文件作為包含目標(biāo)傳遞。

index.php?page=../../etc/passwd

這將導(dǎo)致 /etc/passwd 文件被讀取并展示在瀏覽器上。

要防御此類攻擊,你必須仔細(xì)考慮允許用戶輸入的類型,并刪除可能有害的字符,如輸入字符中的 “.” “/” “”。

如果你真的想使用像這樣的路由系統(tǒng)(我不建議以任何方式),你可以自動(dòng)附加 PHP 擴(kuò)展,刪除任何非 [a-zA-Z0-9-_] 的字符,并指定從專用的模板文件夾中加載,以免被包含任何非模板文件。

我在不同的開(kāi)發(fā)文檔中,多次看到造成此類漏洞的 PHP 代碼。從一開(kāi)始就要有清晰的設(shè)計(jì)思路,允許所需要包含的文件類型,并刪除掉多余的內(nèi)容。你還可以構(gòu)造要讀取文件的絕對(duì)路徑,并驗(yàn)證文件是否存在來(lái)作為保護(hù),而不是任何位置都給予讀取。

5. 不充分的密碼哈希

大部分的 Web 應(yīng)用需要保存用戶的認(rèn)證信息。如果密碼哈希做的足夠好,在你的網(wǎng)站被攻破時(shí),即可保護(hù)用戶的密碼不被非法讀取。

首先,最不應(yīng)該做的事情,就是把用戶密碼明文儲(chǔ)存起來(lái)。大部分的用戶會(huì)在多個(gè)網(wǎng)站上使用同一個(gè)密碼,這是不可改變的事實(shí)。當(dāng)你的網(wǎng)站被攻破,意味著用戶的其他網(wǎng)站的賬號(hào)也被攻破了。

其次,你不應(yīng)該使用簡(jiǎn)單的哈希算法,事實(shí)上所有沒(méi)有專門為密碼哈希優(yōu)化的算法都不應(yīng)使用。哈希算法如 MD5 或者 SHA 設(shè)計(jì)初衷就是執(zhí)行起來(lái)非常快。這不是你需要的,密碼哈希的終極目標(biāo)就是讓黑客花費(fèi)無(wú)窮盡的時(shí)間和精力都無(wú)法破解出來(lái)密碼。

另外一個(gè)比較重要的點(diǎn)是你應(yīng)該為密碼哈希加鹽(Salt),加鹽處理避免了兩個(gè)同樣的密碼會(huì)產(chǎn)生同樣哈希的問(wèn)題。

以下使用 MD5 來(lái)做例子,所以請(qǐng)千萬(wàn)不要使用 MD5 來(lái)哈希你的密碼, MD5 是不安全的。

假如我們的用戶 user1 和 user315 都有相同的密碼 ilovecats123,這個(gè)密碼雖然看起來(lái)是強(qiáng)密碼,有字母有數(shù)字,但是在數(shù)據(jù)庫(kù)里,兩個(gè)用戶的密碼哈希數(shù)據(jù)將會(huì)是相同的:5e2b4d823db9d044ecd5e084b6d33ea5 。

如果一個(gè)如果黑客拿下了你的網(wǎng)站,獲取到了這些哈希數(shù)據(jù),他將不需要去暴力破解用戶 user315 的密碼。我們要盡量讓他花大精力來(lái)破解你的密碼,所以我們對(duì)數(shù)據(jù)進(jìn)行加鹽處理:

<?php

//warning: !!這是一個(gè)很不安全的密碼哈希例子,請(qǐng)不要使用!!

$password = 'cat123';

$salt = random_bytes(20);

$hash = md5($password . $salt);

最后在保存你的唯一密碼哈希數(shù)據(jù)時(shí),請(qǐng)不要忘記連 $salt 也已經(jīng)保存,否則你將無(wú)法驗(yàn)證用戶。

在當(dāng)下,最好的密碼哈希選項(xiàng)是 bcrypt,這是專門為哈希密碼而設(shè)計(jì)的哈希算法,同時(shí)這套哈希算法里還允許你配置一些參數(shù)來(lái)加大破解的難度。

新版的 PHP 中也自帶了安全的密碼哈希函數(shù) password_hash ,此函數(shù)已經(jīng)包含了加鹽處理。對(duì)應(yīng)的密碼驗(yàn)證函數(shù)為 password_verify 用來(lái)檢測(cè)密碼是否正確。password_verify 還可有效防止 時(shí)序攻擊.

以下是使用的例子:

<?php

//user signup

$password = $_POST['password'];

$hashedPassword = password_hash($password, PASSWORD_DEFAULT);

//login

$password = $_POST['password'];

$hash = '1234'; //load this value from your db

if(password_verify($password, $hash)) {

echo 'Password is valid!';

else {

echo 'Invalid password.';

}

需要澄清的一點(diǎn)是:密碼哈希并不是密碼加密。哈希(Hash)是將目標(biāo)文本轉(zhuǎn)換成具有相同長(zhǎng)度的、不可逆的雜湊字符串(或叫做消息摘要),而加密(Encrypt)是將目標(biāo)文本轉(zhuǎn)換成具有不同長(zhǎng)度的、可逆的密文。顯然他們之間最大的區(qū)別是可逆性,在儲(chǔ)存密碼時(shí),我們要的就是哈希這種不可逆的屬性。

分享到:
標(biāo)簽:php
用戶無(wú)頭像

網(wǎng)友整理

注冊(cè)時(shí)間:

網(wǎng)站:5 個(gè)   小程序:0 個(gè)  文章:12 篇

  • 51998

    網(wǎng)站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會(huì)員

趕快注冊(cè)賬號(hào),推廣您的網(wǎng)站吧!
最新入駐小程序

數(shù)獨(dú)大挑戰(zhàn)2018-06-03

數(shù)獨(dú)一種數(shù)學(xué)游戲,玩家需要根據(jù)9

答題星2018-06-03

您可以通過(guò)答題星輕松地創(chuàng)建試卷

全階人生考試2018-06-03

各種考試題,題庫(kù),初中,高中,大學(xué)四六

運(yùn)動(dòng)步數(shù)有氧達(dá)人2018-06-03

記錄運(yùn)動(dòng)步數(shù),積累氧氣值。還可偷

每日養(yǎng)生app2018-06-03

每日養(yǎng)生,天天健康

體育訓(xùn)練成績(jī)?cè)u(píng)定2018-06-03

通用課目體育訓(xùn)練成績(jī)?cè)u(píng)定