-
Notifications
You must be signed in to change notification settings - Fork 2
/
net_xml.py
261 lines (208 loc) · 7.98 KB
/
net_xml.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
#!/usr/bin/python
# -*- coding: utf-8 -*-
# vim:set fileformat=unix shiftwidth=4 softtabstop=4 expandtab:
# kate: end-of-line unix; space-indent on; indent-width 4; remove-trailing-spaces modified;
# Copyright: (c) 2020, Jakob Meng <[email protected]>
# Based on community.libvirt.virt_pool module written by Maciej Delmanowski <[email protected]>
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
from __future__ import absolute_import, division, print_function
__metaclass__ = type
ANSIBLE_METADATA = {'metadata_version': '1.1',
'status': ['preview'],
'supported_by': 'community'}
DOCUMENTATION = r'''
---
module: net_xml
short_description: Create/Modify/Delete a libvirt virtual network.
description:
- "This module allows one to create, modify and delete a libvirt virtual network."
- "For use in addition to M(community.libvirt.virt_net)."
- "Compared to M(community.libvirt.virt_net), this module applies changes to a network whenever necessary while the
former has to be called with 'C(command): I(modify)' explicitly to apply any changes."
requirements: []
options:
ignore:
default:
- /network/mac
- /network/uuid
description:
- "XPath expressions to XML nodes that are ignored when comparing I(xml) to existing network configuration."
- "XPath expressions must return XML nodes only, e.g. '/network/uuid'. Other XPath expressions, such as
'/network/uuid/text()' are not supported."
- "When modifying a network, ignored XML nodes will be taken from existing network, i.e. they will not be
overwritten."
type: list
state:
choices: [present, absent]
default: present
description:
- "Should the network be present or absent."
- "If network does not exist and I(state) is C(present), then the network will be defined, but not started.
Use M(community.libvirt.virt_net) to start the network."
- "If network does exist and I(state) is C(absent), then the network will not be stopped prior to undefine.
Use M(community.libvirt.virt_net) to stop the network."
type: str
xml:
description:
- "XML document used to define or modify the network."
- "Must be raw XML content using C(lookup). XML cannot be reference to a file."
required: true
type: str
notes:
- "For changes to take effect, a modified network might have to be restarted. To do so, e.g. call
M(community.libvirt.virt_net) with 'C(command): I(stop)' and 'C(command): I(start)'."
extends_documentation_fragment:
- jm1.libvirt.libvirt
author: "Jakob Meng (@jm1)"
'''
EXAMPLES = r'''
- name: Create or modify a network
jm1.libvirt.net_xml:
state: present
xml: |
<network>
<name>nat-0</name>
<forward mode='nat'/>
<bridge name='virbr-nat-0' stp='on' delay='0'/>
<mac address='52:54:00:50:00:10'/>
<ip address='192.168.122.1' netmask='255.255.255.0'>
<dhcp>
<range start='192.168.122.2' end='192.168.122.254'/>
</dhcp>
</ip>
</network>
'''
RETURN = r'''
xml:
description: Full XML dump of libvirt's network config, including ignored XML element tags
returned: changed or success
type: str
sample: |
<network>
<name>nat-0</name>
<uuid>363b4985-8284-49ca-8e71-59cfee876a1a</uuid>
<forward mode='nat'/>
<bridge name='virbr-nat-0' stp='on' delay='0'/>
<mac address='52:54:00:50:00:10'/>
<ip address='192.168.122.1' netmask='255.255.255.0'>
<dhcp>
<range start='192.168.122.2' end='192.168.122.254'/>
</dhcp>
</ip>
</network>
'''
# NOTE: Synchronize imports with DOCUMENTATION string above and chapter Requirements in roles/server/README.md
from ansible_collections.jm1.libvirt.plugins.module_utils import libvirt as libvirt_utils
from ansible.module_utils._text import to_native
from ansible.module_utils.basic import AnsibleModule
import traceback
try:
import libvirt
except ImportError:
# error handled in libvirt_utils.try_import() below
pass
try:
from lxml import etree
except ImportError:
# error handled in libvirt_utils.try_import() below
pass
def lookup_uuid(xml):
uuids = xml.xpath('/network/uuid')
if len(uuids) > 1:
raise ValueError("network config is invalid: xml has more than one 'uuid' element")
if uuids:
return uuids[0].text
return None
def lookup_name(xml):
names = xml.xpath('/network/name')
if len(names) > 1:
raise ValueError("network config is invalid: xml has more than one 'name' element")
if names:
return names[0].text
return None
def lookup_uuid_and_name(xml):
uuid = lookup_uuid(xml)
name = lookup_name(xml)
if not uuid and not name:
raise ValueError("network config is invalid: xml requires an 'uuid', an 'name' element or both")
return uuid, name
def lookup_network(conn, uuid, name):
try:
if uuid:
return conn.networkLookupByUUIDString(uuid)
elif name:
return conn.networkLookupByName(name)
except libvirt.libvirtError as e:
if e.get_error_code() != libvirt.VIR_ERR_NO_NETWORK:
raise
return None
def create_or_modify(ignore_xpaths, uri, xml, module):
xml_root = etree.fromstring(xml)
uuid, name = lookup_uuid_and_name(xml_root)
with libvirt_utils.Connection(uri, module) as conn:
network = lookup_network(conn, uuid, name)
if not network:
# create
network = conn.networkDefineXML(xml)
return True, network.XMLDesc(flags=libvirt.VIR_NETWORK_XML_INACTIVE)
else:
# maybe modify
old_xml = network.XMLDesc(flags=libvirt.VIR_NETWORK_XML_INACTIVE)
if libvirt_utils.xml_strings_equal(old_xml, xml, ignore_xpaths):
# network does not require update
return False, old_xml
xml = libvirt_utils.update_xml_desc(old_xml, xml, ignore_xpaths)
network = conn.networkDefineXML(xml)
return True, network.XMLDesc(flags=libvirt.VIR_NETWORK_XML_INACTIVE)
def delete(ignore_xpaths, uri, xml, module):
xml_root = etree.fromstring(xml)
uuid, name = lookup_uuid_and_name(xml_root)
with libvirt_utils.Connection(uri, module) as conn:
network = lookup_network(conn, uuid, name)
if not network:
# network absent already
return False, None
xml = network.XMLDesc(flags=libvirt.VIR_NETWORK_XML_INACTIVE) # fetch xml before deletion
network.undefine()
return True, xml
def core(module):
ignore = module.params['ignore']
state = module.params['state']
uri = module.params['uri']
xml = module.params['xml']
if module.check_mode:
return dict(
changed=False,
ignore=ignore,
state=state,
uri=uri,
xml=xml)
if state == 'present':
changed, xml = create_or_modify(ignore, uri, xml, module)
elif state == 'absent':
changed, xml = delete(ignore, uri, xml, module)
return dict(
changed=changed,
ignore=ignore,
state=state,
uri=uri,
xml=xml)
def main():
module = AnsibleModule(
argument_spec=dict(
ignore=dict(type='list', default=['/network/mac', '/network/uuid']),
state=dict(type='str', choices=['present', 'absent'], default='present'),
uri=dict(default='qemu:///system'),
xml=dict(required=True, type='str')
),
supports_check_mode=True
)
libvirt_utils.try_import(module)
try:
result = core(module)
except Exception as e:
module.fail_json(msg=to_native(e), exception=traceback.format_exc())
else:
module.exit_json(**result)
if __name__ == '__main__':
main()