使用 gin 写网站会涉及到大量的外部文件:

  • HTML 模板文件
  • 静态资源文件:js,css,图片等
  • 多语言翻译文件

发布时,除了要将编译好的二进制文件上传到服务器,还需要将上面提到的三类文件同步到二进制文件所在的目录。这样有两个弊端:

  • 部署麻烦
  • 如果是交付给客户,放在客户服务器上运行,客户能拿到这些文件的明文

embed 打包

go 1.16 引入的新特性 embed 支持将这些外部文件打包入二进制文件。

而且 gin 也支持了 embed,配置起来就更方便了。

首先需要过一遍 go embed 的官方文档,否则 gin 里的 embed 代码不容易理解。

template 及 static 文件打包

main.go

//go:embed templates/* public/*
var f embed.FS

func main() {
	...

	r := gin.Default()

	// apply i18n middleware
	r.Use(i18n.GinI18nLocalize())

	// 自定义模板函数
	funcMap := template.FuncMap{
		"UnescapeHTML": utils.UnescapeHTML,
		"Localize":     ginI18n.GetMessage,
	}

	// embed files
	tmpl := template.New("").Funcs(funcMap)
	tmpl = template.Must(tmpl.ParseFS(f, "templates/*.html"))
	r.SetHTMLTemplate(tmpl)

	fp, _ := fs.Sub(f, "public")
	r.StaticFS("/public", http.FS(fp))

	...
}

模板函数不要忘了配置,否则会报错,例如:

panic: template: article.html:6: function "Localize" not defined

i18n 翻译文件打包

i18n.go

gin i18n 里内置了 embed 的支持,修改一下 loader 即可。

注意,对于 sub package 中的 embed directive,路径是相对于代码文件的,而不是项目根目录。 例如,i18n.go 平级下有个 lang 目录,里面存放了翻译文件。

import (
	"embed"
	ginI18n "github.com/gin-contrib/i18n"
)

//go:embed lang/*
var fs embed.FS

func GinI18nLocalize() gin.HandlerFunc {
	return ginI18n.Localize(
		ginI18n.WithBundle(&ginI18n.BundleCfg{
			RootPath:         "./lang",
			AcceptLanguage:   []language.Tag{language.Chinese, language.English},
			DefaultLanguage:  language.Chinese,
			FormatBundleFile: "toml",
			UnmarshalFunc:    toml.Unmarshal,
			Loader:           &ginI18n.EmbedLoader{fs},
		}),
	)
}

参考

  • https://pkg.go.dev/embed
  • https://github.com/gin-gonic/examples/blob/master/assets-in-binary/example02/main.go
  • https://gist.github.com/rsperl/6990289982130ea5be9b2e330f0a0229
  • https://github.com/gin-contrib/i18n/blob/master/embed.go
  • https://github.com/gin-contrib/i18n/blob/master/embed_test.go

tags: Golang Gin