Usage of variables in a nested template which is also defined as a variable in a go template?

huangapple go评论98阅读模式
英文:

Usage of variables in a nested template which is also defined as a variable in a go template?

问题

  1. package main
  2. import (
  3. "os"
  4. "text/template"
  5. )
  6. type Todo struct {
  7. Name string
  8. Description string
  9. Subtemplate string
  10. }
  11. func main() {
  12. td := Todo{
  13. Name: "Test name",
  14. Description: "Test description",
  15. Subtemplate: "Subtemplate {{.Name}}",
  16. }
  17. t, err := template.New("todos").Parse(`{{template "subtemplate" .}} You have a task named "{{ .Name}}" with description: "{{ .Description}}"`)
  18. if err != nil {
  19. panic(err)
  20. }
  21. // Define the subtemplate
  22. _, err = t.New("subtemplate").Parse(td.Subtemplate)
  23. if err != nil {
  24. panic(err)
  25. }
  26. err = t.Execute(os.Stdout, td)
  27. if err != nil {
  28. panic(err)
  29. }
  30. }

上述代码的结果是:

Subtemplate Test name You have a task named "Test name" with description: "Test description"

这意味着子模板中的变量.Name没有被解析(可能是因为设计上不可能,需要一种递归调用的方式)。有没有其他方法可以实现这个效果?

对于使用template.FuncMap定义的模板函数也应该适用。谢谢。

英文:

The idea is to define a variable for a go template which is also a template using variables (a nested template) like this:

  1. package main
  2. import (
  3. "os"
  4. "text/template"
  5. )
  6. type Todo struct {
  7. Name string
  8. Description string
  9. Subtemplate string
  10. }
  11. func main() {
  12. td := Todo{
  13. Name: "Test name",
  14. Description: "Test description",
  15. Subtemplate: "Subtemplate {{.Name}}",
  16. }
  17. t, err := template.New("todos").Parse("{{.Subtemplate}} You have a task named \"{{ .Name}}\" with description: \"{{ .Description}}\"")
  18. if err != nil {
  19. panic(err)
  20. }
  21. err = t.Execute(os.Stdout, td)
  22. if err != nil {
  23. panic(err)
  24. }
  25. }

The result of the code above is however:

Subtemplate {{.Name}} You have a task named "Test name" with description: "Test description"

means the variable .Name in the subtemplate is not resolved (probably by design not possible, would require some kind of a recursive call). Is there any/other way to achieve this effect?

It should work for the template functions defined using template.FuncMap too. Thanx.

答案1

得分: 2

你可以注册一个函数来解析字符串模板并执行它。

该函数可以像这样:

  1. func exec(body string, data interface{}) (string, error) {
  2. t, err := template.New("").Parse(body)
  3. if err != nil {
  4. return "", err
  5. }
  6. buf := &strings.Builder{}
  7. err = t.Execute(buf, data)
  8. return buf.String(), err
  9. }

你将模板主体文本和用于模板执行的数据传递给它。它会执行并返回结果。

一旦注册,你可以在模板中这样调用它:

  1. {{exec .Subtemplate .}}

完整示例:

  1. td := Todo{
  2. Name: "Test name",
  3. Description: "Test description",
  4. Subtemplate: "Subtemplate {{.Name}}",
  5. }
  6. t, err := template.New("todos").Funcs(template.FuncMap{
  7. "exec": func(body string, data interface{}) (string, error) {
  8. t, err := template.New("").Parse(body)
  9. if err != nil {
  10. return "", err
  11. }
  12. buf := &strings.Builder{}
  13. err = t.Execute(buf, data)
  14. return buf.String(), err
  15. },
  16. }).Parse("{{exec .Subtemplate .}} You have a task named \"{{ .Name}}\" with description: \"{{ .Description}}\"")
  17. if err != nil {
  18. panic(err)
  19. }
  20. err = t.Execute(os.Stdout, td)
  21. if err != nil {
  22. panic(err)
  23. }

这将输出(在Go Playground上尝试):

  1. Subtemplate Test name You have a task named "Test name" with description: "Test description"

