added stateful package for assigning simple workflows to loops objects
git-svn-id: svn://svn.cy55.de/Zope3/src/loops/trunk@2467 fd906abe-77d9-0310-91a1-e0d9ade77398
This commit is contained in:
		
							parent
							
								
									64acbd439d
								
							
						
					
					
						commit
						b7097a09c5
					
				
					 8 changed files with 168 additions and 6 deletions
				
			
		|  | @ -118,10 +118,10 @@ So we are now ready to query the favorites. | ||||||
| 
 | 
 | ||||||
|   >>> favs = favorites.query(userName=johnCId) |   >>> favs = favorites.query(userName=johnCId) | ||||||
|   >>> favs |   >>> favs | ||||||
|   [<Favorite ['27', 1, '33', '...']: {}>] |   [<Favorite ['29', 1, '35', '...']: {}>] | ||||||
| 
 | 
 | ||||||
|   >>> list(favAdapted.list(johnC)) |   >>> list(favAdapted.list(johnC)) | ||||||
|   ['27'] |   ['29'] | ||||||
| 
 | 
 | ||||||
|   >>> util.getObjectForUid(favs[0].taskId) is resources['d001.txt'] |   >>> util.getObjectForUid(favs[0].taskId) is resources['d001.txt'] | ||||||
|   True |   True | ||||||
|  |  | ||||||
|  | @ -27,7 +27,7 @@ from zope.interface import implements, Interface | ||||||
| 
 | 
 | ||||||
| from loops.concept import Concept | from loops.concept import Concept | ||||||
| from loops.interfaces import ITypeConcept | from loops.interfaces import ITypeConcept | ||||||
| from loops.organize.interfaces import IPerson | from loops.organize.interfaces import IPerson, ITask | ||||||
| from loops.setup import SetupManager as BaseSetupManager | from loops.setup import SetupManager as BaseSetupManager | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -46,3 +46,6 @@ class SetupManager(BaseSetupManager): | ||||||
| 
 | 
 | ||||||
|         owner = self.addObject(concepts, Concept, 'ownedby', title=u'owned by', |         owner = self.addObject(concepts, Concept, 'ownedby', title=u'owned by', | ||||||
|                         conceptType=predicate) |                         conceptType=predicate) | ||||||
|  | 
 | ||||||
|  |         task = self.addAndConfigureObject(concepts, Concept, 'task', title=u'Task', | ||||||
|  |                         conceptType=type, typeInterface=ITask) | ||||||
|  |  | ||||||
							
								
								
									
										59
									
								
								stateful/README.txt
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								stateful/README.txt
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,59 @@ | ||||||
|  | =============================================================== | ||||||
|  | loops - Linked Objects for Organization and Processing Services | ||||||
|  | =============================================================== | ||||||
|  | 
 | ||||||
|  |   ($Id$) | ||||||
|  | 
 | ||||||
|  |   >>> from zope import component | ||||||
|  |   >>> from zope.traversing.api import getName | ||||||
|  | 
 | ||||||
|  | First we set up a loops site with basic and example concepts and resources. | ||||||
|  | 
 | ||||||
|  |   >>> from zope.app.testing.setup import placefulSetUp, placefulTearDown | ||||||
|  |   >>> site = placefulSetUp(True) | ||||||
|  | 
 | ||||||
|  |   >>> from loops.organize.setup import SetupManager | ||||||
|  |   >>> component.provideAdapter(SetupManager, name='organize') | ||||||
|  |   >>> from loops.tests.setup import TestSite | ||||||
|  |   >>> t = TestSite(site) | ||||||
|  |   >>> concepts, resources, views = t.setup() | ||||||
|  |   >>> loopsRoot = site['loops'] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | Stateful Objects | ||||||
|  | ================ | ||||||
|  | 
 | ||||||
|  | Let's start with registering the states definitions and adapters needed. | ||||||
|  | The states definition (aka 'workflow') is registered as a utility; for | ||||||
|  | making an object statful we'll use an adapter. | ||||||
|  | 
 | ||||||
|  |   >>> from cybertools.stateful.interfaces import IStatesDefinition, IStateful | ||||||
|  |   >>> from cybertools.stateful.publishing import simplePublishing | ||||||
|  |   >>> component.provideUtility(simplePublishing, IStatesDefinition, | ||||||
|  |   ...                          name='loops.simple_publishing') | ||||||
|  | 
 | ||||||
|  |   >>> from loops.stateful.base import SimplePublishable | ||||||
|  |   >>> component.provideAdapter(SimplePublishable, name='loops.simple_publishing') | ||||||
|  | 
 | ||||||
|  | We may now take a document and adapt it to IStateful so that we may | ||||||
|  | check the document's state and perform transitions to other states. | ||||||
|  | 
 | ||||||
|  |   >>> doc01 = resources['d001.txt'] | ||||||
|  |   >>> statefulDoc01 = component.getAdapter(doc01, IStateful, | ||||||
|  |   ...                                      name='loops.simple_publishing') | ||||||
|  | 
 | ||||||
|  |   >>> statefulDoc01.state | ||||||
|  |   'draft' | ||||||
|  | 
 | ||||||
|  |   >>> statefulDoc01.doTransition('publish') | ||||||
|  |   >>> statefulDoc01.state | ||||||
|  |   'published' | ||||||
|  | 
 | ||||||
|  | Let's check if the state is really stored in the underlying object and | ||||||
|  | not just kept in the adapter. | ||||||
|  | 
 | ||||||
|  |   >>> statefulDoc01_x = component.getAdapter(doc01, IStateful, | ||||||
|  |   ...                                        name='loops.simple_publishing') | ||||||
|  | 
 | ||||||
|  |   >>> statefulDoc01.state | ||||||
|  |   'published' | ||||||
							
								
								
									
										3
									
								
								stateful/__init__.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								stateful/__init__.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,3 @@ | ||||||
|  | """ | ||||||
|  | $Id$ | ||||||
|  | """ | ||||||
							
								
								
									
										47
									
								
								stateful/base.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								stateful/base.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,47 @@ | ||||||
|  | # | ||||||
|  | #  Copyright (c) 2008 Helmut Merz helmutm@cy55.de | ||||||
|  | # | ||||||
|  | #  This program is free software; you can redistribute it and/or modify | ||||||
|  | #  it under the terms of the GNU General Public License as published by | ||||||
|  | #  the Free Software Foundation; either version 2 of the License, or | ||||||
|  | #  (at your option) any later version. | ||||||
|  | # | ||||||
|  | #  This program is distributed in the hope that it will be useful, | ||||||
|  | #  but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | #  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | #  GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | #  You should have received a copy of the GNU General Public License | ||||||
|  | #  along with this program; if not, write to the Free Software | ||||||
|  | #  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA | ||||||
|  | # | ||||||
|  | 
 | ||||||
|  | """ | ||||||
|  | Basic implementations for stateful objects and adapters. | ||||||
|  | 
 | ||||||
|  | $Id$ | ||||||
|  | """ | ||||||
|  | 
 | ||||||
|  | from zope import component | ||||||
|  | from zope.component import adapts | ||||||
|  | 
 | ||||||
|  | from cybertools.stateful.base import Stateful as BaseStateful | ||||||
|  | from cybertools.stateful.base import StatefulAdapter | ||||||
|  | from cybertools.stateful.interfaces import IStatesDefinition | ||||||
|  | from loops.interfaces import ILoopsObject | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class Stateful(BaseStateful): | ||||||
|  | 
 | ||||||
|  |     def getStatesDefinition(self): | ||||||
|  |         return component.getUtility(IStatesDefinition, self.statesDefinition) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class StatefulLoopsObject(Stateful, StatefulAdapter): | ||||||
|  | 
 | ||||||
|  |     adapts(ILoopsObject) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class SimplePublishable(StatefulLoopsObject): | ||||||
|  | 
 | ||||||
|  |     statesDefinition = 'loops.simple_publishing' | ||||||
							
								
								
									
										22
									
								
								stateful/configure.zcml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								stateful/configure.zcml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | ||||||
