use std::path::Path;
use std::sync::atomic::{AtomicBool, Ordering};
use std::sync::Arc;
use crate::modules::shared::blocking::sync_blocker::SyncBlocker;
use notify::{RecommendedWatcher, RecursiveMode, Watcher};
use super::{LiveLogDirHandle, LiveLogDirReaderError, RawLogDirReader};
#[derive(Debug)]
pub struct RawLiveLogDirReader {
blocker: SyncBlocker,
log_dir_reader: RawLogDirReader,
_watcher: RecommendedWatcher,
active: Arc<AtomicBool>,
}
impl RawLiveLogDirReader {
pub fn open<P: AsRef<Path>>(dir_path: P) -> Result<RawLiveLogDirReader, LiveLogDirReaderError> {
let log_dir_reader = RawLogDirReader::open(&dir_path);
let blocker = SyncBlocker::new();
let local_blocker = blocker.clone();
let mut watcher = notify::recommended_watcher(move |_| {
local_blocker.unblock();
})?;
watcher.watch(dir_path.as_ref(), RecursiveMode::NonRecursive)?;
Ok(Self {
blocker,
log_dir_reader,
active: Arc::new(AtomicBool::new(true)),
_watcher: watcher,
})
}
pub fn handle(&self) -> LiveLogDirHandle {
LiveLogDirHandle::new(self.active.clone(), self.blocker.clone())
}
}
impl Iterator for RawLiveLogDirReader {
type Item = Result<serde_json::Value, LiveLogDirReaderError>;
fn next(&mut self) -> Option<Self::Item> {
loop {
if !self.active.load(Ordering::Relaxed) || self.log_dir_reader.is_failing() {
return None;
}
let Some(result) = self.log_dir_reader.next() else {
self.blocker.block();
continue;
};
return Some(result.map_err(|e| e.into()));
}
}
}