loops/organize/party.py

192 lines
6.9 KiB
Python

#
# Copyright (c) 2013 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.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.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 principal is None:
principal = getattr(request, 'principal', None)
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))
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:
raise ValueError(
'Error when creating user %s: There is already a person (%s) assigned to user %s.'
% (getName(self.context), 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', None) or []
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)