forked from third-party-mirrors/zfs_autobackup
fix #190. --exclude-received now expects number of bytes instead that have to be changed for a dataset to not get excluded. (default 0)
this also makes it so that it doesnt conflict with --allow-empty. added regression tests for exclude-unchanged as well.
This commit is contained in:
parent
ab43689a0f
commit
cdd151d45f
@ -95,3 +95,25 @@ test_target1/fs1@test-20101111000000
|
|||||||
test_target1/fs1/sub@test-20101111000000
|
test_target1/fs1/sub@test-20101111000000
|
||||||
test_target1/fs2/sub@test-20101111000000
|
test_target1/fs2/sub@test-20101111000000
|
||||||
""")
|
""")
|
||||||
|
|
||||||
|
|
||||||
|
def test_exclude_unchanged(self):
|
||||||
|
|
||||||
|
shelltest("zfs snapshot -r test_source1@somesnapshot")
|
||||||
|
|
||||||
|
with patch('time.strftime', return_value="test-20101111000000"):
|
||||||
|
self.assertFalse(
|
||||||
|
ZfsAutobackup(
|
||||||
|
"test test_target1 --verbose --allow-empty --exclude-unchanged=1".split(" ")).run())
|
||||||
|
|
||||||
|
#everything should be excluded, but should not return an error (see #190)
|
||||||
|
with patch('time.strftime', return_value="test-20101111000001"):
|
||||||
|
self.assertFalse(
|
||||||
|
ZfsAutobackup(
|
||||||
|
"test test_target1 --verbose --allow-empty --exclude-unchanged=1".split(" ")).run())
|
||||||
|
|
||||||
|
r = shelltest("zfs list -H -o name -r -t snapshot test_target1")
|
||||||
|
self.assertMultiLineEqual(r, """
|
||||||
|
test_target1/test_source2/fs2/sub@test-20101111000000
|
||||||
|
""")
|
||||||
|
|
||||||
|
@ -15,7 +15,9 @@ class TestZfsNode(unittest2.TestCase):
|
|||||||
node = ZfsNode(utc=False, snapshot_time_format="test-%Y%m%d%H%M%S", hold_name="zfs_autobackup:test", logger=logger, description=description)
|
node = ZfsNode(utc=False, snapshot_time_format="test-%Y%m%d%H%M%S", hold_name="zfs_autobackup:test", logger=logger, description=description)
|
||||||
|
|
||||||
with self.subTest("first snapshot"):
|
with self.subTest("first snapshot"):
|
||||||
node.consistent_snapshot(node.selected_datasets(property_name="autobackup:test",exclude_paths=[], exclude_received=False, exclude_unchanged=False, min_change=200000), "test-20101111000001", 100000)
|
(selected_datasets, excluded_datasets)=node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False,
|
||||||
|
exclude_unchanged=0)
|
||||||
|
node.consistent_snapshot(selected_datasets, "test-20101111000001", 100000)
|
||||||
r = shelltest("zfs list -H -o name -r -t all " + TEST_POOLS)
|
r = shelltest("zfs list -H -o name -r -t all " + TEST_POOLS)
|
||||||
self.assertEqual(r, """
|
self.assertEqual(r, """
|
||||||
test_source1
|
test_source1
|
||||||
@ -33,7 +35,9 @@ test_target1
|
|||||||
""")
|
""")
|
||||||
|
|
||||||
with self.subTest("second snapshot, no changes, no snapshot"):
|
with self.subTest("second snapshot, no changes, no snapshot"):
|
||||||
node.consistent_snapshot(node.selected_datasets(property_name="autobackup:test",exclude_paths=[], exclude_received=False, exclude_unchanged=False, min_change=200000), "test-20101111000002", 1)
|
(selected_datasets, excluded_datasets)=node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False,
|
||||||
|
exclude_unchanged=0)
|
||||||
|
node.consistent_snapshot(selected_datasets, "test-20101111000002", 1)
|
||||||
r = shelltest("zfs list -H -o name -r -t all " + TEST_POOLS)
|
r = shelltest("zfs list -H -o name -r -t all " + TEST_POOLS)
|
||||||
self.assertEqual(r, """
|
self.assertEqual(r, """
|
||||||
test_source1
|
test_source1
|
||||||
@ -51,7 +55,8 @@ test_target1
|
|||||||
""")
|
""")
|
||||||
|
|
||||||
with self.subTest("second snapshot, no changes, empty snapshot"):
|
with self.subTest("second snapshot, no changes, empty snapshot"):
|
||||||
node.consistent_snapshot(node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=False, min_change=200000), "test-20101111000002", 0)
|
(selected_datasets, excluded_datasets) =node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=0)
|
||||||
|
node.consistent_snapshot(selected_datasets, "test-20101111000002", 0)
|
||||||
r = shelltest("zfs list -H -o name -r -t all " + TEST_POOLS)
|
r = shelltest("zfs list -H -o name -r -t all " + TEST_POOLS)
|
||||||
self.assertEqual(r, """
|
self.assertEqual(r, """
|
||||||
test_source1
|
test_source1
|
||||||
@ -79,7 +84,8 @@ test_target1
|
|||||||
with self.subTest("Test if all cmds are executed correctly (no failures)"):
|
with self.subTest("Test if all cmds are executed correctly (no failures)"):
|
||||||
with OutputIO() as buf:
|
with OutputIO() as buf:
|
||||||
with redirect_stdout(buf):
|
with redirect_stdout(buf):
|
||||||
node.consistent_snapshot(node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=False, min_change=1), "test-1",
|
(selected_datasets, excluded_datasets) =node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=False, min_change=1)
|
||||||
|
node.consistent_snapshot(selected_datasets, "test-1",
|
||||||
0,
|
0,
|
||||||
pre_snapshot_cmds=["echo pre1", "echo pre2"],
|
pre_snapshot_cmds=["echo pre1", "echo pre2"],
|
||||||
post_snapshot_cmds=["echo post1 >&2", "echo post2 >&2"]
|
post_snapshot_cmds=["echo post1 >&2", "echo post2 >&2"]
|
||||||
@ -95,7 +101,8 @@ test_target1
|
|||||||
with OutputIO() as buf:
|
with OutputIO() as buf:
|
||||||
with redirect_stdout(buf):
|
with redirect_stdout(buf):
|
||||||
with self.assertRaises(ExecuteError):
|
with self.assertRaises(ExecuteError):
|
||||||
node.consistent_snapshot(node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=False, min_change=1), "test-1",
|
(selected_datasets, excluded_datasets) =node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=False, min_change=1)
|
||||||
|
node.consistent_snapshot(selected_datasets, "test-1",
|
||||||
0,
|
0,
|
||||||
pre_snapshot_cmds=["echo pre1", "false", "echo pre2"],
|
pre_snapshot_cmds=["echo pre1", "false", "echo pre2"],
|
||||||
post_snapshot_cmds=["echo post1", "false", "echo post2"]
|
post_snapshot_cmds=["echo post1", "false", "echo post2"]
|
||||||
@ -112,7 +119,8 @@ test_target1
|
|||||||
with redirect_stdout(buf):
|
with redirect_stdout(buf):
|
||||||
with self.assertRaises(ExecuteError):
|
with self.assertRaises(ExecuteError):
|
||||||
#same snapshot name as before so it fails
|
#same snapshot name as before so it fails
|
||||||
node.consistent_snapshot(node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=False, min_change=1), "test-1",
|
(selected_datasets, excluded_datasets) =node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=False, min_change=1)
|
||||||
|
node.consistent_snapshot(selected_datasets, "test-1",
|
||||||
0,
|
0,
|
||||||
pre_snapshot_cmds=["echo pre1", "echo pre2"],
|
pre_snapshot_cmds=["echo pre1", "echo pre2"],
|
||||||
post_snapshot_cmds=["echo post1", "echo post2"]
|
post_snapshot_cmds=["echo post1", "echo post2"]
|
||||||
@ -158,7 +166,7 @@ test_target1
|
|||||||
logger = LogStub()
|
logger = LogStub()
|
||||||
description = "[Source]"
|
description = "[Source]"
|
||||||
node = ZfsNode(utc=False, snapshot_time_format="test-%Y%m%d%H%M%S", hold_name="zfs_autobackup:test", logger=logger, description=description)
|
node = ZfsNode(utc=False, snapshot_time_format="test-%Y%m%d%H%M%S", hold_name="zfs_autobackup:test", logger=logger, description=description)
|
||||||
s = pformat(node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=True, min_change=1))
|
s = pformat(node.selected_datasets(property_name="autobackup:test", exclude_paths=[], exclude_received=False, exclude_unchanged=1))
|
||||||
print(s)
|
print(s)
|
||||||
|
|
||||||
# basics
|
# basics
|
||||||
|
@ -10,7 +10,7 @@ class CliBase(object):
|
|||||||
Overridden in subclasses that add stuff for the specific programs."""
|
Overridden in subclasses that add stuff for the specific programs."""
|
||||||
|
|
||||||
# also used by setup.py
|
# also used by setup.py
|
||||||
VERSION = "3.2-beta1"
|
VERSION = "3.2-beta2"
|
||||||
HEADER = "{} v{} - (c)2022 E.H.Eefting (edwin@datux.nl)".format(os.path.basename(sys.argv[0]), VERSION)
|
HEADER = "{} v{} - (c)2022 E.H.Eefting (edwin@datux.nl)".format(os.path.basename(sys.argv[0]), VERSION)
|
||||||
|
|
||||||
def __init__(self, argv, print_arguments=True):
|
def __init__(self, argv, print_arguments=True):
|
||||||
|
@ -98,8 +98,8 @@ class ZfsAuto(CliBase):
|
|||||||
|
|
||||||
group=parser.add_argument_group("Selection options")
|
group=parser.add_argument_group("Selection options")
|
||||||
group.add_argument('--ignore-replicated', action='store_true', help=argparse.SUPPRESS)
|
group.add_argument('--ignore-replicated', action='store_true', help=argparse.SUPPRESS)
|
||||||
group.add_argument('--exclude-unchanged', action='store_true',
|
group.add_argument('--exclude-unchanged', metavar='BYTES', default=0, type=int,
|
||||||
help='Exclude datasets that have no changes since any last snapshot. (Useful in combination with proxmox HA replication)')
|
help='Exclude datasets that have less than BYTES data changed since any last snapshot. (Use with proxmox HA replication)')
|
||||||
group.add_argument('--exclude-received', action='store_true',
|
group.add_argument('--exclude-received', action='store_true',
|
||||||
help='Exclude datasets that have the origin of their autobackup: property as "received". '
|
help='Exclude datasets that have the origin of their autobackup: property as "received". '
|
||||||
'This can avoid recursive replication between two backup partners.')
|
'This can avoid recursive replication between two backup partners.')
|
||||||
|
@ -67,7 +67,7 @@ class ZfsAutobackup(ZfsAuto):
|
|||||||
help='Only create snapshot if enough bytes are changed. (default %('
|
help='Only create snapshot if enough bytes are changed. (default %('
|
||||||
'default)s)')
|
'default)s)')
|
||||||
group.add_argument('--allow-empty', action='store_true',
|
group.add_argument('--allow-empty', action='store_true',
|
||||||
help='If nothing has changed, still create empty snapshots. (Faster. Same as --min-change=0)')
|
help='If nothing has changed, still create empty snapshots. (Same as --min-change=0)')
|
||||||
group.add_argument('--other-snapshots', action='store_true',
|
group.add_argument('--other-snapshots', action='store_true',
|
||||||
help='Send over other snapshots as well, not just the ones created by this tool.')
|
help='Send over other snapshots as well, not just the ones created by this tool.')
|
||||||
group.add_argument('--set-snapshot-properties', metavar='PROPERTY=VALUE,...', type=str,
|
group.add_argument('--set-snapshot-properties', metavar='PROPERTY=VALUE,...', type=str,
|
||||||
@ -110,7 +110,7 @@ class ZfsAutobackup(ZfsAuto):
|
|||||||
group.add_argument('--zfs-compressed', action='store_true',
|
group.add_argument('--zfs-compressed', action='store_true',
|
||||||
help='Transfer blocks that already have zfs-compression as-is.')
|
help='Transfer blocks that already have zfs-compression as-is.')
|
||||||
|
|
||||||
group = parser.add_argument_group("ZFS send/recv pipes")
|
group = parser.add_argument_group("Data transfer options")
|
||||||
group.add_argument('--compress', metavar='TYPE', default=None, nargs='?', const='zstd-fast',
|
group.add_argument('--compress', metavar='TYPE', default=None, nargs='?', const='zstd-fast',
|
||||||
choices=compressors.choices(),
|
choices=compressors.choices(),
|
||||||
help='Use compression during transfer, defaults to zstd-fast if TYPE is not specified. ({})'.format(
|
help='Use compression during transfer, defaults to zstd-fast if TYPE is not specified. ({})'.format(
|
||||||
@ -443,12 +443,11 @@ class ZfsAutobackup(ZfsAuto):
|
|||||||
|
|
||||||
################# select source datasets
|
################# select source datasets
|
||||||
self.set_title("Selecting")
|
self.set_title("Selecting")
|
||||||
source_datasets = source_node.selected_datasets(property_name=self.property_name,
|
( source_datasets, excluded_datasets) = source_node.selected_datasets(property_name=self.property_name,
|
||||||
exclude_received=self.args.exclude_received,
|
exclude_received=self.args.exclude_received,
|
||||||
exclude_paths=self.exclude_paths,
|
exclude_paths=self.exclude_paths,
|
||||||
exclude_unchanged=self.args.exclude_unchanged,
|
exclude_unchanged=self.args.exclude_unchanged)
|
||||||
min_change=self.args.min_change)
|
if not source_datasets and not excluded_datasets:
|
||||||
if not source_datasets:
|
|
||||||
self.print_error_sources()
|
self.print_error_sources()
|
||||||
return 255
|
return 255
|
||||||
|
|
||||||
|
@ -239,12 +239,11 @@ class ZfsAutoverify(ZfsAuto):
|
|||||||
|
|
||||||
################# select source datasets
|
################# select source datasets
|
||||||
self.set_title("Selecting")
|
self.set_title("Selecting")
|
||||||
source_datasets = source_node.selected_datasets(property_name=self.property_name,
|
( source_datasets, excluded_datasets) = source_node.selected_datasets(property_name=self.property_name,
|
||||||
exclude_received=self.args.exclude_received,
|
exclude_received=self.args.exclude_received,
|
||||||
exclude_paths=self.exclude_paths,
|
exclude_paths=self.exclude_paths,
|
||||||
exclude_unchanged=self.args.exclude_unchanged,
|
exclude_unchanged=self.args.exclude_unchanged)
|
||||||
min_change=0)
|
if not source_datasets and not excluded_datasets:
|
||||||
if not source_datasets:
|
|
||||||
self.print_error_sources()
|
self.print_error_sources()
|
||||||
return 255
|
return 255
|
||||||
|
|
||||||
|
@ -118,7 +118,7 @@ class ZfsDataset:
|
|||||||
"""true if this dataset is a snapshot"""
|
"""true if this dataset is a snapshot"""
|
||||||
return self.name.find("@") != -1
|
return self.name.find("@") != -1
|
||||||
|
|
||||||
def is_selected(self, value, source, inherited, exclude_received, exclude_paths, exclude_unchanged, min_change):
|
def is_selected(self, value, source, inherited, exclude_received, exclude_paths, exclude_unchanged):
|
||||||
"""determine if dataset should be selected for backup (called from
|
"""determine if dataset should be selected for backup (called from
|
||||||
ZfsNode)
|
ZfsNode)
|
||||||
|
|
||||||
@ -128,12 +128,15 @@ class ZfsDataset:
|
|||||||
:type source: str
|
:type source: str
|
||||||
:type inherited: bool
|
:type inherited: bool
|
||||||
:type exclude_received: bool
|
:type exclude_received: bool
|
||||||
:type exclude_unchanged: bool
|
:type exclude_unchanged: int
|
||||||
:type min_change: bool
|
|
||||||
|
|
||||||
:param value: Value of the zfs property ("false"/"true"/"child"/parent/"-")
|
:param value: Value of the zfs property ("false"/"true"/"child"/parent/"-")
|
||||||
:param source: Source of the zfs property ("local"/"received", "-")
|
:param source: Source of the zfs property ("local"/"received", "-")
|
||||||
:param inherited: True of the value/source was inherited from a higher dataset.
|
:param inherited: True of the value/source was inherited from a higher dataset.
|
||||||
|
|
||||||
|
Returns: True : Selected
|
||||||
|
False: Excluded
|
||||||
|
None: No property found
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# sanity checks
|
# sanity checks
|
||||||
@ -149,7 +152,7 @@ class ZfsDataset:
|
|||||||
|
|
||||||
# non specified, ignore
|
# non specified, ignore
|
||||||
if value == "-":
|
if value == "-":
|
||||||
return False
|
return None
|
||||||
|
|
||||||
# only select childs of this dataset, ignore
|
# only select childs of this dataset, ignore
|
||||||
if value == "child" and not inherited:
|
if value == "child" and not inherited:
|
||||||
@ -179,8 +182,8 @@ class ZfsDataset:
|
|||||||
self.verbose("Excluded (dataset already received)")
|
self.verbose("Excluded (dataset already received)")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
if exclude_unchanged and not self.is_changed(min_change):
|
if not self.is_changed(exclude_unchanged):
|
||||||
self.verbose("Excluded (unchanged since last snapshot)")
|
self.verbose("Excluded (by --exclude-unchanged)")
|
||||||
return False
|
return False
|
||||||
|
|
||||||
self.verbose("Selected")
|
self.verbose("Selected")
|
||||||
|
@ -235,10 +235,10 @@ class ZfsNode(ExecuteNode):
|
|||||||
except Exception as e:
|
except Exception as e:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def selected_datasets(self, property_name, exclude_received, exclude_paths, exclude_unchanged, min_change):
|
def selected_datasets(self, property_name, exclude_received, exclude_paths, exclude_unchanged):
|
||||||
"""determine filesystems that should be backed up by looking at the special autobackup-property, systemwide
|
"""determine filesystems that should be backed up by looking at the special autobackup-property, systemwide
|
||||||
|
|
||||||
returns: list of ZfsDataset
|
returns: ( list of selected ZfsDataset, list of excluded ZfsDataset)
|
||||||
"""
|
"""
|
||||||
|
|
||||||
self.debug("Getting selected datasets")
|
self.debug("Getting selected datasets")
|
||||||
@ -249,8 +249,10 @@ class ZfsNode(ExecuteNode):
|
|||||||
property_name
|
property_name
|
||||||
])
|
])
|
||||||
|
|
||||||
|
|
||||||
# The returnlist of selected ZfsDataset's:
|
# The returnlist of selected ZfsDataset's:
|
||||||
selected_filesystems = []
|
selected_filesystems = []
|
||||||
|
excluded_filesystems = []
|
||||||
|
|
||||||
# list of sources, used to resolve inherited sources
|
# list of sources, used to resolve inherited sources
|
||||||
sources = {}
|
sources = {}
|
||||||
@ -270,9 +272,14 @@ class ZfsNode(ExecuteNode):
|
|||||||
source = raw_source
|
source = raw_source
|
||||||
|
|
||||||
# determine it
|
# determine it
|
||||||
if dataset.is_selected(value=value, source=source, inherited=inherited, exclude_received=exclude_received,
|
selected=dataset.is_selected(value=value, source=source, inherited=inherited, exclude_received=exclude_received,
|
||||||
exclude_paths=exclude_paths, exclude_unchanged=exclude_unchanged,
|
exclude_paths=exclude_paths, exclude_unchanged=exclude_unchanged)
|
||||||
min_change=min_change):
|
|
||||||
selected_filesystems.append(dataset)
|
|
||||||
|
|
||||||
return selected_filesystems
|
if selected==True:
|
||||||
|
selected_filesystems.append(dataset)
|
||||||
|
elif selected==False:
|
||||||
|
excluded_filesystems.append(dataset)
|
||||||
|
#returns None when no property is set.
|
||||||
|
|
||||||
|
|
||||||
|
return ( selected_filesystems, excluded_filesystems)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user