Retaining permission of VNF package files
Python zipfile module changes permission of the zip file contents to 644 incorrectly while extracting zip file. This is because of the bug present in the python zipfile module: https://bugs.python.org/issue15795 This patch will resolve this issue and retain the permission of files. Co-Authored-By: Shanu Pandit <shanu.pandit@india.nec.com> Co-Authored-By: Sheel Rana <ranasheel2000@gmail.com> Closes-Bug: #1923540 Change-Id: I71aad5f879e6df21f679187283957c59b73e0643
This commit is contained in:
parent
a98cd4eaa9
commit
89b3b65c7c
@ -521,7 +521,7 @@ def _validate_hash(algorithm, hash_code, csar, artifact_path):
|
|||||||
|
|
||||||
def extract_csar_zip_file(file_path, extract_path):
|
def extract_csar_zip_file(file_path, extract_path):
|
||||||
try:
|
try:
|
||||||
with zipfile.ZipFile(file_path, 'r') as zf:
|
with PreserveZipFilePermissions(file_path, 'r') as zf:
|
||||||
zf.extractall(extract_path)
|
zf.extractall(extract_path)
|
||||||
except (RuntimeError, zipfile.BadZipfile) as exp:
|
except (RuntimeError, zipfile.BadZipfile) as exp:
|
||||||
with excutils.save_and_reraise_exception():
|
with excutils.save_and_reraise_exception():
|
||||||
@ -573,3 +573,25 @@ def delete_csar_data(package_uuid):
|
|||||||
msg = _('Failed to delete csar folder: '
|
msg = _('Failed to delete csar folder: '
|
||||||
'%(csar_path)s, Error: %(exc)s')
|
'%(csar_path)s, Error: %(exc)s')
|
||||||
LOG.warning(msg, {'csar_path': csar_path, 'exc': exc_message})
|
LOG.warning(msg, {'csar_path': csar_path, 'exc': exc_message})
|
||||||
|
|
||||||
|
|
||||||
|
class PreserveZipFilePermissions(zipfile.ZipFile):
|
||||||
|
"""Patched _extract_member function of zipFile.
|
||||||
|
|
||||||
|
zipfile.ZipFile.extractall function internally calls
|
||||||
|
_extract_member function.
|
||||||
|
Here _extract_member function is patched to retain the
|
||||||
|
file permissions using member.external_attr >> 16.
|
||||||
|
|
||||||
|
Note: First 16 bits of external_attr store permission details.
|
||||||
|
"""
|
||||||
|
def _extract_member(self, member, targetpath, pwd):
|
||||||
|
if not isinstance(member, zipfile.ZipInfo):
|
||||||
|
member = self.getinfo(member)
|
||||||
|
|
||||||
|
targetpath = super()._extract_member(member, targetpath, pwd)
|
||||||
|
|
||||||
|
attr = member.external_attr >> 16
|
||||||
|
if attr != 0:
|
||||||
|
os.chmod(targetpath, attr)
|
||||||
|
return targetpath
|
||||||
|
@ -464,3 +464,24 @@ class TestCSARUtils(testtools.TestCase):
|
|||||||
self.assertEqual(flavours[0]['sw_images'][1]['min_disk'], 2000000000)
|
self.assertEqual(flavours[0]['sw_images'][1]['min_disk'], 2000000000)
|
||||||
self.assertEqual(flavours[0]['sw_images'][1]['size'], 2000000000)
|
self.assertEqual(flavours[0]['sw_images'][1]['size'], 2000000000)
|
||||||
self.assertEqual(flavours[0]['sw_images'][1]['min_ram'], 8590458880)
|
self.assertEqual(flavours[0]['sw_images'][1]['min_ram'], 8590458880)
|
||||||
|
|
||||||
|
def test_extract_csar_zip_file_for_file_permissions(self):
|
||||||
|
dir_location = tempfile.mkdtemp()
|
||||||
|
file_obj = tempfile.NamedTemporaryFile(dir=dir_location)
|
||||||
|
file_path = file_obj.name
|
||||||
|
file_name = file_path.split("/")[3]
|
||||||
|
os.chmod(file_path, 0o755)
|
||||||
|
self.addCleanup(shutil.rmtree, dir_location)
|
||||||
|
initial_cwd = os.getcwd()
|
||||||
|
os.chdir(dir_location)
|
||||||
|
zip_file_path = dir_location + '/' + "test_extract_csar_zip_file.zip"
|
||||||
|
with zipfile.ZipFile(zip_file_path, 'w') as zip:
|
||||||
|
zip.write(file_name)
|
||||||
|
self.assertEqual(True, os.path.exists(zip_file_path))
|
||||||
|
zip_extract_path = dir_location + "/zip_extract"
|
||||||
|
extract_file_path = zip_extract_path + "/" + file_name
|
||||||
|
os.chdir(initial_cwd)
|
||||||
|
csar_utils.extract_csar_zip_file(zip_file_path, zip_extract_path)
|
||||||
|
status = os.stat(extract_file_path)
|
||||||
|
permission = oct(status.st_mode)[-3:]
|
||||||
|
self.assertEqual('755', permission)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user