ansible基础-变量

一 变量的命名规范

变量的命名应该符如下合两个规范:

  • 变量应该由字母、数字、下划线组成
  • 变量应该以字母开头

例如:host_port、HOST_PORT、var5是符合命名规范的,foo-port、 foo port、foo.port 、12都不符合命名规范。

变量的定义通常是YAML形式,在inventory host文件中也可以使用INI形式。

ansible变量不仅可以支持简单的key=value格式,而且也支持更复杂数据结构,例如字典类型等。

二 变量的作用域

 变量的作用域可以分为四种:

  • 作用于全局的变量
  • 作用于play的变量
  • 作用于task的变量
  • 作用于host的变量

接下来我们根据变量的作用域,详细分析下ansible变量的定义、使用和调用顺序。

三 作用于全局的变量

3.1 配置文件变量

ansible配置文件会定义一些变量信息,主要是对执行环境、连接信息变量的定义。

例如inventory目录、library目录、与目的主机连接方式、越权信息、连接超时时间等等。

3.2 系统环境变量

在ansible连接到目的主机时,会以non-login shell登陆到目的主机,此时目的主机的/etc/bashrc和~/.basrc的环境变量会被加载,所以这两个文件中设置的环境变量会作用于playbook全局。

3.3 命令行变量

我们可以在执行playbook的命令行指定变量,需要注意的是,命令行指定的变量在所有其他变量中优先级是最高的。也就是说如果命令行指定的变量和其他地方指定的变量有冲突时,那么ansible最终会采用命令行定义的变量。

命令行指定变量示例如下:

ansible-playbook release.yml --extra-vars "version=1.23.45 other_variable=foo"

四 作用于play的变量

4.1 playbook中的变量

vars语句定义全局变量

我们可以在playbook中使用「vars」语句定义变量,该变量作用于整个play。

例如:

---
# inventory/playbooks/test.yaml
- hosts: node1
  vars:
    http_port: 80

上面示例中中「http_port」是一个作用于整个play的变量,对这个play里的tasks、roles、import、include等等之下定义的task均生效。

引用变量文件

除了将变量写在playbook中,我们也可以将变量放在一个单独的YAML文件中,通过「vars_files」语句来导入。

「vars_files」变量只能作用于play全局,不能在某个task中单独被引用。「vars_files」参数可以使用系统绝对路径或playbook文件的相对路径。

举个🌰:

我们在playbooks目录下创建一个vars-files.yaml文件:

---
# playbooks/vars-files.yaml
age: 100

在playbook中使用vars_files语句引用该变量文件:

---
# playbooks/test.yaml
- hosts: node1
  vars_files: ./vars-files.yaml
  tasks:
  - debug:
      msg: "My age is {{ age }}"

4.2 roles中的变量

4.2.1 default变量 

default变量位于roles/defaults/main.yml文件中,该变量作用于role里的所有play,通常作为模版或模块里的默认参数。

default变量与ansible filter变量 「{{ some_variable | default("some_value") }}」具有同样的作用,在所有ansible变量中优先级最低。

4.2.2 dependencies变量

dependencies变量位于roles/meta/main.yaml文件中,该变量与「role」语句同级缩进,作用于本身的role和dependencies role。

举个🌰:

role_A 和 role_B定义了相同的task,debug出「age」变量:

---
# tasks file for role_B
- debug:
    var: age
---
# tasks file for role_B
- debug:
    var: age

role_A/meta/main.yaml定义role_A依赖role_B,并指定「age」变量等于26:

---
dependencies:
  - role: role_B
    vars:
      age: 27

写playbook,引用role_A:

---
# playbooks/test.yaml
- hosts: node1
  roles:
    - role: role_A

执行结果如下:

➜  lab-ansible ansible-playbook playbooks/test.yaml

PLAY [node1] ********************************************************************************************

TASK [Gathering Facts] **********************************************************************************
ok: [node1]

TASK [role_B : debug] ***********************************************************************************
ok: [node1] => {
    "age": 27
}

TASK [role_A : debug] ***********************************************************************************
ok: [node1] => {
    "age": 27
}

PLAY RECAP **********************************************************************************************
node1                      : ok=3    changed=0    unreachable=0    failed=0

输出结果显示,dependencise变量「age」在role_A和role_B均生效。

4.2.3 vars变量

vars变量位于roles/vars/main.yml,该变量作用于role里的所有模块。通常将除了默认变量的其他的变量放在这个文件内。

4.3 register变量

register方法能够将一个task的执行结果注册为一个变量。书写格式要与模块名称对齐,该变量作用于整个play。

通常register变量和when语句联合使用,以达到满足某些条件才运行task的目的。

五 作用于task的变量 

5.1 playbook中的变量

with modules

我们可以为某个模块定义变量,该变量作用于这个task。

