From dff5fc12267856a18215d20e5bde6514c6332923 Mon Sep 17 00:00:00 2001 From: Andrew Date: Sun, 10 Jul 2022 16:20:25 +1000 Subject: [PATCH 1/6] fix diffparam_volume and add diffparam_mount and diffparam_tmpfs This is a fix for bug #355. This compares the arguments podman recieved for the currently existing container with the (effective) arguments to come. This approach was taken over parsing Mounts from inspect because: 1. This line from https://github.com/containers/podman/blob/e084f0ee1e1ded564110e84eefca9f18b5669adf/libpod/container_inspect.go#L224 regarding inspect's Mount value: "Only includes user-specified mounts. Only includes bind mounts and named volumes, not tmpfs volumes." Thus an incomplete solution would result. 2. The code required to parse so that it could be compared with the stuff to come was getting silly-level complex. 3. Anonymous volumes were impossible to decipher as both Name and Source are filled in with the podman-generated values. Thus we compare the arguments podman create was run with to make the existing container with the closest values to those arguments in the new config. This resulted in simpler code that takes care of the issue of anonymous volumes. The downside is that if someone moves, say, a tmpfs from mount to tmpfs (or vice versa) that would reult in exactly the same result this will be considered a different config. This can (possibly) be fixed if and when it becomes an actual issue. Signed-off-by: Andrew --- .../podman/podman_container_lib.py | 66 ++++++++++--------- 1 file changed, 36 insertions(+), 30 deletions(-) diff --git a/plugins/module_utils/podman/podman_container_lib.py b/plugins/module_utils/podman/podman_container_lib.py index 6186e857..52abd46d 100644 --- a/plugins/module_utils/podman/podman_container_lib.py +++ b/plugins/module_utils/podman/podman_container_lib.py @@ -1048,6 +1048,19 @@ def diffparam_mac_address(self): after = before return self._diff_update_and_compare('mac_address', before, after) + def diffparam_mount(self): + before = [] + if self.info['config'].get('createcommand'): + cr_com = self.info['config']['createcommand'] + for i, v in enumerate(cr_com): + if v == '--mount': + before.append(cr_com[i + 1]) + after = self.params.get('mount') + if not after: + after = [] + before, after = sorted(list(set(before))), sorted(list(set(after))) + return self._diff_update_and_compare('mount', before, after) + def diffparam_network(self): net_mode_before = self.info['hostconfig']['networkmode'] net_mode_after = '' @@ -1160,6 +1173,21 @@ def diffparam_timezone(self): after = self.params['timezone'] return self._diff_update_and_compare('timezone', before, after) + def diffparam_tmpfs(self): + before = [] + if self.info['config'].get('createcommand'): + cr_com = self.info['config']['createcommand'] + for i, v in enumerate(cr_com): + if v == '--tmpfs': + before.append(cr_com[i + 1]) + after = [] + tmpfs = self.params.get('tmpfs') + if tmpfs: + for k, v in tmpfs.items(): + after.append('{}:{}'.format(k, v)) + before, after = sorted(list(set(before))), sorted(list(set(after))) + return self._diff_update_and_compare('tmpfs', before, after) + def diffparam_tty(self): before = self.info['config']['tty'] after = self.params['tty'] @@ -1205,37 +1233,15 @@ def diffparam_uts(self): return self._diff_update_and_compare('uts', before, after) def diffparam_volume(self): - def clean_volume(x): - '''Remove trailing and double slashes from volumes.''' - if not x.rstrip("/"): - return "/" - return x.replace("//", "/").rstrip("/") - - before = self.info['mounts'] - before_local_vols = [] - if before: - volumes = [] - local_vols = [] - for m in before: - if m['type'] != 'volume': - volumes.append( - [ - clean_volume(m['source']), - clean_volume(m['destination']) - ]) - elif m['type'] == 'volume': - local_vols.append( - [m['name'], clean_volume(m['destination'])]) - before = [":".join(v) for v in volumes] - before_local_vols = [":".join(v) for v in local_vols] - if self.params['volume'] is not None: - after = [":".join( - [clean_volume(i) for i in v.split(":")[:2]] - ) for v in self.params['volume']] - else: + before = [] + if self.info['config'].get('createcommand'): + cr_com = self.info['config']['createcommand'] + for i, v in enumerate(cr_com): + if v == '--volume': + before.append(cr_com[i + 1]) + after = self.params.get('volume') + if not after: after = [] - if before_local_vols: - after = list(set(after).difference(before_local_vols)) before, after = sorted(list(set(before))), sorted(list(set(after))) return self._diff_update_and_compare('volume', before, after) From d36b4a6edd595270d29574a8c24c83a0d58e87ff Mon Sep 17 00:00:00 2001 From: Andrew Date: Sun, 10 Jul 2022 17:12:53 +1000 Subject: [PATCH 2/6] clean paths in diff checks The old code removed unnecessary slashes via the _clean_volume() function. I accidentally got rid of it and have re-introduced it here as self._clean_path(). The rename is because it is used outside of mere check of volumes but the internal code is as before. Signed-off-by: Andrew --- .../podman/podman_container_lib.py | 35 +++++++++++++++++-- 1 file changed, 32 insertions(+), 3 deletions(-) diff --git a/plugins/module_utils/podman/podman_container_lib.py b/plugins/module_utils/podman/podman_container_lib.py index 52abd46d..3bcdbae6 100644 --- a/plugins/module_utils/podman/podman_container_lib.py +++ b/plugins/module_utils/podman/podman_container_lib.py @@ -713,6 +713,28 @@ def _diff_update_and_compare(self, param_name, before, after): return True return False + def _clean_path(self, x): + '''Remove trailing and double slashes from path.''' + if not x.rstrip("/"): + return "/" + return x.replace("//", "/").rstrip("/") + + def _clean_path_in_mount_str(self, mounts): + mfin = [] + for mstr in mounts: + for mitem in mstr.split(','): + nv = mitem.split('=', maxsplit=1) + miname = nv[0] + mival = nv[1] if len(nv) > 1 else None + if miname in ['src', 'source', 'dst', 'dest', 'destination', 'target']: + if mival: + mival = self._clean_path(mival) + if mival is None: + mfin.append(miname) + else: + mfin.append("{0}={1}".format(miname, mival)) + return mfin + def diffparam_annotation(self): before = self.info['config']['annotations'] or {} after = before.copy() @@ -1054,10 +1076,12 @@ def diffparam_mount(self): cr_com = self.info['config']['createcommand'] for i, v in enumerate(cr_com): if v == '--mount': - before.append(cr_com[i + 1]) + before = self._clean_path_in_mount_str(cr_com[i + 1]) after = self.params.get('mount') if not after: after = [] + else: + after = self._clean_path_in_mount_str(after) before, after = sorted(list(set(before))), sorted(list(set(after))) return self._diff_update_and_compare('mount', before, after) @@ -1184,7 +1208,10 @@ def diffparam_tmpfs(self): tmpfs = self.params.get('tmpfs') if tmpfs: for k, v in tmpfs.items(): - after.append('{}:{}'.format(k, v)) + if v: + after.append('{0}:{1}'.format(self._clean_path(k), self._clean_path(v))) + else: + after.append(self._clean_path(k)) before, after = sorted(list(set(before))), sorted(list(set(after))) return self._diff_update_and_compare('tmpfs', before, after) @@ -1238,10 +1265,12 @@ def diffparam_volume(self): cr_com = self.info['config']['createcommand'] for i, v in enumerate(cr_com): if v == '--volume': - before.append(cr_com[i + 1]) + before = self._clean_path_in_mount_str(cr_com[i + 1]) after = self.params.get('volume') if not after: after = [] + else: + after = self._clean_path_in_mount_str(after) before, after = sorted(list(set(before))), sorted(list(set(after))) return self._diff_update_and_compare('volume', before, after) From 7de11c1b293964950d01cfff9f6ce4ebae2ac507 Mon Sep 17 00:00:00 2001 From: Andrew Date: Mon, 11 Jul 2022 11:49:30 +1000 Subject: [PATCH 3/6] correct string in list context errors and sort mount string Began sorting in self._clean_path_in_mount_str() as an early defense against ordering of mount arguments changing. This steels against false comparison errors later. Fixed lots of embarrassing string in list context errors. Oops! :( Signed-off-by: Andrew --- .../podman/podman_container_lib.py | 24 ++++++++++++------- 1 file changed, 15 insertions(+), 9 deletions(-) diff --git a/plugins/module_utils/podman/podman_container_lib.py b/plugins/module_utils/podman/podman_container_lib.py index 3bcdbae6..41e8e0d9 100644 --- a/plugins/module_utils/podman/podman_container_lib.py +++ b/plugins/module_utils/podman/podman_container_lib.py @@ -720,9 +720,13 @@ def _clean_path(self, x): return x.replace("//", "/").rstrip("/") def _clean_path_in_mount_str(self, mounts): + '''Parse a mount string, using _clean_path() on any known path + values. Sort the resulting mount string as a defense against + order changing as the return is likely used for comparisons.''' mfin = [] - for mstr in mounts: - for mitem in mstr.split(','): + for mstr in [mounts] if type(mounts) is str else mounts: + msfin = [] + for mitem in sorted(mstr.split(',')): nv = mitem.split('=', maxsplit=1) miname = nv[0] mival = nv[1] if len(nv) > 1 else None @@ -730,10 +734,11 @@ def _clean_path_in_mount_str(self, mounts): if mival: mival = self._clean_path(mival) if mival is None: - mfin.append(miname) + msfin.append(miname) else: - mfin.append("{0}={1}".format(miname, mival)) - return mfin + msfin.append("{0}={1}".format(miname, mival)) + mfin.append(','.join(msfin)) + return mfin[0] if type(mounts) is str else mfin def diffparam_annotation(self): before = self.info['config']['annotations'] or {} @@ -1076,7 +1081,7 @@ def diffparam_mount(self): cr_com = self.info['config']['createcommand'] for i, v in enumerate(cr_com): if v == '--mount': - before = self._clean_path_in_mount_str(cr_com[i + 1]) + before.append(self._clean_path_in_mount_str(cr_com[i + 1])) after = self.params.get('mount') if not after: after = [] @@ -1203,13 +1208,14 @@ def diffparam_tmpfs(self): cr_com = self.info['config']['createcommand'] for i, v in enumerate(cr_com): if v == '--tmpfs': - before.append(cr_com[i + 1]) + tp, tm = cr_com[i + 1].split(':') + before.append('{0}:{1}'.format(self._clean_path(tp), self._clean_path_in_mount_str(tm))) after = [] tmpfs = self.params.get('tmpfs') if tmpfs: for k, v in tmpfs.items(): if v: - after.append('{0}:{1}'.format(self._clean_path(k), self._clean_path(v))) + after.append('{0}:{1}'.format(self._clean_path(k), self._clean_path_in_mount_str(v))) else: after.append(self._clean_path(k)) before, after = sorted(list(set(before))), sorted(list(set(after))) @@ -1265,7 +1271,7 @@ def diffparam_volume(self): cr_com = self.info['config']['createcommand'] for i, v in enumerate(cr_com): if v == '--volume': - before = self._clean_path_in_mount_str(cr_com[i + 1]) + before.append(self._clean_path_in_mount_str(cr_com[i + 1])) after = self.params.get('volume') if not after: after = [] From 384f258f3ead3d0adf682cffc106c204db1bd631 Mon Sep 17 00:00:00 2001 From: Andrew Date: Tue, 12 Jul 2022 10:06:16 +1000 Subject: [PATCH 4/6] fix slash squashing in diffparam_volume and _clean_path Made _clean_path() squash multiple consecutive slashes into one as 3+ slashes are just as meaningful as 1. Slash handling is now done properly diffparam_volume(). Signed-off-by: Andrew --- .../podman/podman_container_lib.py | 35 +++++++++++++++---- 1 file changed, 29 insertions(+), 6 deletions(-) diff --git a/plugins/module_utils/podman/podman_container_lib.py b/plugins/module_utils/podman/podman_container_lib.py index 41e8e0d9..f828860d 100644 --- a/plugins/module_utils/podman/podman_container_lib.py +++ b/plugins/module_utils/podman/podman_container_lib.py @@ -714,10 +714,16 @@ def _diff_update_and_compare(self, param_name, before, after): return False def _clean_path(self, x): - '''Remove trailing and double slashes from path.''' + '''Remove trailing and multiple consecutive slashes from path.''' if not x.rstrip("/"): return "/" - return x.replace("//", "/").rstrip("/") + b = x.rstrip("/") + while True: + a = b.replace("//", "/") + if a == b: + break + b = a + return a def _clean_path_in_mount_str(self, mounts): '''Parse a mount string, using _clean_path() on any known path @@ -1266,17 +1272,34 @@ def diffparam_uts(self): return self._diff_update_and_compare('uts', before, after) def diffparam_volume(self): + def clean_volume_def(vdef): + # va = [:|:][:] + va = vdef.split(':', maxsplit=2) + if len(va) == 2: + if len(va[1]) > 0 and va[1][0] == '/': + # va = [:|:] + return '{0}:{1}'.format(self._clean_path(va[0]), self._clean_path(va[1])) + # va = [:] + return '{0}:{1}'.format(self._clean_path(va[0]), self._clean_path_in_mount_str(va[1])) + elif len(va) == 3: + # va = [:|:][:] + return '{0}:{1}:{2}'.format(self._clean_path(va[0]), self._clean_path(va[1]), self._clean_path_in_mount_str(va[2])) + # va = + return '{0}'.format(self._clean_path(va[0])) + before = [] if self.info['config'].get('createcommand'): cr_com = self.info['config']['createcommand'] for i, v in enumerate(cr_com): if v == '--volume': - before.append(self._clean_path_in_mount_str(cr_com[i + 1])) - after = self.params.get('volume') - if not after: + before.append(clean_volume_def(cr_com[i + 1])) + after = [] + params = self.params.get('volume') + if not params: after = [] else: - after = self._clean_path_in_mount_str(after) + for p in params: + after.append(clean_volume_def(p)) before, after = sorted(list(set(before))), sorted(list(set(after))) return self._diff_update_and_compare('volume', before, after) From ca151f50727d10677c27e034c44488a7d300f794 Mon Sep 17 00:00:00 2001 From: Andrew Date: Wed, 12 Oct 2022 15:39:24 +1300 Subject: [PATCH 5/6] check --mount, --tmpfs, --volume and VOLUME for changes Combine diffparam_mount(), diffparam_tmpfs() and diffparam_volume() and add handling of VOLUME into one function that allows for movement of definitions between the various ways of specifying mount points as long as the options are equivalent. This necessitated translations of the various ways of defining mount points into one single "language" so that the definitions may be compared and checked for similarity. prep_mount_args_for_comp() does this work and includes within it the default options for the various CLI arguments and VOLUME. The consequence of this is that the diff returned by the function no longer literally represents the real-life arguments passed but, rather, the translations of those arguments. Signed-off-by: Andrew --- .../podman/podman_container_lib.py | 300 ++++++++++++------ 1 file changed, 208 insertions(+), 92 deletions(-) diff --git a/plugins/module_utils/podman/podman_container_lib.py b/plugins/module_utils/podman/podman_container_lib.py index 49741947..9fdb6b3b 100644 --- a/plugins/module_utils/podman/podman_container_lib.py +++ b/plugins/module_utils/podman/podman_container_lib.py @@ -722,39 +722,6 @@ def _diff_update_and_compare(self, param_name, before, after): return True return False - def _clean_path(self, x): - '''Remove trailing and multiple consecutive slashes from path.''' - if not x.rstrip("/"): - return "/" - b = x.rstrip("/") - while True: - a = b.replace("//", "/") - if a == b: - break - b = a - return a - - def _clean_path_in_mount_str(self, mounts): - '''Parse a mount string, using _clean_path() on any known path - values. Sort the resulting mount string as a defense against - order changing as the return is likely used for comparisons.''' - mfin = [] - for mstr in [mounts] if type(mounts) is str else mounts: - msfin = [] - for mitem in sorted(mstr.split(',')): - nv = mitem.split('=', maxsplit=1) - miname = nv[0] - mival = nv[1] if len(nv) > 1 else None - if miname in ['src', 'source', 'dst', 'dest', 'destination', 'target']: - if mival: - mival = self._clean_path(mival) - if mival is None: - msfin.append(miname) - else: - msfin.append("{0}={1}".format(miname, mival)) - mfin.append(','.join(msfin)) - return mfin[0] if type(mounts) is str else mfin - def diffparam_annotation(self): before = self.info['config']['annotations'] or {} after = before.copy() @@ -1100,21 +1067,6 @@ def diffparam_mac_address(self): after = before return self._diff_update_and_compare('mac_address', before, after) - def diffparam_mount(self): - before = [] - if self.info['config'].get('createcommand'): - cr_com = self.info['config']['createcommand'] - for i, v in enumerate(cr_com): - if v == '--mount': - before.append(self._clean_path_in_mount_str(cr_com[i + 1])) - after = self.params.get('mount') - if not after: - after = [] - else: - after = self._clean_path_in_mount_str(after) - before, after = sorted(list(set(before))), sorted(list(set(after))) - return self._diff_update_and_compare('mount', before, after) - def diffparam_network(self): net_mode_before = self.info['hostconfig']['networkmode'] net_mode_after = '' @@ -1227,25 +1179,6 @@ def diffparam_timezone(self): after = self.params['timezone'] return self._diff_update_and_compare('timezone', before, after) - def diffparam_tmpfs(self): - before = [] - if self.info['config'].get('createcommand'): - cr_com = self.info['config']['createcommand'] - for i, v in enumerate(cr_com): - if v == '--tmpfs': - tp, tm = cr_com[i + 1].split(':') - before.append('{0}:{1}'.format(self._clean_path(tp), self._clean_path_in_mount_str(tm))) - after = [] - tmpfs = self.params.get('tmpfs') - if tmpfs: - for k, v in tmpfs.items(): - if v: - after.append('{0}:{1}'.format(self._clean_path(k), self._clean_path_in_mount_str(v))) - else: - after.append(self._clean_path(k)) - before, after = sorted(list(set(before))), sorted(list(set(after))) - return self._diff_update_and_compare('tmpfs', before, after) - def diffparam_tty(self): before = self.info['config']['tty'] after = self.params['tty'] @@ -1290,37 +1223,220 @@ def diffparam_uts(self): after = before return self._diff_update_and_compare('uts', before, after) - def diffparam_volume(self): - def clean_volume_def(vdef): - # va = [:|:][:] - va = vdef.split(':', maxsplit=2) - if len(va) == 2: - if len(va[1]) > 0 and va[1][0] == '/': - # va = [:|:] - return '{0}:{1}'.format(self._clean_path(va[0]), self._clean_path(va[1])) - # va = [:] - return '{0}:{1}'.format(self._clean_path(va[0]), self._clean_path_in_mount_str(va[1])) - elif len(va) == 3: - # va = [:|:][:] - return '{0}:{1}:{2}'.format(self._clean_path(va[0]), self._clean_path(va[1]), self._clean_path_in_mount_str(va[2])) - # va = - return '{0}'.format(self._clean_path(va[0])) + def diffparam_volumes_all(self): + '''check for difference between old and new uses of --mount, --tmpfs, --volume and + Containerfile's VOLUME.''' + + def clean_path(x): + '''Remove trailing and multiple consecutive slashes from path. Collapse '/./' to '/'.''' + if not x.rstrip("/"): + return "/" + b = x.rstrip("/") + while True: + a = b.replace("//", "/") + a = a.replace("/./", "/") + if a == b: + break + b = a + return a + + def prep_mount_args_for_comp(mounts, mtype=None): + '''Parse a mount string, using clean_path() on any known path + values. Sort the resulting mount string as a defense against + order changing as the return is likely used for comparisons.''' + + # check the code where defaults below are used if changing. unfortunately things + # are not cleanly separated and their handling requires some encoding. :( + # the 'fake' mount options are there to allow for matching with tmpfs/volume/etc + # options whilst keeping them separate from real options. + # all args are assumed to be item[=val] format and comma separated in-code. + default_mount_args = { + 'volume': 'readonly=false,chown=false,idmap=false', + 'bind': 'readonly=false,chown=false,idmap=false,bind-propagation=rprivate', + 'tmpfs': 'readonly=false,chown=false,tmpfs-mode=1777', + 'devpts': 'uid=0,gid=0,mode=600,max=1048576', + 'image': 'readwrite=false', + } + noargs_tmpfs_args = 'rw,noexec,nosuid,nodev' + default_tmpfs_args = 'rw,suid,dev,exec,size=50%' + default_volume_args = 'rw,rprivate,nosuid,nodev' # manpage says "private" but podman inspect says "rprivate" + + mfin = [] + wants_str = isinstance(mounts, str) + for mstr in [mounts] if wants_str else mounts: + mstype = mtype + msparsed = { + 'type': mtype, + } + for mitem in sorted(mstr.split(',')): + nv = mitem.split('=', maxsplit=1) + miname = nv[0] + mival = nv[1] if len(nv) > 1 else None + if miname == 'type': + mstype = mival + msparsed['type'] = mstype + break + if mstype == 'tmpfs': + if not mstr: + mstr = noargs_tmpfs_args + mstr = ','.join([default_mount_args[mstype], default_tmpfs_args, mstr]).strip(',') + elif mstype == 'volume': + mstr = ','.join([default_mount_args[mstype], default_volume_args, mstr]).strip(',') + else: + mstr = ','.join([default_mount_args[mstype], mstr]).strip(',') + if not mstype: + raise ValueError("mount type not set/found for '{}'.".format(mstr)) + for mitem in mstr.split(','): + nv = mitem.split('=', maxsplit=1) + miname = nv[0] + mival = nv[1] if len(nv) > 1 else None + if miname in ['source', 'src']: + miname = 'src' + if mival: + mival = clean_path(mival) + elif miname in ['destination', 'dest', 'target', 'dst']: + miname = 'dst' + if mival: + mival = clean_path(mival) + elif miname in ['ro', 'readonly']: + miname = 'readonly' + if not mival: + mival = 'true' + elif miname in ['rw', 'readwrite']: + miname = 'readonly' + if not mival: + mival = 'false' + elif mival == 'true': + mival = 'false' + elif mival == 'false': + mival = 'true' + elif miname in ['U', 'chown']: + miname = 'chown' + if not mival: + mival = 'true' + elif miname in ['z']: + miname = 'relabel' + mival = 'shared' + elif miname in ['Z']: + miname = 'relabel' + mival = 'private' + elif miname in ['exec', 'dev', 'suid', 'sync']: + if not mival: + mival = 'true' + elif miname in ['noexec', 'nodev', 'nosuid']: + miname = miname[2:] + mival = 'false' + elif miname in ['shared', 'slave', 'private', 'unbindable', 'rshared', 'rslave', 'runbindable', 'rprivate']: + mival = miname + miname = 'bind-propagation' + elif miname in ['idmap']: + if not mival: + mival = 'true' + elif mstype == 'tmpfs' and miname in ['size', 'mode'] and mival: + miname = 'tmpfs-{}'.format(miname) + elif miname in ['type', 'tmpfs-size', 'tmpfs-mode', 'bind-propagation', 'relabel']: + pass + elif miname.startswith('_unparsed_'): + pass + else: + miname = '_unparsed_{}'.format(miname) + # source can be optional and can be specified as empty. If it is empty + # remove it altogether so that comparisons can be made simply. + if miname == 'src' and not mival: + miname = None + if miname: + msparsed[miname] = mival + msfin = [] + for miname, mival in msparsed.items(): + if mival is None: + msfin.append(miname) + else: + msfin.append("{0}={1}".format(miname, mival)) + mfin.append(','.join(sorted(list(set(msfin))))) + return mfin[0] if wants_str else mfin + + def clean_image_vols(iv_type): + '''cleans a list or string of Containerfile VOLUME config(s).''' + if iv_type == 'bind': + return prep_volume_for_comp(image_volumes) + elif iv_type == 'tmpfs': + return prep_tmpfs_for_comp(image_volumes) + raise ValueError("invalid image volume type: {}.".format(iv_type)) + + def prep_tmpfs_for_comp(all_ms): + res = [] + wants_str = isinstance(all_ms, str) + is_dict = isinstance(all_ms, dict) + for ms in [all_ms] if wants_str else all_ms: + if is_dict: + dm = [ms, all_ms[ms]] + else: + dm = ms.split(':', maxsplit=1) + if len(dm) == 1: + res.append('dst={}'.format(dm[0])) + else: + res.append('dst={},{}'.format(dm[0], dm[1])) + return prep_mount_args_for_comp(res[0] if wants_str else res, mtype='tmpfs') + + def prep_volume_for_comp(all_ms): + res = [] + wants_str = isinstance(all_ms, str) + for ms in [all_ms] if wants_str else all_ms: + dm = ms.split(':', maxsplit=2) + mtype = 'volume' + if len(dm) == 1: + # ms is , mtype is volume + dm.append(dm[0]) # + dm.append('') # + dm[0] = '' # + else: + # ms is :: + if dm[0].startswith('/'): + mtype = 'bind' + if len(dm) == 2: + # ms is : + dm.append('') # + dm[2] = prep_mount_args_for_comp(dm[2], mtype=mtype) + res.append('src={},dst={},{}'.format(dm[0], dm[1], dm[2]).strip(',')) + return prep_mount_args_for_comp(res[0] if wants_str else res, mtype='volume') before = [] + iv_type = 'bind' # documented default + image_volumes = list(map(clean_path, self.image_info['config'].get('volumes', {}).keys())) if self.info['config'].get('createcommand'): cr_com = self.info['config']['createcommand'] for i, v in enumerate(cr_com): - if v == '--volume': - before.append(clean_volume_def(cr_com[i + 1])) - after = [] - params = self.params.get('volume') - if not params: - after = [] - else: - for p in params: - after.append(clean_volume_def(p)) + if v == '--image-volume': + if cr_com[i + 1]: + iv_type = cr_com[i + 1] + elif v == '--mount': + f = prep_mount_args_for_comp(cr_com[i + 1]) + if f: + before.append(f) + elif v == '--tmpfs': + f = prep_tmpfs_for_comp(cr_com[i + 1]) + if f: + before.append(f) + elif v == '--volume': + f = prep_volume_for_comp(cr_com[i + 1]) + if f: + before.append(f) + before.extend(clean_image_vols(iv_type)) + + after = list(self.params.get('mount') or set()) + after_iv_type = self.params.get('image_volume', 'bind') or 'bind' + after.extend(clean_image_vols(after_iv_type)) + + tvols = self.params.get('tmpfs') + if tvols: + after.extend(prep_tmpfs_for_comp(tvols)) + tvols = self.params.get('volume') + if tvols: + after.extend(prep_volume_for_comp(tvols)) + after = prep_mount_args_for_comp(after) + before, after = sorted(list(set(before))), sorted(list(set(after))) - return self._diff_update_and_compare('volume', before, after) + return self._diff_update_and_compare('volumes_all', before, after) def diffparam_volumes_from(self): # Possibly volumesfrom is not in config From 84b11d944afd46155b0a066b6cc867de6717a762 Mon Sep 17 00:00:00 2001 From: Andrew Date: Wed, 12 Oct 2022 16:41:10 +1300 Subject: [PATCH 6/6] number fields in string format and key error Fixes: ERROR: Found 7 pylint issue(s) which need to be resolved: ERROR: plugins/module_utils/podman/podman_container_lib.py:1288:37: ansible-format-automatic-specification: Format string contains automatic field numbering specification ... ERROR: plugins/module_utils/podman/podman_container_lib.py:1400:27: ansible-format-automatic-specification: Format string contains automatic field numbering specification also: File "/tmp/ansible_containers.podman.podman_container_payload_6atuqv63/ansible_containers.podman.podman_container_payload.zip/ansible_collections/containers/podman/plugins/module_utils/podman/podman_container_lib.py", line 1405, in diffparam_volumes_all KeyError: 'config' Signed-off-by: Andrew --- .../module_utils/podman/podman_container_lib.py | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/plugins/module_utils/podman/podman_container_lib.py b/plugins/module_utils/podman/podman_container_lib.py index 9fdb6b3b..456a27a9 100644 --- a/plugins/module_utils/podman/podman_container_lib.py +++ b/plugins/module_utils/podman/podman_container_lib.py @@ -1285,7 +1285,7 @@ def prep_mount_args_for_comp(mounts, mtype=None): else: mstr = ','.join([default_mount_args[mstype], mstr]).strip(',') if not mstype: - raise ValueError("mount type not set/found for '{}'.".format(mstr)) + raise ValueError("mount type not set/found for '{0}'.".format(mstr)) for mitem in mstr.split(','): nv = mitem.split('=', maxsplit=1) miname = nv[0] @@ -1333,13 +1333,13 @@ def prep_mount_args_for_comp(mounts, mtype=None): if not mival: mival = 'true' elif mstype == 'tmpfs' and miname in ['size', 'mode'] and mival: - miname = 'tmpfs-{}'.format(miname) + miname = 'tmpfs-{0}'.format(miname) elif miname in ['type', 'tmpfs-size', 'tmpfs-mode', 'bind-propagation', 'relabel']: pass elif miname.startswith('_unparsed_'): pass else: - miname = '_unparsed_{}'.format(miname) + miname = '_unparsed_{0}'.format(miname) # source can be optional and can be specified as empty. If it is empty # remove it altogether so that comparisons can be made simply. if miname == 'src' and not mival: @@ -1361,7 +1361,7 @@ def clean_image_vols(iv_type): return prep_volume_for_comp(image_volumes) elif iv_type == 'tmpfs': return prep_tmpfs_for_comp(image_volumes) - raise ValueError("invalid image volume type: {}.".format(iv_type)) + raise ValueError("invalid image volume type: {0}.".format(iv_type)) def prep_tmpfs_for_comp(all_ms): res = [] @@ -1373,9 +1373,9 @@ def prep_tmpfs_for_comp(all_ms): else: dm = ms.split(':', maxsplit=1) if len(dm) == 1: - res.append('dst={}'.format(dm[0])) + res.append('dst={0}'.format(dm[0])) else: - res.append('dst={},{}'.format(dm[0], dm[1])) + res.append('dst={0},{1}'.format(dm[0], dm[1])) return prep_mount_args_for_comp(res[0] if wants_str else res, mtype='tmpfs') def prep_volume_for_comp(all_ms): @@ -1397,12 +1397,12 @@ def prep_volume_for_comp(all_ms): # ms is : dm.append('') # dm[2] = prep_mount_args_for_comp(dm[2], mtype=mtype) - res.append('src={},dst={},{}'.format(dm[0], dm[1], dm[2]).strip(',')) + res.append('src={0},dst={1},{2}'.format(dm[0], dm[1], dm[2]).strip(',')) return prep_mount_args_for_comp(res[0] if wants_str else res, mtype='volume') before = [] iv_type = 'bind' # documented default - image_volumes = list(map(clean_path, self.image_info['config'].get('volumes', {}).keys())) + image_volumes = list(map(clean_path, self.image_info.get('config', {}).get('volumes', {}).keys())) if self.info['config'].get('createcommand'): cr_com = self.info['config']['createcommand'] for i, v in enumerate(cr_com):