在Go模板中加入范围块

Dee*_*pak 1 json dictionary loops go go-templates

我有一个这样的模板:

 "environment": [
   {{- range $k,$v := .env }}
     {
       "name": "{{ $k }}",
       "value": "{{ $v }}"
     },
   {{- end }}
   ]
Run Code Online (Sandbox Code Playgroud)

我得到以下输出:

     "environment": [
    {
      "name": "name",
      "value": "test"
    },
    {
      "name": "region",
      "value": "us-east-1"
    },
  ]
Run Code Online (Sandbox Code Playgroud)

我想将其渲染如下:

    "environment": [
    {
      "name": "name",
      "value": "bxbd"
    },
    {
      "name": "region",
      "value": "us-east-1"
    }
  ]
Run Code Online (Sandbox Code Playgroud)

我无法去掉最后一个逗号来生成有效的 json。或者是否可以以某种方式将完整的范围块发送到某些自定义连接函数?

icz*_*cza 5

下面是如何使用模板执行此操作的示例,但如果您想生成 JSON,我强烈建议使用第二种方法。

坚持使用模板

由于您正在地图上进行测距,因此您无法(简单地)做到这一点。如果是切片,您可以检查索引变量(示例:Go template 删除 range Loop 中的最后一个逗号;并使用 go-templates 内的 range 检测数组中的最后一项),但如果是映射,则不能这样做。

知道您是否处于第一次(或最后一次)迭代是您必须自己维护的一种状态。例如,为此使用自定义函数或方法。

这是一个示例实现:

type Params struct {
    Env     map[string]string
    Counter int
}

func (p *Params) IncMore() bool {
    p.Counter++
    return p.Counter < len(p.Env)
}

const src = `"environment": [
   {{- range $k,$v := .Env }}
     {
       "name": "{{ $k }}",
       "value": "{{ $v }}"
     }{{if $.IncMore}},{{end}}
   {{- end }}
   ]`
Run Code Online (Sandbox Code Playgroud)

测试它:

func main() {
    t := template.Must(template.New("").Parse(src))
    p := &Params{
        Env: map[string]string{
            "name":   "test",
            "region": "us-east-1",
        },
    }
    err := t.Execute(os.Stdout, p)
    if err != nil {
        panic(err)
    }
}
Run Code Online (Sandbox Code Playgroud)

输出(在Go Playground上尝试):

"environment": [
     {
       "name": "name",
       "value": "test"
     },
     {
       "name": "region",
       "value": "us-east-1"
     }
   ]
Run Code Online (Sandbox Code Playgroud)

用于encoding/json 生成 JSON

如果您的目标是生成 JSON,则应该使用该encoding/json包生成有效的 JSON 文档。上面的模板不了解 JSON 语法和上下文,并且映射条目的值在写入输出时不会转义,因此您可能仍然会得到无效的 JSON。

最好是像这样生成 JSON:

type Entry struct {
    Name  string `json:"name"`
    Value string `json:"value"`
}

type Params struct {
    Env []Entry `json:"environment"`
}

func main() {
    enc := json.NewEncoder(os.Stdout)
    enc.SetIndent("", "  ") // Optional
    p := &Params{
        Env: []Entry{
            {Name: "name", Value: "test"},
            {Name: "region", Value: "us-east-1"},
        },
    }
    err := enc.Encode(p)
    if err != nil {
        panic(err)
    }
}
Run Code Online (Sandbox Code Playgroud)

输出(在Go Playground上尝试):

{
  "environment": [
    {
      "name": "name",
      "value": "test"
    },
    {
      "name": "region",
      "value": "us-east-1"
    }
  ]
}
Run Code Online (Sandbox Code Playgroud)