先看一段代碼:php
<?php $a='aaa'; $aaa='xxx'; echo $$a; //$$a=$($a)=$(aaa)='xxx' ?>
最後回顯的是xxx,講實話,我第一次明白了這個原理後,不由吐槽,啥?這就是世界上最好的語言???這種低級錯誤還能有?html
咱們再回到漏洞自己:數據庫
<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/> <?php $a='aa'; $aa='123'; echo '第一次變量aa的值是'.$aa.'<br>'; $$a=$_GET['name']; echo '第二次變量aa的值是'.$aa.'<br>'; //$$a=$($a)=$aa=$_GET ?>
發現:代碼審計,掃描工具掃不出數組
變量覆蓋漏洞大多數由函數使用不當致使,常常引起變量覆蓋漏洞的函數有:extract(),parse_str()和import_request_variables()安全
織夢dedecms的2015年一次變量覆蓋:函數
/login.php?dopost=login&validate=dcug&useripost
d=帳號&pwd=密碼& _POST[GLOBALS]
[cfg_dbhost]=MYSQL外鏈IP&_POST[GLOBALS] [cfg_dbuser]=MYSQL的帳號網站
&_POST[GLOBALS][cfg_dbpwd]=MYSQL的密碼& ui
_POST[GLOBALS] [cfg_dbname]=本身的dedecms的數據庫
變量覆蓋致使重寫爲任意數據庫配置
這塊內容有點難懂,轉載下來時常翻:
<?php echo "Register_globals: ".(int)ini_get("register_globals")."<br/>"; if ($auth){ echo "private!"; } ?>
當register_globals=OFF時,這段代碼不會出問題。
private!
小記:若是上面的代碼中,已經對變量$auth賦了初始值,好比$auth=0,那麼即便在URL中有/test.php?auth=1,也不會將變量覆蓋,也就是說不會打印出private!
<?php echo "Register_globals:".(int)ini_get("register_globals")."<br/>"; if (ini_get('register_globals')) foreach($_REQUEST as $k=>$v) unset(${$k}); print $a; print $_GET[b]; ?>
<?php $auth = '0'; extract($_GET); if($auth==1){ echo "private!"; }else{ echo "public!"; } ?>
安全的作法是肯定register_globals=OFF後,在調用extract()時使用EXTR_SKIP保證已有變量不會被覆蓋。
小記:PHP extract() 函數從數組中把變量導入到當前的符號表中。對於數組中的每一個元素,鍵名用於變量名,鍵值用於變量值。
3、遍歷初始化變量常見的一些以遍歷的方式釋放變量的代碼,可能會致使變量覆蓋。
<? $chs = ''; if($_POST && $charset != 'utf-8'){ $chs = new Chinese('UTF-8', $charset); foreach($_POST as $key => $value){ $$key = $chs->Convert($value); } unset($chs); } ?>
小記:在代碼審計時須要注意相似「$$k」的變量賦值方式有可能覆蓋已有的變量,從而致使一些不可控制的結果。
4、import_request_variables變量覆蓋<?php $auth = '0'; import_request_variables('G'); if($auth == 1){ echo "private!"; }else{ echo "public!"; } ?>
import_request_variables('G')指定導入GET請求中的變量,從而致使變量覆蓋。
小記:import_request_variables — 將 GET/POST/Cookie 變量導入到全局做用域中。若是你禁止了 register_globals,但又想用到一些全局變量,那麼此函數就頗有用。
5、parse_str()變量覆蓋//var.php?var=new $var='init'; parse_str($_SERVER['QUERY_STRING']); print $var;
與parse_str()相似的函數還有mb_parse_str()
小記:parse_str — 將字符串解析成多個變量,若是參數str是URL傳遞入的查詢字符串(query string),則將它解析爲變量並設置到當前做用域。
修復,用到這些變量的時候,本身檢測可能性,尤爲是當foreach涉及鍵名鍵值時