2016-10-01 15:35:06 +00:00
|
|
|
//
|
|
|
|
// imag - the personal information management suite for the commandline
|
|
|
|
// Copyright (C) 2015, 2016 Matthias Beyer <mail@beyermatthias.de> and contributors
|
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; version
|
|
|
|
// 2.1 of the License.
|
|
|
|
//
|
|
|
|
// This library 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
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
//
|
|
|
|
|
2016-04-14 16:24:33 +00:00
|
|
|
use toml::Value;
|
|
|
|
|
|
|
|
use fs2::FileExt;
|
2016-09-06 12:27:08 +00:00
|
|
|
use std::fs::File;
|
2016-04-14 16:24:33 +00:00
|
|
|
|
|
|
|
use libimagstore::hook::Hook;
|
|
|
|
use libimagstore::hook::accessor::HookDataAccessor as HDA;
|
|
|
|
use libimagstore::hook::accessor::HookDataAccessorProvider;
|
|
|
|
use libimagstore::hook::accessor::StoreIdAccessor;
|
|
|
|
use libimagstore::hook::accessor::MutableHookDataAccessor;
|
|
|
|
use libimagstore::hook::accessor::NonMutableHookDataAccessor;
|
|
|
|
use libimagstore::hook::result::HookResult;
|
2016-09-06 12:24:57 +00:00
|
|
|
use libimagstore::hook::error::{HookError, HookErrorKind};
|
2016-04-14 16:24:33 +00:00
|
|
|
use libimagstore::storeid::StoreId;
|
|
|
|
use libimagstore::store::FileLockEntry;
|
|
|
|
use libimagstore::store::Entry;
|
|
|
|
|
2016-09-06 12:24:57 +00:00
|
|
|
mod error {
|
|
|
|
generate_error_imports!();
|
|
|
|
generate_error_types!(FlockError, FlockErrorKind,
|
|
|
|
IOError => "IO Error",
|
|
|
|
StoreIdPathBufConvertError => "Error while converting StoreId to PathBuf",
|
|
|
|
FileOpenError => "Error on File::open()",
|
|
|
|
LockError => "Error while lock()ing",
|
|
|
|
UnlockError => "Error while unlock()ing"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
use self::error::FlockError as FE;
|
|
|
|
use self::error::FlockErrorKind as FEK;
|
|
|
|
use self::error::MapErrInto;
|
|
|
|
|
2016-04-14 16:24:33 +00:00
|
|
|
trait EntryFlock {
|
2016-09-06 12:24:57 +00:00
|
|
|
fn lock(&self) -> Result<(), FE>;
|
|
|
|
fn unlock(&self) -> Result<(), FE>;
|
2016-04-14 16:24:33 +00:00
|
|
|
}
|
|
|
|
|
2016-09-06 12:27:08 +00:00
|
|
|
fn open_file(id: StoreId) -> Result<File, FE> {
|
|
|
|
id.into_pathbuf()
|
|
|
|
.map_err_into(FEK::StoreIdPathBufConvertError)
|
|
|
|
.and_then(|loc| {
|
|
|
|
File::open(loc)
|
|
|
|
.map_err_into(FEK::FileOpenError)
|
|
|
|
.map_err_into(FEK::IOError)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2016-04-14 16:24:33 +00:00
|
|
|
impl EntryFlock for Entry {
|
|
|
|
|
2016-09-06 12:24:57 +00:00
|
|
|
fn lock(&self) -> Result<(), FE> {
|
2016-09-06 12:27:08 +00:00
|
|
|
open_file(self.get_location().clone())
|
2016-09-06 12:24:57 +00:00
|
|
|
.and_then(|file| {
|
|
|
|
file.lock_exclusive()
|
|
|
|
.map_err_into(FEK::LockError)
|
|
|
|
.map_err_into(FEK::IOError)
|
|
|
|
})
|
2016-04-14 16:24:33 +00:00
|
|
|
}
|
|
|
|
|
2016-09-06 12:24:57 +00:00
|
|
|
fn unlock(&self) -> Result<(), FE> {
|
2016-09-06 12:27:08 +00:00
|
|
|
open_file(self.get_location().clone())
|
2016-09-06 12:24:57 +00:00
|
|
|
.and_then(|file| {
|
|
|
|
file.unlock()
|
|
|
|
.map_err_into(FEK::UnlockError)
|
|
|
|
.map_err_into(FEK::LockError)
|
|
|
|
.map_err_into(FEK::IOError)
|
|
|
|
})
|
2016-04-14 16:24:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(PartialEq, Eq, Debug, Clone)]
|
|
|
|
pub enum Action {
|
|
|
|
Lock,
|
|
|
|
Unlock
|
|
|
|
}
|
|
|
|
|
|
|
|
fn action_to_str(a: &Action) -> &'static str {
|
2016-05-03 21:10:32 +00:00
|
|
|
match *a {
|
|
|
|
Action::Lock => "lock",
|
|
|
|
Action::Unlock => "unlock",
|
2016-04-14 16:24:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct FlockUpdateHook {
|
|
|
|
action: Action,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl FlockUpdateHook {
|
|
|
|
|
2016-08-25 17:26:59 +00:00
|
|
|
pub fn new(action: Action) -> FlockUpdateHook {
|
2016-04-14 16:24:33 +00:00
|
|
|
FlockUpdateHook {
|
|
|
|
action: action,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Hook for FlockUpdateHook {
|
|
|
|
|
|
|
|
fn name(&self) -> &'static str {
|
|
|
|
"stdhook_flock_update"
|
|
|
|
}
|
|
|
|
|
|
|
|
fn set_config(&mut self, _: &Value) {
|
|
|
|
() // We are not configurable here.
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
impl HookDataAccessorProvider for FlockUpdateHook {
|
|
|
|
|
|
|
|
fn accessor(&self) -> HDA {
|
|
|
|
HDA::StoreIdAccess(self)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
impl StoreIdAccessor for FlockUpdateHook {
|
|
|
|
|
|
|
|
fn access(&self, id: &StoreId) -> HookResult<()> {
|
|
|
|
debug!("[FLOCK HOOK][{}] {:?}", action_to_str(&self.action), id);
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
impl MutableHookDataAccessor for FlockUpdateHook {
|
|
|
|
|
|
|
|
fn access_mut(&self, fle: &mut FileLockEntry) -> HookResult<()> {
|
|
|
|
debug!("[FLOCK HOOK][{}] {:?}", action_to_str(&self.action), fle.get_location());
|
2016-08-25 17:26:59 +00:00
|
|
|
fle.lock()
|
2016-04-14 16:24:33 +00:00
|
|
|
.map_err(|e| HookError::new(HookErrorKind::HookExecutionError, Some(Box::new(e))))
|
|
|
|
.map(|_| ())
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
impl NonMutableHookDataAccessor for FlockUpdateHook {
|
|
|
|
|
|
|
|
fn access(&self, fle: &FileLockEntry) -> HookResult<()> {
|
|
|
|
debug!("[FLOCK HOOK][{}] {:?}", action_to_str(&self.action), fle.get_location());
|
2016-08-25 17:26:59 +00:00
|
|
|
fle.unlock()
|
2016-04-14 16:24:33 +00:00
|
|
|
.map_err(|e| HookError::new(HookErrorKind::HookExecutionError, Some(Box::new(e))))
|
|
|
|
.map(|_| ())
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|