##############################################################################
#
# Copyright (c) 2002 Zope Foundation and Contributors.
#
# 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.
#
##############################################################################
"""Objects providing context for product initialization
"""
 
from logging import getLogger
import os
 
from AccessControl.Permission import registerPermissions
from AccessControl.PermissionRole import PermissionRole
from OFS.ObjectManager import ObjectManager
 
from zope.interface import implementedBy
 
from App.FactoryDispatcher import FactoryDispatcher
 
# Waaaa
import Products
if not hasattr(Products, 'meta_types'):
    Products.meta_types=()
if not hasattr(Products, 'meta_classes'):
    Products.meta_classes={}
    Products.meta_class_info={}
 
_marker = []  # Create a new marker object
LOG = getLogger('ProductContext')
 
class ProductContext:
 
    def __init__(self, product, app, package):
        self.__prod = product
        self.__pack = package
 
    def registerClass(self, instance_class=None, meta_type='',
                      permission=None, constructors=(),
                      icon=None, permissions=None, legacy=(),
                      visibility="Global", interfaces=_marker,
                      container_filter=None
        ):
        """Register a constructor
 
        Keyword arguments are used to provide meta data:
 
        instance_class -- The class of the object that will be created.
 
          This is not currently used, but may be used in the future to
          increase object mobility.
 
        meta_type -- The kind of object being created
           This appears in add lists.  If not specified, then the class
           meta_type will be used.
 
        permission -- The permission name for the constructors.
           If not specified, then a permission name based on the
           meta type will be used.
 
        constructors -- A list of constructor methods
          A method can me a callable object with a __name__
          attribute giving the name the method should have in the
          product, or the method may be a tuple consisting of a
          name and a callable object.  The method must be picklable.
 
          The first method will be used as the initial method called
          when creating an object.
 
        icon -- No longer used.
 
        permissions -- Additional permissions to be registered
           If not provided, then permissions defined in the
           class will be registered.
 
        legacy -- A list of legacy methods to be added to ObjectManager
                  for backward compatibility
 
        visibility -- "Global" if the object is globally visible, None else
 
        interfaces -- a list of the interfaces the object supports
 
        container_filter -- function that is called with an ObjectManager
           object as the only parameter, which should return a true object
           if the object is happy to be created in that container. The
           filter is called before showing ObjectManager's Add list,
           and before pasting (after object copy or cut), but not
           before calling an object's constructor.
 
        """
        pack=self.__pack
        initial=constructors[0]
        productObject=self.__prod
        pid=productObject.id
 
        if permissions:
            if isinstance(permissions, basestring): # You goofed it!
                raise TypeError, ('Product context permissions should be a '
                    'list of permissions not a string', permissions)
            for p in permissions:
                if isinstance(p, tuple):
                    p, default= p
                    registerPermissions(((p, (), default),))
                else:
                    registerPermissions(((p, ()),))
 
        ############################################################
        # Constructor permission setup
        if permission is None:
            permission="Add %ss" % (meta_type or instance_class.meta_type)
 
        if isinstance(permission, tuple):
            permission, default = permission
        else:
            default = ('Manager',)
 
        pr = PermissionRole(permission,default)
        registerPermissions(((permission, (), default),))
        ############################################################
 
        OM = ObjectManager
 
        for method in legacy:
            if isinstance(method, tuple):
                name, method = method
                aliased = 1
            else:
                name=method.__name__
                aliased = 0
            if name not in OM.__dict__:
                setattr(OM, name, method)
                setattr(OM, name+'__roles__', pr)
                if aliased:
                    # Set the unaliased method name and its roles
                    # to avoid security holes.  XXX: All "legacy"
                    # methods need to be eliminated.
                    setattr(OM, method.__name__, method)
                    setattr(OM, method.__name__+'__roles__', pr)
 
        if isinstance(initial, tuple):
            name, initial = initial
        else:
            name = initial.__name__
 
        fd = getattr(pack, '__FactoryDispatcher__', None)
        if fd is None:
            class __FactoryDispatcher__(FactoryDispatcher):
                "Factory Dispatcher for a Specific Product"
 
            fd = pack.__FactoryDispatcher__ = __FactoryDispatcher__
 
        if not hasattr(pack, '_m'):
            pack._m = AttrDict(fd)
 
        m = pack._m
 
        if interfaces is _marker:
            if instance_class is None:
                interfaces = ()
            else:
                interfaces = tuple(implementedBy(instance_class))
 
        Products.meta_types = Products.meta_types + (
            { 'name': meta_type or instance_class.meta_type,
              # 'action': The action in the add drop down in the ZMI. This is
              #           currently also required by the _verifyObjectPaste
              #           method of CopyContainers like Folders.
              'action': ('manage_addProduct/%s/%s' % (pid, name)),
              # 'product': product id
              'product': pid,
              # 'permission': Guards the add action.
              'permission': permission,
              # 'visibility': A silly name. Doesn't have much to do with
              #               visibility. Allowed values: 'Global', None
              'visibility': visibility,
              # 'interfaces': A tuple of oldstyle and/or newstyle interfaces.
              'interfaces': interfaces,
              'instance': instance_class,
              'container_filter': container_filter
              },)
 
        m[name]=initial
        m[name+'__roles__']=pr
 
        for method in constructors[1:]:
            if isinstance(method, tuple):
                name, method = method
            else:
                name=os.path.split(method.__name__)[-1]
            if name not in productObject.__dict__:
                m[name]=method
                m[name+'__roles__']=pr
 
    def registerHelp(self, directory=None, clear=None, title_re=None):
        pass
 
    def registerHelpTitle(self, title=None):
        pass
 
    def getProductHelp(self):
        class DummyHelp(object):
            lastRegistered = None
        return DummyHelp()
 
 
class AttrDict:
 
    def __init__(self, ob):
        self.ob = ob
 
    def __setitem__(self, name, v):
        setattr(self.ob, name, v)