pDocumentTemplate.py
上传用户:gyjinxi
上传日期:2007-01-04
资源大小:159k
文件大小:9k
源码类别:

WEB邮件程序

开发平台:

Python

  1. ##############################################################################
  2. # Zope Public License (ZPL) Version 1.0
  3. # -------------------------------------
  4. # Copyright (c) Digital Creations.  All rights reserved.
  5. # This license has been certified as Open Source(tm).
  6. # Redistribution and use in source and binary forms, with or without
  7. # modification, are permitted provided that the following conditions are
  8. # met:
  9. # 1. Redistributions in source code must retain the above copyright
  10. #    notice, this list of conditions, and the following disclaimer.
  11. # 2. Redistributions in binary form must reproduce the above copyright
  12. #    notice, this list of conditions, and the following disclaimer in
  13. #    the documentation and/or other materials provided with the
  14. #    distribution.
  15. # 3. Digital Creations requests that attribution be given to Zope
  16. #    in any manner possible. Zope includes a "Powered by Zope"
  17. #    button that is installed by default. While it is not a license
  18. #    violation to remove this button, it is requested that the
  19. #    attribution remain. A significant investment has been put
  20. #    into Zope, and this effort will continue if the Zope community
  21. #    continues to grow. This is one way to assure that growth.
  22. # 4. All advertising materials and documentation mentioning
  23. #    features derived from or use of this software must display
  24. #    the following acknowledgement:
  25. #      "This product includes software developed by Digital Creations
  26. #      for use in the Z Object Publishing Environment
  27. #      (http://www.zope.org/)."
  28. #    In the event that the product being advertised includes an
  29. #    intact Zope distribution (with copyright and license included)
  30. #    then this clause is waived.
  31. # 5. Names associated with Zope or Digital Creations must not be used to
  32. #    endorse or promote products derived from this software without
  33. #    prior written permission from Digital Creations.
  34. # 6. Modified redistributions of any form whatsoever must retain
  35. #    the following acknowledgment:
  36. #      "This product includes software developed by Digital Creations
  37. #      for use in the Z Object Publishing Environment
  38. #      (http://www.zope.org/)."
  39. #    Intact (re-)distributions of any official Zope release do not
  40. #    require an external acknowledgement.
  41. # 7. Modifications are encouraged but must be packaged separately as
  42. #    patches to official Zope releases.  Distributions that do not
  43. #    clearly separate the patches from the original work must be clearly
  44. #    labeled as unofficial distributions.  Modifications which do not
  45. #    carry the name Zope may be packaged in any form, as long as they
  46. #    conform to all of the clauses above.
  47. # Disclaimer
  48. #   THIS SOFTWARE IS PROVIDED BY DIGITAL CREATIONS ``AS IS'' AND ANY
  49. #   EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  50. #   IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
  51. #   PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL DIGITAL CREATIONS OR ITS
  52. #   CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  53. #   SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  54. #   LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
  55. #   USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  56. #   ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  57. #   OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
  58. #   OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  59. #   SUCH DAMAGE.
  60. # This software consists of contributions made by Digital Creations and
  61. # many individuals on behalf of Digital Creations.  Specific
  62. # attributions are listed in the accompanying credits file.
  63. ##############################################################################
  64. __doc__='''Python implementations of document template some features
  65. $Id: pDocumentTemplate.py,v 1.23 1999/03/10 00:15:08 klm Exp $'''
  66. __version__='$Revision: 1.23 $'[11:-2]
  67. import string, sys, types
  68. from string import join
  69. StringType=type('')
  70. TupleType=type(())
  71. isFunctionType={}
  72. for name in ['BuiltinFunctionType', 'BuiltinMethodType', 'ClassType',
  73.              'FunctionType', 'LambdaType', 'MethodType', 'UnboundMethodType']:
  74.     try: isFunctionType[getattr(types,name)]=1
  75.     except: pass
  76. try: # Add function and method types from Extension Classes
  77.     import ExtensionClass
  78.     isFunctionType[ExtensionClass.PythonMethodType]=1
  79.     isFunctionType[ExtensionClass.ExtensionMethodType]=1
  80. except: pass
  81. isFunctionType=isFunctionType.has_key
  82. isSimpleType={}
  83. for n in dir(types):
  84.     if (n[-4:]=='Type' and n != 'InstanceType' and
  85.         not isFunctionType(getattr(types, n))):
  86.         isSimpleType[getattr(types, n)]=1
  87. isSimpleType=isSimpleType.has_key
  88. class InstanceDict:
  89.     validate=None
  90.     def __init__(self,o,namespace,validate=None):
  91.         self.self=o
  92.         self.cache={}
  93.         self.namespace=namespace
  94.         if validate is None: self.validate=namespace.validate
  95.         else: self.validate=validate
  96.     def has_key(self,key):
  97.         return hasattr(self.self,key)
  98.     def keys(self):
  99.         return self.self.__dict__.keys()
  100.     def __repr__(self): return 'InstanceDict(%s)' % str(self.self)
  101.     def __getitem__(self,key):
  102.         cache=self.cache
  103.         if cache.has_key(key): return cache[key]
  104.         
  105.         inst=self.self
  106.         if key[:1]=='_':
  107.             if key != '__str__':
  108.                 raise KeyError, key # Don't divuldge private data
  109.             r=str(inst)
  110.         else:
  111.             try: r=getattr(inst,key)
  112.             except AttributeError: raise KeyError, key
  113.         v=self.validate
  114.         if v is not None: v(inst,inst,key,r,self.namespace)
  115.         self.cache[key]=r
  116.         return r
  117. class MultiMapping:
  118.     def __init__(self): self.dicts=[]
  119.     def __getitem__(self, key):
  120.         for d in self.dicts:
  121.             try: return d[key]
  122.             except KeyError, AttributeError: pass
  123.         raise KeyError, key
  124.     def push(self,d): self.dicts.insert(0,d)
  125.     def pop(self,n=1): del self.dicts[:n]
  126.     def keys(self):
  127.         kz = []
  128.         for d in self.dicts:
  129.             kz = kz + d.keys()
  130.         return kz
  131. class TemplateDict:
  132.     level=0
  133.     def _pop(self, n=1): return self.dicts.pop(n)
  134.     def _push(self, d): return self.dicts.push(d)
  135.     def __init__(self):
  136.         m=self.dicts=MultiMapping()
  137.         self._pop=m.pop
  138.         self._push=m.push
  139.         try: self.keys=m.keys
  140.         except: pass
  141.     def __getitem__(self,key,call=1,
  142.                     simple=isSimpleType,
  143.                     isFunctionType=isFunctionType,
  144.                     ):
  145.         v=self.dicts[key]
  146.         if call and not simple(type(v)):
  147.             if hasattr(v,'isDocTemp') and v.isDocTemp: return v(None, self)
  148.             elif isFunctionType(type(v)): return v()
  149.             else:
  150.                 try: return v()
  151.                 except AttributeError, ev:
  152.                     try:
  153.                         tb=sys.exc_traceback
  154.                         if hasattr(sys, 'exc_info'): tb=sys.exc_info()[2]
  155.                         if hasattr(v,'__call__'):
  156.                             raise AttributeError, ev, tb
  157.                     finally: tb=None
  158.                 except TypeError: pass
  159.                         
  160.         return v
  161.     def has_key(self,key):
  162.         try:
  163.             v=self.dicts[key]
  164.         except KeyError:
  165.             return 0
  166.         return 1
  167.     
  168.     getitem=__getitem__
  169. def render_blocks(blocks, md):
  170.     rendered = []
  171.     append=rendered.append
  172.     for section in blocks:
  173.         if type(section) is TupleType:
  174.             l=len(section)
  175.             if l==1:
  176.                 # Simple var
  177.                 section=section[0]
  178.                 if type(section) is StringType: section=md[section]
  179.                 else: section=section(md)
  180.                 section=str(section)
  181.             else:
  182.                 # if
  183.                 cache={}
  184.                 md._push(cache)
  185.                 try:
  186.                     i=0
  187.                     m=l-1
  188.                     while i < m:
  189.                         cond=section[i]
  190.                         if type(cond) is StringType:
  191.                             n=cond
  192.                             try:
  193.                                 cond=md[cond]
  194.                                 cache[n]=cond
  195.                             except KeyError, v:
  196.                                 v=str(v)
  197.                                 if n != v: raise KeyError, v, sys.exc_traceback
  198.                                 cond=None
  199.                         else: cond=cond(md)
  200.                         if cond:
  201.                             section=section[i+1]
  202.                             if section: section=render_blocks(section,md)
  203.                             else: section=''
  204.                             m=0
  205.                             break
  206.                         i=i+2
  207.                     if m:
  208.                         if i==m: section=render_blocks(section[i],md)
  209.                         else: section=''
  210.                 finally: md._pop()
  211.         elif type(section) is not StringType:
  212.             section=section(md)
  213.         if section: rendered.append(section)
  214.     l=len(rendered)
  215.     if l==0: return ''
  216.     elif l==1: return rendered[0]
  217.     return join(rendered, '')
  218.     return rendered