VirtualBox

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

Last change on this file since 68071 was 68016, checked in by vboxsync, 7 years ago

VirtualBox::createAppliance: Added two missing return code checks/pickups. sigh.

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