summaryrefslogtreecommitdiff
path: root/lib/sqlalchemy/event.py
diff options
context:
space:
mode:
authorMike Bayer <mike_mp@zzzcomputing.com>2012-06-20 18:55:13 -0400
committerMike Bayer <mike_mp@zzzcomputing.com>2012-06-20 18:55:13 -0400
commit40098941007ff3aa1593e834915c4042c1668dc2 (patch)
treed40068f9a997720d96a2c517717e0267953c75e6 /lib/sqlalchemy/event.py
parente15fc03a16a322298e27c416dce3f3fe869bdf1a (diff)
downloadsqlalchemy-40098941007ff3aa1593e834915c4042c1668dc2.tar.gz
- [feature] Dramatic improvement in memory
usage of the event system; instance-level collections are no longer created for a particular type of event until instance-level listeners are established for that event. [ticket:2516] Also in 0.7.9.
Diffstat (limited to 'lib/sqlalchemy/event.py')
-rw-r--r--lib/sqlalchemy/event.py106
1 files changed, 95 insertions, 11 deletions
diff --git a/lib/sqlalchemy/event.py b/lib/sqlalchemy/event.py
index cd70b3a7c..dfdda3d44 100644
--- a/lib/sqlalchemy/event.py
+++ b/lib/sqlalchemy/event.py
@@ -120,7 +120,8 @@ class _Dispatch(object):
object."""
for ls in _event_descriptors(other):
- getattr(self, ls.name)._update(ls, only_propagate=only_propagate)
+ getattr(self, ls.name).\
+ for_modify(self)._update(ls, only_propagate=only_propagate)
def _event_descriptors(target):
return [getattr(target, k) for k in dir(target) if _is_event_name(k)]
@@ -180,9 +181,11 @@ class Events(object):
@classmethod
def _listen(cls, target, identifier, fn, propagate=False, insert=False):
if insert:
- getattr(target.dispatch, identifier).insert(fn, target, propagate)
+ getattr(target.dispatch, identifier).\
+ for_modify(target.dispatch).insert(fn, target, propagate)
else:
- getattr(target.dispatch, identifier).append(fn, target, propagate)
+ getattr(target.dispatch, identifier).\
+ for_modify(target.dispatch).append(fn, target, propagate)
@classmethod
def _remove(cls, target, identifier, fn):
@@ -201,6 +204,7 @@ class _DispatchDescriptor(object):
self.__name__ = fn.__name__
self.__doc__ = fn.__doc__
self._clslevel = util.defaultdict(list)
+ self._empty_listeners = {}
def insert(self, obj, target, propagate):
assert isinstance(target, type), \
@@ -250,18 +254,91 @@ class _DispatchDescriptor(object):
for dispatcher in self._clslevel.values():
dispatcher[:] = []
+ def for_modify(self, obj):
+ """Return an event collection which can be modified.
+
+ For _DispatchDescriptor at the class level of
+ a dispatcher, this returns self.
+
+ """
+ return self
+
def __get__(self, obj, cls):
if obj is None:
return self
- obj.__dict__[self.__name__] = result = \
- _ListenerCollection(self, obj._parent_cls)
+ elif obj._parent_cls in self._empty_listeners:
+ ret = self._empty_listeners[obj._parent_cls]
+ else:
+ self._empty_listeners[obj._parent_cls] = ret = \
+ _EmptyListener(self, obj._parent_cls)
+ # assigning it to __dict__ means
+ # memoized for fast re-access. but more memory.
+ obj.__dict__[self.__name__] = ret
+ return ret
+
+class _EmptyListener(object):
+ """Serves as a class-level interface to the events
+ served by a _DispatchDescriptor, when there are no
+ instance-level events present.
+
+ Is replaced by _ListenerCollection when instance-level
+ events are added.
+
+ """
+ def __init__(self, parent, target_cls):
+ if target_cls not in parent._clslevel:
+ parent.update_subclass(target_cls)
+ self.parent = parent
+ self.parent_listeners = parent._clslevel[target_cls]
+ self.name = parent.__name__
+ self.propagate = frozenset()
+ self.listeners = ()
+
+ def for_modify(self, obj):
+ """Return an event collection which can be modified.
+
+ For _EmptyListener at the instance level of
+ a dispatcher, this generates a new
+ _ListenerCollection, applies it to the instance,
+ and returns it.
+
+ """
+ obj.__dict__[self.name] = result = _ListenerCollection(
+ self.parent, obj._parent_cls)
return result
+ def _needs_modify(self, *args, **kw):
+ raise NotImplementedError("need to call for_modify()")
+
+ exec_once = insert = append = remove = clear = _needs_modify
+
+ def __call__(self, *args, **kw):
+ """Execute this event."""
+
+ for fn in self.parent_listeners:
+ fn(*args, **kw)
+
+ def __len__(self):
+ return len(self.parent_listeners)
+
+ def __iter__(self):
+ return iter(self.parent_listeners)
+
+ def __getitem__(self, index):
+ return (self.parent_listeners)[index]
+
+ def __nonzero__(self):
+ return bool(self.listeners)
+
+
class _ListenerCollection(object):
"""Instance-level attributes on instances of :class:`._Dispatch`.
Represents a collection of listeners.
+ As of 0.7.9, _ListenerCollection is only first
+ created via the _EmptyListener.for_modify() method.
+
"""
_exec_once = False
@@ -274,6 +351,15 @@ class _ListenerCollection(object):
self.listeners = []
self.propagate = set()
+ def for_modify(self, obj):
+ """Return an event collection which can be modified.
+
+ For _ListenerCollection at the instance level of
+ a dispatcher, this returns self.
+
+ """
+ return self
+
def exec_once(self, *args, **kw):
"""Execute this event, but only if it has not been
executed already for this collection."""
@@ -293,12 +379,10 @@ class _ListenerCollection(object):
# I'm not entirely thrilled about the overhead here,
# but this allows class-level listeners to be added
# at any point.
- #
- # alternatively, _DispatchDescriptor could notify
- # all _ListenerCollection objects, but then we move
- # to a higher memory model, i.e.weakrefs to all _ListenerCollection
- # objects, the _DispatchDescriptor collection repeated
- # for all instances.
+ #
+ # In the absense of instance-level listeners,
+ # we stay with the _EmptyListener object when called
+ # at the instance level.
def __len__(self):
return len(self.parent_listeners + self.listeners)