BugKu_never_give_up

打印 上一主题 下一主题

主题 787|帖子 787|积分 2361

  1. if(!$_GET['id'])
  2. {
  3.         header('Location: hello.php?id=1');
  4.         exit();
  5. }
  6. $id=$_GET['id'];
  7. $a=$_GET['a'];
  8. $b=$_GET['b'];
  9. if(stripos($a,'.'))
  10. {
  11.         echo 'no no no no no no no';
  12.         return ;
  13. }
  14. $data = @file_get_contents($a,'r');
  15. if($data=="bugku is a nice plateform!" and $id==0 and strlen($b)>5 and eregi("111".substr($b,0,1),"1114") and substr($b,0,1)!=4)
  16. {
  17.         $flag = "flag{***********}";
  18. }
  19. else
  20. {
  21.         print "never never never give up !!!";
  22. }
  23. ?>
复制代码
2023年2月23日遇到的一个php代码审计的题目,这个题目的考察知识点非常多,写个wp记录一下。
  1. if(!$_GET['id'])
  2. {
  3.         header('Location: hello.php?id=1');
  4.         exit();
  5. }
复制代码
如果id=0或者没有设置id的值,则跳转id=1,程序退出
  1. $id=$_GET['id'];
  2. $a=$_GET['a'];
  3. $b=$_GET['b'];
  4. if(stripos($a,'.'))
  5. {
  6.         echo 'no no no no no no no';
  7.         return ;
  8. }
复制代码
三个参数 id  a  b ,其中a参数的值不能有'.',如果有,则输出'no no no',并返回。
  1. $data = @file_get_contents($a,'r');
复制代码
file_get_contents从$a里读取数据,然后将数据给变量$data存储。
  1. if($data=="bugku is a nice plateform!" and $id==0 and strlen($b)>5 and eregi("111".substr($b,0,1),"1114") and substr($b,0,1)!=4)
  2. {
  3.         $flag = "flag{***********}";
  4. }
  5. else
  6. {
  7.         print "never never never give up !!!";
  8. }
复制代码
这个判断非常有意思,有很多矛盾的地方。首先data的值必须是"bugku is a nice plateform!",id必须为0(第一段代码就表示过id!=0),b的长度必须大于5,在正则匹配时,“111”和b的第一个字符拼接必须匹配到”1114“,但是b的第一个字符又不能是4。整个过程都非常矛盾,但是这正是考察php的知识点。
首先,data的值必须是"bugku is a nice plateform!",这个怎么办到?我们知道file_get_contents的参数是文件名,本质上是打开一个文件流,从流里读出文件内容。但我们并不知道哪个文件里有这个字符串信息,也没有上传接口,无法自己上传一个文件上去。这时,需要观察到本质,file_get_contents本质上是打开一个文件流,流!!!,重点是流。我们可以用php伪协议来完成。
  1. php://input  // 读取post输入流
复制代码
此时$a=php://input,然后用post传输"bugku is a nice plateform!",当file_get_contents($a)触发时,就是从Post的输入流里获取字符串“bugku is a nice plateform!”。
接着,id==0怎么绕过,毕竟一开始就判断过,id肯定不能是0,要是0,就退出程序了。但仔细看看,==在php中是弱类型比较,0bcde==0这个是成立的。为什么呢?因为数字之间比较时,非数字不参与比较。因此id=0a就可以绕过。
strlen($b) >5 没什么可说的,个数大于5就好。
eregi("111".substr($b,0,1),"1114") and substr($b,0,1)!=4),按照普遍的思路来说
  1. eregi("111".substr($b,0,1),"1114")   
  2. eregi()匹配函数,匹配到则是True
  3. "111".substr($b,0,1)  ->  拼接 “111”和  $b的第一个字符  (.是拼接字符串的意思)
  4. 比如$b=4321    那么  "111".substr($b,0,1) 就是1114
复制代码
但是,绕过就是不走寻常路。令b=*12345。*号是通配符,匹配所有。因此,直接绕过!!!
整体构造如下:


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

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

本版积分规则

八卦阵

金牌会员
这个人很懒什么都没写!

标签云

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