import sys
from array import array
try:
import mx.DateTime.ISO
origISOStr = mx.DateTime.ISO.strGMT
from mx.DateTime import DateTimeType,DateTimeDeltaType
except ImportError:
try:
import DateTime.ISO
origISOStr = DateTime.ISO.strGMT
from DateTime import DateTimeType,DateTimeDeltaType
except ImportError:
origISOStr = None
DateTimeType = None
DateTimeDeltaType = None
import time
import datetime
try:
import Sybase
NumericType=Sybase.NumericType
except ImportError:
NumericType = None
from decimal import Decimal
from types import ClassType,InstanceType,NoneType
########################################
## Quoting
########################################
sqlStringReplace = [
("'", "''"),
('\\', '\\\\'),
('\000', '\\0'),
('\b', '\\b'),
('\n', '\\n'),
('\r', '\\r'),
('\t', '\\t'),
]
def isoStr(val):
"""
Gets rid of time zone information
(@@: should we convert to GMT?)
"""
val = origISOStr(val)
if val.find('+') == -1:
return val
else:
return val[:val.find('+')]
class ConverterRegistry:
def __init__(self):
self.basic = {}
self.klass = {}
def registerConverter(self, typ, func):
if type(typ) is ClassType:
self.klass[typ] = func
else:
self.basic[typ] = func
def lookupConverter(self, value, default=None):
if type(value) is InstanceType:
# lookup on klasses dict
return self.klass.get(value.__class__, default)
return self.basic.get(type(value), default)
converters = ConverterRegistry()
registerConverter = converters.registerConverter
lookupConverter = converters.lookupConverter
def StringLikeConverter(value, db):
if isinstance(value, array):
try:
value = value.tounicode()
except ValueError:
value = value.tostring()
elif isinstance(value, buffer):
value = str(value)
if db in ('mysql', 'postgres'):
for orig, repl in sqlStringReplace:
value = value.replace(orig, repl)
elif db in ('sqlite', 'firebird', 'sybase', 'maxdb', 'mssql'):
value = value.replace("'", "''")
else:
assert 0, "Database %s unknown" % db
return "'%s'" % value
registerConverter(str, StringLikeConverter)
registerConverter(unicode, StringLikeConverter)
registerConverter(array, StringLikeConverter)
registerConverter(buffer, StringLikeConverter)
def IntConverter(value, db):
return repr(int(value))
registerConverter(int, IntConverter)
def LongConverter(value, db):
return str(value)
registerConverter(long, LongConverter)
if NumericType:
registerConverter(NumericType, IntConverter)
def BoolConverter(value, db):
if db in ('postgres',):
if value:
return "'t'"
else:
return "'f'"
else:
if value:
return '1'
else:
return '0'
registerConverter(bool, BoolConverter)
def FloatConverter(value, db):
return repr(value)
registerConverter(float, FloatConverter)
if DateTimeType:
def DateTimeConverter(value, db):
return "'%s'" % isoStr(value)
registerConverter(DateTimeType, DateTimeConverter)
def TimeConverter(value, db):
return "'%s'" % value.strftime("%T")
registerConverter(DateTimeDeltaType, TimeConverter)
def NoneConverter(value, db):
return "NULL"
registerConverter(NoneType, NoneConverter)
def SequenceConverter(value, db):
return "(%s)" % ", ".join([sqlrepr(v, db) for v in value])
registerConverter(tuple, SequenceConverter)
registerConverter(list, SequenceConverter)
registerConverter(dict, SequenceConverter)
registerConverter(set, SequenceConverter)
registerConverter(frozenset, SequenceConverter)
if sys.version_info[:3] < (2, 6, 0): # Module sets was deprecated in Python 2.6
from sets import Set,ImmutableSet
registerConverter(Set, SequenceConverter)
registerConverter(ImmutableSet, SequenceConverter)
if hasattr(time, 'struct_time'):
def StructTimeConverter(value, db):
return time.strftime("'%Y-%m-%d %H:%M:%S'", value)
registerConverter(time.struct_time, StructTimeConverter)
def DateTimeConverter(value, db):
return "'%04d-%02d-%02d %02d:%02d:%02d'" % (
value.year, value.month, value.day,
value.hour, value.minute, value.second)
registerConverter(datetime.datetime, DateTimeConverter)
def DateConverter(value, db):
return "'%04d-%02d-%02d'" % (value.year, value.month, value.day)
registerConverter(datetime.date, DateConverter)
def TimeConverter(value, db):
return "'%02d:%02d:%02d'" % (value.hour, value.minute, value.second)
registerConverter(datetime.time, TimeConverter)
def DecimalConverter(value, db):
# See http://mail.python.org/pipermail/python-dev/2008-March/078189.html
return str(value.to_eng_string()) # Convert to str to work around a bug in Python 2.5.2
registerConverter(Decimal, DecimalConverter)
def sqlrepr(obj, db=None):
try:
reprFunc = obj.__sqlrepr__
except AttributeError:
converter = lookupConverter(obj)
if converter is None:
raise ValueError, "Unknown SQL builtin type: %s for %s" % \
(type(obj), repr(obj))
return converter(obj, db)
else:
return reprFunc(db)
|