1
0
Fork 0
mirror of https://github.com/ansible-collections/community.general.git synced 2026-06-11 10:35:34 +00:00
community.general/plugins/modules/kopia_policy.py
munchtoast bac4e25a86 feat: Add kopia_policy module
- Manage Kopia snapshot policies (set, delete, list, show) via the
  Kopia CLI.
- Extends community.general._kopia doc fragment for shared password
  and config options.
- Uses fixed args for read-only _get() list_policies method.
2026-06-04 10:40:58 -04:00

402 lines
13 KiB
Python

#!/usr/bin/python
# Copyright (c) 2026, Dexter Le <dextersydney2001@gmail.com>
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
# SPDX-License-Identifier: GPL-3.0-or-later
from __future__ import annotations
DOCUMENTATION = r"""
module: kopia_policy
short_description: Manage Kopia snapshot policies
author:
- Dexter Le (@munchtoast)
version_added: "13.1.0"
description:
- Manage Kopia snapshot policies using the Kopia CLI.
- Supports setting, deleting, showing, and listing policies.
- Policies control retention, scheduling, file exclusions, and compression for snapshots.
extends_documentation_fragment:
- community.general._attributes
- community.general._kopia
attributes:
check_mode:
support: full
diff_mode:
support: none
options:
state:
description:
- Desired state of the Kopia policy.
- V(set) creates or updates a policy. At least one policy option must be provided.
- V(deleted) removes the policy for O(target) or the global policy when O(global_policy=true).
- V(listed) lists all defined policies.
- V(shown) displays the effective policy for O(target), including inherited values.
type: str
choices: [set, deleted, listed, shown]
default: set
target:
description:
- Policy target in one of the following forms.
- A per-path policy C(user@host:/path).
- A per-host policy C(@host).
- A per-user policy C(user@host).
- Required if O(state=set), O(state=deleted), or O(state=shown), unless O(global_policy=true).
type: str
global_policy:
description:
- When V(true), operate on the global policy instead of a per-target policy.
- When V(true), O(target) must not be set.
type: bool
default: false
retention:
description:
- Snapshot retention settings. All sub-options accept an integer or the string C(inherit)
to remove an override and fall back to the parent policy.
type: dict
suboptions:
keep_latest:
description:
- Number of most-recent snapshots to keep.
- Pass C(inherit) to remove this override.
type: str
keep_hourly:
description:
- Number of most-recent hourly snapshots to keep.
- Pass C(inherit) to remove this override.
type: str
keep_daily:
description:
- Number of most-recent daily snapshots to keep.
- Pass C(inherit) to remove this override.
type: str
keep_weekly:
description:
- Number of most-recent weekly snapshots to keep.
- Pass C(inherit) to remove this override.
type: str
keep_monthly:
description:
- Number of most-recent monthly snapshots to keep.
- Pass C(inherit) to remove this override.
type: str
keep_annual:
description:
- Number of most-recent annual snapshots to keep.
- Pass C(inherit) to remove this override.
type: str
ignore_identical:
description:
- Do not save a new snapshot if its contents are identical to the previous one.
- Accepts V(true), V(false), or C(inherit).
type: str
scheduling:
description:
- Snapshot scheduling settings.
type: dict
suboptions:
interval:
description:
- Time between automatic snapshots, for example C(1h), C(30m), or C(24h).
type: str
times:
description:
- List of times of day at which to take snapshots, in C(HH:mm) format.
- Pass C(inherit) as the only list entry to remove this override.
type: list
elements: str
manual:
description:
- When V(true), only create snapshots manually and disable automatic scheduling.
type: bool
files:
description:
- File and directory exclusion settings.
type: dict
suboptions:
add_ignore:
description:
- List of path patterns to add to the ignore list.
type: list
elements: str
remove_ignore:
description:
- List of path patterns to remove from the ignore list.
type: list
elements: str
max_file_size:
description:
- Exclude files larger than this size. Accepts a byte count or human-readable string
such as C(100MB).
type: str
one_file_system:
description:
- When V(true), do not cross filesystem boundaries during backup.
- Accepts V(true), V(false), or C(inherit).
type: str
ignore_cache_dirs:
description:
- When V(true), ignore directories containing a C(CACHEDIR.TAG) file.
- Accepts V(true), V(false), or C(inherit).
type: str
compression:
description:
- Compression algorithm to apply to snapshot data.
- Refer to C(kopia policy set --compression help) for the list of supported algorithms.
type: str
"""
EXAMPLES = r"""
- name: Set a retention policy for a path
community.general.kopia_policy:
state: set
target: "user@hostname:/home/user"
retention:
keep_latest: "10"
keep_daily: "7"
keep_weekly: "4"
keep_monthly: "6"
config: /etc/kopia/root.config
- name: Set a scheduled snapshot policy
community.general.kopia_policy:
state: set
target: "user@hostname:/var/www"
scheduling:
times:
- "02:00"
- "14:00"
config: /etc/kopia/root.config
- name: Set the global policy with compression and ignore rules
community.general.kopia_policy:
state: set
global_policy: true
compression: zstd
files:
add_ignore:
- "*.tmp"
- ".cache"
ignore_cache_dirs: "true"
config: /etc/kopia/root.config
- name: Inherit keep_daily from the parent policy
community.general.kopia_policy:
state: set
target: "user@hostname:/home/user"
retention:
keep_daily: inherit
config: /etc/kopia/root.config
- name: Show the effective policy for a target
community.general.kopia_policy:
state: shown
target: "user@hostname:/home/user"
config: /etc/kopia/root.config
- name: List all defined policies
community.general.kopia_policy:
state: listed
config: /etc/kopia/root.config
- name: Delete a policy for a specific target
community.general.kopia_policy:
state: deleted
target: "user@hostname:/home/user"
config: /etc/kopia/root.config
"""
RETURN = r"""
kopia_policy:
description: Output from the Kopia policy command.
type: str
sample: ""
returned: always
"""
from ansible_collections.community.general.plugins.module_utils._cmd_runner import cmd_runner_fmt
from ansible_collections.community.general.plugins.module_utils._kopia import (
KOPIA_COMMON_ARGUMENT_SPEC,
kopia_runner,
)
from ansible_collections.community.general.plugins.module_utils._module_helper import StateModuleHelper
def _fmt_retention(value):
"""Expand the retention dict into --keep-* and --ignore-identical-snapshots flags."""
if not value:
return []
flag_map = {
"keep_latest": "--keep-latest",
"keep_hourly": "--keep-hourly",
"keep_daily": "--keep-daily",
"keep_weekly": "--keep-weekly",
"keep_monthly": "--keep-monthly",
"keep_annual": "--keep-annual",
"ignore_identical": "--ignore-identical-snapshots",
}
result = []
for param, flag in flag_map.items():
v = value.get(param)
if v is not None:
result.extend([flag, str(v)])
return result
def _fmt_scheduling(value):
"""Expand the scheduling dict into --snapshot-interval, --snapshot-time, --manual flags."""
if not value:
return []
result = []
if value.get("interval") is not None:
result.extend(["--snapshot-interval", value["interval"]])
if value.get("times") is not None:
result.extend(["--snapshot-time", ",".join(value["times"])])
if value.get("manual"):
result.append("--manual")
return result
def _fmt_files(value):
"""Expand the files dict into file-handling flags."""
if not value:
return []
result = []
for path in value.get("add_ignore") or []:
result.extend(["--add-ignore", path])
for path in value.get("remove_ignore") or []:
result.extend(["--remove-ignore", path])
if value.get("max_file_size") is not None:
result.extend(["--max-file-size", value["max_file_size"]])
if value.get("one_file_system") is not None:
result.extend(["--one-file-system", value["one_file_system"]])
if value.get("ignore_cache_dirs") is not None:
result.extend(["--ignore-cache-dirs", value["ignore_cache_dirs"]])
return result
class KopiaPolicy(StateModuleHelper):
module = dict(
supports_check_mode=True,
argument_spec=dict(
**KOPIA_COMMON_ARGUMENT_SPEC,
state=dict(
type="str",
default="set",
choices=["set", "deleted", "listed", "shown"],
),
target=dict(type="str"),
global_policy=dict(type="bool", default=False),
retention=dict(
type="dict",
options=dict(
keep_latest=dict(type="str"),
keep_hourly=dict(type="str"),
keep_daily=dict(type="str"),
keep_weekly=dict(type="str"),
keep_monthly=dict(type="str"),
keep_annual=dict(type="str"),
ignore_identical=dict(type="str"),
),
),
scheduling=dict(
type="dict",
options=dict(
interval=dict(type="str"),
times=dict(type="list", elements="str"),
manual=dict(type="bool"),
),
),
files=dict(
type="dict",
options=dict(
add_ignore=dict(type="list", elements="str"),
remove_ignore=dict(type="list", elements="str"),
max_file_size=dict(type="str"),
one_file_system=dict(type="str"),
ignore_cache_dirs=dict(type="str"),
),
),
compression=dict(type="str"),
),
required_if=[
("state", "set", ["target", "global_policy"], True),
("state", "deleted", ["target", "global_policy"], True),
("state", "shown", ["target", "global_policy"], True),
],
)
def __init_module__(self):
self.runner = kopia_runner(
self.module,
extra_formats=dict(
list_policies=cmd_runner_fmt.as_fixed("policy", "list"),
target=cmd_runner_fmt.as_list(),
global_policy=cmd_runner_fmt.as_bool("--global"),
retention=cmd_runner_fmt.as_func(_fmt_retention),
scheduling=cmd_runner_fmt.as_func(_fmt_scheduling),
files=cmd_runner_fmt.as_func(_fmt_files),
compression=cmd_runner_fmt.as_opt_val("--compression"),
),
)
self.vars.set("previous_value", self._get()["out"])
self.vars.set("value", self.vars.previous_value, change=True, diff=True)
def __quit_module__(self):
self.vars.set("value", self._get()["out"])
def _get(self):
with self.runner("list_policies config") as ctx:
result = ctx.run()
return dict(
rc=result[0],
out=(result[1].rstrip() if result[1] else None),
err=result[2],
)
def _process_command_output(self, fail_on_err, ignore_err_msg=""):
def process(rc, out, err):
if fail_on_err and rc != 0 and err and ignore_err_msg not in err:
self.do_raise(f"kopia failed with error (rc={rc}): {err}")
out = out.rstrip() if out else ""
return None if out == "" else out
return process
def state_set(self):
with self.runner(
"cli_action state target global_policy retention scheduling files compression config",
output_process=self._process_command_output(True),
check_mode_skip=True,
) as ctx:
ctx.run(cli_action="policy")
def state_deleted(self):
with self.runner(
"cli_action state target global_policy config",
output_process=self._process_command_output(True, "no such policy"),
check_mode_skip=True,
) as ctx:
ctx.run(cli_action="policy")
def state_listed(self):
with self.runner(
"cli_action state config",
output_process=self._process_command_output(True),
) as ctx:
ctx.run(cli_action="policy")
def state_shown(self):
with self.runner(
"cli_action state target global_policy config",
output_process=self._process_command_output(True),
) as ctx:
ctx.run(cli_action="policy")
def main():
KopiaPolicy.execute()
if __name__ == "__main__":
main()