[Zope-Checkins] CVS: Zope/lib/python/TAL - DummyEngine.py:1.32.4.2 TALGenerator.py:1.55.4.2 TALInterpreter.py:1.69.4.3

Chris McDonough chrism@zope.com
Tue, 8 Oct 2002 14:41:44 -0400


Update of /cvs-repository/Zope/lib/python/TAL
In directory cvs.zope.org:/tmp/cvs-serv20268/lib/python/TAL

Modified Files:
      Tag: chrism-install-branch
	DummyEngine.py TALGenerator.py TALInterpreter.py 
Log Message:
Merging HEAD into chrism-install-branch.


=== Zope/lib/python/TAL/DummyEngine.py 1.32.4.1 => 1.32.4.2 ===
--- Zope/lib/python/TAL/DummyEngine.py:1.32.4.1	Sat Sep 28 21:40:35 2002
+++ Zope/lib/python/TAL/DummyEngine.py	Tue Oct  8 14:41:13 2002
@@ -8,7 +8,7 @@
 # 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.
+# FOR A PARTICULAR PURPOSE
 #
 ##############################################################################
 """
@@ -18,20 +18,13 @@
 import re
 import sys
 
+import driver
+
 from TALDefs import NAME_RE, TALESError, ErrorInfo
-from ITALES import ITALESCompiler, ITALESEngine
-from DocumentTemplate.DT_Util import ustr
-try:
-    from Zope.I18n.ITranslationService import ITranslationService
-    from Zope.I18n.IDomain import IDomain
-except ImportError:
-    # Before 2.7
-    class ITranslationService: pass
-    class IDomain: pass
 
-class _Default:
+class Default:
     pass
-Default = _Default()
+Default = Default()
 
 name_match = re.compile(r"(?s)(%s):(.*)\Z" % NAME_RE).match
 
@@ -43,8 +36,6 @@
     position = None
     source_file = None
 
-    __implements__ = ITALESCompiler, ITALESEngine
-
     def __init__(self, macros=None):
         if macros is None:
             macros = {}
@@ -52,7 +43,6 @@
         dict = {'nothing': None, 'default': Default}
         self.locals = self.globals = dict
         self.stack = [dict]
-        self.translationService = DummyTranslationService()
 
     def getCompilerError(self):
         return CompilerError
@@ -100,7 +90,13 @@
         if type in ("string", "str"):
             return expr
         if type in ("path", "var", "global", "local"):
-            return self.evaluatePathOrVar(expr)
+            expr = expr.strip()
+            if self.locals.has_key(expr):
+                return self.locals[expr]
+            elif self.globals.has_key(expr):
+                return self.globals[expr]
+            else:
+                raise TALESError("unknown variable: %s" % `expr`)
         if type == "not":
             return not self.evaluate(expr)
         if type == "exists":
@@ -120,15 +116,6 @@
             return '%s (%s,%s)' % (self.source_file, lineno, offset)
         raise TALESError("unrecognized expression: " + `expression`)
 
-    def evaluatePathOrVar(self, expr):
-        expr = expr.strip()
-        if self.locals.has_key(expr):
-            return self.locals[expr]
-        elif self.globals.has_key(expr):
-            return self.globals[expr]
-        else:
-            raise TALESError("unknown variable: %s" % `expr`)
-
     def evaluateValue(self, expr):
         return self.evaluate(expr)
 
@@ -138,7 +125,7 @@
     def evaluateText(self, expr):
         text = self.evaluate(expr)
         if text is not None and text is not Default:
-            text = ustr(text)
+            text = str(text)
         return text
 
     def evaluateStructure(self, expr):
@@ -159,7 +146,6 @@
             macro = self.macros[localName]
         else:
             # External macro
-            import driver
             program, macros = driver.compilefile(file)
             macro = macros.get(localName)
             if not macro:
@@ -171,7 +157,6 @@
         file, localName = self.findMacroFile(macroName)
         if not file:
             return file, localName
-        import driver
         doc = driver.parsefile(file)
         return doc, localName
 
@@ -198,10 +183,6 @@
     def getDefault(self):
         return Default
 
-    def translate(self, domain, msgid, mapping):
-        return self.translationService.translate(domain, msgid, mapping)
-    
-
 class Iterator:
 
     def __init__(self, name, seq, engine):
@@ -219,31 +200,3 @@
         self.nextIndex = i+1
         self.engine.setLocal(self.name, item)
         return 1
-
-class DummyDomain:
-    __implements__ = IDomain
-
-    def translate(self, msgid, mapping=None, context=None,
-                  target_language=None):
-        # This is a fake translation service which simply uppercases non
-        # ${name} placeholder text in the message id.
-        #
-        # First, transform a string with ${name} placeholders into a list of
-        # substrings.  Then upcase everything but the placeholders, then glue
-        # things back together.
-        def repl(m, mapping=mapping):
-            return mapping[m.group(m.lastindex).lower()]
-        cre = re.compile(r'\$(?:([_A-Z]\w*)|\{([_A-Z]\w*)\})')
-        return cre.sub(repl, msgid.upper())
-
-class DummyTranslationService:
-    __implements__ = ITranslationService
-
-    def translate(self, domain, msgid, mapping=None, context=None,
-                  target_language=None):
-        # Ignore domain
-        return self.getDomain(domain).translate(msgid, mapping, context,
-                                                target_language)
-
-    def getDomain(self, domain):
-        return DummyDomain()


=== Zope/lib/python/TAL/TALGenerator.py 1.55.4.1 => 1.55.4.2 ===
--- Zope/lib/python/TAL/TALGenerator.py:1.55.4.1	Sat Sep 28 21:40:35 2002
+++ Zope/lib/python/TAL/TALGenerator.py	Tue Oct  8 14:41:13 2002
@@ -8,7 +8,7 @@
 # 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.
+# FOR A PARTICULAR PURPOSE
 #
 ##############################################################################
 """
