cmCPackWIXGenerator.cxx 36 KB

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