如何在Go模板中根据表达式有条件地设置变量,如果没有用if语句包装可能会导致错误

时间:2016-03-24 20:43:25

标签: ruby go-templates

问题

我该怎么做:

{{ $use_ssl := (ne $.Env.CERT_NAME "") }}

其中$.Env.CERT_NAME可能是nil / undefined。如果它为零,则会出现此错误:

at <ne $.Env.CERT_NAME "">: error calling ne: invalid type for comparison

注意:我无法控制传入Go模板的对象,因此必须在模板本身内完全解决这个问题。

我尝试了什么

我试图通过首先检查它是否为非空来解决:

{{ $use_ssl := ( ($.Env.CERT_NAME) && (ne $.Env.CERT_NAME "") ) }}

但它会出现此错误:

unexpected "&" in operand

所以我切换到了这个,这在句法上是允许的:

{{ $use_ssl := (and ($.Env.CERT_NAME) (ne $.Env.CERT_NAME "") ) }}

但是我丢失了&&运算符后我会得到的short-circuit evaluation,我又回来了这个错误:

at <ne $.Env.CERT_NAME ...>: error calling ne: invalid type for comparison

好吧,我想,如果我不能用一个漂亮的单行,Go doesn't have a ternary operator,那很好,我会做idiomatic Go way,显然是{{} 1}}。

if/else

但当然我遇到了范围问题,因为{{ if $.Env.CERT_NAME }} {{ $use_ssl := (ne $.Env.CERT_NAME "") }} {{ else }} {{ $use_ssl := false }} {{ end }} 莫名其妙地(或者至少令人讨厌)创建了一个新的变量范围(与我以前习惯的Ruby / ERB模板不同),所以显然我可以甚至这样做:

if

现在没有收到此错误:

{{ if $.Env.CERT_NAME }}
  {{ $use_ssl := true }}
{{ else }}
  {{ $use_ssl := false }}
{{ end }}
# $use_ssl: {{ $use_ssl }}
我想,“没有汗水”。我只是声明外部作用域中的变量,因此它将具有正确的作用域,内部作用域仍然可以更改该变量(具有外部作用域的变量)。 (顺便说一句,这就是它在Ruby中的工作方式。)

不,显然所有这一切都是创建2个不同的变量,它们具有相同的名称但不同的范围(如何混淆!):

undefined variable "$use_ssl"

我也尝试过这样的if语句:

{{ $use_ssl := false }}
{{ if $.Env.CERT_NAME }}
  {{ $use_ssl := true }}
  # $use_ssl: {{ $use_ssl }}
{{ else }}
  {{ $use_ssl := false }}
{{ end }}
# $use_ssl: {{ $use_ssl }}

  # $use_ssl: true
# $use_ssl: false

但是出现了这个错误:

{{ $use_ssl := if $.Env.CERT_NAME { true } }}

显然unexpected <if> in command 语句不能像在Ruby中那样用作Go中的表达式吗?

如果我尝试What is the idiomatic Go equivalent of C's ternary operator?中建议的三元运算符的各种替代方法,我也会遇到语法错误,例如:

if

当然,您可以使用c := map[bool]int{true: 1, false: 0} [5 > 4] 从内部作用域中的外部作用域读取变量,但是如何在内部作用域中设置 $.something

如果不是那样,那怎么??

那么我错过了什么?这是否可以在Go模板中使用?

Go模板(我是他们的新手)似乎非常有限。几乎你在Go中可以做的一切都不可能在模板中完成。但希望有一个解决方法......

http://play.golang.org/p/SufZdsx-1v有一个聪明的解决方法,涉及使用$.something创建一个新对象,然后使用{{$hasFemale := cell false}}设置一个值,但是如何在不访问代码的情况下执行此类操作评估模板(他们在哪里调用$hasFemale.Set true)?

其他尝试过失败的人

https://groups.google.com/forum/#!topic/golang-nuts/MUzNZ9dbHrg

  

这是我对Go的最大(也是唯一)问题。我根本不明白为什么还没有实现这个功能。

     

在通用上下文中使用模板包时(例如,执行   您可能没有任意模板文件和任意json文件   有预先计算的奢侈。

https://github.com/golang/go/issues/10608

  

这可能是设计的,但如果有办法在条件之外获得更改的$ v,那将会非常好。

     

这是我们在Hugo(https://github.com/spf13/hugo)得到的#1模板问题。我们现在添加了hack [template.FuncMap]来解决它...

3 个答案:

答案 0 :(得分:9)

目前正在使用的解决方案

如果更新范围而不是声明变量,可以使用set或使用merge,也可以在if子句之外访问它。

{{- if eq .podKind "core" -}}
{{- $dummy := set . "matchNodePurpose" .Values.scheduling.corePods.nodeAffinity.matchNodePurpose }}
{{- else if eq .podKind "user" -}}
{{- $dummy := set . "matchNodePurpose" .Values.scheduling.userPods.nodeAffinity.matchNodePurpose }}
{{- end -}}

# Will now output what you decided within an if/else if clause
{{- .matchNodePurpose }}

未来的解决方案(可与Helm 2.10一起使用)

我在Helm模板的上下文中遇到了同样的问题,这些模板是预先安装了一些附加功能的模板,例如Sprig

28 March 2018, in this pr一个Terenary操作员被添加到Sprig,并且Helm会让他们同时解决你和我的问题。

<强> terenary

true | ternary "b" "c"将返回"b"

false | ternary "b" "c"将返回"c"

答案 1 :(得分:4)

我想我可能已经为我的特定问题找到了解决方法,但我希望能够听到更普遍解决问题的其他答案。

https://github.com/jwilder/nginx-proxy/blob/1e0b930/nginx.tmpl#L91中遇到了这一行:

{{ $default_host := or ($.Env.DEFAULT_HOST) "" }}

这给了我另一个尝试的想法。我想解决方案是使用or的组合并设置额外的临时变量...

{{ $cert_name := or $.Env.CERT_NAME "" }}
{{ $use_ssl := ne $cert_name "" }}

答案 2 :(得分:3)

如果您使用赋值运算符而不是定义运算符,则您的第四次尝试将起作用:

{{ $use_ssl := false }}
{{ if $.Env.CERT_NAME }}
  {{ $use_ssl = true }} # notice the missing colon!
{{ else }}
  {{ $use_ssl = false }}
{{ end }}