2013-02-14 15:32:17 +04:00
|
|
|
#!/usr/bin/env python
|
2011-04-15 17:41:50 +04:00
|
|
|
|
|
|
|
"""
|
|
|
|
safe2bin.py - Simple safe(hex) to binary format converter
|
|
|
|
|
2018-01-02 02:48:10 +03:00
|
|
|
Copyright (c) 2006-2018 sqlmap developers (http://sqlmap.org/)
|
2017-10-11 15:50:46 +03:00
|
|
|
See the file 'LICENSE' for copying permission
|
2011-04-15 17:41:50 +04:00
|
|
|
"""
|
|
|
|
|
|
|
|
import binascii
|
|
|
|
import re
|
2011-04-15 17:51:06 +04:00
|
|
|
import string
|
2011-04-15 17:41:50 +04:00
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
|
|
|
|
from optparse import OptionError
|
|
|
|
from optparse import OptionParser
|
|
|
|
|
|
|
|
# Regex used for recognition of hex encoded characters
|
|
|
|
HEX_ENCODED_CHAR_REGEX = r"(?P<result>\\x[0-9A-Fa-f]{2})"
|
|
|
|
|
|
|
|
# Raw chars that will be safe encoded to their slash (\) representations (e.g. newline to \n)
|
2011-06-11 11:54:59 +04:00
|
|
|
SAFE_ENCODE_SLASH_REPLACEMENTS = "\t\n\r\x0b\x0c"
|
|
|
|
|
2011-12-23 00:08:28 +04:00
|
|
|
# Characters that don't need to be safe encoded
|
2016-09-09 12:06:38 +03:00
|
|
|
SAFE_CHARS = "".join(filter(lambda _: _ not in SAFE_ENCODE_SLASH_REPLACEMENTS, string.printable.replace('\\', '')))
|
2011-12-23 00:08:28 +04:00
|
|
|
|
2015-11-16 17:02:30 +03:00
|
|
|
# Prefix used for hex encoded values
|
|
|
|
HEX_ENCODED_PREFIX = r"\x"
|
|
|
|
|
|
|
|
# Strings used for temporary marking of hex encoded prefixes (to prevent double encoding)
|
|
|
|
HEX_ENCODED_PREFIX_MARKER = "__HEX_ENCODED_PREFIX__"
|
|
|
|
|
2011-06-11 11:54:59 +04:00
|
|
|
# String used for temporary marking of slash characters
|
|
|
|
SLASH_MARKER = "__SLASH__"
|
2011-04-15 17:41:50 +04:00
|
|
|
|
2011-04-15 17:51:06 +04:00
|
|
|
def safecharencode(value):
|
|
|
|
"""
|
|
|
|
Returns safe representation of a given basestring value
|
|
|
|
|
|
|
|
>>> safecharencode(u'test123')
|
|
|
|
u'test123'
|
|
|
|
>>> safecharencode(u'test\x01\x02\xff')
|
|
|
|
u'test\\01\\02\\03\\ff'
|
|
|
|
"""
|
|
|
|
|
|
|
|
retVal = value
|
|
|
|
|
|
|
|
if isinstance(value, basestring):
|
2016-09-09 12:06:38 +03:00
|
|
|
if any([_ not in SAFE_CHARS for _ in value]):
|
2015-11-16 17:02:30 +03:00
|
|
|
retVal = retVal.replace(HEX_ENCODED_PREFIX, HEX_ENCODED_PREFIX_MARKER)
|
2012-12-28 01:43:39 +04:00
|
|
|
retVal = retVal.replace('\\', SLASH_MARKER)
|
2011-04-15 17:51:06 +04:00
|
|
|
|
2011-12-23 00:08:28 +04:00
|
|
|
for char in SAFE_ENCODE_SLASH_REPLACEMENTS:
|
|
|
|
retVal = retVal.replace(char, repr(char).strip('\''))
|
2011-06-11 11:54:59 +04:00
|
|
|
|
2014-08-21 01:28:45 +04:00
|
|
|
retVal = reduce(lambda x, y: x + (y if (y in string.printable or isinstance(value, unicode) and ord(y) >= 160) else '\\x%02x' % ord(y)), retVal, (unicode if isinstance(value, unicode) else str)())
|
2012-12-28 01:43:39 +04:00
|
|
|
|
2013-01-16 17:16:22 +04:00
|
|
|
retVal = retVal.replace(SLASH_MARKER, "\\\\")
|
2015-11-16 17:02:30 +03:00
|
|
|
retVal = retVal.replace(HEX_ENCODED_PREFIX_MARKER, HEX_ENCODED_PREFIX)
|
2011-04-15 17:51:06 +04:00
|
|
|
elif isinstance(value, list):
|
|
|
|
for i in xrange(len(value)):
|
|
|
|
retVal[i] = safecharencode(value[i])
|
|
|
|
|
|
|
|
return retVal
|
|
|
|
|
2013-01-16 17:16:22 +04:00
|
|
|
def safechardecode(value, binary=False):
|
2011-04-15 17:41:50 +04:00
|
|
|
"""
|
2011-04-15 17:51:06 +04:00
|
|
|
Reverse function to safecharencode
|
2011-04-15 17:41:50 +04:00
|
|
|
"""
|
|
|
|
|
|
|
|
retVal = value
|
|
|
|
if isinstance(value, basestring):
|
2012-12-28 01:43:39 +04:00
|
|
|
retVal = retVal.replace('\\\\', SLASH_MARKER)
|
2011-04-15 17:41:50 +04:00
|
|
|
|
|
|
|
while True:
|
2012-12-28 01:43:39 +04:00
|
|
|
match = re.search(HEX_ENCODED_CHAR_REGEX, retVal)
|
2011-04-15 17:41:50 +04:00
|
|
|
if match:
|
2013-01-16 17:16:22 +04:00
|
|
|
retVal = retVal.replace(match.group("result"), (unichr if isinstance(value, unicode) else chr)(ord(binascii.unhexlify(match.group("result").lstrip("\\x")))))
|
2011-04-15 17:41:50 +04:00
|
|
|
else:
|
|
|
|
break
|
|
|
|
|
|
|
|
for char in SAFE_ENCODE_SLASH_REPLACEMENTS[::-1]:
|
|
|
|
retVal = retVal.replace(repr(char).strip('\''), char)
|
|
|
|
|
2011-06-11 11:54:59 +04:00
|
|
|
retVal = retVal.replace(SLASH_MARKER, '\\')
|
|
|
|
|
2013-01-16 17:16:22 +04:00
|
|
|
if binary:
|
|
|
|
if isinstance(retVal, unicode):
|
|
|
|
retVal = retVal.encode("utf8")
|
|
|
|
|
2011-04-15 17:41:50 +04:00
|
|
|
elif isinstance(value, (list, tuple)):
|
|
|
|
for i in xrange(len(value)):
|
|
|
|
retVal[i] = safechardecode(value[i])
|
|
|
|
|
|
|
|
return retVal
|
|
|
|
|
|
|
|
def main():
|
|
|
|
usage = '%s -i <input file> [-o <output file>]' % sys.argv[0]
|
2011-04-30 17:20:05 +04:00
|
|
|
parser = OptionParser(usage=usage, version='0.1')
|
2011-04-15 17:41:50 +04:00
|
|
|
|
|
|
|
try:
|
|
|
|
parser.add_option('-i', dest='inputFile', help='Input file')
|
|
|
|
parser.add_option('-o', dest='outputFile', help='Output file')
|
|
|
|
|
|
|
|
(args, _) = parser.parse_args()
|
|
|
|
|
|
|
|
if not args.inputFile:
|
|
|
|
parser.error('Missing the input file, -h for help')
|
|
|
|
|
|
|
|
except (OptionError, TypeError), e:
|
|
|
|
parser.error(e)
|
|
|
|
|
|
|
|
if not os.path.isfile(args.inputFile):
|
|
|
|
print 'ERROR: the provided input file \'%s\' is not a regular file' % args.inputFile
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
f = open(args.inputFile, 'r')
|
|
|
|
data = f.read()
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
if not args.outputFile:
|
|
|
|
args.outputFile = args.inputFile + '.bin'
|
|
|
|
|
|
|
|
f = open(args.outputFile, 'wb')
|
|
|
|
f.write(safechardecode(data))
|
|
|
|
f.close()
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|