cmCPackWIXGenerator.cxx 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170
  1. /*============================================================================
  2. CMake - Cross Platform Makefile Generator
  3. Copyright 2000-2014 Kitware, Inc., Insight Software Consortium
  4. Distributed under the OSI-approved BSD License (the "License");
  5. see accompanying file Copyright.txt for details.
  6. This software is distributed WITHOUT ANY WARRANTY; without even the
  7. implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  8. See the License for more information.
  9. ============================================================================*/
  10. #include "cmCPackWIXGenerator.h"
  11. #include <cmSystemTools.h>
  12. #include <cmGeneratedFileStream.h>
  13. #include <cmCryptoHash.h>
  14. #include <cmInstalledFile.h>
  15. #include <CPack/cmCPackLog.h>
  16. #include <CPack/cmCPackComponentGroup.h>
  17. #include "cmWIXSourceWriter.h"
  18. #include "cmWIXDirectoriesSourceWriter.h"
  19. #include "cmWIXFeaturesSourceWriter.h"
  20. #include "cmWIXFilesSourceWriter.h"
  21. #include "cmWIXRichTextFormatWriter.h"
  22. #include <cmsys/SystemTools.hxx>
  23. #include <cmsys/Directory.hxx>
  24. #include <cmsys/Encoding.hxx>
  25. #include <cmsys/FStream.hxx>
  26. #include <rpc.h> // for GUID generation
  27. cmCPackWIXGenerator::cmCPackWIXGenerator():
  28. HasDesktopShortcuts(false),
  29. Patch(0)
  30. {
  31. }
  32. cmCPackWIXGenerator::~cmCPackWIXGenerator()
  33. {
  34. if(this->Patch)
  35. {
  36. delete this->Patch;
  37. }
  38. }
  39. int cmCPackWIXGenerator::InitializeInternal()
  40. {
  41. componentPackageMethod = ONE_PACKAGE;
  42. this->Patch = new cmWIXPatch(this->Logger);
  43. return this->Superclass::InitializeInternal();
  44. }
  45. bool cmCPackWIXGenerator::RunWiXCommand(std::string const& command)
  46. {
  47. std::string logFileName = this->CPackTopLevel + "/wix.log";
  48. cmCPackLogger(cmCPackLog::LOG_DEBUG,
  49. "Running WiX command: " << command << std::endl);
  50. std::string output;
  51. int returnValue = 0;
  52. bool status = cmSystemTools::RunSingleCommand(command.c_str(), &output,
  53. &returnValue, 0, cmSystemTools::OUTPUT_NONE);
  54. cmsys::ofstream logFile(logFileName.c_str(), std::ios::app);
  55. logFile << command << std::endl;
  56. logFile << output;
  57. logFile.close();
  58. if(!status || returnValue)
  59. {
  60. cmCPackLogger(cmCPackLog::LOG_ERROR,
  61. "Problem running WiX candle. "
  62. "Please check '" << logFileName << "' for errors." << std::endl);
  63. return false;
  64. }
  65. return true;
  66. }
  67. bool cmCPackWIXGenerator::RunCandleCommand(
  68. std::string const& sourceFile, std::string const& objectFile)
  69. {
  70. std::string executable;
  71. if(!RequireOption("CPACK_WIX_CANDLE_EXECUTABLE", executable))
  72. {
  73. return false;
  74. }
  75. std::stringstream command;
  76. command << QuotePath(executable);
  77. command << " -nologo";
  78. command << " -arch " << GetArchitecture();
  79. command << " -out " << QuotePath(objectFile);
  80. for(extension_set_t::const_iterator i = CandleExtensions.begin();
  81. i != CandleExtensions.end(); ++i)
  82. {
  83. command << " -ext " << QuotePath(*i);
  84. }
  85. AddCustomFlags("CPACK_WIX_CANDLE_EXTRA_FLAGS", command);
  86. command << " " << QuotePath(sourceFile);
  87. return RunWiXCommand(command.str());
  88. }
  89. bool cmCPackWIXGenerator::RunLightCommand(std::string const& objectFiles)
  90. {
  91. std::string executable;
  92. if(!RequireOption("CPACK_WIX_LIGHT_EXECUTABLE", executable))
  93. {
  94. return false;
  95. }
  96. std::stringstream command;
  97. command << QuotePath(executable);
  98. command << " -nologo";
  99. command << " -out " << QuotePath(packageFileNames.at(0));
  100. for(extension_set_t::const_iterator i = this->LightExtensions.begin();
  101. i != this->LightExtensions.end(); ++i)
  102. {
  103. command << " -ext " << QuotePath(*i);
  104. }
  105. const char* const cultures = GetOption("CPACK_WIX_CULTURES");
  106. if(cultures)
  107. {
  108. command << " -cultures:" << cultures;
  109. }
  110. AddCustomFlags("CPACK_WIX_LIGHT_EXTRA_FLAGS", command);
  111. command << " " << objectFiles;
  112. return RunWiXCommand(command.str());
  113. }
  114. int cmCPackWIXGenerator::PackageFiles()
  115. {
  116. if(!PackageFilesImpl() || cmSystemTools::GetErrorOccuredFlag())
  117. {
  118. cmCPackLogger(cmCPackLog::LOG_ERROR,
  119. "Fatal WiX Generator Error" << std::endl);
  120. return false;
  121. }
  122. return true;
  123. }
  124. bool cmCPackWIXGenerator::InitializeWiXConfiguration()
  125. {
  126. if(!ReadListFile("CPackWIX.cmake"))
  127. {
  128. cmCPackLogger(cmCPackLog::LOG_ERROR,
  129. "Error while executing CPackWIX.cmake" << std::endl);
  130. return false;
  131. }
  132. if(GetOption("CPACK_WIX_PRODUCT_GUID") == 0)
  133. {
  134. std::string guid = GenerateGUID();
  135. SetOption("CPACK_WIX_PRODUCT_GUID", guid.c_str());
  136. cmCPackLogger(cmCPackLog::LOG_VERBOSE,
  137. "CPACK_WIX_PRODUCT_GUID implicitly set to " << guid << " . "
  138. << std::endl);
  139. }
  140. if(GetOption("CPACK_WIX_UPGRADE_GUID") == 0)
  141. {
  142. std::string guid = GenerateGUID();
  143. SetOption("CPACK_WIX_UPGRADE_GUID", guid.c_str());
  144. cmCPackLogger(cmCPackLog::LOG_WARNING,
  145. "CPACK_WIX_UPGRADE_GUID implicitly set to " << guid << " . "
  146. "Please refer to the documentation on how and why "
  147. "you might want to set this explicitly." << std::endl);
  148. }
  149. if(!RequireOption("CPACK_TOPLEVEL_DIRECTORY", this->CPackTopLevel))
  150. {
  151. return false;
  152. }
  153. if(GetOption("CPACK_WIX_LICENSE_RTF") == 0)
  154. {
  155. std::string licenseFilename = this->CPackTopLevel + "/License.rtf";
  156. SetOption("CPACK_WIX_LICENSE_RTF", licenseFilename.c_str());
  157. if(!CreateLicenseFile())
  158. {
  159. return false;
  160. }
  161. }
  162. if(GetOption("CPACK_PACKAGE_VENDOR") == 0)
  163. {
  164. std::string defaultVendor = "Humanity";
  165. SetOption("CPACK_PACKAGE_VENDOR", defaultVendor.c_str());
  166. cmCPackLogger(cmCPackLog::LOG_VERBOSE,
  167. "CPACK_PACKAGE_VENDOR implicitly set to " << defaultVendor << " . "
  168. << std::endl);
  169. }
  170. if(GetOption("CPACK_WIX_UI_REF") == 0)
  171. {
  172. std::string defaultRef = "WixUI_InstallDir";
  173. if(this->Components.size())
  174. {
  175. defaultRef = "WixUI_FeatureTree";
  176. }
  177. SetOption("CPACK_WIX_UI_REF", defaultRef.c_str());
  178. }
  179. const char* packageContact = GetOption("CPACK_PACKAGE_CONTACT");
  180. if(packageContact != 0 &&
  181. GetOption("CPACK_WIX_PROPERTY_ARPCONTACT") == 0)
  182. {
  183. SetOption("CPACK_WIX_PROPERTY_ARPCONTACT", packageContact);
  184. }
  185. CollectExtensions("CPACK_WIX_EXTENSIONS", this->CandleExtensions);
  186. CollectExtensions("CPACK_WIX_CANDLE_EXTENSIONS", this->CandleExtensions);
  187. this->LightExtensions.insert("WixUIExtension");
  188. CollectExtensions("CPACK_WIX_EXTENSIONS", this->LightExtensions);
  189. CollectExtensions("CPACK_WIX_LIGHT_EXTENSIONS", this->LightExtensions);
  190. const char* patchFilePath = GetOption("CPACK_WIX_PATCH_FILE");
  191. if(patchFilePath)
  192. {
  193. this->Patch->LoadFragments(patchFilePath);
  194. }
  195. return true;
  196. }
  197. bool cmCPackWIXGenerator::PackageFilesImpl()
  198. {
  199. if(!InitializeWiXConfiguration())
  200. {
  201. return false;
  202. }
  203. CreateWiXVariablesIncludeFile();
  204. CreateWiXPropertiesIncludeFile();
  205. if(!CreateWiXSourceFiles())
  206. {
  207. return false;
  208. }
  209. AppendUserSuppliedExtraSources();
  210. std::stringstream objectFiles;
  211. for(size_t i = 0; i < this->WixSources.size(); ++i)
  212. {
  213. std::string const& sourceFilename = this->WixSources[i];
  214. std::string objectFilename =
  215. cmSystemTools::GetFilenameWithoutExtension(sourceFilename) + ".wixobj";
  216. if(!RunCandleCommand(sourceFilename, objectFilename))
  217. {
  218. return false;
  219. }
  220. objectFiles << " " << QuotePath(objectFilename);
  221. }
  222. AppendUserSuppliedExtraObjects(objectFiles);
  223. return RunLightCommand(objectFiles.str());
  224. }
  225. void cmCPackWIXGenerator::AppendUserSuppliedExtraSources()
  226. {
  227. const char *cpackWixExtraSources = GetOption("CPACK_WIX_EXTRA_SOURCES");
  228. if(!cpackWixExtraSources) return;
  229. cmSystemTools::ExpandListArgument(cpackWixExtraSources, this->WixSources);
  230. }
  231. void cmCPackWIXGenerator::AppendUserSuppliedExtraObjects(std::ostream& stream)
  232. {
  233. const char *cpackWixExtraObjects = GetOption("CPACK_WIX_EXTRA_OBJECTS");
  234. if(!cpackWixExtraObjects) return;
  235. std::vector<std::string> expandedExtraObjects;
  236. cmSystemTools::ExpandListArgument(
  237. cpackWixExtraObjects, expandedExtraObjects);
  238. for(size_t i = 0; i < expandedExtraObjects.size(); ++i)
  239. {
  240. stream << " " << QuotePath(expandedExtraObjects[i]);
  241. }
  242. }
  243. void cmCPackWIXGenerator::CreateWiXVariablesIncludeFile()
  244. {
  245. std::string includeFilename =
  246. this->CPackTopLevel + "/cpack_variables.wxi";
  247. cmWIXSourceWriter includeFile(
  248. this->Logger, includeFilename, true);
  249. CopyDefinition(includeFile, "CPACK_WIX_PRODUCT_GUID");
  250. CopyDefinition(includeFile, "CPACK_WIX_UPGRADE_GUID");
  251. CopyDefinition(includeFile, "CPACK_PACKAGE_VENDOR");
  252. CopyDefinition(includeFile, "CPACK_PACKAGE_NAME");
  253. CopyDefinition(includeFile, "CPACK_PACKAGE_VERSION");
  254. CopyDefinition(includeFile, "CPACK_WIX_LICENSE_RTF");
  255. CopyDefinition(includeFile, "CPACK_WIX_PRODUCT_ICON");
  256. CopyDefinition(includeFile, "CPACK_WIX_UI_BANNER");
  257. CopyDefinition(includeFile, "CPACK_WIX_UI_DIALOG");
  258. SetOptionIfNotSet("CPACK_WIX_PROGRAM_MENU_FOLDER",
  259. GetOption("CPACK_PACKAGE_NAME"));
  260. CopyDefinition(includeFile, "CPACK_WIX_PROGRAM_MENU_FOLDER");
  261. CopyDefinition(includeFile, "CPACK_WIX_UI_REF");
  262. }
  263. void cmCPackWIXGenerator::CreateWiXPropertiesIncludeFile()
  264. {
  265. std::string includeFilename =
  266. this->CPackTopLevel + "/properties.wxi";
  267. cmWIXSourceWriter includeFile(
  268. this->Logger, includeFilename, true);
  269. std::string prefix = "CPACK_WIX_PROPERTY_";
  270. std::vector<std::string> options = GetOptions();
  271. for(size_t i = 0; i < options.size(); ++i)
  272. {
  273. std::string const& name = options[i];
  274. if(name.length() > prefix.length() &&
  275. name.substr(0, prefix.length()) == prefix)
  276. {
  277. std::string id = name.substr(prefix.length());
  278. std::string value = GetOption(name.c_str());
  279. includeFile.BeginElement("Property");
  280. includeFile.AddAttribute("Id", id);
  281. includeFile.AddAttribute("Value", value);
  282. includeFile.EndElement("Property");
  283. }
  284. }
  285. if(GetOption("CPACK_WIX_PROPERTY_ARPINSTALLLOCATION") == 0)
  286. {
  287. includeFile.BeginElement("Property");
  288. includeFile.AddAttribute("Id", "INSTALL_ROOT");
  289. includeFile.AddAttribute("Secure", "yes");
  290. includeFile.BeginElement("RegistrySearch");
  291. includeFile.AddAttribute("Id", "FindInstallLocation");
  292. includeFile.AddAttribute("Root", "HKLM");
  293. includeFile.AddAttribute("Key", "Software\\Microsoft\\Windows\\"
  294. "CurrentVersion\\Uninstall\\[WIX_UPGRADE_DETECTED]");
  295. includeFile.AddAttribute("Name", "InstallLocation");
  296. includeFile.AddAttribute("Type", "raw");
  297. includeFile.EndElement("RegistrySearch");
  298. includeFile.EndElement("Property");
  299. includeFile.BeginElement("SetProperty");
  300. includeFile.AddAttribute("Id", "ARPINSTALLLOCATION");
  301. includeFile.AddAttribute("Value", "[INSTALL_ROOT]");
  302. includeFile.AddAttribute("After", "CostFinalize");
  303. includeFile.EndElement("SetProperty");
  304. }
  305. }
  306. void cmCPackWIXGenerator::CopyDefinition(
  307. cmWIXSourceWriter &source, std::string const& name)
  308. {
  309. const char* value = GetOption(name.c_str());
  310. if(value)
  311. {
  312. AddDefinition(source, name, value);
  313. }
  314. }
  315. void cmCPackWIXGenerator::AddDefinition(cmWIXSourceWriter& source,
  316. std::string const& name, std::string const& value)
  317. {
  318. std::stringstream tmp;
  319. tmp << name << "=\"" << value << '"';
  320. source.AddProcessingInstruction("define",
  321. cmWIXSourceWriter::WindowsCodepageToUtf8(tmp.str()));
  322. }
  323. bool cmCPackWIXGenerator::CreateWiXSourceFiles()
  324. {
  325. std::string directoryDefinitionsFilename =
  326. this->CPackTopLevel + "/directories.wxs";
  327. this->WixSources.push_back(directoryDefinitionsFilename);
  328. cmWIXDirectoriesSourceWriter directoryDefinitions(
  329. this->Logger, directoryDefinitionsFilename);
  330. directoryDefinitions.BeginElement("Fragment");
  331. std::string installRoot;
  332. if(!RequireOption("CPACK_PACKAGE_INSTALL_DIRECTORY", installRoot))
  333. {
  334. return false;
  335. }
  336. directoryDefinitions.BeginElement("Directory");
  337. directoryDefinitions.AddAttribute("Id", "TARGETDIR");
  338. directoryDefinitions.AddAttribute("Name", "SourceDir");
  339. size_t installRootSize =
  340. directoryDefinitions.BeginInstallationPrefixDirectory(
  341. GetProgramFilesFolderId(), installRoot);
  342. std::string fileDefinitionsFilename =
  343. this->CPackTopLevel + "/files.wxs";
  344. this->WixSources.push_back(fileDefinitionsFilename);
  345. cmWIXFilesSourceWriter fileDefinitions(
  346. this->Logger, fileDefinitionsFilename);
  347. fileDefinitions.BeginElement("Fragment");
  348. std::string featureDefinitionsFilename =
  349. this->CPackTopLevel +"/features.wxs";
  350. this->WixSources.push_back(featureDefinitionsFilename);
  351. cmWIXFeaturesSourceWriter featureDefinitions(
  352. this->Logger, featureDefinitionsFilename);
  353. featureDefinitions.BeginElement("Fragment");
  354. featureDefinitions.BeginElement("Feature");
  355. featureDefinitions.AddAttribute("Id", "ProductFeature");
  356. featureDefinitions.AddAttribute("Display", "expand");
  357. featureDefinitions.AddAttribute("ConfigurableDirectory", "INSTALL_ROOT");
  358. std::string cpackPackageName;
  359. if(!RequireOption("CPACK_PACKAGE_NAME", cpackPackageName))
  360. {
  361. return false;
  362. }
  363. featureDefinitions.AddAttribute("Title", cpackPackageName);
  364. featureDefinitions.AddAttribute("Level", "1");
  365. const char* package = GetOption("CPACK_WIX_CMAKE_PACKAGE_REGISTRY");
  366. if(package)
  367. {
  368. featureDefinitions.CreateCMakePackageRegistryEntry(
  369. package, GetOption("CPACK_WIX_UPGRADE_GUID"));
  370. }
  371. if(!CreateFeatureHierarchy(featureDefinitions))
  372. {
  373. return false;
  374. }
  375. featureDefinitions.EndElement("Feature");
  376. bool hasShortcuts = false;
  377. shortcut_map_t globalShortcuts;
  378. if(Components.empty())
  379. {
  380. AddComponentsToFeature(toplevel, "ProductFeature",
  381. directoryDefinitions, fileDefinitions, featureDefinitions,
  382. globalShortcuts);
  383. if(globalShortcuts.size())
  384. {
  385. hasShortcuts = true;
  386. }
  387. }
  388. else
  389. {
  390. for(std::map<std::string, cmCPackComponent>::const_iterator
  391. i = this->Components.begin(); i != this->Components.end(); ++i)
  392. {
  393. cmCPackComponent const& component = i->second;
  394. std::string componentPath = toplevel;
  395. componentPath += "/";
  396. componentPath += component.Name;
  397. std::string componentFeatureId = "CM_C_" + component.Name;
  398. shortcut_map_t featureShortcuts;
  399. AddComponentsToFeature(componentPath, componentFeatureId,
  400. directoryDefinitions, fileDefinitions,
  401. featureDefinitions, featureShortcuts);
  402. if(featureShortcuts.size())
  403. {
  404. hasShortcuts = true;
  405. }
  406. if(featureShortcuts.size())
  407. {
  408. if(!CreateStartMenuShortcuts(component.Name, componentFeatureId,
  409. featureShortcuts, fileDefinitions, featureDefinitions))
  410. {
  411. return false;
  412. }
  413. }
  414. }
  415. }
  416. if(hasShortcuts)
  417. {
  418. if(!CreateStartMenuShortcuts(std::string(), "ProductFeature",
  419. globalShortcuts, fileDefinitions, featureDefinitions))
  420. {
  421. return false;
  422. }
  423. }
  424. featureDefinitions.EndElement("Fragment");
  425. fileDefinitions.EndElement("Fragment");
  426. directoryDefinitions.EndInstallationPrefixDirectory(
  427. installRootSize);
  428. if(hasShortcuts)
  429. {
  430. directoryDefinitions.EmitStartMenuFolder(
  431. GetOption("CPACK_WIX_PROGRAM_MENU_FOLDER"));
  432. }
  433. if(this->HasDesktopShortcuts)
  434. {
  435. directoryDefinitions.EmitDesktopFolder();
  436. }
  437. directoryDefinitions.EndElement("Directory");
  438. directoryDefinitions.EndElement("Fragment");
  439. if(!GenerateMainSourceFileFromTemplate())
  440. {
  441. return false;
  442. }
  443. return this->Patch->CheckForUnappliedFragments();
  444. }
  445. std::string cmCPackWIXGenerator::GetProgramFilesFolderId() const
  446. {
  447. if(GetArchitecture() == "x86")
  448. {
  449. return "ProgramFilesFolder";
  450. }
  451. else
  452. {
  453. return "ProgramFiles64Folder";
  454. }
  455. }
  456. bool cmCPackWIXGenerator::GenerateMainSourceFileFromTemplate()
  457. {
  458. std::string wixTemplate = FindTemplate("WIX.template.in");
  459. if(GetOption("CPACK_WIX_TEMPLATE") != 0)
  460. {
  461. wixTemplate = GetOption("CPACK_WIX_TEMPLATE");
  462. }
  463. if(wixTemplate.empty())
  464. {
  465. cmCPackLogger(cmCPackLog::LOG_ERROR,
  466. "Could not find CPack WiX template file WIX.template.in" << std::endl);
  467. return false;
  468. }
  469. std::string mainSourceFilePath = this->CPackTopLevel + "/main.wxs";
  470. if(!ConfigureFile(wixTemplate.c_str(), mainSourceFilePath .c_str()))
  471. {
  472. cmCPackLogger(cmCPackLog::LOG_ERROR,
  473. "Failed creating '" << mainSourceFilePath <<
  474. "'' from template." << std::endl);
  475. return false;
  476. }
  477. this->WixSources.push_back(mainSourceFilePath);
  478. return true;
  479. }
  480. bool cmCPackWIXGenerator::CreateFeatureHierarchy(
  481. cmWIXFeaturesSourceWriter& featureDefinitions)
  482. {
  483. for(std::map<std::string, cmCPackComponentGroup>::const_iterator
  484. i = ComponentGroups.begin(); i != ComponentGroups.end(); ++i)
  485. {
  486. cmCPackComponentGroup const& group = i->second;
  487. if(group.ParentGroup == 0)
  488. {
  489. featureDefinitions.EmitFeatureForComponentGroup(group);
  490. }
  491. }
  492. for(std::map<std::string, cmCPackComponent>::const_iterator
  493. i = this->Components.begin(); i != this->Components.end(); ++i)
  494. {
  495. cmCPackComponent const& component = i->second;
  496. if(!component.Group)
  497. {
  498. featureDefinitions.EmitFeatureForComponent(component);
  499. }
  500. }
  501. return true;
  502. }
  503. bool cmCPackWIXGenerator::AddComponentsToFeature(
  504. std::string const& rootPath,
  505. std::string const& featureId,
  506. cmWIXDirectoriesSourceWriter& directoryDefinitions,
  507. cmWIXFilesSourceWriter& fileDefinitions,
  508. cmWIXFeaturesSourceWriter& featureDefinitions,
  509. shortcut_map_t& shortcutMap)
  510. {
  511. featureDefinitions.BeginElement("FeatureRef");
  512. featureDefinitions.AddAttribute("Id", featureId);
  513. std::vector<std::string> cpackPackageExecutablesList;
  514. const char *cpackPackageExecutables = GetOption("CPACK_PACKAGE_EXECUTABLES");
  515. if(cpackPackageExecutables)
  516. {
  517. cmSystemTools::ExpandListArgument(cpackPackageExecutables,
  518. cpackPackageExecutablesList);
  519. if(cpackPackageExecutablesList.size() % 2 != 0 )
  520. {
  521. cmCPackLogger(cmCPackLog::LOG_ERROR,
  522. "CPACK_PACKAGE_EXECUTABLES should contain pairs of <executable> and "
  523. "<text label>." << std::endl);
  524. return false;
  525. }
  526. }
  527. std::vector<std::string> cpackPackageDesktopLinksList;
  528. const char *cpackPackageDesktopLinks =
  529. GetOption("CPACK_CREATE_DESKTOP_LINKS");
  530. if(cpackPackageDesktopLinks)
  531. {
  532. cmSystemTools::ExpandListArgument(cpackPackageDesktopLinks,
  533. cpackPackageDesktopLinksList);
  534. }
  535. AddDirectoryAndFileDefinitons(
  536. rootPath, "INSTALL_ROOT",
  537. directoryDefinitions, fileDefinitions, featureDefinitions,
  538. cpackPackageExecutablesList, cpackPackageDesktopLinksList,
  539. shortcutMap);
  540. featureDefinitions.EndElement("FeatureRef");
  541. return true;
  542. }
  543. bool cmCPackWIXGenerator::CreateStartMenuShortcuts(
  544. std::string const& cpackComponentName,
  545. std::string const& featureId,
  546. shortcut_map_t& shortcutMap,
  547. cmWIXFilesSourceWriter& fileDefinitions,
  548. cmWIXFeaturesSourceWriter& featureDefinitions)
  549. {
  550. bool thisHasDesktopShortcuts = false;
  551. featureDefinitions.BeginElement("FeatureRef");
  552. featureDefinitions.AddAttribute("Id", featureId);
  553. std::string cpackVendor;
  554. if(!RequireOption("CPACK_PACKAGE_VENDOR", cpackVendor))
  555. {
  556. return false;
  557. }
  558. std::string cpackPackageName;
  559. if(!RequireOption("CPACK_PACKAGE_NAME", cpackPackageName))
  560. {
  561. return false;
  562. }
  563. std::string idSuffix;
  564. if(!cpackComponentName.empty())
  565. {
  566. idSuffix += "_";
  567. idSuffix += cpackComponentName;
  568. }
  569. std::string componentId = "CM_SHORTCUT" + idSuffix;
  570. fileDefinitions.BeginElement("DirectoryRef");
  571. fileDefinitions.AddAttribute("Id", "PROGRAM_MENU_FOLDER");
  572. fileDefinitions.BeginElement("Component");
  573. fileDefinitions.AddAttribute("Id", componentId);
  574. fileDefinitions.AddAttribute("Guid", "*");
  575. for(shortcut_map_t::const_iterator
  576. i = shortcutMap.begin(); i != shortcutMap.end(); ++i)
  577. {
  578. std::string const& id = i->first;
  579. cmWIXShortcut const& shortcut = i->second;
  580. fileDefinitions.EmitShortcut(id, shortcut, false);
  581. if(shortcut.desktop)
  582. {
  583. thisHasDesktopShortcuts = true;
  584. }
  585. }
  586. if(cpackComponentName.empty())
  587. {
  588. fileDefinitions.EmitUninstallShortcut(cpackPackageName);
  589. }
  590. fileDefinitions.EmitRemoveFolder(
  591. "CM_REMOVE_PROGRAM_MENU_FOLDER" + idSuffix);
  592. std::string registryKey =
  593. std::string("Software\\") + cpackVendor + "\\" + cpackPackageName;
  594. fileDefinitions.EmitStartMenuShortcutRegistryValue(
  595. registryKey, cpackComponentName);
  596. fileDefinitions.EndElement("Component");
  597. fileDefinitions.EndElement("DirectoryRef");
  598. featureDefinitions.EmitComponentRef(componentId);
  599. if(thisHasDesktopShortcuts)
  600. {
  601. this->HasDesktopShortcuts = true;
  602. componentId = "CM_DESKTOP_SHORTCUT" + idSuffix;
  603. fileDefinitions.BeginElement("DirectoryRef");
  604. fileDefinitions.AddAttribute("Id", "DesktopFolder");
  605. fileDefinitions.BeginElement("Component");
  606. fileDefinitions.AddAttribute("Id", componentId);
  607. fileDefinitions.AddAttribute("Guid", "*");
  608. for(shortcut_map_t::const_iterator
  609. i = shortcutMap.begin(); i != shortcutMap.end(); ++i)
  610. {
  611. std::string const& id = i->first;
  612. cmWIXShortcut const& shortcut = i->second;
  613. if (!shortcut.desktop)
  614. continue;
  615. fileDefinitions.EmitShortcut(id, shortcut, true);
  616. }
  617. fileDefinitions.EmitDesktopShortcutRegistryValue(
  618. registryKey, cpackComponentName);
  619. fileDefinitions.EndElement("Component");
  620. fileDefinitions.EndElement("DirectoryRef");
  621. featureDefinitions.EmitComponentRef(componentId);
  622. }
  623. featureDefinitions.EndElement("FeatureRef");
  624. return true;
  625. }
  626. bool cmCPackWIXGenerator::CreateLicenseFile()
  627. {
  628. std::string licenseSourceFilename;
  629. if(!RequireOption("CPACK_RESOURCE_FILE_LICENSE", licenseSourceFilename))
  630. {
  631. return false;
  632. }
  633. std::string licenseDestinationFilename;
  634. if(!RequireOption("CPACK_WIX_LICENSE_RTF", licenseDestinationFilename))
  635. {
  636. return false;
  637. }
  638. std::string extension = GetRightmostExtension(licenseSourceFilename);
  639. if(extension == ".rtf")
  640. {
  641. cmSystemTools::CopyAFile(
  642. licenseSourceFilename.c_str(),
  643. licenseDestinationFilename.c_str());
  644. }
  645. else if(extension == ".txt")
  646. {
  647. cmWIXRichTextFormatWriter rtfWriter(licenseDestinationFilename);
  648. cmsys::ifstream licenseSource(licenseSourceFilename.c_str());
  649. std::string line;
  650. while(std::getline(licenseSource, line))
  651. {
  652. rtfWriter.AddText(line);
  653. rtfWriter.AddText("\n");
  654. }
  655. }
  656. else
  657. {
  658. cmCPackLogger(cmCPackLog::LOG_ERROR,
  659. "unsupported WiX License file extension '" <<
  660. extension << "'" << std::endl);
  661. return false;
  662. }
  663. return true;
  664. }
  665. void cmCPackWIXGenerator::AddDirectoryAndFileDefinitons(
  666. std::string const& topdir,
  667. std::string const& directoryId,
  668. cmWIXDirectoriesSourceWriter& directoryDefinitions,
  669. cmWIXFilesSourceWriter& fileDefinitions,
  670. cmWIXFeaturesSourceWriter& featureDefinitions,
  671. const std::vector<std::string>& packageExecutables,
  672. const std::vector<std::string>& desktopExecutables,
  673. shortcut_map_t& shortcutMap)
  674. {
  675. cmsys::Directory dir;
  676. dir.Load(topdir.c_str());
  677. std::string relativeDirectoryPath =
  678. cmSystemTools::RelativePath(toplevel.c_str(), topdir.c_str());
  679. if(relativeDirectoryPath.empty())
  680. {
  681. relativeDirectoryPath = ".";
  682. }
  683. cmInstalledFile const* directoryInstalledFile =
  684. this->GetInstalledFile(relativeDirectoryPath);
  685. bool emptyDirectory = dir.GetNumberOfFiles() == 2;
  686. bool createDirectory = false;
  687. if(emptyDirectory)
  688. {
  689. createDirectory = true;
  690. }
  691. if(directoryInstalledFile)
  692. {
  693. if(directoryInstalledFile->HasProperty("CPACK_WIX_ACL"))
  694. {
  695. createDirectory = true;
  696. }
  697. }
  698. if(createDirectory)
  699. {
  700. std::string componentId = fileDefinitions.EmitComponentCreateFolder(
  701. directoryId, GenerateGUID(), directoryInstalledFile);
  702. featureDefinitions.EmitComponentRef(componentId);
  703. }
  704. if(emptyDirectory)
  705. {
  706. return;
  707. }
  708. for(size_t i = 0; i < dir.GetNumberOfFiles(); ++i)
  709. {
  710. std::string fileName = dir.GetFile(static_cast<unsigned long>(i));
  711. if(fileName == "." || fileName == "..")
  712. {
  713. continue;
  714. }
  715. std::string fullPath = topdir + "/" + fileName;
  716. std::string relativePath = cmSystemTools::RelativePath(
  717. toplevel.c_str(), fullPath.c_str());
  718. std::string id = PathToId(relativePath);
  719. if(cmSystemTools::FileIsDirectory(fullPath.c_str()))
  720. {
  721. std::string subDirectoryId = std::string("CM_D") + id;
  722. directoryDefinitions.BeginElement("Directory");
  723. directoryDefinitions.AddAttribute("Id", subDirectoryId);
  724. directoryDefinitions.AddAttribute("Name", fileName);
  725. AddDirectoryAndFileDefinitons(
  726. fullPath, subDirectoryId,
  727. directoryDefinitions,
  728. fileDefinitions,
  729. featureDefinitions,
  730. packageExecutables,
  731. desktopExecutables,
  732. shortcutMap);
  733. this->Patch->ApplyFragment(subDirectoryId, directoryDefinitions);
  734. directoryDefinitions.EndElement("Directory");
  735. }
  736. else
  737. {
  738. cmInstalledFile const* installedFile =
  739. this->GetInstalledFile(relativePath);
  740. std::string componentId = fileDefinitions.EmitComponentFile(
  741. directoryId, id, fullPath, *(this->Patch), installedFile);
  742. featureDefinitions.EmitComponentRef(componentId);
  743. for(size_t j = 0; j < packageExecutables.size(); ++j)
  744. {
  745. std::string const& executableName = packageExecutables[j++];
  746. std::string const& textLabel = packageExecutables[j];
  747. if(cmSystemTools::LowerCase(fileName) ==
  748. cmSystemTools::LowerCase(executableName) + ".exe")
  749. {
  750. cmWIXShortcut &shortcut = shortcutMap[id];
  751. shortcut.textLabel= textLabel;
  752. shortcut.workingDirectoryId = directoryId;
  753. if(desktopExecutables.size() &&
  754. std::find(desktopExecutables.begin(),
  755. desktopExecutables.end(),
  756. executableName)
  757. != desktopExecutables.end())
  758. {
  759. shortcut.desktop = true;
  760. }
  761. }
  762. }
  763. }
  764. }
  765. }
  766. bool cmCPackWIXGenerator::RequireOption(
  767. std::string const& name, std::string &value) const
  768. {
  769. const char* tmp = GetOption(name.c_str());
  770. if(tmp)
  771. {
  772. value = tmp;
  773. return true;
  774. }
  775. else
  776. {
  777. cmCPackLogger(cmCPackLog::LOG_ERROR,
  778. "Required variable " << name << " not set" << std::endl);
  779. return false;
  780. }
  781. }
  782. std::string cmCPackWIXGenerator::GetArchitecture() const
  783. {
  784. std::string void_p_size;
  785. RequireOption("CPACK_WIX_SIZEOF_VOID_P", void_p_size);
  786. if(void_p_size == "8")
  787. {
  788. return "x64";
  789. }
  790. else
  791. {
  792. return "x86";
  793. }
  794. }
  795. std::string cmCPackWIXGenerator::GenerateGUID()
  796. {
  797. UUID guid;
  798. UuidCreate(&guid);
  799. unsigned short *tmp = 0;
  800. UuidToStringW(&guid, &tmp);
  801. std::string result =
  802. cmsys::Encoding::ToNarrow(reinterpret_cast<wchar_t*>(tmp));
  803. RpcStringFreeW(&tmp);
  804. return cmSystemTools::UpperCase(result);
  805. }
  806. std::string cmCPackWIXGenerator::QuotePath(std::string const& path)
  807. {
  808. return std::string("\"") + path + '"';
  809. }
  810. std::string cmCPackWIXGenerator::GetRightmostExtension(
  811. std::string const& filename)
  812. {
  813. std::string extension;
  814. std::string::size_type i = filename.rfind(".");
  815. if(i != std::string::npos)
  816. {
  817. extension = filename.substr(i);
  818. }
  819. return cmSystemTools::LowerCase(extension);
  820. }
  821. std::string cmCPackWIXGenerator::PathToId(std::string const& path)
  822. {
  823. id_map_t::const_iterator i = PathToIdMap.find(path);
  824. if(i != PathToIdMap.end()) return i->second;
  825. std::string id = CreateNewIdForPath(path);
  826. return id;
  827. }
  828. std::string cmCPackWIXGenerator::CreateNewIdForPath(std::string const& path)
  829. {
  830. std::vector<std::string> components;
  831. cmSystemTools::SplitPath(path.c_str(), components, false);
  832. size_t replacementCount = 0;
  833. std::string identifier;
  834. std::string currentComponent;
  835. for(size_t i = 1; i < components.size(); ++i)
  836. {
  837. if(i != 1) identifier += '.';
  838. currentComponent = NormalizeComponentForId(
  839. components[i], replacementCount);
  840. identifier += currentComponent;
  841. }
  842. std::string idPrefix = "P";
  843. size_t replacementPercent = replacementCount * 100 / identifier.size();
  844. if(replacementPercent > 33 || identifier.size() > 60)
  845. {
  846. identifier = CreateHashedId(path, currentComponent);
  847. idPrefix = "H";
  848. }
  849. std::stringstream result;
  850. result << idPrefix << "_" << identifier;
  851. size_t ambiguityCount = ++IdAmbiguityCounter[identifier];
  852. if(ambiguityCount > 999)
  853. {
  854. cmCPackLogger(cmCPackLog::LOG_ERROR,
  855. "Error while trying to generate a unique Id for '" <<
  856. path << "'" << std::endl);
  857. return std::string();
  858. }
  859. else if(ambiguityCount > 1)
  860. {
  861. result << "_" << ambiguityCount;
  862. }
  863. std::string resultString = result.str();
  864. PathToIdMap[path] = resultString;
  865. return resultString;
  866. }
  867. std::string cmCPackWIXGenerator::CreateHashedId(
  868. std::string const& path, std::string const& normalizedFilename)
  869. {
  870. cmsys::auto_ptr<cmCryptoHash> sha1 = cmCryptoHash::New("SHA1");
  871. std::string hash = sha1->HashString(path.c_str());
  872. std::string identifier;
  873. identifier += hash.substr(0, 7) + "_";
  874. const size_t maxFileNameLength = 52;
  875. if(normalizedFilename.length() > maxFileNameLength)
  876. {
  877. identifier += normalizedFilename.substr(0, maxFileNameLength - 3);
  878. identifier += "...";
  879. }
  880. else
  881. {
  882. identifier += normalizedFilename;
  883. }
  884. return identifier;
  885. }
  886. std::string cmCPackWIXGenerator::NormalizeComponentForId(
  887. std::string const& component, size_t& replacementCount)
  888. {
  889. std::string result;
  890. result.resize(component.size());
  891. for(size_t i = 0; i < component.size(); ++i)
  892. {
  893. char c = component[i];
  894. if(IsLegalIdCharacter(c))
  895. {
  896. result[i] = c;
  897. }
  898. else
  899. {
  900. result[i] = '_';
  901. ++ replacementCount;
  902. }
  903. }
  904. return result;
  905. }
  906. bool cmCPackWIXGenerator::IsLegalIdCharacter(char c)
  907. {
  908. return (c >= '0' && c <= '9') ||
  909. (c >= 'a' && c <= 'z') ||
  910. (c >= 'A' && c <= 'Z') ||
  911. c == '_' || c == '.';
  912. }
  913. void cmCPackWIXGenerator::CollectExtensions(
  914. std::string const& variableName, extension_set_t& extensions)
  915. {
  916. const char *variableContent = GetOption(variableName.c_str());
  917. if(!variableContent) return;
  918. std::vector<std::string> list;
  919. cmSystemTools::ExpandListArgument(variableContent, list);
  920. extensions.insert(list.begin(), list.end());
  921. }
  922. void cmCPackWIXGenerator::AddCustomFlags(
  923. std::string const& variableName, std::ostream& stream)
  924. {
  925. const char *variableContent = GetOption(variableName.c_str());
  926. if(!variableContent) return;
  927. std::vector<std::string> list;
  928. cmSystemTools::ExpandListArgument(variableContent, list);
  929. for(std::vector<std::string>::const_iterator i = list.begin();
  930. i != list.end(); ++i)
  931. {
  932. stream << " " << QuotePath(*i);
  933. }
  934. }