[migration] don't handle data deletion anymore on schema changes
In most cases when we want to drop some entity/relation type, we don't care
whether hooks are called on their deletion. There is even low chances that some
hooks still exists, based on an old version of the schema. Last but not least,
this is horribly inefficient.
So this should be clearly documented and handled by application's programmer if
desired.
This patch removes unnecessary deletion (because table or column will be later
dropped) and reimplements the case of partial deletion (only one relation
definition among several, hence the database structure isn't modified) using
sql.
Only one test regarding deletion of inlined relation def is added as other cases
seem to be covered by existing tests.
Closes #7023315
# copyright 2003-2014 LOGILAB S.A. (Paris, FRANCE), all rights reserved.# contact http://www.logilab.fr/ -- mailto:contact@logilab.fr## This file is part of CubicWeb.## CubicWeb is free software: you can redistribute it and/or modify it under the# terms of the GNU Lesser General Public License as published by the Free# Software Foundation, either version 2.1 of the License, or (at your option)# any later version.## CubicWeb is distributed in the hope that it will be useful, but WITHOUT# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS# FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more# details.## You should have received a copy of the GNU Lesser General Public License along# with CubicWeb. If not, see <http://www.gnu.org/licenses/>."""unit tests for module cubicweb.server.migractions"""fromdatetimeimportdateimportos,os.pathasospfromcontextlibimportcontextmanagerfromlogilab.common.testlibimportunittest_main,Tags,tagfromlogilab.commonimporttempattrfromyams.constraintsimportUniqueConstraintfromcubicwebimportConfigurationError,ValidationError,ExecutionErrorfromcubicweb.devtoolsimportstartpgcluster,stoppgclusterfromcubicweb.devtools.testlibimportCubicWebTCfromcubicweb.server.sqlutilsimportSQL_PREFIXfromcubicweb.server.migractionsimportServerMigrationHelperimportcubicweb.devtoolsHERE=osp.dirname(osp.abspath(__file__))defsetUpModule():startpgcluster(__file__)migrschema=NonedeftearDownModule(*args):globalmigrschemadelmigrschemaifhasattr(MigrationCommandsTC,'origschema'):delMigrationCommandsTC.origschemaifhasattr(MigrationCommandsComputedTC,'origschema'):delMigrationCommandsComputedTC.origschemastoppgcluster(__file__)classMigrationConfig(cubicweb.devtools.TestServerConfiguration):default_sources=cubicweb.devtools.DEFAULT_PSQL_SOURCESCUBES_PATH=cubicweb.devtools.TestServerConfiguration.CUBES_PATH+[osp.join(HERE,'data-migractions','cubes')]classMigrationTC(CubicWebTC):appid='data-migractions'configcls=MigrationConfigtags=CubicWebTC.tags|Tags(('server','migration','migractions'))def_init_repo(self):super(MigrationTC,self)._init_repo()# we have to read schema from the database to get eid for schema entitiesself.repo.set_schema(self.repo.deserialize_schema(),resetvreg=False)# hack to read the schema from data/migrschemaconfig=self.configconfig.appid=osp.join(self.appid,'migratedapp')config._apphome=osp.join(HERE,config.appid)globalmigrschemamigrschema=config.load_schema()config.appid=self.appidconfig._apphome=osp.join(HERE,self.appid)defsetUp(self):self.configcls.cls_adjust_sys_path()super(MigrationTC,self).setUp()deftearDown(self):super(MigrationTC,self).tearDown()self.repo.vreg['etypes'].clear_caches()@contextmanagerdefmh(self):withself.admin_access.repo_cnx()ascnx:yieldcnx,ServerMigrationHelper(self.repo.config,migrschema,repo=self.repo,cnx=cnx,interactive=False)deftable_sql(self,mh,tablename):result=mh.sqlexec("SELECT table_name FROM information_schema.tables WHERE LOWER(table_name)=%(table)s",{'table':tablename.lower()})ifresult:returnresult[0][0]returnNone# no such tabledeftable_schema(self,mh,tablename):result=mh.sqlexec("SELECT column_name, data_type, character_maximum_length FROM information_schema.columns ""WHERE LOWER(table_name) = %(table)s",{'table':tablename.lower()})assertresult,'no table %s'%tablenamereturndict((x[0],(x[1],x[2]))forxinresult)classMigrationCommandsTC(MigrationTC):def_init_repo(self):super(MigrationCommandsTC,self)._init_repo()assert'Folder'inmigrschemadeftest_add_attribute_bool(self):withself.mh()as(cnx,mh):self.assertNotIn('yesno',self.schema)cnx.create_entity('Note')cnx.commit()mh.cmd_add_attribute('Note','yesno')self.assertIn('yesno',self.schema)self.assertEqual(self.schema['yesno'].subjects(),('Note',))self.assertEqual(self.schema['yesno'].objects(),('Boolean',))self.assertEqual(self.schema['Note'].default('yesno'),False)# test default value set on existing entitiesnote=cnx.execute('Note X').get_entity(0,0)self.assertEqual(note.yesno,False)# test default value set for next entitiesself.assertEqual(cnx.create_entity('Note').yesno,False)deftest_add_attribute_int(self):withself.mh()as(cnx,mh):self.assertNotIn('whatever',self.schema)cnx.create_entity('Note')cnx.commit()orderdict=dict(mh.rqlexec('Any RTN, O WHERE X name "Note", RDEF from_entity X, ''RDEF relation_type RT, RDEF ordernum O, RT name RTN'))mh.cmd_add_attribute('Note','whatever')self.assertIn('whatever',self.schema)self.assertEqual(self.schema['whatever'].subjects(),('Note',))self.assertEqual(self.schema['whatever'].objects(),('Int',))self.assertEqual(self.schema['Note'].default('whatever'),0)# test default value set on existing entitiesnote=cnx.execute('Note X').get_entity(0,0)self.assertIsInstance(note.whatever,int)self.assertEqual(note.whatever,0)# test default value set for next entitiesself.assertEqual(cnx.create_entity('Note').whatever,0)# test attribute orderorderdict2=dict(mh.rqlexec('Any RTN, O WHERE X name "Note", RDEF from_entity X, ''RDEF relation_type RT, RDEF ordernum O, RT name RTN'))whateverorder=migrschema['whatever'].rdef('Note','Int').orderfork,vinorderdict.items():ifv>=whateverorder:orderdict[k]=v+1orderdict['whatever']=whateverorderself.assertDictEqual(orderdict,orderdict2)#self.assertEqual([r.type for r in self.schema['Note'].ordered_relations()],# ['modification_date', 'creation_date', 'owned_by',# 'eid', 'ecrit_par', 'inline1', 'date', 'type',# 'whatever', 'date', 'in_basket'])# NB: commit instead of rollback make following test fail with py2.5# this sounds like a pysqlite/2.5 bug (the same eid is affected to# two different entities)deftest_add_attribute_varchar(self):withself.mh()as(cnx,mh):self.assertNotIn('whatever',self.schema)cnx.create_entity('Note')cnx.commit()self.assertNotIn('shortpara',self.schema)mh.cmd_add_attribute('Note','shortpara')self.assertIn('shortpara',self.schema)self.assertEqual(self.schema['shortpara'].subjects(),('Note',))self.assertEqual(self.schema['shortpara'].objects(),('String',))# test created column is actually a varchar(64)fields=self.table_schema(mh,'%sNote'%SQL_PREFIX)self.assertEqual(fields['%sshortpara'%SQL_PREFIX],('character varying',64))# test default value set on existing entitiesself.assertEqual(cnx.execute('Note X').get_entity(0,0).shortpara,'hop')# test default value set for next entitiesself.assertEqual(cnx.create_entity('Note').shortpara,'hop')deftest_add_datetime_with_default_value_attribute(self):withself.mh()as(cnx,mh):self.assertNotIn('mydate',self.schema)self.assertNotIn('oldstyledefaultdate',self.schema)self.assertNotIn('newstyledefaultdate',self.schema)mh.cmd_add_attribute('Note','mydate')mh.cmd_add_attribute('Note','oldstyledefaultdate')mh.cmd_add_attribute('Note','newstyledefaultdate')self.assertIn('mydate',self.schema)self.assertIn('oldstyledefaultdate',self.schema)self.assertIn('newstyledefaultdate',self.schema)self.assertEqual(self.schema['mydate'].subjects(),('Note',))self.assertEqual(self.schema['mydate'].objects(),('Date',))testdate=date(2005,12,13)eid1=mh.rqlexec('INSERT Note N')[0][0]eid2=mh.rqlexec('INSERT Note N: N mydate %(mydate)s',{'mydate':testdate})[0][0]d1=mh.rqlexec('Any D WHERE X eid %(x)s, X mydate D',{'x':eid1})[0][0]d2=mh.rqlexec('Any D WHERE X eid %(x)s, X mydate D',{'x':eid2})[0][0]d3=mh.rqlexec('Any D WHERE X eid %(x)s, X oldstyledefaultdate D',{'x':eid1})[0][0]d4=mh.rqlexec('Any D WHERE X eid %(x)s, X newstyledefaultdate D',{'x':eid1})[0][0]self.assertEqual(d1,date.today())self.assertEqual(d2,testdate)myfavoritedate=date(2013,1,1)self.assertEqual(d3,myfavoritedate)self.assertEqual(d4,myfavoritedate)deftest_drop_chosen_constraints_ctxmanager(self):withself.mh()as(cnx,mh):withmh.cmd_dropped_constraints('Note','unique_id',UniqueConstraint):mh.cmd_add_attribute('Note','unique_id')# make sure the maxsize constraint is not droppedself.assertRaises(ValidationError,mh.rqlexec,'INSERT Note N: N unique_id "xyz"')mh.rollback()# make sure the unique constraint is droppedmh.rqlexec('INSERT Note N: N unique_id "x"')mh.rqlexec('INSERT Note N: N unique_id "x"')mh.rqlexec('DELETE Note N')deftest_drop_required_ctxmanager(self):withself.mh()as(cnx,mh):withmh.cmd_dropped_constraints('Note','unique_id',cstrtype=None,droprequired=True):mh.cmd_add_attribute('Note','unique_id')mh.rqlexec('INSERT Note N')mh.rqlexec('SET N unique_id "x"')# make sure the required=True was restoredself.assertRaises(ValidationError,mh.rqlexec,'INSERT Note N')mh.rollback()deftest_rename_attribute(self):withself.mh()as(cnx,mh):self.assertNotIn('civility',self.schema)eid1=mh.rqlexec('INSERT Personne X: X nom "lui", X sexe "M"')[0][0]eid2=mh.rqlexec('INSERT Personne X: X nom "l\'autre", X sexe NULL')[0][0]mh.cmd_rename_attribute('Personne','sexe','civility')self.assertNotIn('sexe',self.schema)self.assertIn('civility',self.schema)# test data has been backportedc1=mh.rqlexec('Any C WHERE X eid %s, X civility C'%eid1)[0][0]self.assertEqual(c1,'M')c2=mh.rqlexec('Any C WHERE X eid %s, X civility C'%eid2)[0][0]self.assertEqual(c2,None)deftest_workflow_actions(self):withself.mh()as(cnx,mh):wf=mh.cmd_add_workflow(u'foo',('Personne','Email'),ensure_workflowable=False)foretypein('Personne','Email'):s1=mh.rqlexec('Any N WHERE WF workflow_of ET, ET name "%s", WF name N'%etype)[0][0]self.assertEqual(s1,"foo")s1=mh.rqlexec('Any N WHERE ET default_workflow WF, ET name "%s", WF name N'%etype)[0][0]self.assertEqual(s1,"foo")deftest_add_entity_type(self):withself.mh()as(cnx,mh):self.assertNotIn('Folder2',self.schema)self.assertNotIn('filed_under2',self.schema)mh.cmd_add_entity_type('Folder2')self.assertIn('Folder2',self.schema)self.assertIn('Old',self.schema)self.assertTrue(cnx.execute('CWEType X WHERE X name "Folder2"'))self.assertIn('filed_under2',self.schema)self.assertTrue(cnx.execute('CWRType X WHERE X name "filed_under2"'))self.assertEqual(sorted(str(rs)forrsinself.schema['Folder2'].subject_relations()),['created_by','creation_date','cw_source','cwuri','description','description_format','eid','filed_under2','has_text','identity','in_basket','is','is_instance_of','modification_date','name','owned_by'])self.assertCountEqual([str(rs)forrsinself.schema['Folder2'].object_relations()],['filed_under2','identity'])# Old will be missing as it has been renamed into 'New' in the migrated# schema while New hasn't been added here.self.assertEqual(sorted(str(e)foreinself.schema['filed_under2'].subjects()),sorted(str(e)foreinself.schema.entities()ifnote.finalande!='Old'))self.assertEqual(self.schema['filed_under2'].objects(),('Folder2',))eschema=self.schema.eschema('Folder2')forcstrineschema.rdef('name').constraints:self.assertTrue(hasattr(cstr,'eid'))deftest_add_drop_entity_type(self):withself.mh()as(cnx,mh):mh.cmd_add_entity_type('Folder2')wf=mh.cmd_add_workflow(u'folder2 wf','Folder2',ensure_workflowable=False)todo=wf.add_state(u'todo',initial=True)done=wf.add_state(u'done')wf.add_transition(u'redoit',done,todo)wf.add_transition(u'markasdone',todo,done)cnx.commit()eschema=self.schema.eschema('Folder2')mh.cmd_drop_entity_type('Folder2')self.assertNotIn('Folder2',self.schema)self.assertFalse(cnx.execute('CWEType X WHERE X name "Folder2"'))# test automatic workflow deletionself.assertFalse(cnx.execute('Workflow X WHERE NOT X workflow_of ET'))self.assertFalse(cnx.execute('State X WHERE NOT X state_of WF'))self.assertFalse(cnx.execute('Transition X WHERE NOT X transition_of WF'))deftest_rename_entity_type(self):withself.mh()as(cnx,mh):entity=mh.create_entity('Old',name=u'old')self.repo.type_and_source_from_eid(entity.eid,entity._cw)mh.cmd_rename_entity_type('Old','New')mh.cmd_rename_attribute('New','name','new_name')deftest_add_drop_relation_type(self):withself.mh()as(cnx,mh):mh.cmd_add_entity_type('Folder2',auto=False)mh.cmd_add_relation_type('filed_under2')self.assertIn('filed_under2',self.schema)# Old will be missing as it has been renamed into 'New' in the migrated# schema while New hasn't been added here.self.assertEqual(sorted(str(e)foreinself.schema['filed_under2'].subjects()),sorted(str(e)foreinself.schema.entities()ifnote.finalande!='Old'))self.assertEqual(self.schema['filed_under2'].objects(),('Folder2',))mh.cmd_drop_relation_type('filed_under2')self.assertNotIn('filed_under2',self.schema)# this should not crashmh.cmd_drop_relation_type('filed_under2')deftest_add_relation_definition_nortype(self):withself.mh()as(cnx,mh):mh.cmd_add_relation_definition('Personne','concerne2','Affaire')self.assertEqual(self.schema['concerne2'].subjects(),('Personne',))self.assertEqual(self.schema['concerne2'].objects(),('Affaire',))self.assertEqual(self.schema['concerne2'].rdef('Personne','Affaire').cardinality,'1*')mh.cmd_add_relation_definition('Personne','concerne2','Note')self.assertEqual(sorted(self.schema['concerne2'].objects()),['Affaire','Note'])mh.create_entity('Personne',nom=u'tot')mh.create_entity('Affaire')mh.rqlexec('SET X concerne2 Y WHERE X is Personne, Y is Affaire')cnx.commit()mh.cmd_drop_relation_definition('Personne','concerne2','Affaire')self.assertIn('concerne2',self.schema)mh.cmd_drop_relation_definition('Personne','concerne2','Note')self.assertNotIn('concerne2',self.schema)deftest_drop_relation_definition_existant_rtype(self):withself.mh()as(cnx,mh):self.assertEqual(sorted(str(e)foreinself.schema['concerne'].subjects()),['Affaire','Personne'])self.assertEqual(sorted(str(e)foreinself.schema['concerne'].objects()),['Affaire','Division','Note','Societe','SubDivision'])mh.cmd_drop_relation_definition('Personne','concerne','Affaire')self.assertEqual(sorted(str(e)foreinself.schema['concerne'].subjects()),['Affaire'])self.assertEqual(sorted(str(e)foreinself.schema['concerne'].objects()),['Division','Note','Societe','SubDivision'])mh.cmd_add_relation_definition('Personne','concerne','Affaire')self.assertEqual(sorted(str(e)foreinself.schema['concerne'].subjects()),['Affaire','Personne'])self.assertEqual(sorted(str(e)foreinself.schema['concerne'].objects()),['Affaire','Division','Note','Societe','SubDivision'])# trick: overwrite self.maxeid to avoid deletion of just reintroduced typesself.maxeid=cnx.execute('Any MAX(X)')[0][0]deftest_drop_relation_definition_with_specialization(self):withself.mh()as(cnx,mh):self.assertEqual(sorted(str(e)foreinself.schema['concerne'].subjects()),['Affaire','Personne'])self.assertEqual(sorted(str(e)foreinself.schema['concerne'].objects()),['Affaire','Division','Note','Societe','SubDivision'])mh.cmd_drop_relation_definition('Affaire','concerne','Societe')self.assertEqual(sorted(str(e)foreinself.schema['concerne'].subjects()),['Affaire','Personne'])self.assertEqual(sorted(str(e)foreinself.schema['concerne'].objects()),['Affaire','Note'])mh.cmd_add_relation_definition('Affaire','concerne','Societe')self.assertEqual(sorted(str(e)foreinself.schema['concerne'].subjects()),['Affaire','Personne'])self.assertEqual(sorted(str(e)foreinself.schema['concerne'].objects()),['Affaire','Division','Note','Societe','SubDivision'])# trick: overwrite self.maxeid to avoid deletion of just reintroduced typesself.maxeid=cnx.execute('Any MAX(X)')[0][0]deftest_rename_relation(self):self.skipTest('implement me')deftest_change_relation_props_non_final(self):withself.mh()as(cnx,mh):rschema=self.schema['concerne']card=rschema.rdef('Affaire','Societe').cardinalityself.assertEqual(card,'**')try:mh.cmd_change_relation_props('Affaire','concerne','Societe',cardinality='?*')card=rschema.rdef('Affaire','Societe').cardinalityself.assertEqual(card,'?*')finally:mh.cmd_change_relation_props('Affaire','concerne','Societe',cardinality='**')deftest_change_relation_props_final(self):withself.mh()as(cnx,mh):rschema=self.schema['adel']card=rschema.rdef('Personne','String').fulltextindexedself.assertEqual(card,False)try:mh.cmd_change_relation_props('Personne','adel','String',fulltextindexed=True)card=rschema.rdef('Personne','String').fulltextindexedself.assertEqual(card,True)finally:mh.cmd_change_relation_props('Personne','adel','String',fulltextindexed=False)deftest_sync_schema_props_perms_rqlconstraints(self):withself.mh()as(cnx,mh):# Drop one of the RQLConstraint.rdef=self.schema['evaluee'].rdefs[('Personne','Note')]oldconstraints=rdef.constraintsself.assertIn('S created_by U',[cstr.expressionforcstrinoldconstraints])mh.cmd_sync_schema_props_perms('evaluee',commit=True)newconstraints=rdef.constraintsself.assertNotIn('S created_by U',[cstr.expressionforcstrinnewconstraints])# Drop all RQLConstraint.rdef=self.schema['travaille'].rdefs[('Personne','Societe')]oldconstraints=rdef.constraintsself.assertEqual(len(oldconstraints),2)mh.cmd_sync_schema_props_perms('travaille',commit=True)rdef=self.schema['travaille'].rdefs[('Personne','Societe')]newconstraints=rdef.constraintsself.assertEqual(len(newconstraints),0)@tag('longrun')deftest_sync_schema_props_perms(self):withself.mh()as(cnx,mh):nbrqlexpr_start=cnx.execute('Any COUNT(X) WHERE X is RQLExpression')[0][0]migrschema['titre'].rdefs[('Personne','String')].order=7migrschema['adel'].rdefs[('Personne','String')].order=6migrschema['ass'].rdefs[('Personne','String')].order=5migrschema['Personne'].description='blabla bla'migrschema['titre'].description='usually a title'migrschema['titre'].rdefs[('Personne','String')].description='title for this person'delete_concerne_rqlexpr=self._rrqlexpr_rset(cnx,'delete','concerne')add_concerne_rqlexpr=self._rrqlexpr_rset(cnx,'add','concerne')# make sure properties (e.g. etype descriptions) are synced by the# second call to sync_schemamh.cmd_sync_schema_props_perms(syncprops=False,commit=False)mh.cmd_sync_schema_props_perms(commit=False)self.assertEqual(cnx.execute('Any D WHERE X name "Personne", X description D')[0][0],'blabla bla')self.assertEqual(cnx.execute('Any D WHERE X name "titre", X description D')[0][0],'usually a title')self.assertEqual(cnx.execute('Any D WHERE X relation_type RT, RT name "titre",''X from_entity FE, FE name "Personne",''X description D')[0][0],'title for this person')rinorder=[nforn,incnx.execute('Any N ORDERBY O,N WHERE X is CWAttribute, X relation_type RT, RT name N,''X from_entity FE, FE name "Personne",''X ordernum O')]expected=[u'nom',u'prenom',u'sexe',u'promo',u'ass',u'adel',u'titre',u'web',u'tel',u'fax',u'datenaiss',u'test',u'tzdatenaiss',u'description',u'firstname',u'creation_date',u'cwuri',u'modification_date']self.assertEqual(expected,rinorder)# test permissions synchronization ##################################### new rql expr to add note entityeexpr=self._erqlexpr_entity(cnx,'add','Note')self.assertEqual(eexpr.expression,'X ecrit_part PE, U in_group G, ''PE require_permission P, P name "add_note", P require_group G')self.assertEqual([et.nameforetineexpr.reverse_add_permission],['Note'])self.assertEqual(eexpr.reverse_read_permission,())self.assertEqual(eexpr.reverse_delete_permission,())self.assertEqual(eexpr.reverse_update_permission,())self.assertTrue(self._rrqlexpr_rset(cnx,'add','para'))# no rqlexpr to delete para attributeself.assertFalse(self._rrqlexpr_rset(cnx,'delete','para'))# new rql expr to add ecrit_par relationrexpr=self._rrqlexpr_entity(cnx,'add','ecrit_par')self.assertEqual(rexpr.expression,'O require_permission P, P name "add_note", ''U in_group G, P require_group G')self.assertEqual([rdef.rtype.nameforrdefinrexpr.reverse_add_permission],['ecrit_par'])self.assertEqual(rexpr.reverse_read_permission,())self.assertEqual(rexpr.reverse_delete_permission,())# no more rqlexpr to delete and add travaille relationself.assertFalse(self._rrqlexpr_rset(cnx,'add','travaille'))self.assertFalse(self._rrqlexpr_rset(cnx,'delete','travaille'))# no more rqlexpr to delete and update Societe entityself.assertFalse(self._erqlexpr_rset(cnx,'update','Societe'))self.assertFalse(self._erqlexpr_rset(cnx,'delete','Societe'))# no more rqlexpr to read Affaire entityself.assertFalse(self._erqlexpr_rset(cnx,'read','Affaire'))# rqlexpr to update Affaire entity has been updatedeexpr=self._erqlexpr_entity(cnx,'update','Affaire')self.assertEqual(eexpr.expression,'X concerne S, S owned_by U')# no change for rqlexpr to add and delete Affaire entityself.assertEqual(len(self._erqlexpr_rset(cnx,'delete','Affaire')),1)self.assertEqual(len(self._erqlexpr_rset(cnx,'add','Affaire')),1)# no change for rqlexpr to add and delete concerne relationself.assertEqual(len(self._rrqlexpr_rset(cnx,'delete','concerne')),len(delete_concerne_rqlexpr))self.assertEqual(len(self._rrqlexpr_rset(cnx,'add','concerne')),len(add_concerne_rqlexpr))# * migrschema involve:# * 7 erqlexprs deletions (2 in (Affaire + Societe + Note.para) + 1 Note.something# * 2 rrqlexprs deletions (travaille)# * 1 update (Affaire update)# * 2 new (Note add, ecrit_par add)# * 2 implicit new for attributes (Note.para, Person.test)# remaining orphan rql expr which should be deleted at commit (composite relation)# unattached expressions -> pending deletion on commitself.assertEqual(cnx.execute('Any COUNT(X) WHERE X is RQLExpression, X exprtype "ERQLExpression",''NOT ET1 read_permission X, NOT ET2 add_permission X, ''NOT ET3 delete_permission X, NOT ET4 update_permission X')[0][0],7)self.assertEqual(cnx.execute('Any COUNT(X) WHERE X is RQLExpression, X exprtype "RRQLExpression",''NOT ET1 read_permission X, NOT ET2 add_permission X, ''NOT ET3 delete_permission X, NOT ET4 update_permission X')[0][0],2)# finallyself.assertEqual(cnx.execute('Any COUNT(X) WHERE X is RQLExpression')[0][0],nbrqlexpr_start+1+2+2+2)cnx.commit()# unique_together testself.assertEqual(len(self.schema.eschema('Personne')._unique_together),1)self.assertCountEqual(self.schema.eschema('Personne')._unique_together[0],('nom','prenom','datenaiss'))rset=cnx.execute('Any C WHERE C is CWUniqueTogetherConstraint, C constraint_of ET, ET name "Personne"')self.assertEqual(len(rset),1)relations=[r.nameforrinrset.get_entity(0,0).relations]self.assertCountEqual(relations,('nom','prenom','datenaiss'))def_erqlexpr_rset(self,cnx,action,ertype):rql='RQLExpression X WHERE ET is CWEType, ET %s_permission X, ET name %%(name)s'%actionreturncnx.execute(rql,{'name':ertype})def_erqlexpr_entity(self,cnx,action,ertype):rset=self._erqlexpr_rset(cnx,action,ertype)self.assertEqual(len(rset),1)returnrset.get_entity(0,0)def_rrqlexpr_rset(self,cnx,action,ertype):rql='RQLExpression X WHERE RT is CWRType, RDEF %s_permission X, RT name %%(name)s, RDEF relation_type RT'%actionreturncnx.execute(rql,{'name':ertype})def_rrqlexpr_entity(self,cnx,action,ertype):rset=self._rrqlexpr_rset(cnx,action,ertype)self.assertEqual(len(rset),1)returnrset.get_entity(0,0)deftest_set_size_constraint(self):withself.mh()as(cnx,mh):# existing previous valuetry:mh.cmd_set_size_constraint('CWEType','name',128)finally:mh.cmd_set_size_constraint('CWEType','name',64)# non existing previous valuetry:mh.cmd_set_size_constraint('CWEType','description',256)finally:mh.cmd_set_size_constraint('CWEType','description',None)@tag('longrun')deftest_add_drop_cube_and_deps(self):withself.mh()as(cnx,mh):schema=self.repo.schemaself.assertEqual(sorted((str(s),str(o))fors,oinschema['see_also'].rdefs),sorted([('EmailThread','EmailThread'),('Folder','Folder'),('Bookmark','Bookmark'),('Bookmark','Note'),('Note','Note'),('Note','Bookmark')]))try:mh.cmd_drop_cube('fakeemail',removedeps=True)# file was there because it's an email dependancy, should have been removedself.assertNotIn('fakeemail',self.config.cubes())self.assertNotIn(self.config.cube_dir('fakeemail'),self.config.cubes_path())self.assertNotIn('file',self.config.cubes())self.assertNotIn(self.config.cube_dir('file'),self.config.cubes_path())forertypein('Email','EmailThread','EmailPart','File','sender','in_thread','reply_to','data_format'):self.assertNotIn(ertype,schema)self.assertEqual(sorted(schema['see_also'].rdefs),sorted([('Folder','Folder'),('Bookmark','Bookmark'),('Bookmark','Note'),('Note','Note'),('Note','Bookmark')]))self.assertEqual(sorted(schema['see_also'].subjects()),['Bookmark','Folder','Note'])self.assertEqual(sorted(schema['see_also'].objects()),['Bookmark','Folder','Note'])self.assertEqual(cnx.execute('Any X WHERE X pkey "system.version.fakeemail"').rowcount,0)self.assertEqual(cnx.execute('Any X WHERE X pkey "system.version.file"').rowcount,0)finally:mh.cmd_add_cube('fakeemail')self.assertIn('fakeemail',self.config.cubes())self.assertIn(self.config.cube_dir('fakeemail'),self.config.cubes_path())self.assertIn('file',self.config.cubes())self.assertIn(self.config.cube_dir('file'),self.config.cubes_path())forertypein('Email','EmailThread','EmailPart','File','sender','in_thread','reply_to','data_format'):self.assertIn(ertype,schema)self.assertEqual(sorted(schema['see_also'].rdefs),sorted([('EmailThread','EmailThread'),('Folder','Folder'),('Bookmark','Bookmark'),('Bookmark','Note'),('Note','Note'),('Note','Bookmark')]))self.assertEqual(sorted(schema['see_also'].subjects()),['Bookmark','EmailThread','Folder','Note'])self.assertEqual(sorted(schema['see_also'].objects()),['Bookmark','EmailThread','Folder','Note'])fromcubes.fakeemail.__pkginfo__importversionasemail_versionfromcubes.file.__pkginfo__importversionasfile_versionself.assertEqual(cnx.execute('Any V WHERE X value V, X pkey "system.version.fakeemail"')[0][0],email_version)self.assertEqual(cnx.execute('Any V WHERE X value V, X pkey "system.version.file"')[0][0],file_version)# trick: overwrite self.maxeid to avoid deletion of just reintroduced# types (and their associated tables!)self.maxeid=cnx.execute('Any MAX(X)')[0][0]# why this commit is necessary is unclear to me (though without it# next test may fail complaining of missing tablescnx.commit()@tag('longrun')deftest_add_drop_cube_no_deps(self):withself.mh()as(cnx,mh):cubes=set(self.config.cubes())schema=self.repo.schematry:mh.cmd_drop_cube('fakeemail')cubes.remove('fakeemail')self.assertNotIn('fakeemail',self.config.cubes())self.assertIn('file',self.config.cubes())forertypein('Email','EmailThread','EmailPart','sender','in_thread','reply_to'):self.assertNotIn(ertype,schema)finally:mh.cmd_add_cube('fakeemail')self.assertIn('fakeemail',self.config.cubes())# trick: overwrite self.maxeid to avoid deletion of just reintroduced# types (and their associated tables!)self.maxeid=cnx.execute('Any MAX(X)')[0][0]# XXXXXXX KILL KENNY# why this commit is necessary is unclear to me (though without it# next test may fail complaining of missing tablescnx.commit()deftest_drop_dep_cube(self):withself.mh()as(cnx,mh):withself.assertRaises(ConfigurationError)ascm:mh.cmd_drop_cube('file')self.assertEqual(str(cm.exception),"can't remove cube file, used as a dependency")@tag('longrun')deftest_introduce_base_class(self):withself.mh()as(cnx,mh):mh.cmd_add_entity_type('Para')self.assertEqual(sorted(et.typeforetinself.schema['Para'].specialized_by()),['Note'])self.assertEqual(self.schema['Note'].specializes().type,'Para')mh.cmd_add_entity_type('Text')self.assertEqual(sorted(et.typeforetinself.schema['Para'].specialized_by()),['Note','Text'])self.assertEqual(self.schema['Text'].specializes().type,'Para')# test columns have been actually addedtext=cnx.execute('INSERT Text X: X para "hip", X summary "hop", X newattr "momo"').get_entity(0,0)note=cnx.execute('INSERT Note X: X para "hip", X shortpara "hop", X newattr "momo", X unique_id "x"').get_entity(0,0)aff=cnx.execute('INSERT Affaire X').get_entity(0,0)self.assertTrue(cnx.execute('SET X newnotinlined Y WHERE X eid %(x)s, Y eid %(y)s',{'x':text.eid,'y':aff.eid}))self.assertTrue(cnx.execute('SET X newnotinlined Y WHERE X eid %(x)s, Y eid %(y)s',{'x':note.eid,'y':aff.eid}))self.assertTrue(cnx.execute('SET X newinlined Y WHERE X eid %(x)s, Y eid %(y)s',{'x':text.eid,'y':aff.eid}))self.assertTrue(cnx.execute('SET X newinlined Y WHERE X eid %(x)s, Y eid %(y)s',{'x':note.eid,'y':aff.eid}))# XXX remove specializes by ourselves, else tearDown fails when removing# Para because of Note inheritance. This could be fixed by putting the# MemSchemaCWETypeDel(session, name) operation in the# after_delete_entity(CWEType) hook, since in that case the MemSchemaSpecializesDel# operation would be removed before, but I'm not sure this is a desired behaviour.## also we need more tests about introducing/removing base classes or# specialization relationship...cnx.execute('DELETE X specializes Y WHERE Y name "Para"')cnx.commit()self.assertEqual(sorted(et.typeforetinself.schema['Para'].specialized_by()),[])self.assertEqual(self.schema['Note'].specializes(),None)self.assertEqual(self.schema['Text'].specializes(),None)deftest_add_symmetric_relation_type(self):withself.mh()as(cnx,mh):self.assertFalse(self.table_sql(mh,'same_as_relation'))mh.cmd_add_relation_type('same_as')self.assertTrue(self.table_sql(mh,'same_as_relation'))deftest_change_attribute_type(self):withself.mh()as(cnx,mh):mh.cmd_create_entity('Societe',tel=1)mh.commit()mh.change_attribute_type('Societe','tel','Float')self.assertNotIn(('Societe','Int'),self.schema['tel'].rdefs)self.assertIn(('Societe','Float'),self.schema['tel'].rdefs)self.assertEqual(self.schema['tel'].rdefs[('Societe','Float')].object,'Float')tel=mh.rqlexec('Any T WHERE X tel T')[0][0]self.assertEqual(tel,1.0)self.assertIsInstance(tel,float)deftest_drop_required_inlined_relation(self):withself.mh()as(cnx,mh):bob=mh.cmd_create_entity('Personne',nom=u'bob')note=mh.cmd_create_entity('Note',ecrit_par=bob)mh.commit()rdef=mh.fs_schema.rschema('ecrit_par').rdefs[('Note','Personne')]withtempattr(rdef,'cardinality','1*'):mh.sync_schema_props_perms('ecrit_par',syncperms=False)mh.cmd_drop_relation_type('ecrit_par')self.assertNotIn('%secrit_par'%SQL_PREFIX,self.table_schema(mh,'%sPersonne'%SQL_PREFIX))deftest_drop_inlined_rdef_delete_data(self):withself.mh()as(cnx,mh):note=mh.cmd_create_entity('Note',ecrit_par=cnx.user.eid)mh.commit()mh.drop_relation_definition('Note','ecrit_par','CWUser')self.assertFalse(mh.sqlexec('SELECT * FROM cw_Note WHERE cw_ecrit_par IS NOT NULL'))classMigrationCommandsComputedTC(MigrationTC):""" Unit tests for computed relations and attributes """appid='datacomputed'defsetUp(self):MigrationTC.setUp(self)# ensure vregistry is reloaded, needed by generated hooks for computed# attributesself.repo.vreg.set_schema(self.repo.schema)deftest_computed_relation_add_relation_definition(self):self.assertNotIn('works_for',self.schema)withself.mh()as(cnx,mh):withself.assertRaises(ExecutionError)asexc:mh.cmd_add_relation_definition('Employee','works_for','Company')self.assertEqual(str(exc.exception),'Cannot add a relation definition for a computed ''relation (works_for)')deftest_computed_relation_drop_relation_definition(self):self.assertIn('notes',self.schema)withself.mh()as(cnx,mh):withself.assertRaises(ExecutionError)asexc:mh.cmd_drop_relation_definition('Company','notes','Note')self.assertEqual(str(exc.exception),'Cannot drop a relation definition for a computed ''relation (notes)')deftest_computed_relation_add_relation_type(self):self.assertNotIn('works_for',self.schema)withself.mh()as(cnx,mh):mh.cmd_add_relation_type('works_for')self.assertIn('works_for',self.schema)self.assertEqual(self.schema['works_for'].rule,'O employees S, NOT EXISTS (O associates S)')self.assertEqual(self.schema['works_for'].objects(),('Company',))self.assertEqual(self.schema['works_for'].subjects(),('Employee',))self.assertFalse(self.table_sql(mh,'works_for_relation'))e=cnx.create_entity('Employee')a=cnx.create_entity('Employee')cnx.create_entity('Company',employees=e,associates=a)cnx.commit()company=cnx.execute('Company X').get_entity(0,0)self.assertEqual([e.eid],[x.eidforxincompany.reverse_works_for])mh.rollback()deftest_computed_relation_drop_relation_type(self):self.assertIn('notes',self.schema)withself.mh()as(cnx,mh):mh.cmd_drop_relation_type('notes')self.assertNotIn('notes',self.schema)deftest_computed_relation_sync_schema_props_perms(self):self.assertIn('whatever',self.schema)withself.mh()as(cnx,mh):mh.cmd_sync_schema_props_perms('whatever')self.assertEqual(self.schema['whatever'].rule,'S employees E, O associates E')self.assertEqual(self.schema['whatever'].objects(),('Company',))self.assertEqual(self.schema['whatever'].subjects(),('Company',))self.assertFalse(self.table_sql(mh,'whatever_relation'))deftest_computed_relation_sync_schema_props_perms_security(self):withself.mh()as(cnx,mh):rdef=next(iter(self.schema['perm_changes'].rdefs.values()))self.assertEqual(rdef.permissions,{'add':(),'delete':(),'read':('managers','users')})mh.cmd_sync_schema_props_perms('perm_changes')self.assertEqual(self.schema['perm_changes'].permissions,{'read':('managers',)})rdef=next(iter(self.schema['perm_changes'].rdefs.values()))self.assertEqual(rdef.permissions,{'add':(),'delete':(),'read':('managers',)})deftest_computed_relation_sync_schema_props_perms_on_rdef(self):self.assertIn('whatever',self.schema)withself.mh()as(cnx,mh):withself.assertRaises(ExecutionError)asexc:mh.cmd_sync_schema_props_perms(('Company','whatever','Person'))self.assertEqual(str(exc.exception),'Cannot synchronize a relation definition for a computed ''relation (whatever)')deftest_computed_relation_rename_relation_type(self):withself.mh()as(cnx,mh):mh.cmd_rename_relation_type('to_be_renamed','renamed')self.assertIn('renamed',self.schema)self.assertNotIn('to_be_renamed',self.schema)# computed attributes migration ############################################defsetup_add_score(self):withself.admin_access.client_cnx()ascnx:assertnotcnx.execute('Company X')c=cnx.create_entity('Company')e1=cnx.create_entity('Employee',reverse_employees=c)cnx.create_entity('Note',note=2,concerns=e1)e2=cnx.create_entity('Employee',reverse_employees=c)cnx.create_entity('Note',note=4,concerns=e2)cnx.commit()defassert_score_initialized(self,mh):self.assertEqual(self.schema['score'].rdefs['Company','Float'].formula,'Any AVG(NN) WHERE X employees E, N concerns E, N note NN')fields=self.table_schema(mh,'%sCompany'%SQL_PREFIX)self.assertEqual(fields['%sscore'%SQL_PREFIX],('double precision',None))self.assertEqual([[3.0]],mh.rqlexec('Any CS WHERE C score CS, C is Company').rows)deftest_computed_attribute_add_relation_type(self):self.assertNotIn('score',self.schema)self.setup_add_score()withself.mh()as(cnx,mh):mh.cmd_add_relation_type('score')self.assertIn('score',self.schema)self.assertEqual(self.schema['score'].objects(),('Float',))self.assertEqual(self.schema['score'].subjects(),('Company',))self.assert_score_initialized(mh)deftest_computed_attribute_add_attribute(self):self.assertNotIn('score',self.schema)self.setup_add_score()withself.mh()as(cnx,mh):mh.cmd_add_attribute('Company','score')self.assertIn('score',self.schema)self.assert_score_initialized(mh)defassert_computed_attribute_dropped(self):self.assertNotIn('note20',self.schema)withself.mh()as(cnx,mh):fields=self.table_schema(mh,'%sNote'%SQL_PREFIX)self.assertNotIn('%snote20'%SQL_PREFIX,fields)deftest_computed_attribute_drop_type(self):self.assertIn('note20',self.schema)withself.mh()as(cnx,mh):mh.cmd_drop_relation_type('note20')self.assert_computed_attribute_dropped()deftest_computed_attribute_drop_relation_definition(self):self.assertIn('note20',self.schema)withself.mh()as(cnx,mh):mh.cmd_drop_relation_definition('Note','note20','Int')self.assert_computed_attribute_dropped()deftest_computed_attribute_drop_attribute(self):self.assertIn('note20',self.schema)withself.mh()as(cnx,mh):mh.cmd_drop_attribute('Note','note20')self.assert_computed_attribute_dropped()deftest_computed_attribute_sync_schema_props_perms_rtype(self):self.assertIn('note100',self.schema)withself.mh()as(cnx,mh):mh.cmd_sync_schema_props_perms('note100')self.assertEqual(self.schema['note100'].rdefs['Note','Int'].formula,'Any N*100 WHERE X note N')deftest_computed_attribute_sync_schema_props_perms_rdef(self):self.setup_add_score()withself.mh()as(cnx,mh):mh.cmd_sync_schema_props_perms(('Note','note100','Int'))self.assertEqual([[200],[400]],cnx.execute('Any N ORDERBY N WHERE X note100 N').rows)self.assertEqual([[300]],cnx.execute('Any CS WHERE C score100 CS, C is Company').rows)if__name__=='__main__':unittest_main()