代碼審計Day1 – in_array函數缺陷

前言

大家好,我們是紅日安全-代碼審計小組。最近我們小組正在做一個PHP代碼審計的項目,供大家學習交流,我們給這個項目起了一個名字叫 PHP-Audit-Labs 。現在大家所看到的系列文章,屬於項目 第一階段 的內容,本階段的內容題目均來自 PHP SECURITY CALENDAR 2017 。對於每一道題目,我們均給出對應的分析,並結合實際CMS進行解說。在文章的最後,我們還會留一道CTF題目,供大家練習,希望大家喜歡。下面是 第1篇代碼審計文章:Day 1 - Wish List

題目叫做願望清單,代碼如下:

1.png

 

漏洞解析 :

這一關卡考察的是一個任意文件上傳漏洞,而導致這一漏洞的發生則是不安全的使用 in_array() 函數來檢測上傳的文件名,即上圖中的第12行部分。由於該函數並未將第三個參數設置爲 true ,這導致攻擊者可以通過構造的文件名來繞過服務端的檢測,例如文件名爲 7shell.php 。因爲PHP在使用 in_array() 函數判斷時,會將 7shell.php 強制轉換成數字7,而數字7在 range(1,24) 數組中,最終繞過 in_array() 函數判斷,導致任意文件上傳漏洞。(這裏之所以會發生強制類型轉換,是因爲目標數組中的元素爲數字類型)我們來看看PHP手冊對 in_array() 函數的定義。

in_array :(PHP 4, PHP 5, PHP 7)

功能 :檢查數組中是否存在某個值

定義 : 

bool in_array ( mixed $needle , array $haystack [, bool $strict = FALSE ] )

在$haystack中搜索 $needle ,如果第三個參數 $strict 的值爲 TRUE ,則 in_array() 函數會進行強檢 查,檢查 $needle 的類型是否和 $haystack 中的相同。如果找到 $haystack,則返回 TRUE,否則返回 FALSE。

 

實例分析

本次實例分析,我們選取的是 piwigo2.7.1 版本。該版本由於SQL語句直接拼接 $rate 變量,而 $rate 變量也僅是用 in_array() 函數簡單處理,並未使用第三個參數進行嚴格匹配,最終導致sql注入漏洞發生。下面我們來看看具體的漏洞位置。漏洞的入口文件在 include\functions_rate.inc.php 中,具體代碼如下:

2.png

 

當 $_GET['action'] 爲 rate 的時候,就會調用文件 include/functions_rate.inc.php 中的 rate_picture 方法,而漏洞便存在這個方法中。我們可以看到下圖第23行處直接拼接 $rate 變量,而在第2行使用 in_array()函數對 $rate 變量進行檢測,判斷 $rate 是否在 $conf['rate_items'] 中, $conf['rate_items'] 的內容可以在 include\config_default.inc.php 中找到,爲

$conf['rate_items'] = array(0,1,2,3,4,5);

 

3.png

 

由於這裏(上圖第6行)並沒有將in_array()函數的第三個參數設置爲true,所以會進行弱比較,可以繞過。比如我們將 $rate 的值設置成 1,1 and if(ascii(substr((select database()),1,1))=112,1,sleep(3)));# 那麼SQL語句就變成:

INSERT INTO piwigo_rate (user_id,anonymous_id,element_id,rate,date) VALUES (2,'192.168.2',1,1,1 and if(ascii(substr((select database()),1,1))=112,1,sleep(3)));#,NOW()) ;

4.png

 

 

漏洞利用

接下來我們直接用sqlmap進行驗證,payload 如下:

sqlmap -u "http://192.168.2.211/piwigo/picture.php?/1/category/1&action=rate" --data "rate=1" --dbs --batch

 

5.png

 

修復建議

可以看到這個漏洞的原因是弱類型比較問題,那麼我們就可以使用強匹配進行修復。例如將 in_array() 函數的第三個參數設置爲 true ,或者使用 intval() 函數將變量強轉成數字,又或者使用正則匹配來處理變量。這裏我將 in_array() 函數的第三個參數設置爲 true ,代碼及防護效果如下:

6.png

7.png

 

結語

看完了上述分析,不知道大家是否對 in_array() 函數有了更加深入的理解,文中用到的CMS可以從 這裏 下載,當然文中若有不當之處,還望各位斧正。如果你對我們的項目感興趣,歡迎發送郵件到 [email protected] 聯繫我們。Day1 的分析文章就到這裏,我們最後留了一道CTF題目給大家練手,題目如下:

//index.php
<?php
include 'config.php';
$conn = new mysqli($servername, $username, $password, $dbname);
if ($conn->connect_error) {
    die("連接失敗: ");
}
$sql = "SELECT COUNT(*) FROM users";
$whitelist = array();
$result = $conn->query($sql);
if($result->num_rows > 0){
    $row = $result->fetch_assoc();
    $whitelist = range(1, $row['COUNT(*)']);
}
$id = stop_hack($_GET['id']);
 $sql = "SELECT * FROM users WHERE id=$id";
if (!in_array($id, $whitelist)) {
   die("id $id is not in whitelist.");
}
    
$result = $conn->query($sql);
if($result->num_rows > 0){
    $row = $result->fetch_assoc();
    echo "<center><table border='1'>";
    foreach ($row as $key => $value) {
        echo "<tr><td><center>$key</center></td><br>";
        echo "<td><center>$value</center></td></tr><br>";
    }
    echo "</table></center>";
}
    else{
    die($conn->error);
}
    ?>
//config.php<?php
$servername = "localhost";
$username = "fire";
$password = "fire";
$dbname = "day1";
function stop_hack($value){
    $pattern = 
"insert|delete|or|concat|concat_ws|group_concat|join|floor|\/\*|\*|\.\.\/|\.\/|union|into|load_file|outfile|dumpfile|sub|hex|file_put_contents|fwrite|curl|system|eval";
    $back_list = explode("|",$pattern);
    foreach($back_list as $hack){
        if(preg_match("/$hack/i", $value))
            die("$hack detected!");
    }
    return $value;
}
?>
# 搭建CTF環境使用的sql語句
create database day1;
use day1;
create table users (
id int(6) unsigned auto_increment primary key,
name varchar(20) not null,
email varchar(30) not null,
salary int(8) unsigned not null );

INSERT INTO users VALUES(1,'Lucia','[email protected]',3000);
INSERT INTO users VALUES(2,'Danny','[email protected]',4500);
INSERT INTO users VALUES(3,'Alina','[email protected]',2700);
INSERT INTO users VALUES(4,'Jameson','[email protected]',10000);
INSERT INTO users VALUES(5,'Allie','[email protected]',6000);

create table flag(flag varchar(30) not null);
INSERT INTO flag VALUES('HRCTF{1n0rrY_i3_Vu1n3rab13}');

 

題解我們會階段性放出,如果大家有什麼好的解法,可以在文章底下留言,祝大家玩的愉快!

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章