如何将 "key = value;" 对的字符串转换为 ansible 中的字典?
How to convert a string of "key = value;" pairs to a dictionary in ansible?
基本上我正在寻找这个,但是 ansible/jinja2:
我从 mariadb galera 服务器得到一个以分号分隔的键=值对列表 wsrep_provider_options:
mariadb_wsrep_provider_options_output = 'base_dir = /var/lib/mysql/; base_host = 192.168.1.101; base_port = 4567;'
在使用 ansible register
存储该输出后,我试图将其解析为字典,如下所示:
mariadb_wsrep_provider_options_output:
base_dir: /var/lib/mysql/
base_host: 192.168.1.101
base_port: 4567
...
我试过类似的方法,但我不确定如何将它转换回 dict
:
- debug:
msg: "{% for item in mariadb_wsrep_provider_options_output.split(';') %} {{ item.split('=') }} {% endfor %}"
编辑:我用它处理以下任务,但它看起来很笨重:
- set_fact:
new_var: "{{ new_var|d([]) + [{ 'key': item.split('=')[0]|trim, 'value': item.split('=')[1]|trim }] }}"
loop: "{{ mariadb_wsrep_provider_options_output.split(';') }}"
- debug:
var: new_var | items2dict
输出:
"new_var|items2dict": {
"base_dir": "/var/lib/mysql/",
"base_host": "192.168.100.153",
"base_port": "4567"
}
试试这个
- set_fact:
new_var: "{{ dict(_keys|zip(_vals)) }}"
vars:
_arr: "{{ mdb_out.split(';')|map('trim')|select()|list }}"
_keys: "{{ _arr|map('regex_replace', '^(.*?)=(.*)$', '\1')|map('trim')|list }}"
_vals: "{{ _arr|map('regex_replace', '^(.*?)=(.*)$', '\2')|map('trim')|list }}"
给予
new_var:
base_dir: /var/lib/mysql/
base_host: 192.168.1.101
base_port: '4567'
您的代码稍作更改后也可以正常工作
- set_fact:
new_var: "{{ new_var|d({})|combine({_key: _val}) }}"
loop: "{{ mdb_out.split(';') }}"
when: item|length > 0
vars:
_key: "{{ item.split('=')[0]|trim }}"
_val: "{{ item.split('=')[1]|trim }}"
split 自 Ansible 2.11 起可用作过滤器。您可以在此处使用它代替 regex_replace。下面的任务给出了相同的结果
- set_fact:
new_var: "{{ dict(_keys|zip(_vals)) }}"
vars:
_arr: "{{ mdb_out.split(';')|map('trim')|select()|list }}"
_keys: "{{ _arr|map('split', '=')|map('first')|map('trim')|list }}"
_vals: "{{ _arr|map('split', '=')|map('last')|map('trim')|list }}"
基本上我正在寻找这个,但是 ansible/jinja2:
我从 mariadb galera 服务器得到一个以分号分隔的键=值对列表 wsrep_provider_options:
mariadb_wsrep_provider_options_output = 'base_dir = /var/lib/mysql/; base_host = 192.168.1.101; base_port = 4567;'
在使用 ansible register
存储该输出后,我试图将其解析为字典,如下所示:
mariadb_wsrep_provider_options_output:
base_dir: /var/lib/mysql/
base_host: 192.168.1.101
base_port: 4567
...
我试过类似的方法,但我不确定如何将它转换回 dict
:
- debug:
msg: "{% for item in mariadb_wsrep_provider_options_output.split(';') %} {{ item.split('=') }} {% endfor %}"
编辑:我用它处理以下任务,但它看起来很笨重:
- set_fact:
new_var: "{{ new_var|d([]) + [{ 'key': item.split('=')[0]|trim, 'value': item.split('=')[1]|trim }] }}"
loop: "{{ mariadb_wsrep_provider_options_output.split(';') }}"
- debug:
var: new_var | items2dict
输出:
"new_var|items2dict": {
"base_dir": "/var/lib/mysql/",
"base_host": "192.168.100.153",
"base_port": "4567"
}
试试这个
- set_fact:
new_var: "{{ dict(_keys|zip(_vals)) }}"
vars:
_arr: "{{ mdb_out.split(';')|map('trim')|select()|list }}"
_keys: "{{ _arr|map('regex_replace', '^(.*?)=(.*)$', '\1')|map('trim')|list }}"
_vals: "{{ _arr|map('regex_replace', '^(.*?)=(.*)$', '\2')|map('trim')|list }}"
给予
new_var:
base_dir: /var/lib/mysql/
base_host: 192.168.1.101
base_port: '4567'
您的代码稍作更改后也可以正常工作
- set_fact:
new_var: "{{ new_var|d({})|combine({_key: _val}) }}"
loop: "{{ mdb_out.split(';') }}"
when: item|length > 0
vars:
_key: "{{ item.split('=')[0]|trim }}"
_val: "{{ item.split('=')[1]|trim }}"
split 自 Ansible 2.11 起可用作过滤器。您可以在此处使用它代替 regex_replace。下面的任务给出了相同的结果
- set_fact:
new_var: "{{ dict(_keys|zip(_vals)) }}"
vars:
_arr: "{{ mdb_out.split(';')|map('trim')|select()|list }}"
_keys: "{{ _arr|map('split', '=')|map('first')|map('trim')|list }}"
_vals: "{{ _arr|map('split', '=')|map('last')|map('trim')|list }}"