cmCPackWIXGenerator.cxx 36 KB

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