mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-24 21:14:00 -07:00 
			
		
		
		
	Merge pull request #120 from skvidal/master
virt module + idempotents (state=[shutdown|running])
This commit is contained in:
		
				commit
				
					
						c1461bc185
					
				
			
		
					 1 changed files with 449 additions and 0 deletions
				
			
		
							
								
								
									
										449
									
								
								library/virt
									
										
									
									
									
										Executable file
									
								
							
							
						
						
									
										449
									
								
								library/virt
									
										
									
									
									
										Executable file
									
								
							|  | @ -0,0 +1,449 @@ | |||
| #!/usr/bin/python  | ||||
| """ | ||||
| Virt management features | ||||
| 
 | ||||
| Copyright 2007, 2012 Red Hat, Inc | ||||
| Michael DeHaan <mdehaan@redhat.com> | ||||
| Seth Vidal <skvidal@fedoraproject.org> | ||||
| 
 | ||||
| This software may be freely redistributed under the terms of the GNU | ||||
| general public license. | ||||
| 
 | ||||
| You should have received a copy of the GNU General Public License | ||||
| along with this program; if not, write to the Free Software | ||||
| Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | ||||
| """ | ||||
| 
 | ||||
| VIRT_FAILED = 1 | ||||
| VIRT_SUCCESS = 0 | ||||
| VIRT_UNAVAILABLE=2 | ||||
| 
 | ||||
| try: | ||||
|     import json | ||||
| except ImportError: | ||||
|     import simplejson as json | ||||
| 
 | ||||
| 
 | ||||
| # other modules | ||||
| import os | ||||
| import sys | ||||
| import subprocess | ||||
| try: | ||||
|     import libvirt | ||||
| except ImportError: | ||||
|     print json.dumps({ | ||||
|         "failed" : True, | ||||
|         "msg" : "libvirt python module unavailable", | ||||
|         "rc": VIRT_UNAVAILABLE, | ||||
|         })    | ||||
|     sys.exit(1) | ||||
| 
 | ||||
| 
 | ||||
| import shlex | ||||
| 
 | ||||
| VIRT_STATE_NAME_MAP = { | ||||
|    0 : "running", | ||||
|    1 : "running", | ||||
|    2 : "running", | ||||
|    3 : "paused", | ||||
|    4 : "shutdown", | ||||
|    5 : "shutdown", | ||||
|    6 : "crashed" | ||||
| } | ||||
| 
 | ||||
| class LibvirtConnection(object): | ||||
| 
 | ||||
|     def __init__(self): | ||||
| 
 | ||||
|         cmd = subprocess.Popen("uname -r", shell=True, stdout=subprocess.PIPE, | ||||
|                                 close_fds=True) | ||||
|         output = cmd.communicate()[0] | ||||
| 
 | ||||
|         if output.find("xen") != -1: | ||||
|             conn = libvirt.open(None) | ||||
|         else: | ||||
|             conn = libvirt.open("qemu:///system") | ||||
| 
 | ||||
|         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 Exception("virtual machine %s not found" % vmid) | ||||
| 
 | ||||
|     def shutdown(self, vmid): | ||||
|         return self.find_vm(vmid).shutdown() | ||||
| 
 | ||||
|     def pause(self, vmid): | ||||
|         return self.suspend(self.conn,vmid) | ||||
| 
 | ||||
|     def unpause(self, vmid): | ||||
|         return self.resume(self.conn,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] | ||||
|         # print "DEBUG: state: %s" % state | ||||
|         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_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) | ||||
|      | ||||
| 
 | ||||
| 
 | ||||
| class Virt(object): | ||||
| 
 | ||||
|     def __get_conn(self): | ||||
|         self.conn = LibvirtConnection() | ||||
|         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] = { | ||||
|                 "state"     : VIRT_STATE_NAME_MAP.get(data[0],"unknown"), | ||||
|                 "maxMem"    : str(data[1]), | ||||
|                 "memory"    : str(data[2]), | ||||
|                 "nrVirtCpu" : data[3], | ||||
|                 "cpuTime"   : str(data[4]), | ||||
|             } | ||||
|             thisvm = self.conn.find_vm(vm) | ||||
|             if hasattr(thisvm, 'autostart'): | ||||
|                 info[vm]["autostart"] = thisvm.autostart() | ||||
| 
 | ||||
|         return info | ||||
| 
 | ||||
