Patrick Williams | 721dcbd | 2020-04-08 07:32:57 -0500 | [diff] [blame] | 1 | #!/usr/bin/env python3 |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 2 | |
| 3 | """ |
| 4 | This script applies patches to an XML file. |
| 5 | |
| 6 | The patch file is itself an XML file. It can have any root element name, |
| 7 | and uses XML attributes to specify if the elements in the file should replace |
| 8 | existing elements or add new ones. An XPath attribute is used to specify |
| 9 | where the fix should be applied. A <targetFile> element is required in the |
| 10 | patch file to specify the base name of the XML file the patches should be |
| 11 | applied to, though the targetFile element is handled outside of this script. |
| 12 | |
| 13 | The only restriction is that since the type, xpath, and key attributes are |
| 14 | used to specify the patch placement the target XML cannot use those at a |
| 15 | top level element. |
| 16 | |
| 17 | It can apply patches in 5 ways: |
| 18 | |
| 19 | 1) Add an element: |
| 20 | Put in the element to add, along with the type='add' attribute |
| 21 | and an xpath attribute specifying where the new element should go. |
| 22 | |
| 23 | <enumerationType type='add' xpath="./"> |
| 24 | <id>MY_TYPE</id> |
| 25 | </enumerationType> |
| 26 | |
| 27 | This will add a new enumerationType element child to the root element. |
| 28 | |
| 29 | 2) Replace an element: |
| 30 | Put in the new element, with the type='replace' attribute |
| 31 | and the XPath of the element you want to replace. |
| 32 | |
| 33 | <enumerator type='replace' |
| 34 | xpath="enumerationType/[id='TYPE']/enumerator[name='XBUS']"> |
| 35 | <name>XBUS</name> |
| 36 | <value>the new XBUS value</value> |
| 37 | </enumerator> |
| 38 | |
| 39 | This will replace the enumerator element with name XBUS under the |
| 40 | enumerationType element with ID TYPE. |
| 41 | |
| 42 | 3) Remove an element: |
| 43 | Put in the element to remove, with the type='remove' attribute and |
| 44 | the XPath of the element you want to remove. The full element contents |
| 45 | don't need to be specified, as the XPath is what locates the element. |
| 46 | |
| 47 | <enumerator type='remove' |
| 48 | xpath='enumerationType[id='TYPE]/enumerator[name='DIMM']> |
| 49 | </enumerator> |
| 50 | |
| 51 | This will remove the enumerator element with name DIMM under the |
| 52 | enumerationType element with ID TYPE. |
| 53 | |
| 54 | 4) Add child elements to a specific element. Useful when adding several |
| 55 | child elements at once. |
| 56 | |
| 57 | Use a type attribute of 'add-child' and specify the target parent with |
| 58 | the xpath attribute. |
| 59 | |
| 60 | <enumerationType type="add-child" xpath="enumerationType/[id='TYPE']"> |
| 61 | <enumerator> |
| 62 | <name>MY_NEW_ENUMERATOR</name> |
| 63 | <value>23</value> |
| 64 | </enumerator> |
| 65 | <enumerator> |
| 66 | <name>ANOTHER_NEW_ENUMERATOR</name> |
| 67 | <value>99</value> |
| 68 | </enumerator> |
| 69 | </enumerationType> |
| 70 | |
| 71 | This will add 2 new <enumerator> elements to the enumerationType |
| 72 | element with ID TYPE. |
| 73 | |
| 74 | 5) Replace a child element inside another element, useful when replacing |
| 75 | several child elements of the same parent at once. |
| 76 | |
| 77 | Use a type attribute of 'replace-child' and the xpath attribute |
| 78 | as described above, and also use the key attribute to specify which |
| 79 | element should be used to match on so the replace can be done. |
| 80 | |
| 81 | <enumerationType type="replace-child" |
| 82 | key="name" |
| 83 | xpath="enumerationType/[id='TYPE']"> |
| 84 | <enumerator> |
| 85 | <name>OLD_ENUMERATOR</name> |
| 86 | <value>newvalue</value> |
| 87 | </enumerator> |
| 88 | <enumerator> |
| 89 | <name>ANOTHER_OLD_ENUMERATOR</name> |
| 90 | <value>anothernewvalue</value> |
| 91 | </enumerator> |
| 92 | </enumerationType> |
| 93 | |
| 94 | This will replace the <enumerator> elements with the names of |
| 95 | OLD_ENUMERATOR and ANOTHER_OLD_ENUMERATOR with the <enumerator> |
| 96 | elements specified, inside of the enumerationType element with |
| 97 | ID TYPE. |
| 98 | """ |
| 99 | |
| 100 | |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 101 | import argparse |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 102 | import sys |
| 103 | |
| 104 | from lxml import etree |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 105 | |
| 106 | |
| 107 | def delete_attrs(element, attrs): |
| 108 | for a in attrs: |
| 109 | try: |
| 110 | del element.attrib[a] |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 111 | except Exception: |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 112 | pass |
| 113 | |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 114 | |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 115 | if __name__ == "__main__": |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 116 | parser = argparse.ArgumentParser("Applies fixes to XML files") |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 117 | parser.add_argument("-x", dest="xml", help="The input XML file") |
| 118 | parser.add_argument("-p", dest="patch_xml", help="The patch XML file") |
| 119 | parser.add_argument("-o", dest="output_xml", help="The output XML file") |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 120 | args = parser.parse_args() |
| 121 | |
| 122 | if not all([args.xml, args.patch_xml, args.output_xml]): |
| 123 | parser.print_usage() |
| 124 | sys.exit(-1) |
| 125 | |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 126 | errors = [] |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 127 | patch_num = 0 |
| 128 | patch_tree = etree.parse(args.patch_xml) |
| 129 | patch_root = patch_tree.getroot() |
| 130 | tree = etree.parse(args.xml) |
| 131 | root = tree.getroot() |
| 132 | |
| 133 | for node in patch_root: |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 134 | if ( |
| 135 | (node.tag is etree.PI) |
| 136 | or (node.tag is etree.Comment) |
| 137 | or (node.tag == "targetFile") |
| 138 | ): |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 139 | continue |
| 140 | |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 141 | patch_num = patch_num + 1 |
| 142 | |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 143 | xpath = node.get("xpath", None) |
| 144 | patch_type = node.get("type", "add") |
| 145 | patch_key = node.get("key", None) |
| 146 | delete_attrs(node, ["xpath", "type", "key"]) |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 147 | |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 148 | print("Patch " + str(patch_num) + ":") |
| 149 | |
| 150 | try: |
| 151 | if xpath is None: |
| 152 | raise Exception(" E> No XPath attribute found") |
| 153 | |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 154 | target = tree.find(xpath) |
| 155 | |
| 156 | if target is None: |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 157 | raise Exception(" E> Could not find XPath target " + xpath) |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 158 | |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 159 | if patch_type == "add": |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 160 | print(" Adding element " + target.tag + " to " + xpath) |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 161 | |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 162 | # The ServerWiz API is dependent on ordering for the |
| 163 | # elements at the root node, so make sure they get appended |
| 164 | # at the end. |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 165 | if (xpath == "./") or (xpath == "/"): |
| 166 | root.append(node) |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 167 | else: |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 168 | target.append(node) |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 169 | |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 170 | elif patch_type == "remove": |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 171 | print(" Removing element " + xpath) |
| 172 | parent = target.find("..") |
| 173 | if parent is None: |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 174 | raise Exception( |
| 175 | " E> Could not find parent of " |
| 176 | + xpath |
| 177 | + " so can't remove this element" |
| 178 | ) |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 179 | parent.remove(target) |
| 180 | |
| 181 | elif patch_type == "replace": |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 182 | print(" Replacing element " + xpath) |
| 183 | parent = target.find("..") |
| 184 | if parent is None: |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 185 | raise Exception( |
| 186 | " E> Could not find parent of " |
| 187 | + xpath |
| 188 | + " so can't replace this element" |
| 189 | ) |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 190 | parent.remove(target) |
| 191 | parent.append(node) |
| 192 | |
| 193 | elif patch_type == "add-child": |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 194 | for child in node: |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 195 | print( |
| 196 | " Adding a '" |
| 197 | + child.tag |
| 198 | + "' child element to " |
| 199 | + xpath |
| 200 | ) |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 201 | target.append(child) |
| 202 | |
| 203 | elif patch_type == "replace-child": |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 204 | if patch_key is None: |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 205 | raise Exception( |
| 206 | " E> Patch type is replace-child, but" |
| 207 | " 'key' attribute isn't set" |
| 208 | ) |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 209 | updates = [] |
| 210 | for child in node: |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 211 | # Use the key to figure out which element to replace |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 212 | key_element = child.find(patch_key) |
| 213 | for target_child in target: |
| 214 | for grandchild in target_child: |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 215 | if (grandchild.tag == patch_key) and ( |
| 216 | grandchild.text == key_element.text |
| 217 | ): |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 218 | update = {} |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 219 | update["remove"] = target_child |
| 220 | update["add"] = child |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 221 | updates.append(update) |
| 222 | |
| 223 | for update in updates: |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 224 | print( |
| 225 | " Replacing a '" |
| 226 | + update["remove"].tag |
| 227 | + "' element in path " |
| 228 | + xpath |
| 229 | ) |
| 230 | target.remove(update["remove"]) |
| 231 | target.append(update["add"]) |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 232 | |
| 233 | else: |
Patrick Williams | 66367a1 | 2022-12-05 09:58:37 -0600 | [diff] [blame] | 234 | raise Exception( |
| 235 | " E> Unknown patch type attribute found: " + patch_type |
| 236 | ) |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 237 | |
| 238 | except Exception as e: |
Patrick Williams | 721dcbd | 2020-04-08 07:32:57 -0500 | [diff] [blame] | 239 | print(e) |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 240 | errors.append(e) |
Matt Spinler | ab015d7 | 2016-09-06 15:33:17 -0500 | [diff] [blame] | 241 | |
| 242 | tree.write(args.output_xml) |
Matt Spinler | cb99d1e | 2016-09-13 14:47:25 -0500 | [diff] [blame] | 243 | |
| 244 | if errors: |
| 245 | print("Exiting with " + str(len(errors)) + " total errors") |
| 246 | sys.exit(-1) |