diff --git a/changelogs/fragments/5393-archive.yml b/changelogs/fragments/5393-archive.yml new file mode 100644 index 00000000000..a589c935ec0 --- /dev/null +++ b/changelogs/fragments/5393-archive.yml @@ -0,0 +1,2 @@ +bugfixes: + - "archive - avoid crash when ``lzma`` is not present and ``format`` is not ``xz`` (https://github.com/ansible-collections/community.general/pull/5393)." diff --git a/plugins/modules/files/archive.py b/plugins/modules/files/archive.py index 9d82017b3cd..36e00487c75 100644 --- a/plugins/modules/files/archive.py +++ b/plugins/modules/files/archive.py @@ -575,6 +575,11 @@ def py26_filter(path): self.file.add(path, archive_name, recursive=False, exclude=py26_filter) def _get_checksums(self, path): + if HAS_LZMA: + LZMAError = lzma.LZMAError + else: + # Just picking another exception that's also listed below + LZMAError = tarfile.ReadError try: if self.format == 'xz': with lzma.open(_to_native_ascii(path), 'r') as f: @@ -585,7 +590,7 @@ def _get_checksums(self, path): archive = tarfile.open(_to_native_ascii(path), 'r|' + self.format) checksums = set((info.name, info.chksum) for info in archive.getmembers()) archive.close() - except (lzma.LZMAError, tarfile.ReadError, tarfile.CompressionError): + except (LZMAError, tarfile.ReadError, tarfile.CompressionError): try: # The python implementations of gzip, bz2, and lzma do not support restoring compressed files # to their original names so only file checksum is returned diff --git a/tests/integration/targets/archive/tasks/main.yml b/tests/integration/targets/archive/tasks/main.yml index 29742ccf9c9..4043f95b628 100644 --- a/tests/integration/targets/archive/tasks/main.yml +++ b/tests/integration/targets/archive/tasks/main.yml @@ -12,6 +12,37 @@ # Make sure we start fresh # Test setup +- name: prep our files + copy: src={{ item }} dest={{remote_tmp_dir}}/{{ item }} + with_items: + - foo.txt + - bar.txt + - empty.txt + - sub + - sub/subfile.txt + +# Run twice without lzma backport installed, to make sure it does not crash +- name: Archive - pre-test - first run + archive: + path: "{{ remote_tmp_dir }}/*.txt" + dest: "{{ remote_tmp_dir }}/archive_pretest_1.tar" + format: "tar" + register: pretest_1 + +- name: Archive - pre-test - second run + archive: + path: "{{ remote_tmp_dir }}/*.txt" + dest: "{{ remote_tmp_dir }}/archive_pretest_1.tar" + format: "tar" + register: pretest_2 + +- name: Archive - validate pre-test + assert: + that: + - pretest_1 is changed + - pretest_2 is not changed + +# Install dependencies - name: Ensure zip is present to create test archive (yum) yum: name=zip state=latest when: ansible_facts.pkg_mgr == 'yum' @@ -63,15 +94,6 @@ when: ansible_python_version.split('.')[0] == '2' register: backports_lzma_pip -- name: prep our files - copy: src={{ item }} dest={{remote_tmp_dir}}/{{ item }} - with_items: - - foo.txt - - bar.txt - - empty.txt - - sub - - sub/subfile.txt - - name: Define formats to test set_fact: formats: