我该怎么做:
{{ $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
]来解决它...
答案 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模板的上下文中遇到了同样的问题,这些模板是预先安装了一些附加功能的模板,例如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 }}