cmCPackWIXGenerator.cxx 36 KB

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