VirtualBox

source: vbox/trunk/src/VBox/Main/glue/vboxapi.py@ 64664

Last change on this file since 64664 was 62900, checked in by vboxsync, 8 years ago

Main/glue/vboxapi.py: use VirtualBoxClient instead of directly creating a remote VirtualBox object

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 40.4 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: vboxapi.py 62900 2016-08-03 09:51:47Z vboxsync $
3"""
4VirtualBox Python API Glue.
5"""
6
7__copyright__ = \
8 """
9 Copyright (C) 2009-2016 Oracle Corporation
10
11 This file is part of VirtualBox Open Source Edition (OSE), as
12 available from http://www.virtualbox.org. This file is free software;
13 you can redistribute it and/or modify it under the terms of the GNU
14 General Public License (GPL) as published by the Free Software
15 Foundation, in version 2 as it comes in the "COPYING" file of the
16 VirtualBox OSE distribution. VirtualBox OSE is distributed in the
17 hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
18
19 The contents of this file may alternatively be used under the terms
20 of the Common Development and Distribution License Version 1.0
21 (CDDL) only, as it comes in the "COPYING.CDDL" file of the
22 VirtualBox OSE distribution, in which case the provisions of the
23 CDDL are applicable instead of those of the GPL.
24
25 You may elect to license modified versions of this file under the
26 terms and conditions of either the GPL or the CDDL or both.
27 """
28__version__ = "$Revision: 62900 $"
29
30
31# Note! To set Python bitness on OSX use 'export VERSIONER_PYTHON_PREFER_32_BIT=yes'
32
33
34# Standard Python imports.
35import os
36import sys
37import traceback
38
39
40if sys.version_info >= (3, 0):
41 xrange = range
42 long = int
43
44#
45# Globals, environment and sys.path changes.
46#
47import platform;
48VBoxBinDir = os.environ.get("VBOX_PROGRAM_PATH", None)
49VBoxSdkDir = os.environ.get("VBOX_SDK_PATH", None)
50
51if VBoxBinDir is None:
52 if platform.system() == 'Darwin':
53 VBoxBinDir = '/Applications/VirtualBox.app/Contents/MacOS'
54 else: # Will be set by the installer
55 VBoxBinDir = "%VBOX_INSTALL_PATH%"
56else:
57 VBoxBinDir = os.path.abspath(VBoxBinDir)
58
59if VBoxSdkDir is None:
60 if platform.system() == 'Darwin':
61 VBoxSdkDir = '/Applications/VirtualBox.app/Contents/MacOS/sdk'
62 else: # Will be set by the installer
63 VBoxSdkDir = "%VBOX_SDK_PATH%"
64else:
65 VBoxSdkDir = os.path.abspath(VBoxSdkDir)
66
67os.environ["VBOX_PROGRAM_PATH"] = VBoxBinDir
68os.environ["VBOX_SDK_PATH"] = VBoxSdkDir
69sys.path.append(VBoxBinDir)
70
71
72#
73# Import the generated VirtualBox constants.
74#
75from .VirtualBox_constants import VirtualBoxReflectionInfo
76
77
78class PerfCollector(object):
79 """ This class provides a wrapper over IPerformanceCollector in order to
80 get more 'pythonic' interface.
81
82 To begin collection of metrics use setup() method.
83
84 To get collected data use query() method.
85
86 It is possible to disable metric collection without changing collection
87 parameters with disable() method. The enable() method resumes metric
88 collection.
89 """
90
91 def __init__(self, mgr, vbox):
92 """ Initializes the instance.
93
94 """
95 self.mgr = mgr
96 self.isMscom = (mgr.type == 'MSCOM')
97 self.collector = vbox.performanceCollector
98
99 def setup(self, names, objects, period, nsamples):
100 """ Discards all previously collected values for the specified
101 metrics, sets the period of collection and the number of retained
102 samples, enables collection.
103 """
104 self.collector.setupMetrics(names, objects, period, nsamples)
105
106 def enable(self, names, objects):
107 """ Resumes metric collection for the specified metrics.
108 """
109 self.collector.enableMetrics(names, objects)
110
111 def disable(self, names, objects):
112 """ Suspends metric collection for the specified metrics.
113 """
114 self.collector.disableMetrics(names, objects)
115
116 def query(self, names, objects):
117 """ Retrieves collected metric values as well as some auxiliary
118 information. Returns an array of dictionaries, one dictionary per
119 metric. Each dictionary contains the following entries:
120 'name': metric name
121 'object': managed object this metric associated with
122 'unit': unit of measurement
123 'scale': divide 'values' by this number to get float numbers
124 'values': collected data
125 'values_as_string': pre-processed values ready for 'print' statement
126 """
127 # Get around the problem with input arrays returned in output
128 # parameters (see #3953) for MSCOM.
129 if self.isMscom:
130 (values, names, objects, names_out, objects_out, units, scales, sequence_numbers,
131 indices, lengths) = self.collector.queryMetricsData(names, objects)
132 else:
133 (values, names_out, objects_out, units, scales, sequence_numbers,
134 indices, lengths) = self.collector.queryMetricsData(names, objects)
135 out = []
136 for i in xrange(0, len(names_out)):
137 scale = int(scales[i])
138 if scale != 1:
139 fmt = '%.2f%s'
140 else:
141 fmt = '%d %s'
142 out.append({
143 'name': str(names_out[i]),
144 'object': str(objects_out[i]),
145 'unit': str(units[i]),
146 'scale': scale,
147 'values': [int(values[j]) for j in xrange(int(indices[i]), int(indices[i]) + int(lengths[i]))],
148 'values_as_string': '[' + ', '.join([fmt % (int(values[j]) / scale, units[i]) for j in
149 xrange(int(indices[i]), int(indices[i]) + int(lengths[i]))]) + ']'
150 })
151 return out
152
153
154#
155# Attribute hacks.
156#
157def ComifyName(name):
158 return name[0].capitalize() + name[1:]
159
160
161## This is for saving the original DispatchBaseClass __getattr__ and __setattr__
162# method references.
163_g_dCOMForward = {
164 'getattr': None,
165 'setattr': None,
166}
167
168
169def _CustomGetAttr(self, sAttr):
170 """ Our getattr replacement for DispatchBaseClass. """
171 # Fastpath.
172 oRet = self.__class__.__dict__.get(sAttr)
173 if oRet is not None:
174 return oRet
175
176 # Try case-insensitivity workaround for class attributes (COM methods).
177 sAttrLower = sAttr.lower()
178 for k in list(self.__class__.__dict__.keys()):
179 if k.lower() == sAttrLower:
180 setattr(self.__class__, sAttr, self.__class__.__dict__[k])
181 return getattr(self, k)
182
183 # Slow path.
184 try:
185 return _g_dCOMForward['getattr'](self, ComifyName(sAttr))
186 except AttributeError:
187 return _g_dCOMForward['getattr'](self, sAttr)
188
189
190def _CustomSetAttr(self, sAttr, oValue):
191 """ Our setattr replacement for DispatchBaseClass. """
192 try:
193 return _g_dCOMForward['setattr'](self, ComifyName(sAttr), oValue)
194 except AttributeError:
195 return _g_dCOMForward['setattr'](self, sAttr, oValue)
196
197
198class PlatformBase(object):
199 """
200 Base class for the platform specific code.
201 """
202
203 def __init__(self, aoParams):
204 _ = aoParams
205
206 def getVirtualBox(self):
207 """
208 Gets a the IVirtualBox singleton.
209 """
210 return None
211
212 def getSessionObject(self, oIVBox):
213 """
214 Get a session object that can be used for opening machine sessions.
215
216 The oIVBox parameter is an getVirtualBox() return value, i.e. an
217 IVirtualBox reference.
218
219 See also openMachineSession.
220 """
221 _ = oIVBox
222 return None
223
224 def getType(self):
225 """ Returns the platform type (class name sans 'Platform'). """
226 return None
227
228 def isRemote(self):
229 """
230 Returns True if remote (web services) and False if local (COM/XPCOM).
231 """
232 return False
233
234 def getArray(self, oInterface, sAttrib):
235 """
236 Retrives the value of the array attribute 'sAttrib' from
237 interface 'oInterface'.
238
239 This is for hiding platform specific differences in attributes
240 returning arrays.
241 """
242 _ = oInterface
243 _ = sAttrib
244 return None
245
246 def setArray(self, oInterface, sAttrib, aoArray):
247 """
248 Sets the value (aoArray) of the array attribute 'sAttrib' in
249 interface 'oInterface'.
250
251 This is for hiding platform specific differences in attributes
252 setting arrays.
253 """
254 _ = oInterface
255 _ = sAttrib
256 _ = aoArray
257 return None
258
259 def initPerThread(self):
260 """
261 Does backend specific initialization for the calling thread.
262 """
263 return True
264
265 def deinitPerThread(self):
266 """
267 Does backend specific uninitialization for the calling thread.
268 """
269 return True
270
271 def createListener(self, oImplClass, dArgs):
272 """
273 Instantiates and wraps an active event listener class so it can be
274 passed to an event source for registration.
275
276 oImplClass is a class (type, not instance) which implements
277 IEventListener.
278
279 dArgs is a dictionary with string indexed variables. This may be
280 modified by the method to pass platform specific parameters. Can
281 be None.
282
283 This currently only works on XPCOM. COM support is not possible due to
284 shortcuts taken in the COM bridge code, which is not under our control.
285 Use passive listeners for COM and web services.
286 """
287 _ = oImplClass
288 _ = dArgs
289 raise Exception("No active listeners for this platform")
290
291 def waitForEvents(self, cMsTimeout):
292 """
293 Wait for events to arrive and process them.
294
295 The timeout (cMsTimeout) is in milliseconds for how long to wait for
296 events to arrive. A negative value means waiting for ever, while 0
297 does not wait at all.
298
299 Returns 0 if events was processed.
300 Returns 1 if timed out or interrupted in some way.
301 Returns 2 on error (like not supported for web services).
302
303 Raises an exception if the calling thread is not the main thread (the one
304 that initialized VirtualBoxManager) or if the time isn't an integer.
305 """
306 _ = cMsTimeout
307 return 2
308
309 def interruptWaitEvents(self):
310 """
311 Interrupt a waitForEvents call.
312 This is normally called from a worker thread to wake up the main thread.
313
314 Returns True on success, False on failure.
315 """
316 return False
317
318 def deinit(self):
319 """
320 Unitializes the platform specific backend.
321 """
322 return None
323
324 def queryInterface(self, oIUnknown, sClassName):
325 """
326 IUnknown::QueryInterface wrapper.
327
328 oIUnknown is who to ask.
329 sClassName is the name of the interface we're asking for.
330 """
331 return None
332
333 #
334 # Error (exception) access methods.
335 #
336
337 def xcptGetStatus(self, oXcpt):
338 """
339 Returns the COM status code from the VBox API given exception.
340 """
341 return None
342
343 def xcptIsDeadInterface(self, oXcpt):
344 """
345 Returns True if the exception indicates that the interface is dead, False if not.
346 """
347 return False
348
349 def xcptIsEqual(self, oXcpt, hrStatus):
350 """
351 Checks if the exception oXcpt is equal to the COM/XPCOM status code
352 hrStatus.
353
354 The oXcpt parameter can be any kind of object, we'll just return True
355 if it doesn't behave like a our exception class.
356
357 Will not raise any exception as long as hrStatus and self are not bad.
358 """
359 try:
360 hrXcpt = self.xcptGetStatus(oXcpt)
361 except AttributeError:
362 return False
363 if hrXcpt == hrStatus:
364 return True
365
366 # Fudge for 32-bit signed int conversion.
367 if 0x7fffffff < hrStatus <= 0xffffffff and hrXcpt < 0:
368 if (hrStatus - 0x100000000) == hrXcpt:
369 return True
370 return False
371
372 def xcptGetMessage(self, oXcpt):
373 """
374 Returns the best error message found in the COM-like exception.
375 Returns None to fall back on xcptToString.
376 Raises exception if oXcpt isn't our kind of exception object.
377 """
378 return None
379
380 def xcptGetBaseXcpt(self):
381 """
382 Returns the base exception class.
383 """
384 return None
385
386 def xcptSetupConstants(self, oDst):
387 """
388 Copy/whatever all error constants onto oDst.
389 """
390 return oDst
391
392 @staticmethod
393 def xcptCopyErrorConstants(oDst, oSrc):
394 """
395 Copy everything that looks like error constants from oDst to oSrc.
396 """
397 for sAttr in dir(oSrc):
398 if sAttr[0].isupper() and (sAttr[1].isupper() or sAttr[1] == '_'):
399 oAttr = getattr(oSrc, sAttr)
400 if type(oAttr) is int:
401 setattr(oDst, sAttr, oAttr)
402 return oDst
403
404
405class PlatformMSCOM(PlatformBase):
406 """
407 Platform specific code for MS COM.
408 """
409
410 ## @name VirtualBox COM Typelib definitions (should be generate)
411 #
412 # @remarks Must be updated when the corresponding VirtualBox.xidl bits
413 # are changed. Fortunately this isn't very often.
414 # @{
415 VBOX_TLB_GUID = '{D7569351-1750-46F0-936E-BD127D5BC264}'
416 VBOX_TLB_LCID = 0
417 VBOX_TLB_MAJOR = 1
418 VBOX_TLB_MINOR = 3
419 ## @}
420
421 def __init__(self, dParams):
422 PlatformBase.__init__(self, dParams)
423
424 #
425 # Since the code runs on all platforms, we have to do a lot of
426 # importing here instead of at the top of the file where it's normally located.
427 #
428 from win32com import universal
429 from win32com.client import gencache, DispatchBaseClass
430 from win32com.client import constants, getevents
431 import win32com
432 import pythoncom
433 import win32api
434 import winerror
435 from win32con import DUPLICATE_SAME_ACCESS
436 from win32api import GetCurrentThread, GetCurrentThreadId, DuplicateHandle, GetCurrentProcess
437 import threading
438
439 self.winerror = winerror
440
441 pid = GetCurrentProcess()
442 self.tid = GetCurrentThreadId()
443 handle = DuplicateHandle(pid, GetCurrentThread(), pid, 0, 0, DUPLICATE_SAME_ACCESS)
444 self.handles = []
445 self.handles.append(handle)
446
447 # Hack the COM dispatcher base class so we can modify method and
448 # attribute names to match those in xpcom.
449 if _g_dCOMForward['setattr'] is None:
450 _g_dCOMForward['getattr'] = DispatchBaseClass.__dict__['__getattr__']
451 _g_dCOMForward['setattr'] = DispatchBaseClass.__dict__['__setattr__']
452 setattr(DispatchBaseClass, '__getattr__', _CustomGetAttr)
453 setattr(DispatchBaseClass, '__setattr__', _CustomSetAttr)
454
455 # Hack the exception base class so the users doesn't need to check for
456 # XPCOM or COM and do different things.
457 ## @todo
458
459 #
460 # Make sure the gencache is correct (we don't quite follow the COM
461 # versioning rules).
462 #
463 self.flushGenPyCache(win32com.client.gencache)
464 win32com.client.gencache.EnsureDispatch('VirtualBox.Session')
465 win32com.client.gencache.EnsureDispatch('VirtualBox.VirtualBox')
466 win32com.client.gencache.EnsureDispatch('VirtualBox.VirtualBoxClient')
467
468 self.oIntCv = threading.Condition()
469 self.fInterrupted = False
470
471 _ = dParams
472
473 def flushGenPyCache(self, oGenCache):
474 """
475 Flushes VBox related files in the win32com gen_py cache.
476
477 This is necessary since we don't follow the typelib versioning rules
478 that everyeone else seems to subscribe to.
479 """
480 #
481 # The EnsureModule method have broken validation code, it doesn't take
482 # typelib module directories into account. So we brute force them here.
483 # (It's possible the directory approach is from some older pywin
484 # version or the result of runnig makepy or gencache manually, but we
485 # need to cover it as well.)
486 #
487 sName = oGenCache.GetGeneratedFileName(self.VBOX_TLB_GUID, self.VBOX_TLB_LCID,
488 self.VBOX_TLB_MAJOR, self.VBOX_TLB_MINOR)
489 sGenPath = oGenCache.GetGeneratePath()
490 if len(sName) > 36 and len(sGenPath) > 5:
491 sTypelibPath = os.path.join(sGenPath, sName)
492 if os.path.isdir(sTypelibPath):
493 import shutil
494 shutil.rmtree(sTypelibPath, ignore_errors=True)
495
496 #
497 # Ensure that our typelib is valid.
498 #
499 return oGenCache.EnsureModule(self.VBOX_TLB_GUID, self.VBOX_TLB_LCID, self.VBOX_TLB_MAJOR, self.VBOX_TLB_MINOR)
500
501 def getSessionObject(self, oIVBox):
502 _ = oIVBox
503 import win32com
504 from win32com.client import Dispatch
505 return win32com.client.Dispatch("VirtualBox.Session")
506
507 def getVirtualBox(self):
508 import win32com
509 from win32com.client import Dispatch
510 client = win32com.client.Dispatch("VirtualBox.VirtualBoxClient")
511 return client.virtualBox
512
513 def getType(self):
514 return 'MSCOM'
515
516 def getArray(self, oInterface, sAttrib):
517 return oInterface.__getattr__(sAttrib)
518
519 def setArray(self, oInterface, sAttrib, aoArray):
520 #
521 # HACK ALERT!
522 #
523 # With pywin32 build 218, we're seeing type mismatch errors here for
524 # IGuestSession::environmentChanges (safearray of BSTRs). The Dispatch
525 # object (_oleobj_) seems to get some type conversion wrong and COM
526 # gets upset. So, we redo some of the dispatcher work here, picking
527 # the missing type information from the getter.
528 #
529 oOleObj = getattr(oInterface, '_oleobj_');
530 aPropMapGet = getattr(oInterface, '_prop_map_get_');
531 aPropMapPut = getattr(oInterface, '_prop_map_put_');
532 sComAttrib = sAttrib if sAttrib in aPropMapGet else ComifyName(sAttrib);
533 try:
534 aArgs, aDefaultArgs = aPropMapPut[sComAttrib];
535 aGetArgs = aPropMapGet[sComAttrib];
536 except KeyError: # fallback.
537 return oInterface.__setattr__(sAttrib, aoArray);
538
539 import pythoncom;
540 oOleObj.InvokeTypes(aArgs[0], # dispid
541 aArgs[1], # LCID
542 aArgs[2], # DISPATCH_PROPERTYPUT
543 (pythoncom.VT_HRESULT, 0), # retType - or void?
544 (aGetArgs[2],), # argTypes - trick: we get the type from the getter.
545 aoArray,); # The array
546
547 def initPerThread(self):
548 import pythoncom
549 pythoncom.CoInitializeEx(0)
550
551 def deinitPerThread(self):
552 import pythoncom
553 pythoncom.CoUninitialize()
554
555 def createListener(self, oImplClass, dArgs):
556 if True:
557 raise Exception('no active listeners on Windows as PyGatewayBase::QueryInterface() '
558 'returns new gateway objects all the time, thus breaking EventQueue '
559 'assumptions about the listener interface pointer being constants between calls ')
560 # Did this code ever really work?
561 d = {}
562 d['BaseClass'] = oImplClass
563 d['dArgs'] = dArgs
564 d['tlb_guid'] = PlatformMSCOM.VBOX_TLB_GUID
565 d['tlb_major'] = PlatformMSCOM.VBOX_TLB_MAJOR
566 d['tlb_minor'] = PlatformMSCOM.VBOX_TLB_MINOR
567 str_ = ""
568 str_ += "import win32com.server.util\n"
569 str_ += "import pythoncom\n"
570
571 str_ += "class ListenerImpl(BaseClass):\n"
572 str_ += " _com_interfaces_ = ['IEventListener']\n"
573 str_ += " _typelib_guid_ = tlb_guid\n"
574 str_ += " _typelib_version_ = tlb_major, tlb_minor\n"
575 str_ += " _reg_clsctx_ = pythoncom.CLSCTX_INPROC_SERVER\n"
576 # Maybe we'd better implement Dynamic invoke policy, to be more flexible here
577 str_ += " _reg_policy_spec_ = 'win32com.server.policy.EventHandlerPolicy'\n"
578
579 # capitalized version of listener method
580 str_ += " HandleEvent=BaseClass.handleEvent\n"
581 str_ += " def __init__(self): BaseClass.__init__(self, dArgs)\n"
582 str_ += "result = win32com.server.util.wrap(ListenerImpl())\n"
583 exec(str_, d, d)
584 return d['result']
585
586 def waitForEvents(self, timeout):
587 from win32api import GetCurrentThreadId
588 from win32event import INFINITE
589 from win32event import MsgWaitForMultipleObjects, \
590 QS_ALLINPUT, WAIT_TIMEOUT, WAIT_OBJECT_0
591 from pythoncom import PumpWaitingMessages
592 import types
593
594 if not isinstance(timeout, int):
595 raise TypeError("The timeout argument is not an integer")
596 if self.tid != GetCurrentThreadId():
597 raise Exception("wait for events from the same thread you inited!")
598
599 if timeout < 0:
600 cMsTimeout = INFINITE
601 else:
602 cMsTimeout = timeout
603 rc = MsgWaitForMultipleObjects(self.handles, 0, cMsTimeout, QS_ALLINPUT)
604 if WAIT_OBJECT_0 <= rc < WAIT_OBJECT_0 + len(self.handles):
605 # is it possible?
606 rc = 2
607 elif rc == WAIT_OBJECT_0 + len(self.handles):
608 # Waiting messages
609 PumpWaitingMessages()
610 rc = 0
611 else:
612 # Timeout
613 rc = 1
614
615 # check for interruption
616 self.oIntCv.acquire()
617 if self.fInterrupted:
618 self.fInterrupted = False
619 rc = 1
620 self.oIntCv.release()
621
622 return rc
623
624 def interruptWaitEvents(self):
625 """
626 Basically a python implementation of NativeEventQueue::postEvent().
627
628 The magic value must be in sync with the C++ implementation or this
629 won't work.
630
631 Note that because of this method we cannot easily make use of a
632 non-visible Window to handle the message like we would like to do.
633 """
634 from win32api import PostThreadMessage
635 from win32con import WM_USER
636
637 self.oIntCv.acquire()
638 self.fInterrupted = True
639 self.oIntCv.release()
640 try:
641 PostThreadMessage(self.tid, WM_USER, None, 0xf241b819)
642 except:
643 return False
644 return True
645
646 def deinit(self):
647 import pythoncom
648 from win32file import CloseHandle
649
650 for h in self.handles:
651 if h is not None:
652 CloseHandle(h)
653 self.handles = None
654 pythoncom.CoUninitialize()
655 pass
656
657 def queryInterface(self, oIUnknown, sClassName):
658 from win32com.client import CastTo
659 return CastTo(oIUnknown, sClassName)
660
661 def xcptGetStatus(self, oXcpt):
662 # The DISP_E_EXCEPTION + excptinfo fun needs checking up, only
663 # empirical info on it so far.
664 hrXcpt = oXcpt.hresult
665 if hrXcpt == self.winerror.DISP_E_EXCEPTION:
666 try:
667 hrXcpt = oXcpt.excepinfo[5]
668 except:
669 pass
670 return hrXcpt
671
672 def xcptIsDeadInterface(self, oXcpt):
673 return self.xcptGetStatus(oXcpt) in [
674 0x800706ba, -2147023174, # RPC_S_SERVER_UNAVAILABLE.
675 0x800706be, -2147023170, # RPC_S_CALL_FAILED.
676 0x800706bf, -2147023169, # RPC_S_CALL_FAILED_DNE.
677 0x80010108, -2147417848, # RPC_E_DISCONNECTED.
678 0x800706b5, -2147023179, # RPC_S_UNKNOWN_IF
679 ]
680
681 def xcptGetMessage(self, oXcpt):
682 if hasattr(oXcpt, 'excepinfo'):
683 try:
684 if len(oXcpt.excepinfo) >= 3:
685 sRet = oXcpt.excepinfo[2]
686 if len(sRet) > 0:
687 return sRet[0:]
688 except:
689 pass
690 if hasattr(oXcpt, 'strerror'):
691 try:
692 sRet = oXcpt.strerror
693 if len(sRet) > 0:
694 return sRet
695 except:
696 pass
697 return None
698
699 def xcptGetBaseXcpt(self):
700 import pythoncom
701
702 return pythoncom.com_error
703
704 def xcptSetupConstants(self, oDst):
705 import winerror
706
707 oDst = self.xcptCopyErrorConstants(oDst, winerror)
708
709 # XPCOM compatability constants.
710 oDst.NS_OK = oDst.S_OK
711 oDst.NS_ERROR_FAILURE = oDst.E_FAIL
712 oDst.NS_ERROR_ABORT = oDst.E_ABORT
713 oDst.NS_ERROR_NULL_POINTER = oDst.E_POINTER
714 oDst.NS_ERROR_NO_INTERFACE = oDst.E_NOINTERFACE
715 oDst.NS_ERROR_INVALID_ARG = oDst.E_INVALIDARG
716 oDst.NS_ERROR_OUT_OF_MEMORY = oDst.E_OUTOFMEMORY
717 oDst.NS_ERROR_NOT_IMPLEMENTED = oDst.E_NOTIMPL
718 oDst.NS_ERROR_UNEXPECTED = oDst.E_UNEXPECTED
719 return oDst
720
721
722class PlatformXPCOM(PlatformBase):
723 """
724 Platform specific code for XPCOM.
725 """
726
727 def __init__(self, dParams):
728 PlatformBase.__init__(self, dParams)
729 sys.path.append(VBoxSdkDir + '/bindings/xpcom/python/')
730 import xpcom.vboxxpcom
731 import xpcom
732 import xpcom.components
733 _ = dParams
734
735 def getSessionObject(self, oIVBox):
736 _ = oIVBox
737 import xpcom.components
738 return xpcom.components.classes["@virtualbox.org/Session;1"].createInstance()
739
740 def getVirtualBox(self):
741 import xpcom.components
742 client = xpcom.components.classes["@virtualbox.org/VirtualBoxClient;1"].createInstance()
743 return client.virtualBox
744
745 def getType(self):
746 return 'XPCOM'
747
748 def getArray(self, oInterface, sAttrib):
749 return oInterface.__getattr__('get' + ComifyName(sAttrib))()
750
751 def setArray(self, oInterface, sAttrib, aoArray):
752 return oInterface.__getattr__('set' + ComifyName(sAttrib))(aoArray)
753
754 def initPerThread(self):
755 import xpcom
756 xpcom._xpcom.AttachThread()
757
758 def deinitPerThread(self):
759 import xpcom
760 xpcom._xpcom.DetachThread()
761
762 def createListener(self, oImplClass, dArgs):
763 d = {}
764 d['BaseClass'] = oImplClass
765 d['dArgs'] = dArgs
766 str = ""
767 str += "import xpcom.components\n"
768 str += "class ListenerImpl(BaseClass):\n"
769 str += " _com_interfaces_ = xpcom.components.interfaces.IEventListener\n"
770 str += " def __init__(self): BaseClass.__init__(self, dArgs)\n"
771 str += "result = ListenerImpl()\n"
772 exec(str, d, d)
773 return d['result']
774
775 def waitForEvents(self, timeout):
776 import xpcom
777 return xpcom._xpcom.WaitForEvents(timeout)
778
779 def interruptWaitEvents(self):
780 import xpcom
781 return xpcom._xpcom.InterruptWait()
782
783 def deinit(self):
784 import xpcom
785 xpcom._xpcom.DeinitCOM()
786
787 def queryInterface(self, oIUnknown, sClassName):
788 import xpcom.components
789 return oIUnknown.queryInterface(getattr(xpcom.components.interfaces, sClassName))
790
791 def xcptGetStatus(self, oXcpt):
792 return oXcpt.errno
793
794 def xcptIsDeadInterface(self, oXcpt):
795 return self.xcptGetStatus(oXcpt) in [
796 0x80004004, -2147467260, # NS_ERROR_ABORT
797 0x800706be, -2147023170, # NS_ERROR_CALL_FAILED (RPC_S_CALL_FAILED)
798 ]
799
800 def xcptGetMessage(self, oXcpt):
801 if hasattr(oXcpt, 'msg'):
802 try:
803 sRet = oXcpt.msg
804 if len(sRet) > 0:
805 return sRet
806 except:
807 pass
808 return None
809
810 def xcptGetBaseXcpt(self):
811 import xpcom
812 return xpcom.Exception
813
814 def xcptSetupConstants(self, oDst):
815 import xpcom
816 oDst = self.xcptCopyErrorConstants(oDst, xpcom.nsError)
817
818 # COM compatability constants.
819 oDst.E_ACCESSDENIED = -2147024891 # see VBox/com/defs.h
820 oDst.S_OK = oDst.NS_OK
821 oDst.E_FAIL = oDst.NS_ERROR_FAILURE
822 oDst.E_ABORT = oDst.NS_ERROR_ABORT
823 oDst.E_POINTER = oDst.NS_ERROR_NULL_POINTER
824 oDst.E_NOINTERFACE = oDst.NS_ERROR_NO_INTERFACE
825 oDst.E_INVALIDARG = oDst.NS_ERROR_INVALID_ARG
826 oDst.E_OUTOFMEMORY = oDst.NS_ERROR_OUT_OF_MEMORY
827 oDst.E_NOTIMPL = oDst.NS_ERROR_NOT_IMPLEMENTED
828 oDst.E_UNEXPECTED = oDst.NS_ERROR_UNEXPECTED
829 oDst.DISP_E_EXCEPTION = -2147352567 # For COM compatability only.
830 return oDst
831
832
833class PlatformWEBSERVICE(PlatformBase):
834 """
835 VirtualBox Web Services API specific code.
836 """
837
838 def __init__(self, dParams):
839 PlatformBase.__init__(self, dParams)
840 # Import web services stuff. Fix the sys.path the first time.
841 sWebServLib = os.path.join(VBoxSdkDir, 'bindings', 'webservice', 'python', 'lib')
842 if sWebServLib not in sys.path:
843 sys.path.append(sWebServLib)
844 import VirtualBox_wrappers
845 from VirtualBox_wrappers import IWebsessionManager2
846
847 # Initialize instance variables from parameters.
848 if dParams is not None:
849 self.user = dParams.get("user", "")
850 self.password = dParams.get("password", "")
851 self.url = dParams.get("url", "")
852 else:
853 self.user = ""
854 self.password = ""
855 self.url = None
856 self.vbox = None
857 self.wsmgr = None
858
859 #
860 # Base class overrides.
861 #
862
863 def getSessionObject(self, oIVBox):
864 return self.wsmgr.getSessionObject(oIVBox)
865
866 def getVirtualBox(self):
867 return self.connect(self.url, self.user, self.password)
868
869 def getType(self):
870 return 'WEBSERVICE'
871
872 def isRemote(self):
873 """ Returns True if remote VBox host, False if local. """
874 return True
875
876 def getArray(self, oInterface, sAttrib):
877 return oInterface.__getattr__(sAttrib)
878
879 def setArray(self, oInterface, sAttrib, aoArray):
880 return oInterface.__setattr__(sAttrib, aoArray)
881
882 def waitForEvents(self, timeout):
883 # Webservices cannot do that yet
884 return 2
885
886 def interruptWaitEvents(self, timeout):
887 # Webservices cannot do that yet
888 return False
889
890 def deinit(self):
891 try:
892 self.disconnect()
893 except:
894 pass
895
896 def queryInterface(self, oIUnknown, sClassName):
897 d = {}
898 d['oIUnknown'] = oIUnknown
899 str = ""
900 str += "from VirtualBox_wrappers import " + sClassName + "\n"
901 str += "result = " + sClassName + "(oIUnknown.mgr, oIUnknown.handle)\n"
902 # wrong, need to test if class indeed implements this interface
903 exec(str, d, d)
904 return d['result']
905
906 #
907 # Web service specific methods.
908 #
909
910 def connect(self, url, user, passwd):
911 if self.vbox is not None:
912 self.disconnect()
913 from VirtualBox_wrappers import IWebsessionManager2
914
915 if url is None:
916 url = ""
917 self.url = url
918 if user is None:
919 user = ""
920 self.user = user
921 if passwd is None:
922 passwd = ""
923 self.password = passwd
924 self.wsmgr = IWebsessionManager2(self.url)
925 self.vbox = self.wsmgr.logon(self.user, self.password)
926 if not self.vbox.handle:
927 raise Exception("cannot connect to '" + self.url + "' as '" + self.user + "'")
928 return self.vbox
929
930 def disconnect(self):
931 if self.vbox is not None and self.wsmgr is not None:
932 self.wsmgr.logoff(self.vbox)
933 self.vbox = None
934 self.wsmgr = None
935
936
937## The current (last) exception class.
938# This is reinitalized whenever VirtualBoxManager is called, so it will hold
939# the reference to the error exception class for the last platform/style that
940# was used. Most clients does talk to multiple VBox instance on different
941# platforms at the same time, so this should be sufficent for most uses and
942# be way simpler to use than VirtualBoxManager::oXcptClass.
943CurXctpClass = None
944
945
946class VirtualBoxManager(object):
947 """
948 VirtualBox API manager class.
949
950 The API users will have to instantiate this. If no parameters are given,
951 it will default to interface with the VirtualBox running on the local
952 machine. sStyle can be None (default), MSCOM, XPCOM or WEBSERVICES. Most
953 users will either be specifying None or WEBSERVICES.
954
955 The dPlatformParams is an optional dictionary for passing parameters to the
956 WEBSERVICE backend.
957 """
958
959 class Statuses(object):
960 def __init__(self):
961 pass
962
963 def __init__(self, sStyle=None, dPlatformParams=None):
964 if sStyle is None:
965 if sys.platform == 'win32':
966 sStyle = "MSCOM"
967 else:
968 sStyle = "XPCOM"
969 if sStyle == 'XPCOM':
970 self.platform = PlatformXPCOM(dPlatformParams)
971 elif sStyle == 'MSCOM':
972 self.platform = PlatformMSCOM(dPlatformParams)
973 elif sStyle == 'WEBSERVICE':
974 self.platform = PlatformWEBSERVICE(dPlatformParams)
975 else:
976 raise Exception('Unknown sStyle=%s' % (sStyle,))
977 self.style = sStyle
978 self.type = self.platform.getType()
979 self.remote = self.platform.isRemote()
980 ## VirtualBox API constants (for webservices, enums are symbolic).
981 self.constants = VirtualBoxReflectionInfo(sStyle == "WEBSERVICE")
982
983 ## Status constants.
984 self.statuses = self.platform.xcptSetupConstants(VirtualBoxManager.Statuses())
985 ## @todo Add VBOX_E_XXX to statuses? They're already in constants...
986 ## Dictionary for errToString, built on demand.
987 self._dErrorValToName = None
988
989 ## The exception class for the selected platform.
990 self.oXcptClass = self.platform.xcptGetBaseXcpt()
991 global CurXcptClass
992 CurXcptClass = self.oXcptClass
993
994 # Get the virtualbox singleton.
995 try:
996 self.vbox = self.platform.getVirtualBox()
997 except NameError:
998 print("Installation problem: check that appropriate libs in place")
999 traceback.print_exc()
1000 raise
1001 except Exception:
1002 _, e, _ = sys.exc_info()
1003 print("init exception: ", e)
1004 traceback.print_exc()
1005 if self.remote:
1006 self.vbox = None
1007 else:
1008 raise e
1009
1010 def __del__(self):
1011 self.deinit()
1012
1013 def getPythonApiRevision(self):
1014 """
1015 Returns a Python API revision number.
1016 This will be incremented when features are added to this file.
1017 """
1018 return 3
1019
1020 @property
1021 def mgr(self):
1022 """
1023 This used to be an attribute referring to a session manager class with
1024 only one method called getSessionObject. It moved into this class.
1025 """
1026 return self;
1027
1028 #
1029 # Wrappers for self.platform methods.
1030 #
1031 def getVirtualBox(self):
1032 """ See PlatformBase::getVirtualBox(). """
1033 return self.platform.getVirtualBox()
1034
1035 def getSessionObject(self, oIVBox):
1036 """ See PlatformBase::getSessionObject(). """
1037 return self.platform.getSessionObject(oIVBox)
1038
1039 def getArray(self, oInterface, sAttrib):
1040 """ See PlatformBase::getArray(). """
1041 return self.platform.getArray(oInterface, sAttrib)
1042
1043 def setArray(self, oInterface, sAttrib, aoArray):
1044 """ See PlatformBase::setArray(). """
1045 return self.platform.setArray(oInterface, sAttrib, aoArray)
1046
1047 def createListener(self, oImplClass, dArgs=None):
1048 """ See PlatformBase::createListener(). """
1049 return self.platform.createListener(oImplClass, dArgs)
1050
1051 def waitForEvents(self, cMsTimeout):
1052 """ See PlatformBase::waitForEvents(). """
1053 return self.platform.waitForEvents(cMsTimeout)
1054
1055 def interruptWaitEvents(self):
1056 """ See PlatformBase::interruptWaitEvents(). """
1057 return self.platform.interruptWaitEvents()
1058
1059 def queryInterface(self, oIUnknown, sClassName):
1060 """ See PlatformBase::queryInterface(). """
1061 return self.platform.queryInterface(oIUnknown, sClassName)
1062
1063 #
1064 # Init and uninit.
1065 #
1066 def initPerThread(self):
1067 """ See PlatformBase::deinitPerThread(). """
1068 self.platform.initPerThread()
1069
1070 def deinitPerThread(self):
1071 """ See PlatformBase::deinitPerThread(). """
1072 return self.platform.deinitPerThread()
1073
1074 def deinit(self):
1075 """
1076 For unitializing the manager.
1077 Do not access it after calling this method.
1078 """
1079 if hasattr(self, "vbox") and self.vbox is not None:
1080 del self.vbox
1081 self.vbox = None
1082 if hasattr(self, "platform") and self.platform is not None:
1083 self.platform.deinit()
1084 self.platform = None
1085 return True
1086
1087 #
1088 # Utility methods.
1089 #
1090 def openMachineSession(self, oIMachine, fPermitSharing=True):
1091 """
1092 Attempts to open the a session to the machine.
1093 Returns a session object on success.
1094 Raises exception on failure.
1095 """
1096 oSession = self.getSessionObject(self.vbox);
1097 if fPermitSharing:
1098 eType = self.constants.LockType_Shared
1099 else:
1100 eType = self.constants.LockType_Write
1101 oIMachine.lockMachine(oSession, eType)
1102 return oSession
1103
1104 def closeMachineSession(self, oSession):
1105 """
1106 Closes a session opened by openMachineSession.
1107 Ignores None parameters.
1108 """
1109 if oSession is not None:
1110 oSession.unlockMachine()
1111 return True
1112
1113 def getPerfCollector(self, oIVBox):
1114 """
1115 Returns a helper class (PerfCollector) for accessing performance
1116 collector goodies. See PerfCollector for details.
1117 """
1118 return PerfCollector(self, oIVBox)
1119
1120 def getBinDir(self):
1121 """
1122 Returns the VirtualBox binary directory.
1123 """
1124 global VBoxBinDir
1125 return VBoxBinDir
1126
1127 def getSdkDir(self):
1128 """
1129 Returns the VirtualBox SDK directory.
1130 """
1131 global VBoxSdkDir
1132 return VBoxSdkDir
1133
1134 #
1135 # Error code utilities.
1136 #
1137 ## @todo port to webservices!
1138 def xcptGetStatus(self, oXcpt=None):
1139 """
1140 Gets the status code from an exception. If the exception parameter
1141 isn't specified, the current exception is examined.
1142 """
1143 if oXcpt is None:
1144 oXcpt = sys.exc_info()[1]
1145 return self.platform.xcptGetStatus(oXcpt)
1146
1147 def xcptIsDeadInterface(self, oXcpt=None):
1148 """
1149 Returns True if the exception indicates that the interface is dead,
1150 False if not. If the exception parameter isn't specified, the current
1151 exception is examined.
1152 """
1153 if oXcpt is None:
1154 oXcpt = sys.exc_info()[1]
1155 return self.platform.xcptIsDeadInterface(oXcpt)
1156
1157 def xcptIsOurXcptKind(self, oXcpt=None):
1158 """
1159 Checks if the exception is one that could come from the VBox API. If
1160 the exception parameter isn't specified, the current exception is
1161 examined.
1162 """
1163 if self.oXcptClass is None: # @todo find the exception class for web services!
1164 return False
1165 if oXcpt is None:
1166 oXcpt = sys.exc_info()[1]
1167 return isinstance(oXcpt, self.oXcptClass)
1168
1169 def xcptIsEqual(self, oXcpt, hrStatus):
1170 """
1171 Checks if the exception oXcpt is equal to the COM/XPCOM status code
1172 hrStatus.
1173
1174 The oXcpt parameter can be any kind of object, we'll just return True
1175 if it doesn't behave like a our exception class. If it's None, we'll
1176 query the current exception and examine that.
1177
1178 Will not raise any exception as long as hrStatus and self are not bad.
1179 """
1180 if oXcpt is None:
1181 oXcpt = sys.exc_info()[1]
1182 return self.platform.xcptIsEqual(oXcpt, hrStatus)
1183
1184 def xcptIsNotEqual(self, oXcpt, hrStatus):
1185 """
1186 Negated xcptIsEqual.
1187 """
1188 return not self.xcptIsEqual(oXcpt, hrStatus)
1189
1190 def xcptToString(self, hrStatusOrXcpt=None):
1191 """
1192 Converts the specified COM status code, or the status code of the
1193 specified exception, to a C constant string. If the parameter isn't
1194 specified (is None), the current exception is examined.
1195 """
1196
1197 # Deal with exceptions.
1198 if hrStatusOrXcpt is None or self.xcptIsOurXcptKind(hrStatusOrXcpt):
1199 hrStatus = self.xcptGetStatus(hrStatusOrXcpt)
1200 else:
1201 hrStatus = hrStatusOrXcpt
1202
1203 # Build the dictionary on demand.
1204 if self._dErrorValToName is None:
1205 dErrorValToName = dict()
1206 for sKey in dir(self.statuses):
1207 if sKey[0].isupper():
1208 oValue = getattr(self.statuses, sKey)
1209 if type(oValue) is int:
1210 dErrorValToName[oValue] = sKey
1211 self._dErrorValToName = dErrorValToName
1212
1213 # Do the lookup, falling back on formatting the status number.
1214 try:
1215 sStr = self._dErrorValToName[int(hrStatus)]
1216 except KeyError:
1217 hrLong = long(hrStatus)
1218 sStr = '%#x (%d)' % (hrLong, hrLong)
1219 return sStr
1220
1221 def xcptGetMessage(self, oXcpt=None):
1222 """
1223 Returns the best error message found in the COM-like exception. If the
1224 exception parameter isn't specified, the current exception is examined.
1225 """
1226 if oXcpt is None:
1227 oXcpt = sys.exc_info()[1]
1228 sRet = self.platform.xcptGetMessage(oXcpt)
1229 if sRet is None:
1230 sRet = self.xcptToString(oXcpt)
1231 return sRet
1232
1233 # Legacy, remove in a day or two.
1234 errGetStatus = xcptGetStatus
1235 errIsDeadInterface = xcptIsDeadInterface
1236 errIsOurXcptKind = xcptIsOurXcptKind
1237 errGetMessage = xcptGetMessage
1238
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette