2013-02-14 15:32:17 +04:00
|
|
|
#!/usr/bin/env python
|
2008-10-15 19:38:22 +04:00
|
|
|
|
|
|
|
"""
|
2013-01-18 18:07:51 +04:00
|
|
|
Copyright (c) 2006-2013 sqlmap developers (http://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
|
|
|
"""
|
|
|
|
|
2010-05-24 15:00:49 +04:00
|
|
|
import codecs
|
|
|
|
import os
|
2011-12-28 19:59:30 +04:00
|
|
|
import threading
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2012-11-28 15:28:42 +04:00
|
|
|
from xml.dom.minidom import getDOMImplementation
|
|
|
|
|
2011-02-21 01:41:42 +03:00
|
|
|
from lib.core.common import Backend
|
2008-10-15 19:38:22 +04:00
|
|
|
from lib.core.common import dataToDumpFile
|
2010-08-09 01:22:37 +04:00
|
|
|
from lib.core.common import dataToStdout
|
2010-06-02 16:31:36 +04:00
|
|
|
from lib.core.common import getUnicode
|
2012-06-14 17:38:53 +04:00
|
|
|
from lib.core.common import isListLike
|
2011-03-24 23:04:20 +03:00
|
|
|
from lib.core.common import normalizeUnicode
|
2011-01-08 12:30:10 +03:00
|
|
|
from lib.core.common import openFile
|
2012-05-04 02:34:18 +04:00
|
|
|
from lib.core.common import prioritySortColumns
|
2012-12-19 17:42:56 +04:00
|
|
|
from lib.core.common import randomInt
|
2011-07-03 02:48:56 +04:00
|
|
|
from lib.core.common import safeCSValue
|
2011-12-20 17:17:24 +04:00
|
|
|
from lib.core.common import unsafeSQLIdentificatorNaming
|
2008-10-15 19:38:22 +04:00
|
|
|
from lib.core.data import conf
|
2010-09-24 17:19:35 +04:00
|
|
|
from lib.core.data import kb
|
2008-10-15 19:38:22 +04:00
|
|
|
from lib.core.data import logger
|
2012-08-21 13:30:01 +04:00
|
|
|
from lib.core.dicts import DUMP_REPLACEMENTS
|
2013-02-03 15:31:05 +04:00
|
|
|
from lib.core.enums import CONTENT_STATUS
|
2013-01-30 19:30:34 +04:00
|
|
|
from lib.core.enums import CONTENT_TYPE
|
2011-02-21 01:41:42 +03:00
|
|
|
from lib.core.enums import DBMS
|
2012-11-28 13:58:18 +04:00
|
|
|
from lib.core.enums import DUMP_FORMAT
|
2012-12-06 17:14:19 +04:00
|
|
|
from lib.core.exception import SqlmapGenericException
|
|
|
|
from lib.core.exception import SqlmapValueException
|
2010-09-24 17:19:35 +04:00
|
|
|
from lib.core.replication import Replication
|
2012-11-28 15:46:43 +04:00
|
|
|
from lib.core.settings import HTML_DUMP_CSS_STYLE
|
|
|
|
from lib.core.settings import METADB_SUFFIX
|
2013-01-17 14:37:45 +04:00
|
|
|
from lib.core.settings import MIN_BINARY_DISK_DUMP_SIZE
|
2011-07-24 13:19:33 +04:00
|
|
|
from lib.core.settings import TRIM_STDOUT_DUMP_SIZE
|
2011-01-30 14:36:03 +03:00
|
|
|
from lib.core.settings import UNICODE_ENCODING
|
2012-12-19 17:42:56 +04:00
|
|
|
from thirdparty.magic import magic
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2013-01-11 17:31:44 +04:00
|
|
|
from extra.safe2bin.safe2bin import safechardecode
|
|
|
|
|
2012-12-06 13:42:53 +04:00
|
|
|
class Dump(object):
|
2008-10-15 19:38:22 +04:00
|
|
|
"""
|
|
|
|
This class defines methods used to parse and output the results
|
|
|
|
of SQL injection actions
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self):
|
2012-02-16 18:42:28 +04:00
|
|
|
self._outputFile = None
|
|
|
|
self._outputFP = None
|
|
|
|
self._lock = threading.Lock()
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2013-01-29 05:39:27 +04:00
|
|
|
def _write(self, data, newline=True, console=True, content_type=None):
|
|
|
|
if hasattr(conf, "api"):
|
2013-02-03 15:31:05 +04:00
|
|
|
dataToStdout(data, content_type=content_type, status=CONTENT_STATUS.COMPLETE)
|
2013-01-29 05:39:27 +04:00
|
|
|
return
|
|
|
|
|
2012-07-14 13:23:22 +04:00
|
|
|
text = "%s%s" % (data, "\n" if newline else " ")
|
2013-01-29 05:39:27 +04:00
|
|
|
|
2011-07-24 13:19:33 +04:00
|
|
|
if console:
|
|
|
|
dataToStdout(text)
|
2010-10-15 14:28:06 +04:00
|
|
|
|
2011-12-28 19:59:30 +04:00
|
|
|
if kb.get("multiThreadMode"):
|
2012-02-16 18:42:28 +04:00
|
|
|
self._lock.acquire()
|
2011-12-28 19:59:30 +04:00
|
|
|
|
2012-06-06 02:40:55 +04:00
|
|
|
self._outputFP.write(text)
|
2011-12-28 19:59:30 +04:00
|
|
|
|
|
|
|
if kb.get("multiThreadMode"):
|
2012-02-16 18:42:28 +04:00
|
|
|
self._lock.release()
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2011-05-11 00:44:36 +04:00
|
|
|
kb.dataOutputFlag = True
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2008-10-15 19:38:22 +04:00
|
|
|
def setOutputFile(self):
|
2012-12-19 17:45:52 +04:00
|
|
|
self._outputFile = "%s%slog" % (conf.outputPath, os.sep)
|
2012-07-03 02:50:23 +04:00
|
|
|
try:
|
2012-12-19 17:28:54 +04:00
|
|
|
self._outputFP = codecs.open(self._outputFile, "ab" if not conf.flushSession else "wb", UNICODE_ENCODING)
|
2012-07-03 02:50:23 +04:00
|
|
|
except IOError, ex:
|
|
|
|
errMsg = "error occurred while opening log file ('%s')" % ex
|
2013-01-04 02:20:55 +04:00
|
|
|
raise SqlmapGenericException(errMsg)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2010-09-26 18:56:55 +04:00
|
|
|
def getOutputFile(self):
|
2012-02-16 18:42:28 +04:00
|
|
|
return self._outputFile
|
2010-09-26 18:56:55 +04:00
|
|
|
|
2013-01-29 19:34:20 +04:00
|
|
|
def singleString(self, data, content_type=None):
|
|
|
|
self._write(data, content_type=content_type)
|
2012-12-17 17:02:09 +04:00
|
|
|
|
2013-01-29 05:39:27 +04:00
|
|
|
def string(self, header, data, content_type=None, sort=True):
|
2012-07-12 18:04:01 +04:00
|
|
|
kb.stickyLevel = None
|
|
|
|
|
2013-02-06 21:45:25 +04:00
|
|
|
if hasattr(conf, "api"):
|
|
|
|
self._write(data, content_type=content_type)
|
|
|
|
return
|
|
|
|
|
2012-06-14 17:38:53 +04:00
|
|
|
if isListLike(data):
|
2013-02-06 21:45:25 +04:00
|
|
|
self.lister(header, data, content_type, sort)
|
2012-07-12 13:47:51 +04:00
|
|
|
elif data is not None:
|
2012-07-13 12:11:16 +04:00
|
|
|
_ = getUnicode(data)
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2012-07-13 12:11:16 +04:00
|
|
|
if _ and _[-1] == '\n':
|
|
|
|
_ = _[:-1]
|
2011-05-23 18:28:05 +04:00
|
|
|
|
2013-02-06 21:45:25 +04:00
|
|
|
if "\n" in _:
|
2012-07-14 13:23:22 +04:00
|
|
|
self._write("%s:\n---\n%s\n---" % (header, _))
|
2008-10-15 19:38:22 +04:00
|
|
|
else:
|
2012-07-14 13:23:22 +04:00
|
|
|
self._write("%s: %s" % (header, ("'%s'" % _) if isinstance(data, basestring) else _))
|
2008-10-15 19:38:22 +04:00
|
|
|
else:
|
2012-07-14 13:23:22 +04:00
|
|
|
self._write("%s:\tNone" % header)
|
2010-09-24 17:19:35 +04:00
|
|
|
|
2013-01-29 05:39:27 +04:00
|
|
|
def lister(self, header, elements, content_type=None, sort=True):
|
|
|
|
if elements and sort:
|
2009-04-22 15:48:07 +04:00
|
|
|
try:
|
|
|
|
elements = set(elements)
|
|
|
|
elements = list(elements)
|
2011-03-18 19:52:46 +03:00
|
|
|
elements.sort(key=lambda x: x.lower() if isinstance(x, basestring) else x)
|
2009-04-22 15:48:07 +04:00
|
|
|
except:
|
|
|
|
pass
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2013-01-29 05:39:27 +04:00
|
|
|
if hasattr(conf, "api"):
|
|
|
|
self._write(elements, content_type=content_type)
|
|
|
|
return
|
|
|
|
|
|
|
|
if elements:
|
|
|
|
self._write("%s [%d]:" % (header, len(elements)))
|
|
|
|
|
2008-10-15 19:38:22 +04:00
|
|
|
for element in elements:
|
2010-05-25 14:09:35 +04:00
|
|
|
if isinstance(element, basestring):
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("[*] %s" % element)
|
2012-06-14 17:38:53 +04:00
|
|
|
elif isListLike(element):
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("[*] " + ", ".join(getUnicode(e) for e in element))
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2008-11-02 21:23:42 +03:00
|
|
|
if elements:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("")
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2013-01-10 16:18:44 +04:00
|
|
|
def banner(self, data):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.string("banner", data, content_type=CONTENT_TYPE.BANNER)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2013-01-10 16:18:44 +04:00
|
|
|
def currentUser(self, data):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.string("current user", data, content_type=CONTENT_TYPE.CURRENT_USER)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2013-01-10 16:18:44 +04:00
|
|
|
def currentDb(self, data):
|
2011-04-30 18:54:29 +04:00
|
|
|
if Backend.isDbms(DBMS.MAXDB):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.string("current database (no practical usage on %s)" % Backend.getIdentifiedDbms(), data, content_type=CONTENT_TYPE.CURRENT_DB)
|
2011-04-30 18:54:29 +04:00
|
|
|
elif Backend.isDbms(DBMS.ORACLE):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.string("current schema (equivalent to database on %s)" % Backend.getIdentifiedDbms(), data, content_type=CONTENT_TYPE.CURRENT_DB)
|
2011-02-21 01:41:42 +03:00
|
|
|
else:
|
2013-01-30 19:30:34 +04:00
|
|
|
self.string("current database", data, content_type=CONTENT_TYPE.CURRENT_DB)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2013-01-10 16:18:44 +04:00
|
|
|
def hostname(self, data):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.string("hostname", data, content_type=CONTENT_TYPE.HOSTNAME)
|
2012-07-12 03:01:57 +04:00
|
|
|
|
2013-01-10 16:18:44 +04:00
|
|
|
def dba(self, data):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.string("current user is DBA", data, content_type=CONTENT_TYPE.IS_DBA)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2013-01-10 16:18:44 +04:00
|
|
|
def users(self, users):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.lister("database management system users", users, content_type=CONTENT_TYPE.USERS)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2013-01-29 05:39:27 +04:00
|
|
|
def userSettings(self, header, userSettings, subHeader, content_type=None):
|
2012-02-16 18:42:28 +04:00
|
|
|
self._areAdmins = set()
|
2008-10-15 19:38:22 +04:00
|
|
|
|
|
|
|
if isinstance(userSettings, (tuple, list, set)):
|
2012-02-16 18:42:28 +04:00
|
|
|
self._areAdmins = userSettings[1]
|
2008-10-15 19:38:22 +04:00
|
|
|
userSettings = userSettings[0]
|
|
|
|
|
|
|
|
users = userSettings.keys()
|
2011-03-18 19:52:46 +03:00
|
|
|
users.sort(key=lambda x: x.lower() if isinstance(x, basestring) else x)
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2013-01-29 19:34:20 +04:00
|
|
|
if hasattr(conf, "api"):
|
|
|
|
self._write(userSettings, content_type=content_type)
|
|
|
|
return
|
|
|
|
|
|
|
|
if userSettings:
|
|
|
|
self._write("%s:" % header)
|
|
|
|
|
2008-10-15 19:38:22 +04:00
|
|
|
for user in users:
|
|
|
|
settings = userSettings[user]
|
|
|
|
|
2011-02-10 17:24:04 +03:00
|
|
|
if settings is None:
|
|
|
|
stringSettings = ""
|
|
|
|
else:
|
|
|
|
stringSettings = " [%d]:" % len(settings)
|
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
if user in self._areAdmins:
|
|
|
|
self._write("[*] %s (administrator)%s" % (user, stringSettings))
|
2008-10-15 19:38:22 +04:00
|
|
|
else:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("[*] %s%s" % (user, stringSettings))
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2011-02-10 17:24:04 +03:00
|
|
|
if settings:
|
|
|
|
settings.sort()
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2011-02-10 17:24:04 +03:00
|
|
|
for setting in settings:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write(" %s: %s" % (subHeader, setting))
|
2012-12-17 17:35:32 +04:00
|
|
|
|
2013-01-19 02:10:36 +04:00
|
|
|
if userSettings:
|
|
|
|
self.singleString("")
|
2010-01-09 03:05:00 +03:00
|
|
|
|
2013-01-10 16:18:44 +04:00
|
|
|
def dbs(self, dbs):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.lister("available databases", dbs, content_type=CONTENT_TYPE.DBS)
|
2010-01-09 03:05:00 +03:00
|
|
|
|
2013-01-29 19:34:20 +04:00
|
|
|
def dbTables(self, dbTables):
|
2010-11-11 23:37:25 +03:00
|
|
|
if isinstance(dbTables, dict) and len(dbTables) > 0:
|
2013-01-29 19:34:20 +04:00
|
|
|
if hasattr(conf, "api"):
|
2013-01-30 19:30:34 +04:00
|
|
|
self._write(dbTables, content_type=CONTENT_TYPE.TABLES)
|
2013-01-29 19:34:20 +04:00
|
|
|
return
|
|
|
|
|
2010-09-30 16:35:45 +04:00
|
|
|
maxlength = 0
|
|
|
|
|
|
|
|
for tables in dbTables.values():
|
|
|
|
for table in tables:
|
2012-06-14 17:38:53 +04:00
|
|
|
if table and isListLike(table):
|
2010-10-20 02:11:17 +04:00
|
|
|
table = table[0]
|
|
|
|
|
2013-02-15 17:48:24 +04:00
|
|
|
maxlength = max(maxlength, len(unsafeSQLIdentificatorNaming(normalizeUnicode(table) or str(table))))
|
2010-09-30 16:35:45 +04:00
|
|
|
|
|
|
|
lines = "-" * (int(maxlength) + 2)
|
|
|
|
|
|
|
|
for db, tables in dbTables.items():
|
2010-10-20 02:11:17 +04:00
|
|
|
tables.sort()
|
2010-09-30 16:35:45 +04:00
|
|
|
|
2013-02-15 17:05:14 +04:00
|
|
|
self._write("Database: %s" % unsafeSQLIdentificatorNaming(db) if db else "Current database")
|
2010-09-30 16:35:45 +04:00
|
|
|
|
|
|
|
if len(tables) == 1:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("[1 table]")
|
2010-09-30 16:35:45 +04:00
|
|
|
else:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("[%d tables]" % len(tables))
|
2010-09-30 16:35:45 +04:00
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("+%s+" % lines)
|
2010-09-30 16:35:45 +04:00
|
|
|
|
|
|
|
for table in tables:
|
2012-06-14 17:38:53 +04:00
|
|
|
if table and isListLike(table):
|
2010-10-20 02:11:17 +04:00
|
|
|
table = table[0]
|
|
|
|
|
2013-02-15 17:48:24 +04:00
|
|
|
table = unsafeSQLIdentificatorNaming(table)
|
2011-03-29 10:25:17 +04:00
|
|
|
blank = " " * (maxlength - len(normalizeUnicode(table) or str(table)))
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("| %s%s |" % (table, blank))
|
2010-09-30 16:35:45 +04:00
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("+%s+\n" % lines)
|
2012-03-08 19:11:24 +04:00
|
|
|
elif dbTables is None or len(dbTables) == 0:
|
2013-01-30 19:30:34 +04:00
|
|
|
self.singleString("No tables found", content_type=CONTENT_TYPE.TABLES)
|
2010-09-30 16:35:45 +04:00
|
|
|
else:
|
2013-01-30 19:30:34 +04:00
|
|
|
self.string("tables", dbTables, content_type=CONTENT_TYPE.TABLES)
|
2010-12-21 13:31:56 +03:00
|
|
|
|
2013-01-29 19:34:20 +04:00
|
|
|
def dbTableColumns(self, tableColumns, content_type=None):
|
2011-09-19 23:08:08 +04:00
|
|
|
if isinstance(tableColumns, dict) and len(tableColumns) > 0:
|
2013-01-29 19:34:20 +04:00
|
|
|
if hasattr(conf, "api"):
|
|
|
|
self._write(tableColumns, content_type=content_type)
|
|
|
|
return
|
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
for db, tables in tableColumns.items():
|
|
|
|
if not db:
|
|
|
|
db = "All"
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
for table, columns in tables.items():
|
|
|
|
maxlength1 = 0
|
|
|
|
maxlength2 = 0
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
colType = None
|
2011-01-20 00:46:43 +03:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
colList = columns.keys()
|
|
|
|
colList.sort(key=lambda x: x.lower() if isinstance(x, basestring) else x)
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
for column in colList:
|
|
|
|
colType = columns[column]
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2013-02-15 17:48:24 +04:00
|
|
|
column = unsafeSQLIdentificatorNaming(column)
|
2011-09-19 23:08:08 +04:00
|
|
|
maxlength1 = max(maxlength1, len(column or ""))
|
|
|
|
maxlength2 = max(maxlength2, len(colType or ""))
|
2010-01-09 03:05:00 +03:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
maxlength1 = max(maxlength1, len("COLUMN"))
|
|
|
|
lines1 = "-" * (maxlength1 + 2)
|
2010-11-09 19:15:55 +03:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
if colType is not None:
|
|
|
|
maxlength2 = max(maxlength2, len("TYPE"))
|
|
|
|
lines2 = "-" * (maxlength2 + 2)
|
2010-11-09 19:15:55 +03:00
|
|
|
|
2013-02-15 17:05:14 +04:00
|
|
|
self._write("Database: %s\nTable: %s" % (unsafeSQLIdentificatorNaming(db) if db else "Current database", unsafeSQLIdentificatorNaming(table)))
|
2010-11-09 19:15:55 +03:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
if len(columns) == 1:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("[1 column]")
|
2011-09-19 23:08:08 +04:00
|
|
|
else:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("[%d columns]" % len(columns))
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
if colType is not None:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("+%s+%s+" % (lines1, lines2))
|
2011-09-19 23:08:08 +04:00
|
|
|
else:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("+%s+" % lines1)
|
2010-11-09 19:15:55 +03:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
blank1 = " " * (maxlength1 - len("COLUMN"))
|
2010-11-09 19:15:55 +03:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
if colType is not None:
|
|
|
|
blank2 = " " * (maxlength2 - len("TYPE"))
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
if colType is not None:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("| Column%s | Type%s |" % (blank1, blank2))
|
|
|
|
self._write("+%s+%s+" % (lines1, lines2))
|
2011-09-19 23:08:08 +04:00
|
|
|
else:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("| Column%s |" % blank1)
|
|
|
|
self._write("+%s+" % lines1)
|
2010-01-09 03:05:00 +03:00
|
|
|
|
2011-09-19 23:08:08 +04:00
|
|
|
for column in colList:
|
|
|
|
colType = columns[column]
|
2013-02-15 17:48:24 +04:00
|
|
|
|
|
|
|
column = unsafeSQLIdentificatorNaming(column)
|
2011-09-19 23:08:08 +04:00
|
|
|
blank1 = " " * (maxlength1 - len(column))
|
|
|
|
|
|
|
|
if colType is not None:
|
|
|
|
blank2 = " " * (maxlength2 - len(colType))
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("| %s%s | %s%s |" % (column, blank1, colType, blank2))
|
2011-09-19 23:08:08 +04:00
|
|
|
else:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("| %s%s |" % (column, blank1))
|
2010-11-09 19:15:55 +03:00
|
|
|
|
|
|
|
if colType is not None:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("+%s+%s+\n" % (lines1, lines2))
|
2010-11-09 19:15:55 +03:00
|
|
|
else:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("+%s+\n" % lines1)
|
2010-01-09 03:05:00 +03:00
|
|
|
|
2013-01-29 19:34:20 +04:00
|
|
|
def dbTablesCount(self, dbTables):
|
2011-04-30 04:22:22 +04:00
|
|
|
if isinstance(dbTables, dict) and len(dbTables) > 0:
|
2013-01-29 19:34:20 +04:00
|
|
|
if hasattr(conf, "api"):
|
2013-01-30 19:30:34 +04:00
|
|
|
self._write(dbTables, content_type=CONTENT_TYPE.COUNT)
|
2013-01-29 19:34:20 +04:00
|
|
|
return
|
|
|
|
|
2011-04-30 04:22:22 +04:00
|
|
|
maxlength1 = len("Table")
|
|
|
|
maxlength2 = len("Entries")
|
|
|
|
|
|
|
|
for ctables in dbTables.values():
|
|
|
|
for tables in ctables.values():
|
|
|
|
for table in tables:
|
|
|
|
maxlength1 = max(maxlength1, len(normalizeUnicode(table) or str(table)))
|
|
|
|
|
|
|
|
for db, counts in dbTables.items():
|
2013-02-15 17:05:14 +04:00
|
|
|
self._write("Database: %s" % unsafeSQLIdentificatorNaming(db) if db else "Current database")
|
2011-04-30 04:22:22 +04:00
|
|
|
|
|
|
|
lines1 = "-" * (maxlength1 + 2)
|
|
|
|
blank1 = " " * (maxlength1 - len("Table"))
|
|
|
|
lines2 = "-" * (maxlength2 + 2)
|
|
|
|
blank2 = " " * (maxlength2 - len("Entries"))
|
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("+%s+%s+" % (lines1, lines2))
|
|
|
|
self._write("| Table%s | Entries%s |" % (blank1, blank2))
|
|
|
|
self._write("+%s+%s+" % (lines1, lines2))
|
2011-04-30 04:22:22 +04:00
|
|
|
|
|
|
|
sortedCounts = counts.keys()
|
|
|
|
sortedCounts.sort(reverse=True)
|
|
|
|
|
|
|
|
for count in sortedCounts:
|
|
|
|
tables = counts[count]
|
|
|
|
|
|
|
|
if count is None:
|
|
|
|
count = "Unknown"
|
|
|
|
|
|
|
|
tables.sort(key=lambda x: x.lower() if isinstance(x, basestring) else x)
|
|
|
|
|
|
|
|
for table in tables:
|
|
|
|
blank1 = " " * (maxlength1 - len(normalizeUnicode(table) or str(table)))
|
|
|
|
blank2 = " " * (maxlength2 - len(str(count)))
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("| %s%s | %d%s |" % (table, blank1, count, blank2))
|
2011-04-30 04:22:22 +04:00
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("+%s+%s+\n" % (lines1, lines2))
|
2011-04-30 04:22:22 +04:00
|
|
|
else:
|
|
|
|
logger.error("unable to retrieve the number of entries for any table")
|
|
|
|
|
2013-01-29 19:34:20 +04:00
|
|
|
def dbTableValues(self, tableValues):
|
2010-09-24 17:34:46 +04:00
|
|
|
replication = None
|
2011-04-30 17:20:05 +04:00
|
|
|
rtable = None
|
2012-11-28 15:28:42 +04:00
|
|
|
dumpFP = None
|
2010-09-24 17:34:46 +04:00
|
|
|
|
2010-01-09 03:05:00 +03:00
|
|
|
if tableValues is None:
|
|
|
|
return
|
|
|
|
|
2008-10-15 19:38:22 +04:00
|
|
|
db = tableValues["__infos__"]["db"]
|
|
|
|
if not db:
|
|
|
|
db = "All"
|
|
|
|
table = tableValues["__infos__"]["table"]
|
|
|
|
|
2013-01-29 19:34:20 +04:00
|
|
|
if hasattr(conf, "api"):
|
2013-01-30 19:30:34 +04:00
|
|
|
self._write(tableValues, content_type=CONTENT_TYPE.DUMP_TABLE)
|
2013-01-29 19:34:20 +04:00
|
|
|
return
|
|
|
|
|
2013-03-02 12:55:12 +04:00
|
|
|
dumpDbPath = "%s%s%s" % (conf.dumpPath, os.sep, unsafeSQLIdentificatorNaming(db))
|
|
|
|
|
2012-11-28 13:58:18 +04:00
|
|
|
if conf.dumpFormat == DUMP_FORMAT.SQLITE:
|
2011-12-20 17:17:24 +04:00
|
|
|
replication = Replication("%s%s%s.sqlite3" % (conf.dumpPath, os.sep, unsafeSQLIdentificatorNaming(db)))
|
2012-11-28 15:28:42 +04:00
|
|
|
elif conf.dumpFormat in (DUMP_FORMAT.CSV, DUMP_FORMAT.HTML):
|
2008-10-15 19:38:22 +04:00
|
|
|
if not os.path.isdir(dumpDbPath):
|
|
|
|
os.makedirs(dumpDbPath, 0755)
|
|
|
|
|
2012-11-28 15:28:42 +04:00
|
|
|
dumpFileName = "%s%s%s.%s" % (dumpDbPath, os.sep, unsafeSQLIdentificatorNaming(table), conf.dumpFormat.lower())
|
2011-01-08 12:30:10 +03:00
|
|
|
dumpFP = openFile(dumpFileName, "wb")
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2011-04-30 17:20:05 +04:00
|
|
|
count = int(tableValues["__infos__"]["count"])
|
|
|
|
separator = str()
|
|
|
|
field = 1
|
|
|
|
fields = len(tableValues) - 1
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2012-05-04 02:34:18 +04:00
|
|
|
columns = prioritySortColumns(tableValues.keys())
|
2008-10-15 19:38:22 +04:00
|
|
|
|
|
|
|
for column in columns:
|
|
|
|
if column != "__infos__":
|
2011-04-30 17:20:05 +04:00
|
|
|
info = tableValues[column]
|
|
|
|
lines = "-" * (int(info["length"]) + 2)
|
2008-10-15 19:38:22 +04:00
|
|
|
separator += "+%s" % lines
|
|
|
|
|
|
|
|
separator += "+"
|
2013-02-15 17:05:14 +04:00
|
|
|
self._write("Database: %s\nTable: %s" % (unsafeSQLIdentificatorNaming(db) if db else "Current database", unsafeSQLIdentificatorNaming(table)))
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2012-11-28 13:58:18 +04:00
|
|
|
if conf.dumpFormat == DUMP_FORMAT.SQLITE:
|
2010-09-24 18:34:05 +04:00
|
|
|
cols = []
|
2010-11-04 15:21:06 +03:00
|
|
|
|
2010-09-24 18:34:05 +04:00
|
|
|
for column in columns:
|
|
|
|
if column != "__infos__":
|
|
|
|
colType = Replication.INTEGER
|
2010-11-04 15:21:06 +03:00
|
|
|
|
2010-09-24 18:34:05 +04:00
|
|
|
for value in tableValues[column]['values']:
|
|
|
|
try:
|
2012-03-14 17:52:23 +04:00
|
|
|
if not value or value == " ": # NULL
|
2010-09-24 18:34:05 +04:00
|
|
|
continue
|
2010-11-04 15:21:06 +03:00
|
|
|
|
2012-02-22 14:45:10 +04:00
|
|
|
int(value)
|
2010-09-24 18:34:05 +04:00
|
|
|
except ValueError:
|
|
|
|
colType = None
|
|
|
|
break
|
2010-11-04 15:21:06 +03:00
|
|
|
|
2010-09-24 18:34:05 +04:00
|
|
|
if colType is None:
|
|
|
|
colType = Replication.REAL
|
2010-11-04 15:21:06 +03:00
|
|
|
|
2010-09-24 18:34:05 +04:00
|
|
|
for value in tableValues[column]['values']:
|
|
|
|
try:
|
2012-03-14 17:52:23 +04:00
|
|
|
if not value or value == " ": # NULL
|
2010-09-24 18:34:05 +04:00
|
|
|
continue
|
2010-11-04 15:21:06 +03:00
|
|
|
|
2012-02-22 14:45:10 +04:00
|
|
|
float(value)
|
2010-09-24 18:34:05 +04:00
|
|
|
except ValueError:
|
|
|
|
colType = None
|
|
|
|
break
|
2010-11-04 15:21:06 +03:00
|
|
|
|
2010-09-24 18:34:05 +04:00
|
|
|
cols.append((column, colType if colType else Replication.TEXT))
|
2010-11-04 15:21:06 +03:00
|
|
|
|
2010-09-24 18:34:05 +04:00
|
|
|
rtable = replication.createTable(table, cols)
|
2012-11-28 15:28:42 +04:00
|
|
|
elif conf.dumpFormat == DUMP_FORMAT.HTML:
|
|
|
|
documentNode = getDOMImplementation().createDocument(None, "table", None)
|
|
|
|
tableNode = documentNode.documentElement
|
2010-09-24 17:19:35 +04:00
|
|
|
|
2008-10-15 19:38:22 +04:00
|
|
|
if count == 1:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("[1 entry]")
|
2008-10-15 19:38:22 +04:00
|
|
|
else:
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("[%d entries]" % count)
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write(separator)
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2012-11-28 15:28:42 +04:00
|
|
|
if conf.dumpFormat == DUMP_FORMAT.HTML:
|
2012-11-29 18:36:38 +04:00
|
|
|
headNode = documentNode.createElement("thead")
|
2012-11-28 15:28:42 +04:00
|
|
|
rowNode = documentNode.createElement("tr")
|
2012-11-29 18:36:38 +04:00
|
|
|
tableNode.appendChild(headNode)
|
|
|
|
headNode.appendChild(rowNode)
|
|
|
|
bodyNode = documentNode.createElement("tbody")
|
|
|
|
tableNode.appendChild(bodyNode)
|
2012-11-28 15:28:42 +04:00
|
|
|
|
2008-10-15 19:38:22 +04:00
|
|
|
for column in columns:
|
|
|
|
if column != "__infos__":
|
2011-04-30 17:20:05 +04:00
|
|
|
info = tableValues[column]
|
2008-10-15 19:38:22 +04:00
|
|
|
maxlength = int(info["length"])
|
2011-04-30 17:20:05 +04:00
|
|
|
blank = " " * (maxlength - len(column))
|
2009-05-20 17:56:23 +04:00
|
|
|
|
2012-07-14 13:23:22 +04:00
|
|
|
self._write("| %s%s" % (column, blank), newline=False)
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2012-11-28 14:16:00 +04:00
|
|
|
if conf.dumpFormat == DUMP_FORMAT.CSV:
|
2011-11-24 00:56:22 +04:00
|
|
|
if field == fields:
|
2011-07-03 02:48:56 +04:00
|
|
|
dataToDumpFile(dumpFP, "%s" % safeCSValue(column))
|
2011-11-24 00:56:22 +04:00
|
|
|
else:
|
2011-11-30 21:39:41 +04:00
|
|
|
dataToDumpFile(dumpFP, "%s%s" % (safeCSValue(column), conf.csvDel))
|
2012-11-28 15:28:42 +04:00
|
|
|
elif conf.dumpFormat == DUMP_FORMAT.HTML:
|
|
|
|
entryNode = documentNode.createElement("td")
|
|
|
|
rowNode.appendChild(entryNode)
|
|
|
|
entryNode.appendChild(documentNode.createTextNode(column))
|
2008-10-15 19:38:22 +04:00
|
|
|
|
|
|
|
field += 1
|
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("|\n%s" % separator)
|
2009-05-20 17:56:23 +04:00
|
|
|
|
2012-11-28 14:16:00 +04:00
|
|
|
if conf.dumpFormat == DUMP_FORMAT.CSV:
|
2008-10-15 19:38:22 +04:00
|
|
|
dataToDumpFile(dumpFP, "\n")
|
|
|
|
|
2012-11-28 14:16:00 +04:00
|
|
|
elif conf.dumpFormat == DUMP_FORMAT.SQLITE:
|
2011-04-14 18:34:12 +04:00
|
|
|
rtable.beginTransaction()
|
|
|
|
|
2011-07-24 13:19:33 +04:00
|
|
|
if count > TRIM_STDOUT_DUMP_SIZE:
|
2011-12-23 00:54:20 +04:00
|
|
|
warnMsg = "console output will be trimmed to "
|
|
|
|
warnMsg += "last %d rows due to " % TRIM_STDOUT_DUMP_SIZE
|
|
|
|
warnMsg += "large table size"
|
2011-07-24 13:19:33 +04:00
|
|
|
logger.warning(warnMsg)
|
|
|
|
|
2011-10-22 02:34:27 +04:00
|
|
|
for i in xrange(count):
|
2011-07-24 13:19:33 +04:00
|
|
|
console = (i >= count - TRIM_STDOUT_DUMP_SIZE)
|
2008-10-15 19:38:22 +04:00
|
|
|
field = 1
|
2010-09-24 17:19:35 +04:00
|
|
|
values = []
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2012-11-28 15:28:42 +04:00
|
|
|
if conf.dumpFormat == DUMP_FORMAT.HTML:
|
|
|
|
rowNode = documentNode.createElement("tr")
|
2012-11-29 18:36:38 +04:00
|
|
|
bodyNode.appendChild(rowNode)
|
2012-11-28 15:28:42 +04:00
|
|
|
|
2008-10-15 19:38:22 +04:00
|
|
|
for column in columns:
|
|
|
|
if column != "__infos__":
|
|
|
|
info = tableValues[column]
|
|
|
|
|
2011-01-04 02:36:35 +03:00
|
|
|
if len(info["values"]) <= i:
|
|
|
|
continue
|
|
|
|
|
2011-05-12 16:00:17 +04:00
|
|
|
if info["values"][i] is None:
|
|
|
|
value = u''
|
|
|
|
else:
|
|
|
|
value = getUnicode(info["values"][i])
|
2012-08-21 13:30:01 +04:00
|
|
|
value = DUMP_REPLACEMENTS.get(value, value)
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2010-09-24 17:19:35 +04:00
|
|
|
values.append(value)
|
2008-10-15 19:38:22 +04:00
|
|
|
maxlength = int(info["length"])
|
2010-05-24 15:00:49 +04:00
|
|
|
blank = " " * (maxlength - len(value))
|
2012-07-14 13:23:22 +04:00
|
|
|
self._write("| %s%s" % (value, blank), newline=False, console=console)
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2013-01-17 14:37:45 +04:00
|
|
|
if len(value) > MIN_BINARY_DISK_DUMP_SIZE and r'\x' in value:
|
2013-01-16 17:54:37 +04:00
|
|
|
mimetype = magic.from_buffer(value, mime=True)
|
|
|
|
if any(mimetype.startswith(_) for _ in ("application", "image")):
|
2013-03-02 12:55:12 +04:00
|
|
|
if not os.path.isdir(dumpDbPath):
|
|
|
|
os.makedirs(dumpDbPath, 0755)
|
|
|
|
|
2013-01-16 17:54:37 +04:00
|
|
|
filepath = os.path.join(dumpDbPath, "%s-%d.bin" % (column, randomInt(8)))
|
|
|
|
warnMsg = "writing binary ('%s') content to file '%s' " % (mimetype, filepath)
|
|
|
|
logger.warn(warnMsg)
|
|
|
|
|
|
|
|
with open(filepath, "wb") as f:
|
|
|
|
_ = safechardecode(value, True)
|
|
|
|
f.write(_)
|
2012-12-19 17:42:56 +04:00
|
|
|
|
2012-11-28 14:16:00 +04:00
|
|
|
if conf.dumpFormat == DUMP_FORMAT.CSV:
|
2011-11-24 00:56:22 +04:00
|
|
|
if field == fields:
|
2011-07-03 02:48:56 +04:00
|
|
|
dataToDumpFile(dumpFP, "%s" % safeCSValue(value))
|
2011-11-24 00:56:22 +04:00
|
|
|
else:
|
2011-11-30 21:39:41 +04:00
|
|
|
dataToDumpFile(dumpFP, "%s%s" % (safeCSValue(value), conf.csvDel))
|
2012-11-28 15:28:42 +04:00
|
|
|
elif conf.dumpFormat == DUMP_FORMAT.HTML:
|
|
|
|
entryNode = documentNode.createElement("td")
|
|
|
|
rowNode.appendChild(entryNode)
|
|
|
|
entryNode.appendChild(documentNode.createTextNode(value))
|
2008-10-15 19:38:22 +04:00
|
|
|
|
|
|
|
field += 1
|
|
|
|
|
2012-11-28 13:58:18 +04:00
|
|
|
if conf.dumpFormat == DUMP_FORMAT.SQLITE:
|
2011-12-27 15:25:40 +04:00
|
|
|
try:
|
|
|
|
rtable.insert(values)
|
2012-12-06 17:14:19 +04:00
|
|
|
except SqlmapValueException:
|
2011-12-27 15:25:40 +04:00
|
|
|
pass
|
2012-11-28 14:16:00 +04:00
|
|
|
elif conf.dumpFormat == DUMP_FORMAT.CSV:
|
|
|
|
dataToDumpFile(dumpFP, "\n")
|
2010-09-24 17:19:35 +04:00
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("|", console=console)
|
2009-05-20 17:56:23 +04:00
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write("%s\n" % separator)
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2012-11-28 13:58:18 +04:00
|
|
|
if conf.dumpFormat == DUMP_FORMAT.SQLITE:
|
2011-04-14 18:34:12 +04:00
|
|
|
rtable.endTransaction()
|
2012-07-13 17:22:08 +04:00
|
|
|
logger.info("table '%s.%s' dumped to sqlite3 database '%s'" % (db, table, replication.dbpath))
|
2011-04-14 18:34:12 +04:00
|
|
|
|
2012-11-28 15:28:42 +04:00
|
|
|
elif conf.dumpFormat in (DUMP_FORMAT.CSV, DUMP_FORMAT.HTML):
|
|
|
|
if conf.dumpFormat == DUMP_FORMAT.HTML:
|
2012-11-30 14:43:50 +04:00
|
|
|
dataToDumpFile(dumpFP, "<!DOCTYPE html>\n<html>\n<head>\n")
|
|
|
|
dataToDumpFile(dumpFP, "<meta http-equiv=\"Content-type\" content=\"text/html;charset=%s\">\n" % UNICODE_ENCODING)
|
|
|
|
dataToDumpFile(dumpFP, "<title>%s</title>\n" % ("%s%s" % ("%s." % db if METADB_SUFFIX not in db else "", table)))
|
2012-11-28 15:46:43 +04:00
|
|
|
dataToDumpFile(dumpFP, HTML_DUMP_CSS_STYLE)
|
|
|
|
dataToDumpFile(dumpFP, "\n</head>\n")
|
2012-11-28 15:28:42 +04:00
|
|
|
dataToDumpFile(dumpFP, tableNode.toxml())
|
2012-11-28 15:46:43 +04:00
|
|
|
dataToDumpFile(dumpFP, "\n</html>")
|
2012-11-28 15:28:42 +04:00
|
|
|
else:
|
|
|
|
dataToDumpFile(dumpFP, "\n")
|
2008-10-15 19:38:22 +04:00
|
|
|
dumpFP.close()
|
2012-11-28 15:28:42 +04:00
|
|
|
logger.info("table '%s.%s' dumped to %s file '%s'" % (db, table, conf.dumpFormat, dumpFileName))
|
2008-10-15 19:38:22 +04:00
|
|
|
|
2013-01-29 19:34:20 +04:00
|
|
|
def dbColumns(self, dbColumnsDict, colConsider, dbs):
|
|
|
|
if hasattr(conf, "api"):
|
2013-01-30 19:30:34 +04:00
|
|
|
self._write(dbColumnsDict, content_type=CONTENT_TYPE.COLUMNS)
|
2013-01-29 19:34:20 +04:00
|
|
|
return
|
|
|
|
|
2011-06-24 13:29:11 +04:00
|
|
|
for column in dbColumnsDict.keys():
|
2010-05-28 20:43:04 +04:00
|
|
|
if colConsider == "1":
|
2013-02-15 19:48:58 +04:00
|
|
|
colConsiderStr = "s like '%s' were" % unsafeSQLIdentificatorNaming(column)
|
2010-05-28 20:43:04 +04:00
|
|
|
else:
|
2013-02-15 19:48:58 +04:00
|
|
|
colConsiderStr = " '%s' was" % unsafeSQLIdentificatorNaming(column)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2011-04-30 17:20:05 +04:00
|
|
|
msg = "Column%s found in the " % colConsiderStr
|
2010-05-28 20:43:04 +04:00
|
|
|
msg += "following databases:"
|
2012-02-16 18:42:28 +04:00
|
|
|
self._write(msg)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
_ = {}
|
2010-05-28 20:43:04 +04:00
|
|
|
|
|
|
|
for db, tblData in dbs.items():
|
|
|
|
for tbl, colData in tblData.items():
|
|
|
|
for col, dataType in colData.items():
|
|
|
|
if column.lower() in col.lower():
|
2012-02-16 18:42:28 +04:00
|
|
|
if db in _:
|
|
|
|
if tbl in _[db]:
|
|
|
|
_[db][tbl][col] = dataType
|
2010-05-28 20:43:04 +04:00
|
|
|
else:
|
2012-07-13 14:22:37 +04:00
|
|
|
_[db][tbl] = {col: dataType}
|
2010-05-28 20:43:04 +04:00
|
|
|
else:
|
2012-02-16 18:42:28 +04:00
|
|
|
_[db] = {}
|
2012-07-13 14:22:37 +04:00
|
|
|
_[db][tbl] = {col: dataType}
|
2010-05-28 20:43:04 +04:00
|
|
|
|
|
|
|
continue
|
|
|
|
|
2012-02-16 18:42:28 +04:00
|
|
|
self.dbTableColumns(_)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
|
|
|
def query(self, query, queryRes):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.string(query, queryRes, content_type=CONTENT_TYPE.SQL_QUERY)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2012-12-19 18:12:45 +04:00
|
|
|
def rFile(self, fileData):
|
2013-01-30 19:30:34 +04:00
|
|
|
self.lister("files saved to", fileData, sort=False, content_type=CONTENT_TYPE.FILE_READ)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2013-02-04 18:31:28 +04:00
|
|
|
def registerValue(self, registerData):
|
2013-02-14 17:17:30 +04:00
|
|
|
self.string("Registry key value data", registerData, content_type=CONTENT_TYPE.REG_READ, sort=False)
|
2010-05-28 20:43:04 +04:00
|
|
|
|
2008-10-15 19:38:22 +04:00
|
|
|
# object to manage how to print the retrieved queries output to
|
|
|
|
# standard output and sessions file
|
|
|
|
dumper = Dump()
|