#lang scribble/doc @(require "web-server.ss") @(require (for-label web-server/servlet web-server/templates scheme/promise scheme/list xml)) @(define xexpr @tech[#:doc '(lib "xml/xml.scrbl")]{X-expression}) @(define at-reader-ref @secref[#:doc '(lib "scribblings/scribble/scribble.scrbl")]{reader}) @(define text-ref @secref[#:doc '(lib "scribblings/scribble/scribble.scrbl")]{preprocessor}) @title[#:tag "templates"]{Templates: Separation of View} @defmodule[web-server/templates] 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 PLT-based Web applications. @margin-note{Although all the examples here generate HTML, the template language and the @text-ref it is based on can be used to generate any text-based format: C, SQL, form emails, reports, etc.} @section{Static} Suppose we have a file @filepath{static.html} with the contents: @verbatim[#:indent 2]|{ Fastest Templates in the West!

Bang!

Bang!

}| If we write the following in our code: @schemeblock[ (include-template "static.html") ] Then the contents of @filepath{static.html} will be read @emph{at compile time} and compiled into a Scheme program that returns the contents of @filepath{static.html} as a string: @schemeblock[ "\n Fastest Templates in the West!\n \n

Bang!

\n

Bang!

\n \n" ] @section{Dynamic} @scheme[include-template] gives the template access to the @emph{complete lexical context} of the including program. This context can be accessed via the @at-reader-ref syntax. For example, if @filepath{simple.html} contains: @verbatim[#:indent 2]|{ Fastest @thing in the West!

Bang!

Bang!

}| Then @schemeblock[ (let ([thing "Templates"]) (include-template "simple.html")) ] evaluates to the same content as the static example. There are no constraints on how the lexical context of the template is populated. For instance, you can built template abstractions by wrapping the inclusion of a template in a function: @schemeblock[ (define (fast-template thing) (include-template "simple.html")) (fast-template "Templates") (fast-template "Noodles") ] evalutes to two strings with the predictable contents: @verbatim[#:indent 2]|{ Fastest Templates in the West!

Bang!

Bang!

}| and @verbatim[#:indent 2]|{ Fastest Noodles in the West!

Bang!

Bang!

}| Furthermore, there are no constraints on the Scheme used by templates: they can use macros, structs, continuation marks, threads, etc. However, Scheme values that are ultimately returned must be printable by the @|text-ref|. For example, consider the following outputs of the title line of different calls to @scheme[fast-template]: @itemize[ @item{ @schemeblock[ (fast-template 'Templates) ] @verbatim[#:indent 2]|{ ...Fastest Templates in the West!... }| } @item{ @schemeblock[ (fast-template 42) ] @verbatim[#:indent 2]|{ ...Fastest 42 in the West!... }| } @item{ @schemeblock[ (fast-template (list "Noo" "dles")) ] @verbatim[#:indent 2]|{ ...Fastest Noodles in the West!... }| } @item{ @schemeblock[ (fast-template (lambda () "Thunks")) ] @verbatim[#:indent 2]|{ ...Fastest Thunks in the West!... }| } @item{ @schemeblock[ (fast-template (delay "Laziness")) ] @verbatim[#:indent 2]|{ ...Fastest Laziness in the West!... }| } @item{ @schemeblock[ (fast-template (fast-template "Embedding")) ] @verbatim[#:indent 2]|{ ...Fastest ...<title>Fastest Embedding in the West!... in the West!... }| } ] @section{Gotchas} To obtain an @litchar["@"] character in template output, you must escape the it, because it is the escape character of the @at-reader-ref syntax. For example, to obtain: @verbatim[#:indent 2]|{ Fastest @s in the West! }| You must write: @verbatim[#:indent 2]|{ Fastest @"@"s in the West! }| as your template: literal @litchar["@"]s must be replaced with @litchar["@\"@\""]. (Note that the double-quotes are basically a Scheme expression, which can be used for longer strings too.) The @at-reader-ref will read Scheme identifiers, so it does not terminate identifiers on punctuations or XML angle brackets. So, @verbatim[#:indent 2]|{ Fastest @thing in the @place! }| will complain that the identifier @scheme[place!] is undefined. You can subvert this by explicitly delimiting the identifer: @verbatim[#:indent 2]|{ Fastest @thing in the @|place|! }| Another gotcha is that since the template is compiled into a Scheme program, only its results will be printed. For example, suppose we have the template: @verbatim[#:indent 2]|{ @for[([c clients])]{ }
@(car c), @(cdr c)
}| If this is included in a lexical context with @scheme[clients] bound to @schemeblock[(list (cons "Young" "Brigham") (cons "Smith" "Joseph"))] then the template will be printed as: @verbatim[#:indent 2]|{
}| because @scheme[for] does not return the value of the body. Suppose that we change the template to use @scheme[for/list] (which combines them into a list): @verbatim[#:indent 2]|{ @for/list[([c clients])]{ }
@(car c), @(cdr c)
}| Now the result is: @verbatim[#:indent 2]|{
}| because only the final expression of the body of the @scheme[for/list] is included in the result. We can capture all the sub-expressions by using @scheme[list] in the body: @verbatim[#:indent 2]|{ @for/list[([c clients])]{ @list{ } }
@(car c), @(cdr c)
}| Now the result is: @verbatim[#:indent 2]|{
Young, Brigham
Smith, Joseph
}| The templating library provides a syntactic form to deal with this issue for you called @scheme[in]: @verbatim[#:indent 2]|{ @in[c clients]{ }
@(car c), @(cdr c)
}| Notice how it also avoids the absurd amount of punctuation on line two. @section{HTTP Responses} The quickest way to generate an HTTP response from a template is using the @scheme[list] response type: @schemeblock[ (list #"text/html" (include-template "static.html")) ] If you want more control then you can generate a @scheme[response/full] struct: @schemeblock[ (make-response/full 200 #"Okay" (current-seconds) TEXT/HTML-MIME-TYPE empty (list (include-template "static.html"))) ] Finally, if you want to include the contents of a template inside a larger @xexpr : @schemeblock[ `(html ,(include-template "static.html")) ] will result in the literal string being included (and entity-escaped). If you actually want the template to be unescaped, then create a @scheme[cdata] structure: @schemeblock[ `(html ,(make-cdata #f #f (include-template "static.html"))) ] @section{API Details} @defform[(include-template path)]{ Compiles the template at @scheme[path] using the @at-reader-ref syntax within the enclosing lexical context. Example: @schemeblock[ (include-template "static.html") ] } @defform[(in x xs e ...)]{ Expands into @schemeblock[ (for/list ([x xs]) (begin/text e ...)) ] Template Example: @verbatim[#:indent 2]|{ @in[c clients]{ @(car c), @(cdr c) } }| Scheme Example: @schemeblock[ (in c clients "" (car c) ", " (cdr c) "") ] } @section{Conversion Example} Al Church has been maintaining a blog with PLT Scheme for some years and would like to convert to @schememodname[web-server/templates]. The data-structures he uses are defined as: @schemeblock[ (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."))) ] Actually, Al Church-encodes these posts, but for explanatory reasons, we'll use structs. He has divided his code into presentation functions and logic functions. We'll look at the presentation functions first. The first presentation function defines the common layout of all pages. @schemeblock[ (define (template section body) `(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 @scheme[unquote-splicing] on the @scheme[body] argument. This indicates that the @scheme[body] is list of @|xexpr|s. If he had accidentally used only @scheme[unquote] then there would be an error in converting the return value to an HTTP response. @schemeblock[ (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 @|xexpr|s to show the newly posted blog entry, before continuing to redisplay the main page. Let's look at a more complicated body: @schemeblock[ (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 @|xexpr|s. First, @scheme[append] is used to combine different @|xexpr| lists. Second, @scheme[apply append] is used to collapse and combine the results of a @scheme[for/list] where each iteration results in a list of @|xexpr|s. 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 @scheme[template] function to: @schemeblock[ (define (template section body) `(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)))) ] @margin-note{Some of these problems go away by using here strings, as described in the documentation on @secref[#:doc '(lib "scribblings/reference/reference.scrbl")]{parse-string}.} 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. Before moving to templates, let's look at the logic functions: @schemeblock[ (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 @scheme[template], @scheme[blog-posted], and @scheme[blog-posts]: @schemeblock[ (define (template section body) (list TEXT/HTML-MIME-TYPE (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: @filepath{blog.html}: @verbatim[#:indent 2]|{ Al's Church: @|section|

Al's Church: @|section|

@body
}| 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 @scheme[body] is represented as a string, there is no need to remember if splicing is necessary. @filepath{blog-posted.html}: @verbatim[#:indent 2]|{

@|title|

@|body|

Continue

}| @filepath{blog-posts.html}: @verbatim[#:indent 2]|{ @in[p posts]{

@(post-title p)

@(post-body p)

}

New Post

}| Compare this template with the original presentation function: there is no need to worry about managing how lists are nested: the defaults @emph{just work}.