imag/lib/domain/libimagcontact/src/store.rs

122 lines
4 KiB
Rust
Raw Normal View History

2017-09-23 15:51:35 +00:00
//
// imag - the personal information management suite for the commandline
// Copyright (C) 2015-2018 Matthias Beyer <mail@beyermatthias.de> and contributors
2017-09-23 15:51:35 +00:00
//
// 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
//
use std::path::Path;
2017-09-23 15:51:35 +00:00
use std::path::PathBuf;
use std::result::Result as RResult;
2017-09-23 15:51:35 +00:00
2017-09-23 18:18:20 +00:00
use vobject::parse_component;
2017-09-23 15:51:35 +00:00
use libimagstore::store::Store;
use libimagstore::store::FileLockEntry;
use libimagstore::storeid::StoreIdIterator;
2017-09-23 18:18:20 +00:00
use libimagentryref::refstore::RefStore;
use libimagentryref::refstore::UniqueRefPathGenerator;
use libimagentryutil::isa::Is;
2017-09-23 15:51:35 +00:00
use contact::IsContact;
use error::ContactError as CE;
use error::ContactErrorKind as CEK;
2017-09-23 15:51:35 +00:00
use error::Result;
2017-09-23 18:18:20 +00:00
use util;
2017-09-23 15:51:35 +00:00
2018-02-14 11:35:42 +00:00
pub struct UniqueContactPathGenerator;
impl UniqueRefPathGenerator for UniqueContactPathGenerator {
type Error = CE;
/// The collection the `StoreId` should be created for
fn collection() -> &'static str {
"contact"
}
/// A function which should generate a unique string for a Path
fn unique_hash<A: AsRef<Path>>(path: A) -> RResult<String, Self::Error> {
use vobject::vcard::Vcard;
debug!("Generating unique hash for path: {:?}", path.as_ref());
util::read_to_string(path.as_ref())
.and_then(|s| Vcard::build(&s).map_err(CE::from))
.and_then(|card| {
card.uid()
.map(|u| u.raw().clone())
.ok_or_else(|| {
let s = path.as_ref().to_str().unwrap_or("Unknown path");
CEK::UidMissing(String::from(s)).into()
})
})
}
}
pub trait ContactStore<'a> : RefStore<'a> {
2017-09-23 15:51:35 +00:00
// creating
fn create_from_path(&'a self, p: &PathBuf) -> Result<FileLockEntry<'a>>;
2017-09-23 18:18:20 +00:00
/// Create contact ref from buffer
///
/// Needs the `p` argument as we're finally creating a reference by path, the buffer is only for
/// collecting metadata.
fn create_from_buf<P: AsRef<Path>>(&'a self, p: P, buf: &String) -> Result<FileLockEntry<'a>>;
2017-09-23 18:18:20 +00:00
2017-09-23 15:51:35 +00:00
// getting
fn all_contacts(&'a self) -> Result<StoreIdIterator>;
2017-09-23 15:51:35 +00:00
}
/// The extension for the Store to work with contacts
///
/// The contact functionality is implemented by using the `libimagentryref` library, so basically
/// we only reference vcard files from outside the store.
///
/// Because of this, we do not have an own store collection `/contacts` or something like that, but
/// must stress the `libimagentryref` API for everything.
2017-09-23 15:51:35 +00:00
impl<'a> ContactStore<'a> for Store {
fn create_from_path(&'a self, p: &PathBuf) -> Result<FileLockEntry<'a>> {
2017-09-23 18:18:20 +00:00
util::read_to_string(p).and_then(|buf| self.create_from_buf(p, &buf))
}
/// Create contact ref from buffer
fn create_from_buf<P: AsRef<Path>>(&'a self, p: P, buf: &String) -> Result<FileLockEntry<'a>> {
2017-09-23 18:18:20 +00:00
let component = parse_component(&buf)?;
debug!("Parsed: {:?}", component);
RefStore::create_ref::<UniqueContactPathGenerator, P>(self, p)
2017-09-23 18:18:20 +00:00
.map_err(From::from)
.and_then(|mut entry| {
entry.set_isflag::<IsContact>()
2017-09-23 18:18:20 +00:00
.map_err(From::from)
.map(|_| entry)
})
2017-09-23 15:51:35 +00:00
}
fn all_contacts(&'a self) -> Result<StoreIdIterator> {
2018-03-12 11:52:24 +00:00
let iter = self
.entries()?
.without_store()
.filter(|id| id.is_in_collection(&["contact"]));
Ok(StoreIdIterator::new(Box::new(iter)))
2017-09-23 15:51:35 +00:00
}
}