testing scalability of snapshots. optimized performance by making --no-holds also not use holds on the target. (this is also more like expected behavious)
This commit is contained in:
		
							
								
								
									
										25
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								README.md
									
									
									
									
									
								
							| @ -3,27 +3,6 @@ | ||||
|  | ||||
| [](https://github.com/psy0rz/zfs_autobackup/actions?query=workflow%3A%22Regression+tests%22) [](https://coveralls.io/github/psy0rz/zfs_autobackup)  [](https://pypi.org/project/zfs-autobackup/) | ||||
|  | ||||
| ## New in v3 | ||||
|  | ||||
| * Complete rewrite, cleaner object oriented code. | ||||
| * Python 3 and 2 support. | ||||
| * Automated regression test against real ZFS environment. | ||||
| * Installable via [pip](https://pypi.org/project/zfs-autobackup/). | ||||
| * Backwards compatible with your current backups and parameters. | ||||
| * Progressive thinning (via a destroy schedule. default schedule should be fine for most people) | ||||
| * Cleaner output, with optional color support (pip install colorama). | ||||
| * Clear distinction between local and remote output. | ||||
| * Summary at the beginning, displaying what will happen and the current thinning-schedule. | ||||
| * More efficient destroying/skipping snapshots on the fly. (no more space issues if your backup is way behind) | ||||
| * Progress indicator (--progress) | ||||
| * Better property management (--set-properties and --filter-properties) | ||||
| * Better resume handling, automatically abort invalid resumes. | ||||
| * More robust error handling. | ||||
| * Prepared for future enhancements. | ||||
| * Supports raw backups for encryption. | ||||
| * Custom SSH client config. | ||||
|  | ||||
|  | ||||
| ## Introduction | ||||
|  | ||||
| This is a tool I wrote to make replicating ZFS datasets easy and reliable. | ||||
| @ -396,6 +375,10 @@ Snapshots on the source that still have to be send to the target wont be destroy | ||||
| * Use ```--clear-refreservation``` to save space on your backup server. | ||||
| * Use ```--clear-mountpoint``` to prevent the target server from mounting the backupped filesystem in the wrong place during a reboot. | ||||
|  | ||||
| ### Performance tips | ||||
|  | ||||
| * --no-holds and --allow-empty improve performance a lot if you deal with large amounts of datasets or snapshots. | ||||
|  | ||||
| ### Speeding up SSH | ||||
|  | ||||
| You can make your ssh connections persistent and greatly speed up zfs-autobackup: | ||||
|  | ||||
| @ -1182,7 +1182,7 @@ class ZfsDataset: | ||||
|         return allowed_filter_properties, allowed_set_properties | ||||
|  | ||||
|     def sync_snapshots(self, target_dataset, features, show_progress=False, filter_properties=None, set_properties=None, | ||||
|                        ignore_recv_exit_code=False, source_holds=True, rollback=False, raw=False, other_snapshots=False, | ||||
|                        ignore_recv_exit_code=False, holds=True, rollback=False, raw=False, other_snapshots=False, | ||||
|                        no_send=False, destroy_incompatible=False): | ||||
|         """sync this dataset's snapshots to target_dataset, while also thinning out old snapshots along the way.""" | ||||
|  | ||||
| @ -1293,13 +1293,14 @@ class ZfsDataset: | ||||
|                 resume_token = None | ||||
|  | ||||
|                 # hold the new common snapshots and release the previous ones | ||||
|                 target_snapshot.hold() | ||||
|                 if source_holds: | ||||
|                 if holds: | ||||
|                     target_snapshot.hold() | ||||
|                     source_snapshot.hold() | ||||
|  | ||||
|                 if prev_source_snapshot: | ||||
|                     if source_holds: | ||||
|                     if holds: | ||||
|                         prev_source_snapshot.release() | ||||
|                     target_dataset.find_snapshot(prev_source_snapshot).release() | ||||
|                         target_dataset.find_snapshot(prev_source_snapshot).release() | ||||
|  | ||||
|                 # we may now destroy the previous source snapshot if its obsolete | ||||
|                 if prev_source_snapshot in source_obsoletes: | ||||
| @ -1595,8 +1596,7 @@ class ZfsAutobackup: | ||||
|                             help='Ignore datasets that seem to be replicated some other way. (No changes since ' | ||||
|                                  'lastest snapshot. Useful for proxmox HA replication)') | ||||
|         parser.add_argument('--no-holds', action='store_true', | ||||
|                             help='Don\'t lock snapshots on the source. (Useful to allow proxmox HA replication to ' | ||||
|                                  'switches nodes)') | ||||
|                             help='Don\'t hold snapshots. (Faster)') | ||||
|  | ||||
|         parser.add_argument('--resume', action='store_true', help=argparse.SUPPRESS) | ||||
|         parser.add_argument('--strip-path', default=0, type=int, | ||||
| @ -1753,7 +1753,7 @@ class ZfsAutobackup: | ||||
|                                               features=common_features, filter_properties=filter_properties, | ||||
|                                               set_properties=set_properties, | ||||
|                                               ignore_recv_exit_code=self.args.ignore_transfer_errors, | ||||
|                                               source_holds=not self.args.no_holds, rollback=self.args.rollback, | ||||
|                                               holds=not self.args.no_holds, rollback=self.args.rollback, | ||||
|                                               raw=self.args.raw, other_snapshots=self.args.other_snapshots, | ||||
|                                               no_send=self.args.no_send, | ||||
|                                               destroy_incompatible=self.args.destroy_incompatible) | ||||
|  | ||||
							
								
								
									
										57
									
								
								test_scaling.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								test_scaling.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | ||||
