From 816ee3e52fe1b56631f697ccffd5330d1e05b592 Mon Sep 17 00:00:00 2001 From: Rich Megginson Date: Mon, 19 Feb 2024 06:56:10 -0700 Subject: [PATCH] fix: ensure role is idempotent and supports check mode Cause: Role was not idempotent, and did not support check mode. Consequence: Users could not use role to ensure idempotence or check state. Fix: The role has been fixed to return `"changed": true` if the role changed the state of the system, and `false` otherwise. This means that some operations that used to return an error, such as trying to snapshot a volume that already had a snapshot, or trying to remove a snapshot that does not exist, will no longer return an error, but will instead just report `"changed": false`. Users should use the `list` operation to examine state of the system. Result: The role is idempotent and supports check mode. Signed-off-by: Rich Megginson --- README.md | 16 +- tasks/files/snapshot.py | 482 ++++++++++++------- tasks/main.yml | 19 +- tests/tests_basic.yml | 58 +++ tests/tests_check_no_lv_fail.yml | 1 + tests/tests_check_no_vg_fail.yml | 1 + tests/tests_extend_basic.yml | 36 ++ tests/tests_list.yml | 4 + tests/tests_mount.yml | 37 ++ tests/tests_mount_no_vg_fail.yml | 1 + tests/tests_mount_verify.yml | 37 ++ tests/tests_mount_verify_fail.yml | 1 + tests/tests_multi_snapsets.yml | 36 +- tests/tests_no_space_fail.yml | 1 + tests/tests_revert_basic.yml | 17 + tests/tests_set_basic.yml | 30 ++ tests/tests_set_check_no_lv_fail.yml | 1 + tests/tests_set_check_no_vg_fail.yml | 1 + tests/tests_set_extend.yml | 30 ++ tests/tests_set_extend_verify_fail.yml | 1 + tests/tests_set_mount.yml | 61 +++ tests/tests_set_mount_verify_fail.yml | 1 + tests/tests_set_revert.yml | 30 ++ tests/tests_set_revert_no_snapshots_fail.yml | 36 -- tests/tests_set_snapshot_invalid_param.yml | 1 + tests/tests_set_snapshot_missing_param.yml | 1 + tests/tests_set_snapshot_no_space_fail.yml | 1 + tests/tests_umount_verify_fail.yml | 1 + tests/verify-role-failed.yml | 11 +- vars/main.yml | 1 + 30 files changed, 732 insertions(+), 222 deletions(-) delete mode 100644 tests/tests_set_revert_no_snapshots_fail.yml diff --git a/README.md b/README.md index 785ce62..d26c285 100644 --- a/README.md +++ b/README.md @@ -33,12 +33,14 @@ This variable is required. It supports one of the following values: - `remove`: Remove snapshots that conform to the specified prefix and pattern -- `revert`: Revert to snapshots that are specifed by either the pattern or set. If either the source LV or - snapshot are open, the merge is deferred until the next time the server reboots and the - source logical volume is activated. +- `revert`: Revert to snapshots that are specified by either the pattern or set. + If either the source LV or snapshot are open, the merge is deferred + until the next time the server reboots and the source logical volume + is activated. -- `extend`: Extend snapshot to have at least snapshot_lvm_percent_space_required space allocated to the - snapshot. Allocations are rounded up to the next multiple of the volume group extent size. +- `extend`: Extend snapshot to have at least snapshot_lvm_percent_space_required + space allocated to the snapshot. Allocations are rounded up to the + next multiple of the volume group extent size. ### snapshot_lvm_set @@ -108,8 +110,6 @@ that it applies, for example: The mount_origin flag defaults to false, so it is not necessary when the user is mounting the snapshot rather than the origin. -If before running the role, with : - ### snapshot_lvm_snapset_name This variable is required. snapshot_lvm_snapset_name is a string that will be @@ -218,7 +218,7 @@ When used inside of a snapset definition, use mount_origin parameter. ### snapshot_lvm_unmount_all If set to true, unmount all mountpoint for the resulting blockdevice. -Linux allows filesytems to be mounted in multiple locations. Setting +Linux allows filesystems to be mounted in multiple locations. Setting this flag will unmount all locations. ### snapshot_lvm_vg_include diff --git a/tasks/files/snapshot.py b/tasks/files/snapshot.py index 79497ab..a457e6e 100644 --- a/tasks/files/snapshot.py +++ b/tasks/files/snapshot.py @@ -94,9 +94,10 @@ class SnapshotStatus: ERROR_MOUNT_POINT_NOT_EXISTING = 32 ERROR_MOUNT_NOT_BLOCKDEV = 33 ERROR_MOUNT_INVALID_PARAMS = 34 - ERROR_MOUNT_INUSE = 35 + ERROR_MOUNT_POINT_ALREADY_MOUNTED = 35 ERROR_MOUNT_VERIFY_FAILED = 36 ERROR_UMOUNT_VERIFY_FAILED = 37 + ERROR_UMOUNT_NOT_MOUNTED = 38 def makedirs(path): @@ -104,12 +105,42 @@ def makedirs(path): os.makedirs(path, 0o755) -def mount(blockdev, mountpoint, fstype=None, options=None, mountpoint_create=False): +def get_mounted_device(mount_target): + """If mount_target is mounted, return the device that is mounted. + If mount_target is not mounted, return None.""" + with open("/proc/mounts") as pm: + for line in pm: + params = line.split(" ") + if mount_target == params[1]: + return params[0] + return None + + +def mount(blockdev, mountpoint, fstype=None, options=None, mountpoint_create=False, check_mode=False): mount_command = [] mountpoint = os.path.normpath(mountpoint) if options is None: options = "defaults" + mounted_dev = get_mounted_device(mountpoint) + if mounted_dev: + try: + if os.path.samefile(blockdev, mounted_dev): + return ( + SnapshotStatus.ERROR_MOUNT_POINT_ALREADY_MOUNTED, + mountpoint + " is already mounted at " + blockdev, + ) + else: + return ( + SnapshotStatus.ERROR_MOUNT_FAILED, + mountpoint + " is already mounted at different device " + mounted_dev, + ) + except Exception as exc: + return ( + SnapshotStatus.ERROR_MOUNT_FAILED, + "could not verify mountpoint " + mountpoint + " at " + blockdev + ": " + str(exc) + ) + if not os.path.isdir(mountpoint): if not mountpoint_create: return ( @@ -132,17 +163,29 @@ def mount(blockdev, mountpoint, fstype=None, options=None, mountpoint_create=Fal mount_command.append(blockdev) mount_command.append(mountpoint) + if check_mode: + return rc, "Would run command " + " ".join(mount_command) + rc, output = run_command(mount_command) if rc != 0: - logger.info("failed to mount: ".join(mount_command)) - logger.info(output) - return (SnapshotStatus.ERROR_UMOUNT_FAILED, output) + logger.error("failed to mount: ".join(mount_command)) + logger.error(output) + return SnapshotStatus.ERROR_MOUNT_FAILED, output return SnapshotStatus.SNAPSHOT_OK, "" -def umount(umount_target, all_targets): +def umount(umount_target, all_targets, check_mode): + mounted_dev = get_mounted_device(umount_target) + if not mounted_dev: + return ( + SnapshotStatus.ERROR_UMOUNT_NOT_MOUNTED, + "not mounted " + umount_target, + ) + else: + logger.info("umount target %s from device %s", umount_target, mounted_dev) + umount_command = [] umount_command.append("umount") @@ -150,11 +193,14 @@ def umount(umount_target, all_targets): if all_targets: umount_command.append("-A") + if check_mode: + return rc, "Would run command " + " ".join(umount_command) + rc, output = run_command(umount_command) if rc != 0: - logger.info("failed to unmount %s: %s", umount_target, output) - return (SnapshotStatus.ERROR_UMOUNT_FAILED, output) + logger.error("failed to unmount %s: %s", umount_target, output) + return SnapshotStatus.ERROR_UMOUNT_FAILED, output return SnapshotStatus.SNAPSHOT_OK, "" @@ -351,8 +397,7 @@ def lvm_list_json(vg_name, lv_name): top_level["volumes"] = vg_dict top_level["mounts"] = fs_dict - print(json.dumps(top_level, indent=4)) - return SnapshotStatus.SNAPSHOT_OK, "" + return SnapshotStatus.SNAPSHOT_OK, top_level def get_snapshot_name(lv_name, suffix): @@ -473,7 +518,7 @@ def lvm_is_snapshot(vg_name, snapshot_name): return SnapshotStatus.SNAPSHOT_OK, False -def lvm_snapshot_remove(vg_name, snapshot_name): +def lvm_snapshot_remove(vg_name, snapshot_name, check_mode): rc, is_snapshot = lvm_is_snapshot(vg_name, snapshot_name) if rc != SnapshotStatus.SNAPSHOT_OK: @@ -487,6 +532,9 @@ def lvm_snapshot_remove(vg_name, snapshot_name): remove_command = ["lvremove", "-y", vg_name + "/" + snapshot_name] + if check_mode: + return rc, "Would run command " + " ".join(remove_command) + rc, output = run_command(remove_command) if rc: @@ -495,7 +543,7 @@ def lvm_snapshot_remove(vg_name, snapshot_name): return SnapshotStatus.SNAPSHOT_OK, "" -def revert_lv(vg_name, snapshot_name): +def revert_lv(vg_name, snapshot_name, check_mode): rc, _vg_exists, lv_exists = lvm_lv_exists(vg_name, snapshot_name) if rc != SnapshotStatus.SNAPSHOT_OK: raise LvmBug("'lvs' failed '%d'" % rc) @@ -508,12 +556,15 @@ def revert_lv(vg_name, snapshot_name): ) else: return ( - SnapshotStatus.ERROR_REVERT_FAILED, + SnapshotStatus.ERROR_LV_NOTFOUND, "snapshot not found with name: " + vg_name + "/" + snapshot_name, ) revert_command = ["lvconvert", "--merge", vg_name + "/" + snapshot_name] + if check_mode: + return rc, "Would run command " + " ".join(revert_command) + rc, output = run_command(revert_command) if rc: @@ -522,8 +573,9 @@ def revert_lv(vg_name, snapshot_name): return SnapshotStatus.SNAPSHOT_OK, output -def revert_lvs(vg_name, lv_name, suffix): +def revert_lvs(vg_name, lv_name, suffix, check_mode): # Revert snapshots + changed = False for vg, lv_list in vgs_lvs_iterator(vg_name, lv_name): for lv in lv_list: lv_item_name = lv["lv_name"] @@ -539,33 +591,41 @@ def revert_lvs(vg_name, lv_name, suffix): if not lv_item_name.endswith(suffix): continue - rc, message = revert_lv(vg["vg_name"], lv_item_name) + rc, message = revert_lv(vg["vg_name"], lv_item_name, check_mode) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + if rc == SnapshotStatus.ERROR_LV_NOTFOUND: + rc = SnapshotStatus.SNAPSHOT_OK # already removed or reverted + return rc, message, changed - return SnapshotStatus.SNAPSHOT_OK, "" + # if we got here at least 1 snapshot was reverted + changed = True + + return SnapshotStatus.SNAPSHOT_OK, "", changed -def extend_lv_snapshot(vg_name, lv_name, suffix, percent_space_required): +def extend_lv_snapshot(vg_name, lv_name, suffix, percent_space_required, check_mode): snapshot_name = get_snapshot_name(lv_name, suffix) rc, _vg_exists, lv_exists = lvm_lv_exists(vg_name, snapshot_name) + changed = False if lv_exists: if not lvm_is_snapshot(vg_name, snapshot_name): return ( SnapshotStatus.ERROR_EXTEND_NOT_SNAPSHOT, "LV with name: " + vg_name + "/" + snapshot_name + " is not a snapshot", + changed, ) else: return ( SnapshotStatus.ERROR_EXTEND_NOT_FOUND, "snapshot not found with name: " + vg_name + "/" + snapshot_name, + changed, ) rc, _message, current_space_dict = get_current_space_state() if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, "extend_lv get_space_state failure" + return rc, "extend_lv get_space_state failure", changed current_size = current_space_dict[vg_name].lvs[snapshot_name].lv_size required_size = get_space_needed( @@ -573,7 +633,8 @@ def extend_lv_snapshot(vg_name, lv_name, suffix, percent_space_required): ) if current_size >= required_size: - return SnapshotStatus.SNAPSHOT_OK, "" + # rare case of OK return and no change + return SnapshotStatus.SNAPSHOT_OK, "", changed extend_command = [ "lvextend", @@ -582,12 +643,15 @@ def extend_lv_snapshot(vg_name, lv_name, suffix, percent_space_required): vg_name + "/" + snapshot_name, ] + if check_mode: + return rc, "Would run command " + " ".join(extend_command) + rc, output = run_command(extend_command) if rc != SnapshotStatus.SNAPSHOT_OK: - return SnapshotStatus.ERROR_EXTEND_FAILED, output + return SnapshotStatus.ERROR_EXTEND_FAILED, output, changed - return SnapshotStatus.SNAPSHOT_OK, output + return SnapshotStatus.SNAPSHOT_OK, output, True # changed def extend_check_size(vg_name, lv_name, snapshot_name, percent_space_required): @@ -613,22 +677,26 @@ def extend_check_size(vg_name, lv_name, snapshot_name, percent_space_required): return SnapshotStatus.SNAPSHOT_OK, False, "current size too small" -def extend_snapshot_set(snapset_json): +def extend_snapshot_set(snapset_json, check_mode): snapset_name = snapset_json["name"] volume_list = snapset_json["volumes"] logger.info("extend snapsset : %s", snapset_name) + changed = False for list_item in volume_list: vg = list_item["vg"] lv = list_item["lv"] percent_space_required = list_item["percent_space_required"] - rc, message = extend_lv_snapshot(vg, lv, snapset_name, percent_space_required) + rc, message, cmd_changed = extend_lv_snapshot(vg, lv, snapset_name, percent_space_required, check_mode) + + if cmd_changed: + changed = True if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed - return SnapshotStatus.SNAPSHOT_OK, "" + return SnapshotStatus.SNAPSHOT_OK, "", changed def extend_verify_snapshot_set(snapset_json): @@ -752,8 +820,9 @@ def extend_verify_snapshots(vg_name, lv_name, suffix, percent_space_required): return SnapshotStatus.SNAPSHOT_OK, "" -def extend_lvs(vg_name, lv_name, suffix, required_space): +def extend_lvs(vg_name, lv_name, suffix, required_space, check_mode): # Extend snapshots + changed = False for vg, lv_list in vgs_lvs_iterator(vg_name, lv_name): for lv in lv_list: lv = lv["lv_name"] @@ -767,15 +836,18 @@ def extend_lvs(vg_name, lv_name, suffix, required_space): if is_snapshot: continue - rc, message = extend_lv_snapshot(vg["vg_name"], lv, suffix, required_space) + rc, message, cmd_changed = extend_lv_snapshot(vg["vg_name"], lv, suffix, required_space, check_mode) + + if cmd_changed: + changed = True if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed - return SnapshotStatus.SNAPSHOT_OK, "" + return SnapshotStatus.SNAPSHOT_OK, "", changed -def snapshot_lv(vg_name, lv_name, suffix, snap_size): +def snapshot_lv(vg_name, lv_name, suffix, snap_size, check_mode): snapshot_name = get_snapshot_name(lv_name, suffix) rc, _vg_exists, lv_exists = lvm_lv_exists(vg_name, snapshot_name) @@ -802,6 +874,9 @@ def snapshot_lv(vg_name, lv_name, suffix, snap_size): vg_name + "/" + lv_name, ] + if check_mode: + return rc, "Would run command " + " ".join(snapshot_command) + rc, output = run_command(snapshot_command) if rc: @@ -986,21 +1061,27 @@ def check_verify_lvs_completed(snapshot_all, vg_name, lv_name, suffix): return SnapshotStatus.SNAPSHOT_OK, "" -def revert_snapshot_set(snapset_json): +def revert_snapshot_set(snapset_json, check_mode): snapset_name = snapset_json["name"] volume_list = snapset_json["volumes"] logger.info("revert snapsset : %s", snapset_name) + changed = False for list_item in volume_list: vg = list_item["vg"] lv = list_item["lv"] - rc, message = revert_lv(vg, get_snapshot_name(lv, snapset_name)) + rc, message = revert_lv(vg, get_snapshot_name(lv, snapset_name), check_mode) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + if rc == SnapshotStatus.ERROR_LV_NOTFOUND: + rc = SnapshotStatus.SNAPSHOT_OK # already removed or reverted + return rc, message, changed - return SnapshotStatus.SNAPSHOT_OK, "" + # if we got here at least 1 snapshot was reverted + changed = True + + return SnapshotStatus.SNAPSHOT_OK, "", changed def umount_verify(mountpoint, vg_name, lv_to_check): @@ -1025,24 +1106,30 @@ def umount_verify(mountpoint, vg_name, lv_to_check): return SnapshotStatus.SNAPSHOT_OK, "" -def umount_lv(umount_target, vg_name, lv_name, all_targets): +def umount_lv(umount_target, vg_name, lv_name, all_targets, check_mode): logger.info("umount_lv : %s", umount_target) + changed = False if vg_name and lv_name: # Check to make sure all the source vgs/lvs exist rc, message = verify_source_lvs_exist(vg_name, lv_name) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed - return umount(umount_target, all_targets) + rc, message = umount(umount_target, all_targets, check_mode) + changed = rc == SnapshotStatus.SNAPSHOT_OK + if rc == SnapshotStatus.ERROR_UMOUNT_NOT_MOUNTED: + rc = SnapshotStatus.SNAPSHOT_OK # already unmounted - not an error + return rc, message, changed -def umount_snapshot_set(snapset_json, verify_only): +def umount_snapshot_set(snapset_json, verify_only, check_mode): snapset_name = snapset_json["name"] volume_list = snapset_json["volumes"] logger.info("mount verify snapsset : %s", snapset_name) + changed = False for list_item in volume_list: vg_name = list_item["vg"] lv_name = list_item["lv"] @@ -1052,6 +1139,7 @@ def umount_snapshot_set(snapset_json, verify_only): return ( SnapshotStatus.ERROR_UMOUNT_VERIFY_FAILED, "set item must provide a mountpoint for : " + vg_name + "/" + lv_name, + changed, ) if "all_targets" in list_item: @@ -1073,20 +1161,23 @@ def umount_snapshot_set(snapset_json, verify_only): if verify_only: rc, message = umount_verify(mountpoint, vg_name, lv_to_check) else: - rc, message = umount_lv(mountpoint, vg_name, lv_to_check, all_targets) + rc, message, cmd_changed = umount_lv(mountpoint, vg_name, lv_to_check, all_targets, check_mode) + if cmd_changed: + changed = True if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed - return SnapshotStatus.SNAPSHOT_OK, "" + return SnapshotStatus.SNAPSHOT_OK, "", changed -def mount_snapshot_set(snapset_json, verify_only, cmdline_mountpoint_create): +def mount_snapshot_set(snapset_json, verify_only, cmdline_mountpoint_create, check_mode): snapset_name = snapset_json["name"] volume_list = snapset_json["volumes"] logger.info("mount verify snapsset : %s", snapset_name) + changed = False for list_item in volume_list: vg_name = list_item["vg"] lv_name = list_item["lv"] @@ -1120,6 +1211,7 @@ def mount_snapshot_set(snapset_json, verify_only, cmdline_mountpoint_create): return ( SnapshotStatus.ERROR_MOUNT_INVALID_PARAMS, "set item must provide a mountpoint for : " + vg_name + "/" + lv_name, + changed, ) if origin: @@ -1134,7 +1226,7 @@ def mount_snapshot_set(snapset_json, verify_only, cmdline_mountpoint_create): origin, mountpoint, blockdev, vg_name, lv_name, snapset_name ) else: - rc, message = mount_lv( + rc, message, cmd_changed = mount_lv( mountpoint_create, origin, mountpoint, @@ -1144,11 +1236,15 @@ def mount_snapshot_set(snapset_json, verify_only, cmdline_mountpoint_create): vg_name, lv_name, snapset_name, + check_mode, ) + if cmd_changed: + changed = True + if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed - return SnapshotStatus.SNAPSHOT_OK, "" + return SnapshotStatus.SNAPSHOT_OK, "", changed def mount_verify(origin, mountpoint, blockdev, vg_name, lv_name, snapset_name): @@ -1228,13 +1324,16 @@ def mount_lv( vg_name, lv_name, snapset_name, + check_mode, ): logger.info("mount_lv : %s", mountpoint) + changed = False if not blockdev and (not vg_name or not lv_name): return ( SnapshotStatus.ERROR_MOUNT_INVALID_PARAMS, "must provide blockdev or vg/lv for mount source", + changed, ) if vg_name and lv_name: @@ -1246,7 +1345,7 @@ def mount_lv( # Check to make sure all the source vgs/lvs exist rc, message = verify_source_lvs_exist(vg_name, lv_to_mount) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed blockdev = path_join(DEV_PREFIX, vg_name, lv_to_mount) else: @@ -1255,25 +1354,31 @@ def mount_lv( return ( SnapshotStatus.ERROR_MOUNT_NOT_BLOCKDEV, "blockdev parameter is not a block device", + changed, ) if not blockdev: return ( SnapshotStatus.ERROR_MOUNT_NOT_BLOCKDEV, "blockdev or vg/lv is a required", + changed, ) - rc, message = mount(blockdev, mountpoint, fstype, options, create) + rc, message = mount(blockdev, mountpoint, fstype, options, create, check_mode) + changed = rc == SnapshotStatus.SNAPSHOT_OK + if rc == SnapshotStatus.ERROR_MOUNT_POINT_ALREADY_MOUNTED: + rc = SnapshotStatus.SNAPSHOT_OK # this is ok - return rc, message + return rc, message, changed -def remove_snapshot_set(snapset_json): +def remove_snapshot_set(snapset_json, check_mode): snapset_name = snapset_json["name"] volume_list = snapset_json["volumes"] logger.info("remove snapsset : %s", snapset_name) # check to make sure the set is removable before attempting to remove + changed = False for list_item in volume_list: vg = list_item["vg"] lv = list_item["lv"] @@ -1282,7 +1387,7 @@ def remove_snapshot_set(snapset_json): rc, vg_exists, lv_exists = lvm_lv_exists(vg, snapshot_name) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, "failed to get LV status" + return rc, "failed to get LV status", changed # if there is no snapshot, continue (idempotent) if not vg_exists or not lv_exists: @@ -1291,12 +1396,13 @@ def remove_snapshot_set(snapset_json): rc, in_use = lvm_is_inuse(vg, snapshot_name) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, "failed to lvm_is_inuse status" + return rc, "failed to lvm_is_inuse status", changed if in_use: return ( SnapshotStatus.ERROR_REMOVE_FAILED_INUSE, "volume is in use: " + vg + "/" + snapshot_name, + changed, ) for list_item in volume_list: @@ -1307,18 +1413,21 @@ def remove_snapshot_set(snapset_json): rc, vg_exists, lv_exists = lvm_lv_exists(vg, snapshot_name) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, "failed to get LV status" + return rc, "failed to get LV status", changed # if there is no snapshot, continue (idempotent) if not vg_exists or not lv_exists: continue - rc, message = lvm_snapshot_remove(vg, snapshot_name) + rc, message = lvm_snapshot_remove(vg, snapshot_name, check_mode) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed - return SnapshotStatus.SNAPSHOT_OK, "" + # if we got here, at least 1 snapshot was removed + changed = True + + return SnapshotStatus.SNAPSHOT_OK, "", changed def remove_verify_snapshot_set(snapset_json): @@ -1349,7 +1458,7 @@ def remove_verify_snapshot_set(snapset_json): return SnapshotStatus.SNAPSHOT_OK, "" -def remove_snapshots(volume_group, logical_volume, suffix): +def remove_snapshots(volume_group, logical_volume, suffix, check_mode): rc = SnapshotStatus.SNAPSHOT_OK message = "" @@ -1358,6 +1467,7 @@ def remove_snapshots(volume_group, logical_volume, suffix): else: search_lv_name = None + changed = False for vg, lv_list in vgs_lvs_iterator(volume_group, search_lv_name): vg_name = vg["vg_name"] @@ -1367,15 +1477,18 @@ def remove_snapshots(volume_group, logical_volume, suffix): if not lvm_is_owned(lv_name, suffix): continue - rc, message = lvm_snapshot_remove(vg_name, lv_name) + rc, message = lvm_snapshot_remove(vg_name, lv_name, check_mode) if rc != SnapshotStatus.SNAPSHOT_OK: break + # if we got here, at least 1 snapshot was removed + changed = True + if volume_group: break - return rc, message + return rc, message, changed def remove_verify_snapshots(vg_name, lv_name, suffix): @@ -1509,14 +1622,21 @@ def verify_snapset_target_no_existing(snapset_json): if rc != SnapshotStatus.SNAPSHOT_OK: return ( rc, - "volume exists that matches the pattern: " + vg + "/" + snapshot_name, + "could not determine if snapshot exists: " + vg + "/" + snapshot_name, ) if lv_exists: - return ( - SnapshotStatus.ERROR_VERIFY_REMOVE_FAILED, - "volume exists that matches the pattern: " + vg + "/" + snapshot_name, - ) + rc, exists = lvm_is_snapshot(vg, snapshot_name) + if rc == SnapshotStatus.SNAPSHOT_OK and exists: + return ( + SnapshotStatus.ERROR_ALREADY_EXISTS, + "snapshot already exists: " + vg + "/" + snapshot_name, + ) + else: + return ( + SnapshotStatus.ERROR_SNAPSET_CHECK_STATUS_FAILED, + "volume exists that matches the pattern: " + vg + "/" + snapshot_name, + ) return SnapshotStatus.SNAPSHOT_OK, "" @@ -1646,13 +1766,16 @@ def snapshot_precheck_lv_set(snapset_json): return SnapshotStatus.SNAPSHOT_OK, "", current_space_dict -def snapshot_create_set(snapset_json): +def snapshot_create_set(snapset_json, check_mode): snapset_name = snapset_json["name"] volume_list = snapset_json["volumes"] + changed = False rc, message, current_space_dict = snapshot_precheck_lv_set(snapset_json) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + if rc == SnapshotStatus.ERROR_ALREADY_EXISTS: + rc = SnapshotStatus.SNAPSHOT_OK + return rc, message, changed # Take snapshots for list_item in volume_list: @@ -1665,30 +1788,33 @@ def snapshot_create_set(snapset_json): vg, lv, percent_space_required, current_space_dict ) - rc, message = snapshot_lv(vg, lv, snapset_name, required_size) + rc, message = snapshot_lv(vg, lv, snapset_name, required_size, check_mode) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed - return SnapshotStatus.SNAPSHOT_OK, "" + # if we got here, at least 1 snapshot was created + changed = True + + return SnapshotStatus.SNAPSHOT_OK, "", changed -def snapshot_set(snapset_json): +def snapshot_set(snapset_json, check_mode): + changed = False rc, message = verify_snapset_source_lvs_exist(snapset_json) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed - rc, message = snapshot_create_set(snapset_json) - if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + rc, message, changed = snapshot_create_set(snapset_json, check_mode) - return SnapshotStatus.SNAPSHOT_OK, "" + return rc, message, changed -def snapshot_lvs(required_space, snapshot_all, vg_name, lv_name, suffix): +def snapshot_lvs(required_space, snapshot_all, vg_name, lv_name, suffix, check_mode): # check to make sure there is space and no name conflicts + changed = False rc, message = check_lvs(required_space, vg_name, lv_name, suffix) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed vg_found = False lv_found = False @@ -1717,78 +1843,72 @@ def snapshot_lvs(required_space, snapshot_all, vg_name, lv_name, suffix): lv["lv_name"], suffix, snap_size, + check_mode ) - # TODO: Should the exiting snapshot be removed and be updated? + # TODO: Should the existing snapshot be removed and be updated? + # richm - IMO no - Ansible idempotence requires that the task should + # report "changed": false for this snapshot - user can use `list` + # to get list of existing snapshots, and use `remove` if they + # want to remove and recreate if rc == SnapshotStatus.ERROR_ALREADY_EXISTS: continue if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return rc, message, changed + + # if we got here, then at least 1 snapshot was created + changed = True if not snapshot_all: if vg_name and not vg_found: - return SnapshotStatus.ERROR_VG_NOTFOUND, "volume group does not exist" + return SnapshotStatus.ERROR_VG_NOTFOUND, "volume group does not exist: " + vg_name, changed if lv_name and not lv_found: - return SnapshotStatus.ERROR_LV_NOTFOUND, "logical volume does not exist" + return SnapshotStatus.ERROR_LV_NOTFOUND, "logical volume does not exist: " + lv_name, changed - return SnapshotStatus.SNAPSHOT_OK, "" - - -def validate_snapset_args(args): - if args.set_json is None: - print("%s snapset command requires -group parameter", args.operation) - sys.exit(SnapshotStatus.ERROR_CMD_INVALID) + return SnapshotStatus.SNAPSHOT_OK, "", changed def validate_args(args): + rc = SnapshotStatus.ERROR_CMD_INVALID if args.all and args.volume_group: - print("-all and --volume_group are mutually exclusive: ", args.operation) - sys.exit(SnapshotStatus.ERROR_CMD_INVALID) + return rc, "--all and --volume_group are mutually exclusive for operation " + args.operation if not args.all and args.volume_group is None and args.suffix is None: - print( - "must specify either --all, --volume_group or --snapset: ", args.operation - ) - sys.exit(SnapshotStatus.ERROR_CMD_INVALID) + return rc, "must specify either --all, --volume_group or --snapset for operation " + args.operation if not args.all and args.volume_group is None and args.logical_volume: - print("--logical_volume requires --volume_group parameter : ", args.operation) - sys.exit(SnapshotStatus.ERROR_CMD_INVALID) + return rc, "--logical_volume requires --volume_group parameter for operation " + args.operation if not args.suffix: - print("--snapset is required : ", args.operation) - sys.exit(SnapshotStatus.ERROR_CMD_INVALID) + return rc, "--snapset is required for operation " + args.operation if len(args.suffix) == 0: - print("Snapset name must be provided : ", args.operation) - sys.exit(SnapshotStatus.ERROR_CMD_INVALID) + return rc, "Snapset name must be provided for operation " + args.operation # not all commands include required_space if hasattr(args, "required_space"): rc, message, _required_space = get_required_space(args.required_space) if rc != SnapshotStatus.SNAPSHOT_OK: - print(message) - sys.exit(SnapshotStatus.ERROR_CMD_INVALID) + return rc, message - return True + return SnapshotStatus.SNAPSHOT_OK, "" def validate_umount_args(args): if not args.mountpoint and not args.blockdev: - print("--mountpoint or --blockdev is required : ", args.operation) - sys.exit(SnapshotStatus.ERROR_MOUNT_INVALID_PARAMS) + return SnapshotStatus.ERROR_MOUNT_INVALID_PARAMS, "--mountpoint or --blockdev is required" + + return SnapshotStatus.SNAPSHOT_OK, "" def validate_mount_args(args): if not args.blockdev and (not args.volume_group or not args.logical_volume): - print("must provide blockdev or vg/lv for mount source : ", args.operation) - sys.exit(SnapshotStatus.ERROR_MOUNT_INVALID_PARAMS) + return SnapshotStatus.ERROR_MOUNT_INVALID_PARAMS, "must provide blockdev or vg/lv for mount source" if not args.mountpoint: - print("mountpoint is required : ", args.operation) - sys.exit(SnapshotStatus.ERROR_MOUNT_INVALID_PARAMS) + return SnapshotStatus.ERROR_MOUNT_INVALID_PARAMS, "mountpoint is required" return SnapshotStatus.SNAPSHOT_OK, "" @@ -1814,16 +1934,9 @@ def get_required_space(required_space_str): return SnapshotStatus.SNAPSHOT_OK, "", percent_space_required -def print_result(rc, message): - if rc != SnapshotStatus.SNAPSHOT_OK: - # NOTICE - even though this says stderr, if this is called via the - # Ansible builtin.script module, with the ssh connection plugin (the default), - # then the stderr output will be joined with stdout. HOWEVER - if running - # with -c local, then stdout and stderr will be separated into different - # channels. - # See https://docs.ansible.com/ansible/latest/collections/ansible/builtin/script_module.html#notes - print(message, file=sys.stderr) - logger.info("exit code: %d: %s", rc, message) +def print_result(result): + json.dump(result, sys.stdout, indent=4) + logger.info("exit code: %d: %s", result["return_code"], str(result["errors"])) def validate_json_request(snapset_json, check_percent_space_required): @@ -1906,7 +2019,7 @@ def validate_snapset_json(cmd, snapset, verify_only): rc, message = validate_json_umount_request(snapset_json) else: rc = SnapshotStatus.ERROR_UNKNOWN_FAILURE - message = "validate_snapset_json" + message = "validate_snapset_json for command " + cmd logger.info("snapset %s", snapset_json) return rc, message, snapset_json @@ -1914,37 +2027,42 @@ def validate_snapset_json(cmd, snapset, verify_only): def snapshot_cmd(args): logger.info( - "snapshot_cmd: %s %s %s %s %s %s", + "snapshot_cmd: %s %s %s %s %s %s %s", args.operation, args.required_space, args.volume_group, args.logical_volume, args.suffix, args.set_json, + args.check_mode, ) if args.set_json is None: - validate_args(args) + rc, message = validate_args(args) + if rc != SnapshotStatus.SNAPSHOT_OK: + return {"return_code": rc, "errors": [message], "changed": False} + rc, message, required_space = get_required_space(args.required_space) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return {"return_code": rc, "errors": [message], "changed": False} - rc, message = snapshot_lvs( + rc, message, changed = snapshot_lvs( required_space, args.all, args.volume_group, args.logical_volume, args.suffix, + args.check_mode, ) else: rc, message, snapset_json = validate_snapset_json( SnapshotCommand.SNAPSHOT, args.set_json, False ) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message - rc, message = snapshot_set(snapset_json) + return {"return_code": rc, "errors": [message], "changed": False} + rc, message, changed = snapshot_set(snapset_json, args.check_mode) - return rc, message + return {"return_code": rc, "errors": [message], "changed": changed} def check_cmd(args): @@ -1960,7 +2078,9 @@ def check_cmd(args): ) if args.set_json is None: - validate_args(args) + rc, message = validate_args(args) + if rc != SnapshotStatus.SNAPSHOT_OK: + return {"return_code": rc, "errors": [message], "changed": False} if args.verify: rc, message = check_verify_lvs_completed( @@ -1981,14 +2101,14 @@ def check_cmd(args): SnapshotCommand.CHECK, args.set_json, args.verify ) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return {"return_code": rc, "errors": [message], "changed": False} if args.verify: rc, message = check_verify_lvs_set(snapset_json) else: rc, message, _current_space_dict = snapshot_precheck_lv_set(snapset_json) - return rc, message + return {"return_code": rc, "errors": [message], "changed": False} def remove_cmd(args): @@ -2002,31 +2122,37 @@ def remove_cmd(args): args.set_json, ) + changed = False if args.set_json is None: - validate_args(args) + rc, message = validate_args(args) + if rc != SnapshotStatus.SNAPSHOT_OK: + return {"return_code": rc, "errors": [message], "changed": changed} + if args.all and args.volume_group: - print("--all and --volume_group are mutually exclusive: ", args.operation) - sys.exit(1) + return {"return_code": SnapshotStatus.ERROR_CMD_INVALID, + "errors": ["--all and --volume_group are mutually exclusive for operation " + args.operation], + "changed": changed} if args.verify: - return remove_verify_snapshots( + rc, message = remove_verify_snapshots( args.volume_group, args.logical_volume, args.suffix ) else: - return remove_snapshots(args.volume_group, args.logical_volume, args.suffix) + rc, message, changed = remove_snapshots(args.volume_group, args.logical_volume, args.suffix, args.check_mode) else: rc, message, snapset_json = validate_snapset_json( SnapshotCommand.REMOVE, args.set_json, args.verify ) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return {"return_code": rc, "errors": [message], "changed": changed} if args.verify: rc, message = remove_verify_snapshot_set(snapset_json) else: - rc, message = remove_snapshot_set(snapset_json) - return rc, message + rc, message, changed = remove_snapshot_set(snapset_json, args.check_mode) + + return {"return_code": rc, "errors": [message], "changed": changed} def revert_cmd(args): @@ -2040,8 +2166,11 @@ def revert_cmd(args): args.set_json, ) + changed = False if args.set_json is None: - validate_args(args) + rc, message = validate_args(args) + if rc != SnapshotStatus.SNAPSHOT_OK: + return {"return_code": rc, "errors": [message], "changed": changed} if args.verify: rc, message = remove_verify_snapshots( @@ -2050,24 +2179,24 @@ def revert_cmd(args): args.suffix, ) else: - rc, message = revert_lvs( - args.volume_group, args.logical_volume, args.suffix + rc, message, changed = revert_lvs( + args.volume_group, args.logical_volume, args.suffix, args.check_mode ) else: rc, message, snapset_json = validate_snapset_json( SnapshotCommand.CHECK, args.set_json, args.verify ) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return {"return_code": rc, "errors": [message], "changed": changed} if args.verify: # revert re-uses the remove verify since both commands should # cause the snapshot to no longer exist rc, message = remove_verify_snapshot_set(snapset_json) else: - rc, message = revert_snapshot_set(snapset_json) + rc, message, changed = revert_snapshot_set(snapset_json, args.check_mode) - return rc, message + return {"return_code": rc, "errors": [message], "changed": changed} def extend_cmd(args): @@ -2081,8 +2210,11 @@ def extend_cmd(args): args.set_json, ) + changed = False if args.set_json is None: - validate_args(args) + rc, message = validate_args(args) + if rc != SnapshotStatus.SNAPSHOT_OK: + return {"return_code": rc, "errors": [message], "changed": changed} if args.verify: rc, message = extend_verify_snapshots( @@ -2092,25 +2224,26 @@ def extend_cmd(args): args.required_space, ) else: - rc, message = extend_lvs( + rc, message, changed = extend_lvs( args.volume_group, args.logical_volume, args.suffix, args.required_space, + args.check_mode, ) else: rc, message, snapset_json = validate_snapset_json( SnapshotCommand.CHECK, args.set_json, args.verify ) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return {"return_code": rc, "errors": [message], "changed": changed} if args.verify: rc, message = extend_verify_snapshot_set(snapset_json) else: - rc, message = extend_snapshot_set(snapset_json) + rc, message, changed = extend_snapshot_set(snapset_json, args.check_mode) - return rc, message + return {"return_code": rc, "errors": [message], "changed": changed} def list_cmd(args): @@ -2125,16 +2258,19 @@ def list_cmd(args): ) if args.set_json is None: - validate_args(args) - rc, message = lvm_list_json( + rc, message = validate_args(args) + if rc != SnapshotStatus.SNAPSHOT_OK: + return {"return_code": rc, "errors": [message], "changed": False} + + rc, data = lvm_list_json( args.volume_group, args.logical_volume, ) else: # TODO filter the set based on the JSON - rc, message = lvm_list_json(None, None) + rc, data = lvm_list_json(None, None) - return rc, message + return {"return_code": rc, "errors": [], "data": data, "changed": False} def mount_cmd(args): @@ -2153,8 +2289,11 @@ def mount_cmd(args): args.set_json, ) + changed = False if args.set_json is None: - validate_mount_args(args) + rc, message = validate_mount_args(args) + if rc != SnapshotStatus.SNAPSHOT_OK: + return {"return_code": rc, "errors": [message], "changed": changed} if args.verify: rc, message = mount_verify( @@ -2166,7 +2305,7 @@ def mount_cmd(args): args.suffix, ) else: - rc, message = mount_lv( + rc, message, changed = mount_lv( args.create, args.origin, args.mountpoint, @@ -2176,17 +2315,18 @@ def mount_cmd(args): args.volume_group, args.logical_volume, args.suffix, + args.check_mode ) else: rc, message, snapset_json = validate_snapset_json( SnapshotCommand.MOUNT, args.set_json, args.verify ) if rc != SnapshotStatus.SNAPSHOT_OK: - return rc, message + return {"return_code": rc, "errors": [message], "changed": changed} - rc, message = mount_snapshot_set(snapset_json, args.verify, args.create) + rc, message, changed = mount_snapshot_set(snapset_json, args.verify, args.create, args.check_mode) - return rc, message + return {"return_code": rc, "errors": [message], "changed": changed} def umount_cmd(args): @@ -2198,8 +2338,11 @@ def umount_cmd(args): args.volume_group, args.set_json, ) + changed = False if args.set_json is None: - validate_umount_args(args) + rc, message = validate_umount_args(args) + if rc != SnapshotStatus.SNAPSHOT_OK: + return {"return_code": rc, "errors": [message], "changed": changed} if args.verify: rc, message = umount_verify( @@ -2212,16 +2355,16 @@ def umount_cmd(args): umount_target = args.mountpoint else: umount_target = args.blockdev - rc, message = umount_lv( - umount_target, args.volume_group, args.logical_volume, args.all_targets + rc, message, changed = umount_lv( + umount_target, args.volume_group, args.logical_volume, args.all_targets, args.check_mode ) else: rc, message, snapset_json = validate_snapset_json( SnapshotCommand.UMOUNT, args.set_json, False ) - rc, message = umount_snapshot_set(snapset_json, args.verify) + rc, message, changed = umount_snapshot_set(snapset_json, args.verify, args.check_mode) - return rc, message + return {"return_code": rc, "errors": [message], "changed": changed} if __name__ == "__main__": @@ -2265,6 +2408,13 @@ def umount_cmd(args): "pattern. Uses python re.search to match." ), ) + common_parser.add_argument( + "--check-mode", + action="store_true", + default=False, + dest="check_mode", + help="Are we running in Ansible check-mode?", + ) # Group parser group_parser = argparse.ArgumentParser(add_help=False) @@ -2484,7 +2634,7 @@ def umount_cmd(args): VG_INCLUDE = re.compile(args.vg_include) else: VG_INCLUDE = None - return_code, display_message = args.func(args) - print_result(return_code, display_message) + result = args.func(args) + print_result(result) - sys.exit(return_code) + sys.exit(result["return_code"]) diff --git a/tasks/main.yml b/tasks/main.yml index cd05d4a..3cb2283 100644 --- a/tasks/main.yml +++ b/tasks/main.yml @@ -19,22 +19,29 @@ ansible.builtin.script: "{{ __snapshot_cmd }}" args: executable: "{{ __snapshot_python }}" - register: snapshot_cmd + register: snapshot_cmd_raw no_log: true ignore_errors: true # noqa: ignore-errors - handled below + changed_when: false # change handled below too - name: Print out response debug: - var: snapshot_cmd.stdout + var: snapshot_cmd_raw verbosity: 2 +- name: Parse raw output + set_fact: + snapshot_cmd: "{{ snapshot_cmd_raw.stdout | from_json }}" + changed_when: snapshot_cmd["changed"] + - name: Set snapshot_facts to the JSON results set_fact: - snapshot_facts: "{{ snapshot_cmd.stdout | from_json }}" + snapshot_facts: "{{ snapshot_cmd['data'] }}" when: snapshot_lvm_action == "list" - name: Show errors debug: - msg: "{{ snapshot_cmd.stderr }}" - when: snapshot_cmd.rc != 0 - failed_when: snapshot_cmd.rc != 0 + msg: "{{ snapshot_cmd['errors'] }}" + when: snapshot_cmd["return_code"] != 0 + failed_when: + - snapshot_cmd_raw is failed or snapshot_cmd["return_code"] != 0 diff --git a/tests/tests_basic.yml b/tests/tests_basic.yml index cc2dd59..f183d79 100644 --- a/tests/tests_basic.yml +++ b/tests/tests_basic.yml @@ -47,6 +47,10 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_action: snapshot + - name: Assert changes for creation + assert: + that: snapshot_cmd["changed"] + - name: Verify the snapshot LVs are created include_role: name: linux-system-roles.snapshot @@ -56,6 +60,32 @@ snapshot_lvm_verify_only: true snapshot_lvm_action: check + - name: Run the snapshot role again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_percent_space_required: 15 + snapshot_lvm_all_vgs: true + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: snapshot + + - name: Assert no changes for creation + assert: + that: not snapshot_cmd["changed"] + + - name: Verify again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_all_vgs: true + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_verify_only: true + snapshot_lvm_action: check + + - name: Assert no changes for verify + assert: + that: not snapshot_cmd["changed"] + - name: Run the snapshot role remove the snapshot LVs include_role: name: linux-system-roles.snapshot @@ -63,6 +93,10 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_action: remove + - name: Assert changes for removal + assert: + that: snapshot_cmd["changed"] + - name: Use the snapshot_lvm_verify option to make sure remove is done include_role: name: linux-system-roles.snapshot @@ -70,6 +104,30 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_verify_only: true snapshot_lvm_action: remove + + - name: Remove again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: remove + + - name: Assert no changes for remove + assert: + that: not snapshot_cmd["changed"] + + - name: Verify remove again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_verify_only: true + snapshot_lvm_action: remove + + - name: Assert no changes for remove verify + assert: + that: not snapshot_cmd["changed"] + always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_check_no_lv_fail.yml b/tests/tests_check_no_lv_fail.yml index 74c2d30..a78bc0d 100644 --- a/tests/tests_check_no_lv_fail.yml +++ b/tests/tests_check_no_lv_fail.yml @@ -28,6 +28,7 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_verify_only: true snapshot_lvm_action: check + __snapshot_failed_changed: false always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_check_no_vg_fail.yml b/tests/tests_check_no_vg_fail.yml index 2bdda88..c778a98 100644 --- a/tests/tests_check_no_vg_fail.yml +++ b/tests/tests_check_no_vg_fail.yml @@ -26,6 +26,7 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_verify_only: true snapshot_lvm_action: check + __snapshot_failed_changed: false always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_extend_basic.yml b/tests/tests_extend_basic.yml index fd0efdf..c089610 100644 --- a/tests/tests_extend_basic.yml +++ b/tests/tests_extend_basic.yml @@ -65,6 +65,10 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_action: extend + - name: Assert changes for extend + assert: + that: snapshot_cmd["changed"] + - name: Use the snapshot_lvm_verify option to make sure extend is done include_role: name: linux-system-roles.snapshot @@ -74,6 +78,38 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_verify_only: true snapshot_lvm_action: extend + + - name: Assert no changes for extend verify + assert: + that: not snapshot_cmd["changed"] + + - name: Extend again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_percent_space_required: 40 + snapshot_lvm_all_vgs: true + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: extend + + - name: Assert no changes for extend + assert: + that: not snapshot_cmd["changed"] + + - name: Verify extend again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_percent_space_required: 40 + snapshot_lvm_all_vgs: true + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_verify_only: true + snapshot_lvm_action: extend + + - name: Assert no changes for extend verify + assert: + that: not snapshot_cmd["changed"] + always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_list.yml b/tests/tests_list.yml index dbe051e..bbf955a 100644 --- a/tests/tests_list.yml +++ b/tests/tests_list.yml @@ -63,6 +63,10 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_action: list + - name: Assert no changes for list + assert: + that: not snapshot_cmd["changed"] + - name: Run the snapshot role remove the snapshot LVs include_role: name: linux-system-roles.snapshot diff --git a/tests/tests_mount.yml b/tests/tests_mount.yml index 5dc3ffc..9261ce9 100644 --- a/tests/tests_mount.yml +++ b/tests/tests_mount.yml @@ -75,6 +75,10 @@ snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" snapshot_lvm_mountpoint_create: true + - name: Assert changes for mount + assert: + that: snapshot_cmd["changed"] + - name: Mount the snapshot for lv2 include_role: name: linux-system-roles.snapshot @@ -109,6 +113,21 @@ snapshot_lvm_mountpoint_create: true snapshot_lvm_mount_origin: true + - name: Mount the snapshot for lv1 again for idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: mount + snapshot_lvm_vg: test_vg1 + snapshot_lvm_lv: lv1 + snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" + snapshot_lvm_mountpoint_create: true + + - name: Assert no changes for mount + assert: + that: not snapshot_cmd["changed"] + - name: Umount the snapshot for lv1 include_role: name: linux-system-roles.snapshot @@ -119,6 +138,24 @@ snapshot_lvm_lv: lv1 snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" + - name: Assert changes for umount + assert: + that: snapshot_cmd["changed"] + + - name: Umount again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: umount + snapshot_lvm_vg: test_vg1 + snapshot_lvm_lv: lv1 + snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" + + - name: Assert no changes for umount + assert: + that: not snapshot_cmd["changed"] + - name: Umount the snapshot for lv2 include_role: name: linux-system-roles.snapshot diff --git a/tests/tests_mount_no_vg_fail.yml b/tests/tests_mount_no_vg_fail.yml index 6389d70..36a3a4b 100644 --- a/tests/tests_mount_no_vg_fail.yml +++ b/tests/tests_mount_no_vg_fail.yml @@ -40,6 +40,7 @@ snapshot_lvm_lv: lv1 snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" snapshot_lvm_mountpoint_create: true + __snapshot_failed_changed: false - name: Remove the snapshot LVs include_role: diff --git a/tests/tests_mount_verify.yml b/tests/tests_mount_verify.yml index d63c4b8..6056f2c 100644 --- a/tests/tests_mount_verify.yml +++ b/tests/tests_mount_verify.yml @@ -75,6 +75,25 @@ snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" snapshot_lvm_mountpoint_create: true + - name: Assert changes for mount + assert: + that: snapshot_cmd["changed"] + + - name: Mount the snapshot for lv1 again for idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: mount + snapshot_lvm_vg: test_vg1 + snapshot_lvm_lv: lv1 + snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" + snapshot_lvm_mountpoint_create: true + + - name: Assert no changes for mount + assert: + that: not snapshot_cmd["changed"] + - name: Mount the snapshot for lv2 include_role: name: linux-system-roles.snapshot @@ -165,6 +184,24 @@ snapshot_lvm_lv: lv1 snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" + - name: Assert changes for umount + assert: + that: snapshot_cmd["changed"] + + - name: Umount the snapshot for lv1 again for idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: umount + snapshot_lvm_vg: test_vg1 + snapshot_lvm_lv: lv1 + snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" + + - name: Assert no changes for umount + assert: + that: not snapshot_cmd["changed"] + - name: Umount the snapshot for lv2 include_role: name: linux-system-roles.snapshot diff --git a/tests/tests_mount_verify_fail.yml b/tests/tests_mount_verify_fail.yml index 8353b6a..0ccd8a1 100644 --- a/tests/tests_mount_verify_fail.yml +++ b/tests/tests_mount_verify_fail.yml @@ -63,6 +63,7 @@ snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/wrong_mountpoint' }}" snapshot_lvm_verify_only: true + __snapshot_failed_changed: false - name: Umount the snapshot for lv1 include_role: diff --git a/tests/tests_multi_snapsets.yml b/tests/tests_multi_snapsets.yml index 66b59e5..4c0e0af 100644 --- a/tests/tests_multi_snapsets.yml +++ b/tests/tests_multi_snapsets.yml @@ -45,7 +45,11 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_action: snapshot - - name: Verify the snapshot LVs are createdf or snapset1 + - name: Assert changes for snapset1 + assert: + that: snapshot_cmd["changed"] + + - name: Verify the snapshot LVs are created for snapset1 include_role: name: linux-system-roles.snapshot vars: @@ -54,6 +58,19 @@ snapshot_lvm_verify_only: true snapshot_lvm_action: check + - name: Create snapshot LVs for snapset1 to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_percent_space_required: 15 + snapshot_lvm_vg: test_vg1 + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: snapshot + + - name: Assert no changes for snapset1 + assert: + that: not snapshot_cmd["changed"] + - name: Run the snapshot role to create snapshot LVs for snapset2 include_role: name: linux-system-roles.snapshot @@ -63,7 +80,7 @@ snapshot_lvm_snapset_name: snapset2 snapshot_lvm_action: snapshot - - name: Verify the snapshot LVs are createdf or snapset2 + - name: Verify the snapshot LVs are created for snapset2 include_role: name: linux-system-roles.snapshot vars: @@ -79,6 +96,10 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_action: remove + - name: Assert changes for snapset1 removal + assert: + that: snapshot_cmd["changed"] + - name: Use the snapshot_lvm_verify option to make sure remove is done include_role: name: linux-system-roles.snapshot @@ -87,6 +108,17 @@ snapshot_lvm_verify_only: true snapshot_lvm_action: remove + - name: Remove snapshot LVs for snapset1 again for idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: remove + + - name: Assert no changes for snapset1 removal + assert: + that: not snapshot_cmd["changed"] + - name: Run the snapshot role remove the snapshot LVs for snapset2 include_role: name: linux-system-roles.snapshot diff --git a/tests/tests_no_space_fail.yml b/tests/tests_no_space_fail.yml index 9f47dfc..a8b229b 100644 --- a/tests/tests_no_space_fail.yml +++ b/tests/tests_no_space_fail.yml @@ -28,6 +28,7 @@ snapshot_all: true snapshot_lvm_snapset_name: snapset1 snapshot_lvm_action: snapshot + __snapshot_failed_changed: false always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_revert_basic.yml b/tests/tests_revert_basic.yml index 404995d..f56b7f2 100644 --- a/tests/tests_revert_basic.yml +++ b/tests/tests_revert_basic.yml @@ -64,6 +64,10 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_action: revert + - name: Assert changes for revert + assert: + that: snapshot_cmd["changed"] + - name: Use the snapshot_lvm_verify option to make sure revert is done include_role: name: linux-system-roles.snapshot @@ -72,6 +76,19 @@ snapshot_lvm_snapset_name: snapset1 snapshot_lvm_verify_only: true snapshot_lvm_action: revert + + - name: Revert again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_all_vgs: true + snapshot_lvm_snapset_name: snapset1 + snapshot_lvm_action: revert + + - name: Assert no changes for revert + assert: + that: not snapshot_cmd["changed"] + always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_set_basic.yml b/tests/tests_set_basic.yml index c813a87..afe655b 100644 --- a/tests/tests_set_basic.yml +++ b/tests/tests_set_basic.yml @@ -62,6 +62,10 @@ snapshot_lvm_action: snapshot snapshot_lvm_set: "{{ snapshot_test_set }}" + - name: Assert changes for create snapset + assert: + that: snapshot_cmd["changed"] + - name: Run the snapshot role to verify the set of snapshots for the LVs include_role: name: linux-system-roles.snapshot @@ -70,6 +74,17 @@ snapshot_lvm_set: "{{ snapshot_test_set }}" snapshot_lvm_verify_only: true + - name: Create snapset again for idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: snapshot + snapshot_lvm_set: "{{ snapshot_test_set }}" + + - name: Assert no changes for create snapset + assert: + that: not snapshot_cmd["changed"] + - name: Run the snapshot role remove the set include_role: name: linux-system-roles.snapshot @@ -77,6 +92,10 @@ snapshot_lvm_action: remove snapshot_lvm_set: "{{ snapshot_test_set }}" + - name: Assert changes for remove snapset + assert: + that: snapshot_cmd["changed"] + - name: Run the snapshot role to verify the set is removed include_role: name: linux-system-roles.snapshot @@ -84,6 +103,17 @@ snapshot_lvm_action: remove snapshot_lvm_set: "{{ snapshot_test_set }}" snapshot_lvm_verify_only: true + + - name: Remove again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: remove + snapshot_lvm_set: "{{ snapshot_test_set }}" + + - name: Assert no changes for remove snapset + assert: + that: not snapshot_cmd["changed"] always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_set_check_no_lv_fail.yml b/tests/tests_set_check_no_lv_fail.yml index 492af00..a8f252a 100644 --- a/tests/tests_set_check_no_lv_fail.yml +++ b/tests/tests_set_check_no_lv_fail.yml @@ -34,6 +34,7 @@ vg: test_vg1 lv: xxxxx percent_space_required: 20 + __snapshot_failed_changed: false always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_set_check_no_vg_fail.yml b/tests/tests_set_check_no_vg_fail.yml index d8b4a37..300c95c 100644 --- a/tests/tests_set_check_no_vg_fail.yml +++ b/tests/tests_set_check_no_vg_fail.yml @@ -32,6 +32,7 @@ vg: xxxx lv: lv1 percent_space_required: 20 + __snapshot_failed_changed: false always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_set_extend.yml b/tests/tests_set_extend.yml index f13a522..21b701e 100644 --- a/tests/tests_set_extend.yml +++ b/tests/tests_set_extend.yml @@ -96,6 +96,10 @@ snapshot_lvm_action: extend snapshot_lvm_set: "{{ snapshot_extend_set }}" + - name: Assert changes for extend + assert: + that: snapshot_cmd["changed"] + - name: Verify the extend is done include_role: name: linux-system-roles.snapshot @@ -104,6 +108,32 @@ snapshot_lvm_action: extend snapshot_lvm_set: "{{ snapshot_extend_set }}" + - name: Extend the set again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: extend + snapshot_lvm_set: "{{ snapshot_extend_set }}" + + - name: Assert no changes for extend + assert: + that: not snapshot_cmd["changed"] + + - name: Run the snapshot role remove the set + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: remove + snapshot_lvm_set: "{{ snapshot_extend_set }}" + + - name: Run the snapshot role to verify the set is removed + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: remove + snapshot_lvm_set: "{{ snapshot_extend_set }}" + snapshot_lvm_verify_only: true + always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_set_extend_verify_fail.yml b/tests/tests_set_extend_verify_fail.yml index 504ac5b..e6076ed 100644 --- a/tests/tests_set_extend_verify_fail.yml +++ b/tests/tests_set_extend_verify_fail.yml @@ -55,6 +55,7 @@ snapshot_lvm_action: extend snapshot_lvm_verify_only: true __snapshot_lvm_set: "{{ snapshot_test_verify_set }}" + __snapshot_failed_changed: false always: - name: Cleanup diff --git a/tests/tests_set_mount.yml b/tests/tests_set_mount.yml index 84758d8..53dc450 100644 --- a/tests/tests_set_mount.yml +++ b/tests/tests_set_mount.yml @@ -86,6 +86,10 @@ snapshot_lvm_mountpoint_create: true snapshot_lvm_set: "{{ snapshot_test_set }}" + - name: Assert changes for mount + assert: + that: snapshot_cmd["changed"] + - name: Verify the mount is done include_role: name: linux-system-roles.snapshot @@ -94,6 +98,63 @@ snapshot_lvm_action: mount snapshot_lvm_set: "{{ snapshot_test_set }}" + - name: Mount the set again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: mount + snapshot_lvm_mountpoint_create: true + snapshot_lvm_set: "{{ snapshot_test_set }}" + + - name: Assert no changes for mount + assert: + that: not snapshot_cmd["changed"] + + - name: Umount the set + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: umount + snapshot_lvm_set: "{{ snapshot_test_set }}" + + - name: Assert changes for umount + assert: + that: snapshot_cmd["changed"] + + - name: Verify the umount is done + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_verify_only: true + snapshot_lvm_action: umount + snapshot_lvm_set: "{{ snapshot_test_set }}" + + - name: Umount the set again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: umount + snapshot_lvm_set: "{{ snapshot_test_set }}" + + - name: Assert no changes for umount + assert: + that: not snapshot_cmd["changed"] + + - name: Run the snapshot role remove the set + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: remove + snapshot_lvm_set: "{{ snapshot_test_set }}" + + - name: Run the snapshot role to verify the set is removed + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: remove + snapshot_lvm_set: "{{ snapshot_test_set }}" + snapshot_lvm_verify_only: true + always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_set_mount_verify_fail.yml b/tests/tests_set_mount_verify_fail.yml index 504ac5b..e6076ed 100644 --- a/tests/tests_set_mount_verify_fail.yml +++ b/tests/tests_set_mount_verify_fail.yml @@ -55,6 +55,7 @@ snapshot_lvm_action: extend snapshot_lvm_verify_only: true __snapshot_lvm_set: "{{ snapshot_test_verify_set }}" + __snapshot_failed_changed: false always: - name: Cleanup diff --git a/tests/tests_set_revert.yml b/tests/tests_set_revert.yml index 8524b6c..3a82894 100644 --- a/tests/tests_set_revert.yml +++ b/tests/tests_set_revert.yml @@ -77,6 +77,10 @@ snapshot_lvm_action: revert snapshot_lvm_set: "{{ snapshot_test_set }}" + - name: Assert changes for revert + assert: + that: snapshot_cmd["changed"] + - name: Verify the revert is done with snapshot_lvm_verify_only include_role: name: linux-system-roles.snapshot @@ -85,6 +89,32 @@ snapshot_lvm_action: revert snapshot_lvm_set: "{{ snapshot_test_set }}" + - name: Revert the set again to check idempotence + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: revert + snapshot_lvm_set: "{{ snapshot_test_set }}" + + - name: Assert no changes for revert + assert: + that: not snapshot_cmd["changed"] + + - name: Run the snapshot role remove the set + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: remove + snapshot_lvm_set: "{{ snapshot_test_set }}" + + - name: Run the snapshot role to verify the set is removed + include_role: + name: linux-system-roles.snapshot + vars: + snapshot_lvm_action: remove + snapshot_lvm_set: "{{ snapshot_test_set }}" + snapshot_lvm_verify_only: true + always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_set_revert_no_snapshots_fail.yml b/tests/tests_set_revert_no_snapshots_fail.yml deleted file mode 100644 index 40ea366..0000000 --- a/tests/tests_set_revert_no_snapshots_fail.yml +++ /dev/null @@ -1,36 +0,0 @@ ---- -- name: Verify snapshot action fails if no space is available - hosts: all - vars: - test_disk_min_size: "1g" - test_disk_count: 10 - test_storage_pools: - - name: test_vg1 - disks: "{{ range(0, 3) | map('extract', unused_disks) | list }}" - volumes: - - name: lv1 - size: "50%" - tasks: - - name: Run tests - block: - - name: Setup - include_tasks: tasks/setup.yml - - - name: Test failure of reverting snapshot that doesn't exist - include_tasks: verify-role-failed.yml - vars: - __snapshot_failed_regex: "snapshot not found with name:*" - __snapshot_failed_msg: Role did not fail with no space error - __snapshot_failed_params: - snapshot_lvm_action: revert - __snapshot_lvm_set: - name: snapset1 - volumes: - - name: snapshot VG1 LV1 - vg: test_vg1 - lv: lv1 - percent_space_required: 50 - always: - - name: Cleanup - include_tasks: tasks/cleanup.yml - tags: tests::cleanup diff --git a/tests/tests_set_snapshot_invalid_param.yml b/tests/tests_set_snapshot_invalid_param.yml index 281e545..9ec82fb 100644 --- a/tests/tests_set_snapshot_invalid_param.yml +++ b/tests/tests_set_snapshot_invalid_param.yml @@ -33,6 +33,7 @@ vg: test_vg1 lv: lv1 percent_space_required: -20 + __snapshot_failed_changed: false always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_set_snapshot_missing_param.yml b/tests/tests_set_snapshot_missing_param.yml index c421a18..e5c6dc0 100644 --- a/tests/tests_set_snapshot_missing_param.yml +++ b/tests/tests_set_snapshot_missing_param.yml @@ -32,6 +32,7 @@ - name: snapshot VG1 LV1 vg: test_vg1 lv: lv1 + __snapshot_failed_changed: false always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_set_snapshot_no_space_fail.yml b/tests/tests_set_snapshot_no_space_fail.yml index 36c05ea..72ed2ba 100644 --- a/tests/tests_set_snapshot_no_space_fail.yml +++ b/tests/tests_set_snapshot_no_space_fail.yml @@ -31,6 +31,7 @@ vg: test_vg1 lv: lv1 percent_space_required: 50 + __snapshot_failed_changed: false always: - name: Cleanup include_tasks: tasks/cleanup.yml diff --git a/tests/tests_umount_verify_fail.yml b/tests/tests_umount_verify_fail.yml index 275718b..bb94c4f 100644 --- a/tests/tests_umount_verify_fail.yml +++ b/tests/tests_umount_verify_fail.yml @@ -62,6 +62,7 @@ snapshot_lvm_lv: lv1 snapshot_lvm_mountpoint: "{{ test_mnt_parent ~ '/lv1_mp' }}" snapshot_lvm_verify_only: true + __snapshot_failed_changed: false - name: Umount the snapshot for lv1 include_role: diff --git a/tests/verify-role-failed.yml b/tests/verify-role-failed.yml index 1ce4b48..8052bd5 100644 --- a/tests/verify-role-failed.yml +++ b/tests/verify-role-failed.yml @@ -82,11 +82,14 @@ - name: Verify return code and message is correct assert: that: - - snapshot_cmd.failed - - snapshot_cmd.rc != 0 - - snapshot_cmd.stdout_lines is search(__snapshot_failed_regex) or - snapshot_cmd.stderr_lines is search(__snapshot_failed_regex) + - snapshot_cmd_raw is failed + - snapshot_cmd["return_code"] != 0 + - snapshot_cmd["errors"] is search(__snapshot_failed_regex) msg: "{{ __snapshot_failed_msg }}" when: - __snapshot_failed_regex is defined - __snapshot_failed_msg is defined + + - name: Assert role reported changes correctly + assert: + that: snapshot_cmd["changed"] == __snapshot_failed_changed diff --git a/vars/main.yml b/vars/main.yml index e0a7c5d..110f676 100644 --- a/vars/main.yml +++ b/vars/main.yml @@ -17,6 +17,7 @@ __snapshot_required_facts_subsets: "{{ ['!all', '!min'] + __snapshot_required_facts }}" __snapshot_cmd: "{{ 'snapshot.py ' ~ snapshot_lvm_action ~ ' ' ~ + ('--check-mode ' if ansible_check_mode else '') ~ ('-a ' if snapshot_lvm_all_vgs else '') ~ ('-v ' if snapshot_lvm_verify_only else '') ~ ('-O ' if snapshot_lvm_mount_origin else '') ~