|     def nodeinfo(self): | ||||
|         self.__get_conn() | ||||
|         info = dict() | ||||
|         data = self.conn.nodeinfo() | ||||
|         info = { | ||||
|             "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): | ||||
|         self.conn = self.__get_conn() | ||||
|         vms = self.conn.find_vm(-1) | ||||
|         results = [] | ||||
|         for x in vms: | ||||
|             try: | ||||
|                 results.append(x.name()) | ||||
|             except: | ||||
|                 pass | ||||
|         return results | ||||
| 
 | ||||
|     def virttype(self): | ||||
|         return self.__get_conn().get_type() | ||||
|          | ||||
| 
 | ||||
|     def autostart(self, vmid): | ||||
|         self.conn = self.__get_conn() | ||||
|         return self.conn.set_autostart(vmid, True) | ||||
| 
 | ||||
|     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() | ||||
|         self.conn.shutdown(vmid) | ||||
|         return 0 | ||||
| 
 | ||||
| 
 | ||||
|     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 mac address. | ||||
|         """ | ||||
|         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. | ||||
|         by deleting the disk image and it's configuration 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): | ||||
|         """ | ||||
|         Recieve a Vm id as input | ||||
|         Return an xml describing vm config returned by a libvirt call | ||||
|         """ | ||||
|         conn = libvirt.openReadOnly(None) | ||||
|         if conn == None: | ||||
|             return (-1,'Failed to open connection to the hypervisor') | ||||
|         try: | ||||
|             domV = conn.lookupByName(vmid) | ||||
|         except: | ||||
|             return (-1,'Failed to find the main domain') | ||||
|         return domV.XMLDesc(0) | ||||
| 
 | ||||
| 
 | ||||
|     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 main(): | ||||
|     rc = VIRT_SUCCESS | ||||
|     vm_commands = ['create','status', 'start', 'stop', 'pause', 'unpause',  | ||||
|                 'shutdown', 'undefine', 'destroy', 'get_xml', 'autostart'] | ||||
|     host_commands = ['freemem', 'list_vms', 'info', 'nodeinfo', 'virttype'] | ||||
|      | ||||
|     msg = """ | ||||
|     virtmodule arguments: | ||||
|       state=[running|shutdown] guest=guestname | ||||
|       command=some_virt_command  [guest=guestname] | ||||
|     """ | ||||
|      | ||||
|     if len(sys.argv) == 1: | ||||
|         return VIRT_FAILED, msg | ||||
| 
 | ||||
|     argfile = sys.argv[1] | ||||
|     if not os.path.exists(argfile): | ||||
|         msg = "Argument file not found" | ||||
|         return VIRT_FAILED, msg | ||||
| 
 | ||||
|     args = open(argfile, 'r').read() | ||||
|     items = shlex.split(args) | ||||
| 
 | ||||
|     if not len(items): | ||||
|         return VIRT_FAILED, msg | ||||
| 
 | ||||
| 
 | ||||
|     # guest=name state=[running|shutdown|destroyed|undefined] | ||||
|     # command=[some command] [guest=name] | ||||
|      | ||||
|     params = {} | ||||
|     for x in items: | ||||
|         (k, v) = x.split("=") | ||||
|         params[k] = v | ||||
| 
 | ||||
|     state      = params.get('state', None) | ||||
|     guest      = params.get('guest', None) | ||||
|     command    = params.get('command', None) | ||||
|     options    = params.get('options', []) | ||||
| 
 | ||||
|     v = Virt() | ||||
|     res = {} | ||||
|      | ||||
|     if state: | ||||
|         if not guest: | ||||
|             msg = "state change requires a guest specified" | ||||
|             return VIRT_FAILED, msg | ||||
|        | ||||
|         res['changed'] = False | ||||
|         if state == 'running': | ||||
|             if v.status(guest) is not 'running': | ||||
|                 res['changed'] = True  | ||||
|                 res['msg'] = v.start(guest) | ||||
|         elif state == 'shutdown': | ||||
|             if v.status(guest) is not 'shutdown': | ||||
|                 res['changed'] = True | ||||
|                 res['msg'] = v.shutdown(guest) | ||||
|          | ||||
|         return VIRT_SUCCESS, res | ||||
|          | ||||
|     if command: | ||||
|         if command in vm_commands: | ||||
|             if not guest: | ||||
|                 msg = "%s requires 1 argument: guest" % command | ||||
|                 return VIRT_FAILED, msg | ||||
|                               | ||||
|             res = getattr(v, command)(guest) | ||||
|             if type(res) != dict: | ||||
|                 res = { command: res } | ||||
|             return rc, res | ||||
|          | ||||
|         elif hasattr(v, command): | ||||
|             res = getattr(v, command)() | ||||
|             if type(res) != dict: | ||||
|                 res = { command: res } | ||||
|             return rc, res | ||||
|              | ||||
|         else: | ||||
|             msg = "Command %s not recognized" % basecmd | ||||
|             rc = VIRT_FAILED | ||||
|          | ||||
|     return rc, msg | ||||
|      | ||||
| if __name__ == "__main__": | ||||
|     try: | ||||
|         rc, result = main() | ||||
|     except Exception, e: | ||||
|         rc = 1 | ||||
|         result = str(e) | ||||
| 
 | ||||
|     if rc != 0: # something went wrong emit the msg | ||||
|         print json.dumps({ | ||||
|             "failed" : rc, | ||||
|             "msg" : result, | ||||
|             "rc": rc, | ||||
|         })    | ||||
|         sys.exit(rc) | ||||
|     else: | ||||
|         print json.dumps(result) | ||||
|      | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue