附录: YAML技术

本指南大部分都聚焦于编写模板语言。这里,我们要看看YAML格式。作为模板作者,YAML有一些有用的特性 使我们的模板不易出错,更易阅读。

标量和集合

根据 YAML 规范,有两种集合类型和很多标量类型。

两种集合类型是map和sequence:

  1. map:
  2. one: 1
  3. two: 2
  4. three: 3
  5. sequence:
  6. - one
  7. - two
  8. - three

标量值是单个值,(与集合相反)

YAML中的标量类型

在Helm内部的YAML语言中,一个标量值的数据类型是由一组复杂的规则决定的,包含了资源定义的Kubernetes模式。 但在推断类型时,以下规则往往是正确的。

如果整型或浮点型数字没有引号,通常被视为数字类型:

  1. count: 1
  2. size: 2.34

但是如果被引号引起来,会被当做字符串:

  1. count: "1" # <-- string, not int
  2. size: '2.34' # <-- string, not float

布尔函数也是如此:

  1. isGood: true # bool
  2. answer: "true" # string

空字符串是 null (不是 nil)。

注意 port: "80"是合法的YAML,可以通过模板引擎和YAML解释器传值,但是如果Kubernetes希望port是整型,就会失败。

在一些场景中,可以使用YAML节点标签强制推断特定类型:

  1. coffee: "yes, please"
  2. age: !!str 21
  3. port: !!int "80"

如上所示,!!str告诉解释器age是一个字符串,即使它看起来像是整型。即使port被引号括起来,也会被视为int。

YAML中的字符串

YAML中的大多数数据都是字符串。YAML有多种表示字符串的方法。本节解释这些方法并演示如何使用其中一些方法。

有三种单行方式声明一个字符串:

  1. way1: bare words
  2. way2: "double-quoted strings"
  3. way3: 'single-quoted strings'

单行样式必须在一行:

  • 裸字没有引号,也没有转义,因此,必须小心使用字符。
  • 双引号字符串可以使用\转义指定字符。比如,"\"Hello\", she said"。可以使用\n转义换行。
  • 单引号字符串是字面意义的字符串,不用\转义,只有单引号''需要转义,转成单个'

除了单行字符串,可以声明多行字符串:

  1. coffee: |
  2. Latte
  3. Cappuccino
  4. Espresso

上述会被当作coffee的字符串值,等同于Latte\nCappuccino\nEspresso\n

注意在第一行|后面必须正确缩进。可以这样破坏上述示例:

  1. coffee: |
  2. Latte
  3. Cappuccino
  4. Espresso

由于Latte没有正确缩进,会遇到这样的错误:

  1. Error parsing file: error converting YAML to JSON: yaml: line 7: did not find expected key

模板中,有时候为了避免上述错误,在多行文本中添加一个假的“第一行”会更加安全:

  1. coffee: |
  2. # Commented first line
  3. Latte
  4. Cappuccino
  5. Espresso

注意无论第一行是什么,都会保存在字符串的输出中。比如你要这样把文件内容注入到配置映射中,注释应该是读取该条目需要的类型。

控制多行字符串中的空格

在上述示例中,使用了 | 来表示多行字符串。但是注意字符串后面有一个尾随的\n。如果需要YAML处理器去掉末尾的换行符,在| 后面添加-

  1. coffee: |-
  2. Latte
  3. Cappuccino
  4. Espresso

现在 coffee的值变成了: Latte\nCappuccino\nEspresso (没有末尾的\n)。

其他时候,可能希望保留尾随空格。可以使用 |+符号:

  1. coffee: |+
  2. Latte
  3. Cappuccino
  4. Espresso
  5. another: value

现在coffee的值是 Latte\nCappuccino\nEspresso\n\n\n

文本块中的缩进会被保留,也会保留换行符:

  1. coffee: |-
  2. Latte
  3. 12 oz
  4. 16 oz
  5. Cappuccino
  6. Espresso

上述示例中,coffee会变成 Latte\n 12 oz\n 16 oz\nCappuccino\nEspresso

