############################################################################## # # Copyright (c) 2001 Zope Corporation and Contributors. All Rights Reserved. # # This software is subject to the provisions of the Zope Public License, # Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution. # THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED # WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED # WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS # FOR A PARTICULAR PURPOSE # ############################################################################## """WebDAV support - lockable item. $Id: Lockable.py 40281 2005-11-21 00:08:00Z efge $ """ from AccessControl import ClassSecurityInfo from Globals import InitializeClass from Globals import PersistentMapping import Acquisition from EtagSupport import EtagSupport from WriteLockInterface import LockItemInterface from WriteLockInterface import WriteLockInterface class ResourceLockedError(Exception): pass class LockableItem(EtagSupport): """\ Implements the WriteLockInterface, and is inherited by Resource which is then inherited by the majority of Zope objects. For an object to be lockable, however, it should have the WriteLockInterface in its __implements__ list, ie: __implements__ = (WriteLockInterface,) """ # Protect methods using declarative security security = ClassSecurityInfo() security.declarePrivate('wl_lockmapping') security.declarePublic('wl_isLocked', 'wl_getLock', 'wl_isLockedByUser', 'wl_lockItems', 'wl_lockValues', 'wl_lockTokens',) security.declareProtected('WebDAV Lock items', 'wl_setLock') security.declareProtected('WebDAV Unlock items', 'wl_delLock') security.declareProtected('Manage WebDAV Locks', 'wl_clearLocks') # Setting default roles for permissions - we want owners of conent # to be able to lock. security.setPermissionDefault('WebDAV Lock items', ('Manager', 'Owner',)) security.setPermissionDefault('WebDAV Unlock items',('Manager','Owner',)) def wl_lockmapping(self, killinvalids=0, create=0): """ if 'killinvalids' is 1, locks who are no longer valid will be deleted """ try: locks = getattr(self, '_dav_writelocks', None) except: locks = None if locks is None: if create: locks = self._dav_writelocks = PersistentMapping() else: # Don't generate a side effect transaction. locks = {} return locks elif killinvalids: # Delete invalid locks for token, lock in locks.items(): if not lock.isValid(): del locks[token] if (not locks) and hasattr(Acquisition.aq_base(self), '__no_valid_write_locks__'): self.__no_valid_write_locks__() return locks else: return locks def wl_lockItems(self, killinvalids=0): return self.wl_lockmapping(killinvalids).items() def wl_lockValues(self, killinvalids=0): return self.wl_lockmapping(killinvalids).values() def wl_lockTokens(self, killinvalids=0): return self.wl_lockmapping(killinvalids).keys() def wl_hasLock(self, token, killinvalids=0): if not token: return 0 return token in self.wl_lockmapping(killinvalids).keys() def wl_isLocked(self): # returns true if 'self' is locked at all # We set 'killinvalids' to 1 to delete all locks who are no longer # valid (timeout has been exceeded) locks = self.wl_lockmapping(killinvalids=1) if locks.keys(): return 1 else: return 0 def wl_setLock(self, locktoken, lock): locks = self.wl_lockmapping(create=1) if LockItemInterface.isImplementedBy(lock): if locktoken == lock.getLockToken(): locks[locktoken] = lock else: raise ValueError, 'Lock tokens do not match' else: raise ValueError, 'Lock does not implement the LockItem Interface' def wl_getLock(self, locktoken): locks = self.wl_lockmapping(killinvalids=1) return locks.get(locktoken, None) def wl_delLock(self, locktoken): locks = self.wl_lockmapping() if locks.has_key(locktoken): del locks[locktoken] def wl_clearLocks(self): # Called by lock management machinery to quickly and effectively # destroy all locks. try: locks = self.wl_lockmapping() locks.clear() except: # The locks may be totally messed up, so we'll just delete # and replace. if hasattr(self, '_dav_writelocks'): del self._dav_writelocks if WriteLockInterface.isImplementedBy(self): self._dav_writelocks = PersistentMapping() # Call into a special hook used by LockNullResources to delete # themselves. Could be used by other objects who want to deal # with the state of empty locks. if hasattr(Acquisition.aq_base(self), '__no_valid_write_locks__'): self.__no_valid_write_locks__() InitializeClass(LockableItem) ### Utility functions def wl_isLocked(ob): """ Returns true if the object is locked, returns 0 if the object is not locked or does not implement the WriteLockInterface """ return WriteLockInterface.isImplementedBy(ob) and ob.wl_isLocked()