|  | <!-- $Id$ --> | ||||||
|  | 
 | ||||||
|  | <configure | ||||||
|  |    xmlns:zope="http://namespaces.zope.org/zope" | ||||||
|  |    xmlns:browser="http://namespaces.zope.org/browser" | ||||||
|  |    i18n_domain="loops"> | ||||||
|  | 
 | ||||||
|  |   <zope:utility | ||||||
|  |         factory="cybertools.stateful.publishing.simplePublishing" | ||||||
|  |         name="loops.simple_publishing" /> | ||||||
|  | 
 | ||||||
|  |   <zope:adapter | ||||||
|  |         factory="loops.stateful.base.SimplePublishable" | ||||||
|  |         name="loops.simple_publishing" trusted="True" /> | ||||||
|  |   <zope:class class="loops.stateful.base.SimplePublishable"> | ||||||
|  |     <require permission="zope.View" | ||||||
|  |              interface="cybertools.stateful.interfaces.IStateful" /> | ||||||
|  |     <require permission="zope.ManageContent" | ||||||
|  |              set_schema="cybertools.stateful.interfaces.IStateful" /> | ||||||
|  |   </zope:class> | ||||||
|  | 
 | ||||||
|  | </configure> | ||||||
							
								
								
									
										28
									
								
								stateful/tests.py
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										28
									
								
								stateful/tests.py
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,28 @@ | ||||||
|  | #! /usr/bin/python | ||||||
|  | 
 | ||||||
|  | """ | ||||||
|  | Tests for the 'loops.stateful' package. | ||||||
|  | 
 | ||||||
|  | $Id$ | ||||||
|  | """ | ||||||
|  | 
 | ||||||
|  | import unittest, doctest | ||||||
|  | from zope.testing.doctestunit import DocFileSuite | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class Test(unittest.TestCase): | ||||||
|  |     "Basic tests for the storage package." | ||||||
|  | 
 | ||||||
|  |     def testBasicStuff(self): | ||||||
|  |         pass | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def test_suite(): | ||||||
|  |     flags = doctest.NORMALIZE_WHITESPACE | doctest.ELLIPSIS | ||||||
|  |     return unittest.TestSuite(( | ||||||
|  |         unittest.makeSuite(Test), | ||||||
|  |         DocFileSuite('README.txt', optionflags=flags), | ||||||
|  |         )) | ||||||
|  | 
 | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     unittest.main(defaultTest='test_suite') | ||||||
|  | @ -166,14 +166,14 @@ Updating the concept map | ||||||
| 
 | 
 | ||||||
|   >>> topicId = xrf.getObjectByName('topic')['id'] |   >>> topicId = xrf.getObjectByName('topic')['id'] | ||||||
|   >>> xrf.createConcept(topicId, u'zope2', u'Zope 2') |   >>> xrf.createConcept(topicId, u'zope2', u'Zope 2') | ||||||
|   {'description': u'', 'title': u'Zope 2', 'type': '22', 'id': '56', |   {'description': u'', 'title': u'Zope 2', 'type': '24', 'id': '56', | ||||||
|    'name': u'zope2'} |    'name': u'zope2'} | ||||||
| 
 | 
 | ||||||
| The name of the concept is checked by a name chooser; if the corresponding | The name of the concept is checked by a name chooser; if the corresponding | ||||||
| parameter is empty, the name will be generated from the title. | parameter is empty, the name will be generated from the title. | ||||||
| 
 | 
 | ||||||
|   >>> xrf.createConcept(topicId, u'', u'Python') |   >>> xrf.createConcept(topicId, u'', u'Python') | ||||||
|   {'description': u'', 'title': u'Python', 'type': '22', 'id': '58', |   {'description': u'', 'title': u'Python', 'type': '24', 'id': '58', | ||||||
|    'name': u'python'} |    'name': u'python'} | ||||||
| 
 | 
 | ||||||
| Changing the attributes of a concept | Changing the attributes of a concept | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		
		Reference in a new issue
	
	 helmutm
						helmutm