# copyright 2003-2010 LOGILAB S.A. (Paris, FRANCE), all rights reserved.# contact http://www.logilab.fr/ -- mailto:contact@logilab.fr## This file is part of CubicWeb.## CubicWeb 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.## CubicWeb 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 CubicWeb. If not, see <http://www.gnu.org/licenses/>."""Generalities------------Paraphrasing the `emacs`_ documentation, let us say that hooks are an importantmechanism for customizing an application. A hook is basically a list offunctions to be called on some well-defined occasion (this is called `runningthe hook`)... _`emacs`: http://www.gnu.org/software/emacs/manual/html_node/emacs/Hooks.htmlHooks~~~~~In |cubicweb|, hooks are subclasses of the :class:`~cubicweb.server.hook.Hook`class. They are selected over a set of pre-defined `events` (and possibly moreconditions, hooks being selectable appobjects like views and components). Theyshould implement a :meth:`~cubicweb.server.hook.Hook.__call__` method that willbe called when the hook is triggered.There are two families of events: data events (before / after any individualupdate of an entity / or a relation in the repository) and server events (suchas server startup or shutdown). In a typical application, most of the hooks aredefined over data events.Also, some :class:`~cubicweb.server.hook.Operation` may be registered by hooks,which will be fired when the transaction is commited or rollbacked.The purpose of data event hooks is usually to complement the data model asdefined in the schema, which is static by nature and only provide a restrictedbuiltin set of dynamic constraints, with dynamic or value driven behaviours.For instance they can serve the following purposes:* enforcing constraints that the static schema cannot express (spanning several entities/relations, exotic value ranges and cardinalities, etc.)* implement computed attributesIt is functionally equivalent to a `database trigger`_, except that databasetriggers definition languages are not standardized, hence not portable (forinstance, PL/SQL works with Oracle and PostgreSQL but not SqlServer nor Sqlite)... _`database trigger`: http://en.wikipedia.org/wiki/Database_trigger.. hint:: It is a good practice to write unit tests for each hook. See an example in :ref:`hook_test`Operations~~~~~~~~~~Operations are subclasses of the :class:`~cubicweb.server.hook.Operation` classthat may be created by hooks and scheduled to happen just before (or after) the`precommit`, `postcommit` or `rollback` event. Hooks are being fired immediatelyon data operations, and it is sometime necessary to delay the actual work downto a time where all other hooks have run. Also while the order of execution ofhooks is data dependant (and thus hard to predict), it is possible to force anorder on operations.Operations may be used to:* implements a validation check which needs that all relations be already set on an entity* process various side effects associated with a transaction such as filesystem udpates, mail notifications, etc.Events------Hooks are mostly defined and used to handle `dataflow`_ operations. Itmeans as data gets in (entities added, updated, relations set orunset), specific events are issued and the Hooks matching these eventsare called.You can get the event that triggered a hook by accessing its :attr:eventattribute... _`dataflow`: http://en.wikipedia.org/wiki/DataflowEntity modification related events~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~When called for one of these events, hook will have an `entity` attributecontaining the entity instance.* 'before_add_entity', 'before_update_entity': - on those events, you can check what attributes of the entity are modified in `entity.cw_edited` (by definition the database is not yet updated in a before event) - you are allowed to further modify the entity before database operations, using the dictionary notation. By doing this, you'll avoid the need for a whole new rql query processing, the only difference is that the underlying backend query (eg usually sql) will contains the additional data. For example: .. sourcecode:: python self.entity.set_attributes(age=42) will set the `age` attribute of the entity to 42. But to do so, it will generate a rql query that will have to be processed, then trigger some hooks, and so one (potentially leading to infinite hook loops or such awkward situations..) You can avoid this by doing the modification that way: .. sourcecode:: python self.entity.cw_edited['age'] = 42 Here the attribute will simply be edited in the same query that the one that triggered the hook. Similarly, removing an attribute from `cw_edited` will cancel its modification. - on 'before_update_entity' event, you can access to old and new values in this hook, by using `entity.cw_edited.oldnewvalue(attr)`* 'after_add_entity', 'after_update_entity' - on those events, you can still check what attributes of the entity are modified in `entity.cw_edited` but you can't get anymore the old value, nor modify it.* 'before_delete_entity', 'after_delete_entity' - on those events, the entity has no `cw_edited` set.Relation modification related events~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~When called for one of these events, hook will have `eidfrom`, `rtype`, `eidto`attributes containing respectivly the eid of the subject entity, the relationtype and the eid of the object entity.* 'before_add_relation', 'before_delete_relation' - on those events, you can still get original relation by issuing a rql query* 'after_add_relation', 'after_delete_relation'This is an occasion to remind us that relations support the add / deleteoperation, but no update.Non data events~~~~~~~~~~~~~~~Hooks called on server start/maintenance/stop event (eg 'server_startup','server_maintenance', 'server_shutdown') have a `repo` attribute, but *their`_cw` attribute is None*. The `server_startup` is called on regular startup,while `server_maintenance` is called on cubicweb-ctl upgrade or shellcommands. `server_shutdown` is called anyway.Hooks called on backup/restore event (eg 'server_backup', 'server_restore') havea `repo` and a `timestamp` attributes, but *their `_cw` attribute is None*.Hooks called on session event (eg 'session_open', 'session_close') have nospecial attribute.API---Hooks control~~~~~~~~~~~~~It is sometimes convenient to explicitly enable or disable some hooks. Forinstance if you want to disable some integrity checking hook. This can becontrolled more finely through the `category` class attribute, which is a stringgiving a category name. One can then uses the:class:`~cubicweb.server.session.hooks_control` context manager to explicitlyenable or disable some categories... autoclass:: cubicweb.server.session.hooks_controlThe existing categories are:* ``security``, security checking hooks* ``worfklow``, workflow handling hooks* ``metadata``, hooks setting meta-data on newly created entities* ``notification``, email notification hooks* ``integrity``, data integrity checking hooks* ``activeintegrity``, data integrity consistency hooks, that you should *never* want to disable* ``syncsession``, hooks synchronizing existing sessions* ``syncschema``, hooks synchronizing instance schema (including the physical database)* ``email``, email address handling hooks* ``bookmark``, bookmark entities handling hooksNothing precludes one to invent new categories and use the:class:`~cubicweb.server.session.hooks_control` context manager to filter themin or out.Hooks specific selector~~~~~~~~~~~~~~~~~~~~~~~.. autoclass:: cubicweb.server.hook.match_rtype.. autoclass:: cubicweb.server.hook.match_rtype_setsHooks and operations classes~~~~~~~~~~~~~~~~~~~~~~~~~~~~.. autoclass:: cubicweb.server.hook.Hook.. autoclass:: cubicweb.server.hook.Operation.. autoclass:: cubicweb.server.hook.DataOperation.. autoclass:: cubicweb.server.hook.LateOperation"""from__future__importwith_statement__docformat__="restructuredtext en"fromwarningsimportwarnfromloggingimportgetLoggerfromitertoolsimportchainfromlogilab.common.decoratorsimportclasspropertyfromlogilab.common.deprecationimportdeprecated,class_renamedfromlogilab.common.logging_extimportset_log_methodsfromcubicwebimportRegistryNotFoundfromcubicweb.vregistryimportclassidfromcubicweb.cwvregimportCWRegistry,VRegistryfromcubicweb.selectorsimport(objectify_selector,lltrace,ExpectedValueSelector,is_instance)fromcubicweb.appobjectimportAppObjectfromcubicweb.server.sessionimportsecurity_enabledENTITIES_HOOKS=set(('before_add_entity','after_add_entity','before_update_entity','after_update_entity','before_delete_entity','after_delete_entity'))RELATIONS_HOOKS=set(('before_add_relation','after_add_relation','before_delete_relation','after_delete_relation'))SYSTEM_HOOKS=set(('server_backup','server_restore','server_startup','server_maintenance','server_shutdown','session_open','session_close'))ALL_HOOKS=ENTITIES_HOOKS|RELATIONS_HOOKS|SYSTEM_HOOKSclassHooksRegistry(CWRegistry):definitialization_completed(self):forappobjectsinself.values():forclsinappobjects:ifnotcls.enabled:warn('[3.6] %s: enabled is deprecated'%classid(cls))self.unregister(cls)defregister(self,obj,**kwargs):obj.check_events()super(HooksRegistry,self).register(obj,**kwargs)defcall_hooks(self,event,session=None,**kwargs):kwargs['event']=eventifsessionisNone:forhookinsorted(self.possible_objects(session,**kwargs),key=lambdax:x.order):hook()else:# by default, hooks are executed with security turned offwithsecurity_enabled(session,read=False):hooks=sorted(self.possible_objects(session,**kwargs),key=lambdax:x.order)withsecurity_enabled(session,write=False):forhookinhooks:hook()classHooksManager(object):def__init__(self,vreg):self.vreg=vregdefcall_hooks(self,event,session=None,**kwargs):try:self.vreg['%s_hooks'%event].call_hooks(event,session,**kwargs)exceptRegistryNotFound:pass# no hooks for this eventforeventinALL_HOOKS:VRegistry.REGISTRY_FACTORY['%s_hooks'%event]=HooksRegistry@deprecated('[3.10] use entity.cw_edited.oldnewvalue(attr)')defentity_oldnewvalue(entity,attr):returnentity.cw_edited.oldnewvalue(attr)# some hook specific selectors #################################################@objectify_selector@lltracedefenabled_category(cls,req,**kwargs):ifreqisNone:returnTrue# XXX how to deactivate server startup / shutdown eventreturnreq.is_hook_activated(cls)@objectify_selector@lltracedeffrom_dbapi_query(cls,req,**kwargs):ifreq.running_dbapi_query:return1return0classrechain(object):def__init__(self,*iterators):self.iterators=iteratorsdef__iter__(self):returniter(chain(*self.iterators))classmatch_rtype(ExpectedValueSelector):"""accept if parameters specified as initializer arguments are specified in named arguments given to the selector :param *expected: parameters (eg `basestring`) which are expected to be found in named arguments (kwargs) """def__init__(self,*expected,**more):self.expected=expectedself.frometypes=more.pop('frometypes',None)self.toetypes=more.pop('toetypes',None)@lltracedef__call__(self,cls,req,*args,**kwargs):ifkwargs.get('rtype')notinself.expected:return0ifself.frometypesisnotNoneand \req.describe(kwargs['eidfrom'])[0]notinself.frometypes:return0ifself.toetypesisnotNoneand \req.describe(kwargs['eidto'])[0]notinself.toetypes:return0return1classmatch_rtype_sets(ExpectedValueSelector):"""accept if parameters specified as initializer arguments are specified in named arguments given to the selector """def__init__(self,*expected):self.expected=expected@lltracedef__call__(self,cls,req,*args,**kwargs):forrel_setinself.expected:ifkwargs.get('rtype')inrel_set:return1return0# base class for hook ##########################################################classHook(AppObject):"""Base class for hook. Hooks being appobjects like views, they have a `__regid__` and a `__select__` class attribute. Like all appobjects, hooks have the `self._cw` attribute which represents the current session. In entity hooks, a `self.entity` attribute is also present. The `events` tuple is used by the base class selector to dispatch the hook on the right events. It is possible to dispatch on multiple events at once if needed (though take care as hook attribute may vary as described above). .. Note:: Do not forget to extend the base class selectors as in :: .. sourcecode:: python class MyHook(Hook): __regid__ = 'whatever' __select__ = Hook.__select__ & is_instance('Person') else your hooks will be called madly, whatever the event. """__select__=enabled_category()# set this in derivated classesevents=Nonecategory=Noneorder=0# XXX deprecatedenabled=True@classmethoddefcheck_events(cls):try:foreventincls.events:ifeventnotinALL_HOOKS:raiseException('bad event %s on %s.%s'%(event,cls.__module__,cls.__name__))exceptAttributeError:raiseexceptTypeError:raiseException('bad .events attribute %s on %s.%s'%(cls.events,cls.__module__,cls.__name__))@classpropertydef__registries__(cls):cls.check_events()return['%s_hooks'%evforevincls.events]@classpropertydef__regid__(cls):warn('[3.6] %s: please specify an id for your hook'%classid(cls),DeprecationWarning)returnstr(id(cls))@classmethoddef__registered__(cls,reg):super(Hook,cls).__registered__(reg)ifgetattr(cls,'accepts',None):warn('[3.6] %s: accepts is deprecated, define proper __select__'%classid(cls),DeprecationWarning)rtypes=[]forertypeincls.accepts:ifertype.islower():rtypes.append(ertype)else:cls.__select__=cls.__select__&is_instance(ertype)ifrtypes:cls.__select__=cls.__select__&match_rtype(*rtypes)returnclsknown_args=set(('entity','rtype','eidfrom','eidto','repo','timestamp'))def__init__(self,req,event,**kwargs):forarginself.known_args:ifarginkwargs:setattr(self,arg,kwargs.pop(arg))super(Hook,self).__init__(req,**kwargs)self.event=eventdef__call__(self):ifhasattr(self,'call'):warn('[3.6] %s: call is deprecated, implement __call__'%classid(self.__class__),DeprecationWarning)ifself.event.endswith('_relation'):self.call(self._cw,self.eidfrom,self.rtype,self.eidto)elif'delete'inself.event:self.call(self._cw,self.entity.eid)elifself.event.startswith('server_'):self.call(self.repo)elifself.event.startswith('session_'):self.call(self._cw)else:self.call(self._cw,self.entity)set_log_methods(Hook,getLogger('cubicweb.hook'))# abtract hooks for relation propagation ######################################## See example usage in hooks of the nosylist cubeclassPropagateRelationHook(Hook):"""propagate some `main_rtype` relation on entities linked as object of `subject_relations` or as subject of `object_relations` (the watched relations). This hook ensure that when one of the watched relation is added, the `main_rtype` relation is added to the target entity of the relation. Notice there are no default behaviour defined when a watched relation is deleted, you'll have to handle this by yourself. You usually want to use the :class:`match_rtype_sets` selector on concret classes. """events=('after_add_relation',)# to set in concrete classmain_rtype=Nonesubject_relations=Noneobject_relations=Nonedef__call__(self):assertself.main_rtypeforeidin(self.eidfrom,self.eidto):etype=self._cw.describe(eid)[0]ifself.main_rtypenotinself._cw.vreg.schema.eschema(etype).subjrels:returnifself.rtypeinself.subject_relations:meid,seid=self.eidfrom,self.eidtoelse:assertself.rtypeinself.object_relationsmeid,seid=self.eidto,self.eidfromself._cw.execute('SET E %s P WHERE X %s P, X eid %%(x)s, E eid %%(e)s, NOT E %s P'%(self.main_rtype,self.main_rtype,self.main_rtype),{'x':meid,'e':seid})classPropagateRelationAddHook(Hook):"""Propagate to entities at the end of watched relations when a `main_rtype` relation is added. `subject_relations` and `object_relations` attributes should be specified on subclasses and are usually shared references with attributes of the same name on :class:`PropagateRelationHook`. Because of those shared references, you can use `skip_subject_relations` and `skip_object_relations` attributes when you don't want to propagate to entities linked through some particular relations. """events=('after_add_relation',)# to set in concrete class (mandatory)subject_relations=Noneobject_relations=None# to set in concrete class (optionaly)skip_subject_relations=()skip_object_relations=()def__call__(self):eschema=self._cw.vreg.schema.eschema(self._cw.describe(self.eidfrom)[0])execute=self._cw.executeforrelinself.subject_relations:ifrelineschema.subjrelsandnotrelinself.skip_subject_relations:execute('SET R %s P WHERE X eid %%(x)s, P eid %%(p)s, ''X %s R, NOT R %s P'%(self.rtype,rel,self.rtype),{'x':self.eidfrom,'p':self.eidto})forrelinself.object_relations:ifrelineschema.objrelsandnotrelinself.skip_object_relations:execute('SET R %s P WHERE X eid %%(x)s, P eid %%(p)s, ''R %s X, NOT R %s P'%(self.rtype,rel,self.rtype),{'x':self.eidfrom,'p':self.eidto})classPropagateRelationDelHook(PropagateRelationAddHook):"""Propagate to entities at the end of watched relations when a `main_rtype` relation is deleted. This is the opposite of the :class:`PropagateRelationAddHook`, see its documentation for how to use this class. """events=('after_delete_relation',)def__call__(self):eschema=self._cw.vreg.schema.eschema(self._cw.describe(self.eidfrom)[0])execute=self._cw.executeforrelinself.subject_relations:ifrelineschema.subjrelsandnotrelinself.skip_subject_relations:execute('DELETE R %s P WHERE X eid %%(x)s, P eid %%(p)s, ''X %s R'%(self.rtype,rel),{'x':self.eidfrom,'p':self.eidto})forrelinself.object_relations:ifrelineschema.objrelsandnotrelinself.skip_object_relations:execute('DELETE R %s P WHERE X eid %%(x)s, P eid %%(p)s, ''R %s X'%(self.rtype,rel),{'x':self.eidfrom,'p':self.eidto})PropagateSubjectRelationHook=class_renamed('PropagateSubjectRelationHook',PropagateRelationHook,'[3.9] PropagateSubjectRelationHook has been renamed to PropagateRelationHook')PropagateSubjectRelationAddHook=class_renamed('PropagateSubjectRelationAddHook',PropagateRelationAddHook,'[3.9] PropagateSubjectRelationAddHook has been renamed to PropagateRelationAddHook')PropagateSubjectRelationDelHook=class_renamed('PropagateSubjectRelationDelHook',PropagateRelationDelHook,'[3.9] PropagateSubjectRelationDelHook has been renamed to PropagateRelationDelHook')# abstract classes for operation ###############################################classOperation(object):"""Base class for operations. Operation may be instantiated in the hooks' `__call__` method. It always takes a session object as first argument (accessible as `.session` from the operation instance), and optionally all keyword arguments needed by the operation. These keyword arguments will be accessible as attributes from the operation instance. An operation is triggered on connections pool events related to commit / rollback transations. Possible events are: * 'precommit': the transaction is being prepared for commit. You can freely do any heavy computation, raise an exception if the commit can't go. or even add some new operations during this phase. If you do anything which has to be reverted if the commit fails afterwards (eg altering the file system for instance), you'll have to support the 'revertprecommit' event to revert things by yourself * 'revertprecommit': if an operation failed while being pre-commited, this event is triggered for all operations which had their 'precommit' event already fired to let them revert things (including the operation which made the commit fail) * 'rollback': the transaction has been either rollbacked either: * intentionaly * a 'precommit' event failed, in which case all operations are rollbacked once 'revertprecommit'' has been called * 'postcommit': the transaction is over. All the ORM entities accessed by the earlier transaction are invalid. If you need to work on the database, you need to start a new transaction, for instance using a new internal session, which you will need to commit (and close!). For an operation to support an event, one has to implement the `<event name>_event` method with no arguments. Notice order of operations may be important, and is controlled according to the insert_index's method output (whose implementation vary according to the base hook class used). """def__init__(self,session,**kwargs):self.session=sessionself.__dict__.update(kwargs)self.register(session)# execution informationself.processed=None# 'precommit', 'commit'self.failed=Falsedefregister(self,session):session.add_operation(self,self.insert_index())definsert_index(self):"""return the index of the lastest instance which is not a LateOperation instance """# faster by inspecting operation in reverse order for heavy transactionsi=Nonefori,opinenumerate(reversed(self.session.pending_operations)):ifisinstance(op,(LateOperation,SingleLastOperation)):continuereturn-iorNoneifiisNone:returnNonereturn-(i+1)defhandle_event(self,event):"""delegate event handling to the opertaion"""ifevent=='postcommit_event'andhasattr(self,'commit_event'):warn('[3.10] %s: commit_event method has been replaced by postcommit_event'%classid(self.__class__),DeprecationWarning)self.commit_event()getattr(self,event)()defprecommit_event(self):"""the observed connections pool is preparing a commit"""defrevertprecommit_event(self):"""an error went when pre-commiting this operation or a later one should revert pre-commit's changes but take care, they may have not been all considered if it's this operation which failed """defrollback_event(self):"""the observed connections pool has been rollbacked do nothing by default, the operation will just be removed from the pool operation list """defpostcommit_event(self):"""the observed connections pool has committed"""@property@deprecated('[3.6] use self.session.user')defuser(self):returnself.session.user@property@deprecated('[3.6] use self.session.repo')defrepo(self):returnself.session.repo@property@deprecated('[3.6] use self.session.vreg.schema')defschema(self):returnself.session.repo.schema@property@deprecated('[3.6] use self.session.vreg.config')defconfig(self):returnself.session.repo.configset_log_methods(Operation,getLogger('cubicweb.session'))def_container_add(container,value):{set:set.add,list:list.append}[container.__class__](container,value)classDataOperationMixIn(object):"""Mix-in class to ease applying a single operation on a set of data, avoiding to create as many as operation as they are individual modification. The body of the operation must then iterate over the values that have been stored in a single operation instance. You should try to use this instead of creating on operation for each `value`, since handling operations becomes costly on massive data import. Usage looks like: .. sourcecode:: python class MyEntityHook(Hook): __regid__ = 'my.entity.hook' __select__ = Hook.__select__ & is_instance('MyEntity') events = ('after_add_entity',) def __call__(self): MyOperation.get_instance(self._cw).add_data(self.entity) class MyOperation(DataOperation, DataOperationMixIn): def precommit_event(self): for bucket in self.get_data(): process(bucket) You can modify the `containercls` class attribute, which defines the container class that should be instantiated to hold payloads. An instance is created on instantiation, and then the :meth:`add_data` method will add the given data to the existing container. Default to a `set`. Give `list` if you want to keep arrival ordering. You can also use another kind of container by redefining :meth:`_build_container` and :meth:`add_data` More optional parameters can be given to the `get_instance` operation, that will be given to the operation constructer (though those parameters should not vary accross different calls to this method for a same operation for obvious reason). .. Note:: For sanity reason `get_data` will reset the operation, so that once the operation has started its treatment, if some hook want to push additional data to this same operation, a new instance will be created (else that data has a great chance to be never treated). This implies: * you should **always** call `get_data` when starting treatment * you should **never** call `get_data` for another reason. """containercls=set@classpropertydefdata_key(cls):return('cw.dataops',cls.__name__)@classmethoddefget_instance(cls,session,**kwargs):# no need to lock: transaction_data already comes from thread's local storagetry:returnsession.transaction_data[cls.data_key]exceptKeyError:op=session.transaction_data[cls.data_key]=cls(session,**kwargs)returnopdef__init__(self,*args,**kwargs):super(DataOperationMixIn,self).__init__(*args,**kwargs)self._container=self._build_container()self._processed=Falsedef__contains__(self,value):returnvalueinself._containerdef_build_container(self):returnself.containercls()defadd_data(self,data):assertnotself._processed,"""Trying to add data to a closed operation.Iterating over operation data closed it and should be reserved to precommit /postcommit method of the operation."""_container_add(self._container,data)defget_data(self):assertnotself._processed,"""Trying to get data from a closed operation.Iterating over operation data closed it and should be reserved to precommit /postcommit method of the operation."""self._processed=Trueop=self.session.transaction_data.pop(self.data_key)assertopisself,"Bad handling of operation data, found %s instead of %s for key %s"%(op,self,self.data_key)returnself._container@deprecated('[3.10] use opcls.get_instance(session, **opkwargs).add_data(value)')defset_operation(session,datakey,value,opcls,containercls=set,**opkwargs):"""Function to ease applying a single operation on a set of data, avoiding to create as many as operation as they are individual modification. You should try to use this instead of creating on operation for each `value`, since handling operations becomes coslty on massive data import. Arguments are: * the `session` object * `datakey`, a specially forged key that will be used as key in session.transaction_data * `value` that is the actual payload of an individual operation * `opcls`, the class of the operation. An instance is created on the first call for the given key, and then subsequent calls will simply add the payload to the container (hence `opkwargs` is only used on that first call) * `containercls`, the container class that should be instantiated to hold payloads. An instance is created on the first call for the given key, and then subsequent calls will add the data to the existing container. Default to a set. Give `list` if you want to keep arrival ordering. * more optional parameters to give to the operation (here the rtype which do not vary accross operations). The body of the operation must then iterate over the values that have been mapped in the transaction_data dictionary to the forged key, e.g.: .. sourcecode:: python for value in self._cw.transaction_data.pop(datakey): ... .. Note:: **poping** the key from `transaction_data` is not an option, else you may get unexpected data loss in some case of nested hooks. """try:# Search for session.transaction_data[`datakey`] (expected to be a set):# if found, simply append `value`_container_add(session.transaction_data[datakey],value)exceptKeyError:# else, initialize it to containercls([`value`]) and instantiate the given# `opcls` operation class with additional keyword argumentsopcls(session,**opkwargs)session.transaction_data[datakey]=containercls()_container_add(session.transaction_data[datakey],value)classLateOperation(Operation):"""special operation which should be called after all possible (ie non late) operations """definsert_index(self):"""return the index of the lastest instance which is not a SingleLastOperation instance """# faster by inspecting operation in reverse order for heavy transactionsi=Nonefori,opinenumerate(reversed(self.session.pending_operations)):ifisinstance(op,SingleLastOperation):continuereturn-iorNoneifiisNone:returnNonereturn-(i+1)classSingleLastOperation(Operation):"""special operation which should be called once and after all other operations """defregister(self,session):"""override register to handle cases where this operation has already been added """operations=session.pending_operationsindex=self.equivalent_index(operations)ifindexisnotNone:equivalent=operations.pop(index)else:equivalent=Nonesession.add_operation(self,self.insert_index())returnequivalentdefequivalent_index(self,operations):"""return the index of the equivalent operation if any"""fori,opinenumerate(reversed(operations)):ifop.__class__isself.__class__:return-(i+1)returnNonedefinsert_index(self):returnNoneclassSendMailOp(SingleLastOperation):def__init__(self,session,msg=None,recipients=None,**kwargs):# may not specify msg yet, as# `cubicweb.sobjects.supervision.SupervisionMailOp`ifmsgisnotNone:assertrecipientsself.to_send=[(msg,recipients)]else:assertrecipientsisNoneself.to_send=[]super(SendMailOp,self).__init__(session,**kwargs)defregister(self,session):previous=super(SendMailOp,self).register(session)ifprevious:self.to_send=previous.to_send+self.to_senddefpostcommit_event(self):self.session.repo.threaded_task(self.sendmails)defsendmails(self):self.session.vreg.config.sendmails(self.to_send)classRQLPrecommitOperation(Operation):defprecommit_event(self):execute=self.session.executeforrqlinself.rqls:execute(*rql)classCleanupNewEidsCacheOp(DataOperationMixIn,SingleLastOperation):"""on rollback of a insert query we have to remove from repository's type/source cache eids of entities added in that transaction. NOTE: querier's rqlst/solutions cache may have been polluted too with queries such as Any X WHERE X eid 32 if 32 has been rollbacked however generated queries are unpredictable and analysing all the cache probably too expensive. Notice that there is no pb when using args to specify eids instead of giving them into the rql string. """data_key='neweids'defrollback_event(self):"""the observed connections pool has been rollbacked, remove inserted eid from repository type/source cache """try:self.session.repo.clear_caches(self.get_data())exceptKeyError:passclassCleanupDeletedEidsCacheOp(DataOperationMixIn,SingleLastOperation):"""on commit of delete query, we have to remove from repository's type/source cache eids of entities deleted in that transaction. """data_key='pendingeids'defpostcommit_event(self):"""the observed connections pool has been rollbacked, remove inserted eid from repository type/source cache """try:self.session.repo.clear_caches(self.get_data())exceptKeyError:pass