缩进和模板

编写模板时,你可能会想将文件内容插入到模板中。正如在之前的章节中看到的,有两种方法处理:

  • 使用 {{ .Files.Get "FILENAME" }} 获取chart中的文件内容。
  • 使用 {{ include "TEMPLATE" . }} 渲染模板并将其放到chart中。

把文件插入到YAML时,就很好理解上面的多行规则了。通常,插入一个静态文件最简单的方式是像这样:

  1. myfile: |
  2. {{ .Files.Get "myfile.txt" | indent 2 }}

注意上面是怎么做缩进的: indent 2 告诉模板引擎在文件”myfile.txt”中每行缩进两个空格。注意我们没有缩进模板的行。 因为如果缩进了,文件内容的第一行会缩进两次。

折叠多行字符串

有时您想在 YAML 中用多行表示一个字符串,但希望在解释时将其视为一个长行。这被称为”折叠”。要声明一个折叠块,使用 > 代替 |

  1. coffee: >
  2. Latte
  3. Cappuccino
  4. Espresso

上面coffee的值是: Latte Cappuccino Espresso\n。 注意,除了最后一个换行符之外,所有的换行符都将转换成空格。 可以组合空格控制符和折叠字符标记 >- 来替换或取消所有的新行。

注意在折叠语法中,缩进文本将导致保留行。

  1. coffee: >-
  2. Latte
  3. 12 oz
  4. 16 oz
  5. Cappuccino
  6. Espresso

上述结果为:Latte\n 12 oz\n 16 oz\nCappuccino Espresso。注意空格和换行都保存下来了。

在一个文件中嵌入多个文档

可以将多个YAML文档放在单个文件中。 文档前使用 ---,文档后使用 ...

  1. ---
  2. document:1
  3. ...
  4. ---
  5. document: 2
  6. ...

很多情况下,可以省略---或者...

Helm中的有些文件无法包含多个文档。比如,如果values.yaml文件提供了多个文档,只会使用第一个。

但是模板文件可以有多个文档。这种情况下,文件会被当做一个对象进行渲染。但是将结果YAML提供给Kubernetes时,会被分成多个文档。

我们建议在确实需要时才将多个文档写入单个文件。单个文件中的多个文档会变得很难调试。

YAML是JSON的超集

由于YAML是一个JSON的超集,任何合法的JSON文档 都应该 是合法的YAML。

  1. {
  2. "coffee": "yes, please",
  3. "coffees": [
  4. "Latte", "Cappuccino", "Espresso"
  5. ]
  6. }

上述json的另一种表述方式是:

  1. coffee: yes, please
  2. coffees:
  3. - Latte
  4. - Cappuccino
  5. - Espresso

而且两种可以混合(要小心):

  1. coffee: "yes, please"
  2. coffees: [ "Latte", "Cappuccino", "Espresso"]

所有这三个都应该解析为相同的内部表示形式。

这意味着类似 values.yaml 可能包含JSON数据,Helm将.json后缀文件视为不合法的文件。

YAML 锚点

YAML规范存储了一种引用值的方法,然后通过引用指向该值。YAML称之为“锚定”:

  1. coffee: "yes, please"
  2. favorite: &favoriteCoffee "Cappuccino"
  3. coffees:
  4. - Latte
  5. - *favoriteCoffee
  6. - Espresso

上面示例中,&favoriteCoffee 设置成了Cappuccino的引用。之后,通过*favoriteCoffee使用引用。 这样coffees 就变成了 Latte, Cappuccino, Espresso

锚点在一些场景中很有用,但另一方面,锚点可能会引起细微的错误:第一次使用YAML时,将展开引用,然后将其丢弃。

因此,如果我们解码再重新编码上述示例,产生的YAML就会是这样:

  1. coffee: yes, please
  2. favorite: Cappuccino
  3. coffees:
  4. - Latte
  5. - Cappuccino
  6. - Espresso

因为Helm和Kubernetes经常读取,修改和重写YAML文件,锚点会丢失。