# # Copyright (c) 2015 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 # """ Adapters for IConcept providing interfaces from the cybertools.organize package. """ from persistent.mapping import PersistentMapping from zope import interface, component from zope.app.principalannotation import annotations from zope.app.security.interfaces import IAuthentication, PrincipalLookupError from zope.app.security.interfaces import IUnauthenticatedPrincipal from zope.component import adapts from zope.interface import implements from zope.cachedescriptors.property import Lazy from zope.schema.interfaces import ValidationError from zope.app.form.interfaces import WidgetInputError from zope.security.proxy import removeSecurityProxy from zope.traversing.api import getName from cybertools.organize.party import Person as BasePerson from cybertools.relation.interfaces import IRelationRegistry from cybertools.typology.interfaces import IType from loops.common import AdapterBase from loops.concept import Concept from loops.interfaces import IConcept from loops.organize.interfaces import IAddress, IPerson, IHasRole from loops.organize.interfaces import ANNOTATION_KEY from loops.predicate import RelationAdapter from loops.predicate import PredicateInterfaceSourceList from loops.security.common import assignOwner, removeOwner, allowEditingForOwner from loops.security.common import assignPersonRole, removePersonRole from loops.security.common import getCurrentPrincipal from loops.security.interfaces import ISecuritySetter from loops.type import TypeInterfaceSourceList from loops import util # register type interfaces - (TODO: use a function for this) TypeInterfaceSourceList.typeInterfaces += (IPerson, IAddress) PredicateInterfaceSourceList.predicateInterfaces += (IHasRole,) def getPersonForUser(context, request=None, principal=None): if context is None: return None if principal is None: if request is not None: principal = getattr(request, 'principal', None) else: principal = getPrincipal(context) if principal is None: return None loops = context.getLoopsRoot() pa = annotations(principal).get(ANNOTATION_KEY, None) if pa is None: return None if type(pa) == Concept: # backward compatibility if pa.getLoopsRoot() == loops: return pa else: return None return pa.get(util.getUidForObject(loops)) def getPrincipal(context): principal = getCurrentPrincipal() if principal is not None: if IUnauthenticatedPrincipal.providedBy(principal): return None return principal return None class Person(AdapterBase, BasePerson): """ typeInterface adapter for concepts of type 'person'. """ implements(IPerson) _adapterAttributes = ('context', '__parent__', 'userId', 'phoneNumbers') _contextAttributes = list(IPerson) + list(IConcept) def getUserId(self): return getattr(self.context, '_userId', None) def setUserId(self, userId): setter = ISecuritySetter(self) if userId: principal = self.getPrincipalForUserId(userId) if principal is None: return person = getPersonForUser(self.context, principal=principal) if person is not None and person != self.context: name = getName(person) if name: raise ValueError( 'There is already a person (%s) assigned to user %s.' % (getName(person), userId)) pa = annotations(principal) loopsId = util.getUidForObject(self.context.getLoopsRoot()) ann = pa.get(ANNOTATION_KEY) if ann is None: # or not isinstance(ann, PersistentMapping): ann = pa[ANNOTATION_KEY] = PersistentMapping() ann[loopsId] = self.context #assignOwner(self.context, userId) assignPersonRole(self.context, userId) oldUserId = self.userId if oldUserId and oldUserId != userId: self.removeReferenceFromPrincipal(oldUserId) removeOwner(self.context, oldUserId) removePersonRole(self.context, oldUserId) self.context._userId = userId setter.propagateSecurity() allowEditingForOwner(self.context, revert=not userId) # why this? userId = property(getUserId, setUserId) def removeReferenceFromPrincipal(self, userId): principal = self.getPrincipalForUserId(userId) if principal is not None: pa = annotations(principal) ann = pa.get(ANNOTATION_KEY) if type(ann) == Concept: # backward compatibility pa[ANNOTATION_KEY] = None else: if ann is not None: loopsId = util.getUidForObject(self.context.getLoopsRoot()) ann[loopsId] = None def getPhoneNumbers(self): return getattr(self.context, '_phoneNumbers', []) def setPhoneNumbers(self, value): self.context._phoneNumbers = value phoneNumbers = property(getPhoneNumbers, setPhoneNumbers) @Lazy def authentication(self): return getAuthenticationUtility(self.context) @Lazy def principal(self): return self.getPrincipalForUserId() def getPrincipalForUserId(self, userId=None): userId = userId or self.userId if not userId: return None auth = self.authentication try: return auth.getPrincipal(userId) except PrincipalLookupError: return None def getAuthenticationUtility(context): return component.getUtility(IAuthentication, context=context) def removePersonReferenceFromPrincipal(context, event): """ Handles IObjectRemoved event for concepts used as persons. """ if IConcept.providedBy(context): # this does not work as the context is already removed from the # relation registry: #if IType(context).typeInterface == IPerson: # person = IPerson(context) # if person.userId: if getattr(context, '_userId', None): person = IPerson(context) person.removeReferenceFromPrincipal(person.userId) class Address(AdapterBase): """ typeInterface adapter for concepts of type 'address'. """ implements(IAddress) _adapterAttributes = ('context', '__parent__', 'lines') _contextAttributes = list(IAddress) + list(IConcept) def getLines(self): return getattr(self.context, '_lines', []) def setLines(self, value): self.context._lines = value lines = property(getLines, setLines) class HasRole(RelationAdapter): """ Allows specification of a role for a relation. """ implements(IHasRole) _contextAttributes = list(IHasRole)