cmWIXPatchParser.cxx 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  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 "cmWIXPatchParser.h"
  4. #include <utility>
  5. #include <cm/memory>
  6. #include "cm_expat.h"
  7. #include "cmCPackGenerator.h"
  8. cmWIXPatchNode::Type cmWIXPatchText::type()
  9. {
  10. return cmWIXPatchNode::TEXT;
  11. }
  12. cmWIXPatchNode::Type cmWIXPatchElement::type()
  13. {
  14. return cmWIXPatchNode::ELEMENT;
  15. }
  16. cmWIXPatchNode::~cmWIXPatchNode()
  17. {
  18. }
  19. cmWIXPatchElement::cmWIXPatchElement() = default;
  20. cmWIXPatchElement::~cmWIXPatchElement() = default;
  21. cmWIXPatchParser::cmWIXPatchParser(fragment_map_t& fragments,
  22. cmCPackLog* logger)
  23. : Logger(logger)
  24. , State(BEGIN_DOCUMENT)
  25. , Valid(true)
  26. , Fragments(fragments)
  27. {
  28. }
  29. void cmWIXPatchParser::StartElement(const std::string& name, const char** atts)
  30. {
  31. if (State == BEGIN_DOCUMENT) {
  32. if (name == "CPackWiXPatch") {
  33. State = BEGIN_FRAGMENTS;
  34. } else {
  35. ReportValidationError("Expected root element 'CPackWiXPatch'");
  36. }
  37. } else if (State == BEGIN_FRAGMENTS) {
  38. if (name == "CPackWiXFragment") {
  39. State = INSIDE_FRAGMENT;
  40. StartFragment(atts);
  41. } else {
  42. ReportValidationError("Expected 'CPackWixFragment' element");
  43. }
  44. } else if (State == INSIDE_FRAGMENT) {
  45. cmWIXPatchElement& parent = *ElementStack.back();
  46. auto element = cm::make_unique<cmWIXPatchElement>();
  47. element->name = name;
  48. for (size_t i = 0; atts[i]; i += 2) {
  49. std::string key = atts[i];
  50. std::string value = atts[i + 1];
  51. element->attributes[key] = value;
  52. }
  53. ElementStack.push_back(element.get());
  54. parent.children.push_back(std::move(element));
  55. }
  56. }
  57. void cmWIXPatchParser::StartFragment(const char** attributes)
  58. {
  59. cmWIXPatchElement* new_element = nullptr;
  60. /* find the id of for fragment */
  61. for (size_t i = 0; attributes[i]; i += 2) {
  62. const std::string key = attributes[i];
  63. const std::string value = attributes[i + 1];
  64. if (key == "Id") {
  65. if (Fragments.find(value) != Fragments.end()) {
  66. std::ostringstream tmp;
  67. tmp << "Invalid reuse of 'CPackWixFragment' 'Id': " << value;
  68. ReportValidationError(tmp.str());
  69. }
  70. new_element = &Fragments[value];
  71. ElementStack.push_back(new_element);
  72. }
  73. }
  74. /* add any additional attributes for the fragment */
  75. if (!new_element) {
  76. ReportValidationError("No 'Id' specified for 'CPackWixFragment' element");
  77. } else {
  78. for (size_t i = 0; attributes[i]; i += 2) {
  79. const std::string key = attributes[i];
  80. const std::string value = attributes[i + 1];
  81. if (key != "Id") {
  82. new_element->attributes[key] = value;
  83. }
  84. }
  85. }
  86. }
  87. void cmWIXPatchParser::EndElement(const std::string& name)
  88. {
  89. if (State == INSIDE_FRAGMENT) {
  90. if (name == "CPackWiXFragment") {
  91. State = BEGIN_FRAGMENTS;
  92. ElementStack.clear();
  93. } else {
  94. ElementStack.pop_back();
  95. }
  96. }
  97. }
  98. void cmWIXPatchParser::CharacterDataHandler(const char* data, int length)
  99. {
  100. const char* whitespace = "\x20\x09\x0d\x0a";
  101. if (State == INSIDE_FRAGMENT) {
  102. cmWIXPatchElement& parent = *ElementStack.back();
  103. std::string text(data, length);
  104. std::string::size_type first = text.find_first_not_of(whitespace);
  105. std::string::size_type last = text.find_last_not_of(whitespace);
  106. if (first != std::string::npos && last != std::string::npos) {
  107. auto text_node = cm::make_unique<cmWIXPatchText>();
  108. text_node->text = text.substr(first, last - first + 1);
  109. parent.children.push_back(std::move(text_node));
  110. }
  111. }
  112. }
  113. void cmWIXPatchParser::ReportError(int line, int column, const char* msg)
  114. {
  115. cmCPackLogger(cmCPackLog::LOG_ERROR,
  116. "Error while processing XML patch file at "
  117. << line << ":" << column << ": " << msg << std::endl);
  118. Valid = false;
  119. }
  120. void cmWIXPatchParser::ReportValidationError(std::string const& message)
  121. {
  122. ReportError(
  123. XML_GetCurrentLineNumber(static_cast<XML_Parser>(this->Parser)),
  124. XML_GetCurrentColumnNumber(static_cast<XML_Parser>(this->Parser)),
  125. message.c_str());
  126. }
  127. bool cmWIXPatchParser::IsValid() const
  128. {
  129. return Valid;
  130. }