VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/ApplianceImpl.cpp@ 55747

Last change on this file since 55747 was 55505, checked in by vboxsync, 10 years ago

Main: Add method to return list of medium identifiers assoicated with a given password ID. Required by the GUI to verify entered passwords during appliance export

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 58.1 KB
Line 
1/* $Id: ApplianceImpl.cpp 55505 2015-04-29 08:26:44Z vboxsync $ */
2/** @file
3 * IAppliance and IVirtualSystem COM class implementations.
4 */
5
6/*
7 * Copyright (C) 2008-2013 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18#include <iprt/path.h>
19#include <iprt/cpp/utils.h>
20#include <VBox/com/array.h>
21#include <map>
22
23#include "ApplianceImpl.h"
24#include "VFSExplorerImpl.h"
25#include "VirtualBoxImpl.h"
26#include "GuestOSTypeImpl.h"
27#include "Global.h"
28#include "ProgressImpl.h"
29#include "MachineImpl.h"
30#include "SystemPropertiesImpl.h"
31#include "AutoCaller.h"
32#include "Logging.h"
33
34#include "ApplianceImplPrivate.h"
35
36using namespace std;
37
38////////////////////////////////////////////////////////////////////////////////
39//
40// Appliance constructor / destructor
41//
42// ////////////////////////////////////////////////////////////////////////////////
43
44DEFINE_EMPTY_CTOR_DTOR(VirtualSystemDescription)
45
46HRESULT VirtualSystemDescription::FinalConstruct()
47{
48 return BaseFinalConstruct();
49}
50
51void VirtualSystemDescription::FinalRelease()
52{
53 uninit();
54
55 BaseFinalRelease();
56}
57
58Appliance::Appliance()
59 : mVirtualBox(NULL)
60{
61}
62
63Appliance::~Appliance()
64{
65}
66
67
68HRESULT Appliance::FinalConstruct()
69{
70 return BaseFinalConstruct();
71}
72
73void Appliance::FinalRelease()
74{
75 uninit();
76
77 BaseFinalRelease();
78}
79
80
81////////////////////////////////////////////////////////////////////////////////
82//
83// Internal helpers
84//
85////////////////////////////////////////////////////////////////////////////////
86
87static const char* const strISOURI = "http://www.ecma-international.org/publications/standards/Ecma-119.htm";
88static const char* const strVMDKStreamURI = "http://www.vmware.com/interfaces/specifications/vmdk.html#streamOptimized";
89static const char* const strVMDKSparseURI = "http://www.vmware.com/specifications/vmdk.html#sparse";
90static const char* const strVMDKCompressedURI = "http://www.vmware.com/specifications/vmdk.html#compressed";
91static const char* const strVMDKCompressedURI2 = "http://www.vmware.com/interfaces/specifications/vmdk.html#compressed";
92static const char* const strVHDURI = "http://go.microsoft.com/fwlink/?LinkId=137171";
93
94static std::map<Utf8Str, Utf8Str> supportedStandardsURI;
95
96static const char* const applianceIOTarName = "Appliance::IOTar";
97static const char* const applianceIOShaName = "Appliance::IOSha";
98static const char* const applianceIOFileName = "Appliance::IOFile";
99
100static std::map<APPLIANCEIONAME, Utf8Str> applianceIONameMap;
101
102static const struct
103{
104 ovf::CIMOSType_T cim;
105 VBOXOSTYPE osType;
106}
107g_osTypes[] =
108{
109 { ovf::CIMOSType_CIMOS_Unknown, VBOXOSTYPE_Unknown },
110 { ovf::CIMOSType_CIMOS_OS2, VBOXOSTYPE_OS2 },
111 { ovf::CIMOSType_CIMOS_OS2, VBOXOSTYPE_OS2Warp3 },
112 { ovf::CIMOSType_CIMOS_OS2, VBOXOSTYPE_OS2Warp4 },
113 { ovf::CIMOSType_CIMOS_OS2, VBOXOSTYPE_OS2Warp45 },
114 { ovf::CIMOSType_CIMOS_MSDOS, VBOXOSTYPE_DOS },
115 { ovf::CIMOSType_CIMOS_WIN3x, VBOXOSTYPE_Win31 },
116 { ovf::CIMOSType_CIMOS_WIN95, VBOXOSTYPE_Win95 },
117 { ovf::CIMOSType_CIMOS_WIN98, VBOXOSTYPE_Win98 },
118 { ovf::CIMOSType_CIMOS_WINNT, VBOXOSTYPE_WinNT },
119 { ovf::CIMOSType_CIMOS_WINNT, VBOXOSTYPE_WinNT4 },
120 { ovf::CIMOSType_CIMOS_NetWare, VBOXOSTYPE_Netware },
121 { ovf::CIMOSType_CIMOS_NovellOES, VBOXOSTYPE_Netware },
122 { ovf::CIMOSType_CIMOS_Solaris, VBOXOSTYPE_Solaris },
123 { ovf::CIMOSType_CIMOS_SunOS, VBOXOSTYPE_Solaris },
124 { ovf::CIMOSType_CIMOS_FreeBSD, VBOXOSTYPE_FreeBSD },
125 { ovf::CIMOSType_CIMOS_NetBSD, VBOXOSTYPE_NetBSD },
126 { ovf::CIMOSType_CIMOS_QNX, VBOXOSTYPE_QNX },
127 { ovf::CIMOSType_CIMOS_Windows2000, VBOXOSTYPE_Win2k },
128 { ovf::CIMOSType_CIMOS_WindowsMe, VBOXOSTYPE_WinMe },
129 { ovf::CIMOSType_CIMOS_OpenBSD, VBOXOSTYPE_OpenBSD },
130 { ovf::CIMOSType_CIMOS_WindowsXP, VBOXOSTYPE_WinXP },
131 { ovf::CIMOSType_CIMOS_WindowsXPEmbedded, VBOXOSTYPE_WinXP },
132 { ovf::CIMOSType_CIMOS_WindowsEmbeddedforPointofService, VBOXOSTYPE_WinXP },
133 { ovf::CIMOSType_CIMOS_MicrosoftWindowsServer2003, VBOXOSTYPE_Win2k3 },
134 { ovf::CIMOSType_CIMOS_MicrosoftWindowsServer2003_64, VBOXOSTYPE_Win2k3_x64 },
135 { ovf::CIMOSType_CIMOS_WindowsXP_64, VBOXOSTYPE_WinXP_x64 },
136 { ovf::CIMOSType_CIMOS_WindowsVista, VBOXOSTYPE_WinVista },
137 { ovf::CIMOSType_CIMOS_WindowsVista_64, VBOXOSTYPE_WinVista_x64 },
138 { ovf::CIMOSType_CIMOS_MicrosoftWindowsServer2008, VBOXOSTYPE_Win2k8 },
139 { ovf::CIMOSType_CIMOS_MicrosoftWindowsServer2008_64, VBOXOSTYPE_Win2k8_x64 },
140 { ovf::CIMOSType_CIMOS_FreeBSD_64, VBOXOSTYPE_FreeBSD_x64 },
141 { ovf::CIMOSType_CIMOS_MACOS, VBOXOSTYPE_MacOS },
142 { ovf::CIMOSType_CIMOS_MACOS, VBOXOSTYPE_MacOS_x64 }, // there is no CIM 64-bit type for this
143 { ovf::CIMOSType_CIMOS_MACOS, VBOXOSTYPE_MacOS106 },
144 { ovf::CIMOSType_CIMOS_MACOS, VBOXOSTYPE_MacOS106_x64 },
145 { ovf::CIMOSType_CIMOS_MACOS, VBOXOSTYPE_MacOS107_x64 },
146 { ovf::CIMOSType_CIMOS_MACOS, VBOXOSTYPE_MacOS108_x64 },
147 { ovf::CIMOSType_CIMOS_MACOS, VBOXOSTYPE_MacOS109_x64 },
148
149 // Linuxes
150 { ovf::CIMOSType_CIMOS_RedHatEnterpriseLinux, VBOXOSTYPE_RedHat },
151 { ovf::CIMOSType_CIMOS_RedHatEnterpriseLinux_64, VBOXOSTYPE_RedHat_x64 },
152 { ovf::CIMOSType_CIMOS_Solaris_64, VBOXOSTYPE_Solaris_x64 },
153 { ovf::CIMOSType_CIMOS_SUSE, VBOXOSTYPE_OpenSUSE },
154 { ovf::CIMOSType_CIMOS_SLES, VBOXOSTYPE_OpenSUSE },
155 { ovf::CIMOSType_CIMOS_NovellLinuxDesktop, VBOXOSTYPE_OpenSUSE },
156 { ovf::CIMOSType_CIMOS_SUSE_64, VBOXOSTYPE_OpenSUSE_x64 },
157 { ovf::CIMOSType_CIMOS_SLES_64, VBOXOSTYPE_OpenSUSE_x64 },
158 { ovf::CIMOSType_CIMOS_LINUX, VBOXOSTYPE_Linux },
159 { ovf::CIMOSType_CIMOS_LINUX, VBOXOSTYPE_Linux22 },
160 { ovf::CIMOSType_CIMOS_SunJavaDesktopSystem, VBOXOSTYPE_Linux },
161 { ovf::CIMOSType_CIMOS_TurboLinux, VBOXOSTYPE_Turbolinux },
162 { ovf::CIMOSType_CIMOS_TurboLinux_64, VBOXOSTYPE_Turbolinux_x64 },
163 { ovf::CIMOSType_CIMOS_Mandriva, VBOXOSTYPE_Mandriva },
164 { ovf::CIMOSType_CIMOS_Mandriva_64, VBOXOSTYPE_Mandriva_x64 },
165 { ovf::CIMOSType_CIMOS_Ubuntu, VBOXOSTYPE_Ubuntu },
166 { ovf::CIMOSType_CIMOS_Ubuntu_64, VBOXOSTYPE_Ubuntu_x64 },
167 { ovf::CIMOSType_CIMOS_Debian, VBOXOSTYPE_Debian },
168 { ovf::CIMOSType_CIMOS_Debian_64, VBOXOSTYPE_Debian_x64 },
169 { ovf::CIMOSType_CIMOS_Linux_2_4_x, VBOXOSTYPE_Linux24 },
170 { ovf::CIMOSType_CIMOS_Linux_2_4_x_64, VBOXOSTYPE_Linux24_x64 },
171 { ovf::CIMOSType_CIMOS_Linux_2_6_x, VBOXOSTYPE_Linux26 },
172 { ovf::CIMOSType_CIMOS_Linux_2_6_x_64, VBOXOSTYPE_Linux26_x64 },
173 { ovf::CIMOSType_CIMOS_Linux_64, VBOXOSTYPE_Linux26_x64 },
174
175 // types that we have support for but CIM doesn't
176 { ovf::CIMOSType_CIMOS_Linux_2_6_x, VBOXOSTYPE_ArchLinux },
177 { ovf::CIMOSType_CIMOS_Linux_2_6_x_64, VBOXOSTYPE_ArchLinux_x64 },
178 { ovf::CIMOSType_CIMOS_Linux_2_6_x, VBOXOSTYPE_FedoraCore },
179 { ovf::CIMOSType_CIMOS_Linux_2_6_x_64, VBOXOSTYPE_FedoraCore_x64 },
180 { ovf::CIMOSType_CIMOS_Linux_2_6_x, VBOXOSTYPE_Gentoo },
181 { ovf::CIMOSType_CIMOS_Linux_2_6_x_64, VBOXOSTYPE_Gentoo_x64 },
182 { ovf::CIMOSType_CIMOS_Linux_2_6_x, VBOXOSTYPE_Xandros },
183 { ovf::CIMOSType_CIMOS_Linux_2_6_x_64, VBOXOSTYPE_Xandros_x64 },
184 { ovf::CIMOSType_CIMOS_Solaris, VBOXOSTYPE_OpenSolaris },
185 { ovf::CIMOSType_CIMOS_Solaris_64, VBOXOSTYPE_OpenSolaris_x64 },
186
187 // types added with CIM 2.25.0 follow:
188 { ovf::CIMOSType_CIMOS_WindowsServer2008R2, VBOXOSTYPE_Win2k8 }, // duplicate, see above
189// { ovf::CIMOSType_CIMOS_VMwareESXi = 104, // we can't run ESX in a VM
190 { ovf::CIMOSType_CIMOS_Windows7, VBOXOSTYPE_Win7 },
191 { ovf::CIMOSType_CIMOS_Windows7, VBOXOSTYPE_Win7_x64 }, // there is no
192 // CIM 64-bit type for this
193 { ovf::CIMOSType_CIMOS_CentOS, VBOXOSTYPE_RedHat },
194 { ovf::CIMOSType_CIMOS_CentOS_64, VBOXOSTYPE_RedHat_x64 },
195 { ovf::CIMOSType_CIMOS_OracleEnterpriseLinux, VBOXOSTYPE_Oracle },
196 { ovf::CIMOSType_CIMOS_OracleEnterpriseLinux_64, VBOXOSTYPE_Oracle_x64 },
197 { ovf::CIMOSType_CIMOS_eComStation, VBOXOSTYPE_ECS }
198
199 // there are no CIM types for these, so these turn to "other" on export:
200 // VBOXOSTYPE_OpenBSD
201 // VBOXOSTYPE_OpenBSD_x64
202 // VBOXOSTYPE_NetBSD
203 // VBOXOSTYPE_NetBSD_x64
204
205};
206
207/* Pattern structure for matching the OS type description field */
208struct osTypePattern
209{
210 const char *pcszPattern;
211 VBOXOSTYPE osType;
212};
213
214/* These are the 32-Bit ones. They are sorted by priority. */
215static const osTypePattern g_osTypesPattern[] =
216{
217 {"Windows NT", VBOXOSTYPE_WinNT4},
218 {"Windows XP", VBOXOSTYPE_WinXP},
219 {"Windows 2000", VBOXOSTYPE_Win2k},
220 {"Windows 2003", VBOXOSTYPE_Win2k3},
221 {"Windows Vista", VBOXOSTYPE_WinVista},
222 {"Windows 2008", VBOXOSTYPE_Win2k8},
223 {"SUSE", VBOXOSTYPE_OpenSUSE},
224 {"Novell", VBOXOSTYPE_OpenSUSE},
225 {"Red Hat", VBOXOSTYPE_RedHat},
226 {"Mandriva", VBOXOSTYPE_Mandriva},
227 {"Ubuntu", VBOXOSTYPE_Ubuntu},
228 {"Debian", VBOXOSTYPE_Debian},
229 {"QNX", VBOXOSTYPE_QNX},
230 {"Linux 2.4", VBOXOSTYPE_Linux24},
231 {"Linux 2.6", VBOXOSTYPE_Linux26},
232 {"Linux", VBOXOSTYPE_Linux},
233 {"OpenSolaris", VBOXOSTYPE_OpenSolaris},
234 {"Solaris", VBOXOSTYPE_OpenSolaris},
235 {"FreeBSD", VBOXOSTYPE_FreeBSD},
236 {"NetBSD", VBOXOSTYPE_NetBSD},
237 {"Windows 95", VBOXOSTYPE_Win95},
238 {"Windows 98", VBOXOSTYPE_Win98},
239 {"Windows Me", VBOXOSTYPE_WinMe},
240 {"Windows 3.", VBOXOSTYPE_Win31},
241 {"DOS", VBOXOSTYPE_DOS},
242 {"OS2", VBOXOSTYPE_OS2}
243};
244
245/* These are the 64-Bit ones. They are sorted by priority. */
246static const osTypePattern g_osTypesPattern64[] =
247{
248 {"Windows XP", VBOXOSTYPE_WinXP_x64},
249 {"Windows 2003", VBOXOSTYPE_Win2k3_x64},
250 {"Windows Vista", VBOXOSTYPE_WinVista_x64},
251 {"Windows 2008", VBOXOSTYPE_Win2k8_x64},
252 {"SUSE", VBOXOSTYPE_OpenSUSE_x64},
253 {"Novell", VBOXOSTYPE_OpenSUSE_x64},
254 {"Red Hat", VBOXOSTYPE_RedHat_x64},
255 {"Mandriva", VBOXOSTYPE_Mandriva_x64},
256 {"Ubuntu", VBOXOSTYPE_Ubuntu_x64},
257 {"Debian", VBOXOSTYPE_Debian_x64},
258 {"Linux 2.4", VBOXOSTYPE_Linux24_x64},
259 {"Linux 2.6", VBOXOSTYPE_Linux26_x64},
260 {"Linux", VBOXOSTYPE_Linux26_x64},
261 {"OpenSolaris", VBOXOSTYPE_OpenSolaris_x64},
262 {"Solaris", VBOXOSTYPE_OpenSolaris_x64},
263 {"FreeBSD", VBOXOSTYPE_FreeBSD_x64},
264};
265
266/**
267 * Private helper func that suggests a VirtualBox guest OS type
268 * for the given OVF operating system type.
269 * @param osTypeVBox
270 * @param c
271 * @param cStr
272 */
273void convertCIMOSType2VBoxOSType(Utf8Str &strType, ovf::CIMOSType_T c, const Utf8Str &cStr)
274{
275 /* First check if the type is other/other_64 */
276 if (c == ovf::CIMOSType_CIMOS_Other)
277 {
278 for (size_t i=0; i < RT_ELEMENTS(g_osTypesPattern); ++i)
279 if (cStr.contains (g_osTypesPattern[i].pcszPattern, Utf8Str::CaseInsensitive))
280 {
281 strType = Global::OSTypeId(g_osTypesPattern[i].osType);
282 return;
283 }
284 }
285 else if (c == ovf::CIMOSType_CIMOS_Other_64)
286 {
287 for (size_t i=0; i < RT_ELEMENTS(g_osTypesPattern64); ++i)
288 if (cStr.contains (g_osTypesPattern64[i].pcszPattern, Utf8Str::CaseInsensitive))
289 {
290 strType = Global::OSTypeId(g_osTypesPattern64[i].osType);
291 return;
292 }
293 }
294
295 for (size_t i = 0; i < RT_ELEMENTS(g_osTypes); ++i)
296 {
297 if (c == g_osTypes[i].cim)
298 {
299 strType = Global::OSTypeId(g_osTypes[i].osType);
300 return;
301 }
302 }
303
304 if (c == ovf::CIMOSType_CIMOS_Other_64)
305 strType = Global::OSTypeId(VBOXOSTYPE_Unknown_x64);
306 else
307 strType = Global::OSTypeId(VBOXOSTYPE_Unknown);
308}
309
310/**
311 * Private helper func that suggests a VirtualBox guest OS type
312 * for the given OVF operating system type.
313 * @returns CIM OS type.
314 * @param pcszVBox Our guest OS type identifier string.
315 * @param fLongMode Whether long mode is enabled and a 64-bit CIM type is
316 * preferred even if the VBox guest type isn't 64-bit.
317 */
318ovf::CIMOSType_T convertVBoxOSType2CIMOSType(const char *pcszVBox, BOOL fLongMode)
319{
320 for (size_t i = 0; i < RT_ELEMENTS(g_osTypes); ++i)
321 {
322 if (!RTStrICmp(pcszVBox, Global::OSTypeId(g_osTypes[i].osType)))
323 {
324 if (fLongMode && !(g_osTypes[i].osType & VBOXOSTYPE_x64))
325 {
326 VBOXOSTYPE enmDesiredOsType = (VBOXOSTYPE)((int)g_osTypes[i].osType | (int)VBOXOSTYPE_x64);
327 size_t j = i;
328 while (++j < RT_ELEMENTS(g_osTypes))
329 if (g_osTypes[j].osType == enmDesiredOsType)
330 return g_osTypes[j].cim;
331 j = i;
332 while (--j > 0)
333 if (g_osTypes[j].osType == enmDesiredOsType)
334 return g_osTypes[j].cim;
335 /* Not all OSes have 64-bit versions, so just return the 32-bit variant. */
336 }
337 return g_osTypes[i].cim;
338 }
339 }
340
341 return fLongMode ? ovf::CIMOSType_CIMOS_Other_64 : ovf::CIMOSType_CIMOS_Other;
342}
343
344Utf8Str convertNetworkAttachmentTypeToString(NetworkAttachmentType_T type)
345{
346 Utf8Str strType;
347 switch (type)
348 {
349 case NetworkAttachmentType_NAT: strType = "NAT"; break;
350 case NetworkAttachmentType_Bridged: strType = "Bridged"; break;
351 case NetworkAttachmentType_Internal: strType = "Internal"; break;
352 case NetworkAttachmentType_HostOnly: strType = "HostOnly"; break;
353 case NetworkAttachmentType_Generic: strType = "Generic"; break;
354 case NetworkAttachmentType_NATNetwork: strType = "NATNetwork"; break;
355 case NetworkAttachmentType_Null: strType = "Null"; break;
356 }
357 return strType;
358}
359
360////////////////////////////////////////////////////////////////////////////////
361//
362// IVirtualBox public methods
363//
364////////////////////////////////////////////////////////////////////////////////
365
366// This code is here so we won't have to include the appliance headers in the
367// IVirtualBox implementation.
368
369/**
370 * Implementation for IVirtualBox::createAppliance.
371 *
372 * @param anAppliance IAppliance object created if S_OK is returned.
373 * @return S_OK or error.
374 */
375HRESULT VirtualBox::createAppliance(ComPtr<IAppliance> &aAppliance)
376{
377 HRESULT rc;
378
379 ComObjPtr<Appliance> appliance;
380 appliance.createObject();
381 rc = appliance->init(this);
382
383 if (SUCCEEDED(rc))
384 appliance.queryInterfaceTo(aAppliance.asOutParam());
385
386 return rc;
387}
388
389/**
390 * Appliance COM initializer.
391 * @param
392 * @return
393 */
394HRESULT Appliance::init(VirtualBox *aVirtualBox)
395{
396 HRESULT rc = S_OK;
397 /* Enclose the state transition NotReady->InInit->Ready */
398 AutoInitSpan autoInitSpan(this);
399 AssertReturn(autoInitSpan.isOk(), E_FAIL);
400
401 /* Weak reference to a VirtualBox object */
402 unconst(mVirtualBox) = aVirtualBox;
403
404 // initialize data
405 m = new Data;
406 m->m_pSecretKeyStore = new SecretKeyStore(false /* fRequireNonPageable*/);
407 AssertReturn(m->m_pSecretKeyStore, E_FAIL);
408
409 i_initApplianceIONameMap();
410
411 rc = i_initSetOfSupportedStandardsURI();
412
413 /* Confirm a successful initialization */
414 autoInitSpan.setSucceeded();
415
416 return rc;
417}
418
419/**
420 * Appliance COM uninitializer.
421 * @return
422 */
423void Appliance::uninit()
424{
425 /* Enclose the state transition Ready->InUninit->NotReady */
426 AutoUninitSpan autoUninitSpan(this);
427 if (autoUninitSpan.uninitDone())
428 return;
429
430 if (m->m_pSecretKeyStore)
431 delete m->m_pSecretKeyStore;
432
433 delete m;
434 m = NULL;
435}
436
437////////////////////////////////////////////////////////////////////////////////
438//
439// IAppliance public methods
440//
441////////////////////////////////////////////////////////////////////////////////
442
443/**
444 * Public method implementation.
445 * @param
446 * @return
447 */
448HRESULT Appliance::getPath(com::Utf8Str &aPath)
449{
450 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
451
452 if (!i_isApplianceIdle())
453 return E_ACCESSDENIED;
454
455 aPath = m->locInfo.strPath;
456
457 return S_OK;
458}
459
460/**
461 * Public method implementation.
462 * @param
463 * @return
464 */
465HRESULT Appliance::getDisks(std::vector<com::Utf8Str> &aDisks)
466{
467 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
468
469 aDisks.resize(0);
470 if (!i_isApplianceIdle())
471 return E_ACCESSDENIED;
472
473 if (m->pReader) // OVFReader instantiated?
474 {
475 aDisks.resize(m->pReader->m_mapDisks.size());
476
477 ovf::DiskImagesMap::const_iterator it;
478 size_t i = 0;
479 for (it = m->pReader->m_mapDisks.begin();
480 it != m->pReader->m_mapDisks.end();
481 ++it, ++i)
482 {
483 // create a string representing this disk
484 const ovf::DiskImage &d = it->second;
485 char *psz = NULL;
486 RTStrAPrintf(&psz,
487 "%s\t"
488 "%RI64\t"
489 "%RI64\t"
490 "%s\t"
491 "%s\t"
492 "%RI64\t"
493 "%RI64\t"
494 "%s",
495 d.strDiskId.c_str(),
496 d.iCapacity,
497 d.iPopulatedSize,
498 d.strFormat.c_str(),
499 d.strHref.c_str(),
500 d.iSize,
501 d.iChunkSize,
502 d.strCompression.c_str());
503 Utf8Str utf(psz);
504 aDisks[i] = utf;
505 RTStrFree(psz);
506 }
507 }
508
509 return S_OK;
510}
511
512/**
513 * Public method implementation.
514 * @param
515 * @return
516 */
517HRESULT Appliance::getVirtualSystemDescriptions(std::vector<ComPtr<IVirtualSystemDescription> > &aVirtualSystemDescriptions)
518{
519 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
520
521 if (!i_isApplianceIdle())
522 return E_ACCESSDENIED;
523
524 aVirtualSystemDescriptions.resize(m->virtualSystemDescriptions.size());
525 std::list< ComObjPtr<VirtualSystemDescription> > vsds(m->virtualSystemDescriptions);
526 size_t i = 0;
527 for (std::list< ComObjPtr<VirtualSystemDescription> >::iterator it = vsds.begin(); it != vsds.end(); ++it, ++i)
528 {
529 (*it).queryInterfaceTo(aVirtualSystemDescriptions[i].asOutParam());
530 }
531 return S_OK;
532}
533
534/**
535 * Public method implementation.
536 * @param aDisks
537 * @return
538 */
539HRESULT Appliance::getMachines(std::vector<com::Utf8Str> &aMachines)
540{
541 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
542
543 if (!i_isApplianceIdle())
544 return E_ACCESSDENIED;
545
546 aMachines.resize(m->llGuidsMachinesCreated.size());
547 size_t i = 0;
548 for (std::list<Guid>::const_iterator it = m->llGuidsMachinesCreated.begin();
549 it != m->llGuidsMachinesCreated.end();
550 ++it, ++i)
551 {
552 const Guid &uuid = *it;
553 aMachines[i] = uuid.toUtf16();
554 }
555 return S_OK;
556}
557
558HRESULT Appliance::createVFSExplorer(const com::Utf8Str &aURI, ComPtr<IVFSExplorer> &aExplorer)
559{
560 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
561
562 ComObjPtr<VFSExplorer> explorer;
563 HRESULT rc = S_OK;
564 try
565 {
566 Utf8Str uri(aURI);
567 /* Check which kind of export the user has requested */
568 LocationInfo li;
569 i_parseURI(aURI, li);
570 /* Create the explorer object */
571 explorer.createObject();
572 rc = explorer->init(li.storageType, li.strPath, li.strHostname, li.strUsername, li.strPassword, mVirtualBox);
573 }
574 catch (HRESULT aRC)
575 {
576 rc = aRC;
577 }
578
579 if (SUCCEEDED(rc))
580 /* Return explorer to the caller */
581 explorer.queryInterfaceTo(aExplorer.asOutParam());
582
583 return rc;
584}
585
586/**
587* Public method implementation.
588 * @return
589 */
590HRESULT Appliance::getWarnings(std::vector<com::Utf8Str> &aWarnings)
591{
592 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
593
594 aWarnings.resize(m->llWarnings.size());
595
596 list<Utf8Str>::const_iterator it;
597 size_t i = 0;
598 for (it = m->llWarnings.begin();
599 it != m->llWarnings.end();
600 ++it, ++i)
601 {
602 aWarnings[i] = *it;
603 }
604
605 return S_OK;
606}
607
608HRESULT Appliance::getPasswordIds(std::vector<com::Utf8Str> &aIdentifiers)
609{
610 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
611
612 aIdentifiers = m->m_vecPasswordIdentifiers;
613 return S_OK;
614}
615
616HRESULT Appliance::getMediumIdsForPasswordId(const com::Utf8Str &aPasswordId, std::vector<com::Guid> &aIdentifiers)
617{
618 HRESULT hrc = S_OK;
619 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
620
621 std::map<com::Utf8Str, GUIDVEC>::const_iterator it = m->m_mapPwIdToMediumIds.find(aPasswordId);
622 if (it != m->m_mapPwIdToMediumIds.end())
623 aIdentifiers = it->second;
624 else
625 hrc = setError(E_FAIL, tr("The given password identifier is not associated with any medium"));
626
627 return hrc;
628}
629
630HRESULT Appliance::addPasswords(const std::vector<com::Utf8Str> &aIdentifiers,
631 const std::vector<com::Utf8Str> &aPasswords)
632{
633 HRESULT hrc = S_OK;
634
635 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
636
637 /* Check that the IDs do not exist already before changing anything. */
638 for (unsigned i = 0; i < aIdentifiers.size(); i++)
639 {
640 SecretKey *pKey = NULL;
641 int rc = m->m_pSecretKeyStore->retainSecretKey(aIdentifiers[i], &pKey);
642 if (rc != VERR_NOT_FOUND)
643 {
644 AssertPtr(pKey);
645 if (pKey)
646 pKey->release();
647 return setError(VBOX_E_OBJECT_IN_USE, tr("A password with the given ID already exists"));
648 }
649 }
650
651 for (unsigned i = 0; i < aIdentifiers.size() && SUCCEEDED(hrc); i++)
652 {
653 size_t cbKey = aPasswords[i].length() + 1; /* Include terminator */
654 const uint8_t *pbKey = (const uint8_t *)aPasswords[i].c_str();
655
656 int rc = m->m_pSecretKeyStore->addSecretKey(aIdentifiers[i], pbKey, cbKey);
657 if (RT_SUCCESS(rc))
658 m->m_cPwProvided++;
659 else if (rc == VERR_NO_MEMORY)
660 hrc = setError(E_OUTOFMEMORY, tr("Failed to allocate enough secure memory for the key"));
661 else
662 hrc = setError(E_FAIL, tr("Unknown error happened while adding a password (%Rrc)"), rc);
663 }
664
665 return hrc;
666}
667
668////////////////////////////////////////////////////////////////////////////////
669//
670// Appliance private methods
671//
672////////////////////////////////////////////////////////////////////////////////
673//
674HRESULT Appliance::i_initSetOfSupportedStandardsURI()
675{
676 HRESULT rc = S_OK;
677 if (!supportedStandardsURI.empty())
678 return rc;
679
680 /* Get the system properties. */
681 SystemProperties *pSysProps = mVirtualBox->i_getSystemProperties();
682 {
683 ComObjPtr<MediumFormat> trgFormat = pSysProps->i_mediumFormatFromExtension("iso");
684 if (trgFormat.isNull())
685 return setError(E_FAIL, tr("Can't find appropriate medium format for ISO type of a virtual disk."));
686
687 Bstr bstrFormatName;
688 rc = trgFormat->COMGETTER(Name)(bstrFormatName.asOutParam());
689 if (FAILED(rc)) return rc;
690
691 Utf8Str strTrgFormat = Utf8Str(bstrFormatName);
692
693 supportedStandardsURI.insert(std::make_pair(Utf8Str(strISOURI), strTrgFormat));
694 }
695
696 {
697 ComObjPtr<MediumFormat> trgFormat = pSysProps->i_mediumFormatFromExtension("vmdk");
698 if (trgFormat.isNull())
699 return setError(E_FAIL, tr("Can't find appropriate medium format for VMDK type of a virtual disk."));
700
701 Bstr bstrFormatName;
702 rc = trgFormat->COMGETTER(Name)(bstrFormatName.asOutParam());
703 if (FAILED(rc)) return rc;
704
705 Utf8Str strTrgFormat = Utf8Str(bstrFormatName);
706
707 supportedStandardsURI.insert(std::make_pair(Utf8Str(strVMDKStreamURI), strTrgFormat));
708 supportedStandardsURI.insert(std::make_pair(Utf8Str(strVMDKSparseURI), strTrgFormat));
709 supportedStandardsURI.insert(std::make_pair(Utf8Str(strVMDKCompressedURI), strTrgFormat));
710 supportedStandardsURI.insert(std::make_pair(Utf8Str(strVMDKCompressedURI2), strTrgFormat));
711 }
712
713 {
714 ComObjPtr<MediumFormat> trgFormat = pSysProps->i_mediumFormatFromExtension("vhd");
715 if (trgFormat.isNull())
716 return setError(E_FAIL, tr("Can't find appropriate medium format for VHD type of a virtual disk."));
717
718 Bstr bstrFormatName;
719 rc = trgFormat->COMGETTER(Name)(bstrFormatName.asOutParam());
720 if (FAILED(rc)) return rc;
721
722 Utf8Str strTrgFormat = Utf8Str(bstrFormatName);
723
724 supportedStandardsURI.insert(std::make_pair(Utf8Str(strVHDURI), strTrgFormat));
725 }
726
727 return rc;
728}
729
730Utf8Str Appliance::i_typeOfVirtualDiskFormatFromURI(Utf8Str uri) const
731{
732 Utf8Str type;
733 std::map<Utf8Str, Utf8Str>::const_iterator cit = supportedStandardsURI.find(uri);
734 if (cit != supportedStandardsURI.end())
735 {
736 type = cit->second;
737 }
738
739 return type;
740}
741
742std::set<Utf8Str> Appliance::i_URIFromTypeOfVirtualDiskFormat(Utf8Str type)
743{
744 std::set<Utf8Str> uri;
745 std::map<Utf8Str, Utf8Str>::const_iterator cit = supportedStandardsURI.begin();
746 while(cit != supportedStandardsURI.end())
747 {
748 if (cit->second.compare(type,Utf8Str::CaseInsensitive) == 0)
749 uri.insert(cit->first);
750 ++cit;
751 }
752
753 return uri;
754}
755
756HRESULT Appliance::i_initApplianceIONameMap()
757{
758 HRESULT rc = S_OK;
759 if (!applianceIONameMap.empty())
760 return rc;
761
762 applianceIONameMap.insert(std::make_pair(applianceIOTar, applianceIOTarName));
763 applianceIONameMap.insert(std::make_pair(applianceIOFile, applianceIOFileName));
764 applianceIONameMap.insert(std::make_pair(applianceIOSha, applianceIOShaName));
765
766 return rc;
767}
768
769Utf8Str Appliance::i_applianceIOName(APPLIANCEIONAME type) const
770{
771 Utf8Str name;
772 std::map<APPLIANCEIONAME, Utf8Str>::const_iterator cit = applianceIONameMap.find(type);
773 if (cit != applianceIONameMap.end())
774 {
775 name = cit->second;
776 }
777
778 return name;
779}
780
781
782/**
783 * Returns a medium format object corresponding to the given
784 * disk image or null if no such format.
785 *
786 * @param di Disk Image
787 * @param mf Medium Format
788 *
789 * @return ComObjPtr<MediumFormat>
790 */
791
792HRESULT Appliance::i_findMediumFormatFromDiskImage(const ovf::DiskImage &di, ComObjPtr<MediumFormat>& mf)
793{
794 HRESULT rc = S_OK;
795
796 /* Get the system properties. */
797 SystemProperties *pSysProps = mVirtualBox->i_getSystemProperties();
798
799 /* We need a proper source format description */
800 /* Which format to use? */
801 Utf8Str strSrcFormat = i_typeOfVirtualDiskFormatFromURI(di.strFormat);
802
803 /*
804 * fallback, if we can't determine virtual disk format using URI from the attribute ovf:format
805 * in the corresponding section <Disk> in the OVF file.
806 */
807 if (strSrcFormat.isEmpty())
808 {
809 strSrcFormat = di.strHref;
810
811 /* check either file gzipped or not
812 * if "yes" then remove last extension,
813 * i.e. "image.vmdk.gz"->"image.vmdk"
814 */
815 if (di.strCompression == "gzip")
816 {
817 if (RTPathHasSuffix(strSrcFormat.c_str()))
818 {
819 strSrcFormat.stripSuffix();
820 }
821 else
822 {
823 mf.setNull();
824 rc = setError(E_FAIL,
825 tr("Internal inconsistency looking up medium format for the disk image '%s'"),
826 di.strHref.c_str());
827 return rc;
828 }
829 }
830 /* Figure out from extension which format the image of disk has. */
831 if (RTPathHasSuffix(strSrcFormat.c_str()))
832 {
833 const char *pszExt = RTPathSuffix(strSrcFormat.c_str());
834 if (pszExt)
835 pszExt++;
836 mf = pSysProps->i_mediumFormatFromExtension(pszExt);
837 }
838 else
839 mf.setNull();
840 }
841 else
842 mf = pSysProps->i_mediumFormat(strSrcFormat);
843
844 if (mf.isNull())
845 {
846 rc = setError(E_FAIL,
847 tr("Internal inconsistency looking up medium format for the disk image '%s'"),
848 di.strHref.c_str());
849 }
850
851 return rc;
852}
853
854/**
855 * Returns true if the appliance is in "idle" state. This should always be the
856 * case unless an import or export is currently in progress. Similar to machine
857 * states, this permits the Appliance implementation code to let go of the
858 * Appliance object lock while a time-consuming disk conversion is in progress
859 * without exposing the appliance to conflicting calls.
860 *
861 * This sets an error on "this" (the appliance) and returns false if the appliance
862 * is busy. The caller should then return E_ACCESSDENIED.
863 *
864 * Must be called from under the object lock!
865 *
866 * @return
867 */
868bool Appliance::i_isApplianceIdle()
869{
870 if (m->state == Data::ApplianceImporting)
871 setError(VBOX_E_INVALID_OBJECT_STATE, tr("The appliance is busy importing files"));
872 else if (m->state == Data::ApplianceExporting)
873 setError(VBOX_E_INVALID_OBJECT_STATE, tr("The appliance is busy exporting files"));
874 else
875 return true;
876
877 return false;
878}
879
880HRESULT Appliance::i_searchUniqueVMName(Utf8Str& aName) const
881{
882 IMachine *machine = NULL;
883 char *tmpName = RTStrDup(aName.c_str());
884 int i = 1;
885 /** @todo: Maybe too cost-intensive; try to find a lighter way */
886 while (mVirtualBox->FindMachine(Bstr(tmpName).raw(), &machine) != VBOX_E_OBJECT_NOT_FOUND)
887 {
888 RTStrFree(tmpName);
889 RTStrAPrintf(&tmpName, "%s_%d", aName.c_str(), i);
890 ++i;
891 }
892 aName = tmpName;
893 RTStrFree(tmpName);
894
895 return S_OK;
896}
897
898HRESULT Appliance::i_searchUniqueDiskImageFilePath(Utf8Str& aName) const
899{
900 IMedium *harddisk = NULL;
901 char *tmpName = RTStrDup(aName.c_str());
902 int i = 1;
903 /* Check if the file exists or if a file with this path is registered
904 * already */
905 /** @todo: Maybe too cost-intensive; try to find a lighter way */
906 while ( RTPathExists(tmpName)
907 || mVirtualBox->OpenMedium(Bstr(tmpName).raw(), DeviceType_HardDisk, AccessMode_ReadWrite,
908 FALSE /* fForceNewUuid */, &harddisk) != VBOX_E_OBJECT_NOT_FOUND)
909 {
910 RTStrFree(tmpName);
911 char *tmpDir = RTStrDup(aName.c_str());
912 RTPathStripFilename(tmpDir);;
913 char *tmpFile = RTStrDup(RTPathFilename(aName.c_str()));
914 RTPathStripSuffix(tmpFile);
915 const char *pszTmpSuff = RTPathSuffix(aName.c_str());
916 RTStrAPrintf(&tmpName, "%s%c%s_%d%s", tmpDir, RTPATH_DELIMITER, tmpFile, i, pszTmpSuff);
917 RTStrFree(tmpFile);
918 RTStrFree(tmpDir);
919 ++i;
920 }
921 aName = tmpName;
922 RTStrFree(tmpName);
923
924 return S_OK;
925}
926
927/**
928 * Called from Appliance::importImpl() and Appliance::writeImpl() to set up a
929 * progress object with the proper weights and maximum progress values.
930 *
931 * @param pProgress
932 * @param bstrDescription
933 * @param mode
934 * @return
935 */
936HRESULT Appliance::i_setUpProgress(ComObjPtr<Progress> &pProgress,
937 const Bstr &bstrDescription,
938 SetUpProgressMode mode)
939{
940 HRESULT rc;
941
942 /* Create the progress object */
943 pProgress.createObject();
944
945 // compute the disks weight (this sets ulTotalDisksMB and cDisks in the instance data)
946 i_disksWeight();
947
948 m->ulWeightForManifestOperation = 0;
949
950 ULONG cOperations;
951 ULONG ulTotalOperationsWeight;
952
953 cOperations = 1 // one for XML setup
954 + m->cDisks; // plus one per disk
955 if (m->ulTotalDisksMB)
956 {
957 m->ulWeightForXmlOperation = (ULONG)((double)m->ulTotalDisksMB * 1 / 100); // use 1% of the progress for the XML
958 ulTotalOperationsWeight = m->ulTotalDisksMB + m->ulWeightForXmlOperation;
959 }
960 else
961 {
962 // no disks to export:
963 m->ulWeightForXmlOperation = 1;
964 ulTotalOperationsWeight = 1;
965 }
966
967 switch (mode)
968 {
969 case ImportFile:
970 {
971 break;
972 }
973 case WriteFile:
974 {
975 // assume that creating the manifest will take .1% of the time it takes to export the disks
976 if (m->fManifest)
977 {
978 ++cOperations; // another one for creating the manifest
979
980 m->ulWeightForManifestOperation = (ULONG)((double)m->ulTotalDisksMB * .1 / 100); // use .5% of the
981 // progress for the manifest
982 ulTotalOperationsWeight += m->ulWeightForManifestOperation;
983 }
984 break;
985 }
986 case ImportS3:
987 {
988 cOperations += 1 + 1; // another one for the manifest file & another one for the import
989 ulTotalOperationsWeight = m->ulTotalDisksMB;
990 if (!m->ulTotalDisksMB)
991 // no disks to export:
992 ulTotalOperationsWeight = 1;
993
994 ULONG ulImportWeight = (ULONG)((double)ulTotalOperationsWeight * 50 / 100); // use 50% for import
995 ulTotalOperationsWeight += ulImportWeight;
996
997 m->ulWeightForXmlOperation = ulImportWeight; /* save for using later */
998
999 ULONG ulInitWeight = (ULONG)((double)ulTotalOperationsWeight * 0.1 / 100); // use 0.1% for init
1000 ulTotalOperationsWeight += ulInitWeight;
1001 break;
1002 }
1003 case WriteS3:
1004 {
1005 cOperations += 1 + 1; // another one for the mf & another one for temporary creation
1006
1007 if (m->ulTotalDisksMB)
1008 {
1009 m->ulWeightForXmlOperation = (ULONG)((double)m->ulTotalDisksMB * 1 / 100); // use 1% of the progress
1010 // for OVF file upload
1011 // (we didn't know the
1012 // size at this point)
1013 ulTotalOperationsWeight = m->ulTotalDisksMB + m->ulWeightForXmlOperation;
1014 }
1015 else
1016 {
1017 // no disks to export:
1018 ulTotalOperationsWeight = 1;
1019 m->ulWeightForXmlOperation = 1;
1020 }
1021 ULONG ulOVFCreationWeight = (ULONG)((double)ulTotalOperationsWeight * 50.0 / 100.0); /* Use 50% for the
1022 creation of the OVF
1023 & the disks */
1024 ulTotalOperationsWeight += ulOVFCreationWeight;
1025 break;
1026 }
1027 }
1028 Log(("Setting up progress object: ulTotalMB = %d, cDisks = %d, => cOperations = %d, ulTotalOperationsWeight = %d, m->ulWeightForXmlOperation = %d\n",
1029 m->ulTotalDisksMB, m->cDisks, cOperations, ulTotalOperationsWeight, m->ulWeightForXmlOperation));
1030
1031 rc = pProgress->init(mVirtualBox, static_cast<IAppliance*>(this),
1032 bstrDescription.raw(),
1033 TRUE /* aCancelable */,
1034 cOperations, // ULONG cOperations,
1035 ulTotalOperationsWeight, // ULONG ulTotalOperationsWeight,
1036 bstrDescription.raw(), // CBSTR bstrFirstOperationDescription,
1037 m->ulWeightForXmlOperation); // ULONG ulFirstOperationWeight,
1038 return rc;
1039}
1040
1041/**
1042 * Called from the import and export background threads to synchronize the second
1043 * background disk thread's progress object with the current progress object so
1044 * that the user interface sees progress correctly and that cancel signals are
1045 * passed on to the second thread.
1046 * @param pProgressThis Progress object of the current thread.
1047 * @param pProgressAsync Progress object of asynchronous task running in background.
1048 */
1049void Appliance::i_waitForAsyncProgress(ComObjPtr<Progress> &pProgressThis,
1050 ComPtr<IProgress> &pProgressAsync)
1051{
1052 HRESULT rc;
1053
1054 // now loop until the asynchronous operation completes and then report its result
1055 BOOL fCompleted;
1056 BOOL fCanceled;
1057 ULONG currentPercent;
1058 ULONG cOp = 0;
1059 while (SUCCEEDED(pProgressAsync->COMGETTER(Completed(&fCompleted))))
1060 {
1061 rc = pProgressThis->COMGETTER(Canceled)(&fCanceled);
1062 if (FAILED(rc)) throw rc;
1063 if (fCanceled)
1064 pProgressAsync->Cancel();
1065 /* Check if the current operation has changed. It is also possible
1066 that in the meantime more than one async operation was finished. So
1067 we have to loop as long as we reached the same operation count. */
1068 ULONG curOp;
1069 for (;;)
1070 {
1071 rc = pProgressAsync->COMGETTER(Operation(&curOp));
1072 if (FAILED(rc)) throw rc;
1073 if (cOp != curOp)
1074 {
1075 Bstr bstr;
1076 ULONG currentWeight;
1077 rc = pProgressAsync->COMGETTER(OperationDescription(bstr.asOutParam()));
1078 if (FAILED(rc)) throw rc;
1079 rc = pProgressAsync->COMGETTER(OperationWeight(&currentWeight));
1080 if (FAILED(rc)) throw rc;
1081 rc = pProgressThis->SetNextOperation(bstr.raw(), currentWeight);
1082 if (FAILED(rc)) throw rc;
1083 ++cOp;
1084 }
1085 else
1086 break;
1087 }
1088
1089 rc = pProgressAsync->COMGETTER(OperationPercent(&currentPercent));
1090 if (FAILED(rc)) throw rc;
1091 pProgressThis->SetCurrentOperationProgress(currentPercent);
1092 if (fCompleted)
1093 break;
1094
1095 /* Make sure the loop is not too tight */
1096 rc = pProgressAsync->WaitForCompletion(100);
1097 if (FAILED(rc)) throw rc;
1098 }
1099 // report result of asynchronous operation
1100 LONG iRc;
1101 rc = pProgressAsync->COMGETTER(ResultCode)(&iRc);
1102 if (FAILED(rc)) throw rc;
1103
1104
1105 // if the thread of the progress object has an error, then
1106 // retrieve the error info from there, or it'll be lost
1107 if (FAILED(iRc))
1108 {
1109 ProgressErrorInfo info(pProgressAsync);
1110 Utf8Str str(info.getText());
1111 const char *pcsz = str.c_str();
1112 HRESULT rc2 = setError(iRc, pcsz);
1113 throw rc2;
1114 }
1115}
1116
1117void Appliance::i_addWarning(const char* aWarning, ...)
1118{
1119 va_list args;
1120 va_start(args, aWarning);
1121 Utf8Str str(aWarning, args);
1122 va_end(args);
1123 m->llWarnings.push_back(str);
1124}
1125
1126/**
1127 * Refreshes the cDisks and ulTotalDisksMB members in the instance data.
1128 * Requires that virtual system descriptions are present.
1129 */
1130void Appliance::i_disksWeight()
1131{
1132 m->ulTotalDisksMB = 0;
1133 m->cDisks = 0;
1134 // weigh the disk images according to their sizes
1135 list< ComObjPtr<VirtualSystemDescription> >::const_iterator it;
1136 for (it = m->virtualSystemDescriptions.begin();
1137 it != m->virtualSystemDescriptions.end();
1138 ++it)
1139 {
1140 ComObjPtr<VirtualSystemDescription> vsdescThis = (*it);
1141 /* One for every hard disk of the Virtual System */
1142 std::list<VirtualSystemDescriptionEntry*> avsdeHDs = vsdescThis->i_findByType(VirtualSystemDescriptionType_HardDiskImage);
1143 std::list<VirtualSystemDescriptionEntry*>::const_iterator itH;
1144 for (itH = avsdeHDs.begin();
1145 itH != avsdeHDs.end();
1146 ++itH)
1147 {
1148 const VirtualSystemDescriptionEntry *pHD = *itH;
1149 m->ulTotalDisksMB += pHD->ulSizeMB;
1150 ++m->cDisks;
1151 }
1152
1153 avsdeHDs = vsdescThis->i_findByType(VirtualSystemDescriptionType_CDROM);
1154 for (itH = avsdeHDs.begin();
1155 itH != avsdeHDs.end();
1156 ++itH)
1157 {
1158 const VirtualSystemDescriptionEntry *pHD = *itH;
1159 m->ulTotalDisksMB += pHD->ulSizeMB;
1160 ++m->cDisks;
1161 }
1162 }
1163
1164}
1165
1166void Appliance::i_parseBucket(Utf8Str &aPath, Utf8Str &aBucket)
1167{
1168 /* Buckets are S3 specific. So parse the bucket out of the file path */
1169 if (!aPath.startsWith("/"))
1170 throw setError(E_INVALIDARG,
1171 tr("The path '%s' must start with /"), aPath.c_str());
1172 size_t bpos = aPath.find("/", 1);
1173 if (bpos != Utf8Str::npos)
1174 {
1175 aBucket = aPath.substr(1, bpos - 1); /* The bucket without any slashes */
1176 aPath = aPath.substr(bpos); /* The rest of the file path */
1177 }
1178 /* If there is no bucket name provided reject it */
1179 if (aBucket.isEmpty())
1180 throw setError(E_INVALIDARG,
1181 tr("You doesn't provide a bucket name in the URI '%s'"), aPath.c_str());
1182}
1183
1184/**
1185 * Starts the worker thread for the task.
1186 *
1187 * @return COM status code.
1188 */
1189HRESULT Appliance::TaskOVF::startThread()
1190{
1191 /* Pick a thread name suitable for logging (<= 8 chars). */
1192 const char *pszTaskNm;
1193 switch (taskType)
1194 {
1195 case TaskOVF::Read: pszTaskNm = "ApplRead"; break;
1196 case TaskOVF::Import: pszTaskNm = "ApplImp"; break;
1197 case TaskOVF::Write: pszTaskNm = "ApplWrit"; break;
1198 default: pszTaskNm = "ApplTask"; break;
1199 }
1200
1201 int vrc = RTThreadCreate(NULL, Appliance::i_taskThreadImportOrExport, this,
1202 0, RTTHREADTYPE_MAIN_HEAVY_WORKER, 0, pszTaskNm);
1203 if (RT_SUCCESS(vrc))
1204 return S_OK;
1205 return Appliance::i_setErrorStatic(E_FAIL, Utf8StrFmt("Could not create OVF task thread (%Rrc)\n", vrc));
1206}
1207
1208/**
1209 * Thread function for the thread started in Appliance::readImpl() and Appliance::importImpl()
1210 * and Appliance::writeImpl().
1211 * This will in turn call Appliance::readFS() or Appliance::readS3() or Appliance::importFS()
1212 * or Appliance::importS3() or Appliance::writeFS() or Appliance::writeS3().
1213 *
1214 * @param aThread
1215 * @param pvUser
1216 */
1217/* static */
1218DECLCALLBACK(int) Appliance::i_taskThreadImportOrExport(RTTHREAD /* aThread */, void *pvUser)
1219{
1220 std::auto_ptr<TaskOVF> task(static_cast<TaskOVF*>(pvUser));
1221 AssertReturn(task.get(), VERR_GENERAL_FAILURE);
1222
1223 Appliance *pAppliance = task->pAppliance;
1224
1225 LogFlowFuncEnter();
1226 LogFlowFunc(("Appliance %p taskType=%d\n", pAppliance, task->taskType));
1227
1228 HRESULT taskrc = S_OK;
1229
1230 switch (task->taskType)
1231 {
1232 case TaskOVF::Read:
1233 if (task->locInfo.storageType == VFSType_File)
1234 taskrc = pAppliance->i_readFS(task.get());
1235 else if (task->locInfo.storageType == VFSType_S3)
1236#ifdef VBOX_WITH_S3
1237 taskrc = pAppliance->i_readS3(task.get());
1238#else
1239 taskrc = VERR_NOT_IMPLEMENTED;
1240#endif
1241 break;
1242
1243 case TaskOVF::Import:
1244 if (task->locInfo.storageType == VFSType_File)
1245 taskrc = pAppliance->i_importFS(task.get());
1246 else if (task->locInfo.storageType == VFSType_S3)
1247#ifdef VBOX_WITH_S3
1248 taskrc = pAppliance->i_importS3(task.get());
1249#else
1250 taskrc = VERR_NOT_IMPLEMENTED;
1251#endif
1252 break;
1253
1254 case TaskOVF::Write:
1255 if (task->locInfo.storageType == VFSType_File)
1256 taskrc = pAppliance->i_writeFS(task.get());
1257 else if (task->locInfo.storageType == VFSType_S3)
1258#ifdef VBOX_WITH_S3
1259 taskrc = pAppliance->i_writeS3(task.get());
1260#else
1261 taskrc = VERR_NOT_IMPLEMENTED;
1262#endif
1263 break;
1264 }
1265
1266 task->rc = taskrc;
1267
1268 if (!task->pProgress.isNull())
1269 task->pProgress->i_notifyComplete(taskrc);
1270
1271 LogFlowFuncLeave();
1272
1273 return VINF_SUCCESS;
1274}
1275
1276/* static */
1277int Appliance::TaskOVF::updateProgress(unsigned uPercent, void *pvUser)
1278{
1279 Appliance::TaskOVF* pTask = *(Appliance::TaskOVF**)pvUser;
1280
1281 if ( pTask
1282 && !pTask->pProgress.isNull())
1283 {
1284 BOOL fCanceled;
1285 pTask->pProgress->COMGETTER(Canceled)(&fCanceled);
1286 if (fCanceled)
1287 return -1;
1288 pTask->pProgress->SetCurrentOperationProgress(uPercent);
1289 }
1290 return VINF_SUCCESS;
1291}
1292
1293void i_parseURI(Utf8Str strUri, LocationInfo &locInfo)
1294{
1295 /* Check the URI for the protocol */
1296 if (strUri.startsWith("file://", Utf8Str::CaseInsensitive)) /* File based */
1297 {
1298 locInfo.storageType = VFSType_File;
1299 strUri = strUri.substr(sizeof("file://") - 1);
1300 }
1301 else if (strUri.startsWith("SunCloud://", Utf8Str::CaseInsensitive)) /* Sun Cloud service */
1302 {
1303 locInfo.storageType = VFSType_S3;
1304 strUri = strUri.substr(sizeof("SunCloud://") - 1);
1305 }
1306 else if (strUri.startsWith("S3://", Utf8Str::CaseInsensitive)) /* S3 service */
1307 {
1308 locInfo.storageType = VFSType_S3;
1309 strUri = strUri.substr(sizeof("S3://") - 1);
1310 }
1311 else if (strUri.startsWith("webdav://", Utf8Str::CaseInsensitive)) /* webdav service */
1312 throw E_NOTIMPL;
1313
1314 /* Not necessary on a file based URI */
1315 if (locInfo.storageType != VFSType_File)
1316 {
1317 size_t uppos = strUri.find("@"); /* username:password combo */
1318 if (uppos != Utf8Str::npos)
1319 {
1320 locInfo.strUsername = strUri.substr(0, uppos);
1321 strUri = strUri.substr(uppos + 1);
1322 size_t upos = locInfo.strUsername.find(":");
1323 if (upos != Utf8Str::npos)
1324 {
1325 locInfo.strPassword = locInfo.strUsername.substr(upos + 1);
1326 locInfo.strUsername = locInfo.strUsername.substr(0, upos);
1327 }
1328 }
1329 size_t hpos = strUri.find("/"); /* hostname part */
1330 if (hpos != Utf8Str::npos)
1331 {
1332 locInfo.strHostname = strUri.substr(0, hpos);
1333 strUri = strUri.substr(hpos);
1334 }
1335 }
1336
1337 locInfo.strPath = strUri;
1338}
1339
1340////////////////////////////////////////////////////////////////////////////////
1341//
1342// IVirtualSystemDescription constructor / destructor
1343//
1344////////////////////////////////////////////////////////////////////////////////
1345
1346
1347/**
1348 * COM initializer.
1349 * @return
1350 */
1351HRESULT VirtualSystemDescription::init()
1352{
1353 /* Enclose the state transition NotReady->InInit->Ready */
1354 AutoInitSpan autoInitSpan(this);
1355 AssertReturn(autoInitSpan.isOk(), E_FAIL);
1356
1357 /* Initialize data */
1358 m = new Data();
1359 m->pConfig = NULL;
1360
1361 /* Confirm a successful initialization */
1362 autoInitSpan.setSucceeded();
1363 return S_OK;
1364}
1365
1366/**
1367* COM uninitializer.
1368*/
1369
1370void VirtualSystemDescription::uninit()
1371{
1372 if (m->pConfig)
1373 delete m->pConfig;
1374 delete m;
1375 m = NULL;
1376}
1377
1378////////////////////////////////////////////////////////////////////////////////
1379//
1380// IVirtualSystemDescription public methods
1381//
1382////////////////////////////////////////////////////////////////////////////////
1383
1384/**
1385 * Public method implementation.
1386 * @param
1387 * @return
1388 */
1389HRESULT VirtualSystemDescription::getCount(ULONG *aCount)
1390{
1391 if (!aCount)
1392 return E_POINTER;
1393
1394 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1395
1396 *aCount = (ULONG)m->maDescriptions.size();
1397 return S_OK;
1398}
1399
1400/**
1401 * Public method implementation.
1402 * @return
1403 */
1404HRESULT VirtualSystemDescription::getDescription(std::vector<VirtualSystemDescriptionType_T> &aTypes,
1405 std::vector<com::Utf8Str> &aRefs,
1406 std::vector<com::Utf8Str> &aOVFValues,
1407 std::vector<com::Utf8Str> &aVBoxValues,
1408 std::vector<com::Utf8Str> &aExtraConfigValues)
1409
1410{
1411 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1412 size_t c = m->maDescriptions.size();
1413 aTypes.resize(c);
1414 aRefs.resize(c);
1415 aOVFValues.resize(c);
1416 aVBoxValues.resize(c);
1417 aExtraConfigValues.resize(c);
1418
1419 for (size_t i = 0; i < c; i++)
1420 {
1421 const VirtualSystemDescriptionEntry &vsde = m->maDescriptions[i];
1422 aTypes[i] = vsde.type;
1423 aRefs[i] = vsde.strRef;
1424 aOVFValues[i] = vsde.strOvf;
1425 aVBoxValues[i] = vsde.strVBoxCurrent;
1426 aExtraConfigValues[i] = vsde.strExtraConfigCurrent;
1427 }
1428 return S_OK;
1429}
1430
1431/**
1432 * Public method implementation.
1433 * @return
1434 */
1435HRESULT VirtualSystemDescription::getDescriptionByType(VirtualSystemDescriptionType_T aType,
1436 std::vector<VirtualSystemDescriptionType_T> &aTypes,
1437 std::vector<com::Utf8Str> &aRefs,
1438 std::vector<com::Utf8Str> &aOVFValues,
1439 std::vector<com::Utf8Str> &aVBoxValues,
1440 std::vector<com::Utf8Str> &aExtraConfigValues)
1441{
1442 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1443 std::list<VirtualSystemDescriptionEntry*> vsd = i_findByType(aType);
1444
1445 size_t c = vsd.size();
1446 aTypes.resize(c);
1447 aRefs.resize(c);
1448 aOVFValues.resize(c);
1449 aVBoxValues.resize(c);
1450 aExtraConfigValues.resize(c);
1451
1452 size_t i = 0;
1453 for (list<VirtualSystemDescriptionEntry*>::const_iterator it = vsd.begin(); it != vsd.end(); ++it, ++i)
1454 {
1455 const VirtualSystemDescriptionEntry *vsde = (*it);
1456 aTypes[i] = vsde->type;
1457 aRefs[i] = vsde->strRef;
1458 aOVFValues[i] = vsde->strOvf;
1459 aVBoxValues[i] = vsde->strVBoxCurrent;
1460 aExtraConfigValues[i] = vsde->strExtraConfigCurrent;
1461 }
1462
1463 return S_OK;
1464}
1465
1466/**
1467 * Public method implementation.
1468 * @return
1469 */
1470HRESULT VirtualSystemDescription::getValuesByType(VirtualSystemDescriptionType_T aType,
1471 VirtualSystemDescriptionValueType_T aWhich,
1472 std::vector<com::Utf8Str> &aValues)
1473{
1474 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS);
1475
1476 std::list<VirtualSystemDescriptionEntry*> vsd = i_findByType (aType);
1477 aValues.resize((ULONG)vsd.size());
1478
1479 list<VirtualSystemDescriptionEntry*>::const_iterator it;
1480 size_t i = 0;
1481 for (it = vsd.begin();
1482 it != vsd.end();
1483 ++it, ++i)
1484 {
1485 const VirtualSystemDescriptionEntry *vsde = (*it);
1486
1487 Bstr bstr;
1488 switch (aWhich)
1489 {
1490 case VirtualSystemDescriptionValueType_Reference: aValues[i] = vsde->strRef; break;
1491 case VirtualSystemDescriptionValueType_Original: aValues[i] = vsde->strOvf; break;
1492 case VirtualSystemDescriptionValueType_Auto: aValues[i] = vsde->strVBoxCurrent; break;
1493 case VirtualSystemDescriptionValueType_ExtraConfig: aValues[i] = vsde->strExtraConfigCurrent; break;
1494 }
1495 }
1496
1497 return S_OK;
1498}
1499
1500/**
1501 * Public method implementation.
1502 * @return
1503 */
1504HRESULT VirtualSystemDescription::setFinalValues(const std::vector<BOOL> &aEnabled,
1505 const std::vector<com::Utf8Str> &aVBoxValues,
1506 const std::vector<com::Utf8Str> &aExtraConfigValues)
1507{
1508#ifndef RT_OS_WINDOWS
1509 // NOREF(aEnabledSize);
1510#endif /* RT_OS_WINDOWS */
1511 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1512
1513 if ( (aEnabled.size() != m->maDescriptions.size())
1514 || (aVBoxValues.size() != m->maDescriptions.size())
1515 || (aExtraConfigValues.size() != m->maDescriptions.size())
1516 )
1517 return E_INVALIDARG;
1518
1519 size_t i = 0;
1520 for (vector<VirtualSystemDescriptionEntry>::iterator it = m->maDescriptions.begin();
1521 it != m->maDescriptions.end();
1522 ++it, ++i)
1523 {
1524 VirtualSystemDescriptionEntry& vsde = *it;
1525
1526 if (aEnabled[i])
1527 {
1528 vsde.strVBoxCurrent = aVBoxValues[i];
1529 vsde.strExtraConfigCurrent = aExtraConfigValues[i];
1530 }
1531 else
1532 vsde.type = VirtualSystemDescriptionType_Ignore;
1533 }
1534
1535 return S_OK;
1536}
1537
1538/**
1539 * Public method implementation.
1540 * @return
1541 */
1542HRESULT VirtualSystemDescription::addDescription(VirtualSystemDescriptionType_T aType,
1543 const com::Utf8Str &aVBoxValue,
1544 const com::Utf8Str &aExtraConfigValue)
1545
1546{
1547 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS);
1548 i_addEntry(aType, "", aVBoxValue, aVBoxValue, 0, aExtraConfigValue);
1549 return S_OK;
1550}
1551
1552/**
1553 * Internal method; adds a new description item to the member list.
1554 * @param aType Type of description for the new item.
1555 * @param strRef Reference item; only used with hard disk controllers.
1556 * @param aOrigValue Corresponding original value from OVF.
1557 * @param aAutoValue Initial configuration value (can be overridden by caller with setFinalValues).
1558 * @param ulSizeMB Weight for IProgress
1559 * @param strExtraConfig Extra configuration; meaning dependent on type.
1560 */
1561void VirtualSystemDescription::i_addEntry(VirtualSystemDescriptionType_T aType,
1562 const Utf8Str &strRef,
1563 const Utf8Str &aOvfValue,
1564 const Utf8Str &aVBoxValue,
1565 uint32_t ulSizeMB,
1566 const Utf8Str &strExtraConfig /*= ""*/)
1567{
1568 VirtualSystemDescriptionEntry vsde;
1569 vsde.ulIndex = (uint32_t)m->maDescriptions.size(); // each entry gets an index so the client side can reference them
1570 vsde.type = aType;
1571 vsde.strRef = strRef;
1572 vsde.strOvf = aOvfValue;
1573 vsde.strVBoxSuggested // remember original value
1574 = vsde.strVBoxCurrent // and set current value which can be overridden by setFinalValues()
1575 = aVBoxValue;
1576 vsde.strExtraConfigSuggested
1577 = vsde.strExtraConfigCurrent
1578 = strExtraConfig;
1579 vsde.ulSizeMB = ulSizeMB;
1580
1581 vsde.skipIt = false;
1582
1583 m->maDescriptions.push_back(vsde);
1584}
1585
1586/**
1587 * Private method; returns a list of description items containing all the items from the member
1588 * description items of this virtual system that match the given type.
1589 * @param aType
1590 * @return
1591 */
1592std::list<VirtualSystemDescriptionEntry*> VirtualSystemDescription::i_findByType(VirtualSystemDescriptionType_T aType)
1593{
1594 std::list<VirtualSystemDescriptionEntry*> vsd;
1595 for (vector<VirtualSystemDescriptionEntry>::iterator it = m->maDescriptions.begin();
1596 it != m->maDescriptions.end();
1597 ++it)
1598 {
1599 if (it->type == aType)
1600 vsd.push_back(&(*it));
1601 }
1602
1603 return vsd;
1604}
1605
1606/* Private method; delete all records from the list
1607 * m->llDescriptions that match the given type.
1608 * @param aType
1609 * @return
1610 */
1611void VirtualSystemDescription::i_removeByType(VirtualSystemDescriptionType_T aType)
1612{
1613 std::vector<VirtualSystemDescriptionEntry>::iterator it = m->maDescriptions.begin();
1614 while (it != m->maDescriptions.end())
1615 {
1616 if (it->type == aType)
1617 it = m->maDescriptions.erase(it);
1618 else
1619 ++it;
1620 }
1621}
1622
1623/**
1624 * Private method; looks thru the member hardware items for the IDE, SATA, or SCSI controller with
1625 * the given reference ID. Useful when needing the controller for a particular
1626 * virtual disk.
1627 * @param id
1628 * @return
1629 */
1630const VirtualSystemDescriptionEntry* VirtualSystemDescription::i_findControllerFromID(uint32_t id)
1631{
1632 Utf8Str strRef = Utf8StrFmt("%RI32", id);
1633 vector<VirtualSystemDescriptionEntry>::const_iterator it;
1634 for (it = m->maDescriptions.begin();
1635 it != m->maDescriptions.end();
1636 ++it)
1637 {
1638 const VirtualSystemDescriptionEntry &d = *it;
1639 switch (d.type)
1640 {
1641 case VirtualSystemDescriptionType_HardDiskControllerIDE:
1642 case VirtualSystemDescriptionType_HardDiskControllerSATA:
1643 case VirtualSystemDescriptionType_HardDiskControllerSCSI:
1644 case VirtualSystemDescriptionType_HardDiskControllerSAS:
1645 if (d.strRef == strRef)
1646 return &d;
1647 break;
1648 }
1649 }
1650
1651 return NULL;
1652}
1653
1654/**
1655 * Method called from Appliance::Interpret() if the source OVF for a virtual system
1656 * contains a <vbox:Machine> element. This method then attempts to parse that and
1657 * create a MachineConfigFile instance from it which is stored in this instance data
1658 * and can then be used to create a machine.
1659 *
1660 * This must only be called once per instance.
1661 *
1662 * This rethrows all XML and logic errors from MachineConfigFile.
1663 *
1664 * @param elmMachine <vbox:Machine> element with attributes and subelements from some
1665 * DOM tree.
1666 */
1667void VirtualSystemDescription::i_importVBoxMachineXML(const xml::ElementNode &elmMachine)
1668{
1669 settings::MachineConfigFile *pConfig = NULL;
1670
1671 Assert(m->pConfig == NULL);
1672
1673 try
1674 {
1675 pConfig = new settings::MachineConfigFile(NULL);
1676 pConfig->importMachineXML(elmMachine);
1677
1678 m->pConfig = pConfig;
1679 }
1680 catch (...)
1681 {
1682 if (pConfig)
1683 delete pConfig;
1684 throw;
1685 }
1686}
1687
1688/**
1689 * Returns the machine config created by importVBoxMachineXML() or NULL if there's none.
1690 * @return
1691 */
1692const settings::MachineConfigFile* VirtualSystemDescription::i_getMachineConfig() const
1693{
1694 return m->pConfig;
1695}
1696
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