#!/usr/bin/python # # Copyright (C) 2009 Sun Microsystems, Inc. # # This file is part of VirtualBox Open Source Edition (OSE), as # available from http://www.virtualbox.org. This file is free software; # you can redistribute it and/or modify it under the terms of the GNU # General Public License (GPL) as published by the Free Software # Foundation, in version 2 as it comes in the "COPYING" file of the # VirtualBox OSE distribution. VirtualBox OSE is distributed in the # hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. # # Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa # Clara, CA 95054 USA or visit http://www.sun.com if you need # additional information or have any questions. # # ################################################################################# # This program is a simple interactive shell for VirtualBox. You can query # # information and issue commands from a simple command line. # # # # It also provides you with examples on how to use VirtualBox's Python API. # # This shell is even somewhat documented and supports TAB-completion and # # history if you have Python readline installed. # # # # Enjoy. # ################################################################################ import os,sys import traceback class PerfCollector: """ This class provides a wrapper over IPerformanceCollector in order to get more 'pythonic' interface. To begin collection of metrics use setup() method. To get collected data use query() method. It is possible to disable metric collection without changing collection parameters with disable() method. The enable() method resumes metric collection. """ def __init__(self, vb): """ Initializes the instance. Pass an instance of IVirtualBox as parameter. """ self.collector = vb.performanceCollector def setup(self, names, objects, period, nsamples): """ Discards all previously collected values for the specified metrics, sets the period of collection and the number of retained samples, enables collection. """ self.collector.setupMetrics(names, objects, period, nsamples) def enable(self, names, objects): """ Resumes metric collection for the specified metrics. """ self.collector.enableMetrics(names, objects) def disable(self, names, objects): """ Suspends metric collection for the specified metrics. """ self.collector.disableMetrics(names, objects) def query(self, names, objects): """ Retrieves collected metric values as well as some auxiliary information. Returns an array of dictionaries, one dictionary per metric. Each dictionary contains the following entries: 'name': metric name 'object': managed object this metric associated with 'unit': unit of measurement 'scale': divide 'values' by this number to get float numbers 'values': collected data 'values_as_string': pre-processed values ready for 'print' statement """ (values, names_out, objects_out, units, scales, sequence_numbers, indices, lengths) = self.collector.queryMetricsData(names, objects) out = [] for i in xrange(0, len(names_out)): scale = int(scales[i]) if scale != 1: fmt = '%.2f%s' else: fmt = '%d %s' out.append({ 'name':str(names_out[i]), 'object':str(objects_out[i]), 'unit':str(units[i]), 'scale':scale, 'values':[int(values[j]) for j in xrange(int(indices[i]), int(indices[i])+int(lengths[i]))], 'values_as_string':'['+', '.join([fmt % (int(values[j])/scale, units[i]) for j in xrange(int(indices[i]), int(indices[i])+int(lengths[i]))])+']' }) return out # Simple implementation of IConsoleCallback, one can use it as skeleton # for custom implementations class GuestMonitor: def __init__(self, mach): self.mach = mach def onMousePointerShapeChange(self, visible, alpha, xHot, yHot, width, height, shape): print "%s: onMousePointerShapeChange: visible=%d" %(self.mach.name, visible) def onMouseCapabilityChange(self, supportsAbsolute, needsHostCursor): print "%s: onMouseCapabilityChange: needsHostCursor=%d" %(self.mach.name, needsHostCursor) def onKeyboardLedsChange(self, numLock, capsLock, scrollLock): print "%s: onKeyboardLedsChange capsLock=%d" %(self.mach.name, capsLock) def onStateChange(self, state): print "%s: onStateChange state=%d" %(self.mach.name, state) def onAdditionsStateChange(self): print "%s: onAdditionsStateChange" %(self.mach.name) def onDVDDriveChange(self): print "%s: onDVDDriveChange" %(self.mach.name) def onFloppyDriveChange(self): print "%s: onFloppyDriveChange" %(self.mach.name) def onNetworkAdapterChange(self, adapter): print "%s: onNetworkAdapterChange" %(self.mach.name) def onSerialPortChange(self, port): print "%s: onSerialPortChange" %(self.mach.name) def onParallelPortChange(self, port): print "%s: onParallelPortChange" %(self.mach.name) def onStorageControllerChange(self): print "%s: onStorageControllerChange" %(self.mach.name) def onVRDPServerChange(self): print "%s: onVRDPServerChange" %(self.mach.name) def onUSBControllerChange(self): print "%s: onUSBControllerChange" %(self.mach.name) def onUSBDeviceStateChange(self, device, attached, error): print "%s: onUSBDeviceStateChange" %(self.mach.name) def onSharedFolderChange(self, scope): print "%s: onSharedFolderChange" %(self.mach.name) def onRuntimeError(self, fatal, id, message): print "%s: onRuntimeError fatal=%d message=%s" %(self.mach.name, fatal, message) def onCanShowWindow(self): print "%s: onCanShowWindow" %(self.mach.name) return true def onShowWindow(self, winId): print "%s: onShowWindow: %d" %(self.mach.name, winId) class VBoxMonitor: def __init__(self, vbox): self.vbox = vbox pass def onMachineStateChange(self, id, state): print "onMachineStateChange: %s %d" %(id, state) def onMachineDataChange(self,id): print "onMachineDataChange: %s" %(id) def onExtraDataCanChange(self, id, key, value): print "onExtraDataCanChange: %s %s=>%s" %(id, key, value) return true def onExtraDataChange(self, id, key, value): print "onExtraDataChange: %s %s=>%s" %(id, key, value) def onMediaRegistred(self, id, type, registred): print "onMediaRegistred: %s" %(id) def onMachineRegistred(self, id, registred): print "onMachineRegistred: %s" %(id) def onSessionStateChange(self, id, state): print "onSessionStateChange: %s %d" %(id, state) def onSnapshotTaken(self, mach, id): print "onSnapshotTaken: %s %s" %(mach, id) def onSnapshotDiscarded(self, mach, id): print "onSnapshotDiscarded: %s %s" %(mach, id) def onSnapshotChange(self, mach, id): print "onSnapshotChange: %s %s" %(mach, id) def onGuestPropertyChange(self, id, val1, val2, val3): print "onGuestPropertyChange: %s" %(id) g_hasreadline = 1 try: import readline import rlcompleter except: g_hasreadline = 0 if g_hasreadline: class CompleterNG(rlcompleter.Completer): def __init__(self, dic, ctx): self.ctx = ctx return rlcompleter.Completer.__init__(self,dic) def complete(self, text, state): """ taken from: http://aspn.activestate.com/ASPN/Cookbook/Python/Recipe/496812 """ if text == "": return ['\t',None][state] else: return rlcompleter.Completer.complete(self,text,state) def global_matches(self, text): """ Compute matches when text is a simple name. Return a list of all names currently defined in self.namespace that match. """ matches = [] n = len(text) for list in [ self.namespace ]: for word in list: if word[:n] == text: matches.append(word) try: for m in getMachines(self.ctx): # although it has autoconversion, we need to cast # explicitly for subscripts to work word = str(m.name) if word[:n] == text: matches.append(word) word = str(m.id) if word[0] == '{': word = word[1:-1] if word[:n] == text: matches.append(word) except Exception,e: traceback.print_exc() print e return matches def autoCompletion(commands, ctx): if not g_hasreadline: return comps = {} for (k,v) in commands.items(): comps[k] = None completer = CompleterNG(comps, ctx) readline.set_completer(completer.complete) readline.parse_and_bind("tab: complete") g_verbose = True def split_no_quotes(s): return s.split() def createVm(ctx,name,kind,base): mgr = ctx['mgr'] vb = ctx['vb'] mach = vb.createMachine(name, kind, base, "00000000-0000-0000-0000-000000000000") mach.saveSettings() print "created machine with UUID",mach.id vb.registerMachine(mach) def removeVm(ctx,mach): mgr = ctx['mgr'] vb = ctx['vb'] id = mach.id print "removing machine ",mach.name,"with UUID",id session = ctx['global'].openMachineSession(id) mach=session.machine for d in mach.getHardDiskAttachments(): mach.detachHardDisk(d.controller, d.port, d.device) ctx['global'].closeMachineSession(session) mach = vb.unregisterMachine(id) if mach: mach.deleteSettings() def startVm(ctx,mach,type): mgr = ctx['mgr'] vb = ctx['vb'] perf = ctx['perf'] session = mgr.getSessionObject(vb) uuid = mach.id progress = vb.openRemoteSession(session, uuid, type, "") progress.waitForCompletion(-1) completed = progress.completed rc = int(progress.resultCode) print "Completed:", completed, "rc:",hex(rc&0xffffffff) if rc == 0: # we ignore exceptions to allow starting VM even if # perf collector cannot be started if perf: try: perf.setup(['*'], [mach], 10, 15) except Exception,e: print e if g_verbose: traceback.print_exc() pass # if session not opened, close doesn't make sense session.close() else: # Not yet implemented error string query API for remote API if not ctx['remote']: print session.QueryErrorObject(rc) def getMachines(ctx): return ctx['global'].getArray(ctx['vb'], 'machines') def asState(var): if var: return 'on' else: return 'off' def guestStats(ctx,mach): if not ctx['perf']: return for metric in ctx['perf'].query(["*"], [mach]): print metric['name'], metric['values_as_string'] def guestExec(ctx, machine, console, cmds): exec cmds def monitorGuest(ctx, machine, console, dur): import time cb = ctx['global'].createCallback('IConsoleCallback', GuestMonitor, machine) console.registerCallback(cb) if dur == -1: # not infinity, but close enough dur = 100000 try: end = time.time() + dur while time.time() < end: ctx['global'].waitForEvents(500) # We need to catch all exceptions here, otherwise callback will never be unregistered except: pass console.unregisterCallback(cb) def monitorVbox(ctx, dur): import time vbox = ctx['vb'] cb = ctx['global'].createCallback('IVirtualBoxCallback', VBoxMonitor, vbox) vbox.registerCallback(cb) if dur == -1: # not infinity, but close enough dur = 100000 try: end = time.time() + dur while time.time() < end: ctx['global'].waitForEvents(500) # We need to catch all exceptions here, otherwise callback will never be unregistered except Exception,e: print e if g_verbose: traceback.print_exc() pass vbox.unregisterCallback(cb) def cmdExistingVm(ctx,mach,cmd,args): mgr=ctx['mgr'] vb=ctx['vb'] session = mgr.getSessionObject(vb) uuid = mach.id try: progress = vb.openExistingSession(session, uuid) except Exception,e: print "Session to '%s' not open: %s" %(mach.name,e) if g_verbose: traceback.print_exc() return if session.state != ctx['ifaces'].SessionState_Open: print "Session to '%s' in wrong state: %s" %(mach.name, session.state) return # unfortunately IGuest is suppressed, thus WebServices knows not about it # this is an example how to handle local only functionality if ctx['remote'] and cmd == 'stats2': print 'Trying to use local only functionality, ignored' return console=session.console ops={'pause' : lambda: console.pause(), 'resume': lambda: console.resume(), 'powerdown': lambda: console.powerDown(), 'stats': lambda: guestStats(ctx, mach), 'guest': lambda: guestExec(ctx, mach, console, args), 'monitorGuest': lambda: monitorGuest(ctx, mach, console, args) } try: ops[cmd]() except Exception, e: print 'failed: ',e if g_verbose: traceback.print_exc() session.close() # can cache known machines, if needed def machById(ctx,id): mach = None for m in getMachines(ctx): if m.name == id: mach = m break mid = str(m.id) if mid[0] == '{': mid = mid[1:-1] if mid == id: mach = m break return mach def argsToMach(ctx,args): if len(args) < 2: print "usage: %s [vmname|uuid]" %(args[0]) return None id = args[1] m = machById(ctx, id) if m == None: print "Machine '%s' is unknown, use list command to find available machines" %(id) return m def helpCmd(ctx, args): if len(args) == 1: print "Help page:" for i in commands: print " ",i,":", commands[i][0] else: c = commands.get(args[1], None) if c == None: print "Command '%s' not known" %(args[1]) else: print " ",args[1],":", c[0] return 0 def listCmd(ctx, args): for m in getMachines(ctx): print "Machine '%s' [%s], state=%s" %(m.name,m.id,m.sessionState) return 0 def infoCmd(ctx,args): if (len(args) < 2): print "usage: info [vmname|uuid]" return 0 mach = argsToMach(ctx,args) if mach == None: return 0 os = ctx['vb'].getGuestOSType(mach.OSTypeId) print " Name: ",mach.name print " ID: ",mach.id print " OS Type: ",os.description print " RAM: %dM" %(mach.memorySize) print " VRAM: %dM" %(mach.VRAMSize) print " Clipboard mode: %d" %(mach.clipboardMode) print " Machine status: " ,mach.sessionState bios = mach.BIOSSettings print " BIOS ACPI: ",bios.ACPIEnabled print " PAE: ",mach.PAEEnabled print " Hardware virtualization: ",asState(mach.HWVirtExEnabled) print " Nested paging: ",asState(mach.HWVirtExNestedPagingEnabled) print " Last changed: ",mach.lastStateChange return 0 def startCmd(ctx, args): mach = argsToMach(ctx,args) if mach == None: return 0 if len(args) > 2: type = args[2] else: type = "gui" startVm(ctx, mach, type) return 0 def createCmd(ctx, args): if (len(args) < 3 or len(args) > 4): print "usage: create name ostype " return 0 name = args[1] oskind = args[2] if len(args) == 4: base = args[3] else: base = '' try: ctx['vb'].getGuestOSType(oskind) except Exception, e: print 'Unknown OS type:',oskind return 0 createVm(ctx, name, oskind, base) return 0 def removeCmd(ctx, args): mach = argsToMach(ctx,args) if mach == None: return 0 removeVm(ctx, mach) return 0 def pauseCmd(ctx, args): mach = argsToMach(ctx,args) if mach == None: return 0 cmdExistingVm(ctx, mach, 'pause', '') return 0 def powerdownCmd(ctx, args): mach = argsToMach(ctx,args) if mach == None: return 0 cmdExistingVm(ctx, mach, 'powerdown', '') return 0 def resumeCmd(ctx, args): mach = argsToMach(ctx,args) if mach == None: return 0 cmdExistingVm(ctx, mach, 'resume', '') return 0 def statsCmd(ctx, args): mach = argsToMach(ctx,args) if mach == None: return 0 cmdExistingVm(ctx, mach, 'stats', '') return 0 def guestCmd(ctx, args): if (len(args) < 3): print "usage: guest name commands" return 0 mach = argsToMach(ctx,args) if mach == None: return 0 cmdExistingVm(ctx, mach, 'guest', ' '.join(args[2:])) return 0 def setvarCmd(ctx, args): if (len(args) < 4): print "usage: setvar [vmname|uuid] expr value" return 0 mach = argsToMach(ctx,args) if mach == None: return 0 session = ctx['mgr'].getSessionObject(vbox) vbox.openSession(session, mach.id) mach = session.machine expr = 'mach.'+args[2]+' = '+args[3] print "Executing",expr try: exec expr except Exception, e: print 'failed: ',e if g_verbose: traceback.print_exc() mach.saveSettings() session.close() return 0 def quitCmd(ctx, args): return 1 def aliasesCmd(ctx, args): for (k,v) in aliases.items(): print "'%s' is an alias for '%s'" %(k,v) return 0 def verboseCmd(ctx, args): global g_verbose g_verbose = not g_verbose return 0 def hostCmd(ctx, args): host = ctx['vb'].host cnt = host.processorCount print "Processor count:",cnt for i in range(0,cnt): print "Processor #%d speed: %dMHz" %(i,host.getProcessorSpeed(i)) if ctx['perf']: for metric in ctx['perf'].query(["*"], [host]): print metric['name'], metric['values_as_string'] return 0 def monitorGuestCmd(ctx, args): if (len(args) < 2): print "usage: monitorGuest name (duration)" return 0 mach = argsToMach(ctx,args) if mach == None: return 0 dur = 5 if len(args) > 2: dur = float(args[2]) cmdExistingVm(ctx, mach, 'monitorGuest', dur) return 0 def monitorVboxCmd(ctx, args): if (len(args) > 2): print "usage: monitorVbox (duration)" return 0 dur = 5 if len(args) > 1: dur = float(args[1]) monitorVbox(ctx, dur) return 0 def evalCmd(ctx, args): expr = ' '.join(args[1:]) try: exec expr except Exception, e: print 'failed: ',e if g_verbose: traceback.print_exc() return 0 aliases = {'s':'start', 'i':'info', 'l':'list', 'h':'help', 'a':'aliases', 'q':'quit', 'exit':'quit', 'v':'verbose'} commands = {'help':['Prints help information', helpCmd], 'start':['Start virtual machine by name or uuid', startCmd], 'create':['Create virtual machine', createCmd], 'remove':['Remove virtual machine', removeCmd], 'pause':['Pause virtual machine', pauseCmd], 'resume':['Resume virtual machine', resumeCmd], 'stats':['Stats for virtual machine', statsCmd], 'powerdown':['Power down virtual machine', powerdownCmd], 'list':['Shows known virtual machines', listCmd], 'info':['Shows info on machine', infoCmd], 'aliases':['Shows aliases', aliasesCmd], 'verbose':['Toggle verbosity', verboseCmd], 'setvar':['Set VMs variable: setvar Fedora BIOSSettings.ACPIEnabled True', setvarCmd], 'eval':['Evaluate arbitrary Python construction: eval for m in getMachines(ctx): print m.name,"has",m.memorySize,"M"', evalCmd], 'quit':['Exits', quitCmd], 'host':['Show host information', hostCmd], 'guest':['Execute command for guest: guest Win32 console.mouse.putMouseEvent(20, 20, 0, 0)', guestCmd], 'monitorGuest':['Monitor what happens with the guest for some time: monitorGuest Win32 10', monitorGuestCmd], 'monitorVbox':['Monitor what happens with Virtual Box for some time: monitorVbox 10', monitorVboxCmd], } def runCommand(ctx, cmd): if len(cmd) == 0: return 0 args = split_no_quotes(cmd) if len(args) == 0: return 0 c = args[0] if aliases.get(c, None) != None: c = aliases[c] ci = commands.get(c,None) if ci == None: print "Unknown command: '%s', type 'help' for list of known commands" %(c) return 0 return ci[1](ctx, args) def interpret(ctx): vbox = ctx['vb'] print "Running VirtualBox version %s" %(vbox.version) ctx['perf'] = PerfCollector(vbox) autoCompletion(commands, ctx) # to allow to print actual host information, we collect info for # last 150 secs maximum, (sample every 10 secs and keep up to 15 samples) if ctx['perf']: try: ctx['perf'].setup(['*'], [vbox.host], 10, 15) except: pass while True: try: cmd = raw_input("vbox> ") done = runCommand(ctx, cmd) if done != 0: break except KeyboardInterrupt: print '====== You can type quit or q to leave' break except EOFError: break; except Exception,e: print e if g_verbose: traceback.print_exc() try: # There is no need to disable metric collection. This is just an example. if ct['perf']: ctx['perf'].disable(['*'], [vbox.host]) except: pass from vboxapi import VirtualBoxManager def main(argv): style = None if len(argv) > 1: if argv[1] == "-w": style = "WEBSERVICE" g_virtualBoxManager = VirtualBoxManager(style, None) ctx = {'global':g_virtualBoxManager, 'mgr':g_virtualBoxManager.mgr, 'vb':g_virtualBoxManager.vbox, 'ifaces':g_virtualBoxManager.constants, 'remote':g_virtualBoxManager.remote, 'type':g_virtualBoxManager.type } interpret(ctx) g_virtualBoxManager.deinit() del g_virtualBoxManager if __name__ == '__main__': main(sys.argv)