VirtualBox

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

Last change on this file since 66714 was 66630, checked in by vboxsync, 8 years ago

VBox/Main: ​​bugref:3300: VBoxSVC from terminal server session is not 'visible' - dispose the client before calling CoUninitialize, that should fix hang in Validation Kit.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 41.5 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: vboxapi.py 66630 2017-04-20 14:27:34Z 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: 66630 $"
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 # Setup client impersonation in COM calls.
442 try:
443 pythoncom.CoInitializeSecurity(None,
444 None,
445 None,
446 pythoncom.RPC_C_AUTHN_LEVEL_DEFAULT,
447 pythoncom.RPC_C_IMP_LEVEL_IMPERSONATE,
448 None,
449 pythoncom.EOAC_NONE,
450 None)
451 except:
452 # handle RPC_E_TOO_LATE (repeat call of CoInitializeSecurity)
453 print("Warning: CoInitializeSecurity was already called")
454
455 pid = GetCurrentProcess()
456 self.tid = GetCurrentThreadId()
457 handle = DuplicateHandle(pid, GetCurrentThread(), pid, 0, 0, DUPLICATE_SAME_ACCESS)
458 self.handles = []
459 self.handles.append(handle)
460
461 # Hack the COM dispatcher base class so we can modify method and
462 # attribute names to match those in xpcom.
463 if _g_dCOMForward['setattr'] is None:
464 _g_dCOMForward['getattr'] = DispatchBaseClass.__dict__['__getattr__']
465 _g_dCOMForward['setattr'] = DispatchBaseClass.__dict__['__setattr__']
466 setattr(DispatchBaseClass, '__getattr__', _CustomGetAttr)
467 setattr(DispatchBaseClass, '__setattr__', _CustomSetAttr)
468
469 # Hack the exception base class so the users doesn't need to check for
470 # XPCOM or COM and do different things.
471 ## @todo
472
473 #
474 # Make sure the gencache is correct (we don't quite follow the COM
475 # versioning rules).
476 #
477 self.flushGenPyCache(win32com.client.gencache)
478 win32com.client.gencache.EnsureDispatch('VirtualBox.Session')
479 win32com.client.gencache.EnsureDispatch('VirtualBox.VirtualBox')
480 win32com.client.gencache.EnsureDispatch('VirtualBox.VirtualBoxClient')
481
482 self.oClient = None ##< instance of client used to support lifetime of VBoxSDS
483 self.oIntCv = threading.Condition()
484 self.fInterrupted = False
485
486 _ = dParams
487
488 def flushGenPyCache(self, oGenCache):
489 """
490 Flushes VBox related files in the win32com gen_py cache.
491
492 This is necessary since we don't follow the typelib versioning rules
493 that everyeone else seems to subscribe to.
494 """
495 #
496 # The EnsureModule method have broken validation code, it doesn't take
497 # typelib module directories into account. So we brute force them here.
498 # (It's possible the directory approach is from some older pywin
499 # version or the result of runnig makepy or gencache manually, but we
500 # need to cover it as well.)
501 #
502 sName = oGenCache.GetGeneratedFileName(self.VBOX_TLB_GUID, self.VBOX_TLB_LCID,
503 self.VBOX_TLB_MAJOR, self.VBOX_TLB_MINOR)
504 sGenPath = oGenCache.GetGeneratePath()
505 if len(sName) > 36 and len(sGenPath) > 5:
506 sTypelibPath = os.path.join(sGenPath, sName)
507 if os.path.isdir(sTypelibPath):
508 import shutil
509 shutil.rmtree(sTypelibPath, ignore_errors=True)
510
511 #
512 # Ensure that our typelib is valid.
513 #
514 return oGenCache.EnsureModule(self.VBOX_TLB_GUID, self.VBOX_TLB_LCID, self.VBOX_TLB_MAJOR, self.VBOX_TLB_MINOR)
515
516 def getSessionObject(self, oIVBox):
517 _ = oIVBox
518 import win32com
519 from win32com.client import Dispatch
520 return win32com.client.Dispatch("VirtualBox.Session")
521
522 def getVirtualBox(self):
523 # Caching self.oClient is the trick for SDS. It allows to keep the
524 # VBoxSDS in the memory until the end of PlatformMSCOM lifetme.
525 if self.oClient is None:
526 import win32com
527 from win32com.client import Dispatch
528 self.oClient = win32com.client.Dispatch("VirtualBox.VirtualBoxClient")
529 return self.oClient.virtualBox
530
531 def getType(self):
532 return 'MSCOM'
533
534 def getArray(self, oInterface, sAttrib):
535 return oInterface.__getattr__(sAttrib)
536
537 def setArray(self, oInterface, sAttrib, aoArray):
538 #
539 # HACK ALERT!
540 #
541 # With pywin32 build 218, we're seeing type mismatch errors here for
542 # IGuestSession::environmentChanges (safearray of BSTRs). The Dispatch
543 # object (_oleobj_) seems to get some type conversion wrong and COM
544 # gets upset. So, we redo some of the dispatcher work here, picking
545 # the missing type information from the getter.
546 #
547 oOleObj = getattr(oInterface, '_oleobj_');
548 aPropMapGet = getattr(oInterface, '_prop_map_get_');
549 aPropMapPut = getattr(oInterface, '_prop_map_put_');
550 sComAttrib = sAttrib if sAttrib in aPropMapGet else ComifyName(sAttrib);
551 try:
552 aArgs, aDefaultArgs = aPropMapPut[sComAttrib];
553 aGetArgs = aPropMapGet[sComAttrib];
554 except KeyError: # fallback.
555 return oInterface.__setattr__(sAttrib, aoArray);
556
557 import pythoncom;
558 oOleObj.InvokeTypes(aArgs[0], # dispid
559 aArgs[1], # LCID
560 aArgs[2], # DISPATCH_PROPERTYPUT
561 (pythoncom.VT_HRESULT, 0), # retType - or void?
562 (aGetArgs[2],), # argTypes - trick: we get the type from the getter.
563 aoArray,); # The array
564
565 def initPerThread(self):
566 import pythoncom
567 pythoncom.CoInitializeEx(0)
568
569 def deinitPerThread(self):
570 import pythoncom
571 pythoncom.CoUninitialize()
572
573 def createListener(self, oImplClass, dArgs):
574 if True:
575 raise Exception('no active listeners on Windows as PyGatewayBase::QueryInterface() '
576 'returns new gateway objects all the time, thus breaking EventQueue '
577 'assumptions about the listener interface pointer being constants between calls ')
578 # Did this code ever really work?
579 d = {}
580 d['BaseClass'] = oImplClass
581 d['dArgs'] = dArgs
582 d['tlb_guid'] = PlatformMSCOM.VBOX_TLB_GUID
583 d['tlb_major'] = PlatformMSCOM.VBOX_TLB_MAJOR
584 d['tlb_minor'] = PlatformMSCOM.VBOX_TLB_MINOR
585 str_ = ""
586 str_ += "import win32com.server.util\n"
587 str_ += "import pythoncom\n"
588
589 str_ += "class ListenerImpl(BaseClass):\n"
590 str_ += " _com_interfaces_ = ['IEventListener']\n"
591 str_ += " _typelib_guid_ = tlb_guid\n"
592 str_ += " _typelib_version_ = tlb_major, tlb_minor\n"
593 str_ += " _reg_clsctx_ = pythoncom.CLSCTX_INPROC_SERVER\n"
594 # Maybe we'd better implement Dynamic invoke policy, to be more flexible here
595 str_ += " _reg_policy_spec_ = 'win32com.server.policy.EventHandlerPolicy'\n"
596
597 # capitalized version of listener method
598 str_ += " HandleEvent=BaseClass.handleEvent\n"
599 str_ += " def __init__(self): BaseClass.__init__(self, dArgs)\n"
600 str_ += "result = win32com.server.util.wrap(ListenerImpl())\n"
601 exec(str_, d, d)
602 return d['result']
603
604 def waitForEvents(self, timeout):
605 from win32api import GetCurrentThreadId
606 from win32event import INFINITE
607 from win32event import MsgWaitForMultipleObjects, \
608 QS_ALLINPUT, WAIT_TIMEOUT, WAIT_OBJECT_0
609 from pythoncom import PumpWaitingMessages
610 import types
611
612 if not isinstance(timeout, int):
613 raise TypeError("The timeout argument is not an integer")
614 if self.tid != GetCurrentThreadId():
615 raise Exception("wait for events from the same thread you inited!")
616
617 if timeout < 0:
618 cMsTimeout = INFINITE
619 else:
620 cMsTimeout = timeout
621 rc = MsgWaitForMultipleObjects(self.handles, 0, cMsTimeout, QS_ALLINPUT)
622 if WAIT_OBJECT_0 <= rc < WAIT_OBJECT_0 + len(self.handles):
623 # is it possible?
624 rc = 2
625 elif rc == WAIT_OBJECT_0 + len(self.handles):
626 # Waiting messages
627 PumpWaitingMessages()
628 rc = 0
629 else:
630 # Timeout
631 rc = 1
632
633 # check for interruption
634 self.oIntCv.acquire()
635 if self.fInterrupted:
636 self.fInterrupted = False
637 rc = 1
638 self.oIntCv.release()
639
640 return rc
641
642 def interruptWaitEvents(self):
643 """
644 Basically a python implementation of NativeEventQueue::postEvent().
645
646 The magic value must be in sync with the C++ implementation or this
647 won't work.
648
649 Note that because of this method we cannot easily make use of a
650 non-visible Window to handle the message like we would like to do.
651 """
652 from win32api import PostThreadMessage
653 from win32con import WM_USER
654
655 self.oIntCv.acquire()
656 self.fInterrupted = True
657 self.oIntCv.release()
658 try:
659 PostThreadMessage(self.tid, WM_USER, None, 0xf241b819)
660 except:
661 return False
662 return True
663
664 def deinit(self):
665 import pythoncom
666 from win32file import CloseHandle
667
668 for h in self.handles:
669 if h is not None:
670 CloseHandle(h)
671 self.handles = None
672 del self.oClient
673 oClient = None
674 pythoncom.CoUninitialize()
675 pass
676
677 def queryInterface(self, oIUnknown, sClassName):
678 from win32com.client import CastTo
679 return CastTo(oIUnknown, sClassName)
680
681 def xcptGetStatus(self, oXcpt):
682 # The DISP_E_EXCEPTION + excptinfo fun needs checking up, only
683 # empirical info on it so far.
684 hrXcpt = oXcpt.hresult
685 if hrXcpt == self.winerror.DISP_E_EXCEPTION:
686 try:
687 hrXcpt = oXcpt.excepinfo[5]
688 except:
689 pass
690 return hrXcpt
691
692 def xcptIsDeadInterface(self, oXcpt):
693 return self.xcptGetStatus(oXcpt) in [
694 0x800706ba, -2147023174, # RPC_S_SERVER_UNAVAILABLE.
695 0x800706be, -2147023170, # RPC_S_CALL_FAILED.
696 0x800706bf, -2147023169, # RPC_S_CALL_FAILED_DNE.
697 0x80010108, -2147417848, # RPC_E_DISCONNECTED.
698 0x800706b5, -2147023179, # RPC_S_UNKNOWN_IF
699 ]
700
701 def xcptGetMessage(self, oXcpt):
702 if hasattr(oXcpt, 'excepinfo'):
703 try:
704 if len(oXcpt.excepinfo) >= 3:
705 sRet = oXcpt.excepinfo[2]
706 if len(sRet) > 0:
707 return sRet[0:]
708 except:
709 pass
710 if hasattr(oXcpt, 'strerror'):
711 try:
712 sRet = oXcpt.strerror
713 if len(sRet) > 0:
714 return sRet
715 except:
716 pass
717 return None
718
719 def xcptGetBaseXcpt(self):
720 import pythoncom
721
722 return pythoncom.com_error
723
724 def xcptSetupConstants(self, oDst):
725 import winerror
726
727 oDst = self.xcptCopyErrorConstants(oDst, winerror)
728
729 # XPCOM compatability constants.
730 oDst.NS_OK = oDst.S_OK
731 oDst.NS_ERROR_FAILURE = oDst.E_FAIL
732 oDst.NS_ERROR_ABORT = oDst.E_ABORT
733 oDst.NS_ERROR_NULL_POINTER = oDst.E_POINTER
734 oDst.NS_ERROR_NO_INTERFACE = oDst.E_NOINTERFACE
735 oDst.NS_ERROR_INVALID_ARG = oDst.E_INVALIDARG
736 oDst.NS_ERROR_OUT_OF_MEMORY = oDst.E_OUTOFMEMORY
737 oDst.NS_ERROR_NOT_IMPLEMENTED = oDst.E_NOTIMPL
738 oDst.NS_ERROR_UNEXPECTED = oDst.E_UNEXPECTED
739 return oDst
740
741
742class PlatformXPCOM(PlatformBase):
743 """
744 Platform specific code for XPCOM.
745 """
746
747 def __init__(self, dParams):
748 PlatformBase.__init__(self, dParams)
749 sys.path.append(VBoxSdkDir + '/bindings/xpcom/python/')
750 import xpcom.vboxxpcom
751 import xpcom
752 import xpcom.components
753 _ = dParams
754
755 def getSessionObject(self, oIVBox):
756 _ = oIVBox
757 import xpcom.components
758 return xpcom.components.classes["@virtualbox.org/Session;1"].createInstance()
759
760 def getVirtualBox(self):
761 import xpcom.components
762 client = xpcom.components.classes["@virtualbox.org/VirtualBoxClient;1"].createInstance()
763 return client.virtualBox
764
765 def getType(self):
766 return 'XPCOM'
767
768 def getArray(self, oInterface, sAttrib):
769 return oInterface.__getattr__('get' + ComifyName(sAttrib))()
770
771 def setArray(self, oInterface, sAttrib, aoArray):
772 return oInterface.__getattr__('set' + ComifyName(sAttrib))(aoArray)
773
774 def initPerThread(self):
775 import xpcom
776 xpcom._xpcom.AttachThread()
777
778 def deinitPerThread(self):
779 import xpcom
780 xpcom._xpcom.DetachThread()
781
782 def createListener(self, oImplClass, dArgs):
783 d = {}
784 d['BaseClass'] = oImplClass
785 d['dArgs'] = dArgs
786 str = ""
787 str += "import xpcom.components\n"
788 str += "class ListenerImpl(BaseClass):\n"
789 str += " _com_interfaces_ = xpcom.components.interfaces.IEventListener\n"
790 str += " def __init__(self): BaseClass.__init__(self, dArgs)\n"
791 str += "result = ListenerImpl()\n"
792 exec(str, d, d)
793 return d['result']
794
795 def waitForEvents(self, timeout):
796 import xpcom
797 return xpcom._xpcom.WaitForEvents(timeout)
798
799 def interruptWaitEvents(self):
800 import xpcom
801 return xpcom._xpcom.InterruptWait()
802
803 def deinit(self):
804 import xpcom
805 xpcom._xpcom.DeinitCOM()
806
807 def queryInterface(self, oIUnknown, sClassName):
808 import xpcom.components
809 return oIUnknown.queryInterface(getattr(xpcom.components.interfaces, sClassName))
810
811 def xcptGetStatus(self, oXcpt):
812 return oXcpt.errno
813
814 def xcptIsDeadInterface(self, oXcpt):
815 return self.xcptGetStatus(oXcpt) in [
816 0x80004004, -2147467260, # NS_ERROR_ABORT
817 0x800706be, -2147023170, # NS_ERROR_CALL_FAILED (RPC_S_CALL_FAILED)
818 ]
819
820 def xcptGetMessage(self, oXcpt):
821 if hasattr(oXcpt, 'msg'):
822 try:
823 sRet = oXcpt.msg
824 if len(sRet) > 0:
825 return sRet
826 except:
827 pass
828 return None
829
830 def xcptGetBaseXcpt(self):
831 import xpcom
832 return xpcom.Exception
833
834 def xcptSetupConstants(self, oDst):
835 import xpcom
836 oDst = self.xcptCopyErrorConstants(oDst, xpcom.nsError)
837
838 # COM compatability constants.
839 oDst.E_ACCESSDENIED = -2147024891 # see VBox/com/defs.h
840 oDst.S_OK = oDst.NS_OK
841 oDst.E_FAIL = oDst.NS_ERROR_FAILURE
842 oDst.E_ABORT = oDst.NS_ERROR_ABORT
843 oDst.E_POINTER = oDst.NS_ERROR_NULL_POINTER
844 oDst.E_NOINTERFACE = oDst.NS_ERROR_NO_INTERFACE
845 oDst.E_INVALIDARG = oDst.NS_ERROR_INVALID_ARG
846 oDst.E_OUTOFMEMORY = oDst.NS_ERROR_OUT_OF_MEMORY
847 oDst.E_NOTIMPL = oDst.NS_ERROR_NOT_IMPLEMENTED
848 oDst.E_UNEXPECTED = oDst.NS_ERROR_UNEXPECTED
849 oDst.DISP_E_EXCEPTION = -2147352567 # For COM compatability only.
850 return oDst
851
852
853class PlatformWEBSERVICE(PlatformBase):
854 """
855 VirtualBox Web Services API specific code.
856 """
857
858 def __init__(self, dParams):
859 PlatformBase.__init__(self, dParams)
860 # Import web services stuff. Fix the sys.path the first time.
861 sWebServLib = os.path.join(VBoxSdkDir, 'bindings', 'webservice', 'python', 'lib')
862 if sWebServLib not in sys.path:
863 sys.path.append(sWebServLib)
864 import VirtualBox_wrappers
865 from VirtualBox_wrappers import IWebsessionManager2
866
867 # Initialize instance variables from parameters.
868 if dParams is not None:
869 self.user = dParams.get("user", "")
870 self.password = dParams.get("password", "")
871 self.url = dParams.get("url", "")
872 else:
873 self.user = ""
874 self.password = ""
875 self.url = None
876 self.vbox = None
877 self.wsmgr = None
878
879 #
880 # Base class overrides.
881 #
882
883 def getSessionObject(self, oIVBox):
884 return self.wsmgr.getSessionObject(oIVBox)
885
886 def getVirtualBox(self):
887 return self.connect(self.url, self.user, self.password)
888
889 def getType(self):
890 return 'WEBSERVICE'
891
892 def isRemote(self):
893 """ Returns True if remote VBox host, False if local. """
894 return True
895
896 def getArray(self, oInterface, sAttrib):
897 return oInterface.__getattr__(sAttrib)
898
899 def setArray(self, oInterface, sAttrib, aoArray):
900 return oInterface.__setattr__(sAttrib, aoArray)
901
902 def waitForEvents(self, timeout):
903 # Webservices cannot do that yet
904 return 2
905
906 def interruptWaitEvents(self, timeout):
907 # Webservices cannot do that yet
908 return False
909
910 def deinit(self):
911 try:
912 self.disconnect()
913 except:
914 pass
915
916 def queryInterface(self, oIUnknown, sClassName):
917 d = {}
918 d['oIUnknown'] = oIUnknown
919 str = ""
920 str += "from VirtualBox_wrappers import " + sClassName + "\n"
921 str += "result = " + sClassName + "(oIUnknown.mgr, oIUnknown.handle)\n"
922 # wrong, need to test if class indeed implements this interface
923 exec(str, d, d)
924 return d['result']
925
926 #
927 # Web service specific methods.
928 #
929
930 def connect(self, url, user, passwd):
931 if self.vbox is not None:
932 self.disconnect()
933 from VirtualBox_wrappers import IWebsessionManager2
934
935 if url is None:
936 url = ""
937 self.url = url
938 if user is None:
939 user = ""
940 self.user = user
941 if passwd is None:
942 passwd = ""
943 self.password = passwd
944 self.wsmgr = IWebsessionManager2(self.url)
945 self.vbox = self.wsmgr.logon(self.user, self.password)
946 if not self.vbox.handle:
947 raise Exception("cannot connect to '" + self.url + "' as '" + self.user + "'")
948 return self.vbox
949
950 def disconnect(self):
951 if self.vbox is not None and self.wsmgr is not None:
952 self.wsmgr.logoff(self.vbox)
953 self.vbox = None
954 self.wsmgr = None
955
956
957## The current (last) exception class.
958# This is reinitalized whenever VirtualBoxManager is called, so it will hold
959# the reference to the error exception class for the last platform/style that
960# was used. Most clients does talk to multiple VBox instance on different
961# platforms at the same time, so this should be sufficent for most uses and
962# be way simpler to use than VirtualBoxManager::oXcptClass.
963CurXctpClass = None
964
965
966class VirtualBoxManager(object):
967 """
968 VirtualBox API manager class.
969
970 The API users will have to instantiate this. If no parameters are given,
971 it will default to interface with the VirtualBox running on the local
972 machine. sStyle can be None (default), MSCOM, XPCOM or WEBSERVICES. Most
973 users will either be specifying None or WEBSERVICES.
974
975 The dPlatformParams is an optional dictionary for passing parameters to the
976 WEBSERVICE backend.
977 """
978
979 class Statuses(object):
980 def __init__(self):
981 pass
982
983 def __init__(self, sStyle=None, dPlatformParams=None):
984 if sStyle is None:
985 if sys.platform == 'win32':
986 sStyle = "MSCOM"
987 else:
988 sStyle = "XPCOM"
989 if sStyle == 'XPCOM':
990 self.platform = PlatformXPCOM(dPlatformParams)
991 elif sStyle == 'MSCOM':
992 self.platform = PlatformMSCOM(dPlatformParams)
993 elif sStyle == 'WEBSERVICE':
994 self.platform = PlatformWEBSERVICE(dPlatformParams)
995 else:
996 raise Exception('Unknown sStyle=%s' % (sStyle,))
997 self.style = sStyle
998 self.type = self.platform.getType()
999 self.remote = self.platform.isRemote()
1000 ## VirtualBox API constants (for webservices, enums are symbolic).
1001 self.constants = VirtualBoxReflectionInfo(sStyle == "WEBSERVICE")
1002
1003 ## Status constants.
1004 self.statuses = self.platform.xcptSetupConstants(VirtualBoxManager.Statuses())
1005 ## @todo Add VBOX_E_XXX to statuses? They're already in constants...
1006 ## Dictionary for errToString, built on demand.
1007 self._dErrorValToName = None
1008
1009 ## The exception class for the selected platform.
1010 self.oXcptClass = self.platform.xcptGetBaseXcpt()
1011 global CurXcptClass
1012 CurXcptClass = self.oXcptClass
1013
1014 # Get the virtualbox singleton.
1015 try:
1016 self.vbox = self.platform.getVirtualBox()
1017 except NameError:
1018 print("Installation problem: check that appropriate libs in place")
1019 traceback.print_exc()
1020 raise
1021 except Exception:
1022 _, e, _ = sys.exc_info()
1023 print("init exception: ", e)
1024 traceback.print_exc()
1025 if self.remote:
1026 self.vbox = None
1027 else:
1028 raise e
1029
1030 def __del__(self):
1031 self.deinit()
1032
1033 def getPythonApiRevision(self):
1034 """
1035 Returns a Python API revision number.
1036 This will be incremented when features are added to this file.
1037 """
1038 return 3
1039
1040 @property
1041 def mgr(self):
1042 """
1043 This used to be an attribute referring to a session manager class with
1044 only one method called getSessionObject. It moved into this class.
1045 """
1046 return self;
1047
1048 #
1049 # Wrappers for self.platform methods.
1050 #
1051 def getVirtualBox(self):
1052 """ See PlatformBase::getVirtualBox(). """
1053 return self.platform.getVirtualBox()
1054
1055 def getSessionObject(self, oIVBox):
1056 """ See PlatformBase::getSessionObject(). """
1057 return self.platform.getSessionObject(oIVBox)
1058
1059 def getArray(self, oInterface, sAttrib):
1060 """ See PlatformBase::getArray(). """
1061 return self.platform.getArray(oInterface, sAttrib)
1062
1063 def setArray(self, oInterface, sAttrib, aoArray):
1064 """ See PlatformBase::setArray(). """
1065 return self.platform.setArray(oInterface, sAttrib, aoArray)
1066
1067 def createListener(self, oImplClass, dArgs=None):
1068 """ See PlatformBase::createListener(). """
1069 return self.platform.createListener(oImplClass, dArgs)
1070
1071 def waitForEvents(self, cMsTimeout):
1072 """ See PlatformBase::waitForEvents(). """
1073 return self.platform.waitForEvents(cMsTimeout)
1074
1075 def interruptWaitEvents(self):
1076 """ See PlatformBase::interruptWaitEvents(). """
1077 return self.platform.interruptWaitEvents()
1078
1079 def queryInterface(self, oIUnknown, sClassName):
1080 """ See PlatformBase::queryInterface(). """
1081 return self.platform.queryInterface(oIUnknown, sClassName)
1082
1083 #
1084 # Init and uninit.
1085 #
1086 def initPerThread(self):
1087 """ See PlatformBase::deinitPerThread(). """
1088 self.platform.initPerThread()
1089
1090 def deinitPerThread(self):
1091 """ See PlatformBase::deinitPerThread(). """
1092 return self.platform.deinitPerThread()
1093
1094 def deinit(self):
1095 """
1096 For unitializing the manager.
1097 Do not access it after calling this method.
1098 """
1099 if hasattr(self, "vbox") and self.vbox is not None:
1100 del self.vbox
1101 self.vbox = None
1102 if hasattr(self, "platform") and self.platform is not None:
1103 self.platform.deinit()
1104 self.platform = None
1105 return True
1106
1107 #
1108 # Utility methods.
1109 #
1110 def openMachineSession(self, oIMachine, fPermitSharing=True):
1111 """
1112 Attempts to open the a session to the machine.
1113 Returns a session object on success.
1114 Raises exception on failure.
1115 """
1116 oSession = self.getSessionObject(self.vbox);
1117 if fPermitSharing:
1118 eType = self.constants.LockType_Shared
1119 else:
1120 eType = self.constants.LockType_Write
1121 oIMachine.lockMachine(oSession, eType)
1122 return oSession
1123
1124 def closeMachineSession(self, oSession):
1125 """
1126 Closes a session opened by openMachineSession.
1127 Ignores None parameters.
1128 """
1129 if oSession is not None:
1130 oSession.unlockMachine()
1131 return True
1132
1133 def getPerfCollector(self, oIVBox):
1134 """
1135 Returns a helper class (PerfCollector) for accessing performance
1136 collector goodies. See PerfCollector for details.
1137 """
1138 return PerfCollector(self, oIVBox)
1139
1140 def getBinDir(self):
1141 """
1142 Returns the VirtualBox binary directory.
1143 """
1144 global VBoxBinDir
1145 return VBoxBinDir
1146
1147 def getSdkDir(self):
1148 """
1149 Returns the VirtualBox SDK directory.
1150 """
1151 global VBoxSdkDir
1152 return VBoxSdkDir
1153
1154 #
1155 # Error code utilities.
1156 #
1157 ## @todo port to webservices!
1158 def xcptGetStatus(self, oXcpt=None):
1159 """
1160 Gets the status code from an exception. If the exception parameter
1161 isn't specified, the current exception is examined.
1162 """
1163 if oXcpt is None:
1164 oXcpt = sys.exc_info()[1]
1165 return self.platform.xcptGetStatus(oXcpt)
1166
1167 def xcptIsDeadInterface(self, oXcpt=None):
1168 """
1169 Returns True if the exception indicates that the interface is dead,
1170 False if not. If the exception parameter isn't specified, the current
1171 exception is examined.
1172 """
1173 if oXcpt is None:
1174 oXcpt = sys.exc_info()[1]
1175 return self.platform.xcptIsDeadInterface(oXcpt)
1176
1177 def xcptIsOurXcptKind(self, oXcpt=None):
1178 """
1179 Checks if the exception is one that could come from the VBox API. If
1180 the exception parameter isn't specified, the current exception is
1181 examined.
1182 """
1183 if self.oXcptClass is None: # @todo find the exception class for web services!
1184 return False
1185 if oXcpt is None:
1186 oXcpt = sys.exc_info()[1]
1187 return isinstance(oXcpt, self.oXcptClass)
1188
1189 def xcptIsEqual(self, oXcpt, hrStatus):
1190 """
1191 Checks if the exception oXcpt is equal to the COM/XPCOM status code
1192 hrStatus.
1193
1194 The oXcpt parameter can be any kind of object, we'll just return True
1195 if it doesn't behave like a our exception class. If it's None, we'll
1196 query the current exception and examine that.
1197
1198 Will not raise any exception as long as hrStatus and self are not bad.
1199 """
1200 if oXcpt is None:
1201 oXcpt = sys.exc_info()[1]
1202 return self.platform.xcptIsEqual(oXcpt, hrStatus)
1203
1204 def xcptIsNotEqual(self, oXcpt, hrStatus):
1205 """
1206 Negated xcptIsEqual.
1207 """
1208 return not self.xcptIsEqual(oXcpt, hrStatus)
1209
1210 def xcptToString(self, hrStatusOrXcpt=None):
1211 """
1212 Converts the specified COM status code, or the status code of the
1213 specified exception, to a C constant string. If the parameter isn't
1214 specified (is None), the current exception is examined.
1215 """
1216
1217 # Deal with exceptions.
1218 if hrStatusOrXcpt is None or self.xcptIsOurXcptKind(hrStatusOrXcpt):
1219 hrStatus = self.xcptGetStatus(hrStatusOrXcpt)
1220 else:
1221 hrStatus = hrStatusOrXcpt
1222
1223 # Build the dictionary on demand.
1224 if self._dErrorValToName is None:
1225 dErrorValToName = dict()
1226 for sKey in dir(self.statuses):
1227 if sKey[0].isupper():
1228 oValue = getattr(self.statuses, sKey)
1229 if type(oValue) is int:
1230 dErrorValToName[oValue] = sKey
1231 self._dErrorValToName = dErrorValToName
1232
1233 # Do the lookup, falling back on formatting the status number.
1234 try:
1235 sStr = self._dErrorValToName[int(hrStatus)]
1236 except KeyError:
1237 hrLong = long(hrStatus)
1238 sStr = '%#x (%d)' % (hrLong, hrLong)
1239 return sStr
1240
1241 def xcptGetMessage(self, oXcpt=None):
1242 """
1243 Returns the best error message found in the COM-like exception. If the
1244 exception parameter isn't specified, the current exception is examined.
1245 """
1246 if oXcpt is None:
1247 oXcpt = sys.exc_info()[1]
1248 sRet = self.platform.xcptGetMessage(oXcpt)
1249 if sRet is None:
1250 sRet = self.xcptToString(oXcpt)
1251 return sRet
1252
1253 # Legacy, remove in a day or two.
1254 errGetStatus = xcptGetStatus
1255 errIsDeadInterface = xcptIsDeadInterface
1256 errIsOurXcptKind = xcptIsOurXcptKind
1257 errGetMessage = xcptGetMessage
1258
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