2017-09-14 18:08:14 +03:00
|
|
|
/*
|
|
|
|
* meli - accounts module.
|
|
|
|
*
|
|
|
|
* Copyright 2017 Manos Pitsidianakis
|
|
|
|
*
|
|
|
|
* This file is part of meli.
|
|
|
|
*
|
|
|
|
* meli 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.
|
|
|
|
*
|
|
|
|
* meli 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 meli. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2018-08-06 22:20:34 +03:00
|
|
|
/*!
|
|
|
|
* Account management from user configuration.
|
|
|
|
*/
|
|
|
|
|
2018-08-07 15:01:15 +03:00
|
|
|
use async::*;
|
2018-08-19 14:08:20 +03:00
|
|
|
use conf::AccountConf;
|
2018-09-05 16:08:11 +03:00
|
|
|
use mailbox::backends::{
|
2018-09-07 15:36:42 +03:00
|
|
|
Backends, Folder, MailBackend, NotifyFn, RefreshEvent, RefreshEventConsumer, RefreshEventKind,
|
2018-09-05 16:08:11 +03:00
|
|
|
};
|
2018-07-27 21:37:56 +03:00
|
|
|
use mailbox::*;
|
2018-08-23 15:36:52 +03:00
|
|
|
use melib::error::Result;
|
2019-02-15 09:06:42 +02:00
|
|
|
use melib::AddressBook;
|
|
|
|
|
|
|
|
use std::fs;
|
|
|
|
use std::io;
|
2018-09-23 19:55:29 +03:00
|
|
|
use std::mem;
|
2017-09-16 15:05:28 +03:00
|
|
|
use std::ops::{Index, IndexMut};
|
2018-08-03 13:46:08 +03:00
|
|
|
use std::result;
|
2018-09-07 15:36:42 +03:00
|
|
|
use std::sync::Arc;
|
2018-09-05 16:08:11 +03:00
|
|
|
use types::UIEventType::{self, Notification};
|
2018-08-03 13:46:08 +03:00
|
|
|
|
2018-09-17 07:53:16 +03:00
|
|
|
pub type Worker = Option<Async<Result<Mailbox>>>;
|
2018-07-11 18:58:57 +03:00
|
|
|
|
2018-10-14 19:49:16 +03:00
|
|
|
macro_rules! mailbox {
|
|
|
|
($idx:expr, $folders:expr) => {
|
|
|
|
$folders[$idx].as_mut().unwrap().as_mut().unwrap()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-09-14 18:08:14 +03:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Account {
|
|
|
|
name: String,
|
|
|
|
folders: Vec<Option<Result<Mailbox>>>,
|
2019-02-15 09:06:42 +02:00
|
|
|
sent_folder: Option<usize>,
|
2017-09-14 18:08:14 +03:00
|
|
|
|
2019-02-15 09:06:42 +02:00
|
|
|
pub(crate) address_book: AddressBook,
|
2018-08-03 13:46:08 +03:00
|
|
|
|
2019-02-15 09:06:42 +02:00
|
|
|
pub(crate) workers: Vec<Worker>,
|
2017-09-14 18:08:14 +03:00
|
|
|
|
2019-02-15 09:06:42 +02:00
|
|
|
|
|
|
|
pub(crate) settings: AccountConf,
|
|
|
|
pub(crate) runtime_settings: AccountConf,
|
|
|
|
pub(crate) backend: Box<MailBackend>,
|
2018-09-07 15:36:42 +03:00
|
|
|
notify_fn: Arc<NotifyFn>,
|
2017-09-14 18:08:14 +03:00
|
|
|
}
|
|
|
|
|
2019-02-15 09:06:42 +02:00
|
|
|
impl Drop for Account {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
let data_dir =
|
|
|
|
xdg::BaseDirectories::with_profile("meli", &self.name)
|
|
|
|
.unwrap();
|
|
|
|
if let Ok(data) = data_dir.place_data_file("addressbook") {
|
|
|
|
/* place result in cache directory */
|
|
|
|
let f = match fs::File::create(data) {
|
|
|
|
Ok(f) => f,
|
|
|
|
Err(e) => {
|
|
|
|
panic!("{}", e);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
let writer = io::BufWriter::new(f);
|
|
|
|
bincode::serialize_into(writer, &self.address_book).unwrap();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-14 18:08:14 +03:00
|
|
|
impl Account {
|
2018-09-07 15:36:42 +03:00
|
|
|
pub fn new(name: String, settings: AccountConf, map: &Backends, notify_fn: NotifyFn) -> Self {
|
2018-08-19 14:08:20 +03:00
|
|
|
let mut backend = map.get(settings.account().format())(settings.account());
|
2018-08-11 18:00:21 +03:00
|
|
|
let ref_folders: Vec<Folder> = backend.folders();
|
|
|
|
let mut folders: Vec<Option<Result<Mailbox>>> = Vec::with_capacity(ref_folders.len());
|
|
|
|
let mut workers: Vec<Worker> = Vec::new();
|
|
|
|
let sent_folder = ref_folders
|
2017-09-16 15:05:28 +03:00
|
|
|
.iter()
|
2018-08-19 14:08:20 +03:00
|
|
|
.position(|x: &Folder| x.name() == settings.account().sent_folder);
|
2018-09-07 15:36:42 +03:00
|
|
|
let notify_fn = Arc::new(notify_fn);
|
2018-08-11 18:00:21 +03:00
|
|
|
for f in ref_folders {
|
2017-09-16 15:05:28 +03:00
|
|
|
folders.push(None);
|
2018-10-14 19:49:16 +03:00
|
|
|
workers.push(Account::new_worker(f, &mut backend, notify_fn.clone()));
|
2017-09-16 15:05:28 +03:00
|
|
|
}
|
2019-02-15 09:06:42 +02:00
|
|
|
let data_dir =
|
|
|
|
xdg::BaseDirectories::with_profile("meli", &name)
|
|
|
|
.unwrap();
|
|
|
|
let address_book = if let Ok(data) = data_dir.place_data_file("addressbook") {
|
|
|
|
if data.exists() {
|
|
|
|
let reader = io::BufReader::new(fs::File::open(data).unwrap());
|
|
|
|
let result: result::Result<AddressBook, _> = bincode::deserialize_from(reader);
|
|
|
|
if let Ok(mut data_t) = result {
|
|
|
|
data_t
|
|
|
|
} else {
|
|
|
|
AddressBook::new(name.clone())
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
AddressBook::new(name.clone())
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
AddressBook::new(name.clone())
|
|
|
|
};
|
|
|
|
|
2017-09-14 18:08:14 +03:00
|
|
|
Account {
|
2018-08-23 15:36:52 +03:00
|
|
|
name,
|
|
|
|
folders,
|
2019-02-15 09:06:42 +02:00
|
|
|
address_book,
|
2018-08-23 15:36:52 +03:00
|
|
|
sent_folder,
|
2019-02-15 09:06:42 +02:00
|
|
|
workers,
|
2018-07-22 14:05:41 +03:00
|
|
|
settings: settings.clone(),
|
|
|
|
runtime_settings: settings,
|
2018-08-23 15:36:52 +03:00
|
|
|
backend,
|
2018-09-07 15:36:42 +03:00
|
|
|
notify_fn,
|
2017-09-14 18:08:14 +03:00
|
|
|
}
|
|
|
|
}
|
2018-09-17 07:53:16 +03:00
|
|
|
fn new_worker(
|
|
|
|
folder: Folder,
|
|
|
|
backend: &mut Box<MailBackend>,
|
|
|
|
notify_fn: Arc<NotifyFn>,
|
|
|
|
) -> Worker {
|
2018-10-14 19:49:16 +03:00
|
|
|
let mailbox_handle = backend.get(&folder);
|
2018-09-17 07:53:16 +03:00
|
|
|
let mut builder = AsyncBuilder::new();
|
|
|
|
let tx = builder.tx();
|
2018-10-14 19:49:16 +03:00
|
|
|
Some(builder.build(Box::new(move || {
|
|
|
|
let mut handle = mailbox_handle.clone();
|
|
|
|
let folder = folder.clone();
|
|
|
|
let work = handle.work().unwrap();
|
|
|
|
work.compute();
|
|
|
|
handle.join();
|
|
|
|
let envelopes = handle.extract();
|
|
|
|
let ret = Mailbox::new(folder, envelopes);
|
|
|
|
tx.send(AsyncStatus::Payload(ret));
|
|
|
|
notify_fn.notify();
|
|
|
|
})))
|
2018-09-17 07:53:16 +03:00
|
|
|
}
|
2018-09-05 16:08:11 +03:00
|
|
|
pub fn reload(&mut self, event: RefreshEvent, idx: usize) -> Option<UIEventType> {
|
|
|
|
let kind = event.kind();
|
2018-09-23 19:55:29 +03:00
|
|
|
{
|
2018-10-14 19:49:16 +03:00
|
|
|
//let mailbox: &mut Mailbox = self.folders[idx].as_mut().unwrap().as_mut().unwrap();
|
2018-09-23 19:55:29 +03:00
|
|
|
match kind {
|
|
|
|
RefreshEventKind::Update(old_hash, envelope) => {
|
2018-10-14 19:49:16 +03:00
|
|
|
mailbox!(idx, self.folders).update(old_hash, *envelope);
|
|
|
|
}
|
|
|
|
RefreshEventKind::Rename(old_hash, new_hash) => {
|
|
|
|
mailbox!(idx, self.folders).rename(old_hash, new_hash);
|
2018-09-23 19:55:29 +03:00
|
|
|
}
|
|
|
|
RefreshEventKind::Create(envelope) => {
|
2018-10-14 19:49:16 +03:00
|
|
|
eprintln!("create {}", envelope.hash());
|
|
|
|
let env: &Envelope = mailbox!(idx, self.folders).insert(*envelope);
|
2018-09-23 19:55:29 +03:00
|
|
|
let ref_folders: Vec<Folder> = self.backend.folders();
|
|
|
|
return Some(Notification(
|
|
|
|
Some("new mail".into()),
|
|
|
|
format!(
|
2018-10-14 19:49:16 +03:00
|
|
|
"{:<15}:\nSubject: {:<15}\nFrom: {:<15}",
|
2018-09-23 19:55:29 +03:00
|
|
|
ref_folders[idx].name(),
|
|
|
|
env.subject(),
|
2018-10-14 19:49:16 +03:00
|
|
|
env.field_from_to_string(),
|
2018-09-23 19:55:29 +03:00
|
|
|
),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
RefreshEventKind::Remove(envelope_hash) => {
|
2018-10-14 19:49:16 +03:00
|
|
|
mailbox!(idx, self.folders).remove(envelope_hash);
|
2018-09-23 19:55:29 +03:00
|
|
|
}
|
|
|
|
RefreshEventKind::Rescan => {
|
|
|
|
let ref_folders: Vec<Folder> = self.backend.folders();
|
|
|
|
let handle = Account::new_worker(
|
|
|
|
ref_folders[idx].clone(),
|
|
|
|
&mut self.backend,
|
|
|
|
self.notify_fn.clone(),
|
|
|
|
);
|
|
|
|
self.workers[idx] = handle;
|
|
|
|
}
|
2018-09-05 16:08:11 +03:00
|
|
|
}
|
|
|
|
}
|
2018-09-12 15:10:19 +03:00
|
|
|
None
|
2018-08-11 19:19:30 +03:00
|
|
|
}
|
2017-09-28 18:06:35 +03:00
|
|
|
pub fn watch(&self, r: RefreshEventConsumer) -> () {
|
2018-08-11 18:00:21 +03:00
|
|
|
self.backend.watch(r).unwrap();
|
2017-09-28 18:06:35 +03:00
|
|
|
}
|
2018-07-11 18:58:57 +03:00
|
|
|
/* This doesn't represent the number of correctly parsed mailboxes though */
|
|
|
|
pub fn len(&self) -> usize {
|
|
|
|
self.folders.len()
|
|
|
|
}
|
2018-08-23 15:36:52 +03:00
|
|
|
pub fn is_empty(&self) -> bool {
|
|
|
|
self.folders.is_empty()
|
|
|
|
}
|
2018-07-11 23:12:25 +03:00
|
|
|
pub fn list_folders(&self) -> Vec<Folder> {
|
2018-08-19 14:54:32 +03:00
|
|
|
let mut folders = self.backend.folders();
|
2018-09-05 08:22:10 +03:00
|
|
|
if let Some(folder_confs) = self.settings.conf().folders() {
|
2018-08-23 14:39:54 +03:00
|
|
|
//eprintln!("folder renames: {:?}", folder_renames);
|
2018-08-19 14:54:32 +03:00
|
|
|
for f in &mut folders {
|
2018-09-05 08:22:10 +03:00
|
|
|
if let Some(r) = folder_confs.get(&f.name().to_ascii_lowercase()) {
|
|
|
|
if let Some(rename) = r.rename() {
|
|
|
|
f.change_name(rename);
|
|
|
|
}
|
2018-08-19 14:54:32 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
folders
|
2018-07-11 18:58:57 +03:00
|
|
|
}
|
2018-07-20 12:44:04 +03:00
|
|
|
pub fn name(&self) -> &str {
|
2018-07-11 23:12:25 +03:00
|
|
|
&self.name
|
|
|
|
}
|
2018-08-03 13:46:08 +03:00
|
|
|
pub fn workers(&mut self) -> &mut Vec<Worker> {
|
|
|
|
&mut self.workers
|
2017-09-14 18:08:14 +03:00
|
|
|
}
|
2018-08-23 15:36:52 +03:00
|
|
|
|
2018-09-17 07:53:16 +03:00
|
|
|
fn load_mailbox(&mut self, index: usize, mailbox: Result<Mailbox>) {
|
2018-10-14 19:49:16 +03:00
|
|
|
self.folders[index] = Some(mailbox);
|
|
|
|
/*
|
2018-09-10 02:21:46 +03:00
|
|
|
if self.sent_folder.is_some() && self.sent_folder.unwrap() == index {
|
2018-09-17 07:53:16 +03:00
|
|
|
self.folders[index] = Some(mailbox);
|
2018-09-10 02:21:46 +03:00
|
|
|
/* Add our replies to other folders */
|
|
|
|
for id in (0..self.folders.len()).filter(|i| *i != index) {
|
|
|
|
self.add_replies_to_folder(id);
|
|
|
|
}
|
|
|
|
} else {
|
2018-09-17 07:53:16 +03:00
|
|
|
self.folders[index] = Some(mailbox);
|
2018-09-10 02:21:46 +03:00
|
|
|
self.add_replies_to_folder(index);
|
|
|
|
};
|
2018-10-14 19:49:16 +03:00
|
|
|
*/
|
2018-09-10 02:21:46 +03:00
|
|
|
}
|
|
|
|
|
2019-02-11 14:55:29 +02:00
|
|
|
/*
|
2018-09-10 02:21:46 +03:00
|
|
|
fn add_replies_to_folder(&mut self, folder_index: usize) {
|
|
|
|
if let Some(sent_index) = self.sent_folder.as_ref() {
|
|
|
|
if self.folders[*sent_index]
|
|
|
|
.as_ref()
|
|
|
|
.map(|v| v.is_ok())
|
|
|
|
.unwrap_or(false)
|
|
|
|
&& self.folders[folder_index]
|
|
|
|
.as_ref()
|
|
|
|
.map(|v| v.is_ok())
|
|
|
|
.unwrap_or(false)
|
|
|
|
{
|
2018-08-07 15:01:15 +03:00
|
|
|
let (sent, cur) = {
|
|
|
|
let ptr = self.folders.as_mut_ptr();
|
|
|
|
unsafe {
|
|
|
|
use std::slice::from_raw_parts_mut;
|
|
|
|
(
|
2018-09-10 02:21:46 +03:00
|
|
|
from_raw_parts_mut(ptr.offset(*sent_index as isize), *sent_index + 1)
|
2018-09-23 19:55:29 +03:00
|
|
|
[0].as_mut()
|
|
|
|
.unwrap()
|
|
|
|
.as_mut()
|
|
|
|
.unwrap(),
|
2018-09-10 02:21:46 +03:00
|
|
|
from_raw_parts_mut(ptr.offset(folder_index as isize), folder_index + 1)
|
2018-09-23 19:55:29 +03:00
|
|
|
[0].as_mut()
|
|
|
|
.unwrap()
|
|
|
|
.as_mut()
|
|
|
|
.unwrap(),
|
2018-08-07 15:01:15 +03:00
|
|
|
)
|
2017-09-14 18:08:14 +03:00
|
|
|
}
|
2018-08-07 15:01:15 +03:00
|
|
|
};
|
2018-09-10 02:21:46 +03:00
|
|
|
cur.insert_sent_folder(&sent);
|
2018-08-07 15:01:15 +03:00
|
|
|
}
|
2018-09-10 02:21:46 +03:00
|
|
|
}
|
2018-08-03 13:46:08 +03:00
|
|
|
}
|
2019-02-11 14:55:29 +02:00
|
|
|
*/
|
2018-08-03 13:46:08 +03:00
|
|
|
|
2018-09-05 16:08:11 +03:00
|
|
|
pub fn status(&mut self, index: usize) -> result::Result<(), usize> {
|
2018-08-03 13:46:08 +03:00
|
|
|
match self.workers[index].as_mut() {
|
2018-08-07 15:01:15 +03:00
|
|
|
None => {
|
2018-09-05 16:08:11 +03:00
|
|
|
return Ok(());
|
2018-08-07 15:01:15 +03:00
|
|
|
}
|
2018-10-14 19:49:16 +03:00
|
|
|
Some(ref mut w) if self.folders[index].is_none() => match w.poll() {
|
2018-08-07 15:01:15 +03:00
|
|
|
Ok(AsyncStatus::NoUpdate) => {
|
|
|
|
return Err(0);
|
|
|
|
}
|
|
|
|
Ok(AsyncStatus::Finished) => {}
|
|
|
|
Ok(AsyncStatus::ProgressReport(n)) => {
|
|
|
|
return Err(n);
|
|
|
|
}
|
2018-10-14 19:49:16 +03:00
|
|
|
_ => {
|
2018-08-07 15:01:15 +03:00
|
|
|
return Err(0);
|
2018-08-03 13:46:08 +03:00
|
|
|
}
|
|
|
|
},
|
2018-10-14 19:49:16 +03:00
|
|
|
Some(_) => return Ok(()),
|
2018-08-03 13:46:08 +03:00
|
|
|
};
|
2018-10-14 19:49:16 +03:00
|
|
|
let m = mem::replace(&mut self.workers[index], None)
|
|
|
|
.unwrap()
|
|
|
|
.extract();
|
2018-08-03 13:46:08 +03:00
|
|
|
self.workers[index] = None;
|
2018-09-12 15:10:19 +03:00
|
|
|
self.load_mailbox(index, m);
|
|
|
|
Ok(())
|
2018-08-03 13:46:08 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Index<usize> for Account {
|
|
|
|
type Output = Result<Mailbox>;
|
|
|
|
fn index(&self, index: usize) -> &Result<Mailbox> {
|
2018-08-07 15:01:15 +03:00
|
|
|
&self.folders[index]
|
|
|
|
.as_ref()
|
|
|
|
.expect("BUG: Requested mailbox that is not yet available.")
|
2018-08-03 13:46:08 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Will panic if mailbox hasn't loaded, ask `status()` first.
|
|
|
|
impl IndexMut<usize> for Account {
|
|
|
|
fn index_mut(&mut self, index: usize) -> &mut Result<Mailbox> {
|
2018-08-07 15:01:15 +03:00
|
|
|
self.folders[index]
|
|
|
|
.as_mut()
|
|
|
|
.expect("BUG: Requested mailbox that is not yet available.")
|
2017-09-14 18:08:14 +03:00
|
|
|
}
|
|
|
|
}
|