@@ -18,16 +18,7 @@
 import re
 import cgi
 
-import TALDefs
-
-from TALDefs import NAME_RE, TAL_VERSION
-from TALDefs import I18NError, METALError, TALError
-from TALDefs import parseSubstitution
-from TranslationContext import TranslationContext, DEFAULT_DOMAIN
-
-I18N_REPLACE = 1
-I18N_CONTENT = 2
-I18N_EXPRESSION = 3
+from TALDefs import *
 
 class TALGenerator:
 
@@ -41,15 +32,8 @@
             expressionCompiler = DummyEngine()
         self.expressionCompiler = expressionCompiler
         self.CompilerError = expressionCompiler.getCompilerError()
-        # This holds the emitted opcodes representing the input
         self.program = []
-        # The program stack for when we need to do some sub-evaluation for an
-        # intermediate result.  E.g. in an i18n:name tag for which the
-        # contents describe the ${name} value.
         self.stack = []
-        # Another stack of postponed actions.  Elements on this stack are a
-        # dictionary; key/values contain useful information that
-        # emitEndElement needs to finish its calculations
         self.todoStack = []
         self.macros = {}
         self.slots = {}
@@ -60,7 +44,6 @@
         if source_file is not None:
             self.source_file = source_file
             self.emit("setSourceFile", source_file)
-        self.i18nContext = TranslationContext()
 
     def getCode(self):
         assert not self.stack
@@ -99,12 +82,6 @@
                 # instructions to be joined together.
                 output.append(self.optimizeArgsList(item))
                 continue
-            if opcode == 'noop':
-                # This is a spacer for end tags in the face of i18n:name
-                # attributes.  We can't let the optimizer collect immediately
-                # following end tags into the same rawtextOffset.
-                opcode = None
-                pass
             text = "".join(collect)
             if text:
                 i = text.rfind("\n")
@@ -125,28 +102,9 @@
         else:
             return item[0], tuple(item[1:])
 
-    # These codes are used to indicate what sort of special actions
-    # are needed for each special attribute.  (Simple attributes don't
-    # get action codes.)
-    #
-    # The special actions (which are modal) are handled by
-    # TALInterpreter.attrAction() and .attrAction_tal().
-    #
-    # Each attribute is represented by a tuple:
-    #
-    # (name, value)                 -- a simple name/value pair, with
-    #                                  no special processing
-    #
-    # (name, value, action, *extra) -- attribute with special
-    #                                  processing needs, action is a
-    #                                  code that indicates which
-    #                                  branch to take, and *extra
-    #                                  contains additional,
-    #                                  action-specific information
-    #                                  needed by the processing
-    #
+    actionIndex = {"replace":0, "insert":1, "metal":2, "tal":3, "xmlns":4,
+                   0: 0, 1: 1, 2: 2, 3: 3, 4: 4}
     def optimizeStartTag(self, collect, name, attrlist, end):
