From 1a4e66aa9dab4f14e0ffb5a92fdf230f82c7dee7 Mon Sep 17 00:00:00 2001 From: Randy Wilkey <37119201+rwilkey@users.noreply.github.com> Date: Mon, 14 May 2018 22:31:00 -0700 Subject: [PATCH] changes for macOS Startup (#358) These changes are proposed as a solution to issue #356 and maybe issue #322. 1. Change znapzend startup behavior to die if no zfs snap or backup sets are found. 2. Update init directory to add a launchd template and modify README.md to explain use. 3. Update main README.md to mention XCode command-line tools requirement. --- .gitignore | 2 ++ README.md | 5 +++++ configure.ac | 1 + init/README.md | 13 +++++++++++++ init/org.znapzend.plist.in | 30 ++++++++++++++++++++++++++++++ lib/ZnapZend.pm | 25 ++++++++++++------------- t/znapzend.t | 10 +++++++--- 7 files changed, 70 insertions(+), 16 deletions(-) create mode 100644 init/org.znapzend.plist.in diff --git a/.gitignore b/.gitignore index 3d56d93e..f63154c0 100644 --- a/.gitignore +++ b/.gitignore @@ -26,3 +26,5 @@ init/znapzend.service init/znapzend.sysv init/znapzend.upstart init/znapzend.xml +init/org.znapzend.plist +.DS_Store diff --git a/README.md b/README.md index a7e8fe6b..b75b0493 100644 --- a/README.md +++ b/README.md @@ -45,6 +45,11 @@ since the installed perl version is probably very old. On OmniOS/SmartOS you will need perl and gnu-make. +On macOS, if you have not already installed the Xcode command line tools, you can +get them from the command line (Terminal app) with: + + xcode-select --install (or just install the full Xcode app from the Apple app store). + With that in place you can now utter: ```sh diff --git a/configure.ac b/configure.ac index ef1736a6..ea0509e6 100644 --- a/configure.ac +++ b/configure.ac @@ -194,6 +194,7 @@ AC_CONFIG_FILES([ init/znapzend.sysv init/znapzend.upstart init/znapzend.xml + init/org.znapzend.plist ]) AC_SUBST(VERSION) diff --git a/init/README.md b/init/README.md index 652119a2..c1eeb2d8 100644 --- a/init/README.md +++ b/init/README.md @@ -13,6 +13,19 @@ replacing the tags in the corresponding ```.in``` template file to match your existing system layout (replace ```@BINDIR@``` usually with ```/usr/local/bin``` to match other setup documentation). +## macOS/launchd + +For macOS launchd, you can copy the generated ```org.znapzend.plist``` +file to ```/Library/LaunchDaemons``` and then start the daemon with: + +```sh +launchctl load /Library/LaunchDaemons/org.znapzend.plist +``` + +```Note:``` It is recommended to ```not``` set the ```--daemonize``` flag of ```znapzend``` +as launchd will lose control of the process. Check out ```init/org.znapzend.plist.in``` +for an example plist. + ## Solaris/Illumos For solaris/illumos OSes you can tell configure to install a znapzend diff --git a/init/org.znapzend.plist.in b/init/org.znapzend.plist.in new file mode 100644 index 00000000..241300f9 --- /dev/null +++ b/init/org.znapzend.plist.in @@ -0,0 +1,30 @@ + + + + + EnvironmentVariables + + PATH + /usr/local/bin:/usr/bin:/bin:/usr/sbin:/sbin + + KeepAlive + + Crashed + + + Label + org.znapzend + ProgramArguments + + @BINDIR@/org.znapzend + + RunAtLoad + + StandardErrorPath + /var/log/org.znapzend.stderr + StandardOutPath + /var/log/org.znapzend.stdout + ThrottleInterval + 30 + + diff --git a/lib/ZnapZend.pm b/lib/ZnapZend.pm index d3917215..ca3312cc 100644 --- a/lib/ZnapZend.pm +++ b/lib/ZnapZend.pm @@ -129,7 +129,7 @@ my $refreshBackupPlans = sub { $self->backupSets($self->zConfig->getBackupSetEnabled($dataSet)); @{$self->backupSets} - or $self->zLog->warn("No backup set defined or enabled, yet. run 'znapzendzetup' to setup znapzend"); + or die "No backup set defined or enabled, yet. run 'znapzendzetup' to setup znapzend\n"; for my $backupSet (@{$self->backupSets}){ $backupSet->{srcPlanHash} = $self->zTime->backupPlanToHash($backupSet->{src_plan}); @@ -157,7 +157,7 @@ my $refreshBackupPlans = sub { for (keys %$backupSet){ my ($key) = /^dst_([^_]+)_plan$/ or next; - #check if destination exists (i.e. is valid) otherwise recheck as dst might be online, now + #check if destination exists (i.e. is valid) otherwise recheck as dst might be online, now if (!$backupSet->{"dst_$key" . '_valid'}){ $backupSet->{"dst_$key" . '_valid'} = @@ -175,7 +175,7 @@ my $refreshBackupPlans = sub { }; } $backupSet->{"dst_$key" . '_valid'} or - $self->zLog->warn("destination '" . $backupSet->{"dst_$key"} + $self->zLog->warn("destination '" . $backupSet->{"dst_$key"} . "' does not exist or is offline. will be rechecked every run..."); }; } @@ -227,7 +227,7 @@ my $sendRecvCleanup = sub { my $sendFailed = 0; my $startTime = time; $self->zLog->info('starting work on backupSet ' . $backupSet->{src}); - + #get all sub datasets of source filesystem; need to send them all individually if recursive my $srcSubDataSets = $backupSet->{recursive} eq 'on' ? $self->zZfs->listSubDataSets($backupSet->{src}) : [ $backupSet->{src} ]; @@ -254,7 +254,7 @@ my $sendRecvCleanup = sub { } } - #recheck non valid dst as it might be online, now + #recheck non valid dst as it might be online, now if (!$backupSet->{"dst_$key" . '_valid'}) { $backupSet->{"dst_$key" . '_valid'} = @@ -397,7 +397,7 @@ my $createSnapshot = sub { local $ENV{ZNAP_TIME} = $timeStamp; my $skip = 0; - + if ($backupSet->{pre_znap_cmd} && $backupSet->{pre_znap_cmd} ne 'off'){ $self->zLog->info("running pre snapshot command on $backupSet->{src}"); @@ -468,7 +468,7 @@ my $sendWorker = sub { $fc->on( spawn => sub { my ($fc, $pid) = @_; - + $self->zLog->debug('send/receive worker for ' . $backupSet->{src} . " spawned ($pid)"); $backupSet->{send_pid} = $pid; @@ -505,7 +505,7 @@ my $snapWorker = sub { #snapshot worker callback sub { my ($fc, $err) = @_; - + $self->zLog->warn('taking snapshot on ' . $backupSet->{src} . ' failed: ' . $err) if $err; @@ -528,7 +528,7 @@ my $snapWorker = sub { $fc->on( spawn => sub { my ($fc, $pid) = @_; - + $self->zLog->debug('snapshot worker for ' . $backupSet->{src} . " spawned ($pid)"); $backupSet->{snap_pid} = $pid; @@ -662,7 +662,7 @@ sub start { $self->$refreshBackupPlans($self->dataset); $self->$createWorkers; }; - + $self->$refreshBackupPlans($self->dataset); $self->$createWorkers; @@ -670,12 +670,12 @@ sub start { $self->zLog->info("znapzend (PID=$$) initialized -- resuming normal operations."); # if Mojo is running with EV, signals will not be received if the IO loop - # is sleeping so lets activate it periodically + # is sleeping so lets activate it periodically ### RM_COMM_4_TEST ### # remove ### RM_COMM_4_TEST ### comments for testing purpose. ### RM_COMM_4_TEST ### if (0) { Mojo::IOLoop->recurring(1 => sub { }) if not $self->runonce; ### RM_COMM_4_TEST ### } - + #start eventloop Mojo::IOLoop->start; @@ -755,4 +755,3 @@ Shadfl@cpan.orgE> 2014-05-30 had Initial Version =cut - diff --git a/t/znapzend.t b/t/znapzend.t index 755d994a..a25d2ba2 100755 --- a/t/znapzend.t +++ b/t/znapzend.t @@ -54,6 +54,7 @@ sub runCommand { } use Test::More; +use Test::Exception; use_ok 'ZnapZend'; @@ -65,12 +66,15 @@ is (runCommand('--help'), 1, 'znapzend help'); is (runCommand(), 1, 'znapzend'); -is (runCommand(qw(--runonce=tank/source)), 1, 'znapzend --runonce'); +throws_ok { runCommand(qw(--runonce=nosets) ) } qr/No backup set defined or enabled/, + 'znapzend dies with no backup sets defined or enabled at startup'; + +# seems to allow tests to continue so why not? +is (runCommand('--help'), 1, 'znapzend help'); is (runCommand(qw(--daemonize --debug),'--features=oracleMode,recvu', qw( --pidfile=znapzend.pid)), 1, 'znapzend --daemonize'); done_testing; - -1; +1;