# -*- coding: utf-8 -*- # # AWL simulator - object reference manager # # Copyright 2015 Michael Buesch # # This program is free software; you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation; either version 2 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 General Public License for more details. # # You should have received a copy of the GNU General Public License along # with this program; if not, write to the Free Software Foundation, Inc., # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. # from __future__ import division, absolute_import, print_function, unicode_literals from awlsim.common.compat import * from awlsim.common.exceptions import * __all__ = [ "ObjRef", "ObjRefManager", ] class ObjRef(object): """An object reference.""" @classmethod def make(cls, name, managerOrRef, obj = None, inheritRef = False): """Make a new ObjRef instance. name -> A name string (or callable returning a string). managerOrRef -> An ObjRefManager instance or ObjRef instance. obj -> The object that is associated with this ref. inheritRef -> If False, a new ref is created. If True and managerOrRef is an ObjRef, the ref is inherited. """ if managerOrRef is None: return None elif isinstance(managerOrRef, ObjRefManager): manager = managerOrRef return cls(name, manager, obj) elif isinstance(managerOrRef, ObjRef): oldRef = managerOrRef newRef = cls(name, oldRef.__manager, obj) if inheritRef and oldRef.alive: oldRef.destroy() return newRef else: assert(0) def __init__(self, name, manager, obj = None): """Contruct object reference. name: Informational name string or callable returing a string. manager: An ObjRefManager instance. obj: The object that is associated with this ref (optional). """ self.__name = name self.__obj = obj self.__manager = manager self.__manager._addRef(self) def destroy(self): """Destroy (unref) this reference. This removes the reference from the manager. """ if self.alive: self.__manager.refDestroyed(self) self.__name = None self.__obj = None self.__manager = None @property def name(self): """The reference name string. """ if callable(self.__name): return self.__name(self) return self.__name @property def obj(self): """The object that is associated with this ref. """ return self.__obj @property def manager(self): """Get the manager that this ref belongs to. """ return self.__manager @property def alive(self): """True, if this reference is alive. False, if this reference was destroyed. """ return self.__manager is not None def __repr__(self): return str(self.name) class ObjRefManager(object): """Object reference manager. The manager belongs to the object that actually is referenced. """ def __init__(self, name, oneDestroyedCallback = None, allDestroyedCallback = None): """Contruct reference manager. name: Informational name string or callable returing a string. oneDestroyedCallback: Optional callback. Called, if one ref was destroyed. allDestroyedCallback: Optional callback. Called, if all refs were destroyed. """ self.__name = name self.__oneDestroyedCallback = oneDestroyedCallback self.__allDestroyedCallback = allDestroyedCallback self.__refs = set() @property def name(self): """The manager name string. """ if callable(self.__name): return self.__name(self) return self.__name @property def hasReferences(self): """Returns true, if this manager holds references. """ return bool(self.__refs) @property def refs(self): """Get a set of all references to this object. """ return set(self.__refs) def _addRef(self, objRef): self.__refs.add(objRef) def refDestroyed(self, objRef): """Callback: Called if one reference was destroyed. Override this method or set oneDestroyedCallback, if you want to be notified. """ self.__refs.remove(objRef) if self.__oneDestroyedCallback: self.__oneDestroyedCallback(objRef) if not self.__refs: self.allRefsDestroyed() def allRefsDestroyed(self): """Callback: Called if all references were destroyed. Override this method or set allDestroyedCallback, if you want to be notified. """ if self.__allDestroyedCallback: self.__allDestroyedCallback() def __repr__(self): return str(self.name)