-        # return true if the tag can be converted to plain text
         if not attrlist:
             collect.append("<%s%s" % (name, end))
             return 1
@@ -157,15 +115,18 @@
             if len(item) > 2:
                 opt = 0
                 name, value, action = item[:3]
+                action = self.actionIndex[action]
                 attrlist[i] = (name, value, action) + item[3:]
             else:
                 if item[1] is None:
                     s = item[0]
                 else:
-                    s = '%s="%s"' % (item[0], cgi.escape(item[1], 1))
+                    s = "%s=%s" % (item[0], quote(item[1]))
                 attrlist[i] = item[0], s
-                new.append(" " + s)
-        # if no non-optimizable attributes were found, convert to plain text
+            if item[1] is None:
+                new.append(" " + item[0])
+            else:
+                new.append(" %s=%s" % (item[0], quote(item[1])))
         if opt:
             new.append(end)
             collect.extend(new)
@@ -261,7 +222,7 @@
         self.emitRawText(cgi.escape(text))
 
     def emitDefines(self, defines):
-        for part in TALDefs.splitParts(defines):
+        for part in splitParts(defines):
             m = re.match(
                 r"(?s)\s*(?:(global|local)\s+)?(%s)\s+(.*)\Z" % NAME_RE, part)
             if not m:
@@ -313,49 +274,6 @@
             assert key == "structure"
             self.emit("insertStructure", cexpr, attrDict, program)
 
-    def emitI18nVariable(self, varname, action, expression):
-        # Used for i18n:name attributes.  arg is extra information describing
-        # how the contents of the variable should get filled in, and it will
-        # either be a 1-tuple or a 2-tuple.  If arg[0] is None, then the
-        # i18n:name value is taken implicitly from the contents of the tag,
-        # e.g. "I live in <span i18n:name="country">the USA</span>".  In this
-        # case, arg[1] is the opcode sub-program describing the contents of
-        # the tag.
-        #
-        # When arg[0] is not None, it contains the tal expression used to
-        # calculate the contents of the variable, e.g.
-        # "I live in <span i18n:name="country"
-        #                  tal:replace="here/countryOfOrigin" />"
-        key = cexpr = None
-        program = self.popProgram()
-        if action == I18N_REPLACE:
-            # This is a tag with an i18n:name and a tal:replace (implicit or
-            # explicit).  Get rid of the first and last elements of the
-            # program, which are the start and end tag opcodes of the tag.
-            program = program[1:-1]
-        elif action == I18N_CONTENT:
-            # This is a tag with an i18n:name and a tal:content
-            # (explicit-only).  Keep the first and last elements of the
-            # program, so we keep the start and end tag output.
-            pass
-        else:
-            assert action == I18N_EXPRESSION
-            key, expr = parseSubstitution(expression)
-            cexpr = self.compileExpression(expr)
-        # XXX Would key be anything but 'text' or None?
-        assert key in ('text', None)
-        self.emit('i18nVariable', varname, program, cexpr)
-
-    def emitTranslation(self, msgid, i18ndata):
-        program = self.popProgram()
-        if i18ndata is None:
-            self.emit('insertTranslation', msgid, program)
-        else:
-            key, expr = parseSubstitution(i18ndata)
-            cexpr = self.compileExpression(expr)
-            assert key == 'text'
-            self.emit('insertTranslation', msgid, program, cexpr)
-
     def emitDefineMacro(self, macroName):
         program = self.popProgram()
         macroName = macroName.strip()
@@ -443,30 +361,23 @@
         return None
 
     def replaceAttrs(self, attrlist, repldict):
-        # Each entry in attrlist starts like (name, value).
-        # Result is (name, value, action, expr, xlat) if there is a
-        # tal:attributes entry for that attribute.  Additional attrs
-        # defined only by tal:attributes are added here.
-        #
-        # (name, value, action, expr, xlat)
         if not repldict:
             return attrlist
         newlist = []
         for item in attrlist:
             key = item[0]
             if repldict.has_key(key):
-                expr, xlat = repldict[key]
-                item = item[:2] + ("replace", expr, xlat)
+                item = item[:2] + ("replace", repldict[key])
                 del repldict[key]
             newlist.append(item)
