meli/melib/src/mailbox/collection.rs

172 lines
5.5 KiB
Rust
Raw Normal View History

2018-09-06 10:05:35 +00:00
use super::*;
use std::collections::BTreeMap;
2018-09-07 12:36:42 +00:00
use std::fs;
use std::io;
2018-09-06 10:05:35 +00:00
use std::ops::{Deref, DerefMut};
2019-04-04 11:21:52 +00:00
use fnv::FnvHashMap;
2018-09-06 10:05:35 +00:00
/// `Mailbox` represents a folder of mail.
#[derive(Debug, Clone, Default)]
pub struct Collection {
folder: Folder,
2018-09-06 10:05:35 +00:00
pub envelopes: FnvHashMap<EnvelopeHash, Envelope>,
date_index: BTreeMap<UnixTimestamp, EnvelopeHash>,
subject_index: Option<BTreeMap<String, EnvelopeHash>>,
pub threads: Threads,
}
impl Drop for Collection {
fn drop(&mut self) {
let cache_dir =
xdg::BaseDirectories::with_profile("meli", format!("{}_Thread", self.folder.hash()))
.unwrap();
if let Ok(cached) = cache_dir.place_cache_file("threads") {
/* place result in cache directory */
let f = match fs::File::create(cached) {
Ok(f) => f,
Err(e) => {
panic!("{}", e);
}
};
let writer = io::BufWriter::new(f);
bincode::serialize_into(writer, &self.threads).unwrap();
}
}
}
2018-09-06 10:05:35 +00:00
impl Collection {
2018-09-17 04:53:16 +00:00
pub fn new(vec: Vec<Envelope>, folder: &Folder) -> Collection {
2018-09-06 10:05:35 +00:00
let mut envelopes: FnvHashMap<EnvelopeHash, Envelope> =
FnvHashMap::with_capacity_and_hasher(vec.len(), Default::default());
for e in vec {
envelopes.insert(e.hash(), e);
}
let date_index = BTreeMap::new();
let subject_index = None;
2018-10-14 16:49:16 +00:00
/* Scrap caching for now. When a cached threads file is loaded, we must remove/rehash the
* thread nodes that shouldn't exist anymore (e.g. because their file moved from /new to
* /cur, or it was deleted).
*/
let threads = Threads::new(&mut envelopes);
2019-02-11 12:55:29 +00:00
/*let cache_dir =
2019-05-01 16:20:33 +00:00
xdg::BaseDirectories::with_profile("meli", format!("{}_Thread", folder.hash()))
.unwrap();
if let Some(cached) = cache_dir.find_cache_file("threads") {
let reader = io::BufReader::new(fs::File::open(cached).unwrap());
let result: result::Result<Threads, _> = bincode::deserialize_from(reader);
let ret = if let Ok(mut cached_t) = result {
use std::iter::FromIterator;
debug!("loaded cache, our hash set is {:?}\n and the cached one is {:?}", FnvHashSet::from_iter(envelopes.keys().cloned()), cached_t.hash_set);
cached_t.amend(&mut envelopes);
cached_t
} else {
Threads::new(&mut envelopes)
};
ret
} else {
Threads::new(&mut envelopes)
};
*/
2018-10-14 16:49:16 +00:00
2018-09-06 10:05:35 +00:00
Collection {
folder: folder.clone(),
2018-09-06 10:05:35 +00:00
envelopes,
date_index,
subject_index,
threads,
}
}
pub fn len(&self) -> usize {
self.envelopes.len()
}
pub fn is_empty(&self) -> bool {
self.envelopes.is_empty()
}
2018-09-07 07:16:36 +00:00
2018-10-14 16:49:16 +00:00
pub fn remove(&mut self, envelope_hash: EnvelopeHash) {
2019-05-01 16:20:33 +00:00
debug!("DEBUG: Removing {}", envelope_hash);
2018-10-14 16:49:16 +00:00
self.envelopes.remove(&envelope_hash);
self.threads.remove(envelope_hash, &mut self.envelopes);
}
pub fn rename(&mut self, old_hash: EnvelopeHash, new_hash: EnvelopeHash) {
if !self.envelopes.contains_key(&old_hash) {
return;
}
2018-10-14 16:49:16 +00:00
let mut env = self.envelopes.remove(&old_hash).unwrap();
env.set_hash(new_hash);
self.envelopes.insert(new_hash, env);
{
2019-04-04 11:24:05 +00:00
if self
.threads
.update_envelope(old_hash, new_hash, &self.envelopes)
.is_ok()
{
2018-10-14 16:49:16 +00:00
return;
}
}
/* envelope is not in threads, so insert it */
let env = self.envelopes.entry(new_hash).or_default() as *mut Envelope;
unsafe {
self.threads.insert(&mut (*env), &self.envelopes);
}
}
pub fn update_envelope(&mut self, old_hash: EnvelopeHash, envelope: Envelope) {
self.envelopes.remove(&old_hash);
let new_hash = envelope.hash();
self.envelopes.insert(new_hash, envelope);
{
2019-04-04 11:24:05 +00:00
if self
.threads
.update_envelope(old_hash, new_hash, &self.envelopes)
.is_ok()
{
2018-10-14 16:49:16 +00:00
return;
}
}
/* envelope is not in threads, so insert it */
let env = self.envelopes.entry(new_hash).or_default() as *mut Envelope;
unsafe {
self.threads.insert(&mut (*env), &self.envelopes);
}
}
pub fn insert(&mut self, envelope: Envelope) {
2018-09-17 04:53:16 +00:00
let hash = envelope.hash();
2019-05-01 16:20:33 +00:00
debug!("DEBUG: Inserting hash {} in {}", hash, self.folder.name());
2018-09-07 07:16:36 +00:00
self.envelopes.insert(hash, envelope);
let env = self.envelopes.entry(hash).or_default() as *mut Envelope;
unsafe {
self.threads.insert(&mut (*env), &self.envelopes);
}
2018-09-07 07:16:36 +00:00
}
2018-10-14 16:49:16 +00:00
pub(crate) fn insert_reply(&mut self, _envelope: &Envelope) {
return;
/*
2019-05-01 16:20:33 +00:00
//self.insert(envelope);
debug!("insert_reply in collections");
self.threads.insert_reply(envelope, &mut self.envelopes);
*/
}
2018-09-06 10:05:35 +00:00
}
impl Deref for Collection {
type Target = FnvHashMap<EnvelopeHash, Envelope>;
fn deref(&self) -> &FnvHashMap<EnvelopeHash, Envelope> {
&self.envelopes
}
}
impl DerefMut for Collection {
fn deref_mut(&mut self) -> &mut FnvHashMap<EnvelopeHash, Envelope> {
&mut self.envelopes
}
}