From c3b36bbb3b4cb67f1abba18c658a439d29607cb1 Mon Sep 17 00:00:00 2001
From: Andreas Fuchs
Date: Sun, 13 Dec 2020 20:50:59 -0500
Subject: Use notify crate to learn about deletion of the activation sentinel
We should now be able to deploy to non-Linux systems, and build all
the required binaries too.
---
src/activate.rs | 72 +++++++++++++++++++++++++++++++--------------------------
1 file changed, 39 insertions(+), 33 deletions(-)
diff --git a/src/activate.rs b/src/activate.rs
index 193f617..00b637d 100644
--- a/src/activate.rs
+++ b/src/activate.rs
@@ -1,22 +1,20 @@
// SPDX-FileCopyrightText: 2020 Serokell
+// SPDX-FileCopyrightText: 2020 Andreas Fuchs
//
// SPDX-License-Identifier: MPL-2.0
use clap::Clap;
-use futures_util::FutureExt;
-use std::process::Stdio;
use tokio::fs;
use tokio::process::Command;
+use tokio::sync::mpsc;
use tokio::time::timeout;
use std::time::Duration;
-use futures_util::StreamExt;
-
use std::path::Path;
-use inotify::Inotify;
+use notify::{RecommendedWatcher, RecursiveMode, Watcher};
use thiserror::Error;
@@ -159,42 +157,36 @@ pub enum ActivationConfirmationError {
CreateConfirmDirError(std::io::Error),
#[error("Failed to create activation confirmation file: {0}")]
CreateConfirmFileError(std::io::Error),
- #[error("Failed to create inotify instance: {0}")]
- CreateInotifyError(std::io::Error),
- #[error("Failed to create inotify watcher: {0}")]
- CreateInotifyWatcherError(std::io::Error),
+ #[error("Failed to create file system watcher instance: {0}")]
+ CreateWatcherError(notify::Error),
#[error("Error forking process: {0}")]
ForkError(i32),
+ #[error("Could not watch for activation sentinel")]
+ WatcherError(#[from] notify::Error),
}
#[derive(Error, Debug)]
pub enum DangerZoneError {
- #[error("Timeout elapsed for confirmation: {0}")]
- TimesUp(#[from] tokio::time::Elapsed),
+ #[error("Timeout elapsed for confirmation")]
+ TimesUp,
#[error("inotify stream ended without activation confirmation")]
NoConfirmation,
- #[error("There was some kind of error waiting for confirmation (todo figure it out)")]
- SomeKindOfError(std::io::Error),
+ #[error("inotify encountered an error: {0}")]
+ WatchError(notify::Error),
}
async fn danger_zone(
- profile_path: &str,
- mut inotify: Inotify,
+ mut events: mpsc::Receiver>,
confirm_timeout: u16,
) -> Result<(), DangerZoneError> {
info!("Waiting for confirmation event...");
- let mut buffer = [0; 32];
- let mut stream = inotify
- .event_stream(&mut buffer)
- .map_err(DangerZoneError::SomeKindOfError)?;
-
- timeout(Duration::from_secs(confirm_timeout as u64), stream.next())
- .await?
- .ok_or(DangerZoneError::NoConfirmation)?
- .map_err(DangerZoneError::SomeKindOfError)?;
-
- Ok(())
+ match timeout(Duration::from_secs(confirm_timeout as u64), events.recv()).await {
+ Ok(Some(Ok(_))) => Ok(()),
+ Ok(Some(Err(e))) => Err(DangerZoneError::WatchError(e)),
+ Ok(None) => Err(DangerZoneError::NoConfirmation),
+ Err(_) => Err(DangerZoneError::TimesUp),
+ }
}
pub async fn activation_confirmation(
@@ -216,20 +208,34 @@ pub async fn activation_confirmation(
.await
.map_err(ActivationConfirmationError::CreateConfirmDirError)?;
- let mut inotify =
- Inotify::init().map_err(ActivationConfirmationError::CreateConfirmDirError)?;
- inotify
- .add_watch(lock_path, inotify::WatchMask::DELETE)
- .map_err(ActivationConfirmationError::CreateConfirmDirError)?;
+ let (deleted, done) = mpsc::channel(1);
+ let mut watcher: RecommendedWatcher =
+ Watcher::new_immediate(move |res: Result| {
+ let send_result = match res {
+ Ok(e) if e.kind == notify::EventKind::Remove(notify::event::RemoveKind::File) => {
+ deleted.blocking_send(Ok(e))
+ }
+ Ok(_) => Ok(()), // ignore non-removal events
+ Err(e) => deleted.blocking_send(Err(e)),
+ };
+ if let Err(e) = send_result {
+ // We can't communicate our error, but panic-ing would
+ // be bad; let's write an error and trust that the
+ // activate function will realize we aren't sending
+ // data.
+ eprintln!("Could not send file system event to watcher: {}", e);
+ }
+ })?;
+ watcher.watch(lock_path, RecursiveMode::Recursive)?;
if let fork::Fork::Child =
fork::daemon(false, false).map_err(ActivationConfirmationError::ForkError)?
{
std::thread::spawn(move || {
- let mut rt = tokio::runtime::Runtime::new().unwrap();
+ let rt = tokio::runtime::Runtime::new().unwrap();
rt.block_on(async move {
- if let Err(err) = danger_zone(&profile_path, inotify, confirm_timeout).await {
+ if let Err(err) = danger_zone(done, confirm_timeout).await {
if let Err(err) = deactivate(&profile_path).await {
good_panic!("Error de-activating due to another error in confirmation thread, oh no...: {}", err);
}
--
cgit v1.2.3