请注意,如果子模板在运行时不会更改,你应该预先解析它并将生成的模板(*template.Template)存储起来,以避免每次执行模板时都进行解析。

英文:

You can register a function which parses a string template and executes it.

The function could look like this:

  1. func exec(body string, data any) (string, error) {
  2. t, err := template.New("").Parse(body)
  3. if err != nil {
  4. return "", err
  5. }
  6. buf := &strings.Builder{}
  7. err = t.Execute(buf, data)
  8. return buf.String(), err
  9. }

You pass the template body text and the data for template execution to it. It executes it and returns the result.

Once registered, you can call it like this from a template:

  1. {{exec .Subtemplate .}}

Full example:

  1. td := Todo{
  2. Name: "Test name",
  3. Description: "Test description",
  4. Subtemplate: "Subtemplate {{.Name}}",
  5. }
  6. t, err := template.New("todos").Funcs(template.FuncMap{
  7. "exec": func(body string, data any) (string, error) {
  8. t, err := template.New("").Parse(body)
  9. if err != nil {
  10. return "", err
  11. }
  12. buf := &strings.Builder{}
  13. err = t.Execute(buf, data)
  14. return buf.String(), err
  15. },
  16. }).Parse("{{exec .Subtemplate .}} You have a task named \"{{ .Name}}\" with description: \"{{ .Description}}\"")
  17. if err != nil {
  18. panic(err)
  19. }
  20. err = t.Execute(os.Stdout, td)
  21. if err != nil {
  22. panic(err)
  23. }

This will output (try it on the Go Playground):

  1. Subtemplate Test name You have a task named "Test name" with description: "Test description"

Note that if the subtemplate does not change during runtime, you should pre-parse it and store the resulting template (*template.Template) to avoid having to parse it each time you execute the template.

答案2

得分: 1

如果你不介意性能问题,你可以渲染两次...

  1. package main
  2. import (
  3. "os"
  4. "text/template"
  5. "bytes"
  6. )
  7. type Todo struct {
  8. Name string
  9. Description string
  10. Subtemplate string
  11. }
  12. func main() {
  13. td := Todo{
  14. Name: "测试名称",
  15. Description: "测试描述",
  16. Subtemplate: "子模板 {{.Name}}",
  17. }
  18. t, err := template.New("todos").Parse("{{.Subtemplate}} 你有一个名为“{{.Name}}”的任务,描述为:“{{.Description}}”")
  19. if err != nil {
  20. panic(err)
  21. }
  22. buffer := &bytes.Buffer{}
  23. err = t.Execute(buffer, td)
  24. tc, err := template.New("todo2").Parse(string(buffer.Bytes()))
  25. if err != nil {
  26. panic(err)
  27. }
  28. err = tc.Execute(os.Stdout, td)
  29. if err != nil {
  30. panic(err)
  31. }
  32. }
英文:

You can render twice if you don't mind for peformance ...

  1. package main
  2. import (
  3. "os"
  4. "text/template"
  5. "bytes"
  6. )
  7. type Todo struct {
  8. Name string
  9. Description string
  10. Subtemplate string
  11. }
  12. func main() {
  13. td := Todo{
  14. Name: "Test name",
  15. Description: "Test description",
  16. Subtemplate: "Subtemplate {{.Name}}",
  17. }
  18. t, err := template.New("todos").Parse("{{.Subtemplate}} You have a task named \"{{ .Name}}\" with description: \"{{ .Description}}\"")
  19. if err != nil {
  20. panic(err)
  21. }
  22. buffer := &bytes.Buffer{}
  23. err = t.Execute(buffer, td)
  24. tc, err := template.New ("todo2").Parse(string (buffer.Bytes ()))
  25. if err != nil {
  26. panic (err)
  27. }
  28. err = tc.Execute(os.Stdout, td)
  29. if err != nil {
  30. panic(err)
  31. }
  32. }

huangapple
  • 本文由 发表于 2023年1月10日 15:48:34
  • 转载请务必保留本文链接:https://go.coder-hub.com/75066909.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定