196 lines
7.1 KiB
Python
196 lines
7.1 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.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 principal is None:
|
|
if request is None:
|
|
principal = getCurrentPrincipal()
|
|
else:
|
|
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)
|