Placeholders ============ Simple placeholders ------------------- Let's add a few $placeholders to our template: :: >>> from Cheetah.Template import Template >>> values = {'what': 'surreal', 'punctuation': '?'} >>> t = Template("""\ ... Hello, $what world$punctuation ... One of Python's least-used functions is $xrange. ... """, [values]) >>> print t Hello, surreal world? One of Python's least-used functions is . >>> print t.generatedModuleCode() 1 #!/usr/bin/env python 2 """ 3 Autogenerated by CHEETAH: The Python-Powered Template Engine 4 CHEETAH VERSION: 0.9.12 5 Generation time: Sun Apr 21 00:53:01 2002 6 """ 7 __CHEETAH_genTime__ = 'Sun Apr 21 00:53:01 2002' 8 __CHEETAH_version__ = '0.9.12' 9 ################################################## 10 ## DEPENDENCIES 11 import sys 12 import os 13 import os.path 14 from os.path import getmtime, exists 15 import time 16 import types 17 from Cheetah.Template import Template 18 from Cheetah.DummyTransaction import DummyTransaction 19 from Cheetah.NameMapper import NotFound, valueForName, valueFromSearchList 20 import Cheetah.Filters as Filters 21 import Cheetah.ErrorCatchers as ErrorCatchers 22 ################################################## 23 ## MODULE CONSTANTS 24 try: 25 True, False 26 except NameError: 27 True, False = (1==1), (1==0) 28 ################################################## 29 ## CLASSES 30 class GenTemplate(Template): 31 """ 32 33 Autogenerated by CHEETAH: The Python-Powered Template Engine 34 """ 35 ################################################## 36 ## GENERATED METHODS :: 37 def __init__(self, *args, **KWs): 38 """ 39 40 """ 41 Template.__init__(self, *args, **KWs) 42 def respond(self, 43 trans=None, 44 dummyTrans=False, 45 VFS=valueFromSearchList, 46 VFN=valueForName, 47 getmtime=getmtime, 48 currentTime=time.time): 49 """ 50 This is the main method generated by Cheetah 51 """ 52 if not trans: 53 trans = DummyTransaction() 54 dummyTrans = True 55 write = trans.response().write 56 SL = self._searchList 57 filter = self._currentFilter 58 globalSetVars = self._globalSetVars 59 60 ######################################## 61 ## START - generated method body 62 63 write('Hello, ') 64 write(filter(VFS(SL,"what",1))) # generated from '$what' at # line 1, col 8. 65 write(' world') 66 write(filter(VFS(SL,"punctuation",1))) # generated from # '$punctuation' at line 1, col 19. 67 write("\nOne of Python's least-used methods is ") 68 write(filter(xrange)) # generated from '$xrange' at line 2, # col 39. 69 write('.\n') 70 71 ######################################## 72 ## END - generated method body 73 74 if dummyTrans: 75 return trans.response().getvalue() 76 else: 77 return "" :: 78 79 ################################################## 80 ## GENERATED ATTRIBUTES 81 __str__ = respond 82 _mainCheetahMethod_for_GenTemplate= 'respond' 83 # CHEETAH was developed by Tavis Rudd, Chuck Esterbrook, Ian Bicking # and Mike Orr; 84 # with code, advice and input from many other volunteers. 85 # For more information visit https://cheetahtemplate.org/ 86 ################################################## 87 ## if run from command line: 88 if __name__ == '__main__': 89 GenTemplate().runAsMainProgram() (Again, I have added line numbers and split the lines as in the previous chapter.) This generated template module is different from the previous one in several trivial respects and one important respect. Trivially, {.\_filePath} and {.\_fileMtime} are not updated in {.\_\_init\_\_}, so they inherit the value {None} from {Template}. Also, that if-stanza in {.respond} that recompiles the template if the source file changes is missing - because there is no source file. So this module is several lines shorter than the other one. But the important way this module is different is that instead of the one {write} call outputting a string literal, this module has a series of {write} calls (lines 63-69) outputting successive chunks of the template. Regular text has been translated into a string literal, and placeholders into function calls. Every placeholder is wrapped inside a {filter} call to apply the current output filter. (The default output filter converts all objects to strings, and {None} to {""}.) Placeholders referring to a Python builtin like {xrange} (line 68) generate a bare variable name. Placeholders to be looked up in the searchList have a nested function call; e.g., :: write(filter(VFS(SL,"what",1))) # generated from '$what' at line 1, col 8. {VFS}, remember, is a function imported from {Cheetah.NameMapper} that looks up a value in a searchList. So we pass it the searchList, the name to look up, and a boolean (1) indicating we want autocalling. (It's {1} rather than {True} because it's generated from an {and} expression, and that's what Python 2.2 outputs for true {and} expressions.) Complex placeholders -------------------- Placeholders can get far more complicated than that. This example shows what kind of code the various NameMapper features produce. The formulas are taken from Cheetah's test suite, in the {Cheetah.Tests.SyntaxAndOutput.Placeholders} class. :: 1 placeholder: $aStr 2 placeholders: $aStr $anInt 2 placeholders, back-to-back: $aStr$anInt 1 placeholder enclosed in {}: ${aStr} 1 escaped placeholder: \$var func placeholder - with (): $aFunc() func placeholder - with (int): $aFunc(1234) func placeholder - with (string): $aFunc('aoeu') func placeholder - with ('''\nstring'\n'''): $aFunc('''\naoeu'\n''') func placeholder - with (string*int): $aFunc('aoeu'*2) func placeholder - with (int*float): $aFunc(2*2.0) Python builtin values: $None $True $False func placeholder - with ($arg=float): $aFunc($arg=4.0) deeply nested argstring: $aFunc( $arg = $aMeth( $arg = $aFunc( 1 ) ) ): function with None: $aFunc(None) autocalling: $aFunc! $aFunc(). nested autocalling: $aFunc($aFunc). list subscription: $aList[0] list slicing: $aList[:2] list slicing and subcription combined: $aList[:2][0] dict - NameMapper style: $aDict.one dict - Python style: $aDict['one'] dict combined with autocalled string method: $aDict.one.upper dict combined with string method: $aDict.one.upper() nested dict - NameMapper style: $aDict.nestedDict.two nested dict - Python style: $aDict['nestedDict']['two'] nested dict - alternating style: $aDict['nestedDict'].two nested dict - NameMapper style + method: $aDict.nestedDict.two.upper nested dict - alternating style + method: $aDict['nestedDict'].two.upper nested dict - NameMapper style + method + slice: $aDict.nestedDict.two.upper[:4] nested dict - Python style, variable key: $aDict[$anObj.meth('nestedDict')].two object method: $anObj.meth1 object method + complex slice: $anObj.meth1[0: ((4/4*2)*2)/$anObj.meth1(2) ] very complex slice: $( anObj.meth1[0: ((4/4*2)*2)/$anObj.meth1(2) ] ) $_('a call to gettext') We'll need a big program to set up the placeholder values. Here it is: :: #!/usr/bin/env python from ComplexExample import ComplexExample try: # Python >= 2.2.1 True, False except NameError: # Older Python True, False = (1==1), (1==0) class DummyClass: _called = False def __str__(self): return 'object' def meth(self, arg="arff"): return str(arg) def meth1(self, arg="doo"): return arg def meth2(self, arg1="a1", arg2="a2"): return str(arg1) + str(arg2) def callIt(self, arg=1234): self._called = True self._callArg = arg def dummyFunc(arg="Scooby"): return arg defaultTestNameSpace = { 'aStr':'blarg', 'anInt':1, 'aFloat':1.5, 'aList': ['item0','item1','item2'], 'aDict': {'one':'item1', 'two':'item2', 'nestedDict':{1:'nestedItem1', 'two':'nestedItem2' }, 'nestedFunc':dummyFunc, }, 'aFunc': dummyFunc, 'anObj': DummyClass(), 'aMeth': DummyClass().meth1, '_': lambda x: 'translated ' + x } print ComplexExample( searchList=[defaultTestNameSpace] ) Here's the output: :: 1 placeholder: blarg 2 placeholders: blarg 1 2 placeholders, back-to-back: blarg1 1 placeholder enclosed in {}: blarg 1 escaped placeholder: $var func placeholder - with (): Scooby func placeholder - with (int): 1234 func placeholder - with (string): aoeu func placeholder - with ('''\nstring'\n'''): aoeu' func placeholder - with (string*int): aoeuaoeu func placeholder - with (int*float): 4.0 Python builtin values: 1 0 func placeholder - with ($arg=float): 4.0 deeply nested argstring: 1: function with None: autocalling: Scooby! Scooby. nested autocalling: Scooby. list subscription: item0 list slicing: ['item0', 'item1'] list slicing and subcription combined: item0 dict - NameMapper style: item1 dict - Python style: item1 dict combined with autocalled string method: ITEM1 dict combined with string method: ITEM1 nested dict - NameMapper style: nestedItem2 nested dict - Python style: nestedItem2 nested dict - alternating style: nestedItem2 nested dict - NameMapper style + method: NESTEDITEM2 nested dict - alternating style + method: NESTEDITEM2 nested dict - NameMapper style + method + slice: NEST nested dict - Python style, variable key: nestedItem2 object method: doo object method + complex slice: do very complex slice: do translated a call to gettext And here - tada! - is the generated module. To save space, I've included only the lines containing the {write} calls. The rest of the module is the same as in the first example, chapter pyModules.example. I've split some of the lines to make them fit on the page. :: 1 write('1 placeholder: ') 2 write(filter(VFS(SL,"aStr",1))) # generated from '$aStr' at line 1, col 16. 3 write('\n2 placeholders: ') 4 write(filter(VFS(SL,"aStr",1))) # generated from '$aStr' at line 2, col 17. 5 write(' ') 6 write(filter(VFS(SL,"anInt",1))) # generated from '$anInt' at line 2, col 23. 7 write('\n2 placeholders, back-to-back: ') 8 write(filter(VFS(SL,"aStr",1))) # generated from '$aStr' at line 3, col 31. 9 write(filter(VFS(SL,"anInt",1))) # generated from '$anInt' at line 3, col 36. 10 write('\n1 placeholder enclosed in {}: ') 11 write(filter(VFS(SL,"aStr",1))) # generated from '${aStr}' at line 4, # col 31. 12 write('\n1 escaped placeholder: $var\nfunc placeholder - with (): ') 13 write(filter(VFS(SL,"aFunc",0)())) # generated from '$aFunc()' at line 6, # col 29. 14 write('\nfunc placeholder - with (int): ') 15 write(filter(VFS(SL,"aFunc",0)(1234))) # generated from '$aFunc(1234)' at # line 7, col 32. 16 write('\nfunc placeholder - with (string): ') 17 write(filter(VFS(SL,"aFunc",0)('aoeu'))) # generated from "$aFunc('aoeu')" # at line 8, col 35. 18 write("\nfunc placeholder - with ('''\\nstring'\\n'''): ") 19 write(filter(VFS(SL,"aFunc",0)('''\naoeu'\n'''))) # generated from # "$aFunc('''\\naoeu'\\n''')" at line 9, col 46. 20 write('\nfunc placeholder - with (string*int): ') 21 write(filter(VFS(SL,"aFunc",0)('aoeu'*2))) # generated from # "$aFunc('aoeu'*2)" at line 10, col 39. 22 write('\nfunc placeholder - with (int*float): ') 23 write(filter(VFS(SL,"aFunc",0)(2*2.0))) # generated from '$aFunc(2*2.0)' # at line 11, col 38. 24 write('\nPython builtin values: ') 25 write(filter(None)) # generated from '$None' at line 12, col 24. 26 write(' ') 27 write(filter(True)) # generated from '$True' at line 12, col 30. 28 write(' ') 29 write(filter(False)) # generated from '$False' at line 12, col 36. 30 write('\nfunc placeholder - with ($arg=float): ') 31 write(filter(VFS(SL,"aFunc",0)(arg=4.0))) # generated from # '$aFunc($arg=4.0)' at line 13, col 40. 32 write('\ndeeply nested argstring: ') 33 write(filter(VFS(SL,"aFunc",0)( arg = VFS(SL,"aMeth",0)( arg = VFS(SL,"aFunc",0)( 1 ) ) ))) # generated from '$aFunc( $arg = $aMeth( $arg = $aFunc( 1 ) ) )' # at line 14, col 26. 34 write(':\nfunction with None: ') 35 write(filter(VFS(SL,"aFunc",0)(None))) # generated from '$aFunc(None)' at # line 15, col 21. 36 write('\nautocalling: ') 37 write(filter(VFS(SL,"aFunc",1))) # generated from '$aFunc' at line 16, # col 14. 38 write('! ') 39 write(filter(VFS(SL,"aFunc",0)())) # generated from '$aFunc()' at line 16, # col 22. :: 40 write('.\nnested autocalling: ') 41 write(filter(VFS(SL,"aFunc",0)(VFS(SL,"aFunc",1)))) # generated from # '$aFunc($aFunc)' at line 17, col 21. 42 write('.\nlist subscription: ') 43 write(filter(VFS(SL,"aList",1)[0])) # generated from '$aList[0]' at line # 18, col 20. 44 write('\nlist slicing: ') 45 write(filter(VFS(SL,"aList",1)[:2])) # generated from '$aList[:2]' at # line 19, col 15. 46 write('\nlist slicing and subcription combined: ') 47 write(filter(VFS(SL,"aList",1)[:2][0])) # generated from '$aList[:2][0]' # at line 20, col 40. 48 write('\ndict - NameMapper style: ') 49 write(filter(VFS(SL,"aDict.one",1))) # generated from '$aDict.one' at line # 21, col 26. 50 write('\ndict - Python style: ') 51 write(filter(VFS(SL,"aDict",1)['one'])) # generated from "$aDict['one']" # at line 22, col 22. 52 write('\ndict combined with autocalled string method: ') 53 write(filter(VFS(SL,"aDict.one.upper",1))) # generated from # '$aDict.one.upper' at line 23, col 46. 54 write('\ndict combined with string method: ') 55 write(filter(VFN(VFS(SL,"aDict.one",1),"upper",0)())) # generated from # '$aDict.one.upper()' at line 24, col 35. 56 write('\nnested dict - NameMapper style: ') 57 write(filter(VFS(SL,"aDict.nestedDict.two",1))) # generated from # '$aDict.nestedDict.two' at line 25, col 33. 58 write('\nnested dict - Python style: ') 59 write(filter(VFS(SL,"aDict",1)['nestedDict']['two'])) # generated from # "$aDict['nestedDict']['two']" at line 26, col 29. 60 write('\nnested dict - alternating style: ') 61 write(filter(VFN(VFS(SL,"aDict",1)['nestedDict'],"two",1))) # generated # from "$aDict['nestedDict'].two" at line 27, col 34. 62 write('\nnested dict - NameMapper style + method: ') 63 write(filter(VFS(SL,"aDict.nestedDict.two.upper",1))) # generated from # '$aDict.nestedDict.two.upper' at line 28, col 42. 64 write('\nnested dict - alternating style + method: ') 65 write(filter(VFN(VFS(SL,"aDict",1)['nestedDict'],"two.upper",1))) # generated from "$aDict['nestedDict'].two.upper" at line 29, col 43. 66 write('\nnested dict - NameMapper style + method + slice: ') :: 67 write(filter(VFN(VFS(SL,"aDict.nestedDict.two",1),"upper",1)[:4])) # generated from '$aDict.nestedDict.two.upper[:4]' at line 30, col 50. 68 write('\nnested dict - Python style, variable key: ') 69 write(filter(VFN(VFS(SL,"aDict",1) [VFN(VFS(SL,"anObj",1),"meth",0)('nestedDict')],"two",1))) # generated from "$aDict[$anObj.meth('nestedDict')].two" at line 31, # col 43. 70 write('\nobject method: ') 71 write(filter(VFS(SL,"anObj.meth1",1))) # generated from '$anObj.meth1' at # line 32, col 16. 72 write('\nobject method + complex slice: ') 73 write(filter(VFN(VFS(SL,"anObj",1),"meth1",1) [0: ((4/4*2)*2)/VFN(VFS(SL,"anObj",1),"meth1",0)(2) ])) # generated from '$anObj.meth1[0: ((4/4*2)*2)/$anObj.meth1(2) ]' # at line 33, col 32. 74 write('\nvery complex slice: ') 75 write(filter(VFN(VFS(SL,"anObj",1),"meth1",1) [0: ((4/4*2)*2)/VFN(VFS(SL,"anObj",1),"meth1",0)(2) ] )) # generated from '$( anObj.meth1[0: ((4/4*2)*2)/$anObj.meth1(2) ] )' # at line 34, col 21. 76 if False: 77 _('foo') 78 write(filter(VFS(SL,"_",0)("a call to gettext"))) # generated from "$_('a call to gettext')" # at line 35, col 1. 79 write('\n') For each placeholder lookup, the the innermost level of nesting is a {VFS} call, which looks up the first (leftmost) placeholder component in the searchList. This is wrapped by zero or more {VFN} calls, which perform Universal Dotted Notation lookup on the next dotted component of the placeholder, looking for an attribute or key by that name within the previous object (not in the searchList). Autocalling is performed by {VFS} and {VFN}: that's the reason for their third argument. Explicit function/method arguments, subscripts and keys (which are all expressions) are left unchanged, besides expanding any embedded $placeholders in them. This means they must result in valid Python expressions, following the standard Python quoting rules. Built-in Python values ({None}, {True} and {False}) are converted to {filter(None)}, etc. They use normal Python variable lookup rather than {VFS}. (Cheetah emulates {True} and {False} using global variables for Python < 2.2.1, when they weren't builtins yet.) Notice the last line is a call to {\_} (i.e. {gettext}) which is used for internationalization (see http://docs.python.org/lib/module-gettext.html). The code is converted normally, but an {if False} block is used so that gettext can successfully mark the string for translation when parsing the generated Python. Otherwise, the NameMapper syntax would get in the way of this.