Embedding Files into Go Binaries
Learn how to embed static files into Go binaries using the embed package introduced in Go 1.16
Embedding static files directly into Go binaries is a handy feature provided by the embed
package, introduced in Go 1.16. This feature allows you to package any file as part of your Go application, eliminating the need for separate file deployments.
Basic File Embedding
Here's a practical example demonstrating how to embed an HTML template into your web application:
package main
import (
_ "embed"
"fmt"
"html/template"
"net/http"
)
//go:embed templates/dashboard.html
var dashboardTemplate string
func main() {
// Parse the embedded template.
tmpl, err := template.New("dashboard").Parse(dashboardTemplate)
if err != nil {
panic(err)
}
http.HandleFunc("/dashboard", func(w http.ResponseWriter, r *http.Request) {
data := struct {
Username string
Stats map[string]int
}{
Username: "JohnDoe",
Stats: map[string]int{
"ActiveUsers": 1250,
"TotalPosts": 5432,
"NewSignups": 127,
},
}
if err := tmpl.Execute(w, data); err != nil {
http.Error(w, "Template execution failed.", http.StatusInternalServerError)
return
}
})
fmt.Println("Server starting on :8080...")
http.ListenAndServe(":8080", nil)
}
Embedding Multiple Files
Here's how to embed and serve multiple static assets in a web application:
package main
import (
"embed"
"fmt"
"html/template"
"io/fs"
"net/http"
"path"
)
//go:embed static/* templates/*
var content embed.FS
// Template cache for better performance.
var templates map[string]*template.Template
func init() {
// Initialize template cache.
templates = make(map[string]*template.Template)
// Load all templates from the embedded filesystem.
templateFiles, err := fs.Glob(content, "templates/*.html")
if err != nil {
panic(fmt.Sprintf("Failed to load templates: %v.", err))
}
for _, tmpl := range templateFiles {
name := path.Base(tmpl)
templates[name], err = template.ParseFS(content, tmpl)
if err != nil {
panic(fmt.Sprintf("Failed to parse template %s: %v.", name, err))
}
}
}
func main() {
// Serve static files from the embedded filesystem.
staticFS, err := fs.Sub(content, "static")
if err != nil {
panic(fmt.Sprintf("Failed to create sub-filesystem: %v.", err))
}
http.Handle("/static/", http.StripPrefix("/static/", http.FileServer(http.FS(staticFS))))
// Handle dynamic pages.
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
tmpl, ok := templates["index.html"]
if !ok {
http.Error(w, "Template not found.", http.StatusNotFound)
return
}
data := struct {
Title string
Navigation []struct {
Name string
URL string
}
}{
Title: "Welcome to Our App",
Navigation: []struct {
Name string
URL string
}{
{"Home", "/"},
{"Dashboard", "/dashboard"},
{"Profile", "/profile"},
{"Settings", "/settings"},
},
}
if err := tmpl.Execute(w, data); err != nil {
http.Error(w, "Failed to render template.", http.StatusInternalServerError)
return
}
})
fmt.Println("Server starting on :8080...")
http.ListenAndServe(":8080", nil)
}
Best Practices
- Organize Static Assets: Structure your embedded files in logical directories (e.g.,
templates/
,static/css/
,static/js/
). - Cache Templates: Parse templates once during initialization and cache them for better performance.
- Use Type-Safe Templates: Consider using typed template data structures to catch errors at compile time.
- Implement Asset Versioning: Include version numbers in static asset paths for cache busting.
- Monitor Binary Size: Keep track of embedded asset sizes and optimize large files when necessary.
Common Pitfalls
- Missing Template Functions: Remember to register custom template functions before parsing templates.
- Incorrect MIME Types: Ensure proper content types are set when serving embedded files.
- Memory Usage: Be mindful of memory usage when embedding large assets or caching templates.
- Development Workflow: Set up a development mode that reads files from disk for faster iteration.
Performance Tips
- Minify Assets: Compress and minify CSS, JavaScript, and images before embedding.
- Use Efficient Template Parsing: Parse templates at startup rather than per-request.
- Implement Caching Headers: Set appropriate cache headers for static assets.
- Consider Lazy Loading: Load large assets on-demand rather than embedding everything.
- Profile Memory Usage: Monitor memory consumption, especially with large embedded files.