HTML Templates
Discover how to use HTML templates in Go for web development using the html/template package
Go provides robust support for HTML templating through the html/template
package. This guide demonstrates how to use HTML templates effectively to create dynamic web content in Go.
Basic HTML Template Rendering
Here's a simple example that shows how to load and render an HTML template:
package main
import (
"html/template"
"net/http"
)
func handler(w http.ResponseWriter, r *http.Request) {
tmpl, err := template.ParseFiles("template.html")
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
data := struct {
Title string
}{
Title: "Hello, Go Templates!",
}
err = tmpl.Execute(w, data)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
func main() {
http.HandleFunc("/", handler)
http.ListenAndServe(":8080", nil)
}
Using Template Functions
You can enhance your HTML templates with custom functions:
package main
import (
"html/template"
"net/http"
"strings"
)
func handler(w http.ResponseWriter, r *http.Request) {
funcMap := template.FuncMap{
"ToUpper": strings.ToUpper,
}
tmpl, err := template.New("template.html").Funcs(funcMap).ParseFiles("template.html")
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
data := struct {
Name string
}{
Name: "gopher",
}
err = tmpl.Execute(w, data)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
func main() {
http.HandleFunc("/", handler)
http.ListenAndServe(":8080", nil)
}
Nested Templates
Create modular templates with inheritance to manage complex layouts:
package main
import (
"html/template"
"net/http"
)
func handler(w http.ResponseWriter, r *http.Request) {
templates, err := template.ParseFiles("base.html", "content.html")
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
data := struct {
Content string
}{
Content: "Nested Templates in Go",
}
err = templates.ExecuteTemplate(w, "base", data)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
func main() {
http.HandleFunc("/", handler)
http.ListenAndServe(":8080", nil)
}
Best Practices
- Always escape HTML inputs to avoid injection attacks
- Use template inheritance and nested templates for better code organization
- Leverage
template.FuncMap
to extend template functionalities - Keep logic minimal within templates; complex logic should reside in Go code
Common Pitfalls
- Forgetting to handle errors gracefully, which can expose server internals
- Not considering template caching for improved performance
- Mishandling of unescaped user input, leading to security vulnerabilities
- Repeated parsing of templates instead of reusing parsed templates
Performance Tips
- Cache parsed templates for reuse to avoid repeated loading and parsing
- Use goroutines to handle simultaneous requests efficiently
- Minimize data passed to templates to avoid large allocations
- Use buffered responses, especially when working with large templates
- Profile your template rendering path to identify bottlenecks in complex applications