VirtualBox

source: vbox/trunk/src/VBox/Frontends/VBoxManage/VBoxManageImport.cpp@ 20554

Last change on this file since 20554 was 20221, checked in by vboxsync, 16 years ago

VBoxManage: IProgess::GetResultCode() gcc warnings

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 41.4 KB
Line 
1/* $Id: VBoxManageImport.cpp 20221 2009-06-03 08:54:39Z vboxsync $ */
2/** @file
3 * VBoxManage - The appliance-related commands.
4 */
5
6/*
7 * Copyright (C) 2009 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22#ifndef VBOX_ONLY_DOCS
23
24/*******************************************************************************
25* Header Files *
26*******************************************************************************/
27#ifndef VBOX_ONLY_DOCS
28#include <VBox/com/com.h>
29#include <VBox/com/string.h>
30#include <VBox/com/Guid.h>
31#include <VBox/com/array.h>
32#include <VBox/com/ErrorInfo.h>
33#include <VBox/com/errorprint2.h>
34#include <VBox/com/EventQueue.h>
35
36#include <VBox/com/VirtualBox.h>
37
38#include <list>
39#include <map>
40#endif /* !VBOX_ONLY_DOCS */
41
42#include <iprt/stream.h>
43#include <iprt/getopt.h>
44#include <iprt/ctype.h>
45#include <iprt/path.h>
46#include <iprt/file.h>
47
48#include <VBox/log.h>
49
50#include "VBoxManage.h"
51using namespace com;
52
53
54// funcs
55///////////////////////////////////////////////////////////////////////////////
56
57typedef std::map<Utf8Str, Utf8Str> ArgsMap; // pairs of strings like "vmname" => "newvmname"
58typedef std::map<uint32_t, ArgsMap> ArgsMapsMap; // map of maps, one for each virtual system, sorted by index
59
60typedef std::map<uint32_t, bool> IgnoresMap; // pairs of numeric description entry indices
61typedef std::map<uint32_t, IgnoresMap> IgnoresMapsMap; // map of maps, one for each virtual system, sorted by index
62
63static bool findArgValue(Utf8Str &strOut,
64 ArgsMap *pmapArgs,
65 const Utf8Str &strKey)
66{
67 if (pmapArgs)
68 {
69 ArgsMap::iterator it;
70 it = pmapArgs->find(strKey);
71 if (it != pmapArgs->end())
72 {
73 strOut = it->second;
74 pmapArgs->erase(it);
75 return true;
76 }
77 }
78
79 return false;
80}
81
82static const RTGETOPTDEF g_aImportApplianceOptions[] =
83{
84 { "--dry-run", 'n', RTGETOPT_REQ_NOTHING },
85 { "-dry-run", 'n', RTGETOPT_REQ_NOTHING }, // deprecated
86 { "--dryrun", 'n', RTGETOPT_REQ_NOTHING },
87 { "-dryrun", 'n', RTGETOPT_REQ_NOTHING }, // deprecated
88 { "--detailed-progress", 'P', RTGETOPT_REQ_NOTHING },
89 { "-detailed-progress", 'P', RTGETOPT_REQ_NOTHING }, // deprecated
90 { "--vsys", 's', RTGETOPT_REQ_UINT32 },
91 { "-vsys", 's', RTGETOPT_REQ_UINT32 }, // deprecated
92 { "--ostype", 'o', RTGETOPT_REQ_STRING },
93 { "-ostype", 'o', RTGETOPT_REQ_STRING }, // deprecated
94 { "--vmname", 'V', RTGETOPT_REQ_STRING },
95 { "-vmname", 'V', RTGETOPT_REQ_STRING }, // deprecated
96 { "--description", 'd', RTGETOPT_REQ_STRING },
97 { "--eula", 'L', RTGETOPT_REQ_STRING },
98 { "-eula", 'L', RTGETOPT_REQ_STRING }, // deprecated
99 { "--unit", 'u', RTGETOPT_REQ_UINT32 },
100 { "-unit", 'u', RTGETOPT_REQ_UINT32 }, // deprecated
101 { "--ignore", 'x', RTGETOPT_REQ_NOTHING },
102 { "-ignore", 'x', RTGETOPT_REQ_NOTHING }, // deprecated
103 { "--scsitype", 'T', RTGETOPT_REQ_UINT32 },
104 { "-scsitype", 'T', RTGETOPT_REQ_UINT32 }, // deprecated
105 { "--type", 'T', RTGETOPT_REQ_UINT32 }, // deprecated
106 { "-type", 'T', RTGETOPT_REQ_UINT32 }, // deprecated
107};
108
109int handleImportAppliance(HandlerArg *a)
110{
111 HRESULT rc = S_OK;
112
113 Utf8Str strOvfFilename;
114 bool fExecute = true; // if true, then we actually do the import
115 uint32_t ulCurVsys = (uint32_t)-1;
116 uint32_t ulCurUnit = (uint32_t)-1;
117 // for each --vsys X command, maintain a map of command line items
118 // (we'll parse them later after interpreting the OVF, when we can
119 // actually check whether they make sense semantically)
120 ArgsMapsMap mapArgsMapsPerVsys;
121 IgnoresMapsMap mapIgnoresMapsPerVsys;
122
123 int c;
124 RTGETOPTUNION ValueUnion;
125 RTGETOPTSTATE GetState;
126 // start at 0 because main() has hacked both the argc and argv given to us
127 RTGetOptInit(&GetState, a->argc, a->argv, g_aImportApplianceOptions, RT_ELEMENTS(g_aImportApplianceOptions), 0, 0 /* fFlags */);
128 while ((c = RTGetOpt(&GetState, &ValueUnion)))
129 {
130 switch (c)
131 {
132 case 'n': // --dry-run
133 fExecute = false;
134 break;
135
136 case 'P': // --detailed-progress
137 g_fDetailedProgress = true;
138 break;
139
140 case 's': // --vsys
141 ulCurVsys = ValueUnion.u32;
142 ulCurUnit = (uint32_t)-1;
143 break;
144
145 case 'o': // --ostype
146 if (ulCurVsys == (uint32_t)-1)
147 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
148 mapArgsMapsPerVsys[ulCurVsys]["ostype"] = ValueUnion.psz;
149 break;
150
151 case 'V': // --vmname
152 if (ulCurVsys == (uint32_t)-1)
153 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
154 mapArgsMapsPerVsys[ulCurVsys]["vmname"] = ValueUnion.psz;
155 break;
156
157 case 'd': // --description
158 if (ulCurVsys == (uint32_t)-1)
159 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
160 mapArgsMapsPerVsys[ulCurVsys]["description"] = ValueUnion.psz;
161 break;
162
163 case 'L': // --eula
164 if (ulCurVsys == (uint32_t)-1)
165 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
166 mapArgsMapsPerVsys[ulCurVsys]["eula"] = ValueUnion.psz;
167 break;
168
169 case 'm': // --memory
170 if (ulCurVsys == (uint32_t)-1)
171 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
172 mapArgsMapsPerVsys[ulCurVsys]["memory"] = ValueUnion.psz;
173 break;
174
175 case 'u': // --unit
176 ulCurUnit = ValueUnion.u32;
177 break;
178
179 case 'x': // --ignore
180 if (ulCurVsys == (uint32_t)-1)
181 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
182 if (ulCurUnit == (uint32_t)-1)
183 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --unit argument.", GetState.pDef->pszLong);
184 mapIgnoresMapsPerVsys[ulCurVsys][ulCurUnit] = true;
185 break;
186
187 case 'T': // --scsitype
188 if (ulCurVsys == (uint32_t)-1)
189 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
190 if (ulCurUnit == (uint32_t)-1)
191 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --unit argument.", GetState.pDef->pszLong);
192 mapArgsMapsPerVsys[ulCurVsys][Utf8StrFmt("scsitype%u", ulCurUnit)] = ValueUnion.psz;
193 break;
194
195 case 'C': // --controller
196 if (ulCurVsys == (uint32_t)-1)
197 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
198 if (ulCurUnit == (uint32_t)-1)
199 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Option \"%s\" requires preceding --unit argument.", GetState.pDef->pszLong);
200 mapArgsMapsPerVsys[ulCurVsys][Utf8StrFmt("controller%u", ulCurUnit)] = ValueUnion.psz;
201 break;
202
203 case VINF_GETOPT_NOT_OPTION:
204 if (!strOvfFilename)
205 strOvfFilename = ValueUnion.psz;
206 else
207 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Invalid parameter '%s'", ValueUnion.psz);
208 break;
209
210 default:
211 if (c > 0)
212 {
213 if (RT_C_IS_PRINT(c))
214 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Invalid option -%c", c);
215 else
216 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Invalid option case %i", c);
217 }
218 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
219 return errorSyntax(USAGE_IMPORTAPPLIANCE, "unknown option: %s\n", ValueUnion.psz);
220 else if (ValueUnion.pDef)
221 return errorSyntax(USAGE_IMPORTAPPLIANCE, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
222 else
223 return errorSyntax(USAGE_IMPORTAPPLIANCE, "error: %Rrs", c);
224 }
225 }
226
227 if (!strOvfFilename)
228 return errorSyntax(USAGE_IMPORTAPPLIANCE, "Not enough arguments for \"import\" command.");
229
230 do
231 {
232 ComPtr<IAppliance> pAppliance;
233 CHECK_ERROR_BREAK(a->virtualBox, CreateAppliance(pAppliance.asOutParam()));
234
235 char *pszAbsFilePath = RTPathAbsDup(strOvfFilename.c_str());
236 CHECK_ERROR_BREAK(pAppliance, Read(Bstr(pszAbsFilePath)));
237 RTStrFree(pszAbsFilePath);
238
239 // call interpret(); this can yield both warnings and errors, so we need
240 // to tinker with the error info a bit
241 RTPrintf("Interpreting %s...\n", strOvfFilename.c_str());
242 rc = pAppliance->Interpret();
243 com::ErrorInfo info0(pAppliance);
244
245 com::SafeArray<BSTR> aWarnings;
246 if (SUCCEEDED(pAppliance->GetWarnings(ComSafeArrayAsOutParam(aWarnings))))
247 {
248 size_t cWarnings = aWarnings.size();
249 for (unsigned i = 0; i < cWarnings; ++i)
250 {
251 Bstr bstrWarning(aWarnings[i]);
252 RTPrintf("WARNING: %ls.\n", bstrWarning.raw());
253 }
254 }
255
256 if (FAILED(rc)) // during interpret, after printing warnings
257 {
258 com::GluePrintErrorInfo(info0);
259 com::GluePrintErrorContext("Interpret", __FILE__, __LINE__);
260 break;
261 }
262
263 RTPrintf("OK.\n");
264
265 // fetch all disks
266 com::SafeArray<BSTR> retDisks;
267 CHECK_ERROR_BREAK(pAppliance,
268 COMGETTER(Disks)(ComSafeArrayAsOutParam(retDisks)));
269 if (retDisks.size() > 0)
270 {
271 RTPrintf("Disks:");
272 for (unsigned i = 0; i < retDisks.size(); i++)
273 RTPrintf(" %ls", retDisks[i]);
274 RTPrintf("\n");
275 }
276
277 // fetch virtual system descriptions
278 com::SafeIfaceArray<IVirtualSystemDescription> aVirtualSystemDescriptions;
279 CHECK_ERROR_BREAK(pAppliance,
280 COMGETTER(VirtualSystemDescriptions)(ComSafeArrayAsOutParam(aVirtualSystemDescriptions)));
281
282 size_t cVirtualSystemDescriptions = aVirtualSystemDescriptions.size();
283
284 // match command line arguments with virtual system descriptions;
285 // this is only to sort out invalid indices at this time
286 ArgsMapsMap::const_iterator it;
287 for (it = mapArgsMapsPerVsys.begin();
288 it != mapArgsMapsPerVsys.end();
289 ++it)
290 {
291 uint32_t ulVsys = it->first;
292 if (ulVsys >= cVirtualSystemDescriptions)
293 return errorSyntax(USAGE_IMPORTAPPLIANCE,
294 "Invalid index %RI32 with -vsys option; the OVF contains only %zu virtual system(s).",
295 ulVsys, cVirtualSystemDescriptions);
296 }
297
298 uint32_t cLicensesInTheWay = 0;
299
300 // dump virtual system descriptions and match command-line arguments
301 if (cVirtualSystemDescriptions > 0)
302 {
303 for (unsigned i = 0; i < cVirtualSystemDescriptions; ++i)
304 {
305 com::SafeArray<VirtualSystemDescriptionType_T> retTypes;
306 com::SafeArray<BSTR> aRefs;
307 com::SafeArray<BSTR> aOvfValues;
308 com::SafeArray<BSTR> aVboxValues;
309 com::SafeArray<BSTR> aExtraConfigValues;
310 CHECK_ERROR_BREAK(aVirtualSystemDescriptions[i],
311 GetDescription(ComSafeArrayAsOutParam(retTypes),
312 ComSafeArrayAsOutParam(aRefs),
313 ComSafeArrayAsOutParam(aOvfValues),
314 ComSafeArrayAsOutParam(aVboxValues),
315 ComSafeArrayAsOutParam(aExtraConfigValues)));
316
317 RTPrintf("Virtual system %u:\n", i);
318
319 // look up the corresponding command line options, if any
320 ArgsMap *pmapArgs = NULL;
321 ArgsMapsMap::iterator itm = mapArgsMapsPerVsys.find(i);
322 if (itm != mapArgsMapsPerVsys.end())
323 pmapArgs = &itm->second;
324
325 // this collects the final values for setFinalValues()
326 com::SafeArray<BOOL> aEnabled(retTypes.size());
327 com::SafeArray<BSTR> aFinalValues(retTypes.size());
328
329 for (unsigned a = 0; a < retTypes.size(); ++a)
330 {
331 VirtualSystemDescriptionType_T t = retTypes[a];
332
333 Utf8Str strOverride;
334
335 Bstr bstrFinalValue = aVboxValues[a];
336
337 bool fIgnoreThis = mapIgnoresMapsPerVsys[i][a];
338
339 aEnabled[a] = true;
340
341 switch (t)
342 {
343 case VirtualSystemDescriptionType_OS:
344 if (findArgValue(strOverride, pmapArgs, "ostype"))
345 {
346 bstrFinalValue = strOverride;
347 RTPrintf("%2u: OS type specified with --ostype: \"%ls\"\n",
348 a, bstrFinalValue.raw());
349 }
350 else
351 RTPrintf("%2u: Suggested OS type: \"%ls\""
352 "\n (change with \"--vsys %u --ostype <type>\"; use \"list ostypes\" to list all possible values)\n",
353 a, bstrFinalValue.raw(), i);
354 break;
355
356 case VirtualSystemDescriptionType_Name:
357 if (findArgValue(strOverride, pmapArgs, "vmname"))
358 {
359 bstrFinalValue = strOverride;
360 RTPrintf("%2u: VM name specified with --vmname: \"%ls\"\n",
361 a, bstrFinalValue.raw());
362 }
363 else
364 RTPrintf("%2u: Suggested VM name \"%ls\""
365 "\n (change with \"--vsys %u --vmname <name>\")\n",
366 a, bstrFinalValue.raw(), i);
367 break;
368
369 case VirtualSystemDescriptionType_Product:
370 RTPrintf("%2u: Product (ignored): %ls\n",
371 a, aVboxValues[a]);
372 break;
373
374 case VirtualSystemDescriptionType_ProductUrl:
375 RTPrintf("%2u: ProductUrl (ignored): %ls\n",
376 a, aVboxValues[a]);
377 break;
378
379 case VirtualSystemDescriptionType_Vendor:
380 RTPrintf("%2u: Vendor (ignored): %ls\n",
381 a, aVboxValues[a]);
382 break;
383
384 case VirtualSystemDescriptionType_VendorUrl:
385 RTPrintf("%2u: VendorUrl (ignored): %ls\n",
386 a, aVboxValues[a]);
387 break;
388
389 case VirtualSystemDescriptionType_Version:
390 RTPrintf("%2u: Version (ignored): %ls\n",
391 a, aVboxValues[a]);
392 break;
393
394 case VirtualSystemDescriptionType_Description:
395 if (findArgValue(strOverride, pmapArgs, "description"))
396 {
397 bstrFinalValue = strOverride;
398 RTPrintf("%2u: Description specified with --description: \"%ls\"\n",
399 a, bstrFinalValue.raw());
400 }
401 else
402 RTPrintf("%2u: Description \"%ls\""
403 "\n (change with \"--vsys %u --description <desc>\")\n",
404 a, bstrFinalValue.raw(), i);
405 break;
406
407 case VirtualSystemDescriptionType_License:
408 ++cLicensesInTheWay;
409 if (findArgValue(strOverride, pmapArgs, "eula"))
410 {
411 if (strOverride == "show")
412 {
413 RTPrintf("%2u: End-user license agreement"
414 "\n (accept with \"--vsys %u --eula accept\"):"
415 "\n\n%ls\n\n",
416 a, i, bstrFinalValue.raw());
417 }
418 else if (strOverride == "accept")
419 {
420 RTPrintf("%2u: End-user license agreement (accepted)\n",
421 a);
422 --cLicensesInTheWay;
423 }
424 else
425 return errorSyntax(USAGE_IMPORTAPPLIANCE,
426 "Argument to --eula must be either \"show\" or \"accept\".");
427 }
428 else
429 RTPrintf("%2u: End-user license agreement"
430 "\n (display with \"--vsys %u --eula show\";"
431 "\n accept with \"--vsys %u --eula accept\")\n",
432 a, i, i);
433 break;
434
435 case VirtualSystemDescriptionType_CPU:
436 RTPrintf("%2u: Number of CPUs (ignored): %ls\n",
437 a, aVboxValues[a]);
438 break;
439
440 case VirtualSystemDescriptionType_Memory:
441 {
442 if (findArgValue(strOverride, pmapArgs, "memory"))
443 {
444 uint32_t ulMemMB;
445 if (VINF_SUCCESS == strOverride.toInt(ulMemMB))
446 {
447 bstrFinalValue = strOverride;
448 RTPrintf("%2u: Guest memory specified with --memory: %ls MB\n",
449 a, bstrFinalValue.raw());
450 }
451 else
452 return errorSyntax(USAGE_IMPORTAPPLIANCE,
453 "Argument to --memory option must be a non-negative number.");
454 }
455 else
456 RTPrintf("%2u: Guest memory: %ls MB\n (change with \"--vsys %u --memory <MB>\")\n",
457 a, bstrFinalValue.raw(), i);
458 }
459 break;
460
461 case VirtualSystemDescriptionType_HardDiskControllerIDE:
462 if (fIgnoreThis)
463 {
464 RTPrintf("%2u: IDE controller, type %ls -- disabled\n",
465 a,
466 aVboxValues[a]);
467 aEnabled[a] = false;
468 }
469 else
470 RTPrintf("%2u: IDE controller, type %ls"
471 "\n (disable with \"--vsys %u --unit %u --ignore\")\n",
472 a,
473 aVboxValues[a],
474 i, a);
475 break;
476
477 case VirtualSystemDescriptionType_HardDiskControllerSATA:
478 if (fIgnoreThis)
479 {
480 RTPrintf("%2u: SATA controller, type %ls -- disabled\n",
481 a,
482 aVboxValues[a]);
483 aEnabled[a] = false;
484 }
485 else
486 RTPrintf("%2u: SATA controller, type %ls"
487 "\n (disable with \"--vsys %u --unit %u --ignore\")\n",
488 a,
489 aVboxValues[a],
490 i, a);
491 break;
492
493 case VirtualSystemDescriptionType_HardDiskControllerSCSI:
494 if (fIgnoreThis)
495 {
496 RTPrintf("%2u: SCSI controller, type %ls -- disabled\n",
497 a,
498 aVboxValues[a]);
499 aEnabled[a] = false;
500 }
501 else
502 {
503 Utf8StrFmt strTypeArg("scsitype%u", a);
504 if (findArgValue(strOverride, pmapArgs, strTypeArg))
505 {
506 bstrFinalValue = strOverride;
507 RTPrintf("%2u: SCSI controller, type set with --unit %u --scsitype: \"%ls\"\n",
508 a,
509 a,
510 bstrFinalValue.raw());
511 }
512 else
513 RTPrintf("%2u: SCSI controller, type %ls"
514 "\n (change with \"--vsys %u --unit %u --scsitype {BusLogic|LsiLogic}\";"
515 "\n disable with \"--vsys %u --unit %u --ignore\")\n",
516 a,
517 aVboxValues[a],
518 i, a, i, a);
519 }
520 break;
521
522 case VirtualSystemDescriptionType_HardDiskImage:
523 if (fIgnoreThis)
524 {
525 RTPrintf("%2u: Hard disk image: source image=%ls -- disabled\n",
526 a,
527 aOvfValues[a]);
528 aEnabled[a] = false;
529 }
530 else
531 {
532 Utf8StrFmt strTypeArg("controller%u", a);
533 if (findArgValue(strOverride, pmapArgs, strTypeArg))
534 {
535 // strOverride now has the controller index as a number, but we
536 // need a "controller=X" format string
537 strOverride = Utf8StrFmt("controller=%s", strOverride.c_str());
538 Bstr bstrExtraConfigValue = strOverride;
539 bstrExtraConfigValue.detachTo(&aExtraConfigValues[a]);
540 RTPrintf("%2u: Hard disk image: source image=%ls, target path=%ls, %ls\n",
541 a,
542 aOvfValues[a],
543 aVboxValues[a],
544 aExtraConfigValues[a]);
545 }
546 else
547 RTPrintf("%2u: Hard disk image: source image=%ls, target path=%ls, %ls"
548 "\n (change controller with \"--vsys %u --unit %u --controller <id>\";"
549 "\n disable with \"--vsys %u --unit %u --ignore\")\n",
550 a,
551 aOvfValues[a],
552 aVboxValues[a],
553 aExtraConfigValues[a],
554 i, a, i, a);
555 }
556 break;
557
558 case VirtualSystemDescriptionType_CDROM:
559 if (fIgnoreThis)
560 {
561 RTPrintf("%2u: CD-ROM -- disabled\n",
562 a);
563 aEnabled[a] = false;
564 }
565 else
566 RTPrintf("%2u: CD-ROM"
567 "\n (disable with \"--vsys %u --unit %u --ignore\")\n",
568 a, i, a);
569 break;
570
571 case VirtualSystemDescriptionType_Floppy:
572 if (fIgnoreThis)
573 {
574 RTPrintf("%2u: Floppy -- disabled\n",
575 a);
576 aEnabled[a] = false;
577 }
578 else
579 RTPrintf("%2u: Floppy"
580 "\n (disable with \"--vsys %u --unit %u --ignore\")\n",
581 a, i, a);
582 break;
583
584 case VirtualSystemDescriptionType_NetworkAdapter:
585 RTPrintf("%2u: Network adapter: orig %ls, config %ls, extra %ls\n", // @todo implement once we have a plan for the back-end
586 a,
587 aOvfValues[a],
588 aVboxValues[a],
589 aExtraConfigValues[a]);
590 break;
591
592 case VirtualSystemDescriptionType_USBController:
593 if (fIgnoreThis)
594 {
595 RTPrintf("%2u: USB controller -- disabled\n",
596 a);
597 aEnabled[a] = false;
598 }
599 else
600 RTPrintf("%2u: USB controller"
601 "\n (disable with \"--vsys %u --unit %u --ignore\")\n",
602 a, i, a);
603 break;
604
605 case VirtualSystemDescriptionType_SoundCard:
606 if (fIgnoreThis)
607 {
608 RTPrintf("%2u: Sound card \"%ls\" -- disabled\n",
609 a,
610 aOvfValues[a]);
611 aEnabled[a] = false;
612 }
613 else
614 RTPrintf("%2u: Sound card (appliance expects \"%ls\", can change on import)"
615 "\n (disable with \"--vsys %u --unit %u --ignore\")\n",
616 a,
617 aOvfValues[a],
618 i,
619 a);
620 break;
621 }
622
623 bstrFinalValue.detachTo(&aFinalValues[a]);
624 }
625
626 if (fExecute)
627 CHECK_ERROR_BREAK(aVirtualSystemDescriptions[i],
628 SetFinalValues(ComSafeArrayAsInParam(aEnabled),
629 ComSafeArrayAsInParam(aFinalValues),
630 ComSafeArrayAsInParam(aExtraConfigValues)));
631
632 } // for (unsigned i = 0; i < cVirtualSystemDescriptions; ++i)
633
634 if (cLicensesInTheWay == 1)
635 RTPrintf("ERROR: Cannot import until the license agreement listed above is accepted.\n");
636 else if (cLicensesInTheWay > 1)
637 RTPrintf("ERROR: Cannot import until the %c license agreements listed above are accepted.\n", cLicensesInTheWay);
638
639 if (!cLicensesInTheWay && fExecute)
640 {
641 // go!
642 ComPtr<IProgress> progress;
643 CHECK_ERROR_BREAK(pAppliance,
644 ImportMachines(progress.asOutParam()));
645
646 showProgress(progress);
647
648 if (SUCCEEDED(rc))
649 {
650 LONG iRc;
651 progress->COMGETTER(ResultCode)(&iRc);
652 rc = iRc;
653 }
654
655 if (FAILED(rc))
656 {
657 com::ProgressErrorInfo info(progress);
658 com::GluePrintErrorInfo(info);
659 com::GluePrintErrorContext("ImportAppliance", __FILE__, __LINE__);
660 }
661 else
662 RTPrintf("Successfully imported the appliance.\n");
663 }
664 } // end if (aVirtualSystemDescriptions.size() > 0)
665 } while (0);
666
667 return SUCCEEDED(rc) ? 0 : 1;
668}
669
670static const RTGETOPTDEF g_aExportOptions[]
671 = {
672 { "--output", 'o', RTGETOPT_REQ_STRING },
673 { "--legacy09", 'l', RTGETOPT_REQ_NOTHING },
674 { "--vsys", 's', RTGETOPT_REQ_UINT32 },
675 { "--product", 'p', RTGETOPT_REQ_STRING },
676 { "--producturl", 'P', RTGETOPT_REQ_STRING },
677 { "--vendor", 'd', RTGETOPT_REQ_STRING },
678 { "--vendorurl", 'D', RTGETOPT_REQ_STRING },
679 { "--version", 'v', RTGETOPT_REQ_STRING },
680 { "--eula", 'e', RTGETOPT_REQ_STRING },
681 { "--eulafile", 'E', RTGETOPT_REQ_STRING },
682 };
683
684int handleExportAppliance(HandlerArg *a)
685{
686 HRESULT rc = S_OK;
687
688 Utf8Str strOutputFile;
689 Utf8Str strOvfFormat("ovf-1.0"); // the default export version
690 std::list< ComPtr<IMachine> > llMachines;
691
692 uint32_t ulCurVsys = (uint32_t)-1;
693 // for each --vsys X command, maintain a map of command line items
694 ArgsMapsMap mapArgsMapsPerVsys;
695 do
696 {
697 int c;
698
699 RTGETOPTUNION ValueUnion;
700 RTGETOPTSTATE GetState;
701 // start at 0 because main() has hacked both the argc and argv given to us
702 RTGetOptInit(&GetState, a->argc, a->argv, g_aExportOptions,
703 RT_ELEMENTS(g_aExportOptions), 0, 0 /* fFlags */);
704
705 Utf8Str strProductUrl;
706 while ((c = RTGetOpt(&GetState, &ValueUnion)))
707 {
708 switch (c)
709 {
710 case 'o': // --output
711 if (strOutputFile.length())
712 return errorSyntax(USAGE_EXPORTAPPLIANCE, "You can only specify --output once.");
713 else
714 strOutputFile = ValueUnion.psz;
715 break;
716
717 case 'l': // --legacy09
718 strOvfFormat = "ovf-0.9";
719 break;
720
721 case 's': // --vsys
722 ulCurVsys = ValueUnion.u32;
723 break;
724
725 case 'p': // --product
726 if (ulCurVsys == (uint32_t)-1)
727 return errorSyntax(USAGE_EXPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
728 mapArgsMapsPerVsys[ulCurVsys]["product"] = ValueUnion.psz;
729 break;
730
731 case 'P': // --producturl
732 if (ulCurVsys == (uint32_t)-1)
733 return errorSyntax(USAGE_EXPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
734 mapArgsMapsPerVsys[ulCurVsys]["producturl"] = ValueUnion.psz;
735 break;
736
737 case 'd': // --vendor
738 if (ulCurVsys == (uint32_t)-1)
739 return errorSyntax(USAGE_EXPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
740 mapArgsMapsPerVsys[ulCurVsys]["vendor"] = ValueUnion.psz;
741 break;
742
743 case 'D': // --vendorurl
744 if (ulCurVsys == (uint32_t)-1)
745 return errorSyntax(USAGE_EXPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
746 mapArgsMapsPerVsys[ulCurVsys]["vendorurl"] = ValueUnion.psz;
747 break;
748
749 case 'v': // --version
750 if (ulCurVsys == (uint32_t)-1)
751 return errorSyntax(USAGE_EXPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
752 mapArgsMapsPerVsys[ulCurVsys]["version"] = ValueUnion.psz;
753 break;
754
755 case 'e': // --eula
756 if (ulCurVsys == (uint32_t)-1)
757 return errorSyntax(USAGE_EXPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
758 mapArgsMapsPerVsys[ulCurVsys]["eula"] = ValueUnion.psz;
759 break;
760
761 case 'E': // --eulafile
762 if (ulCurVsys == (uint32_t)-1)
763 return errorSyntax(USAGE_EXPORTAPPLIANCE, "Option \"%s\" requires preceding --vsys argument.", GetState.pDef->pszLong);
764 mapArgsMapsPerVsys[ulCurVsys]["eulafile"] = ValueUnion.psz;
765 break;
766
767 case VINF_GETOPT_NOT_OPTION:
768 {
769 Utf8Str strMachine(ValueUnion.psz);
770 // must be machine: try UUID or name
771 ComPtr<IMachine> machine;
772 /* assume it's a UUID */
773 rc = a->virtualBox->GetMachine(Bstr(strMachine), machine.asOutParam());
774 if (FAILED(rc) || !machine)
775 {
776 /* must be a name */
777 CHECK_ERROR_BREAK(a->virtualBox, FindMachine(Bstr(strMachine), machine.asOutParam()));
778 }
779
780 if (machine)
781 llMachines.push_back(machine);
782 }
783 break;
784
785 default:
786 if (c > 0)
787 {
788 if (RT_C_IS_GRAPH(c))
789 return errorSyntax(USAGE_EXPORTAPPLIANCE, "unhandled option: -%c", c);
790 else
791 return errorSyntax(USAGE_EXPORTAPPLIANCE, "unhandled option: %i", c);
792 }
793 else if (c == VERR_GETOPT_UNKNOWN_OPTION)
794 return errorSyntax(USAGE_EXPORTAPPLIANCE, "unknown option: %s", ValueUnion.psz);
795 else if (ValueUnion.pDef)
796 return errorSyntax(USAGE_EXPORTAPPLIANCE, "%s: %Rrs", ValueUnion.pDef->pszLong, c);
797 else
798 return errorSyntax(USAGE_EXPORTAPPLIANCE, "%Rrs", c);
799 }
800
801 if (FAILED(rc))
802 break;
803 }
804
805 if (FAILED(rc))
806 break;
807
808 if (llMachines.size() == 0)
809 return errorSyntax(USAGE_EXPORTAPPLIANCE, "At least one machine must be specified with the export command.");
810 if (!strOutputFile.length())
811 return errorSyntax(USAGE_EXPORTAPPLIANCE, "Missing --output argument with export command.");
812
813 // match command line arguments with the machines count
814 // this is only to sort out invalid indices at this time
815 ArgsMapsMap::const_iterator it;
816 for (it = mapArgsMapsPerVsys.begin();
817 it != mapArgsMapsPerVsys.end();
818 ++it)
819 {
820 uint32_t ulVsys = it->first;
821 if (ulVsys >= llMachines.size())
822 return errorSyntax(USAGE_EXPORTAPPLIANCE,
823 "Invalid index %RI32 with -vsys option; you specified only %zu virtual system(s).",
824 ulVsys, llMachines.size());
825 }
826
827 ComPtr<IAppliance> pAppliance;
828 CHECK_ERROR_BREAK(a->virtualBox, CreateAppliance(pAppliance.asOutParam()));
829
830 std::list< ComPtr<IMachine> >::iterator itM;
831 uint32_t i=0;
832 for (itM = llMachines.begin();
833 itM != llMachines.end();
834 ++itM, ++i)
835 {
836 ComPtr<IMachine> pMachine = *itM;
837 ComPtr<IVirtualSystemDescription> pVSD;
838 CHECK_ERROR_BREAK(pMachine, Export(pAppliance, pVSD.asOutParam()));
839 // Add additional info to the virtal system description if the user wants so
840 ArgsMap *pmapArgs = NULL;
841 ArgsMapsMap::iterator itm = mapArgsMapsPerVsys.find(i);
842 if (itm != mapArgsMapsPerVsys.end())
843 pmapArgs = &itm->second;
844 if (pmapArgs)
845 {
846 ArgsMap::iterator itD;
847 for (itD = pmapArgs->begin();
848 itD != pmapArgs->end();
849 ++itD)
850 {
851 if (itD->first == "product")
852 pVSD->AddDescription (VirtualSystemDescriptionType_Product, Bstr(itD->second), Bstr(itD->second));
853 else if (itD->first == "producturl")
854 pVSD->AddDescription (VirtualSystemDescriptionType_ProductUrl, Bstr(itD->second), Bstr(itD->second));
855 else if (itD->first == "vendor")
856 pVSD->AddDescription (VirtualSystemDescriptionType_Vendor, Bstr(itD->second), Bstr(itD->second));
857 else if (itD->first == "vendorurl")
858 pVSD->AddDescription (VirtualSystemDescriptionType_VendorUrl, Bstr(itD->second), Bstr(itD->second));
859 else if (itD->first == "version")
860 pVSD->AddDescription (VirtualSystemDescriptionType_Version, Bstr(itD->second), Bstr(itD->second));
861 else if (itD->first == "eula")
862 pVSD->AddDescription (VirtualSystemDescriptionType_License, Bstr(itD->second), Bstr(itD->second));
863 else if (itD->first == "eulafile")
864 {
865 Utf8Str strContent;
866 void *pvFile;
867 size_t cbFile;
868 int rc = RTFileReadAll(itD->second.c_str(), &pvFile, &cbFile);
869 if (RT_SUCCESS(rc))
870 {
871 Bstr strContent((char*)pvFile);
872 pVSD->AddDescription (VirtualSystemDescriptionType_License, strContent, strContent);
873 RTFileReadAllFree(pvFile, cbFile);
874 }
875 else
876 {
877 RTPrintf("ERROR: Cannot read license file \"%s\" which should be included in the virtual system %u.\n",
878 itD->second.c_str(),
879 i);
880 return 1;
881 }
882 }
883 }
884 }
885 }
886
887 if (FAILED(rc))
888 break;
889
890 ComPtr<IProgress> progress;
891 char *pszAbsFilePath = RTPathAbsDup(strOutputFile.c_str());
892 CHECK_ERROR_BREAK(pAppliance, Write(Bstr(strOvfFormat), Bstr(pszAbsFilePath), progress.asOutParam()));
893 RTStrFree(pszAbsFilePath);
894
895 showProgress(progress);
896
897 if (SUCCEEDED(rc))
898 {
899 LONG iRc;
900 progress->COMGETTER(ResultCode)(&iRc);
901 rc = iRc;
902 }
903
904 if (FAILED(rc))
905 {
906 com::ProgressErrorInfo info(progress);
907 com::GluePrintErrorInfo(info);
908 com::GluePrintErrorContext("Write", __FILE__, __LINE__);
909 }
910 else
911 RTPrintf("Successfully exported %d machine(s).\n", llMachines.size());
912
913 } while (0);
914
915 return SUCCEEDED(rc) ? 0 : 1;
916}
917
918#endif /* !VBOX_ONLY_DOCS */
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