举个🌰,示例中为「debug」模块定义了「name」和「age」变量并在「msg」参数后使用了这两个变量:

---
# playbooks/test.yaml
- hosts: node1
  tasks:
    - debug:
        msg: "My name is {{ name }} and I'm {{ age }} years old"
      vars:
        name: Maurice
        age: 27

with import*/include*

在使用import_playbook、import_tasks、include_tasks、import_role、include_role时可以在import*/include*的同级位置指定变量,该变量作用于导入的所有play。

使用import_role举个🌰:

playbook导入role_A,并定义变量「age」,这样role_A内的play就可以使用「age」变量了:

---
# playbooks/test.yaml
- hosts: node1
  tasks:
  - import_role:
      name: role_A
    vars:
      age: 1000

其他import*/include*的语句使用方法类似,只要记住缩进与import*/include*语句保持一致即可。

with roles

在playbook中使用roles语句来导入role时也可以定义变量,该变量作用于role包含的所有play。 

举个🌰:

playbook使用roles语句导入role_A,并定义变量「age」:

---
# playbooks/test.yaml
- hosts: node1
  roles:
    - role: role_A
      vars:
       age: 1000

通过上面两个示例我们发现,roles和import_role语句定义变量写法上很相似,其实import_role和include_role是新版本的语法,功能上完全可以代替roles语句。如果你使用的ansible版本>=2.4,建议使用include_role和import_role语句。

5.2 roles中的变量

指的是在tasks/main.yaml或handlers/main.yaml内书写task时指定的变量,该变量作用于某个task,这个变量类型和上述章节中「5.1playbook中的变量—with modules」类似,这里就不再举例说明。

六 作用于host的变量

6.1 系统变量Facts

6.1.1 facts变量

ansible中有个特殊的变量,这些变量不是开发者定义的,而是ansible根据目的主机环境信息自动收集的,称之为fact变量。

fact变量很实用,和「when」语句配合使用会让你的代码更加健壮。

举个🌰,如果当前的操作系统为「RedHat」类型,则通过yum安装需要的软件包:

---
# playbooks/test.yaml
- hosts: node1
  tasks:
    - yum:
        name: firewalld
        state: present
      when: ansible_os_family == 'RedHat'

6.1.2 facts缓存

在执行playbook时,我们发现在「Gathering Facts」步骤时总会卡住一会,如果定义的play多了,会非常耗时。其实这步就是ansible在收集目的主机的facts信息。

如果我们定义的playbook中并没有使用到fact变量,那么我们可以选择将其关闭,只需添加「gather_facts: false」即可。

如果必须要使用facts信息,我们可以将fact信息缓存到redis服务或本地json文件中,这样当我们第二次执行playbook时,ansible就会读取缓存信息,从而加快运行速度。

假设本地redis服务正常运行,我们只需更改ansible配置文件即可达到缓存fact的目的。

redis缓存:

[defaults]
gathering = smart
fact_caching = redis
fact_caching_timeout = 86400

json文件缓存:

[defaults]
gathering = smart
fact_caching = jsonfile
fact_caching_connection = /path/to/cachedir
fact_caching_timeout = 86400

6.2 inventory中的变量

6.2.1 主机变量

关于主机和主机组变量,我们在「ansible基础-安装与配置:3.1 主机与主机组」中有介绍,当时介绍了组和主机变量的定义方法、变量的分离、优先级等知识点。其实主机变量的知识点不复杂,这里做下总结。主机变量是指作用在某一台主机上的变量。位置可以与主机定义写在一起也可以写在inventory/host_vars/a_host_name.yaml文件里。通常前者使用使用INI格式,后者使用YAML格式。这里要注意一下YAML的语法,在「:」后面要留有一个空格。如果组变量和主机变量都对同一个主机定义了相同的变量,那么ansible最终会采用主机变量而放弃组变量。主机变量示例:INI格式:

# inventory/hosts
[nodes]
node1
node2
node3
[nodes:vars]
http_port=80
database_port=3306

转换为YAML格式:

---
# inventory/hosts
nodes:
  hosts:
    node1:
    node2:
    node3:
  vars:
    http_port: \'80\'
    database_port: \'3306\'

不含节点定义的主机变量定义:

---
# inventory/host_vars/node1.yaml
http_port: \'80\'
database_port: \'3306\'

6.2.2 组变量 

和主机变量类似,组变量作用于主机组,即多个主机。位置可以与主机组定义写在一起也可以写在inventory/group_vars/a_group_name.yaml文件里。通常前者使用使用INI格式,后者使用YAML格式。

INI格式:

# inventory/hosts
[nodes]
node1
node2
node3
[nodes:vars]
http_port=80
database_port=3306

转换为YAML格式:

---
# inventory/hosts
nodes:
  hosts:
    node1:
    node2:
    node3:
  vars:
    http_port: \'80\'
    database_port: \'3306\'

定义一个字典变量,位于inventory/group_vars/nodes.yaml:

---
# 一位职工记录
name: Maurice
job: Developer
skill: Develop program
employed: True
foods:
    - Apple
    - Orange
languages:
    shell: Elite
    python: Elite
    c++: Lame

将上面示例转换为一行:

---
# 一位职工记录
{name: Maurice,job: Developer,skill: Develop program,employed: True,foods: [\'Apple\',\'Orange\'],languages: {shell: Elite,python: Elite,c++: Lame}}

很显然,YANL格式分行来写会更加直观和美观。

七 变量的调用顺序

通过上面描述,我们发现ansible能够定义变量的地方真的是太多太多了,我个人觉得ansible变量这块的设计有点复杂和冗余。

在生产中,我们要读懂别人的代码或者让自己的代码更加健壮,就必须清楚的知道ansible最终会使用哪个变量。这里我总结下ansible的调用变量的顺序,当小伙伴迷茫时可以回来看下这个列表(越靠后变量优先级越高,越会被ansible采用)

  • 命令行参数(非-e指定的参数,eg: "-u user -b yes")
  • roles defaults目录下的变量
  • 组变量:inventory 文件
  • 组变量:inventory/group_vars/all
  • 组变量:playbook/group_vars/all
  • 组变量:inventory/group_vars/*
  • 组变量:playbook/group_vars/*
  • 主机变量:inventory 文件
  • 主机变量:inventory/group_vars/*
  • 主机变量:playbook/group_vars/*
  • facts变量
  • play变量:vars定义的
  • play变量:vars_prompt定义的
  • play变量:vars_files导入的
  • roles vars目录下的变量
  • block中task定义的变量
  • playbook中task定义的变量
  • include_vars导入的变量
  • set_facts/register注册的变量
  • 使用roles/include_role/import_role语句时定义的变量
  • 使用include语句(ansible旧版本)时定义的变量
  • 命令行-e参数指定的额外变量(优先级最高)

八 变量的使用

8.1 模块使用变量

一个变量被定义后,在它的作用域内的play可以直接调用,例如:

我们定义了整个play作用域的变量「name」和「age」,那么在之后的两个debug模块内可以直接调用。

---
# playbooks/test.yaml
- hosts: node1
  vars:
    name: Maurice
    age: 27
  tasks:
    - debug:
        msg: "My name is {{ name }} and I'm {{ age }} years old"
    - debug:
        msg: "Hello Maurice"
      when: name == 'Maurice'

输出结果展示:

➜  lab-ansible ansible-playbook playbooks/test.yaml


PLAY [node1] ********************************************************************************************

TASK [Gathering Facts] **********************************************************************************
ok: [node1]

TASK [debug] ********************************************************************************************
ok: [node1] => {
    "msg": "My name is Maurice and I'm 27 years old"
}

TASK [debug] ********************************************************************************************
ok: [node1] => {
    "msg": "Hello Maurice"
}

PLAY RECAP **********************************************************************************************
node1                      : ok=3    changed=0    unreachable=0    failed=0

8.2 模版使用变量

变量被频繁使用的还有roles里的模版,位于roles/template/xxx.j2,该模版使用python的Jinja2模版语法。

roles模版多被用于生成服务的配置文件,所以会调用很多的变量。

示例如下:

  "customerMonthInfo": "{{ cmp_server_customerMonthInfo }}",
  "type_black_list": [{% for type_black in cmp_server_type_black_list %}"{{ type_black }}"{{ '' if loop.last else ',' }}{% endfor %}],

上述示例中「customerMonthInfo」的参数比较简单,就是变量「cmp_server_customerMonthInfo」的值

「type_black_list」参数从列表变量「cmp_server_type_black_list」中获取,执行结果是个字符串,字符串由该列表的元素以逗号为间隔组成,最后一个参数后没有逗号。

九 本节应该掌握的技能

  • 掌握变量的命名规范
  • 掌握变量定义的方法
  • 掌握变量的作用域及调用顺序
  • 会在模块和模版里使用变量
  • 熟悉Jinja2模版的语法规则

十 参考链接

  • https://docs.ansible.com/ansible/latest/user_guide/playbooks_variables.html#passing-variables-on-the-command-line
  • https://docs.ansible.com/ansible/latest/user_guide/playbooks_reuse_roles.html
  • https://docs.ansible.com/ansible/latest/user_guide/intro_inventory.html
  • 红帽DO407 Automation with Ansible 教材

 

欢迎大家关注我的公众号:

posted @ 2018-12-02 17:49  MauriceWei  阅读(11549)  评论(1编辑  收藏  举报