-        # Add dynamic-only attributes
-        for key, (expr, xlat) in repldict.items():
-            newlist.append((key, None, "insert", expr, xlat))
+        for key, value in repldict.items(): # Add dynamic-only attributes
+            item = (key, None, "insert", value)
+            newlist.append(item)
         return newlist
 
-    def emitStartElement(self, name, attrlist, taldict, metaldict, i18ndict,
+    def emitStartElement(self, name, attrlist, taldict, metaldict,
                          position=(None, None), isend=0):
-        if not taldict and not metaldict and not i18ndict:
+        if not taldict and not metaldict:
             # Handle the simple, common case
             self.emitStartTag(name, attrlist, isend)
             self.todoPush({})
@@ -476,24 +387,18 @@
 
         self.position = position
         for key, value in taldict.items():
-            if key not in TALDefs.KNOWN_TAL_ATTRIBUTES:
+            if key not in KNOWN_TAL_ATTRIBUTES:
                 raise TALError("bad TAL attribute: " + `key`, position)
             if not (value or key == 'omit-tag'):
                 raise TALError("missing value for TAL attribute: " +
                                `key`, position)
         for key, value in metaldict.items():
-            if key not in TALDefs.KNOWN_METAL_ATTRIBUTES:
+            if key not in KNOWN_METAL_ATTRIBUTES:
                 raise METALError("bad METAL attribute: " + `key`,
-                                 position)
+                position)
             if not value:
                 raise TALError("missing value for METAL attribute: " +
                                `key`, position)
-        for key, value in i18ndict.items():
-            if key not in TALDefs.KNOWN_I18N_ATTRIBUTES:
-                raise I18NError("bad i18n attribute: " + `key`, position)
-            if not value and key in ("attributes", "data", "id"):
-                raise I18NError("missing value for i18n attribute: " +
-                                `key`, position)
         todo = {}
         defineMacro = metaldict.get("define-macro")
         useMacro = metaldict.get("use-macro")
@@ -508,31 +413,13 @@
         onError = taldict.get("on-error")
         omitTag = taldict.get("omit-tag")
         TALtag = taldict.get("tal tag")
-        i18nattrs = i18ndict.get("attributes")
-        # Preserve empty string if implicit msgids are used.  We'll generate
-        # code with the msgid='' and calculate the right implicit msgid during
-        # interpretation phase.
-        msgid = i18ndict.get("translate")
-        varname = i18ndict.get('name')
-        i18ndata = i18ndict.get('data')
-
-        if i18ndata and not msgid:
-            raise I18NError("i18n:data must be accompanied by i18n:translate",
-                            position)
-
         if len(metaldict) > 1 and (defineMacro or useMacro):
             raise METALError("define-macro and use-macro cannot be used "
                              "together or with define-slot or fill-slot",
                              position)
-        if replace:
-            if content:
-                raise TALError(
-                    "tal:content and tal:replace are mutually exclusive",
-                    position)
-            if msgid is not None:
-                raise I18NError(
-                    "i18n:translate and tal:replace are mutually exclusive",
-                    position)
+        if content and replace:
+            raise TALError("content and replace are mutually exclusive",
+                           position)
 
         repeatWhitespace = None
         if repeat:
@@ -550,8 +437,8 @@
                 self.inMacroUse = 0
         else:
             if fillSlot:
-                raise METALError("fill-slot must be within a use-macro",
-                                 position)
+                raise METALError, ("fill-slot must be within a use-macro",
+                                   position)
         if not self.inMacroUse:
             if defineMacro:
                 self.pushProgram()
@@ -568,29 +455,13 @@
                 self.inMacroUse = 1
             if defineSlot:
                 if not self.inMacroDef:
