##############################################################################
#
# Copyright (c) 2004 Zope Corporation and Contributors. All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.1 (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.
#
##############################################################################
"""Unique Id Generation and Handling

These interfaces are intentionaly kept close to those from Zope3. Additionaly
handling (IUniqueIdHandler) and generation (IUniqueIdGenerator) of unique ids 
are kept separate.

$Id: interfaces.py 66701 2006-04-09 00:53:45Z jens $
"""

from Interface import Interface, Attribute

from Products.CMFCore.interfaces.IOpaqueItems \
    import ICallableOpaqueItem, ICallableOpaqueItemEvents

class UniqueIdError(Exception): pass


class IUniqueIdGenerator(Interface):
    """Generate a unique id.
    """
    
    def __call__():
        """Return a unique id value.
        """

    def convert(uid):
        """Converts the unique id from string type to the appropriate
           uid type.
           
        The resulting type has to be the same as the one '__call__'
        returns.
        """

class IUniqueIdAnnotationManagement(Interface):
    """Manage unique id annotations.
    """
    
    def __call__(obj, id):
        """Attach an unique id attribute of 'id' to the passed object.
        
        Return a unique id object implementing 'IUniqueIdAnnotation'.
        """

class IUniqueIdAnnotation(ICallableOpaqueItem, ICallableOpaqueItemEvents):
    """Opaque unique id item handling adding, copying, and deletion events.
    """
    
    def setUid(uid):
        """Set the uid value the unique id annotation shall return.
        """


class IUniqueIdSet(Interface):
    """(Un)register unique ids on objects.
    """

    def register(obj):
        """Register the object and return the unique id generated for it.
        
        If the object is already registered, its unique id is returned 
        anyway.
        
        UniqueIdError is raised if object can not be registered a unique id.
        """

    def unregister(obj):
        """Remove the object from the indexes.
        
        UniqueIdError is raised if object was not registered previously.
        """

    def setUid(obj, uid, check_uniqueness=True):
        """Set the unique id of an object.
        
        By default a check ensuring uniqueness is enabled. Be aware when
        disabling this check: You really need to know what you do !!!
        """


class IUniqueIdQuery(Interface):
    """Querying unique ids.
    """
    def queryUid(obj, default=None):
        """Return the unique id of the object.
        
        If the object doesn't have a unique, the default value is returned.
        """
        
    def getUid(obj):
        """Return the unique id of the object.
        
        If the object doesn't have a unique, a UniqueIdError is raised.
        """
        
    def queryObject(uid, default=None):
        """Return the object with the given uid.
        
        If no object exist with the given unique id, the default value is 
        returned.
        """
        
    def getObject(uid):
        """Return the object with the given uid.
        
        If no object exist with the given unique id, a UniqueIdError is raised.
        """

# This is the main API for playing with unique ids
class IUniqueIdHandler(IUniqueIdSet, IUniqueIdQuery):
    """Handle registering, querying unique ids and objects.
    """

# This is a secondary API enhancement
class IUniqueIdBrainQuery(Interface):
    """Querying unique ids returning brains for efficiency sake.
    
    Returning a brain is more efficient than returning the object.
    A brain usually exposes only parts of the object and should only 
    be read from. 
    
    If the implementing class doesn't support returning a catalog 
    brain it may fallback to return the object. To be entirely
    compatible it must implement the (non existing) interface 
    catalog brains implement.
    """
    
    def queryBrain(uid, default=None):
        """Return the brain of object with the given uid.
        
        If no object exist with the given unique id, the default value
        is returned.
        """
        
    def getBrain(uid):
        """Return a brain of the object with the given uid.
        
        If no object exist with the given unique id, a UniqueIdError 
        is raised.
        """

# This is another secondary API enhancement
class IUniqueIdUnrestrictedQuery(Interface):
    """Querying unique ids unrestricted.
    
    The below methods return not yet effective and already expired 
    objects regardless of the roles the caller has.
    
    CAUTION: Care must be taken not to open security holes by exposing 
    the results of these methods to non authorized callers!
        
    If you're in doubth if you should use this method signature or
    the restricted ones ('IUniqueIdQuery' and 'IUniqueIdBrainQuery') 
    use the latter!
    
    Returning a brain is more efficient than returning the object.
    A brain usually exposes only parts of the object and should only 
    be read from. 
    
    If the implementing class doesn't support returning a catalog 
    brain it may fallback to return the object. To be entirely
    compatible it must implement the (non existing) interface 
    catalog brains implement.
    """
    
    def unrestrictedQueryObject(uid, default=None):
        """Return the object with the given uid.
        
        If no object exist with the given unique id, the default value 
        is returned.
        """
        
    def unrestrictedGetObject(uid):
        """Return a brain of the object with the given uid.
        
        If no object exist with the given unique id, a UniqueIdError 
        is raised.
        """

    def unrestrictedQueryBrain(uid, default=None):
        """Return the brain of the object with the given uid.
        
        If no object exist with the given unique id, the default value
        is returned.
        """
        
    def unrestrictedGetBrain(uid):
        """Return a brain of the object with the given uid.
        
        If no object exist with the given unique id, a UniqueIdError 
        is raised.
        """
