Update SmackManager for enablepermissions()
[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 //      _PackageInfoImpl* pPackageInfoImpl = pContext->GetPackageInfoImpl();
315 //      TryReturn(pPackageInfoImpl, false, "[osp-installer] pPackageInfoImpl is null.");
316 //
317 //      ArrayList* pAppList = pPackageInfoImpl->GetAppInfoList();
318 //      for (int i = 0 ; i < pAppList->GetCount(); i++)
319 //      {
320 //              _PackageAppInfoImpl* pAppInfoImpl = dynamic_cast<_PackageAppInfoImpl*>(pAppList->GetAt(i));
321 //              if (pAppInfoImpl)
322 //              {
323 //                      String packageName = pAppInfoImpl->GetPackageName();
324 //                      String destPath;
325 //                      char    dest[772] = {0};
326 //
327 //                      destPath.Format(1024, L"%ls/%ls", SLP_APP_PATH, packageName.GetPointer());
328 //                      sprintf(dest, "%ls", destPath.GetPointer());
329 //                      unlink(dest);
330 //              }
331 //      }
332
333         return true;
334 }
335
336 bool
337 ConfigurationManager::PostInstall(InstallationContext* pContext, bool error) const
338 {
339         app2ext_handle* pHandle = (app2ext_handle*)pContext->__pApp2ExtHandle;
340
341         PackageId packageId = pContext->__packageId;
342         std::unique_ptr<char[]> pPackageId(_StringConverter::CopyToCharArrayN(packageId));
343         TryReturn(pPackageId, false, "pPackageId is null");
344
345         if (error == false)
346         {
347                 std::unique_ptr<char[]> pXmlPath(_StringConverter::CopyToCharArrayN(pContext->__coreXmlPath));
348                 TryReturn(pXmlPath != null, false, "pXmlPath is null");
349
350                 AppLog("------------------------------------------");
351                 AppLog("pkgmgr_parser_parse_manifest_for_installation() - START");
352                 int err = pkgmgr_parser_parse_manifest_for_installation(pXmlPath.get(), null);
353                 TryReturn(err == 0, false, "pkgmgr_parser_parse_manifest_for_installation() is failed. error = [%d][%s]", err, pXmlPath.get());
354
355                 AppLog("pkgmgr_parser_parse_manifest_for_installation() - END");
356                 AppLog("------------------------------------------");
357
358                 SmackManager smackManager;
359                 smackManager.Construct(pContext);
360                 smackManager.EnablePermissions(packageId);
361         }
362
363         if (pHandle)
364         {
365                 if (error == true)
366                 {
367                         pHandle->interface.post_install(pPackageId.get(), APP2EXT_STATUS_FAILED);
368                         AppLog("[app2sd] post_install(%s, APP2EXT_STATUS_FAILED)", pPackageId.get());
369                 }
370                 else
371                 {
372                         pHandle->interface.post_install(pPackageId.get(), APP2EXT_STATUS_SUCCESS);
373                         AppLog("[app2sd] post_install(%s, APP2EXT_STATUS_SUCCESS)", pPackageId.get());
374                 }
375
376                 app2ext_deinit(pHandle);
377         }
378
379         return true;
380 }
381
382 bool
383 ConfigurationManager::PostUninstall(InstallationContext* pContext) const
384 {
385         app2ext_handle* pHandle = (app2ext_handle*)pContext->__pApp2ExtHandle;
386
387         PackageId packageId = pContext->__packageId;
388         std::unique_ptr<char[]> pPackageId(_StringConverter::CopyToCharArrayN(packageId));
389         TryReturn(pPackageId, false, "pPackageId is null.");
390
391         if (pHandle)
392         {
393                 pHandle->interface.post_uninstall(pPackageId.get());
394                 AppLog("[app2sd] post_uninstall(%s)", pPackageId.get());
395
396                 app2ext_deinit(pHandle);
397         }
398
399         SmackManager smackManager;
400         smackManager.Construct(pContext);
401         smackManager.RevokePermissions(packageId);
402
403         return true;
404 }
405
406 bool
407 ConfigurationManager::CreateCoreXmlFile(InstallationContext* pContext)
408 {
409         AppLog("------------------------------------------");
410         AppLog("CreateCoreXmlFile() - START");
411         bool ret = false;
412
413         ManifestGenerator manifestGenerator;
414         manifestGenerator.Construct(pContext);
415         ret = manifestGenerator.Write();
416
417         if (ret == false)
418         {
419                 AppLog("manifestGenerator.Write() is failed.");
420         }
421         AppLog("CreateCoreXmlFile() - END");
422         AppLog("------------------------------------------");
423
424         return ret;
425 }
426
427 //bool
428 //ConfigurationManager::CreateHybridServiceDesktopFile(InstallationContext* pContext)
429 //{
430 //      AppLog("HybridServiceDesktopFile - START");
431 //
432 //      bool ret = true;
433 //      _PackageInfoImpl *pPackageInfoImpl = pContext->GetPackageInfoImpl();
434 //      _PackageAppInfoImpl* pAppInfoImpl = null;
435 //      ArrayList* pAppList = null;
436 //      pAppList = pPackageInfoImpl->GetAppInfoList();
437 //
438 //      for (int i = 0 ; i < pAppList->GetCount(); i++)
439 //      {
440 //              pAppInfoImpl = dynamic_cast<_PackageAppInfoImpl*>(pAppList->GetAt(i));
441 //
442 //              if (pAppInfoImpl)
443 //              {
444 //                      String name;
445 //                      name.Format(1024, L"%ls", pPackageInfoImpl->GetAppName().GetPointer());
446 //
447 //                      String binaryPath;
448 //                      binaryPath.Format(1024, L"%ls%ls/%ls", pPackageInfoImpl->GetAppRootPath().GetPointer(), DIR_BIN, pAppInfoImpl->GetName().GetPointer());
449 //
450 //                      String iconPath;
451 //                      iconPath.Format(1024, L"%ls%ls/%ls", pPackageInfoImpl->GetAppRootPath().GetPointer(), DIR_ICONS, pAppInfoImpl->GetMainmenuIcon().GetPointer());
452 //
453 //                      String version;
454 //                      version.Format(1024, L"%ls", pPackageInfoImpl->GetVersion().GetPointer());
455 //
456 //                      String desktop;
457 //                      desktop.Format(1024, L"[Desktop Entry]\n"
458 //                                                                                                        "Name=%ls\n"
459 //                                                                                                              "Name[en_GB]=%ls\n"
460 //                                                                                                              "Type=Application\n"
461 //                                                                                                      "Exec=%ls\n"
462 //                                                                                                        "Icon=%ls\n"
463 //                                                                                                        "Version=%ls\n"
464 //                                                                                                        "NoDisplay=true\n"
465 //                                                                                                        "X-TIZEN-TaskManage=False\n"
466 //                                                                                                        "X-TIZEN-PackageType=tpk\n",
467 //                                                                                                        name.GetPointer(), name.GetPointer(), binaryPath.GetPointer(), iconPath.GetPointer(), version.GetPointer());
468 //
469 //                      String desktopPath;
470 //                      desktopPath.Format(1024, L"/opt/share/applications/%ls.desktop", pAppInfoImpl->GetPackageName().GetPointer());
471 //
472 //                      if (File::IsFileExist(desktopPath) == true)
473 //                      {
474 //                              AppLog("removing desktop filePath=[%ls]", desktopPath.GetPointer());
475 //                              InstallerUtil::Remove(desktopPath);
476 //                      }
477 //
478 //                      InstallerUtil::CreateInfoFile(desktopPath, &desktop);
479 //              }
480 //      }
481 //
482 //      AppLog("HybridServiceDesktopFile - END");
483 //
484 //      return ret;
485 //}
486
487 bool
488 ConfigurationManager::MergeToSystemXmlFile(const String& systemXmlPath, const String& webXmlPath, const String& serviceXmlPath)
489 {
490         result r = E_SUCCESS;
491
492         FileAttributes webXmlAttr;
493         r = File::GetAttributes(webXmlPath, webXmlAttr);
494         TryReturn(!IsFailed(r), false, "File::GetAttributes() failed, webXmlPath=%ls", webXmlPath.GetPointer());
495
496         FileAttributes serviceAttr;
497         r = File::GetAttributes(serviceXmlPath, serviceAttr);
498         TryReturn(!IsFailed(r), false, "File::GetAttributes() failed, serviceXmlPath=%ls", serviceXmlPath.GetPointer());
499
500         long long webXmlFileSize = webXmlAttr.GetFileSize();
501         long long serviceXmlFileSize = serviceAttr.GetFileSize();
502         long long mergedSize = webXmlFileSize + serviceXmlFileSize;
503
504         File webXml;
505         r = webXml.Construct(webXmlPath, L"r");
506         TryReturn(!IsFailed(r), false, "webXmlPath.Construct is failed");
507
508         std::unique_ptr<char[]> pMergedBuf(new (std::nothrow) char[mergedSize + 1]);
509         TryReturn(pMergedBuf, false, "pMergedBuf is null");
510         memset(pMergedBuf.get(), 0, mergedSize + 1);
511
512         int readBytes = webXml.Read(pMergedBuf.get(), webXmlFileSize);
513         TryReturn(readBytes >= 0, false, "webXml.Read is failed");
514
515         File serviceXml;
516         r = serviceXml.Construct(serviceXmlPath, L"r");
517         TryReturn(!IsFailed(r), false, "serviceXmlPath.Construct is failed");
518
519         std::unique_ptr<char[]> pServiceBuf(new (std::nothrow) char[serviceXmlFileSize + 1]);
520         TryReturn(pServiceBuf, false, "pServiceBuf is null");
521         memset(pServiceBuf.get(), 0, serviceXmlFileSize + 1);
522
523         readBytes = serviceXml.Read(pServiceBuf.get(), serviceXmlFileSize);
524         TryReturn(readBytes >= 0, false, "serviceXml.Read is failed");
525
526         char* pManifestTag = strcasestr(pMergedBuf.get(), "</manifest>");
527         TryReturn(pManifestTag, false, "pManifestTag is null");
528
529         char* pAppTagStart = strcasestr(pServiceBuf.get(), "<ui-application");
530         TryReturn(pAppTagStart, false, "pAppTagStart is null");
531
532         char* pAppTagEnd = strcasestr(pServiceBuf.get(), "</ui-application>");
533         TryReturn(pAppTagEnd, false, "pAppTagEnd is null");
534
535         int serviceAppLen = pAppTagEnd - pAppTagStart + strlen("</ui-application>");
536
537         memcpy(pManifestTag, pAppTagStart, serviceAppLen);
538
539         char* pManifestEndTag = pManifestTag + serviceAppLen;
540         strcpy(pManifestEndTag, "\n</manifest>");
541
542         int fileSize = pManifestEndTag - pMergedBuf.get() + strlen("\n</manifest>");
543
544         File systemXml;
545         r = systemXml.Construct(systemXmlPath, L"w");
546         TryReturn(!IsFailed(r), false, "systemXmlPath.Construct() is failed.");
547
548         r = systemXml.Write(pMergedBuf.get(), fileSize);
549         TryReturn(!IsFailed(r), false, "systemXmlPath.Write() is failed.");
550
551         AppLog("pMergedBuf.get()=0x%0x, length=%d", (unsigned int)pMergedBuf.get(), fileSize);
552         InstallerUtil::DumpLogData(pMergedBuf.get(), fileSize);
553
554         return true;
555 }
556
557 bool
558 ConfigurationManager::CreateImeSymlink(const String& binaryPath, const String& appId)
559 {
560         bool res = true;
561         int err = 0;
562         const char* pExePath = null;
563         const char* pSymlinkPath = null;
564
565         Directory::Create(IME_PATH, true);
566
567         String exePath;
568         exePath.Format(1024, L"%ls.exe", binaryPath.GetPointer());
569
570         String symlinkPath;
571         symlinkPath.Format(1024, L"%s/%ls.so", IME_PATH, appId.GetPointer());
572
573         pExePath = _StringConverter::CopyToCharArrayN(exePath);
574         TryCatch(pExePath, res = false, "pExePath is null");
575
576         pSymlinkPath = _StringConverter::CopyToCharArrayN(symlinkPath);
577         TryCatch(pSymlinkPath, res = false, "pSymlinkPath is null");
578
579         err = symlink(pExePath, pSymlinkPath);
580
581         AppLog("[%s] -> [%s]", pSymlinkPath, pExePath);
582
583 CATCH:
584         delete[] pExePath;
585         delete[] pSymlinkPath;
586
587         return res;
588 }
589
590 bool
591 ConfigurationManager::FindPrivilege(InstallationContext* pContext, const String& privilege) const
592 {
593         TryReturn(pContext, false, "pContext is null.");
594
595         bool ret = false;
596         const ArrayList* pPrivilegeList = pContext->GetPrivilegeList();
597
598         if (pPrivilegeList)
599         {
600                 if (pPrivilegeList->Contains(privilege) == true)
601                 {
602                         AppLog("Privilege = [%ls]", privilege.GetPointer());
603                         ret = true;
604                 }
605         }
606
607         return ret;
608 }