25

问题

我该怎么做这样的事情:

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

where$.Env.CERT_NAME可能是 nil/undefined。如果它是 nil,它会给出这个错误:

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 "") ) }}

但后来我失去了与操作员一起获得的短路评估,我&&又回到了这个错误:

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

好吧,我想,如果我不能在一个不错的单行中做到这一点,而且Go 没有三元运算符,那很好,我会用惯用的 Go 方式来做,这显然是if/else.

{{ if $.Env.CERT_NAME }}
  {{ $use_ssl := (ne $.Env.CERT_NAME "") }}
{{ else }}
  {{ $use_ssl := false }}
{{ end }}

但是当然我遇到了范围问题,因为if莫名其妙(或至少令人讨厌)创建了一个新的变量范围(与我更习惯的 Ruby/ERB 模板不同),所以显然我什至不能这样做:

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

现在没有收到此错误:

undefined variable "$use_ssl"

“没有汗水”,我想。我将在外部范围内声明变量,以便它具有正确的范围,内部范围仍然能够更改该变量(具有外部范围的变量)。(顺便说一下,这就是它在 Ruby 中的工作方式。)

不,显然所做的只是创建两个具有相同名称但范围不同的不同变量(这如何令人困惑!):

{{ $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

我也尝试过像这样内联 if 语句:

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

但这给出了这个错误:

unexpected <if> in command

显然if语句不能像在 Ruby 中那样在 Go 中用作表达式?

如果我尝试在什么是 C 的三元运算符的惯用 Go 等效项中建议的三元运算符的各种替代方案,我也会遇到语法错误?, 喜欢:

c := map[bool]int{true: 1, false: 0} [5 > 4]

当然,您可以在内部范围内从外部范围读取变量,$.something但是如何在内部范围中设置 $.something

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

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

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

http://play.golang.org/p/SufZdsx-1v有一个聪明的解决方法,包括创建一个新对象,{{$hasFemale := cell false}}然后设置一个$hasFemale.Set true值(他们在哪里打电话template.FuncMap)?

其他尝试过但失败的人

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 ) 得到的第一个模板问题。我们现在添加了一个hack [ $.Scratch.Set "v1" 123] 来解决它......

4

3 回答 3

20

当前工作解决方案

如果您更新范围而不是声明变量,则通过使用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 模板的上下文中遇到了同样的问题,这些模板是预先安装了一些附加功能的 go 模板,例如Sprig

2018 年 3 月 28 日,在这个 pr 中,向 Sprig 添加了一个 Terenary 操作员,Helm 将及时让他们解决你和我都有的问题。

三元组

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

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

于 2018-04-02T03:32:58.077 回答
8

我想我可能已经为我的特定问题找到了解决方法,尽管我很想听到其他更普遍地解决问题的答案。

在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 "" }}
于 2016-03-24T21:16:13.753 回答
7

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

{{ $use_ssl := false }}
{{ if $.Env.CERT_NAME }}
  {{ $use_ssl = true }} # notice the missing colon!
{{ else }}
  {{ $use_ssl = false }}
{{ end }}
于 2019-11-06T08:34:53.897 回答