-                    raise METALError(
+                    raise METALError, (
                         "define-slot must be within a define-macro",
                         position)
                 self.pushProgram()
                 todo["defineSlot"] = defineSlot
 
-        if defineSlot or i18ndict:
-            
-            domain = i18ndict.get("domain") or self.i18nContext.domain
-            source = i18ndict.get("source") or self.i18nContext.source
-            target = i18ndict.get("target") or self.i18nContext.target
-            if (  domain != DEFAULT_DOMAIN
-                  or source is not None
-                  or target is not None):
-                self.i18nContext = TranslationContext(self.i18nContext,
-                                                      domain=domain,
-                                                      source=source,
-                                                      target=target)
-                self.emit("beginI18nContext",
-                          {"domain": domain, "source": source,
-                           "target": target})
-                todo["i18ncontext"] = 1
-        if taldict or i18ndict:
+        if taldict:
             dict = {}
             for item in attrlist:
                 key, value = item[:2]
@@ -616,43 +487,16 @@
         if content:
             todo["content"] = content
         if replace:
-            # tal:replace w/ i18n:name has slightly different semantics.  What
-            # we're actually replacing then is the contents of the ${name}
-            # placeholder.
-            if varname:
-                todo['i18nvar'] = (varname, replace)
-            else:
-                todo["replace"] = replace
+            todo["replace"] = replace
             self.pushProgram()
-        # i18n:name w/o tal:replace uses the content as the interpolation
-        # dictionary values
-        elif varname:
-            todo['i18nvar'] = (varname, None)
-            self.pushProgram()
-        if msgid is not None:
-            todo['msgid'] = msgid
-        if i18ndata:
-            todo['i18ndata'] = i18ndata
         optTag = omitTag is not None or TALtag
         if optTag:
             todo["optional tag"] = omitTag, TALtag
             self.pushProgram()
-        if attrsubst or i18nattrs:
-            if attrsubst:
-                repldict = TALDefs.parseAttributeReplacements(attrsubst)
-            else:
-                repldict = {}
-            if i18nattrs:
-                i18nattrs = i18nattrs.split()
-            else:
-                i18nattrs = ()
-            # Convert repldict's name-->expr mapping to a
-            # name-->(compiled_expr, translate) mapping
+        if attrsubst:
+            repldict = parseAttributeReplacements(attrsubst)
             for key, value in repldict.items():
-                repldict[key] = self.compileExpression(value), key in i18nattrs
-            for key in i18nattrs:
-                if not repldict.has_key(key):
-                    repldict[key] = None, 1
+                repldict[key] = self.compileExpression(value)
         else:
             repldict = {}
         if replace:
@@ -663,8 +507,6 @@
             self.pushProgram()
         if content:
             self.pushProgram()
-        if msgid is not None:
-            self.pushProgram()
         if todo and position != (None, None):
             todo["position"] = position
         self.todoPush(todo)
@@ -693,10 +535,6 @@
         repldict = todo.get("repldict", {})
         scope = todo.get("scope")
         optTag = todo.get("optional tag")
-        msgid = todo.get('msgid')
-        i18ncontext = todo.get("i18ncontext")
-        varname = todo.get('i18nvar')
-        i18ndata = todo.get('i18ndata')
 
         if implied > 0:
             if defineMacro or useMacro or defineSlot or fillSlot:
@@ -708,51 +546,14 @@
             raise exc("%s attributes on <%s> require explicit </%s>" %
                       (what, name, name), position)
 
-        # If there's no tal:content or tal:replace in the tag with the
-        # i18n:name, tal:replace is the default.
-        i18nNameAction = I18N_REPLACE
         if content:
-            if varname:
-                i18nNameAction = I18N_CONTENT
             self.emitSubstitution(content, {})
-        # If we're looking at an implicit msgid, emit the insertTranslation
-        # opcode now, so that the end tag doesn't become part of the implicit
-        # msgid.  If we're looking at an explicit msgid, it's better to emit
-        # the opcode after the i18nVariable opcode so we can better handle
-        # tags with both of them in them (and in the latter case, the contents
-        # would be thrown away for msgid purposes).
-        if msgid is not None and not varname:
-            self.emitTranslation(msgid, i18ndata)
         if optTag:
             self.emitOptTag(name, optTag, isend)
         elif not isend:
-            # If we're processing the end tag for a tag that contained
-            # i18n:name, we need to make sure that optimize() won't collect
-            # immediately following end tags into the same rawtextOffset, so
-            # put a spacer here that the optimizer will recognize.
-            if varname:
-                self.emit('noop')
             self.emitEndTag(name)
-        # If i18n:name appeared in the same tag as tal:replace then we're
-        # going to do the substitution a little bit differently.  The results
-        # of the expression go into the i18n substitution dictionary.
         if replace:
             self.emitSubstitution(replace, repldict)
-        elif varname:
-            if varname[1] is not None:
-                i18nNameAction = I18N_EXPRESSION
-            # o varname[0] is the variable name
-            # o i18nNameAction is either
-            #   - I18N_REPLACE for implicit tal:replace
-            #   - I18N_CONTENT for tal:content
-            #   - I18N_EXPRESSION for explicit tal:replace
-            # o varname[1] will be None for the first two actions and the
-            #   replacement tal expression for the third action.
-            self.emitI18nVariable(varname[0], i18nNameAction, varname[1])
-        # Do not test for "msgid is not None", i.e. we only want to test for
-        # explicit msgids here.  See comment above.
-        if msgid is not None and varname:
-            self.emitTranslation(msgid, i18ndata)
         if repeat:
             self.emitRepeat(repeat)
         if condition:
@@ -761,10 +562,6 @@
             self.emitOnError(name, onError)
         if scope:
             self.emit("endScope")
-        if i18ncontext:
-            self.emit("endI18nContext")
-            assert self.i18nContext.parent is not None
-            self.i18nContext = self.i18nContext.parent
         if defineSlot:
             self.emitDefineSlot(defineSlot)
         if fillSlot:


=== Zope/lib/python/TAL/TALInterpreter.py 1.69.4.2 => 1.69.4.3 === (460/560 lines abridged)
--- Zope/lib/python/TAL/TALInterpreter.py:1.69.4.2	Sat Sep 28 21:40:35 2002
+++ Zope/lib/python/TAL/TALInterpreter.py	Tue Oct  8 14:41:13 2002
@@ -8,7 +8,7 @@
 # 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.
+# FOR A PARTICULAR PURPOSE
 #
 ##############################################################################
 """
@@ -17,17 +17,17 @@
 
 import sys
 import getopt
-import re
-from types import ListType
+
 from cgi import escape
-# Do not use cStringIO here!  It's not unicode aware. :(
-from StringIO import StringIO
-from DocumentTemplate.DT_Util import ustr
 
-from TALDefs import TAL_VERSION, TALError, METALError
+try:
+    from cStringIO import StringIO
+except ImportError:
+    from StringIO import StringIO
+
+from TALDefs import quote, TAL_VERSION, TALError, METALError
 from TALDefs import isCurrentVersion, getProgramVersion, getProgramMode
 from TALGenerator import TALGenerator
-from TranslationContext import TranslationContext
 
 BOOLEAN_HTML_ATTRS = [
     # List of Boolean attributes in HTML that should be rendered in
@@ -40,12 +40,13 @@
     "defer"
 ]
 
-def normalize(text):
-    # Now we need to normalize the whitespace in implicit message ids and
-    # implicit $name substitution values by stripping leading and trailing
-    # whitespace, and folding all internal whitespace to a single space.
-    return ' '.join(text.split())
-
+EMPTY_HTML_TAGS = [
+    # List of HTML tags with an empty content model; these are
+    # rendered in minimized form, e.g. <img />.
+    # From http://www.w3.org/TR/xhtml1/#dtds

[-=- -=- -=- 460 lines omitted -=- -=- -=-]

-        self.stream = stream = self.StringIO()
+        self.stream = stream = StringIO()
         self._stream_write = stream.write
         try:
             self.interpret(block)
@@ -732,24 +598,24 @@
     bytecode_handlers_tal["optTag"] = do_optTag_tal
 
 
-class FasterStringIO(StringIO):
-    """Append-only version of StringIO.
-
-    This let's us have a much faster write() method.
-    """
-    def close(self):
-        if not self.closed:
-            self.write = _write_ValueError
-            StringIO.close(self)
-
-    def seek(self, pos, mode=0):
-        raise RuntimeError("FasterStringIO.seek() not allowed")
-
-    def write(self, s):
-        #assert self.pos == self.len
-        self.buflist.append(s)
-        self.len = self.pos = self.pos + len(s)
-
+def test():
+    from driver import FILE, parsefile
+    from DummyEngine import DummyEngine
+    try:
+        opts, args = getopt.getopt(sys.argv[1:], "")
+    except getopt.error, msg:
+        print msg
+        sys.exit(2)
+    if args:
+        file = args[0]
+    else:
+        file = FILE
+    doc = parsefile(file)
+    compiler = TALCompiler(doc)
+    program, macros = compiler()
+    engine = DummyEngine()
+    interpreter = TALInterpreter(program, macros, engine)
+    interpreter()
 
-def _write_ValueError(s):
-    raise ValueError, "I/O operation on closed file"
+if __name__ == "__main__":
+    test()