cmCPackWIXGenerator.cxx 34 KB

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