// This file is part of Moonfire NVR, a security camera network video recorder. // Copyright (C) 2018 Scott Lamb // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // In addition, as a special exception, the copyright holders give // permission to link the code of portions of this program with the // OpenSSL library under certain conditions as described in each // individual source file, and distribute linked combinations including // the two. // // You must obey the GNU General Public License in all respects for all // of the code used other than OpenSSL. If you modify file(s) with this // exception, you may extend this exception to your version of the // file(s), but you are not obligated to do so. If you do not wish to do // so, delete this exception statement from your version. If you delete // this exception statement from all source files in the program, then // also delete it here. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . //! Sample file directory management. //! //! This includes opening files for serving, rotating away old files, and saving new files. use crate::coding; use crate::db::CompositeId; use crate::schema; use cstr::*; use failure::{Error, Fail, bail, format_err}; use log::warn; use protobuf::Message; use nix::{NixPath, fcntl::{FlockArg, OFlag}, sys::stat::Mode}; use nix::sys::statvfs::Statvfs; use std::ffi::{CStr, CString}; use std::fs; use std::io::{Read, Write}; use std::os::unix::io::{AsRawFd, RawFd}; use std::sync::Arc; /// The fixed length of a directory's `meta` file. /// /// See DirMeta comments within proto/schema.proto for more explanation. const FIXED_DIR_META_LEN: usize = 512; /// A sample file directory. Typically one per physical disk drive. /// /// If the directory is used for writing, the `start_syncer` function should be called to start /// a background thread. This thread manages deleting files and writing new files. It synces the /// directory and commits these operations to the database in the correct order to maintain the /// invariants described in `design/schema.md`. #[derive(Debug)] pub struct SampleFileDir { /// The open file descriptor for the directory. The worker uses it to create files and sync the /// directory. Other threads use it to open sample files for reading during video serving. pub(crate) fd: Fd, } pub(crate) struct CompositeIdPath([u8; 17]); impl CompositeIdPath { pub(crate) fn from(id: CompositeId) -> Self { let mut buf = [0u8; 17]; write!(&mut buf[..16], "{:016x}", id.0).expect("can't format id to pathname buf"); CompositeIdPath(buf) } } impl NixPath for CompositeIdPath { fn len(&self) -> usize { 16 } fn with_nix_path(&self, f: F) -> Result where F: FnOnce(&CStr) -> T { let p = CStr::from_bytes_with_nul(&self.0[..]).expect("no interior nuls"); Ok(f(p)) } } /// A file descriptor associated with a directory (not necessarily the sample file dir). #[derive(Debug)] pub struct Fd(std::os::unix::io::RawFd); impl std::os::unix::io::AsRawFd for Fd { fn as_raw_fd(&self) -> std::os::unix::io::RawFd { self.0 } } impl Drop for Fd { fn drop(&mut self) { if let Err(e) = nix::unistd::close(self.0) { warn!("Unable to close sample file dir: {}", e); } } } impl Fd { /// Opens the given path as a directory. pub fn open(path: &str, mkdir: bool) -> Result { let cstring = CString::new(path).map_err(|_| nix::Error::InvalidPath)?; if mkdir { match nix::unistd::mkdir(cstring.as_c_str(), nix::sys::stat::Mode::S_IRWXU) { Ok(()) | Err(nix::Error::Sys(nix::errno::Errno::EEXIST)) => {}, Err(e) => return Err(e), } } let fd = nix::fcntl::open(cstring.as_c_str(), OFlag::O_DIRECTORY | OFlag::O_RDONLY, Mode::empty())?; Ok(Fd(fd)) } pub(crate) fn sync(&self) -> Result<(), nix::Error> { nix::unistd::fsync(self.0) } /// Locks the directory with the specified `flock` operation. pub fn lock(&self, arg: FlockArg) -> Result<(), nix::Error> { nix::fcntl::flock(self.0, arg) } pub fn statfs(&self) -> Result { nix::sys::statvfs::fstatvfs(self) } } /// Reads `dir`'s metadata. If none is found, returns an empty proto. pub(crate) fn read_meta(dir: &Fd) -> Result { let mut meta = schema::DirMeta::default(); let mut f = match crate::fs::openat(dir.0, cstr!("meta"), OFlag::O_RDONLY, Mode::empty()) { Err(e) => { if e == nix::Error::Sys(nix::errno::Errno::ENOENT) { return Ok(meta); } return Err(e.into()); }, Ok(f) => f, }; let mut data = Vec::new(); f.read_to_end(&mut data)?; let (len, pos) = coding::decode_varint32(&data, 0) .map_err(|_| format_err!("Unable to decode varint length in meta file"))?; if data.len() != FIXED_DIR_META_LEN || len as usize + pos > FIXED_DIR_META_LEN { bail!("Expected a {}-byte file with a varint length of a DirMeta message; got \ a {}-byte file with length {}", FIXED_DIR_META_LEN, data.len(), len); } let data = &data[pos..pos+len as usize]; let mut s = protobuf::CodedInputStream::from_bytes(&data); meta.merge_from(&mut s).map_err(|e| e.context("Unable to parse metadata proto: {}"))?; Ok(meta) } /// Write `dir`'s metadata, clobbering existing data. pub(crate) fn write_meta(dirfd: RawFd, meta: &schema::DirMeta) -> Result<(), Error> { let mut data = meta.write_length_delimited_to_bytes().expect("proto3->vec is infallible"); if data.len() > FIXED_DIR_META_LEN { bail!("Length-delimited DirMeta message requires {} bytes, over limit of {}", data.len(), FIXED_DIR_META_LEN); } data.resize(FIXED_DIR_META_LEN, 0); // pad to required length. let mut f = crate::fs::openat(dirfd, cstr!("meta"), OFlag::O_CREAT | OFlag::O_WRONLY, Mode::S_IRUSR | Mode::S_IWUSR)?; let stat = f.metadata()?; if stat.len() == 0 { // Need to sync not only the data but also the file metadata and dirent. f.write_all(&data)?; f.sync_all()?; nix::unistd::fsync(dirfd)?; } else if stat.len() == FIXED_DIR_META_LEN as u64 { // Just syncing the data will suffice; existing metadata and dirent are fine. f.write_all(&data)?; f.sync_data()?; } else { bail!("Existing meta file is {}-byte; expected {}", stat.len(), FIXED_DIR_META_LEN); } Ok(()) } impl SampleFileDir { /// Opens the directory using the given metadata. /// /// `db_meta.in_progress_open` should be filled if the directory should be opened in read/write /// mode; absent in read-only mode. pub fn open(path: &str, db_meta: &schema::DirMeta) -> Result, Error> { let read_write = db_meta.in_progress_open.is_some(); let s = SampleFileDir::open_self(path, false)?; s.fd.lock(if read_write { FlockArg::LockExclusiveNonblock } else { FlockArg::LockSharedNonblock })?; let dir_meta = read_meta(&s.fd)?; if !SampleFileDir::consistent(db_meta, &dir_meta) { let serialized = db_meta.write_length_delimited_to_bytes().expect("proto3->vec is infallible"); bail!("metadata mismatch.\ndb: {:#?}\ndir: {:#?}\nserialized db: {:#?}", db_meta, &dir_meta, &serialized); } if db_meta.in_progress_open.is_some() { s.write_meta(db_meta)?; } Ok(s) } /// Returns true if the existing directory and database metadata are consistent; the directory /// is then openable. pub(crate) fn consistent(db_meta: &schema::DirMeta, dir_meta: &schema::DirMeta) -> bool { if dir_meta.db_uuid != db_meta.db_uuid { return false; } if dir_meta.dir_uuid != db_meta.dir_uuid { return false; } if db_meta.last_complete_open.is_some() && (db_meta.last_complete_open != dir_meta.last_complete_open && db_meta.last_complete_open != dir_meta.in_progress_open) { return false; } if db_meta.last_complete_open.is_none() && dir_meta.last_complete_open.is_some() { return false; } true } pub(crate) fn create(path: &str, db_meta: &schema::DirMeta) -> Result, Error> { let s = SampleFileDir::open_self(path, true)?; s.fd.lock(FlockArg::LockExclusiveNonblock)?; let old_meta = read_meta(&s.fd)?; // Verify metadata. We only care that it hasn't been completely opened. // Partial opening by this or another database is fine; we won't overwrite anything. if old_meta.last_complete_open.is_some() { bail!("Can't create dir at path {}: is already in use:\n{:?}", path, old_meta); } if !s.is_empty()? { bail!("Can't create dir at path {} with existing files", path); } s.write_meta(db_meta)?; Ok(s) } pub(crate) fn opendir(&self) -> Result { nix::dir::Dir::openat(self.fd.as_raw_fd(), ".", OFlag::O_DIRECTORY | OFlag::O_RDONLY, Mode::empty()) } /// Determines if the directory is empty, aside form metadata. pub(crate) fn is_empty(&self) -> Result { let mut dir = self.opendir()?; for e in dir.iter() { let e = e?; match e.file_name().to_bytes() { b"." | b".." => continue, b"meta" => continue, // existing metadata is fine. _ => return Ok(false), } } Ok(true) } fn open_self(path: &str, create: bool) -> Result, Error> { let fd = Fd::open(path, create) .map_err(|e| format_err!("unable to open sample file dir {}: {}", path, e))?; Ok(Arc::new(SampleFileDir { fd, })) } /// Opens the given sample file for reading. pub fn open_file(&self, composite_id: CompositeId) -> Result { let p = CompositeIdPath::from(composite_id); crate::fs::openat(self.fd.0, &p, OFlag::O_RDONLY, Mode::empty()) } pub fn create_file(&self, composite_id: CompositeId) -> Result { let p = CompositeIdPath::from(composite_id); crate::fs::openat(self.fd.0, &p, OFlag::O_WRONLY | OFlag::O_EXCL | OFlag::O_CREAT, Mode::S_IRUSR | Mode::S_IWUSR) } pub(crate) fn write_meta(&self, meta: &schema::DirMeta) -> Result<(), Error> { write_meta(self.fd.0, meta) } pub fn statfs(&self) -> Result { self.fd.statfs() } /// Unlinks the given sample file within this directory. pub(crate) fn unlink_file(&self, id: CompositeId) -> Result<(), nix::Error> { let p = CompositeIdPath::from(id); nix::unistd::unlinkat(Some(self.fd.0), &p, nix::unistd::UnlinkatFlags::NoRemoveDir) } /// Syncs the directory itself. pub(crate) fn sync(&self) -> Result<(), nix::Error> { self.fd.sync() } } /// Parses a composite id filename. /// /// These are exactly 16 bytes, lowercase hex. pub(crate) fn parse_id(id: &[u8]) -> Result { if id.len() != 16 { return Err(()); } let mut v: u64 = 0; for i in 0..16 { v = (v << 4) | match id[i] { b @ b'0'..=b'9' => b - b'0', b @ b'a'..=b'f' => b - b'a' + 10, _ => return Err(()), } as u64; } Ok(CompositeId(v as i64)) } #[cfg(test)] mod tests { use protobuf::prelude::MessageField; use super::*; #[test] fn parse_id() { use super::parse_id; assert_eq!(parse_id(b"0000000000000000").unwrap().0, 0); assert_eq!(parse_id(b"0000000100000002").unwrap().0, 0x0000000100000002); parse_id(b"").unwrap_err(); parse_id(b"meta").unwrap_err(); parse_id(b"0").unwrap_err(); parse_id(b"000000010000000x").unwrap_err(); } /// Ensures that a DirMeta with all fields filled fits within the maximum size. #[test] fn max_len_meta() { let mut meta = schema::DirMeta::new(); let fake_uuid = &[0u8; 16][..]; meta.db_uuid.extend_from_slice(fake_uuid); meta.dir_uuid.extend_from_slice(fake_uuid); meta.last_complete_open.mut_message().id = u32::max_value(); meta.last_complete_open.mut_message().id = u32::max_value(); meta.in_progress_open.mut_message().uuid.extend_from_slice(fake_uuid); meta.in_progress_open.mut_message().uuid.extend_from_slice(fake_uuid); let data = meta.write_length_delimited_to_bytes().expect("proto3->vec is infallible"); assert!(data.len() <= FIXED_DIR_META_LEN, "{} vs {}", data.len(), FIXED_DIR_META_LEN); } }