Generating Code with Templates
Learn how to generate Go code using text templates for dynamic content and code generation tasks.
The Go programming language provides powerful templating support through the text/template
and html/template
packages. These packages can be used for generating code by composing templates that render dynamic content. This is particularly useful for code generators, report creation, and template-based system configuration.
Basic Code Generation with text/template
Here's a simple example that demonstrates how to generate code using the text/template
package:
package main
import (
"os"
"text/template"
)
func main() {
const tpl = `
package main
import "fmt"
func main() {
fmt.Println("Hello, {{.Name}}!")
}
`
data := struct {
Name string
}{
Name: "World",
}
tmpl, err := template.New("hello").Parse(tpl)
if err != nil {
panic(err)
}
if err = tmpl.Execute(os.Stdout, data); err != nil {
panic(err)
}
}
Advanced Custom Template Functions
You can enhance templates with custom functions to extend functionality:
package main
import (
"os"
"text/template"
"strings"
)
func main() {
const tpl = `
package main
import "strings"
func main() {
uppercase := {{. | uppercase}}
fmt.Println(uppercase)
}
`
funcMap := template.FuncMap{
"uppercase": strings.ToUpper,
}
data := "hello, world"
tmpl, err := template.New("uppercase").Funcs(funcMap).Parse(tpl)
if err != nil {
panic(err)
}
err = tmpl.Execute(os.Stdout, data)
if err != nil {
panic(err)
}
}
Generating Structs from Templates
You can also generate Go structs with predefined fields from templates:
package main
import (
"os"
"text/template"
)
func main() {
const tpl = `
package models
type {{.StructName}} struct {
{{range .Fields}}
{{.Name}} {{.Type}}
{{end}}
}
`
type Field struct {
Name string
Type string
}
data := struct {
StructName string
Fields []Field
}{
StructName: "User",
Fields: []Field{
{"ID", "int"},
{"Name", "string"},
{"Email", "string"},
},
}
tmpl, err := template.New("struct").Parse(tpl)
if err != nil {
panic(err)
}
err = tmpl.Execute(os.Stdout, data)
if err != nil {
panic(err)
}
}
Best Practices
- Leverage
template.FuncMap
to enhance the utility of your templates by adding custom functions. - Aim to keep templates as simple as possible but powerful by using logical constructs {like if/else} and iteration wisely.
- Store templates in separate files for ease of management and reuse, loading them using
template.ParseFiles
.
Common Pitfalls
- Avoid ignoring errors from template parsing or execution, as they can contain valuable information about syntax issues.
- Be mindful of template injection vulnerabilities, particularly when using templates with user-provided data in web applications.
- Ensure your templates do not become overly complex; complex templates can be hard to maintain and debug.
Performance Tips
- Compile your templates once and reuse them if possible, as parsing can be expensive for large templates.
- Use buffers to collect template output rather than writing directly to
os.Stdout
, especially in performance-sensitive contexts. - Consider caching generated code if you're repeating generation tasks with similar data for better performance.