From 0243eabd3007ec0bbc127fa8b0cc9bc0dddffc02 Mon Sep 17 00:00:00 2001
From: mklassen <lmklassen@gmail.com>
Date: Wed, 30 Sep 2020 07:11:28 -0400
Subject: [PATCH] Fix xml reports changed when node is not deleted (#1007)

* Fix xml reports changed when node is not deleted

* Added changelog fragment

* Added tests for xml no change remove

* Added PR to changeling fragment

Co-authored-by: Felix Fontein <felix@fontein.de>

Co-authored-by: Felix Fontein <felix@fontein.de>
---
 changelogs/fragments/xml-remove-changed.yml   |  2 ++
 plugins/modules/files/xml.py                  |  4 ++-
 tests/integration/targets/xml/tasks/main.yml  |  3 ++
 .../tasks/test-remove-attribute-nochange.yml  | 28 ++++++++++++++++
 .../tasks/test-remove-element-nochange.yml    | 28 ++++++++++++++++
 ...t-remove-namespaced-attribute-nochange.yml | 33 +++++++++++++++++++
 .../test-remove-namespaced-attribute.yml      |  2 +-
 ...est-remove-namespaced-element-nochange.yml | 33 +++++++++++++++++++
 8 files changed, 131 insertions(+), 2 deletions(-)
 create mode 100644 changelogs/fragments/xml-remove-changed.yml
 create mode 100644 tests/integration/targets/xml/tasks/test-remove-attribute-nochange.yml
 create mode 100644 tests/integration/targets/xml/tasks/test-remove-element-nochange.yml
 create mode 100644 tests/integration/targets/xml/tasks/test-remove-namespaced-attribute-nochange.yml
 create mode 100644 tests/integration/targets/xml/tasks/test-remove-namespaced-element-nochange.yml

diff --git a/changelogs/fragments/xml-remove-changed.yml b/changelogs/fragments/xml-remove-changed.yml
new file mode 100644
index 0000000000..f1fc50b32c
--- /dev/null
+++ b/changelogs/fragments/xml-remove-changed.yml
@@ -0,0 +1,2 @@
+minor_changes:
+  - xml - fixed issue were changed was returned when removing non-existent xpath (https://github.com/ansible-collections/community.general/pull/1007).
diff --git a/plugins/modules/files/xml.py b/plugins/modules/files/xml.py
index 7bf99e29ba..53b84b127b 100644
--- a/plugins/modules/files/xml.py
+++ b/plugins/modules/files/xml.py
@@ -411,8 +411,10 @@ def xpath_matches(tree, xpath, namespaces):
 
 def delete_xpath_target(module, tree, xpath, namespaces):
     """ Delete an attribute or element from a tree """
+    changed = False
     try:
         for result in tree.xpath(xpath, namespaces=namespaces):
+            changed = True
             # Get the xpath for this result
             if is_attribute(tree, xpath, namespaces):
                 # Delete an attribute
@@ -429,7 +431,7 @@ def delete_xpath_target(module, tree, xpath, namespaces):
     except Exception as e:
         module.fail_json(msg="Couldn't delete xpath target: %s (%s)" % (xpath, e))
     else:
-        finish(module, tree, xpath, namespaces, changed=True)
+        finish(module, tree, xpath, namespaces, changed=changed)
 
 
 def replace_children_of(children, match):
diff --git a/tests/integration/targets/xml/tasks/main.yml b/tests/integration/targets/xml/tasks/main.yml
index 48e72ebe0f..a5c6e87ac2 100644
--- a/tests/integration/targets/xml/tasks/main.yml
+++ b/tests/integration/targets/xml/tasks/main.yml
@@ -43,7 +43,9 @@
   - include_tasks: test-count.yml
   - include_tasks: test-mutually-exclusive-attributes.yml
   - include_tasks: test-remove-attribute.yml
+  - include_tasks: test-remove-attribute-nochange.yml
   - include_tasks: test-remove-element.yml
+  - include_tasks: test-remove-element-nochange.yml
   - include_tasks: test-set-attribute-value.yml
   - include_tasks: test-set-children-elements.yml
   - include_tasks: test-set-children-elements-level.yml
@@ -53,6 +55,7 @@
   - include_tasks: test-pretty-print-only.yml
   - include_tasks: test-add-namespaced-children-elements.yml
   - include_tasks: test-remove-namespaced-attribute.yml
+  - include_tasks: test-remove-namespaced-attribute-nochange.yml
   - include_tasks: test-set-namespaced-attribute-value.yml
   - include_tasks: test-set-namespaced-element-value.yml
   - include_tasks: test-set-namespaced-children-elements.yml
diff --git a/tests/integration/targets/xml/tasks/test-remove-attribute-nochange.yml b/tests/integration/targets/xml/tasks/test-remove-attribute-nochange.yml
new file mode 100644
index 0000000000..d09dee405c
--- /dev/null
+++ b/tests/integration/targets/xml/tasks/test-remove-attribute-nochange.yml
@@ -0,0 +1,28 @@
+---
+  - name: Setup test fixture
+    copy:
+      src: results/test-remove-attribute.xml
+      dest: /tmp/ansible-xml-beers.xml
+
+
+  - name: Remove non-existing '/business/rating/@subjective'
+    xml:
+      path: /tmp/ansible-xml-beers.xml
+      xpath: /business/rating/@subjective
+      state: absent
+    register: remove_attribute
+
+  - name: Compare to expected result
+    copy:
+      src: results/test-remove-attribute.xml
+      dest: /tmp/ansible-xml-beers.xml
+    check_mode: yes
+    diff: yes
+    register: comparison
+
+  - name: Test expected result
+    assert:
+      that:
+      - remove_attribute.changed == false
+      - comparison.changed == false  # identical
+    #command: diff -u {{ role_path }}/results/test-remove-attribute.xml /tmp/ansible-xml-beers.xml
diff --git a/tests/integration/targets/xml/tasks/test-remove-element-nochange.yml b/tests/integration/targets/xml/tasks/test-remove-element-nochange.yml
new file mode 100644
index 0000000000..2debc80d51
--- /dev/null
+++ b/tests/integration/targets/xml/tasks/test-remove-element-nochange.yml
@@ -0,0 +1,28 @@
+---
+  - name: Setup test fixture
+    copy:
+      src: results/test-remove-element.xml
+      dest: /tmp/ansible-xml-beers.xml
+
+
+  - name: Remove non-existing '/business/rating'
+    xml:
+      path: /tmp/ansible-xml-beers.xml
+      xpath: /business/rating
+      state: absent
+    register: remove_element
+
+  - name: Compare to expected result
+    copy:
+      src: results/test-remove-element.xml
+      dest: /tmp/ansible-xml-beers.xml
+    check_mode: yes
+    diff: yes
+    register: comparison
+
+  - name: Test expected result
+    assert:
+      that:
+      - remove_element.changed == false
+      - comparison.changed == false  # identical
+    #command: diff -u {{ role_path }}/results/test-remove-element.xml /tmp/ansible-xml-beers.xml
diff --git a/tests/integration/targets/xml/tasks/test-remove-namespaced-attribute-nochange.yml b/tests/integration/targets/xml/tasks/test-remove-namespaced-attribute-nochange.yml
new file mode 100644
index 0000000000..291536d3bf
--- /dev/null
+++ b/tests/integration/targets/xml/tasks/test-remove-namespaced-attribute-nochange.yml
@@ -0,0 +1,33 @@
+---
+  - name: Setup test fixture
+    copy:
+      src: results/test-remove-namespaced-attribute.xml
+      dest: /tmp/ansible-xml-namespaced-beers.xml
+
+
+  - name: Remove non-existing namespaced '/bus:business/rat:rating/@attr:subjective'
+    xml:
+      path: /tmp/ansible-xml-namespaced-beers.xml
+      xpath: /bus:business/rat:rating/@attr:subjective
+      namespaces:
+        bus: http://test.business
+        ber: http://test.beers
+        rat: http://test.rating
+        attr: http://test.attribute
+      state: absent
+    register: remove_namespaced_attribute
+
+  - name: Compare to expected result
+    copy:
+      src: results/test-remove-namespaced-attribute.xml
+      dest: /tmp/ansible-xml-namespaced-beers.xml
+    check_mode: yes
+    diff: yes
+    register: comparison
+
+  - name: Test expected result
+    assert:
+      that:
+      - remove_namespaced_attribute.changed == false
+      - comparison.changed == false  # identical
+    #command: diff -u {{ role_path }}/results/test-remove-namespaced-attribute.xml /tmp/ansible-xml-namespaced-beers.xml
diff --git a/tests/integration/targets/xml/tasks/test-remove-namespaced-attribute.yml b/tests/integration/targets/xml/tasks/test-remove-namespaced-attribute.yml
index 36682b2202..a7ccdac4e3 100644
--- a/tests/integration/targets/xml/tasks/test-remove-namespaced-attribute.yml
+++ b/tests/integration/targets/xml/tasks/test-remove-namespaced-attribute.yml
@@ -28,6 +28,6 @@
   - name: Test expected result
     assert:
       that:
-      - remove_element.changed == true
+      - remove_namespaced_attribute.changed == true
       - comparison.changed == false  # identical
     #command: diff -u {{ role_path }}/results/test-remove-namespaced-attribute.xml /tmp/ansible-xml-namespaced-beers.xml
diff --git a/tests/integration/targets/xml/tasks/test-remove-namespaced-element-nochange.yml b/tests/integration/targets/xml/tasks/test-remove-namespaced-element-nochange.yml
new file mode 100644
index 0000000000..b1938e45b7
--- /dev/null
+++ b/tests/integration/targets/xml/tasks/test-remove-namespaced-element-nochange.yml
@@ -0,0 +1,33 @@
+---
+  - name: Setup test fixture
+    copy:
+      src: results/test-remove-element.xml
+      dest: /tmp/ansible-xml-namespaced-beers.xml
+
+
+  - name: Remove non-existing namespaced '/bus:business/rat:rating'
+    xml:
+      path: /tmp/ansible-xml-namespaced-beers.xml
+      xpath: /bus:business/rat:rating
+      namespaces:
+        bus: http://test.business
+        ber: http://test.beers
+        rat: http://test.rating
+        attr: http://test.attribute
+      state: absent
+    register: remove_namespaced_element
+
+  - name: Compare to expected result
+    copy:
+      src: results/test-remove-element.xml
+      dest: /tmp/ansible-xml-namespaced-beers.xml
+    check_mode: yes
+    diff: yes
+    register: comparison
+
+  - name: Test expected result
+    assert:
+      that:
+      - remove_namespaced_element.changed == false
+      - comparison.changed == false  # identical
+    #command: diff -u {{ role_path }}/results/test-remove-element.xml /tmp/ansible-xml-namespaced-beers.xml