author | Alexandre Fayolle <alexandre.fayolle@logilab.fr> |
Fri, 12 Feb 2010 14:34:38 +0100 | |
branch | oldstable |
changeset 4562 | 7e781ff78726 |
parent 4496 | 14cbf2570ce8 |
child 4514 | 5bb1d39ea0da |
permissions | -rw-r--r-- |
0 | 1 |
"""Core hooks: check schema validity, unsure we are not deleting necessary |
2 |
entities... |
|
3 |
||
4 |
:organization: Logilab |
|
4212
ab6573088b4a
update copyright: welcome 2010
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
4119
diff
changeset
|
5 |
:copyright: 2001-2010 LOGILAB S.A. (Paris, FRANCE), license is LGPL v2. |
0 | 6 |
:contact: http://www.logilab.fr/ -- mailto:contact@logilab.fr |
1977
606923dff11b
big bunch of copyright / docstring update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1802
diff
changeset
|
7 |
:license: GNU Lesser General Public License, v2.1 - http://www.gnu.org/licenses |
0 | 8 |
""" |
9 |
__docformat__ = "restructuredtext en" |
|
10 |
||
4489
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
11 |
from threading import Lock |
1016
26387b836099
use datetime instead of mx.DateTime
sylvain.thenault@logilab.fr
parents:
479
diff
changeset
|
12 |
from datetime import datetime |
0 | 13 |
|
14 |
from cubicweb import UnknownProperty, ValidationError, BadConnectionId |
|
3978
2c95e3033f64
finish yesterday work on rql constraints:
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3887
diff
changeset
|
15 |
from cubicweb.schema import RQLConstraint, RQLUniqueConstraint |
0 | 16 |
from cubicweb.server.pool import Operation, LateOperation, PreCommitOperation |
3773
14fde27a70a2
don't allow that even with cow powers
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3693
diff
changeset
|
17 |
from cubicweb.server.hookhelper import (check_internal_entity, |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
18 |
get_user_sessions, rproperty) |
0 | 19 |
from cubicweb.server.repository import FTIndexEntityOp |
20 |
||
2605
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
21 |
# special relations that don't have to be checked for integrity, usually |
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
22 |
# because they are handled internally by hooks (so we trust ourselves) |
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
23 |
DONT_CHECK_RTYPES_ON_ADD = set(('owned_by', 'created_by', |
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
24 |
'is', 'is_instance_of', |
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
25 |
'wf_info_for', 'from_state', 'to_state')) |
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
26 |
DONT_CHECK_RTYPES_ON_DEL = set(('is', 'is_instance_of', |
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
27 |
'wf_info_for', 'from_state', 'to_state')) |
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
28 |
|
4489
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
29 |
_UNIQUE_CONSTRAINTS_LOCK = Lock() |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
30 |
_UNIQUE_CONSTRAINTS_HOLDER = None |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
31 |
|
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
32 |
class _ReleaseUniqueConstraintsHook(Operation): |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
33 |
def commit_event(self): |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
34 |
pass |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
35 |
def postcommit_event(self): |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
36 |
_release_unique_cstr_lock(self.session) |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
37 |
def rollback_event(self): |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
38 |
_release_unique_cstr_lock(self.session) |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
39 |
|
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
40 |
def _acquire_unique_cstr_lock(session): |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
41 |
"""acquire the _UNIQUE_CONSTRAINTS_LOCK for the session. |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
42 |
|
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
43 |
This lock used to avoid potential integrity pb when checking |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
44 |
RQLUniqueConstraint in two different transactions, as explained in |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
45 |
http://intranet.logilab.fr/jpl/ticket/36564 |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
46 |
""" |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
47 |
global _UNIQUE_CONSTRAINTS_HOLDER |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
48 |
asession = session.actual_session() |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
49 |
if _UNIQUE_CONSTRAINTS_HOLDER is asession: |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
50 |
return |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
51 |
_UNIQUE_CONSTRAINTS_LOCK.acquire() |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
52 |
_UNIQUE_CONSTRAINTS_HOLDER = asession |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
53 |
# register operation responsible to release the lock on commit/rollback |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
54 |
_ReleaseUniqueConstraintsHook(asession) |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
55 |
|
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
56 |
def _release_unique_cstr_lock(session): |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
57 |
global _UNIQUE_CONSTRAINTS_HOLDER |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
58 |
if _UNIQUE_CONSTRAINTS_HOLDER is session: |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
59 |
_UNIQUE_CONSTRAINTS_HOLDER = None |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
60 |
_UNIQUE_CONSTRAINTS_LOCK.release() |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
61 |
|
2605
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
62 |
|
0 | 63 |
def relation_deleted(session, eidfrom, rtype, eidto): |
2101
08003e0354a7
update transaction data api
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
1977
diff
changeset
|
64 |
session.transaction_data.setdefault('pendingrelations', []).append( |
08003e0354a7
update transaction data api
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
1977
diff
changeset
|
65 |
(eidfrom, rtype, eidto)) |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
66 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
67 |
def eschema_type_eid(session, etype): |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
68 |
"""get eid of the CWEType entity for the given yams type""" |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
69 |
eschema = session.repo.schema.eschema(etype) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
70 |
# eschema.eid is None if schema has been readen from the filesystem, not |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
71 |
# from the database (eg during tests) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
72 |
if eschema.eid is None: |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
73 |
eschema.eid = session.unsafe_execute( |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
74 |
'Any X WHERE X is CWEType, X name %(name)s', |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
75 |
{'name': str(etype)})[0][0] |
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
76 |
return eschema.eid |
0 | 77 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
78 |
|
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
79 |
# base meta-data handling ###################################################### |
0 | 80 |
|
81 |
def setctime_before_add_entity(session, entity): |
|
82 |
"""before create a new entity -> set creation and modification date |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
83 |
|
0 | 84 |
this is a conveniency hook, you shouldn't have to disable it |
85 |
""" |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
86 |
timestamp = datetime.now() |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
87 |
entity.setdefault('creation_date', timestamp) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
88 |
entity.setdefault('modification_date', timestamp) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
89 |
if not session.get_shared_data('do-not-insert-cwuri'): |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
90 |
entity.setdefault('cwuri', u'%seid/%s' % (session.base_url(), entity.eid)) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
91 |
|
0 | 92 |
|
93 |
def setmtime_before_update_entity(session, entity): |
|
94 |
"""update an entity -> set modification date""" |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
95 |
entity.setdefault('modification_date', datetime.now()) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
96 |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
97 |
|
0 | 98 |
class SetCreatorOp(PreCommitOperation): |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
99 |
|
0 | 100 |
def precommit_event(self): |
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
101 |
session = self.session |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
102 |
if self.entity.eid in session.transaction_data.get('pendingeids', ()): |
0 | 103 |
# entity have been created and deleted in the same transaction |
104 |
return |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
105 |
if not self.entity.created_by: |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
106 |
session.add_relation(self.entity.eid, 'created_by', session.user.eid) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
107 |
|
0 | 108 |
|
109 |
def setowner_after_add_entity(session, entity): |
|
110 |
"""create a new entity -> set owner and creator metadata""" |
|
111 |
asession = session.actual_session() |
|
112 |
if not asession.is_internal_session: |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
113 |
session.add_relation(entity.eid, 'owned_by', asession.user.eid) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
114 |
SetCreatorOp(asession, entity=entity) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
115 |
|
0 | 116 |
|
117 |
def setis_after_add_entity(session, entity): |
|
118 |
"""create a new entity -> set is relation""" |
|
1250
5c20a7f13c84
new recreate argument to extid2eid when an external source want to recreate entities previously imported with a predictable ext id
sylvain.thenault@logilab.fr
parents:
479
diff
changeset
|
119 |
if hasattr(entity, '_cw_recreating'): |
5c20a7f13c84
new recreate argument to extid2eid when an external source want to recreate entities previously imported with a predictable ext id
sylvain.thenault@logilab.fr
parents:
479
diff
changeset
|
120 |
return |
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
121 |
try: |
3693
92581287726f
[optimisation] insert is / is_instance_of using sql
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3689
diff
changeset
|
122 |
#session.add_relation(entity.eid, 'is', |
92581287726f
[optimisation] insert is / is_instance_of using sql
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3689
diff
changeset
|
123 |
# eschema_type_eid(session, entity.id)) |
92581287726f
[optimisation] insert is / is_instance_of using sql
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3689
diff
changeset
|
124 |
session.system_sql('INSERT INTO is_relation(eid_from,eid_to) VALUES (%s,%s)' |
92581287726f
[optimisation] insert is / is_instance_of using sql
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3689
diff
changeset
|
125 |
% (entity.eid, eschema_type_eid(session, entity.id))) |
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
126 |
except IndexError: |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
127 |
# during schema serialization, skip |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
128 |
return |
3693
92581287726f
[optimisation] insert is / is_instance_of using sql
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3689
diff
changeset
|
129 |
for etype in entity.e_schema.ancestors() + [entity.e_schema]: |
92581287726f
[optimisation] insert is / is_instance_of using sql
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3689
diff
changeset
|
130 |
#session.add_relation(entity.eid, 'is_instance_of', |
92581287726f
[optimisation] insert is / is_instance_of using sql
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3689
diff
changeset
|
131 |
# eschema_type_eid(session, etype)) |
92581287726f
[optimisation] insert is / is_instance_of using sql
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3689
diff
changeset
|
132 |
session.system_sql('INSERT INTO is_instance_of_relation(eid_from,eid_to) VALUES (%s,%s)' |
92581287726f
[optimisation] insert is / is_instance_of using sql
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3689
diff
changeset
|
133 |
% (entity.eid, eschema_type_eid(session, etype))) |
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
134 |
|
0 | 135 |
|
136 |
def setowner_after_add_user(session, entity): |
|
137 |
"""when a user has been created, add owned_by relation on itself""" |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
138 |
session.add_relation(entity.eid, 'owned_by', entity.eid) |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
139 |
|
0 | 140 |
|
141 |
def fti_update_after_add_relation(session, eidfrom, rtype, eidto): |
|
142 |
"""sync fulltext index when relevant relation is added. Reindexing the |
|
143 |
contained entity is enough since it will implicitly reindex the container |
|
144 |
entity. |
|
145 |
""" |
|
146 |
ftcontainer = session.repo.schema.rschema(rtype).fulltext_container |
|
147 |
if ftcontainer == 'subject': |
|
2195
58bef4f707ce
update calls to a deprecated method
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2101
diff
changeset
|
148 |
FTIndexEntityOp(session, entity=session.entity_from_eid(eidto)) |
0 | 149 |
elif ftcontainer == 'object': |
2195
58bef4f707ce
update calls to a deprecated method
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2101
diff
changeset
|
150 |
FTIndexEntityOp(session, entity=session.entity_from_eid(eidfrom)) |
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
151 |
|
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
152 |
|
0 | 153 |
def fti_update_after_delete_relation(session, eidfrom, rtype, eidto): |
154 |
"""sync fulltext index when relevant relation is deleted. Reindexing both |
|
155 |
entities is necessary. |
|
156 |
""" |
|
157 |
if session.repo.schema.rschema(rtype).fulltext_container: |
|
2195
58bef4f707ce
update calls to a deprecated method
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2101
diff
changeset
|
158 |
FTIndexEntityOp(session, entity=session.entity_from_eid(eidto)) |
58bef4f707ce
update calls to a deprecated method
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2101
diff
changeset
|
159 |
FTIndexEntityOp(session, entity=session.entity_from_eid(eidfrom)) |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
160 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
161 |
|
0 | 162 |
class SyncOwnersOp(PreCommitOperation): |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
163 |
|
0 | 164 |
def precommit_event(self): |
165 |
self.session.unsafe_execute('SET X owned_by U WHERE C owned_by U, C eid %(c)s,' |
|
166 |
'NOT EXISTS(X owned_by U, X eid %(x)s)', |
|
167 |
{'c': self.compositeeid, 'x': self.composedeid}, |
|
168 |
('c', 'x')) |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
169 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
170 |
|
0 | 171 |
def sync_owner_after_add_composite_relation(session, eidfrom, rtype, eidto): |
172 |
"""when adding composite relation, the composed should have the same owners |
|
173 |
has the composite |
|
174 |
""" |
|
175 |
if rtype == 'wf_info_for': |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
176 |
# skip this special composite relation # XXX (syt) why? |
0 | 177 |
return |
178 |
composite = rproperty(session, rtype, eidfrom, eidto, 'composite') |
|
179 |
if composite == 'subject': |
|
180 |
SyncOwnersOp(session, compositeeid=eidfrom, composedeid=eidto) |
|
181 |
elif composite == 'object': |
|
182 |
SyncOwnersOp(session, compositeeid=eidto, composedeid=eidfrom) |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
183 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
184 |
|
0 | 185 |
def _register_metadata_hooks(hm): |
186 |
"""register meta-data related hooks on the hooks manager""" |
|
187 |
hm.register_hook(setctime_before_add_entity, 'before_add_entity', '') |
|
188 |
hm.register_hook(setmtime_before_update_entity, 'before_update_entity', '') |
|
189 |
hm.register_hook(setowner_after_add_entity, 'after_add_entity', '') |
|
190 |
hm.register_hook(sync_owner_after_add_composite_relation, 'after_add_relation', '') |
|
191 |
hm.register_hook(fti_update_after_add_relation, 'after_add_relation', '') |
|
192 |
hm.register_hook(fti_update_after_delete_relation, 'after_delete_relation', '') |
|
193 |
if 'is' in hm.schema: |
|
194 |
hm.register_hook(setis_after_add_entity, 'after_add_entity', '') |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
195 |
if 'CWUser' in hm.schema: |
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
196 |
hm.register_hook(setowner_after_add_user, 'after_add_entity', 'CWUser') |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
197 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
198 |
|
0 | 199 |
# core hooks ################################################################## |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
200 |
|
0 | 201 |
class DelayedDeleteOp(PreCommitOperation): |
202 |
"""delete the object of composite relation except if the relation |
|
203 |
has actually been redirected to another composite |
|
204 |
""" |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
205 |
|
0 | 206 |
def precommit_event(self): |
207 |
session = self.session |
|
2607
5d889b4928bb
[F hooks] skip new eids as well as pending ones in DelayedDelete operation
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2606
diff
changeset
|
208 |
# don't do anything if the entity is being created or deleted |
5d889b4928bb
[F hooks] skip new eids as well as pending ones in DelayedDelete operation
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2606
diff
changeset
|
209 |
if not (self.eid in session.transaction_data.get('pendingeids', ()) or |
5d889b4928bb
[F hooks] skip new eids as well as pending ones in DelayedDelete operation
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2606
diff
changeset
|
210 |
self.eid in session.transaction_data.get('neweids', ())): |
0 | 211 |
etype = session.describe(self.eid)[0] |
3568
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
212 |
if self.role == 'subject': |
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
213 |
rql = 'DELETE %s X WHERE X eid %%(x)s, NOT X %s Y' |
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
214 |
else: # self.role == 'object': |
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
215 |
rql = 'DELETE %s X WHERE X eid %%(x)s, NOT Y %s X' |
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
216 |
session.unsafe_execute(rql % (etype, self.rtype), {'x': self.eid}, 'x') |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
217 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
218 |
|
0 | 219 |
def handle_composite_before_del_relation(session, eidfrom, rtype, eidto): |
220 |
"""delete the object of composite relation""" |
|
3568
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
221 |
# if the relation is being delete, don't delete composite's components |
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
222 |
# automatically |
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
223 |
pendingrdefs = session.transaction_data.get('pendingrdefs', ()) |
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
224 |
if (session.describe(eidfrom)[0], rtype, session.describe(eidto)[0]) in pendingrdefs: |
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
225 |
return |
0 | 226 |
composite = rproperty(session, rtype, eidfrom, eidto, 'composite') |
227 |
if composite == 'subject': |
|
3568
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
228 |
DelayedDeleteOp(session, eid=eidto, rtype=rtype, role='object') |
0 | 229 |
elif composite == 'object': |
3568
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
230 |
DelayedDeleteOp(session, eid=eidfrom, rtype=rtype, role='subject') |
0 | 231 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
232 |
|
0 | 233 |
def before_del_group(session, eid): |
234 |
"""check that we don't remove the owners group""" |
|
235 |
check_internal_entity(session, eid, ('owners',)) |
|
236 |
||
237 |
||
238 |
# schema validation hooks ##################################################### |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
239 |
|
0 | 240 |
class CheckConstraintsOperation(LateOperation): |
241 |
"""check a new relation satisfy its constraints |
|
242 |
""" |
|
243 |
def precommit_event(self): |
|
244 |
eidfrom, rtype, eidto = self.rdef |
|
245 |
# first check related entities have not been deleted in the same |
|
246 |
# transaction |
|
2101
08003e0354a7
update transaction data api
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
1977
diff
changeset
|
247 |
pending = self.session.transaction_data.get('pendingeids', ()) |
0 | 248 |
if eidfrom in pending: |
249 |
return |
|
250 |
if eidto in pending: |
|
251 |
return |
|
252 |
for constraint in self.constraints: |
|
4489
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
253 |
# XXX |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
254 |
# * lock RQLConstraint as well? |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
255 |
# * use a constraint id to use per constraint lock and avoid |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
256 |
# unnecessary commit serialization ? |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
257 |
if isinstance(constraint, RQLUniqueConstraint): |
63128e8b9af9
fix security issue #36564 (integrity error w/ RQLUniqueConstraint) by using a global lock serializing commit when some RQLUniqueConstraint is encountered.
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
4212
diff
changeset
|
258 |
_acquire_unique_cstr_lock(self.session) |
0 | 259 |
try: |
260 |
constraint.repo_check(self.session, eidfrom, rtype, eidto) |
|
261 |
except NotImplementedError: |
|
262 |
self.critical('can\'t check constraint %s, not supported', |
|
263 |
constraint) |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
264 |
|
0 | 265 |
def commit_event(self): |
266 |
pass |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
267 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
268 |
|
0 | 269 |
def cstrcheck_after_add_relation(session, eidfrom, rtype, eidto): |
270 |
"""check the relation satisfy its constraints |
|
271 |
||
272 |
this is delayed to a precommit time operation since other relation which |
|
273 |
will make constraint satisfied may be added later. |
|
274 |
""" |
|
3685
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
275 |
if session.is_super_session: |
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
276 |
return |
0 | 277 |
constraints = rproperty(session, rtype, eidfrom, eidto, 'constraints') |
278 |
if constraints: |
|
3978
2c95e3033f64
finish yesterday work on rql constraints:
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3887
diff
changeset
|
279 |
# XXX get only RQL[Unique]Constraints? |
0 | 280 |
CheckConstraintsOperation(session, constraints=constraints, |
281 |
rdef=(eidfrom, rtype, eidto)) |
|
282 |
||
283 |
def uniquecstrcheck_before_modification(session, entity): |
|
3685
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
284 |
if session.is_super_session: |
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
285 |
return |
0 | 286 |
eschema = entity.e_schema |
4111
1fda1d356741
use edited_attributes and uniformize attribute schema retreival
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3978
diff
changeset
|
287 |
for attr in entity.edited_attributes: |
4116 | 288 |
val = entity[attr] |
0 | 289 |
if val is None: |
290 |
continue |
|
3689
deb13e88e037
follow yams 0.25 api changes to improve performance
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3685
diff
changeset
|
291 |
if eschema.subjrels[attr].final and \ |
0 | 292 |
eschema.has_unique_values(attr): |
293 |
rql = '%s X WHERE X %s %%(val)s' % (entity.e_schema, attr) |
|
294 |
rset = session.unsafe_execute(rql, {'val': val}) |
|
295 |
if rset and rset[0][0] != entity.eid: |
|
296 |
msg = session._('the value "%s" is already used, use another one') |
|
297 |
raise ValidationError(entity.eid, {attr: msg % val}) |
|
298 |
||
3978
2c95e3033f64
finish yesterday work on rql constraints:
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3887
diff
changeset
|
299 |
|
3554
26e586f3c15c
[schema] make RQL* constraints usable w/ attributes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3528
diff
changeset
|
300 |
def cstrcheck_after_update_attributes(session, entity): |
3685
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
301 |
if session.is_super_session: |
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
302 |
return |
4119
85bb30fb6d9b
fix name error
Alexandre Fayolle <alexandre.fayolle@logilab.fr>
parents:
4116
diff
changeset
|
303 |
eschema = entity.e_schema |
3554
26e586f3c15c
[schema] make RQL* constraints usable w/ attributes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3528
diff
changeset
|
304 |
for attr in entity.edited_attributes: |
4111
1fda1d356741
use edited_attributes and uniformize attribute schema retreival
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3978
diff
changeset
|
305 |
if eschema.subjrels[attr].final: |
3554
26e586f3c15c
[schema] make RQL* constraints usable w/ attributes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3528
diff
changeset
|
306 |
constraints = [c for c in entity.e_schema.constraints(attr) |
3978
2c95e3033f64
finish yesterday work on rql constraints:
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3887
diff
changeset
|
307 |
if isinstance(c, (RQLConstraint, RQLUniqueConstraint))] |
3554
26e586f3c15c
[schema] make RQL* constraints usable w/ attributes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3528
diff
changeset
|
308 |
if constraints: |
26e586f3c15c
[schema] make RQL* constraints usable w/ attributes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3528
diff
changeset
|
309 |
CheckConstraintsOperation(session, rdef=(entity.eid, attr, None), |
26e586f3c15c
[schema] make RQL* constraints usable w/ attributes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3528
diff
changeset
|
310 |
constraints=constraints) |
26e586f3c15c
[schema] make RQL* constraints usable w/ attributes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3528
diff
changeset
|
311 |
|
0 | 312 |
|
313 |
class CheckRequiredRelationOperation(LateOperation): |
|
314 |
"""checking relation cardinality has to be done after commit in |
|
315 |
case the relation is being replaced |
|
316 |
""" |
|
317 |
eid, rtype = None, None |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
318 |
|
0 | 319 |
def precommit_event(self): |
320 |
# recheck pending eids |
|
2101
08003e0354a7
update transaction data api
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
1977
diff
changeset
|
321 |
if self.eid in self.session.transaction_data.get('pendingeids', ()): |
0 | 322 |
return |
3887
130ac9aee402
remove bad invisible character
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
3886
diff
changeset
|
323 |
if self.rtype in self.session.transaction_data.get('pendingrtypes', ()): |
3886
6389e5d6edcb
check we're not deleting the relation before checking its cardinality...
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3859
diff
changeset
|
324 |
return |
0 | 325 |
if self.session.unsafe_execute(*self._rql()).rowcount < 1: |
326 |
etype = self.session.describe(self.eid)[0] |
|
2241
fcf08ac5f8c0
translate schema types
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2195
diff
changeset
|
327 |
_ = self.session._ |
fcf08ac5f8c0
translate schema types
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2195
diff
changeset
|
328 |
msg = _('at least one relation %(rtype)s is required on %(etype)s (%(eid)s)') |
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
329 |
msg %= {'rtype': _(self.rtype), 'etype': _(etype), 'eid': self.eid} |
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
330 |
raise ValidationError(self.eid, {self.rtype: msg}) |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
331 |
|
0 | 332 |
def commit_event(self): |
333 |
pass |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
334 |
|
0 | 335 |
def _rql(self): |
336 |
raise NotImplementedError() |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
337 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
338 |
|
0 | 339 |
class CheckSRelationOp(CheckRequiredRelationOperation): |
340 |
"""check required subject relation""" |
|
341 |
def _rql(self): |
|
342 |
return 'Any O WHERE S eid %%(x)s, S %s O' % self.rtype, {'x': self.eid}, 'x' |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
343 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
344 |
|
0 | 345 |
class CheckORelationOp(CheckRequiredRelationOperation): |
346 |
"""check required object relation""" |
|
347 |
def _rql(self): |
|
348 |
return 'Any S WHERE O eid %%(x)s, S %s O' % self.rtype, {'x': self.eid}, 'x' |
|
349 |
||
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
350 |
|
0 | 351 |
def checkrel_if_necessary(session, opcls, rtype, eid): |
352 |
"""check an equivalent operation has not already been added""" |
|
353 |
for op in session.pending_operations: |
|
354 |
if isinstance(op, opcls) and op.rtype == rtype and op.eid == eid: |
|
355 |
break |
|
356 |
else: |
|
357 |
opcls(session, rtype=rtype, eid=eid) |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
358 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
359 |
|
0 | 360 |
def cardinalitycheck_after_add_entity(session, entity): |
361 |
"""check cardinalities are satisfied""" |
|
3685
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
362 |
if session.is_super_session: |
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
363 |
return |
0 | 364 |
eid = entity.eid |
365 |
for rschema, targetschemas, x in entity.e_schema.relation_definitions(): |
|
366 |
# skip automatically handled relations |
|
2605
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
367 |
if rschema.type in DONT_CHECK_RTYPES_ON_ADD: |
0 | 368 |
continue |
369 |
if x == 'subject': |
|
370 |
subjtype = entity.e_schema |
|
371 |
objtype = targetschemas[0].type |
|
372 |
cardindex = 0 |
|
373 |
opcls = CheckSRelationOp |
|
374 |
else: |
|
375 |
subjtype = targetschemas[0].type |
|
376 |
objtype = entity.e_schema |
|
377 |
cardindex = 1 |
|
378 |
opcls = CheckORelationOp |
|
379 |
card = rschema.rproperty(subjtype, objtype, 'cardinality') |
|
380 |
if card[cardindex] in '1+': |
|
381 |
checkrel_if_necessary(session, opcls, rschema.type, eid) |
|
382 |
||
383 |
def cardinalitycheck_before_del_relation(session, eidfrom, rtype, eidto): |
|
384 |
"""check cardinalities are satisfied""" |
|
3685
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
385 |
if session.is_super_session: |
6f807804f1d0
no integrity check for super session
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3662
diff
changeset
|
386 |
return |
2605
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
387 |
if rtype in DONT_CHECK_RTYPES_ON_DEL: |
c4f6a53884ec
[R hooks] use DONT_CHECK_RTYPES_ON_[ADD|DEL] constant, don't check wf related internal relations
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2603
diff
changeset
|
388 |
return |
0 | 389 |
card = rproperty(session, rtype, eidfrom, eidto, 'cardinality') |
2745
0dafa29ace1f
[schema migration] test reproducing pb when deleting a relation definition with mandatory card and without removing the associated rtype + fix
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2695
diff
changeset
|
390 |
pendingrdefs = session.transaction_data.get('pendingrdefs', ()) |
0dafa29ace1f
[schema migration] test reproducing pb when deleting a relation definition with mandatory card and without removing the associated rtype + fix
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2695
diff
changeset
|
391 |
if (session.describe(eidfrom)[0], rtype, session.describe(eidto)[0]) in pendingrdefs: |
0dafa29ace1f
[schema migration] test reproducing pb when deleting a relation definition with mandatory card and without removing the associated rtype + fix
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2695
diff
changeset
|
392 |
return |
2101
08003e0354a7
update transaction data api
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
1977
diff
changeset
|
393 |
pendingeids = session.transaction_data.get('pendingeids', ()) |
0 | 394 |
if card[0] in '1+' and not eidfrom in pendingeids: |
395 |
checkrel_if_necessary(session, CheckSRelationOp, rtype, eidfrom) |
|
396 |
if card[1] in '1+' and not eidto in pendingeids: |
|
397 |
checkrel_if_necessary(session, CheckORelationOp, rtype, eidto) |
|
398 |
||
399 |
||
400 |
def _register_core_hooks(hm): |
|
401 |
hm.register_hook(handle_composite_before_del_relation, 'before_delete_relation', '') |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
402 |
hm.register_hook(before_del_group, 'before_delete_entity', 'CWGroup') |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
403 |
|
0 | 404 |
#hm.register_hook(cstrcheck_before_update_entity, 'before_update_entity', '') |
405 |
hm.register_hook(cardinalitycheck_after_add_entity, 'after_add_entity', '') |
|
406 |
hm.register_hook(cardinalitycheck_before_del_relation, 'before_delete_relation', '') |
|
407 |
hm.register_hook(cstrcheck_after_add_relation, 'after_add_relation', '') |
|
408 |
hm.register_hook(uniquecstrcheck_before_modification, 'before_add_entity', '') |
|
409 |
hm.register_hook(uniquecstrcheck_before_modification, 'before_update_entity', '') |
|
3554
26e586f3c15c
[schema] make RQL* constraints usable w/ attributes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3528
diff
changeset
|
410 |
hm.register_hook(cstrcheck_after_update_attributes, 'after_add_entity', '') |
26e586f3c15c
[schema] make RQL* constraints usable w/ attributes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3528
diff
changeset
|
411 |
hm.register_hook(cstrcheck_after_update_attributes, 'after_update_entity', '') |
0 | 412 |
|
413 |
# user/groups synchronisation ################################################# |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
414 |
|
0 | 415 |
class GroupOperation(Operation): |
416 |
"""base class for group operation""" |
|
417 |
geid = None |
|
418 |
def __init__(self, session, *args, **kwargs): |
|
419 |
"""override to get the group name before actual groups manipulation: |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
420 |
|
0 | 421 |
we may temporarily loose right access during a commit event, so |
422 |
no query should be emitted while comitting |
|
423 |
""" |
|
424 |
rql = 'Any N WHERE G eid %(x)s, G name N' |
|
425 |
result = session.execute(rql, {'x': kwargs['geid']}, 'x', build_descr=False) |
|
426 |
Operation.__init__(self, session, *args, **kwargs) |
|
427 |
self.group = result[0][0] |
|
428 |
||
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
429 |
|
0 | 430 |
class DeleteGroupOp(GroupOperation): |
431 |
"""synchronize user when a in_group relation has been deleted""" |
|
432 |
def commit_event(self): |
|
433 |
"""the observed connections pool has been commited""" |
|
434 |
groups = self.cnxuser.groups |
|
435 |
try: |
|
436 |
groups.remove(self.group) |
|
437 |
except KeyError: |
|
438 |
self.error('user %s not in group %s', self.cnxuser, self.group) |
|
439 |
return |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
440 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
441 |
|
0 | 442 |
def after_del_in_group(session, fromeid, rtype, toeid): |
443 |
"""modify user permission, need to update users""" |
|
444 |
for session_ in get_user_sessions(session.repo, fromeid): |
|
445 |
DeleteGroupOp(session, cnxuser=session_.user, geid=toeid) |
|
446 |
||
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
447 |
|
0 | 448 |
class AddGroupOp(GroupOperation): |
449 |
"""synchronize user when a in_group relation has been added""" |
|
450 |
def commit_event(self): |
|
451 |
"""the observed connections pool has been commited""" |
|
452 |
groups = self.cnxuser.groups |
|
453 |
if self.group in groups: |
|
454 |
self.warning('user %s already in group %s', self.cnxuser, |
|
455 |
self.group) |
|
456 |
return |
|
457 |
groups.add(self.group) |
|
458 |
||
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
459 |
|
0 | 460 |
def after_add_in_group(session, fromeid, rtype, toeid): |
461 |
"""modify user permission, need to update users""" |
|
462 |
for session_ in get_user_sessions(session.repo, fromeid): |
|
463 |
AddGroupOp(session, cnxuser=session_.user, geid=toeid) |
|
464 |
||
465 |
||
466 |
class DelUserOp(Operation): |
|
467 |
"""synchronize user when a in_group relation has been added""" |
|
468 |
def __init__(self, session, cnxid): |
|
469 |
self.cnxid = cnxid |
|
470 |
Operation.__init__(self, session) |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
471 |
|
0 | 472 |
def commit_event(self): |
473 |
"""the observed connections pool has been commited""" |
|
474 |
try: |
|
475 |
self.repo.close(self.cnxid) |
|
476 |
except BadConnectionId: |
|
477 |
pass # already closed |
|
478 |
||
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
479 |
|
0 | 480 |
def after_del_user(session, eid): |
481 |
"""modify user permission, need to update users""" |
|
482 |
for session_ in get_user_sessions(session.repo, eid): |
|
483 |
DelUserOp(session, session_.id) |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
484 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
485 |
|
0 | 486 |
def _register_usergroup_hooks(hm): |
487 |
"""register user/group related hooks on the hooks manager""" |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
488 |
hm.register_hook(after_del_user, 'after_delete_entity', 'CWUser') |
0 | 489 |
hm.register_hook(after_add_in_group, 'after_add_relation', 'in_group') |
490 |
hm.register_hook(after_del_in_group, 'after_delete_relation', 'in_group') |
|
491 |
||
492 |
||
493 |
# workflow handling ########################################################### |
|
494 |
||
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
495 |
from cubicweb.entities.wfobjs import WorkflowTransition, WorkflowException |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
496 |
|
2949
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
497 |
def _change_state(session, x, oldstate, newstate): |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
498 |
nocheck = session.transaction_data.setdefault('skip-security', set()) |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
499 |
nocheck.add((x, 'in_state', oldstate)) |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
500 |
nocheck.add((x, 'in_state', newstate)) |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
501 |
# delete previous state first in case we're using a super session |
3039
7d5a4d27d052
[multi-sources] in ms config, don't try to delete previous state in some condition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2992
diff
changeset
|
502 |
fromsource = session.describe(x)[1] |
7d5a4d27d052
[multi-sources] in ms config, don't try to delete previous state in some condition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2992
diff
changeset
|
503 |
# don't try to remove previous state if in_state isn't stored in the system |
7d5a4d27d052
[multi-sources] in ms config, don't try to delete previous state in some condition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2992
diff
changeset
|
504 |
# source |
7d5a4d27d052
[multi-sources] in ms config, don't try to delete previous state in some condition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2992
diff
changeset
|
505 |
if fromsource == 'system' or \ |
7d5a4d27d052
[multi-sources] in ms config, don't try to delete previous state in some condition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2992
diff
changeset
|
506 |
not session.repo.sources_by_uri[fromsource].support_relation('in_state'): |
7d5a4d27d052
[multi-sources] in ms config, don't try to delete previous state in some condition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2992
diff
changeset
|
507 |
session.delete_relation(x, 'in_state', oldstate) |
2949
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
508 |
session.add_relation(x, 'in_state', newstate) |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
509 |
|
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
510 |
|
3528
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
511 |
class FireAutotransitionOp(PreCommitOperation): |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
512 |
"""try to fire auto transition after state changes""" |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
513 |
|
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
514 |
def precommit_event(self): |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
515 |
session = self.session |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
516 |
entity = self.entity |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
517 |
autotrs = list(entity.possible_transitions('auto')) |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
518 |
if autotrs: |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
519 |
assert len(autotrs) == 1 |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
520 |
entity.fire_transition(autotrs[0]) |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
521 |
|
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
522 |
|
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
523 |
def before_add_trinfo(session, entity): |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
524 |
"""check the transition is allowed, add missing information. Expect that: |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
525 |
* wf_info_for inlined relation is set |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
526 |
* by_transition or to_state (managers only) inlined relation is set |
0 | 527 |
""" |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
528 |
# first retreive entity to which the state change apply |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
529 |
try: |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
530 |
foreid = entity['wf_info_for'] |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
531 |
except KeyError: |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
532 |
msg = session._('mandatory relation') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
533 |
raise ValidationError(entity.eid, {'wf_info_for': msg}) |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
534 |
forentity = session.entity_from_eid(foreid) |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
535 |
# then check it has a workflow set, unless we're in the process of changing |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
536 |
# entity's workflow |
2978
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
537 |
if session.transaction_data.get((forentity.eid, 'customwf')): |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
538 |
wfeid = session.transaction_data[(forentity.eid, 'customwf')] |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
539 |
wf = session.entity_from_eid(wfeid) |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
540 |
else: |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
541 |
wf = forentity.current_workflow |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
542 |
if wf is None: |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
543 |
msg = session._('related entity has no workflow set') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
544 |
raise ValidationError(entity.eid, {None: msg}) |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
545 |
# then check it has a state set |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
546 |
fromstate = forentity.current_state |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
547 |
if fromstate is None: |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
548 |
msg = session._('related entity has no state') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
549 |
raise ValidationError(entity.eid, {None: msg}) |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
550 |
# True if we are coming back from subworkflow |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
551 |
swtr = session.transaction_data.pop((forentity.eid, 'subwfentrytr'), None) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
552 |
cowpowers = session.is_super_session or 'managers' in session.user.groups |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
553 |
# no investigate the requested state change... |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
554 |
try: |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
555 |
treid = entity['by_transition'] |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
556 |
except KeyError: |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
557 |
# no transition set, check user is a manager and destination state is |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
558 |
# specified (and valid) |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
559 |
if not cowpowers: |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
560 |
msg = session._('mandatory relation') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
561 |
raise ValidationError(entity.eid, {'by_transition': msg}) |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
562 |
deststateeid = entity.get('to_state') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
563 |
if not deststateeid: |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
564 |
msg = session._('mandatory relation') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
565 |
raise ValidationError(entity.eid, {'by_transition': msg}) |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
566 |
deststate = wf.state_by_eid(deststateeid) |
3773
14fde27a70a2
don't allow that even with cow powers
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3693
diff
changeset
|
567 |
if deststate is None: |
14fde27a70a2
don't allow that even with cow powers
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3693
diff
changeset
|
568 |
msg = entity.req._("state doesn't belong to entity's current workflow") |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
569 |
raise ValidationError(entity.eid, {'to_state': msg}) |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
570 |
else: |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
571 |
# check transition is valid and allowed, unless we're coming back from |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
572 |
# subworkflow |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
573 |
tr = session.entity_from_eid(treid) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
574 |
if swtr is None: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
575 |
if tr is None: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
576 |
msg = session._("transition doesn't belong to entity's workflow") |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
577 |
raise ValidationError(entity.eid, {'by_transition': msg}) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
578 |
if not tr.has_input_state(fromstate): |
3859
85e6ba89837a
more helpful error message
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3773
diff
changeset
|
579 |
_ = session._ |
4098
eb4722dd5f86
[i18n] use named substitutions for proper localization
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
3978
diff
changeset
|
580 |
msg = _("transition %(tr)s isn't allowed from %(st)s") % {'tr': _(tr.name), |
eb4722dd5f86
[i18n] use named substitutions for proper localization
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
3978
diff
changeset
|
581 |
'st': _(fromstate.name), |
eb4722dd5f86
[i18n] use named substitutions for proper localization
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
3978
diff
changeset
|
582 |
} |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
583 |
raise ValidationError(entity.eid, {'by_transition': msg}) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
584 |
if not tr.may_be_fired(foreid): |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
585 |
msg = session._("transition may not be fired") |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
586 |
raise ValidationError(entity.eid, {'by_transition': msg}) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
587 |
if entity.get('to_state'): |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
588 |
deststateeid = entity['to_state'] |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
589 |
if not cowpowers and deststateeid != tr.destination().eid: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
590 |
msg = session._("transition isn't allowed") |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
591 |
raise ValidationError(entity.eid, {'by_transition': msg}) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
592 |
if swtr is None: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
593 |
deststate = session.entity_from_eid(deststateeid) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
594 |
if not cowpowers and deststate is None: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
595 |
msg = entity.req._("state doesn't belong to entity's workflow") |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
596 |
raise ValidationError(entity.eid, {'to_state': msg}) |
0 | 597 |
else: |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
598 |
deststateeid = tr.destination().eid |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
599 |
# everything is ok, add missing information on the trinfo entity |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
600 |
entity['from_state'] = fromstate.eid |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
601 |
entity['to_state'] = deststateeid |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
602 |
nocheck = session.transaction_data.setdefault('skip-security', set()) |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
603 |
nocheck.add((entity.eid, 'from_state', fromstate.eid)) |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
604 |
nocheck.add((entity.eid, 'to_state', deststateeid)) |
3528
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
605 |
FireAutotransitionOp(session, entity=forentity) |
77a69de16709
support for automatic transition
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3508
diff
changeset
|
606 |
|
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
607 |
|
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
608 |
def after_add_trinfo(session, entity): |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
609 |
"""change related entity state""" |
2949
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
610 |
_change_state(session, entity['wf_info_for'], |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
611 |
entity['from_state'], entity['to_state']) |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
612 |
forentity = session.entity_from_eid(entity['wf_info_for']) |
3568
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
613 |
assert forentity.current_state.eid == entity['to_state'], ( |
87a7ca9d8ce6
[hooks] don't delete composed of a composite where relation is being removed from the schema
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
3554
diff
changeset
|
614 |
forentity.eid, forentity.current_state.name) |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
615 |
if forentity.main_workflow.eid != forentity.current_workflow.eid: |
3662
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
616 |
SubWorkflowExitOp(session, forentity=forentity, trinfo=entity) |
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
617 |
|
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
618 |
class SubWorkflowExitOp(PreCommitOperation): |
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
619 |
def precommit_event(self): |
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
620 |
session = self.session |
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
621 |
forentity = self.forentity |
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
622 |
trinfo = self.trinfo |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
623 |
# we're in a subworkflow, check if we've reached an exit point |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
624 |
wftr = forentity.subworkflow_input_transition() |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
625 |
if wftr is None: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
626 |
# inconsistency detected |
3662
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
627 |
msg = session._("state doesn't belong to entity's current workflow") |
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
628 |
raise ValidationError(self.trinfo.eid, {'to_state': msg}) |
9eeadad82d93
defer trinfo creation when cancelling subworkflow
Aurelien Campeas <aurelien.campeas@logilab.fr>
parents:
3628
diff
changeset
|
629 |
tostate = wftr.get_exit_point(forentity, trinfo['to_state']) |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
630 |
if tostate is not None: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
631 |
# reached an exit point |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
632 |
msg = session._('exiting from subworkflow %s') |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
633 |
msg %= session._(forentity.current_workflow.name) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
634 |
session.transaction_data[(forentity.eid, 'subwfentrytr')] = True |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
635 |
# XXX iirk |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
636 |
req = forentity.req |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
637 |
forentity.req = session.super_session |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
638 |
try: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
639 |
trinfo = forentity.change_state(tostate, msg, u'text/plain', |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
640 |
tr=wftr) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
641 |
finally: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
642 |
forentity.req = req |
0 | 643 |
|
644 |
||
645 |
class SetInitialStateOp(PreCommitOperation): |
|
646 |
"""make initial state be a default state""" |
|
647 |
||
648 |
def precommit_event(self): |
|
649 |
session = self.session |
|
650 |
entity = self.entity |
|
651 |
# if there is an initial state and the entity's state is not set, |
|
652 |
# use the initial state as a default state |
|
2101
08003e0354a7
update transaction data api
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
1977
diff
changeset
|
653 |
pendingeids = session.transaction_data.get('pendingeids', ()) |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
654 |
if not entity.eid in pendingeids and not entity.in_state and \ |
2978
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
655 |
entity.main_workflow: |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
656 |
state = entity.main_workflow.initial |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
657 |
if state: |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
658 |
# use super session to by-pass security checks |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
659 |
session.super_session.add_relation(entity.eid, 'in_state', |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
660 |
state.eid) |
0 | 661 |
|
662 |
||
663 |
def set_initial_state_after_add(session, entity): |
|
664 |
SetInitialStateOp(session, entity=entity) |
|
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
665 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
666 |
|
2983
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
667 |
def before_add_in_state(session, eidfrom, rtype, eidto): |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
668 |
"""check state apply, in case of direct in_state change using unsafe_execute |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
669 |
""" |
3508
d9915224f9a5
[B] server: fix initial data structure in session.transaction_data
Julien Jehannet <julien.jehannet@logilab.fr>
parents:
3071
diff
changeset
|
670 |
nocheck = session.transaction_data.setdefault('skip-security', set()) |
2983
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
671 |
if (eidfrom, 'in_state', eidto) in nocheck: |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
672 |
# state changed through TrInfo insertion, so we already know it's ok |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
673 |
return |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
674 |
entity = session.entity_from_eid(eidfrom) |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
675 |
mainwf = entity.main_workflow |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
676 |
if mainwf is None: |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
677 |
msg = session._('entity has no workflow set') |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
678 |
raise ValidationError(entity.eid, {None: msg}) |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
679 |
for wf in mainwf.iter_workflows(): |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
680 |
if wf.state_by_eid(eidto): |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
681 |
break |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
682 |
else: |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
683 |
msg = session._("state doesn't belong to entity's workflow. You may " |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
684 |
"want to set a custom workflow for this entity first.") |
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
685 |
raise ValidationError(eidfrom, {'in_state': msg}) |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
686 |
if entity.current_workflow and wf.eid != entity.current_workflow.eid: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
687 |
msg = session._("state doesn't belong to entity's current workflow") |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
688 |
raise ValidationError(eidfrom, {'in_state': msg}) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
689 |
|
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
690 |
|
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
691 |
class CheckTrExitPoint(PreCommitOperation): |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
692 |
|
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
693 |
def precommit_event(self): |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
694 |
tr = self.session.entity_from_eid(self.treid) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
695 |
outputs = set() |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
696 |
for ep in tr.subworkflow_exit: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
697 |
if ep.subwf_state.eid in outputs: |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
698 |
msg = self.session._("can't have multiple exits on the same state") |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
699 |
raise ValidationError(self.treid, {'subworkflow_exit': msg}) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
700 |
outputs.add(ep.subwf_state.eid) |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
701 |
|
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
702 |
|
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
703 |
def after_add_subworkflow_exit(session, eidfrom, rtype, eidto): |
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
704 |
CheckTrExitPoint(session, treid=eidfrom) |
2983
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
705 |
|
2949
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
706 |
|
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
707 |
class WorkflowChangedOp(PreCommitOperation): |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
708 |
"""fix entity current state when changing its workflow""" |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
709 |
|
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
710 |
def precommit_event(self): |
2985 | 711 |
# notice that enforcement that new workflow apply to the entity's type is |
712 |
# done by schema rule, no need to check it here |
|
2949
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
713 |
session = self.session |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
714 |
pendingeids = session.transaction_data.get('pendingeids', ()) |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
715 |
if self.eid in pendingeids: |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
716 |
return |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
717 |
entity = session.entity_from_eid(self.eid) |
2978
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
718 |
# check custom workflow has not been rechanged to another one in the same |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
719 |
# transaction |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
720 |
mainwf = entity.main_workflow |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
721 |
if mainwf.eid == self.wfeid: |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
722 |
deststate = mainwf.initial |
2949
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
723 |
if not deststate: |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
724 |
msg = session._('workflow has no initial state') |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
725 |
raise ValidationError(entity.eid, {'custom_workflow': msg}) |
2978
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
726 |
if mainwf.state_by_eid(entity.current_state.eid): |
2949
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
727 |
# nothing to do |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
728 |
return |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
729 |
# if there are no history, simply go to new workflow's initial state |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
730 |
if not entity.workflow_history: |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
731 |
if entity.current_state.eid != deststate.eid: |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
732 |
_change_state(session, entity.eid, |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
733 |
entity.current_state.eid, deststate.eid) |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
734 |
return |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
735 |
msg = session._('workflow changed to "%s"') |
2978
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
736 |
msg %= session._(mainwf.name) |
d8c5ad14ab8e
[wf]Â distinguish main workflow / current workflow
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2949
diff
changeset
|
737 |
session.transaction_data[(entity.eid, 'customwf')] = self.wfeid |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
738 |
entity.change_state(deststate, msg, u'text/plain') |
2949
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
739 |
|
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
740 |
|
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
741 |
def set_custom_workflow(session, eidfrom, rtype, eidto): |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
742 |
WorkflowChangedOp(session, eid=eidfrom, wfeid=eidto) |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
743 |
|
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
744 |
|
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
745 |
def del_custom_workflow(session, eidfrom, rtype, eidto): |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
746 |
entity = session.entity_from_eid(eidfrom) |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
747 |
typewf = entity.cwetype_workflow() |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
748 |
if typewf is not None: |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
749 |
WorkflowChangedOp(session, eid=eidfrom, wfeid=typewf.eid) |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
750 |
|
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
751 |
|
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
752 |
def after_del_workflow(session, eid): |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
753 |
# workflow cleanup |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
754 |
session.execute('DELETE State X WHERE NOT X state_of Y') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
755 |
session.execute('DELETE Transition X WHERE NOT X transition_of Y') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
756 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
757 |
|
0 | 758 |
def _register_wf_hooks(hm): |
759 |
"""register workflow related hooks on the hooks manager""" |
|
760 |
if 'in_state' in hm.schema: |
|
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
761 |
hm.register_hook(before_add_trinfo, 'before_add_entity', 'TrInfo') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
762 |
hm.register_hook(after_add_trinfo, 'after_add_entity', 'TrInfo') |
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
763 |
#hm.register_hook(relation_deleted, 'before_delete_relation', 'in_state') |
0 | 764 |
for eschema in hm.schema.entities(): |
765 |
if 'in_state' in eschema.subject_relations(): |
|
766 |
hm.register_hook(set_initial_state_after_add, 'after_add_entity', |
|
767 |
str(eschema)) |
|
2949
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
768 |
hm.register_hook(set_custom_workflow, 'after_add_relation', 'custom_workflow') |
a2aa2c51f3be
test and implements workflow changes
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2920
diff
changeset
|
769 |
hm.register_hook(del_custom_workflow, 'after_delete_relation', 'custom_workflow') |
2920
64322aa83a1d
start a new workflow engine
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2909
diff
changeset
|
770 |
hm.register_hook(after_del_workflow, 'after_delete_entity', 'Workflow') |
2983
b458631fe347
[wf] test in_state consistency
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2978
diff
changeset
|
771 |
hm.register_hook(before_add_in_state, 'before_add_relation', 'in_state') |
2992
a5b8bf107a1a
[wf] test and hooks for WorkflowTransition support
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2985
diff
changeset
|
772 |
hm.register_hook(after_add_subworkflow_exit, 'after_add_relation', 'subworkflow_exit') |
0 | 773 |
|
774 |
||
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
775 |
# CWProperty hooks ############################################################# |
0 | 776 |
|
777 |
||
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
778 |
class DelCWPropertyOp(Operation): |
0 | 779 |
"""a user's custom properties has been deleted""" |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
780 |
|
0 | 781 |
def commit_event(self): |
782 |
"""the observed connections pool has been commited""" |
|
783 |
try: |
|
784 |
del self.epropdict[self.key] |
|
785 |
except KeyError: |
|
786 |
self.error('%s has no associated value', self.key) |
|
787 |
||
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
788 |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
789 |
class ChangeCWPropertyOp(Operation): |
0 | 790 |
"""a user's custom properties has been added/changed""" |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
791 |
|
0 | 792 |
def commit_event(self): |
793 |
"""the observed connections pool has been commited""" |
|
794 |
self.epropdict[self.key] = self.value |
|
795 |
||
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
796 |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
797 |
class AddCWPropertyOp(Operation): |
0 | 798 |
"""a user's custom properties has been added/changed""" |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
799 |
|
0 | 800 |
def commit_event(self): |
801 |
"""the observed connections pool has been commited""" |
|
802 |
eprop = self.eprop |
|
803 |
if not eprop.for_user: |
|
804 |
self.repo.vreg.eprop_values[eprop.pkey] = eprop.value |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
805 |
# if for_user is set, update is handled by a ChangeCWPropertyOp operation |
0 | 806 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
807 |
|
0 | 808 |
def after_add_eproperty(session, entity): |
809 |
key, value = entity.pkey, entity.value |
|
810 |
try: |
|
811 |
value = session.vreg.typed_value(key, value) |
|
812 |
except UnknownProperty: |
|
813 |
raise ValidationError(entity.eid, {'pkey': session._('unknown property key')}) |
|
814 |
except ValueError, ex: |
|
815 |
raise ValidationError(entity.eid, {'value': session._(str(ex))}) |
|
816 |
if not session.user.matching_groups('managers'): |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
817 |
session.add_relation(entity.eid, 'for_user', session.user.eid) |
0 | 818 |
else: |
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
819 |
AddCWPropertyOp(session, eprop=entity) |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
820 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
821 |
|
0 | 822 |
def after_update_eproperty(session, entity): |
2695
56439c45781c
[hooks] check key or value is actually edited
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2607
diff
changeset
|
823 |
if not ('pkey' in entity.edited_attributes or |
56439c45781c
[hooks] check key or value is actually edited
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2607
diff
changeset
|
824 |
'value' in entity.edited_attributes): |
56439c45781c
[hooks] check key or value is actually edited
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2607
diff
changeset
|
825 |
return |
0 | 826 |
key, value = entity.pkey, entity.value |
827 |
try: |
|
828 |
value = session.vreg.typed_value(key, value) |
|
829 |
except UnknownProperty: |
|
830 |
return |
|
831 |
except ValueError, ex: |
|
832 |
raise ValidationError(entity.eid, {'value': session._(str(ex))}) |
|
833 |
if entity.for_user: |
|
834 |
for session_ in get_user_sessions(session.repo, entity.for_user[0].eid): |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
835 |
ChangeCWPropertyOp(session, epropdict=session_.user.properties, |
0 | 836 |
key=key, value=value) |
837 |
else: |
|
838 |
# site wide properties |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
839 |
ChangeCWPropertyOp(session, epropdict=session.vreg.eprop_values, |
0 | 840 |
key=key, value=value) |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
841 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
842 |
|
0 | 843 |
def before_del_eproperty(session, eid): |
2101
08003e0354a7
update transaction data api
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
1977
diff
changeset
|
844 |
for eidfrom, rtype, eidto in session.transaction_data.get('pendingrelations', ()): |
0 | 845 |
if rtype == 'for_user' and eidfrom == eid: |
846 |
# if for_user was set, delete has already been handled |
|
847 |
break |
|
848 |
else: |
|
849 |
key = session.execute('Any K WHERE P eid %(x)s, P pkey K', |
|
850 |
{'x': eid}, 'x')[0][0] |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
851 |
DelCWPropertyOp(session, epropdict=session.vreg.eprop_values, key=key) |
0 | 852 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
853 |
|
0 | 854 |
def after_add_for_user(session, fromeid, rtype, toeid): |
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
855 |
if not session.describe(fromeid)[0] == 'CWProperty': |
0 | 856 |
return |
857 |
key, value = session.execute('Any K,V WHERE P eid %(x)s,P pkey K,P value V', |
|
858 |
{'x': fromeid}, 'x')[0] |
|
859 |
if session.vreg.property_info(key)['sitewide']: |
|
860 |
raise ValidationError(fromeid, |
|
861 |
{'for_user': session._("site-wide property can't be set for user")}) |
|
862 |
for session_ in get_user_sessions(session.repo, toeid): |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
863 |
ChangeCWPropertyOp(session, epropdict=session_.user.properties, |
0 | 864 |
key=key, value=value) |
1802
d628defebc17
delete-trailing-whitespace + some copyright update
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
1398
diff
changeset
|
865 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
866 |
|
0 | 867 |
def before_del_for_user(session, fromeid, rtype, toeid): |
868 |
key = session.execute('Any K WHERE P eid %(x)s, P pkey K', |
|
869 |
{'x': fromeid}, 'x')[0][0] |
|
870 |
relation_deleted(session, fromeid, rtype, toeid) |
|
871 |
for session_ in get_user_sessions(session.repo, toeid): |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
872 |
DelCWPropertyOp(session, epropdict=session_.user.properties, key=key) |
0 | 873 |
|
2603
e47d63351891
[R hooks] optmize insertion by using direct repository methods when possible
Sylvain Thénault <sylvain.thenault@logilab.fr>
parents:
2463
diff
changeset
|
874 |
|
0 | 875 |
def _register_eproperty_hooks(hm): |
876 |
"""register workflow related hooks on the hooks manager""" |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
877 |
hm.register_hook(after_add_eproperty, 'after_add_entity', 'CWProperty') |
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
878 |
hm.register_hook(after_update_eproperty, 'after_update_entity', 'CWProperty') |
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1263
diff
changeset
|
879 |
hm.register_hook(before_del_eproperty, 'before_delete_entity', 'CWProperty') |
0 | 880 |
hm.register_hook(after_add_for_user, 'after_add_relation', 'for_user') |
881 |
hm.register_hook(before_del_for_user, 'before_delete_relation', 'for_user') |