Most templating systems provide commands that you embed in the HTML to repeat elements or include fragments from other files. This works fairly well for simple constructs; however, as soon as the programmer wants to make the logic even slightly more complicated, the templating system must be bent in ways it was never meant to be used.
The theory behind DOMTemplate
is that Python code, rather than special syntax in the HTML template, should be used to manipulate the structure of the HTML. DOMTemplate
uses the Document Object Model (DOM), a W3 standard tree-based representation of an HTML document. The DOM provides an API that allows you to traverse nodes in the tree, examine their attributes, move, add, and delete them. For more information on using DOM in Python, see The xml.dom module documentation.
You'll need to start a plain Twisted Web server.
mktap web --path ~/public_html twistd -f web.tap
For more detailed information on this, see the guide to Installing and Using Twisted.Web.
Once you do this you should be able to throw any files in ~/public_html
, and they'll be served on localhost:8080. Twisted Web also supports a number of special script types; the one we'll be using here is an .rpy
, or Resource Script.
A Resource Script is simply a python file ending with the extension .rpy
, which is required to create an instance of a (subclass of a) twisted.web.resource.Resource
. The Resource
subclass we'll be using in this example is, of course, DOMTemplate
.
Make sure the TwistedQuotes
directory is on your PYTHONPATH
, put webquotes.rpy
and WebQuotes.html
in your ~/public_html
directory, and you are ready to go.
There are three files involved in this example; webquoteresource.py
, WebQuotes.html
, and webquotes.rpy
. webquoteresource.py
is a normal python module and contains the class definition which will be used, a DOMTemplate
subclass. WebQuotes.html
is placed in the web directory and is looked up at runtime by the DOMTemplate
machinery. It is converted into a DOM tree which is iterated during page rendering. Finally, webquotes.rpy
is placed in the web directory; each time the URL is visited, the file is executed; it import
s webquoteresource.QuoteResource
and instantiates it. This instance is asked to render the page.
A DOMTemplate
subclass must do two things: specify a template, and provide methods to handle specific nodes in the template. The first simply requires either a template
attribute, which should be a string, or a templateFile
attribute, which should be a file name, specifying the XHTML template. To accomplish the second, we define methods with the prefix factory_
in our subclass. When the template is rendered, DOMTemplate
will look for the view
attribute on any HTML node. If one is found, the corresponding factory method will be called to handle the node.
Listing 1: webquoteresource.py: Twisted Quotes Web Resource module
In our example template, we insert a view
attribute onto the <title>
node with the value 'getTitle'
. We also insert a view
attribute on a <h1>
node with the value 'getTitle'
; this shows the ability of DOMTemplate
to reuse functionality while applying formatting defined by the template to the output. We also insert a view
attribute onto a <pre>
node with the value 'getQuote'
. This is where the real action will take place.
When the DOM is iterated and nodes with view attributes are found, DOMTemplate
will look in the instance's namespace for a method with the corresponding name, prefixed
by factory_
.
Finally, we need an .rpy
file in the web directory that twisted can find and execute. This is simply a matter of importing our module, instantiating our class, and assigning the instance to a variable named resource
. Twisted will discover this instance and call render
on it, causing the DOMTemplate
to be rendered.