Installer SignatureValidator implementation
[platform/framework/native/installer.git] / src / Manager / ConfigurationManager.cpp
1 //
2 // Open Service Platform
3 // Copyright (c) 2012 Samsung Electronics Co., Ltd.
4 //
5 // Licensed under the Apache License, Version 2.0 (the License);
6 // you may not use this file except in compliance with the License.
7 // You may obtain a copy of the License at
8 //
9 //     http://www.apache.org/licenses/LICENSE-2.0
10 //
11 // Unless required by applicable law or agreed to in writing, software
12 // distributed under the License is distributed on an "AS IS" BASIS,
13 // WITHOUT  WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 // See the License for the specific language governing permissions and
15 // limitations under the License.
16 //
17 /**
18  * @file        ConfigurationManager.cpp
19  * @brief       This is the implementation file for %ConfigurationManager class.
20  */
21
22 #include <stdio.h>
23 #include <sys/stat.h>
24 #include <unistd.h>
25 #include <unique_ptr.h>
26
27 #include <pkgmgr_parser.h>
28 #include <pkgmgr_installer.h>
29 #include <app2ext_interface.h>
30
31 #include <FBaseUtilStringUtil.h>
32 #include <FIoFile.h>
33 #include <FIoDirectory.h>
34 #include <FSecCertX509CertificatePath.h>
35 #include <FApp_Aul.h>
36 #include <FBase_StringConverter.h>
37
38 #include "ConfigurationManager.h"
39 #include "ManifestGenerator.h"
40 #include "InstallerUtil.h"
41 #include "SmackManager.h"
42
43 using namespace Tizen::Base;
44 using namespace Tizen::Base::Collection;
45 using namespace Tizen::Base::Utility;
46 using namespace Tizen::Security::Cert;
47 using namespace Tizen::App;
48 using namespace Tizen::Io;
49
50 ConfigurationManager::ConfigurationManager(void)
51 {
52 }
53
54 ConfigurationManager::~ConfigurationManager(void)
55 {
56 }
57
58 bool
59 ConfigurationManager::CreateFile(InstallationContext* pContext)
60 {
61         int err = 0;
62
63         String rootPath;
64         rootPath = pContext->__rootPath;
65
66         String apiVersion;
67         apiVersion = pContext->__apiVersion;
68         String versionInfoFile;
69         versionInfoFile.Format(1024, L"%ls%ls", rootPath.GetPointer(), VERSION_INFO_FILE);
70         InstallerUtil::CreateInfoFile(versionInfoFile, &apiVersion);
71
72         if (pContext->__isOspCompat == true)
73         {
74                 AppLog("[OspCompat] is detected");
75
76                 String compatInfoFile;
77                 compatInfoFile.Format(1024, L"%ls%ls", rootPath.GetPointer(), COMPAT_INFO_FILE);
78                 InstallerUtil::CreateInfoFile(compatInfoFile, null);
79         }
80
81         String webServicePrivilege(TIZEN_PRIVILEGE_WEB_SERVICE);
82         if (FindPrivilege(pContext, webServicePrivilege) == true)
83         {
84                 AppLog("WEB_SERVICE privilege is detected. rootPath=[%ls]", rootPath.GetPointer());
85
86                 String webServiceInfoFile;
87                 webServiceInfoFile.Format(1024, L"%ls%ls", rootPath.GetPointer(), WEBSERVICE_INFO_FILE);
88                 InstallerUtil::CreateInfoFile(webServiceInfoFile, null);
89         }
90
91         IListT<AppData*>* pAppDataList = pContext->__pAppDataList;
92         TryReturn(pAppDataList, false, "pAppDataList is null");
93
94         for (int i = 0 ; i < pAppDataList->GetCount(); i++)
95         {
96                 AppData* pAppData = null;
97                 pAppDataList->GetAt(i, pAppData);
98                 TryReturn(pAppData, false, "pAppData is null");
99
100                 String appName = pAppData->__name;
101                 String appType = pAppData->__type;
102                 AppId appId = pAppData->__appId;
103
104                 String binaryPath;
105                 binaryPath.Format(1024, L"%ls%ls/%ls", rootPath.GetPointer(), DIR_BIN, appName.GetPointer());
106
107                 if (File::IsFileExist(binaryPath) == true)
108                 {
109                         InstallerUtil::Remove(binaryPath);
110                 }
111
112                 if (appType == L"UiApp")
113                 {
114                         //err = symlink(UIAPP_LOADER_PATH, pBinaryPath);
115
116                         AppLog("copy ui app loader");
117                         InstallerUtil::Copy(UIAPP_LOADER_PATH, binaryPath);
118
119                         HashMap* pFeatureList = pAppData->__pFeatureList;
120                         TryReturn(pFeatureList, false, "pFeatureList is null");
121
122                         String coordinateSystem;
123                         String baseScreenSize;
124                         String logicalCoordinate;
125
126                         std::unique_ptr< IMapEnumerator > pEnum(pFeatureList->GetMapEnumeratorN());
127                         TryReturn(pEnum, false, "GetMapEnumeratorN() failed. [%s]", GetErrorMessage(GetLastResult()));
128
129                         while (pEnum->MoveNext() == E_SUCCESS)
130                         {
131                                 String* pKey = static_cast< String* > (pEnum->GetKey());
132                                 TryReturn(pEnum, false, "GetKey() failed. [%s]", GetErrorMessage(GetLastResult()));
133
134                                 String* pValue = static_cast< String* > (pEnum->GetValue());
135                                 TryReturn(pEnum, false, "GetValue() failed. [%s]", GetErrorMessage(GetLastResult()));
136
137                                 if ((*pKey) == L"CoordinateSystem")
138                                 {
139                                         coordinateSystem = (*pValue);
140                                 }
141
142                                 if ((*pKey) == L"BaseScreenSize")
143                                 {
144                                         baseScreenSize = (*pValue);
145                                 }
146
147                                 if ((*pKey) == L"LogicalCoordinate")
148                                 {
149                                         logicalCoordinate = (*pValue);
150                                 }
151                         }
152
153                         String uiScalability;
154                         uiScalability.Format(1024, UISCALABILITY_INFO, coordinateSystem.GetPointer(), baseScreenSize.GetPointer(), logicalCoordinate.GetPointer());
155
156                         String uiScalabilityInfoFile;
157                         uiScalabilityInfoFile.Format(1024, L"%ls%ls", rootPath.GetPointer(), UISCALABILITY_INFO_FILE);
158
159                         InstallerUtil::CreateInfoFile(uiScalabilityInfoFile, &uiScalability);
160
161                         int categoryType = pAppData->__feature;
162                         if (categoryType != CATEGORY_TYPE_NONE)
163                         {
164                                 String category = InstallerUtil::GetCategory(categoryType);
165                                 category.ToLowerCase();
166
167                                 int type = _Aul::GetAppType(category);
168
169                                 if (category == L"ime")
170                                 {
171                                         CreateImeSymlink(binaryPath, appId);
172                                 }
173
174                                 String typeInfo;
175                                 typeInfo.Format(1024, L"%d", type);
176
177                                 String typeInfoFile;
178                                 typeInfoFile.Format(1024, L"%ls%ls", rootPath.GetPointer(), TYPE_INFO_FILE);
179
180                                 InstallerUtil::CreateInfoFile(typeInfoFile, &typeInfo);
181                         }
182                 }
183                 else if (appType == L"ServiceApp")
184                 {
185                         if (pContext->__isPreloaded == true || pAppData->__isSystemService == true)
186                         {
187                                 AppLog("copy system service loader");
188                                 InstallerUtil::Copy(SYSTEMSERIVCE_LOADER_PATH, binaryPath);
189                         }
190                         else
191                         {
192                                 AppLog("copy service app loader");
193                                 InstallerUtil::Copy(SERVICEAPP_LOADER_PATH, binaryPath);
194                         }
195                 }
196
197                 InstallerUtil::ChangeMode(binaryPath, S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH);
198         }
199
200         ArrayList* pContentDataList = pContext->GetContentDataList();
201         if (pContentDataList)
202         {
203                 int contentCount = pContentDataList->GetCount();
204                 for (int i = 0 ; i < contentCount; i++)
205                 {
206                         ContentData* pContentData = static_cast<ContentData*>(pContentDataList->GetAt(i));
207                         if (pContentData)
208                         {
209                                 String contentId = pContentData->GetContentId();
210                                 String oldPath = rootPath + DIR_CONTENTS + L"/" + contentId;
211
212                                 if (File::IsFileExist(SLP_FONT_PATH) == false)
213                                 {
214                                         Directory::Create(SLP_FONT_PATH, false);
215                                         InstallerUtil::ChangeMode(SLP_FONT_PATH, S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH);
216                                 }
217
218                                 String newPath = SLP_FONT_PATH;
219                                 newPath += L"/" + pContext->__packageId;
220                                 Directory::Create(newPath, false);
221                                 InstallerUtil::ChangeMode(newPath, S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH);
222
223                                 newPath += L"/";
224                                 newPath += contentId;
225                                 InstallerUtil::CreateSymlink(oldPath, newPath);
226                         }
227                 }
228         }
229
230         if (pContext->__isPreloaded == true)
231         {
232                 pContext->__coreXmlPath.Format(1024, DIR_RO_PACKAGE_SYSTEM_MANIFEST, pContext->__packageId.GetPointer());
233         }
234         else
235         {
236                 pContext->__coreXmlPath.Format(1024, DIR_RW_PACKAGE_SYSTEM_MANIFEST, pContext->__packageId.GetPointer());
237         }
238
239         std::unique_ptr<char[]> pXmlPath(_StringConverter::CopyToCharArrayN(pContext->__coreXmlPath));
240         TryReturn(pXmlPath != null, false, "pXmlPath is null");
241
242         if (pContext->__isHybridService == true)
243         {
244                 String webXmlPath = pXmlPath.get();
245                 webXmlPath += L".wgt";
246                 InstallerUtil::Copy(pXmlPath.get(), webXmlPath);
247
248                 AppLog("pkgmgr_parser_parse_manifest_for_uninstallation() - START");
249                 pkgmgr_parser_parse_manifest_for_uninstallation(pXmlPath.get(), null);
250                 AppLog("pkgmgr_parser_parse_manifest_for_uninstallation() - END");
251                 CreateCoreXmlFile(pContext);
252
253                 String serviceXmlPath = pXmlPath.get();
254                 serviceXmlPath += L".tpk";
255                 InstallerUtil::Copy(pXmlPath.get(), serviceXmlPath);
256                 InstallerUtil::Remove(pXmlPath.get());
257
258                 MergeToSystemXmlFile(pXmlPath.get(), webXmlPath, serviceXmlPath);
259                 InstallerUtil::Remove(webXmlPath);
260                 InstallerUtil::Remove(serviceXmlPath);
261         }
262         else
263         {
264                 AppLog("pkgmgr_parser_parse_manifest_for_uninstallation() - START");
265                 pkgmgr_parser_parse_manifest_for_uninstallation(pXmlPath.get(), null);
266                 AppLog("pkgmgr_parser_parse_manifest_for_uninstallation() - END");
267                 CreateCoreXmlFile(pContext);
268         }
269         
270         AppLog("------------------------------------------");
271         AppLog("sync() - START");
272         sync();
273         AppLog("sync() - END");
274         AppLog("------------------------------------------");
275         
276         err = pkgmgr_parser_check_manifest_validation(pXmlPath.get());
277         TryReturn(err == 0, false, "pkgmgr_parser_check_manifest_validation() is failed. error = [%d][%s]", err, pXmlPath.get());
278
279         return true;
280 }
281
282 bool
283 ConfigurationManager::RemoveFile(InstallationContext* pContext)
284 {
285         if (pContext->__isPreloaded == true)
286         {
287                 pContext->__coreXmlPath.Format(1024, DIR_RO_PACKAGE_SYSTEM_MANIFEST, pContext->__packageId.GetPointer());
288         }
289         else
290         {
291                 pContext->__coreXmlPath.Format(1024, DIR_RW_PACKAGE_SYSTEM_MANIFEST, pContext->__packageId.GetPointer());
292         }
293
294         std::unique_ptr<char[]> pXmlPath(_StringConverter::CopyToCharArrayN(pContext->__coreXmlPath));
295         TryReturn(pXmlPath != null, false, "pXmlPath is null");
296
297         if (pContext->__isHybridService == true)
298         {
299                 AppLog("Uninstallation for HybridService - skip");
300
301         }
302         else
303         {
304                 AppLog("pkgmgr_parser_parse_manifest_for_uninstallation() START");
305                 if (pkgmgr_parser_parse_manifest_for_uninstallation(pXmlPath.get(), null) != 0)
306                 {
307                         AppLog("pkgmgr_parser_parse_manifest_for_uninstallation() is failed.[%s]", pXmlPath.get());
308                 }
309                 AppLog("pkgmgr_parser_parse_manifest_for_uninstallation() END");
310
311                 InstallerUtil::Remove(pXmlPath.get());
312         }
313
314         return true;
315 }
316
317 bool
318 ConfigurationManager::PostInstall(InstallationContext* pContext, bool error) const
319 {
320         app2ext_handle* pHandle = (app2ext_handle*)pContext->__pApp2ExtHandle;
321
322         PackageId packageId = pContext->__packageId;
323         std::unique_ptr<char[]> pPackageId(_StringConverter::CopyToCharArrayN(packageId));
324         TryReturn(pPackageId, false, "pPackageId is null");
325
326         if (error == false)
327         {
328                 std::unique_ptr<char[]> pXmlPath(_StringConverter::CopyToCharArrayN(pContext->__coreXmlPath));
329                 TryReturn(pXmlPath != null, false, "pXmlPath is null");
330
331                 AppLog("------------------------------------------");
332                 AppLog("pkgmgr_parser_parse_manifest_for_installation() - START");
333                 int err = pkgmgr_parser_parse_manifest_for_installation(pXmlPath.get(), null);
334                 TryReturn(err == 0, false, "pkgmgr_parser_parse_manifest_for_installation() is failed. error = [%d][%s]", err, pXmlPath.get());
335
336                 AppLog("pkgmgr_parser_parse_manifest_for_installation() - END");
337                 AppLog("------------------------------------------");
338
339                 SmackManager smackManager;
340                 smackManager.Construct(pContext);
341                 smackManager.EnablePermissions(packageId);
342         }
343
344         if (pHandle)
345         {
346                 if (error == true)
347                 {
348                         pHandle->interface.post_install(pPackageId.get(), APP2EXT_STATUS_FAILED);
349                         AppLog("[app2sd] post_install(%s, APP2EXT_STATUS_FAILED)", pPackageId.get());
350                 }
351                 else
352                 {
353                         pHandle->interface.post_install(pPackageId.get(), APP2EXT_STATUS_SUCCESS);
354                         AppLog("[app2sd] post_install(%s, APP2EXT_STATUS_SUCCESS)", pPackageId.get());
355                 }
356
357                 app2ext_deinit(pHandle);
358         }
359
360         return true;
361 }
362
363 bool
364 ConfigurationManager::PostUninstall(InstallationContext* pContext) const
365 {
366         app2ext_handle* pHandle = (app2ext_handle*)pContext->__pApp2ExtHandle;
367
368         PackageId packageId = pContext->__packageId;
369         std::unique_ptr<char[]> pPackageId(_StringConverter::CopyToCharArrayN(packageId));
370         TryReturn(pPackageId, false, "pPackageId is null.");
371
372         if (pHandle)
373         {
374                 pHandle->interface.post_uninstall(pPackageId.get());
375                 AppLog("[app2sd] post_uninstall(%s)", pPackageId.get());
376
377                 app2ext_deinit(pHandle);
378         }
379
380         SmackManager smackManager;
381         smackManager.Construct(pContext);
382         smackManager.RevokePermissions(packageId);
383
384         return true;
385 }
386
387 bool
388 ConfigurationManager::CreateCoreXmlFile(InstallationContext* pContext)
389 {
390         AppLog("------------------------------------------");
391         AppLog("CreateCoreXmlFile() - START");
392         bool ret = false;
393
394         ManifestGenerator manifestGenerator;
395         manifestGenerator.Construct(pContext);
396         ret = manifestGenerator.Write();
397
398         if (ret == false)
399         {
400                 AppLog("manifestGenerator.Write() is failed.");
401         }
402         AppLog("CreateCoreXmlFile() - END");
403         AppLog("------------------------------------------");
404
405         return ret;
406 }
407
408 //bool
409 //ConfigurationManager::CreateHybridServiceDesktopFile(InstallationContext* pContext)
410 //{
411 //      AppLog("HybridServiceDesktopFile - START");
412 //
413 //      bool ret = true;
414 //      _PackageInfoImpl *pPackageInfoImpl = pContext->GetPackageInfoImpl();
415 //      _PackageAppInfoImpl* pAppInfoImpl = null;
416 //      ArrayList* pAppList = null;
417 //      pAppList = pPackageInfoImpl->GetAppInfoList();
418 //
419 //      for (int i = 0 ; i < pAppList->GetCount(); i++)
420 //      {
421 //              pAppInfoImpl = dynamic_cast<_PackageAppInfoImpl*>(pAppList->GetAt(i));
422 //
423 //              if (pAppInfoImpl)
424 //              {
425 //                      String name;
426 //                      name.Format(1024, L"%ls", pPackageInfoImpl->GetAppName().GetPointer());
427 //
428 //                      String binaryPath;
429 //                      binaryPath.Format(1024, L"%ls%ls/%ls", pPackageInfoImpl->GetAppRootPath().GetPointer(), DIR_BIN, pAppInfoImpl->GetName().GetPointer());
430 //
431 //                      String iconPath;
432 //                      iconPath.Format(1024, L"%ls%ls/%ls", pPackageInfoImpl->GetAppRootPath().GetPointer(), DIR_ICONS, pAppInfoImpl->GetMainmenuIcon().GetPointer());
433 //
434 //                      String version;
435 //                      version.Format(1024, L"%ls", pPackageInfoImpl->GetVersion().GetPointer());
436 //
437 //                      String desktop;
438 //                      desktop.Format(1024, L"[Desktop Entry]\n"
439 //                                                                                                        "Name=%ls\n"
440 //                                                                                                              "Name[en_GB]=%ls\n"
441 //                                                                                                              "Type=Application\n"
442 //                                                                                                      "Exec=%ls\n"
443 //                                                                                                        "Icon=%ls\n"
444 //                                                                                                        "Version=%ls\n"
445 //                                                                                                        "NoDisplay=true\n"
446 //                                                                                                        "X-TIZEN-TaskManage=False\n"
447 //                                                                                                        "X-TIZEN-PackageType=tpk\n",
448 //                                                                                                        name.GetPointer(), name.GetPointer(), binaryPath.GetPointer(), iconPath.GetPointer(), version.GetPointer());
449 //
450 //                      String desktopPath;
451 //                      desktopPath.Format(1024, L"/opt/share/applications/%ls.desktop", pAppInfoImpl->GetPackageName().GetPointer());
452 //
453 //                      if (File::IsFileExist(desktopPath) == true)
454 //                      {
455 //                              AppLog("removing desktop filePath=[%ls]", desktopPath.GetPointer());
456 //                              InstallerUtil::Remove(desktopPath);
457 //                      }
458 //
459 //                      InstallerUtil::CreateInfoFile(desktopPath, &desktop);
460 //              }
461 //      }
462 //
463 //      AppLog("HybridServiceDesktopFile - END");
464 //
465 //      return ret;
466 //}
467
468 bool
469 ConfigurationManager::MergeToSystemXmlFile(const String& systemXmlPath, const String& webXmlPath, const String& serviceXmlPath)
470 {
471         result r = E_SUCCESS;
472
473         FileAttributes webXmlAttr;
474         r = File::GetAttributes(webXmlPath, webXmlAttr);
475         TryReturn(!IsFailed(r), false, "File::GetAttributes() failed, webXmlPath=%ls", webXmlPath.GetPointer());
476
477         FileAttributes serviceAttr;
478         r = File::GetAttributes(serviceXmlPath, serviceAttr);
479         TryReturn(!IsFailed(r), false, "File::GetAttributes() failed, serviceXmlPath=%ls", serviceXmlPath.GetPointer());
480
481         long long webXmlFileSize = webXmlAttr.GetFileSize();
482         long long serviceXmlFileSize = serviceAttr.GetFileSize();
483         long long mergedSize = webXmlFileSize + serviceXmlFileSize;
484
485         File webXml;
486         r = webXml.Construct(webXmlPath, L"r");
487         TryReturn(!IsFailed(r), false, "webXmlPath.Construct is failed");
488
489         std::unique_ptr<char[]> pMergedBuf(new (std::nothrow) char[mergedSize + 1]);
490         TryReturn(pMergedBuf, false, "pMergedBuf is null");
491         memset(pMergedBuf.get(), 0, mergedSize + 1);
492
493         int readBytes = webXml.Read(pMergedBuf.get(), webXmlFileSize);
494         TryReturn(readBytes >= 0, false, "webXml.Read is failed");
495
496         File serviceXml;
497         r = serviceXml.Construct(serviceXmlPath, L"r");
498         TryReturn(!IsFailed(r), false, "serviceXmlPath.Construct is failed");
499
500         std::unique_ptr<char[]> pServiceBuf(new (std::nothrow) char[serviceXmlFileSize + 1]);
501         TryReturn(pServiceBuf, false, "pServiceBuf is null");
502         memset(pServiceBuf.get(), 0, serviceXmlFileSize + 1);
503
504         readBytes = serviceXml.Read(pServiceBuf.get(), serviceXmlFileSize);
505         TryReturn(readBytes >= 0, false, "serviceXml.Read is failed");
506
507         char* pManifestTag = strcasestr(pMergedBuf.get(), "</manifest>");
508         TryReturn(pManifestTag, false, "pManifestTag is null");
509
510         char* pAppTagStart = strcasestr(pServiceBuf.get(), "<ui-application");
511         TryReturn(pAppTagStart, false, "pAppTagStart is null");
512
513         char* pAppTagEnd = strcasestr(pServiceBuf.get(), "</ui-application>");
514         TryReturn(pAppTagEnd, false, "pAppTagEnd is null");
515
516         int serviceAppLen = pAppTagEnd - pAppTagStart + strlen("</ui-application>");
517
518         memcpy(pManifestTag, pAppTagStart, serviceAppLen);
519
520         char* pManifestEndTag = pManifestTag + serviceAppLen;
521         strcpy(pManifestEndTag, "\n</manifest>");
522
523         int fileSize = pManifestEndTag - pMergedBuf.get() + strlen("\n</manifest>");
524
525         File systemXml;
526         r = systemXml.Construct(systemXmlPath, L"w");
527         TryReturn(!IsFailed(r), false, "systemXmlPath.Construct() is failed.");
528
529         r = systemXml.Write(pMergedBuf.get(), fileSize);
530         TryReturn(!IsFailed(r), false, "systemXmlPath.Write() is failed.");
531
532         AppLog("pMergedBuf.get()=0x%0x, length=%d", (unsigned int)pMergedBuf.get(), fileSize);
533         InstallerUtil::DumpLogData(pMergedBuf.get(), fileSize);
534
535         return true;
536 }
537
538 bool
539 ConfigurationManager::CreateImeSymlink(const String& binaryPath, const String& appId)
540 {
541         bool res = true;
542         int err = 0;
543         const char* pExePath = null;
544         const char* pSymlinkPath = null;
545
546         Directory::Create(IME_PATH, true);
547
548         String exePath;
549         exePath.Format(1024, L"%ls.exe", binaryPath.GetPointer());
550
551         String symlinkPath;
552         symlinkPath.Format(1024, L"%s/%ls.so", IME_PATH, appId.GetPointer());
553
554         pExePath = _StringConverter::CopyToCharArrayN(exePath);
555         TryCatch(pExePath, res = false, "pExePath is null");
556
557         pSymlinkPath = _StringConverter::CopyToCharArrayN(symlinkPath);
558         TryCatch(pSymlinkPath, res = false, "pSymlinkPath is null");
559
560         err = symlink(pExePath, pSymlinkPath);
561
562         AppLog("[%s] -> [%s]", pSymlinkPath, pExePath);
563
564 CATCH:
565         delete[] pExePath;
566         delete[] pSymlinkPath;
567
568         return res;
569 }
570
571 bool
572 ConfigurationManager::FindPrivilege(InstallationContext* pContext, const String& privilege) const
573 {
574         TryReturn(pContext, false, "pContext is null.");
575
576         bool ret = false;
577         const ArrayList* pPrivilegeList = pContext->GetPrivilegeList();
578
579         if (pPrivilegeList)
580         {
581                 if (pPrivilegeList->Contains(privilege) == true)
582                 {
583                         AppLog("Privilege = [%ls]", privilege.GetPointer());
584                         ret = true;
585                 }
586         }
587
588         return ret;
589 }