| from basetest import * | ||||
| import time | ||||
| from bin.zfs_autobackup import * | ||||
|  | ||||
| run_orig=ExecuteNode.run | ||||
| run_counter=0 | ||||
|  | ||||
| def run_count(*args, **kwargs): | ||||
|     global run_counter | ||||
|     run_counter=run_counter+1 | ||||
|     return (run_orig(*args, **kwargs)) | ||||
|  | ||||
| class TestZfsScaling(unittest2.TestCase): | ||||
|  | ||||
|     def setUp(self): | ||||
|         prepare_zpools() | ||||
|         self.longMessage = True | ||||
|  | ||||
|     def test_manysnaps(self): | ||||
|         """count the number of commands when there are many snapshots.""" | ||||
|  | ||||
|         snapshot_count=100 | ||||
|  | ||||
|         # create bunch of snapshots | ||||
|         s="" | ||||
|         for i in range(1970,1970+snapshot_count): | ||||
|             s=s+"zfs snapshot test_source1/fs1@test-{:04}1111000000;".format(i) | ||||
|  | ||||
|         shelltest(s) | ||||
|  | ||||
|         global run_counter | ||||
|  | ||||
|         run_counter=0 | ||||
|         with patch.object(ExecuteNode,'run', run_count) as p: | ||||
|  | ||||
|             with patch('time.strftime', return_value="20101112000000"): | ||||
|                 self.assertFalse(ZfsAutobackup("test test_target1 --verbose --keep-source=10000 --keep-target=10000 --no-holds --allow-empty".split(" ")).run()) | ||||
|  | ||||
|  | ||||
|             #this triggers if you make a change with an impact of more than O(snapshot_count/2) | ||||
|             expected_runs=343 | ||||
|             print("ACTUAL RUNS: {}".format(run_counter)) | ||||
|             self.assertLess(abs(run_counter-expected_runs), snapshot_count/2) | ||||
|  | ||||
|  | ||||
|         run_counter=0 | ||||
|         with patch.object(ExecuteNode,'run', run_count) as p: | ||||
|  | ||||
|             with patch('time.strftime', return_value="20101112000001"): | ||||
|                 self.assertFalse(ZfsAutobackup("test test_target1 --verbose --keep-source=10000 --keep-target=10000 --no-holds --allow-empty".split(" ")).run()) | ||||
|  | ||||
|  | ||||
|             #this triggers if you make a change with an impact of more than O(snapshot_count/2) | ||||
|             expected_runs=47 | ||||
|             print("ACTUAL RUNS: {}".format(run_counter)) | ||||
|             self.assertLess(abs(run_counter-expected_runs), snapshot_count/2) | ||||
|  | ||||
| @ -370,13 +370,13 @@ test_source2/fs3/sub                                   userrefs  -         - | ||||
| test_target1                                           userrefs  -         - | ||||
| test_target1/test_source1                              userrefs  -         - | ||||
| test_target1/test_source1/fs1                          userrefs  -         - | ||||
| test_target1/test_source1/fs1@test-20101111000000      userrefs  1         - | ||||
| test_target1/test_source1/fs1@test-20101111000000      userrefs  0         - | ||||
| test_target1/test_source1/fs1/sub                      userrefs  -         - | ||||
| test_target1/test_source1/fs1/sub@test-20101111000000  userrefs  1         - | ||||
| test_target1/test_source1/fs1/sub@test-20101111000000  userrefs  0         - | ||||
| test_target1/test_source2                              userrefs  -         - | ||||
| test_target1/test_source2/fs2                          userrefs  -         - | ||||
| test_target1/test_source2/fs2/sub                      userrefs  -         - | ||||
| test_target1/test_source2/fs2/sub@test-20101111000000  userrefs  1         - | ||||
| test_target1/test_source2/fs2/sub@test-20101111000000  userrefs  0         - | ||||
| """) | ||||
|  | ||||
|  | ||||
|  | ||||
		Reference in New Issue
	
	Block a user