Files
sqlalchemy/lib/sqlalchemy/schema.py
T
Mike Bayer a726477344
2005-09-03 03:15:31 +00:00

183 lines
5.8 KiB
Python

# schema.py
# Copyright (C) 2005 Michael Bayer mike_mp@zzzcomputing.com
#
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2.1 of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this library; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
from sqlalchemy.util import *
import copy
__ALL__ = ['Table', 'Column', 'Sequence',
'INT', 'CHAR', 'VARCHAR', 'TEXT', 'FLOAT', 'DECIMAL',
'TIMESTAMP', 'DATETIME', 'CLOB', 'BLOB', 'BOOLEAN'
]
class INT:
"""integer datatype"""
pass
class CHAR:
"""character datatype"""
def __init__(self, length):
self.length = length
class VARCHAR:
def __init__(self, length):
self.length = length
class FLOAT:
def __init__(self, precision, length):
self.precision = precision
self.length = length
class TEXT: pass
class DECIMAL: pass
class TIMESTAMP: pass
class DATETIME: pass
class CLOB: pass
class BLOB: pass
class BOOLEAN: pass
class SchemaItem(object):
"""base class for items that define a database schema."""
def _init_items(self, *args):
for item in args:
item._set_parent(self)
def accept_visitor(self, visitor):
raise NotImplementedError()
def _set_parent(self, parent):
"""a child item attaches itself to its parent via this method."""
raise NotImplementedError()
def hash_key(self):
"""returns a string that identifies this SchemaItem uniquely"""
return repr(self)
def __getattr__(self, key):
"""proxies method calls to an underlying implementation object for methods not found locally"""
return getattr(self._impl, key)
class Table(SchemaItem):
"""represents a relational database table."""
def __init__(self, name, engine, *args, **kwargs):
self.name = name
self.columns = OrderedProperties()
self.c = self.columns
self.relations = []
self.primary_keys = []
self.engine = engine
self._impl = self.engine.tableimpl(self)
self._init_items(*args)
# load column definitions from the database if 'autoload' is defined
if kwargs.get('autoload', False):
self.engine.reflecttable(self)
def append_item(self, item):
self._init_items(item)
def _set_parent(self, schema):
schema.tables[self.name] = self
self.schema = schema
def accept_visitor(self, visitor):
for c in self.columns:
c.accept_visitor(visitor)
return visitor.visit_table(self)
class Column(SchemaItem):
"""represents a column in a database table."""
def __init__(self, name, type, key = None, primary_key = False, *args):
self.name = name
self.type = type
self.sequences = OrderedProperties()
self.key = key or name
self.primary_key = primary_key
self._items = args
self._orig = None
original = property(lambda s: s._orig or s)
def _set_parent(self, table):
table.columns[self.key] = self
if self.primary_key:
table.primary_keys.append(self)
self.table = table
self.engine = table.engine
self._impl = self.engine.columnimpl(self)
self._init_items(*self._items)
def _make_proxy(self, selectable, name = None):
"""creates a copy of this Column for use in a new selectable unit"""
# using copy.copy(c) seems to add a full second to the select.py unittest package
#c = copy.copy(self)
#if name is not None:
# c.name = name
# c.key = name
c = Column(name or self.name, self.type, key = name or self.key, primary_key = self.primary_key)
c.table = selectable
c.engine = self.engine
c._orig = self.original
c._items = self._items
selectable.columns[c.key] = c
c._impl = self.engine.columnimpl(c)
return c
def accept_visitor(self, visitor):
return visitor.visit_column(self)
def __lt__(self, other): return self._impl.__lt__(other)
def __le__(self, other): return self._impl.__le__(other)
def __eq__(self, other): return self._impl.__eq__(other)
def __ne__(self, other): return self._impl.__ne__(other)
def __gt__(self, other): return self._impl.__gt__(other)
def __ge__(self, other): return self._impl.__ge__(other)
def __str__(self): return self._impl.__str__()
class Sequence(SchemaItem):
"""represents a sequence, which applies to Oracle and Postgres databases."""
def set_parent(self, column, key):
column.sequences[key] = self
self.column = column
def accept_visitor(self, visitor):
return visitor.visit_sequence(self)
class SchemaEngine(object):
"""a factory object used to create implementations for schema objects"""
def tableimpl(self, table):
raise NotImplementedError()
def columnimpl(self, column):
raise NotImplementedError()
class SchemaVisitor(object):
"""base class for an object that traverses across Schema objects"""
def visit_schema(self, schema):pass
def visit_table(self, table):pass
def visit_column(self, column):pass
def visit_relation(self, join):pass
def visit_index(self, index):pass
def visit_sequence(self, sequence):pass