author | sylvain.thenault@logilab.fr |
Thu, 23 Apr 2009 16:11:30 +0200 | |
branch | tls-sprint |
changeset 1451 | 982e8616d9a2 |
parent 1398 | 5fe84a5f7035 |
child 1642 | 12a98b17fb05 |
permissions | -rw-r--r-- |
0 | 1 |
# -*- coding: utf-8 -*- |
2 |
"""unit tests for cubicweb.entities.base module""" |
|
3 |
||
4 |
from logilab.common.testlib import unittest_main |
|
5 |
from logilab.common.decorators import clear_cache |
|
6 |
from logilab.common.interface import implements |
|
7 |
||
8 |
from cubicweb.devtools.apptest import EnvBasedTC |
|
9 |
||
10 |
from cubicweb import ValidationError |
|
11 |
from cubicweb.interfaces import IMileStone, IWorkflowable |
|
12 |
from cubicweb.entities import AnyEntity |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
13 |
from cubicweb.entities.authobjs import CWUser |
0 | 14 |
from cubicweb.web.widgets import AutoCompletionWidget |
15 |
||
16 |
||
17 |
class BaseEntityTC(EnvBasedTC): |
|
18 |
||
19 |
def setup_database(self): |
|
20 |
self.member = self.create_user('member') |
|
21 |
||
22 |
||
23 |
||
24 |
class MetadataTC(BaseEntityTC): |
|
25 |
||
26 |
def test_creator(self): |
|
27 |
self.login(u'member') |
|
28 |
card = self.add_entity('Card', title=u"hello") |
|
29 |
self.commit() |
|
30 |
self.assertEquals(card.creator.eid, self.member.eid) |
|
31 |
self.assertEquals(card.dc_creator(), u'member') |
|
32 |
||
33 |
def test_type(self): |
|
34 |
self.assertEquals(self.member.dc_type(), 'euser') |
|
35 |
||
36 |
||
1102
054dfee29bfc
some test for meta attributes (XXX should be in yams)
sylvain.thenault@logilab.fr
parents:
1016
diff
changeset
|
37 |
def test_entity_meta_attributes(self): |
054dfee29bfc
some test for meta attributes (XXX should be in yams)
sylvain.thenault@logilab.fr
parents:
1016
diff
changeset
|
38 |
# XXX move to yams |
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
39 |
self.assertEquals(self.schema['CWUser'].meta_attributes(), {}) |
1102
054dfee29bfc
some test for meta attributes (XXX should be in yams)
sylvain.thenault@logilab.fr
parents:
1016
diff
changeset
|
40 |
self.assertEquals(dict((str(k), v) for k, v in self.schema['Card'].meta_attributes().iteritems()), |
054dfee29bfc
some test for meta attributes (XXX should be in yams)
sylvain.thenault@logilab.fr
parents:
1016
diff
changeset
|
41 |
{'content_format': ('format', 'content')}) |
054dfee29bfc
some test for meta attributes (XXX should be in yams)
sylvain.thenault@logilab.fr
parents:
1016
diff
changeset
|
42 |
|
0 | 43 |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
44 |
class CWUserTC(BaseEntityTC): |
0 | 45 |
def test_dc_title_and_name(self): |
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
46 |
e = self.entity('CWUser U WHERE U login "member"') |
0 | 47 |
self.assertEquals(e.dc_title(), 'member') |
48 |
self.assertEquals(e.name(), 'member') |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
49 |
self.execute(u'SET X firstname "bouah" WHERE X is CWUser, X login "member"') |
0 | 50 |
self.assertEquals(e.dc_title(), 'member') |
51 |
self.assertEquals(e.name(), u'bouah') |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
52 |
self.execute(u'SET X surname "lôt" WHERE X is CWUser, X login "member"') |
0 | 53 |
self.assertEquals(e.dc_title(), 'member') |
54 |
self.assertEquals(e.name(), u'bouah lôt') |
|
55 |
||
56 |
||
57 |
class StateAndTransitionsTC(BaseEntityTC): |
|
58 |
||
59 |
def test_transitions(self): |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
60 |
user = self.entity('CWUser X') |
0 | 61 |
e = self.entity('State S WHERE S name "activated"') |
62 |
trs = list(e.transitions(user)) |
|
63 |
self.assertEquals(len(trs), 1) |
|
64 |
self.assertEquals(trs[0].name, u'deactivate') |
|
65 |
self.assertEquals(trs[0].destination().name, u'deactivated') |
|
66 |
self.assert_(user.can_pass_transition('deactivate')) |
|
67 |
self.assert_(not user.can_pass_transition('activate')) |
|
68 |
# test a std user get no possible transition |
|
69 |
self.login('member') |
|
70 |
# fetch the entity using the new session |
|
71 |
e = self.entity('State S WHERE S name "activated"') |
|
72 |
trs = list(e.transitions(user)) |
|
73 |
self.assertEquals(len(trs), 0) |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
74 |
user = self.entity('CWUser X') |
0 | 75 |
self.assert_(not user.can_pass_transition('deactivate')) |
76 |
self.assert_(not user.can_pass_transition('activate')) |
|
77 |
||
78 |
def test_transitions_with_dest_specfied(self): |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
79 |
user = self.entity('CWUser X') |
0 | 80 |
e = self.entity('State S WHERE S name "activated"') |
81 |
e2 = self.entity('State S WHERE S name "deactivated"') |
|
82 |
trs = list(e.transitions(user, e2.eid)) |
|
83 |
self.assertEquals(len(trs), 1) |
|
84 |
self.assertEquals(trs[0].name, u'deactivate') |
|
85 |
self.assertEquals(trs[0].destination().name, u'deactivated') |
|
86 |
trs = list(e.transitions(user, e.eid)) |
|
87 |
self.assertEquals(len(trs), 0) |
|
88 |
||
89 |
def test_transitions_maybe_passed(self): |
|
90 |
self.execute('INSERT RQLExpression X: X exprtype "ERQLExpression", ' |
|
91 |
'X expression "X owned_by U", T condition X ' |
|
92 |
'WHERE T name "deactivate"') |
|
93 |
self._test_deactivated() |
|
94 |
||
95 |
def test_transitions_maybe_passed_using_has_update_perm(self): |
|
96 |
self.execute('INSERT RQLExpression X: X exprtype "ERQLExpression", ' |
|
97 |
'X expression "U has_update_permission X", T condition X ' |
|
98 |
'WHERE T name "deactivate"') |
|
99 |
self._test_deactivated() |
|
100 |
||
101 |
||
102 |
def _test_deactivated(self): |
|
103 |
ueid = self.create_user('toto').eid |
|
104 |
self.create_user('tutu') |
|
105 |
cnx = self.login('tutu') |
|
106 |
cu = cnx.cursor() |
|
107 |
self.assertRaises(ValidationError, |
|
108 |
cu.execute, 'SET X in_state S WHERE X eid %(x)s, S name "deactivated"', |
|
109 |
{'x': ueid}, 'x') |
|
110 |
cnx.close() |
|
111 |
cnx = self.login('toto') |
|
112 |
cu = cnx.cursor() |
|
113 |
cu.execute('SET X in_state S WHERE X eid %(x)s, S name "deactivated"', |
|
114 |
{'x': ueid}, 'x') |
|
115 |
cnx.commit() |
|
116 |
self.assertRaises(ValidationError, |
|
117 |
cu.execute, 'SET X in_state S WHERE X eid %(x)s, S name "activated"', |
|
118 |
{'x': ueid}, 'x') |
|
119 |
||
120 |
||
121 |
def test_transitions_selection(self): |
|
122 |
""" |
|
123 |
------------------------ tr1 ----------------- |
|
124 |
| state1 (Card, Bookmark) | ------> | state2 (Card) | |
|
125 |
------------------------ ----------------- |
|
126 |
| tr2 ------------------ |
|
127 |
`------> | state3 (Bookmark) | |
|
128 |
------------------ |
|
129 |
""" |
|
130 |
state1 = self.add_entity('State', name=u'state1') |
|
131 |
state2 = self.add_entity('State', name=u'state2') |
|
132 |
state3 = self.add_entity('State', name=u'state3') |
|
133 |
tr1 = self.add_entity('Transition', name=u'tr1') |
|
134 |
tr2 = self.add_entity('Transition', name=u'tr2') |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
135 |
self.execute('SET X state_of Y WHERE X eid in (%s, %s), Y is CWEType, Y name "Card"' % |
0 | 136 |
(state1.eid, state2.eid)) |
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
137 |
self.execute('SET X state_of Y WHERE X eid in (%s, %s), Y is CWEType, Y name "Bookmark"' % |
0 | 138 |
(state1.eid, state3.eid)) |
139 |
self.execute('SET X transition_of Y WHERE X eid %s, Y name "Card"' % tr1.eid) |
|
140 |
self.execute('SET X transition_of Y WHERE X eid %s, Y name "Bookmark"' % tr2.eid) |
|
141 |
self.execute('SET X allowed_transition Y WHERE X eid %s, Y eid %s' % |
|
142 |
(state1.eid, tr1.eid)) |
|
143 |
self.execute('SET X allowed_transition Y WHERE X eid %s, Y eid %s' % |
|
144 |
(state1.eid, tr2.eid)) |
|
145 |
self.execute('SET X destination_state Y WHERE X eid %s, Y eid %s' % |
|
146 |
(tr1.eid, state2.eid)) |
|
147 |
self.execute('SET X destination_state Y WHERE X eid %s, Y eid %s' % |
|
148 |
(tr2.eid, state3.eid)) |
|
149 |
self.execute('SET X initial_state Y WHERE Y eid %s, X name "Card"' % state1.eid) |
|
150 |
self.execute('SET X initial_state Y WHERE Y eid %s, X name "Bookmark"' % state1.eid) |
|
151 |
card = self.add_entity('Card', title=u't1') |
|
152 |
bookmark = self.add_entity('Bookmark', title=u'111', path=u'/view') |
|
153 |
||
154 |
transitions = list(state1.transitions(card)) |
|
155 |
self.assertEquals(len(transitions), 1) |
|
156 |
self.assertEquals(transitions[0].name, 'tr1') |
|
157 |
transitions = list(state1.transitions(bookmark)) |
|
158 |
self.assertEquals(len(transitions), 1) |
|
159 |
self.assertEquals(transitions[0].name, 'tr2') |
|
160 |
||
161 |
||
162 |
def test_transitions_selection2(self): |
|
163 |
""" |
|
164 |
------------------------ tr1 (Bookmark) ----------------------- |
|
165 |
| state1 (Card, Bookmark) | -------------> | state2 (Card,Bookmark) | |
|
166 |
------------------------ ----------------------- |
|
167 |
| tr2 (Card) | |
|
168 |
`---------------------------------/ |
|
169 |
""" |
|
170 |
state1 = self.add_entity('State', name=u'state1') |
|
171 |
state2 = self.add_entity('State', name=u'state2') |
|
172 |
tr1 = self.add_entity('Transition', name=u'tr1') |
|
173 |
tr2 = self.add_entity('Transition', name=u'tr2') |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
174 |
self.execute('SET X state_of Y WHERE X eid in (%s, %s), Y is CWEType, Y name "Card"' % |
0 | 175 |
(state1.eid, state2.eid)) |
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
176 |
self.execute('SET X state_of Y WHERE X eid in (%s, %s), Y is CWEType, Y name "Bookmark"' % |
0 | 177 |
(state1.eid, state2.eid)) |
178 |
self.execute('SET X transition_of Y WHERE X eid %s, Y name "Card"' % tr1.eid) |
|
179 |
self.execute('SET X transition_of Y WHERE X eid %s, Y name "Bookmark"' % tr2.eid) |
|
180 |
self.execute('SET X allowed_transition Y WHERE X eid %s, Y eid %s' % |
|
181 |
(state1.eid, tr1.eid)) |
|
182 |
self.execute('SET X allowed_transition Y WHERE X eid %s, Y eid %s' % |
|
183 |
(state1.eid, tr2.eid)) |
|
184 |
self.execute('SET X destination_state Y WHERE X eid %s, Y eid %s' % |
|
185 |
(tr1.eid, state2.eid)) |
|
186 |
self.execute('SET X destination_state Y WHERE X eid %s, Y eid %s' % |
|
187 |
(tr2.eid, state2.eid)) |
|
188 |
self.execute('SET X initial_state Y WHERE Y eid %s, X name "Card"' % state1.eid) |
|
189 |
self.execute('SET X initial_state Y WHERE Y eid %s, X name "Bookmark"' % state1.eid) |
|
190 |
card = self.add_entity('Card', title=u't1') |
|
191 |
bookmark = self.add_entity('Bookmark', title=u'111', path=u'/view') |
|
192 |
||
193 |
transitions = list(state1.transitions(card)) |
|
194 |
self.assertEquals(len(transitions), 1) |
|
195 |
self.assertEquals(transitions[0].name, 'tr1') |
|
196 |
transitions = list(state1.transitions(bookmark)) |
|
197 |
self.assertEquals(len(transitions), 1) |
|
198 |
self.assertEquals(transitions[0].name, 'tr2') |
|
199 |
||
200 |
||
201 |
class EmailAddressTC(BaseEntityTC): |
|
202 |
def test_canonical_form(self): |
|
203 |
eid1 = self.execute('INSERT EmailAddress X: X address "maarten.ter.huurne@philips.com"')[0][0] |
|
204 |
eid2 = self.execute('INSERT EmailAddress X: X address "maarten@philips.com", X canonical TRUE')[0][0] |
|
205 |
self.execute('SET X identical_to Y WHERE X eid %s, Y eid %s' % (eid1, eid2)) |
|
206 |
email1 = self.entity('Any X WHERE X eid %(x)s', {'x':eid1}, 'x') |
|
207 |
email2 = self.entity('Any X WHERE X eid %(x)s', {'x':eid2}, 'x') |
|
208 |
self.assertEquals(email1.canonical_form().eid, eid2) |
|
209 |
self.assertEquals(email2.canonical_form(), email2) |
|
210 |
eid3 = self.execute('INSERT EmailAddress X: X address "toto@logilab.fr"')[0][0] |
|
211 |
email3 = self.entity('Any X WHERE X eid %s'%eid3) |
|
212 |
self.assertEquals(email3.canonical_form(), None) |
|
213 |
||
214 |
def test_mangling(self): |
|
215 |
eid = self.execute('INSERT EmailAddress X: X address "maarten.ter.huurne@philips.com"')[0][0] |
|
216 |
email = self.entity('Any X WHERE X eid %(x)s', {'x':eid}, 'x') |
|
217 |
self.assertEquals(email.display_address(), 'maarten.ter.huurne@philips.com') |
|
218 |
self.assertEquals(email.printable_value('address'), 'maarten.ter.huurne@philips.com') |
|
219 |
self.vreg.config.global_set_option('mangle-emails', True) |
|
220 |
self.assertEquals(email.display_address(), 'maarten.ter.huurne at philips dot com') |
|
221 |
self.assertEquals(email.printable_value('address'), 'maarten.ter.huurne at philips dot com') |
|
222 |
eid = self.execute('INSERT EmailAddress X: X address "syt"')[0][0] |
|
223 |
email = self.entity('Any X WHERE X eid %(x)s', {'x':eid}, 'x') |
|
224 |
self.assertEquals(email.display_address(), 'syt') |
|
225 |
self.assertEquals(email.printable_value('address'), 'syt') |
|
226 |
||
227 |
||
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
228 |
class CWUserTC(BaseEntityTC): |
0 | 229 |
|
230 |
def test_complete(self): |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
231 |
e = self.entity('CWUser X WHERE X login "admin"') |
0 | 232 |
e.complete() |
233 |
||
234 |
||
235 |
def test_matching_groups(self): |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
236 |
e = self.entity('CWUser X WHERE X login "admin"') |
0 | 237 |
self.failUnless(e.matching_groups('managers')) |
238 |
self.failIf(e.matching_groups('xyz')) |
|
239 |
self.failUnless(e.matching_groups(('xyz', 'managers'))) |
|
240 |
self.failIf(e.matching_groups(('xyz', 'abcd'))) |
|
241 |
||
242 |
def test_workflow_base(self): |
|
243 |
e = self.create_user('toto') |
|
244 |
self.assertEquals(e.state, 'activated') |
|
245 |
activatedeid = self.execute('State X WHERE X name "activated"')[0][0] |
|
246 |
deactivatedeid = self.execute('State X WHERE X name "deactivated"')[0][0] |
|
247 |
e.change_state(deactivatedeid, u'deactivate 1') |
|
248 |
self.commit() |
|
249 |
e.change_state(activatedeid, u'activate 1') |
|
250 |
self.commit() |
|
251 |
e.change_state(deactivatedeid, u'deactivate 2') |
|
252 |
self.commit() |
|
253 |
# get a fresh user to avoid potential cache issues |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
254 |
e = self.entity('CWUser X WHERE X eid %s' % e.eid) |
0 | 255 |
self.assertEquals([tr.comment for tr in e.reverse_wf_info_for], |
256 |
[None, 'deactivate 1', 'activate 1', 'deactivate 2']) |
|
257 |
self.assertEquals(e.latest_trinfo().comment, 'deactivate 2') |
|
258 |
||
259 |
||
260 |
class InterfaceTC(EnvBasedTC): |
|
261 |
||
262 |
def test_nonregr_subclasses_and_mixins_interfaces(self): |
|
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
263 |
class MyUser(CWUser): |
0 | 264 |
__implements__ = (IMileStone,) |
1358 | 265 |
self.vreg._loadedmods[__name__] = {} |
0 | 266 |
self.vreg.register_vobject_class(MyUser) |
1398
5fe84a5f7035
rename internal entity types to have CW prefix instead of E
sylvain.thenault@logilab.fr
parents:
1358
diff
changeset
|
267 |
self.failUnless(implements(CWUser, IWorkflowable)) |
0 | 268 |
self.failUnless(implements(MyUser, IMileStone)) |
269 |
self.failUnless(implements(MyUser, IWorkflowable)) |
|
270 |
||
271 |
||
272 |
class SpecializedEntityClassesTC(EnvBasedTC): |
|
273 |
||
274 |
def select_eclass(self, etype): |
|
275 |
# clear selector cache |
|
276 |
clear_cache(self.vreg, 'etype_class') |
|
277 |
return self.vreg.etype_class(etype) |
|
278 |
||
279 |
def test_etype_class_selection_and_specialization(self): |
|
280 |
# no specific class for Subdivisions, the default one should be selected |
|
281 |
eclass = self.select_eclass('SubDivision') |
|
282 |
self.failUnless(eclass.__autogenerated__) |
|
283 |
#self.assertEquals(eclass.__bases__, (AnyEntity,)) |
|
284 |
# build class from most generic to most specific and make |
|
285 |
# sure the most specific is always selected |
|
1358 | 286 |
self.vreg._loadedmods[__name__] = {} |
0 | 287 |
for etype in ('Company', 'Division', 'SubDivision'): |
288 |
class Foo(AnyEntity): |
|
289 |
id = etype |
|
290 |
self.vreg.register_vobject_class(Foo) |
|
291 |
eclass = self.select_eclass('SubDivision') |
|
292 |
if etype == 'SubDivision': |
|
293 |
self.failUnless(eclass is Foo) |
|
294 |
else: |
|
295 |
self.failUnless(eclass.__autogenerated__) |
|
296 |
self.assertEquals(eclass.__bases__, (Foo,)) |
|
297 |
# check Division eclass is still selected for plain Division entities |
|
298 |
eclass = self.select_eclass('Division') |
|
299 |
self.assertEquals(eclass.id, 'Division') |
|
300 |
||
301 |
if __name__ == '__main__': |
|
302 |
unittest_main() |