VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/UnattendedScript.cpp@ 87635

Last change on this file since 87635 was 86660, checked in by vboxsync, 4 years ago

scm fix

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 26.2 KB
Line 
1/* $Id: UnattendedScript.cpp 86660 2020-10-21 06:06:30Z vboxsync $ */
2/** @file
3 * Classes for reading/parsing/saving scripts for unattended installation.
4 */
5
6/*
7 * Copyright (C) 2006-2020 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
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#define LOG_GROUP LOG_GROUP_MAIN_UNATTENDED
23#include "LoggingNew.h"
24#include "VirtualBoxBase.h"
25#include "AutoCaller.h"
26#include <VBox/com/ErrorInfo.h>
27
28#include "UnattendedScript.h"
29#include "UnattendedImpl.h"
30
31#include <iprt/errcore.h>
32
33#include <iprt/ctype.h>
34#include <iprt/file.h>
35#include <iprt/vfs.h>
36#include <iprt/getopt.h>
37#include <iprt/path.h>
38
39using namespace std;
40
41#ifdef VBOX_WITH_UNATTENDED
42
43
44/*********************************************************************************************************************************
45* UnattendedScriptTemplate Implementation *
46*********************************************************************************************************************************/
47
48UnattendedScriptTemplate::UnattendedScriptTemplate(Unattended *pUnattended, const char *pszDefaultTemplateFilename,
49 const char *pszDefaultFilename)
50 : BaseTextScript(pUnattended, pszDefaultTemplateFilename, pszDefaultFilename), mpUnattended(pUnattended)
51{
52}
53
54
55HRESULT UnattendedScriptTemplate::saveToString(Utf8Str &rStrDst)
56{
57 static const char s_szPrefix[] = "@@VBOX_";
58 static const char s_szPrefixInsert[] = "@@VBOX_INSERT_";
59 static const char s_szPrefixCond[] = "@@VBOX_COND_";
60 static const char s_szPrefixCondEnd[] = "@@VBOX_COND_END@@";
61
62 struct
63 {
64 bool fSavedOutputting;
65 } aConds[8];
66 unsigned cConds = 0;
67 bool fOutputting = true;
68 HRESULT hrc = E_FAIL;
69 size_t offTemplate = 0;
70 size_t cchTemplate = mStrScriptFullContent.length();
71 rStrDst.setNull();
72 for (;;)
73 {
74 /*
75 * Find the next placeholder and add any text before it to the output.
76 */
77 size_t offPlaceholder = mStrScriptFullContent.find(s_szPrefix, offTemplate);
78 size_t cchToCopy = offPlaceholder != RTCString::npos ? offPlaceholder - offTemplate : cchTemplate - offTemplate;
79 if (cchToCopy > 0)
80 {
81 if (fOutputting)
82 {
83 try
84 {
85 rStrDst.append(mStrScriptFullContent, offTemplate , cchToCopy);
86 }
87 catch (std::bad_alloc &)
88 {
89 hrc = E_OUTOFMEMORY;
90 break;
91 }
92 }
93 offTemplate += cchToCopy;
94 }
95
96 /*
97 * Process placeholder.
98 */
99 if (offPlaceholder != RTCString::npos)
100 {
101 /*
102 * First we must find the end of the placeholder string.
103 */
104 const char *pszPlaceholder = mStrScriptFullContent.c_str() + offPlaceholder;
105 size_t cchPlaceholder = sizeof(s_szPrefix) - 1;
106 char ch;
107 while ( offPlaceholder + cchPlaceholder < cchTemplate
108 && (ch = pszPlaceholder[cchPlaceholder]) != '\0'
109 && ( ch == '_'
110 || ch == '['
111 || ch == ']'
112 || ch == '.'
113 || ch == '>'
114 || ch == '<'
115 || RT_C_IS_UPPER(ch)
116 || RT_C_IS_DIGIT(ch)) )
117 cchPlaceholder++;
118
119 if ( offPlaceholder + cchPlaceholder < cchTemplate
120 && pszPlaceholder[cchPlaceholder] == '@')
121 {
122 cchPlaceholder++;
123 if ( offPlaceholder + cchPlaceholder < cchTemplate
124 && pszPlaceholder[cchPlaceholder] == '@')
125 cchPlaceholder++;
126 }
127
128 if ( pszPlaceholder[cchPlaceholder - 1] != '@'
129 || pszPlaceholder[cchPlaceholder - 2] != '@'
130 || ( strncmp(pszPlaceholder, s_szPrefixInsert, sizeof(s_szPrefixInsert) - 1) != 0
131 && strncmp(pszPlaceholder, s_szPrefixCond, sizeof(s_szPrefixCond) - 1) != 0 ) )
132 {
133 hrc = mpSetError->setError(E_FAIL, mpSetError->tr("Malformed template placeholder '%.*s'"),
134 cchPlaceholder, pszPlaceholder);
135 break;
136 }
137
138 offTemplate += cchPlaceholder;
139
140 /*
141 * @@VBOX_INSERT_XXX@@:
142 */
143 if ( strncmp(pszPlaceholder, s_szPrefixInsert, sizeof(s_szPrefixInsert) - 1) == 0 )
144 {
145 /*
146 * Get the placeholder value and add it to the output.
147 */
148 RTCString strValue;
149 hrc = getReplacement(pszPlaceholder, cchPlaceholder, fOutputting, strValue);
150 if (SUCCEEDED(hrc))
151 {
152 if (fOutputting)
153 {
154 try
155 {
156 rStrDst.append(strValue);
157 }
158 catch (std::bad_alloc &)
159 {
160 hrc = E_OUTOFMEMORY;
161 break;
162 }
163 }
164 }
165 else
166 break;
167 }
168 /*
169 * @@VBOX_COND_END@@: Pop one item of the conditional stack.
170 */
171 else if ( strncmp(pszPlaceholder, s_szPrefixCondEnd, sizeof(s_szPrefixCondEnd) - 1U) == 0 )
172 {
173 if (cConds > 0)
174 {
175 cConds--;
176 fOutputting = aConds[cConds].fSavedOutputting;
177 }
178 else
179 {
180 hrc = mpSetError->setErrorBoth(E_FAIL, VERR_PARSE_ERROR,
181 mpSetError->tr("%s without @@VBOX_COND_XXX@@ at offset %zu (%#zx)"),
182 s_szPrefixCondEnd, offPlaceholder, offPlaceholder);
183 break;
184 }
185 }
186 /*
187 * @@VBOX_COND_XXX@@: Push the previous outputting state and combine it with the
188 * one from the condition.
189 */
190 else
191 {
192 Assert(strncmp(pszPlaceholder, s_szPrefixCond, sizeof(s_szPrefixCond) - 1) == 0);
193 if (cConds + 1 < RT_ELEMENTS(aConds))
194 {
195 aConds[cConds].fSavedOutputting = fOutputting;
196 bool fNewOutputting = fOutputting;
197 hrc = getConditional(pszPlaceholder, cchPlaceholder, &fNewOutputting);
198 if (SUCCEEDED(hrc))
199 fOutputting = fOutputting && fNewOutputting;
200 else
201 break;
202 cConds++;
203 }
204 else
205 {
206 hrc = mpSetError->setErrorBoth(E_FAIL, VERR_PARSE_ERROR,
207 mpSetError->tr("Too deep conditional nesting at offset %zu (%#zx)"),
208 offPlaceholder, offPlaceholder);
209 break;
210 }
211 }
212 }
213
214 /*
215 * Done?
216 */
217 if (offTemplate >= cchTemplate)
218 {
219 if (cConds == 0)
220 return S_OK;
221 if (cConds == 1)
222 hrc = mpSetError->setErrorBoth(E_FAIL, VERR_PARSE_ERROR, mpSetError->tr("Missing @@VBOX_COND_END@@"));
223 else
224 hrc = mpSetError->setErrorBoth(E_FAIL, VERR_PARSE_ERROR, mpSetError->tr("Missing %u @@VBOX_COND_END@@"), cConds);
225 break;
226 }
227 }
228
229 /* failed */
230 rStrDst.setNull();
231 return hrc;
232}
233
234HRESULT UnattendedScriptTemplate::getReplacement(const char *pachPlaceholder, size_t cchPlaceholder,
235 bool fOutputting, RTCString &rValue)
236{
237 /*
238 * Check for an escaping suffix. Drop the '@@'.
239 */
240 size_t const cchFullPlaceholder = cchPlaceholder;
241 enum
242 {
243 kValueEscaping_None,
244 kValueEscaping_Bourne,
245 kValueEscaping_XML_Element,
246 kValueEscaping_XML_Attribute_Double_Quotes
247 } enmEscaping;
248
249#define PLACEHOLDER_ENDS_WITH(a_szSuffix) \
250 ( cchPlaceholder > sizeof(a_szSuffix) - 1U \
251 && memcmp(&pachPlaceholder[cchPlaceholder - sizeof(a_szSuffix) + 1U], a_szSuffix, sizeof(a_szSuffix) - 1U) == 0)
252 if (PLACEHOLDER_ENDS_WITH("_SH@@"))
253 {
254 cchPlaceholder -= 3 + 2;
255 enmEscaping = kValueEscaping_Bourne;
256 }
257 else if (PLACEHOLDER_ENDS_WITH("_ELEMENT@@"))
258 {
259 cchPlaceholder -= 8 + 2;
260 enmEscaping = kValueEscaping_XML_Element;
261 }
262 else if (PLACEHOLDER_ENDS_WITH("_ATTRIB_DQ@@"))
263 {
264 cchPlaceholder -= 10 + 2;
265 enmEscaping = kValueEscaping_XML_Attribute_Double_Quotes;
266 }
267 else
268 {
269 Assert(PLACEHOLDER_ENDS_WITH("@@"));
270 cchPlaceholder -= 2;
271 enmEscaping = kValueEscaping_None;
272 }
273
274 /*
275 * Resolve and escape the value.
276 */
277 HRESULT hrc;
278 try
279 {
280 switch (enmEscaping)
281 {
282 case kValueEscaping_None:
283 hrc = getUnescapedReplacement(pachPlaceholder, cchPlaceholder, cchFullPlaceholder, fOutputting, rValue);
284 if (SUCCEEDED(hrc))
285 return hrc;
286 break;
287
288 case kValueEscaping_Bourne:
289 case kValueEscaping_XML_Element:
290 case kValueEscaping_XML_Attribute_Double_Quotes:
291 {
292 RTCString strUnescaped;
293 hrc = getUnescapedReplacement(pachPlaceholder, cchPlaceholder, cchFullPlaceholder, fOutputting, strUnescaped);
294 if (SUCCEEDED(hrc))
295 {
296 switch (enmEscaping)
297 {
298 case kValueEscaping_Bourne:
299 {
300 const char * const papszArgs[2] = { strUnescaped.c_str(), NULL };
301 char *pszEscaped = NULL;
302 int vrc = RTGetOptArgvToString(&pszEscaped, papszArgs, RTGETOPTARGV_CNV_QUOTE_BOURNE_SH);
303 if (RT_SUCCESS(vrc))
304 {
305 try
306 {
307 rValue = pszEscaped;
308 RTStrFree(pszEscaped);
309 return S_OK;
310 }
311 catch (std::bad_alloc &)
312 {
313 hrc = E_OUTOFMEMORY;
314 }
315 RTStrFree(pszEscaped);
316 }
317 break;
318 }
319
320 case kValueEscaping_XML_Element:
321 rValue.printf("%RMes", strUnescaped.c_str());
322 return S_OK;
323
324 case kValueEscaping_XML_Attribute_Double_Quotes:
325 {
326 RTCString strTmp;
327 strTmp.printf("%RMas", strUnescaped.c_str());
328 rValue = RTCString(strTmp, 1, strTmp.length() - 2);
329 return S_OK;
330 }
331
332 default:
333 hrc = E_FAIL;
334 break;
335 }
336 }
337 break;
338 }
339
340 default:
341 AssertFailedStmt(hrc = E_FAIL);
342 break;
343 }
344 }
345 catch (std::bad_alloc &)
346 {
347 hrc = E_OUTOFMEMORY;
348 }
349 rValue.setNull();
350 return hrc;
351}
352
353HRESULT UnattendedScriptTemplate::getUnescapedReplacement(const char *pachPlaceholder, size_t cchPlaceholder,
354 size_t cchFullPlaceholder, bool fOutputting, RTCString &rValue)
355{
356 RT_NOREF(fOutputting);
357#define IS_PLACEHOLDER_MATCH(a_szMatch) \
358 ( cchPlaceholder == sizeof("@@VBOX_INSERT_" a_szMatch) - 1U \
359 && memcmp(pachPlaceholder, "@@VBOX_INSERT_" a_szMatch, sizeof("@@VBOX_INSERT_" a_szMatch) - 1U) == 0)
360
361 if (IS_PLACEHOLDER_MATCH("USER_LOGIN"))
362 rValue = mpUnattended->i_getUser();
363 else if (IS_PLACEHOLDER_MATCH("USER_PASSWORD"))
364 rValue = mpUnattended->i_getPassword();
365 else if (IS_PLACEHOLDER_MATCH("ROOT_PASSWORD"))
366 rValue = mpUnattended->i_getPassword();
367 else if (IS_PLACEHOLDER_MATCH("USER_FULL_NAME"))
368 rValue = mpUnattended->i_getFullUserName();
369 else if (IS_PLACEHOLDER_MATCH("PRODUCT_KEY"))
370 rValue = mpUnattended->i_getProductKey();
371 else if (IS_PLACEHOLDER_MATCH("POST_INSTALL_COMMAND"))
372 rValue = mpUnattended->i_getPostInstallCommand();
373 else if (IS_PLACEHOLDER_MATCH("IMAGE_INDEX"))
374 rValue.printf("%u", mpUnattended->i_getImageIndex());
375 else if (IS_PLACEHOLDER_MATCH("OS_ARCH"))
376 rValue = mpUnattended->i_isGuestOs64Bit() ? "amd64" : "x86";
377 else if (IS_PLACEHOLDER_MATCH("OS_ARCH2"))
378 rValue = mpUnattended->i_isGuestOs64Bit() ? "x86_64" : "x86";
379 else if (IS_PLACEHOLDER_MATCH("OS_ARCH3"))
380 rValue = mpUnattended->i_isGuestOs64Bit() ? "x86_64" : "i386";
381 else if (IS_PLACEHOLDER_MATCH("OS_ARCH4"))
382 rValue = mpUnattended->i_isGuestOs64Bit() ? "x86_64" : "i486";
383 else if (IS_PLACEHOLDER_MATCH("OS_ARCH6"))
384 rValue = mpUnattended->i_isGuestOs64Bit() ? "x86_64" : "i686";
385 else if (IS_PLACEHOLDER_MATCH("GUEST_OS_VERSION"))
386 rValue = mpUnattended->i_getDetectedOSVersion();
387 else if (IS_PLACEHOLDER_MATCH("GUEST_OS_MAJOR_VERSION"))
388 {
389 Utf8Str strOsVer(mpUnattended->i_getDetectedOSVersion());
390 RTCList<RTCString> partList = strOsVer.split(".");
391 if (partList.size() < 1)
392 {
393 rValue.setNull();
394 return mpSetError->setErrorBoth(E_FAIL, VERR_NOT_FOUND, mpSetError->tr("Unknown guest OS major version '%s'"),
395 partList.at(0).c_str());
396 }
397 rValue = partList.at(0);
398 }
399 else if (IS_PLACEHOLDER_MATCH("TIME_ZONE_UX"))
400 rValue = mpUnattended->i_getTimeZoneInfo()
401 ? mpUnattended->i_getTimeZoneInfo()->pszUnixName : mpUnattended->i_getTimeZone();
402 else if (IS_PLACEHOLDER_MATCH("TIME_ZONE_WIN_NAME"))
403 {
404 PCRTTIMEZONEINFO pInfo = mpUnattended->i_getTimeZoneInfo();
405 if (pInfo)
406 rValue = pInfo->pszWindowsName ? pInfo->pszWindowsName : "GMT";
407 else
408 rValue = mpUnattended->i_getTimeZone();
409 }
410 else if (IS_PLACEHOLDER_MATCH("TIME_ZONE_WIN_INDEX"))
411 {
412 PCRTTIMEZONEINFO pInfo = mpUnattended->i_getTimeZoneInfo();
413 if (pInfo)
414 rValue.printf("%u", pInfo->idxWindows ? pInfo->idxWindows : 85 /*GMT*/);
415 else
416 rValue = mpUnattended->i_getTimeZone();
417 }
418 else if (IS_PLACEHOLDER_MATCH("LOCALE"))
419 rValue = mpUnattended->i_getLocale();
420 else if (IS_PLACEHOLDER_MATCH("DASH_LOCALE"))
421 {
422 rValue = mpUnattended->i_getLocale();
423 Assert(rValue[2] == '_');
424 rValue.replace(2, 1, "-");
425 }
426 else if (IS_PLACEHOLDER_MATCH("LANGUAGE"))
427 rValue = mpUnattended->i_getLanguage();
428 else if (IS_PLACEHOLDER_MATCH("COUNTRY"))
429 rValue = mpUnattended->i_getCountry();
430 else if (IS_PLACEHOLDER_MATCH("HOSTNAME_FQDN"))
431 rValue = mpUnattended->i_getHostname();
432 else if (IS_PLACEHOLDER_MATCH("HOSTNAME_WITHOUT_DOMAIN"))
433 rValue.assign(mpUnattended->i_getHostname(), 0, mpUnattended->i_getHostname().find("."));
434 else if (IS_PLACEHOLDER_MATCH("HOSTNAME_WITHOUT_DOMAIN_MAX_15"))
435 rValue.assign(mpUnattended->i_getHostname(), 0, RT_MIN(mpUnattended->i_getHostname().find("."), 15));
436 else if (IS_PLACEHOLDER_MATCH("HOSTNAME_DOMAIN"))
437 rValue.assign(mpUnattended->i_getHostname(), mpUnattended->i_getHostname().find(".") + 1);
438 else if (IS_PLACEHOLDER_MATCH("PROXY"))
439 rValue = mpUnattended->i_getProxy();
440 else
441 {
442 rValue.setNull();
443 return mpSetError->setErrorBoth(E_FAIL, VERR_NOT_FOUND, mpSetError->tr("Unknown template placeholder '%.*s'"),
444 cchFullPlaceholder, pachPlaceholder);
445 }
446 return S_OK;
447#undef IS_PLACEHOLDER_MATCH
448}
449
450HRESULT UnattendedScriptTemplate::getConditional(const char *pachPlaceholder, size_t cchPlaceholder, bool *pfOutputting)
451{
452#define IS_PLACEHOLDER_MATCH(a_szMatch) \
453 ( cchPlaceholder == sizeof("@@VBOX_COND_" a_szMatch "@@") - 1U \
454 && memcmp(pachPlaceholder, "@@VBOX_COND_" a_szMatch "@@", sizeof("@@VBOX_COND_" a_szMatch "@@") - 1U) == 0)
455#define IS_PLACEHOLDER_PARTIALLY_MATCH(a_szMatch) \
456 (memcmp(pachPlaceholder, "@@VBOX_COND_" a_szMatch, sizeof("@@VBOX_COND_" a_szMatch) - 1U) == 0)
457
458 /* Install Guest Additions: */
459 if (IS_PLACEHOLDER_MATCH("IS_INSTALLING_ADDITIONS"))
460 *pfOutputting = mpUnattended->i_getInstallGuestAdditions();
461 else if (IS_PLACEHOLDER_MATCH("IS_NOT_INSTALLING_ADDITIONS"))
462 *pfOutputting = !mpUnattended->i_getInstallGuestAdditions();
463 /* User == Administrator: */
464 else if (IS_PLACEHOLDER_MATCH("IS_USER_LOGIN_ADMINISTRATOR"))
465 *pfOutputting = mpUnattended->i_getUser().compare("Administrator", RTCString::CaseInsensitive) == 0;
466 else if (IS_PLACEHOLDER_MATCH("IS_USER_LOGIN_NOT_ADMINISTRATOR"))
467 *pfOutputting = mpUnattended->i_getUser().compare("Administrator", RTCString::CaseInsensitive) != 0;
468 /* Install TXS: */
469 else if (IS_PLACEHOLDER_MATCH("IS_INSTALLING_TEST_EXEC_SERVICE"))
470 *pfOutputting = mpUnattended->i_getInstallTestExecService();
471 else if (IS_PLACEHOLDER_MATCH("IS_NOT_INSTALLING_TEST_EXEC_SERVICE"))
472 *pfOutputting = !mpUnattended->i_getInstallTestExecService();
473 /* Post install command: */
474 else if (IS_PLACEHOLDER_MATCH("HAS_POST_INSTALL_COMMAND"))
475 *pfOutputting = mpUnattended->i_getPostInstallCommand().isNotEmpty();
476 else if (IS_PLACEHOLDER_MATCH("HAS_NO_POST_INSTALL_COMMAND"))
477 *pfOutputting = !mpUnattended->i_getPostInstallCommand().isNotEmpty();
478 /* Product key: */
479 else if (IS_PLACEHOLDER_MATCH("HAS_PRODUCT_KEY"))
480 *pfOutputting = mpUnattended->i_getProductKey().isNotEmpty();
481 else if (IS_PLACEHOLDER_MATCH("HAS_NO_PRODUCT_KEY"))
482 *pfOutputting = !mpUnattended->i_getProductKey().isNotEmpty();
483 /* Minimal installation: */
484 else if (IS_PLACEHOLDER_MATCH("IS_MINIMAL_INSTALLATION"))
485 *pfOutputting = mpUnattended->i_isMinimalInstallation();
486 else if (IS_PLACEHOLDER_MATCH("IS_NOT_MINIMAL_INSTALLATION"))
487 *pfOutputting = !mpUnattended->i_isMinimalInstallation();
488 /* Is RTC using UTC (i.e. set to UTC time on startup): */
489 else if (IS_PLACEHOLDER_MATCH("IS_RTC_USING_UTC"))
490 *pfOutputting = mpUnattended->i_isRtcUsingUtc();
491 else if (IS_PLACEHOLDER_MATCH("IS_NOT_RTC_USING_UTC"))
492 *pfOutputting = !mpUnattended->i_isRtcUsingUtc();
493 else if (IS_PLACEHOLDER_MATCH("HAS_PROXY"))
494 *pfOutputting = mpUnattended->i_getProxy().isNotEmpty();
495 else if (IS_PLACEHOLDER_PARTIALLY_MATCH("GUEST_VERSION"))
496 {
497 //parse the placeholder and extract the OS version from there
498 RTCString strPlaceHolder(pachPlaceholder);
499 size_t startPos = sizeof("@@VBOX_COND_GUEST_VERSION") - 1;//-1 is for '\n'
500 size_t endPos = strPlaceHolder.find("@@", startPos + 2);
501 //next part should look like [>8.0.0] for example where:
502 // - "[,]" is just the brackets to wrap up the condition;
503 // - ">" is "greater". Also possible comparison is "<";
504 // - 8.0.0 is required guest OS version.
505 //The end of placeholder is "@@" like for others.
506
507 if ( strPlaceHolder[endPos] == '@'
508 && strPlaceHolder[endPos+1] == '@' )
509 {
510 if ( strPlaceHolder[startPos++] == '[' && strPlaceHolder[--endPos] == ']' )
511 {
512 char chComp = strPlaceHolder[startPos++];
513 RTCString strRequiredOSVersion = strPlaceHolder.substr(startPos, endPos - startPos);
514 RTCString strDetectedOSVersion = mpUnattended->i_getDetectedOSVersion();
515 int res = RTStrVersionCompare(strDetectedOSVersion.c_str(), strRequiredOSVersion.c_str());
516 if ( res >= 0 && chComp == '>' )
517 *pfOutputting = true;
518 else if ( res < 0 && chComp == '<' )
519 *pfOutputting = true;
520 else
521 *pfOutputting = false;
522 }
523 }
524 else
525 *pfOutputting = false;//initially is set to "false"
526 }
527 else
528 return mpSetError->setErrorBoth(E_FAIL, VERR_NOT_FOUND, mpSetError->tr("Unknown conditional placeholder '%.*s'"),
529 cchPlaceholder, pachPlaceholder);
530 return S_OK;
531#undef IS_PLACEHOLDER_MATCH
532}
533
534#endif /* VBOX_WITH_UNATTENDED */
535#if 0 /* Keeping this a reference */
536
537
538/*********************************************************************************************************************************
539* UnattendedSUSEXMLScript Implementation *
540*********************************************************************************************************************************/
541
542HRESULT UnattendedSUSEXMLScript::parse()
543{
544 HRESULT hrc = UnattendedXMLScript::parse();
545 if (SUCCEEDED(hrc))
546 {
547 /*
548 * Check that we've got the right root element type.
549 */
550 const xml::ElementNode *pelmRoot = mDoc.getRootElement();
551 if ( pelmRoot
552 && strcmp(pelmRoot->getName(), "profile") == 0)
553 {
554 /*
555 * Work thought the sections.
556 */
557 try
558 {
559 LoopThruSections(pelmRoot);
560 hrc = S_OK;
561 }
562 catch (std::bad_alloc &)
563 {
564 hrc = E_OUTOFMEMORY;
565 }
566 }
567 else if (pelmRoot)
568 hrc = mpSetError->setError(E_FAIL, mpSetError->tr("XML document root element is '%s' instead of 'profile'"),
569 pelmRoot->getName());
570 else
571 hrc = mpSetError->setError(E_FAIL, mpSetError->tr("Missing XML root element"));
572 }
573 return hrc;
574}
575
576HRESULT UnattendedSUSEXMLScript::setFieldInElement(xml::ElementNode *pElement, const DataId enmDataId, const Utf8Str &rStrValue)
577{
578 /*
579 * Don't set empty values.
580 */
581 if (rStrValue.isEmpty())
582 {
583 Utf8Str strProbableValue;
584 try
585 {
586 strProbableValue = createProbableValue(enmDataId, pElement);
587 }
588 catch (std::bad_alloc &)
589 {
590 return E_OUTOFMEMORY;
591 }
592 return UnattendedXMLScript::setFieldInElement(pElement, enmDataId, strProbableValue);
593 }
594 return UnattendedXMLScript::setFieldInElement(pElement, enmDataId, rStrValue);
595}
596
597HRESULT UnattendedSUSEXMLScript::LoopThruSections(const xml::ElementNode *pelmRoot)
598{
599 xml::NodesLoop loopChildren(*pelmRoot);
600 const xml::ElementNode *pelmOuterLoop;
601 while ((pelmOuterLoop = loopChildren.forAllNodes()) != NULL)
602 {
603 const char *pcszElemName = pelmOuterLoop->getName();
604 if (!strcmp(pcszElemName, "users"))
605 {
606 xml::NodesLoop loopUsers(*pelmOuterLoop);
607 const xml::ElementNode *pelmUser;
608 while ((pelmUser = loopUsers.forAllNodes()) != NULL)
609 {
610 HRESULT hrc = HandleUserAccountsSection(pelmUser);
611 if (FAILED(hrc))
612 return hrc;
613 }
614 }
615 }
616 return S_OK;
617}
618
619HRESULT UnattendedSUSEXMLScript::HandleUserAccountsSection(const xml::ElementNode *pelmSection)
620{
621 xml::NodesLoop loopUser(*pelmSection);
622
623 const xml::ElementNode *pelmCur;
624 while ((pelmCur = loopUser.forAllNodes()) != NULL)
625 {
626 const char *pszValue = pelmCur->getValue();
627#ifdef LOG_ENABLED
628 if (!RTStrCmp(pelmCur->getName(), "uid"))
629 LogRelFunc(("UnattendedSUSEXMLScript::HandleUserAccountsSection profile/users/%s/%s = %s\n",
630 pelmSection->getName(), pelmCur->getName(), pszValue));
631#endif
632
633 if (!RTStrCmp(pszValue, "$homedir"))
634 mNodesForCorrectionMap.insert(make_pair(USERHOMEDIR_ID, pelmCur));
635
636 if (!RTStrCmp(pszValue, "$user"))
637 mNodesForCorrectionMap.insert(make_pair(USERNAME_ID, pelmCur));
638
639 if (!RTStrCmp(pszValue, "$password"))
640 mNodesForCorrectionMap.insert(make_pair(USERPASSWORD_ID, pelmCur));
641 }
642 return S_OK;
643}
644
645Utf8Str UnattendedSUSEXMLScript::createProbableValue(const DataId enmDataId, const xml::ElementNode *pCurElem)
646{
647 const xml::ElementNode *pElem = pCurElem;
648
649 switch (enmDataId)
650 {
651 case USERHOMEDIR_ID:
652// if ((pElem = pElem->findChildElement("home")))
653// {
654 return createProbableUserHomeDir(pElem);
655// }
656 break;
657 default:
658 break;
659 }
660
661 return Utf8Str::Empty;
662}
663
664Utf8Str UnattendedSUSEXMLScript::createProbableUserHomeDir(const xml::ElementNode *pCurElem)
665{
666 Utf8Str strCalcValue;
667 const xml::ElementNode *pElem = pCurElem->findNextSibilingElement("username");
668 if (pElem)
669 {
670 const char *pszValue = pElem->getValue();
671 strCalcValue = "/home/";
672 strCalcValue.append(pszValue);
673 }
674
675 return strCalcValue;
676}
677#endif /* just for reference */
678
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