Skip to content

Instantly share code, notes, and snippets.

@victorock
Last active July 6, 2018 01:39
Show Gist options
  • Save victorock/c8319b62b7ce11b4bb51d2666fc9556f to your computer and use it in GitHub Desktop.
Save victorock/c8319b62b7ce11b4bb51d2666fc9556f to your computer and use it in GitHub Desktop.
#!/usr/bin/python
# -*- coding: utf-8 -*-
# Copyright: (c) 2007, 2012 Red Hat, Inc
# Michael DeHaan <michael.dehaan@gmail.com>
# Seth Vidal <skvidal@fedoraproject.org>
# Victor da Costa <victorockeiro@gmail.com>
# 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 = '''
---
module: virt
short_description: Manages virtual machines supported by libvirt
description:
- Manages virtual machines supported by I(libvirt).
version_added: "0.2"
options:
name:
description:
- name of the guest VM being managed. Note that name must match the domain name defined in the xml.
required: true
state:
description:
- Note that there may be some lag for state requests like C(shutdown)
since these refer only to VM states. After starting a guest, it may not
be immediately accessible.
choices: [ 'present', 'absent', 'running', 'stopped', 'poweredoff', 'paused' ]
comments:
- Presence of Domain is enforced to ensure idempotency.
- `present` ensure domain presence without changing the current state.
- `absent` poweroff -> then remove domain.
- `running` present -> then unpause (if required) before starting the domain.
- `stopped` present -> then gracefully shutdown the domain.
- `poweredoff` present -> then unplug the virtual cable of the domain.
- `paused` present -> then pause the domain.
autostart:
description:
- start VM at host startup.
type: bool
version_added: "2.3"
uri:
description:
- libvirt connection uri.
default: qemu:///system
xml:
description:
- XML document used with the states.
- Must be raw XML content using C(lookup). XML cannot be reference to a file.
requirements:
- python >= 2.6
- libvirt-python
author:
- Ansible Core Team
- Michael DeHaan
- Seth Vidal
- Victor da Costa
'''
EXAMPLES = '''
# a playbook task line:
- virt:
name: alpha
xml: "{{ lookup('template', 'machine-template.xml.j2') }}"
state: running
'''
RETURN = '''
# for list_vms command
list_vms:
description: The list of vms defined on the remote system
type: dictionary
returned: success
sample: [
"build.example.org",
"dev.example.org"
]
# for status command
status:
description: The status of the VM, among running, crashed, paused and shutdown
type: string
sample: "success"
returned: success
'''
import traceback
import time
try:
import libvirt
except ImportError:
HAS_VIRT = False
else:
HAS_VIRT = True
from ansible.module_utils.basic import AnsibleModule
from ansible.module_utils._text import to_native
VIRT_FAILED = 1
VIRT_SUCCESS = 0
VIRT_UNAVAILABLE = 2
STATES = [ 'present', 'absent', 'running', 'poweredoff', 'stopped', 'paused' ]
VIRT_STATE_NAME_MAP = {
0: 'running',
1: 'running',
2: 'running',
3: 'paused',
4: 'shutdown',
5: 'shutdown',
6: 'crashed',
}
class VMNotFound(Exception):
pass
class WaitForStatusTimeout(Exception):
pass
class LibvirtConnection(object):
def __init__(self, uri, module):
self.module = module
cmd = "uname -r"
rc, stdout, stderr = self.module.run_command(cmd)
if "xen" in stdout:
conn = libvirt.open(None)
elif "esx" in uri:
auth = [[libvirt.VIR_CRED_AUTHNAME, libvirt.VIR_CRED_NOECHOPROMPT], [], None]
conn = libvirt.openAuth(uri, auth)
else:
conn = libvirt.open(uri)
if not conn:
raise Exception("hypervisor connection failure")
self.conn = conn
def find_vm(self, vmid):
"""
Extra bonus feature: vmid = -1 returns a list of everything
"""
conn = self.conn
vms = []
# this block of code borrowed from virt-manager:
# get working domain's name
ids = conn.listDomainsID()
for id in ids:
vm = conn.lookupByID(id)
vms.append(vm)
# get defined domain
names = conn.listDefinedDomains()
for name in names:
vm = conn.lookupByName(name)
vms.append(vm)
if vmid == -1:
return vms
for vm in vms:
if vm.name() == vmid:
return vm
raise VMNotFound("virtual machine %s not found" % vmid)
def shutdown(self, vmid):
return self.find_vm(vmid).shutdown()
def pause(self, vmid):
return self.suspend(vmid)
def unpause(self, vmid):
return self.resume(vmid)
def suspend(self, vmid):
return self.find_vm(vmid).suspend()
def resume(self, vmid):
return self.find_vm(vmid).resume()
def create(self, vmid):
return self.find_vm(vmid).create()
def destroy(self, vmid):
return self.find_vm(vmid).destroy()
def undefine(self, vmid):
return self.find_vm(vmid).undefine()
def get_status2(self, vm):
state = vm.info()[0]
return VIRT_STATE_NAME_MAP.get(state, "unknown")
def get_status(self, vmid):
state = self.find_vm(vmid).info()[0]
return VIRT_STATE_NAME_MAP.get(state, "unknown")
def nodeinfo(self):
return self.conn.getInfo()
def get_type(self):
return self.conn.getType()
def get_xml(self, vmid):
vm = self.conn.lookupByName(vmid)
return vm.XMLDesc(0)
def get_maxVcpus(self, vmid):
vm = self.conn.lookupByName(vmid)
return vm.maxVcpus()
def get_maxMemory(self, vmid):
vm = self.conn.lookupByName(vmid)
return vm.maxMemory()
def getFreeMemory(self):
return self.conn.getFreeMemory()
def get_autostart(self, vmid):
vm = self.conn.lookupByName(vmid)
return vm.autostart()
def set_autostart(self, vmid, val):
vm = self.conn.lookupByName(vmid)
return vm.setAutostart(val)
def define_from_xml(self, xml):
return self.conn.defineXML(xml)
class Virt(object):
def __init__(self, module):
self.module = module
self.uri = module.params.get('uri')
def __get_conn(self):
self.conn = LibvirtConnection(self.uri, self.module)
return self.conn
def get_vm(self, vmid):
self.__get_conn()
return self.conn.find_vm(vmid)
def state(self):
vms = self.list_vms()
state = []
for vm in vms:
state_blurb = self.conn.get_status(vm)
state.append("%s %s" % (vm, state_blurb))
return state
def info(self):
vms = self.list_vms()
info = dict()
for vm in vms:
data = self.conn.find_vm(vm).info()
# libvirt returns maxMem, memory, and cpuTime as long()'s, which
# xmlrpclib tries to convert to regular int's during serialization.
# This throws exceptions, so convert them to strings here and
# assume the other end of the xmlrpc connection can figure things
# out or doesn't care.
info[vm] = dict(
state=VIRT_STATE_NAME_MAP.get(data[0], "unknown"),
maxMem=str(data[1]),
memory=str(data[2]),
nrVirtCpu=data[3],
cpuTime=str(data[4]),
autostart=self.conn.get_autostart(vm),
)
return info
def nodeinfo(self):
self.__get_conn()
data = self.conn.nodeinfo()
info = dict(
cpumodel=str(data[0]),
phymemory=str(data[1]),
cpus=str(data[2]),
cpumhz=str(data[3]),
numanodes=str(data[4]),
sockets=str(data[5]),
cpucores=str(data[6]),
cputhreads=str(data[7])
)
return info
def list_vms(self, state=None):
self.conn = self.__get_conn()
vms = self.conn.find_vm(-1)
results = []
for x in vms:
try:
if state:
vmstate = self.conn.get_status2(x)
if vmstate == state:
results.append(x.name())
else:
results.append(x.name())
except:
pass
return results
def virttype(self):
return self.__get_conn().get_type()
def autostart(self, vmid, as_flag):
self.conn = self.__get_conn()
# Change autostart flag only if needed
if self.conn.get_autostart(vmid) != as_flag:
self.conn.set_autostart(vmid, as_flag)
return True
return False
def freemem(self):
self.conn = self.__get_conn()
return self.conn.getFreeMemory()
def shutdown(self, vmid):
""" Make the machine with the given vmid stop running. Whatever that takes. """
self.__get_conn()
return self.conn.shutdown(vmid)
def pause(self, vmid):
""" Pause the machine with the given vmid. """
self.__get_conn()
return self.conn.suspend(vmid)
def unpause(self, vmid):
""" Unpause the machine with the given vmid. """
self.__get_conn()
return self.conn.resume(vmid)
def create(self, vmid):
""" Start the machine via the given vmid """
self.__get_conn()
return self.conn.create(vmid)
def start(self, vmid):
""" Start the machine via the given id/name """
self.__get_conn()
return self.conn.create(vmid)
def destroy(self, vmid):
""" Pull the virtual power from the virtual domain, giving it virtually no time to virtually shut down. """
self.__get_conn()
return self.conn.destroy(vmid)
def undefine(self, vmid):
""" Stop a domain, and then wipe it from the face of the earth. (delete disk/config file) """
self.__get_conn()
return self.conn.undefine(vmid)
def status(self, vmid):
"""
Return a state suitable for server consumption. Aka, codes.py values, not XM output.
"""
self.__get_conn()
return self.conn.get_status(vmid)
def get_xml(self, vmid):
"""
Receive a Vm id as input
Return an xml describing vm config returned by a libvirt call
"""
self.__get_conn()
return self.conn.get_xml(vmid)
def get_maxVcpus(self, vmid):
"""
Gets the max number of VCPUs on a guest
"""
self.__get_conn()
return self.conn.get_maxVcpus(vmid)
def get_max_memory(self, vmid):
"""
Gets the max memory on a guest
"""
self.__get_conn()
return self.conn.get_MaxMemory(vmid)
def define(self, xml):
"""
Define a guest with the given xml
"""
self.__get_conn()
return self.conn.define_from_xml(xml)
# https://libvirt.org/guide/html/Application_Development_Guide-Guest_Domains-Lifecycle.html
class DomainStateMachine(object):
def __init__(self, virt, module):
self._result = dict()
self._virt = virt
self._module = module
self._guest = module.params.get('name')
self._xml = module.params.get('xml')
self._autostart = module.params.get('autostart')
def result(self):
return self._result
def waitfor_status(self, status="running", timeout=1):
"""
Ensure statefull operations are sync
"""
for count in range(0, timeout):
time.sleep(1)
if self._virt.status(self._guest) == status:
return True
raise WaitForTimeout(
"""
State didn't transition to {status} after {timeout}
""".format(
status=status,
timeout=timeout
)
)
def shutdown(self):
if self._virt.status(self._guest) == "running":
self._virt.shutdown(self._guest)
self._result['changed'] = self.waitfor_status("shutdown", 50)
self._result['virt'] = self._virt.info()
return self
def poweroff(self):
if self._virt.status(self._guest) == "running":
self._virt.destroy(self._guest)
self._result['changed'] = self.waitfor_status("shutdown", 30)
self._result['virt'] = self._virt.info()
return self
def start(self):
if self._virt.status(self._guest) == "shutdown":
self._virt.start(self._guest)
self._result['changed'] = self.waitfor_status("running", 10)
self._result['virt'] = self._virt.info()
return self
def autostart(self):
self._result['changed'] = self._virt.autostart(self._guest, self._autostart)
self._result['virt'] = self._virt.info()
return self
def pause(self):
if self._virt.status(self._guest) == "running":
self._virt.pause(self._guest)
self._result['changed'] = self.waitfor_status("paused", 50)
self._result['virt'] = self._virt.info()
return self
def unpause(self):
if self._virt.status(self._guest) == "paused":
self._virt.unpause(self._guest)
self._result['changed'] = self.waitfor_status("running", 50)
self._result['virt'] = self._virt.info()
return self
def paused(self):
try:
self._result['changed'] = False
self.present()
self.pause()
except WaitForStatusTimeout:
raise
return self.result()
def absent(self):
try:
self._result['changed'] = False
self._virt.status(self._guest)
except VMNotFound:
pass
else:
try:
self.poweredoff()
# Check if domain still exists after being poweredoff
# Transitient domains will simply disappear
self._virt.status(self._guest)
self._virt.undefine(self._guest)
# Check if the domain was undefined
self._virt.status(self._guest)
except VMNotFound:
# The domain doesn't exists anymore
self._result['changed'] = True
self._result['virt'] = self._virt.info()
return self.result()
def present(self):
try:
self._result['changed'] = False
self._virt.status(self._guest)
except VMNotFound:
self._virt.define(self._xml)
self._result['changed'] = self.waitfor_status("shutdown", 10)
self._result['virt'] = self._virt.info()
return self.result()
def poweredoff(self):
try:
self._result['changed'] = False
self.present()
self.poweroff()
except WaitForStatusTimeout:
raise
finally:
return self.result()
def stopped(self):
try:
self._result['changed'] = False
self.present()
self.shutdown()
except WaitForStatusTimeout:
# Pull the vpower-cable (poweroff) if domain is not transitioning to shutdown
self.poweroff()
finally:
return self.result()
def running(self):
try:
self._result['changed'] = False
self.present()
self.unpause()
self.start()
self.autostart()
except WaitForStatusTimeout:
raise
return self.result()
def main():
module = AnsibleModule(
argument_spec=dict(
name=dict(required=True, type='str', aliases=['guest']),
state=dict(type='str', default='present', choices=STATES),
autostart=dict(type='bool', default=False),
uri=dict(type='str', default='qemu:///system'),
xml=dict(required=True, type='str'),
),
)
if not HAS_VIRT:
module.fail_json(msg='The `libvirt` module is not importable. Check the requirements.')
try:
virt = Virt(module)
fsm = DomainStateMachine(virt, module)
state = module.params.get('state')
result = getattr(fsm, state)()
except Exception as e:
module.fail_json(msg=to_native(e), exception=traceback.format_exc())
finally:
module.exit_json(**result)
if __name__ == '__main__':
main()
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment