7 Templates: Separation of View
(require web-server/templates) | package: web-server-lib |
The Web Server provides a powerful Web template system for separating the presentation logic of a Web application and enabling non-programmers to contribute to Racket-based Web applications.
Although all the examples here generate HTML, the template language and the Text Generation it is based on can be used to generate any text-based format: C, SQL, form emails, reports, etc.
7.1 Static
<html> |
<head><title>Fastest Templates in the West!</title></head> |
<body> |
<h1>Bang!</h1> |
<h2>Bang!</h2> |
</body> |
</html> |
(include-template "static.html")
"<html>\n <head><title>Fastest Templates in the West!</title></head>\n <body>\n <h1>Bang!</h1>\n <h2>Bang!</h2>\n </body>\n</html>"
7.2 Dynamic
<html> |
<head><title>Fastest @thing in the West!</title></head> |
<body> |
<h1>Bang!</h1> |
<h2>Bang!</h2> |
</body> |
</html> |
(let ([thing "Templates"]) (include-template "simple.html"))
(define (fast-template thing) (include-template "simple.html")) (fast-template "Templates") (fast-template "Noodles")
<html> |
<head><title>Fastest Templates in the West!</title></head> |
<body> |
<h1>Bang!</h1> |
<h2>Bang!</h2> |
</body> |
</html> |
and
<html> |
<head><title>Fastest Noodles in the West!</title></head> |
<body> |
<h1>Bang!</h1> |
<h2>Bang!</h2> |
</body> |
</html> |
Furthermore, there are no constraints on the Racket used by templates: they can use macros, structs, continuation marks, threads, etc. However, Racket values that are ultimately returned must be printable by the Text Generation. For example, consider the following outputs of the title line of different calls to fast-template:
(fast-template 'Templates)
...<title>Fastest Templates in the West!</title>...
(fast-template 42)
...<title>Fastest 42 in the West!</title>...
(fast-template (list "Noo" "dles"))
...<title>Fastest Noodles in the West!</title>...
(fast-template (lambda () "Thunks"))
...<title>Fastest Thunks in the West!</title>...
(fast-template (delay "Laziness"))
...<title>Fastest Laziness in the West!</title>...
(fast-template (fast-template "Embedding"))
...<title>Fastest ...<title>Fastest Embedding in the West!</title>... in the West!</title>...
7.3 Gotchas
<head><title>Fastest @s in the West!</title></head> |
<head><title>Fastest @"@"s in the West!</title></head> |
<head><title>Fastest @thing in the @place!</title></head> |
<head><title>Fastest @thing in the @|place|!</title></head> |
<table> |
@for[([c clients])]{ |
<tr><td>@(car c), @(cdr c)</td></tr> |
} |
</table> |
<table> |
</table> |
<table> |
@for/list[([c clients])]{ |
<tr><td>@(car c), @(cdr c)</td></tr> |
} |
</table> |
<table> |
</tr> |
</tr> |
</table> |
<table> |
@for/list[([c clients])]{ |
@list{ |
<tr><td>@(car c), @(cdr c)</td></tr> |
} |
} |
</table> |
<table> |
<tr><td>Young, Brigham</td></tr> |
<tr><td>Smith, Joseph</td></tr> |
</table> |
<table> |
@in[c clients]{ |
<tr><td>@(car c), @(cdr c)</td></tr> |
} |
</table> |
7.4 Escaping
Thanks to Michael W. for this section.
Because templates are useful for many things (scripts, CSS, HTML, etc), the Web Server does not assume that the template is for XML-like content. Therefore when when templates are expanded, no XML escaping is done by default. Beware of cross-site scripting vulnerabilities! For example, suppose a servlet serves the following template where some-variable is an input string supplied by the client:
<html> |
<head><title>Fastest Templates in the West!</title></head> |
<body> |
@some-variable |
</body> |
</html> |
If the servlet contains something like the following:
(let ([some-variable (get-input-from-user)]) (include-template "static.htm"))
There is nothing to prevent an attacker from entering <script type="text/javascript">...</script> to make the template expand into:
<html> |
<head><title>Fastest Templates in the West!</title></head> |
<body> |
<script type="text/javascript">...</script> |
</body> |
</html> |
Now the server will send the attacker’s code to millions of innocent users. To keep this from happening when serving HTML, use the xexpr->string function from the xml module.
This can be done in the servlet:
(require xml) (let ([some-variable (xexpr->string (get-input-from-user))]) (include-template "static.htm"))
Alternatively, make the template responsible for its own escaping:
<html> |
<head><title>Fastest Templates in the West!</title></head> |
<body> |
@(xexpr->string some-variable) |
</body> |
</html> |
The improved version renders as:
<html> |
<head><title>Fastest Templates in the West!</title></head> |
<body> |
<script type=\"text/javascript\">...</script> |
</body> |
</html> |
When writing templates, always remember to escape user-supplied input.
7.5 HTTP Responses
(response/full 200 #"Okay" (current-seconds) TEXT/HTML-MIME-TYPE empty (list (string->bytes/utf-8 (include-template "static.html"))))
`(html ,(include-template "static.html"))
`(html ,(make-cdata #f #f (include-template "static.html")))
7.6 API Details
语法
(include-template path-spec)
(include-template #:command-char command-char path-spec)
(include-template "static.html") (include-template #:command-char #\$ "dollar-static.html")
语法
(in x xs e ...)
@in[c clients]{ |
<tr><td>@(car c), @(cdr c)</td></tr> |
} |
7.7 Conversion Example
Al Church has been maintaining a blog with Racket for some years and would like to convert to web-server/templates.
(define-struct post (title body)) (define posts (list (make-post "(Y Y) Works: The Why of Y" "Why is Y, that is the question.") (make-post "Church and the States" "As you may know, I grew up in DC, not technically a state.")))
He has divided his code into presentation functions and logic functions. We’ll look at the presentation functions first.
(define (template section body) (response/xexpr `(html (head (title "Al's Church: " ,section)) (body (h1 "Al's Church: " ,section) (div ([id "main"]) ,@body)))))
One of the things to notice here is the unquote-splicing on the body argument. This indicates that the body is list of X-expressions. If he had accidentally used only unquote then there would be an error in converting the return value to an HTTP response.
(define (blog-posted title body k-url) `((h2 ,title) (p ,body) (h1 (a ([href ,k-url]) "Continue"))))
Here’s an example of simple body that uses a list of X-expressions to show the newly posted blog entry, before continuing to redisplay the main page. Let’s look at a more complicated body:
(define (blog-posts k-url) (append (apply append (for/list ([p posts]) `((h2 ,(post-title p)) (p ,(post-body p))))) `((h1 "New Post") (form ([action ,k-url]) (input ([name "title"])) (input ([name "body"])) (input ([type "submit"]))))))
This function shows a number of common patterns that are required by X-expressions. First, append is used to combine different X-expression lists. Second, apply append is used to collapse and combine the results of a for/list where each iteration results in a list of X-expressions. We’ll see that these patterns are unnecessary with templates. Another annoying patterns shows up when Al tries to add CSS styling and some JavaScript from Google Analytics to all the pages of his blog. He changes the template function to:
(define (template section body) (response/xexpr `(html (head (title "Al's Church: " ,section) (style ([type "text/css"]) "body {margin: 0px; padding: 10px;}" "#main {background: #dddddd;}")) (body (script ([type "text/javascript"]) ,(make-cdata #f #f "var gaJsHost = ((\"https:\" ==" "document.location.protocol)" "? \"https://ssl.\" : \"http://www.\");" "document.write(unescape(\"%3Cscript src='\" + gaJsHost" "+ \"google-analytics.com/ga.js' " "type='text/javascript'%3E%3C/script%3E\"));")) (script ([type "text/javascript"]) ,(make-cdata #f #f "var pageTracker = _gat._getTracker(\"UA-YYYYYYY-Y\");" "pageTracker._trackPageview();")) (h1 "Al's Church: " ,section) (div ([id "main"]) ,@body)))))
Some of these problems go away by using here strings, as described in the documentation on Reading Strings.
The first thing we notice is that encoding CSS as a string is rather primitive. Encoding JavaScript with strings is even worse for two reasons: first, we are more likely to need to manually escape characters such as "; second, we need to use a CDATA object, because most JavaScript code uses characters that "need" to be escaped in XML, such as &, but most browsers will fail if these characters are entity-encoded. These are all problems that go away with templates.
(define (extract-post req) (define binds (request-bindings req)) (define title (extract-binding/single 'title binds)) (define body (extract-binding/single 'body binds)) (set! posts (list* (make-post title body) posts)) (send/suspend (lambda (k-url) (template "Posted" (blog-posted title body k-url)))) (display-posts)) (define (display-posts) (extract-post (send/suspend (lambda (k-url) (template "Posts" (blog-posts k-url)))))) (define (start req) (display-posts))
To use templates, we need only change template, blog-posted, and blog-posts:
(define (template section body) (response/full 200 #"Okay" (current-seconds) TEXT/HTML-MIME-TYPE empty (list (string->bytes/utf-8 (include-template "blog.html"))))) (define (blog-posted title body k-url) (include-template "blog-posted.html")) (define (blog-posts k-url) (include-template "blog-posts.html"))
Each of the templates are given below:
<html> |
<head> |
<title>Al's Church: @|section|</title> |
<style type="text/css"> |
body { |
margin: 0px; |
padding: 10px; |
} |
|
#main { |
background: #dddddd; |
} |
</style> |
</head> |
<body> |
<script type="text/javascript"> |
var gaJsHost = (("https:" == document.location.protocol) ? |
"https://ssl." : "http://www."); |
document.write(unescape("%3Cscript src='" + gaJsHost + |
"google-analytics.com/ga.js' |
type='text/javascript'%3E%3C/script%3E")); |
</script> |
<script type="text/javascript"> |
var pageTracker = _gat._getTracker("UA-YYYYYYY-Y"); |
pageTracker._trackPageview(); |
</script> |
|
<h1>Al's Church: @|section|</h1> |
<div id="main"> |
@body |
</div> |
</body> |
</html> |
Notice that this part of the presentation is much simpler, because the CSS and JavaScript can be included verbatim, without resorting to any special escape-escaping patterns. Similarly, since the body is represented as a string, there is no need to remember if splicing is necessary.
<h2>@|title|</h2> |
<p>@|body|</p> |
|
<h1><a href="@|k-url|">Continue</a></h1> |
@in[p posts]{ |
<h2>@(post-title p)</h2> |
<p>@(post-body p)</p> |
} |
|
<h1>New Post</h1> |
<form action="@|k-url|"> |
<input name="title" /> |
<input name="body" /> |
<input type="submit" /> |
</form> |
Compare this template with the original presentation function: there is no need to worry about managing how lists are nested: the defaults just work.