This commit is contained in:
holger krekel
2019-11-07 15:54:37 +01:00
parent 4cebdf3b84
commit 7b7bead869
2 changed files with 78 additions and 111 deletions

View File

@@ -380,10 +380,8 @@ impl Imap {
self.should_reconnect.load(Ordering::Relaxed) self.should_reconnect.load(Ordering::Relaxed)
} }
fn setup_handle_if_needed(&self, context: &Context) -> bool { fn setup_handle_if_needed(&self, context: &Context) -> Result<(), Error> {
if self.config.read().unwrap().imap_server.is_empty() { ensure!(!self.config.read().unwrap().imap_server.is_empty(), "no imap server configured");
return false;
}
if self.should_reconnect() { if self.should_reconnect() {
self.unsetup_handle(context); self.unsetup_handle(context);
@@ -391,7 +389,7 @@ impl Imap {
if self.is_connected() && self.stream.read().unwrap().is_some() { if self.is_connected() && self.stream.read().unwrap().is_some() {
self.should_reconnect.store(false, Ordering::Relaxed); self.should_reconnect.store(false, Ordering::Relaxed);
return true; return Ok(());
} }
let server_flags = self.config.read().unwrap().server_flags as i32; let server_flags = self.config.read().unwrap().server_flags as i32;
@@ -437,7 +435,7 @@ impl Imap {
}; };
client.authenticate("XOAUTH2", &auth) client.authenticate("XOAUTH2", &auth)
} else { } else {
return false; bail!("could not authenticate with XOAUTH2");
} }
} else { } else {
client.login(imap_user, imap_pw) client.login(imap_user, imap_pw)
@@ -455,7 +453,7 @@ impl Imap {
emit_event!(context, Event::ErrorNetwork(message)); emit_event!(context, Event::ErrorNetwork(message));
return false; bail!("login error: {}", err);
} }
}; };
@@ -465,7 +463,7 @@ impl Imap {
Ok((session, stream)) => { Ok((session, stream)) => {
*self.session.lock().unwrap() = Some(session); *self.session.lock().unwrap() = Some(session);
*self.stream.write().unwrap() = Some(stream); *self.stream.write().unwrap() = Some(stream);
true Ok(())
} }
Err((err, _)) => { Err((err, _)) => {
let imap_user = self.config.read().unwrap().imap_user.to_owned(); let imap_user = self.config.read().unwrap().imap_user.to_owned();
@@ -477,7 +475,7 @@ impl Imap {
); );
self.unsetup_handle(context); self.unsetup_handle(context);
false bail!("{}", err)
} }
} }
} }
@@ -549,7 +547,7 @@ impl Imap {
config.server_flags = server_flags; config.server_flags = server_flags;
} }
if !self.setup_handle_if_needed(context) { if self.setup_handle_if_needed(context).is_err() {
self.free_connect_params(); self.free_connect_params();
return false; return false;
} }
@@ -622,7 +620,7 @@ impl Imap {
// get any more. if IDLE is called directly after, there is only a small chance that // get any more. if IDLE is called directly after, there is only a small chance that
// messages are missed and delayed until the next IDLE call // messages are missed and delayed until the next IDLE call
loop { loop {
if self.fetch_from_single_folder(context, watch_folder) == 0 { if self.fetch_from_single_folder(context, watch_folder).is_err() {
break; break;
} }
} }
@@ -656,19 +654,19 @@ impl Imap {
true true
} }
fn select_folder(&self, context: &Context, folder: &str) -> usize { fn select_folder(&self, context: &Context, folder: &str) -> Result<(), Error> {
if self.session.lock().unwrap().is_none() { if self.session.lock().unwrap().is_none() {
let mut cfg = self.config.write().unwrap(); let mut cfg = self.config.write().unwrap();
cfg.selected_folder = None; cfg.selected_folder = None;
cfg.selected_folder_needs_expunge = false; cfg.selected_folder_needs_expunge = false;
return 0; bail!("select_folder: session is closed");
} }
// if there is a new folder and the new folder is equal to the selected one, there's nothing to do. // if there is a new folder and the new folder is equal to the selected one, there's nothing to do.
// if there is _no_ new folder, we continue as we might want to expunge below. // if there is _no_ new folder, we continue as we might want to expunge below.
if let Some(ref selected_folder) = self.config.read().unwrap().selected_folder { if let Some(ref selected_folder) = self.config.read().unwrap().selected_folder {
if folder == selected_folder { if folder == selected_folder {
return 1; return Ok(());
} }
} }
@@ -688,23 +686,16 @@ impl Imap {
config.selected_mailbox = Some(mailbox); config.selected_mailbox = Some(mailbox);
} }
Err(err) => { Err(err) => {
info!(
context,
"Cannot select folder: {}; {:?}.",
folder,
err
);
self.config.write().unwrap().selected_folder = None; self.config.write().unwrap().selected_folder = None;
self.should_reconnect.store(true, Ordering::Relaxed); self.should_reconnect.store(true, Ordering::Relaxed);
return 0; bail!("Cannot select folder: {}; {:?}.", folder, err);
} }
} }
} else { } else {
unreachable!(); unreachable!();
} }
1 Ok(())
} }
fn get_config_last_seen_uid(&self, context: &Context, folder: &str) -> (u32, u32) { fn get_config_last_seen_uid(&self, context: &Context, folder: &str) -> (u32, u32) {
@@ -729,49 +720,25 @@ impl Imap {
} }
} }
fn fetch_from_single_folder(&self, context: &Context, folder: &str) -> usize {
if !self.is_connected() {
info!(
context,
"Cannot fetch from \"{}\" - not connected.",
folder
);
return 0; fn
}
if self.select_folder(context, &folder) == 0 {
info!(
context,
"Cannot select folder \"{}\" for fetching.",
folder
);
return 0;
}
// compare last seen UIDVALIDITY against the current one // compare last seen UIDVALIDITY against the current one
let (mut uid_validity, mut last_seen_uid) = self.get_config_last_seen_uid(context, &folder); let (last_uid_validity, last_seen_uid) = self.get_config_last_seen_uid(context, &folder);
/* let mailbox = self.config.read().unwrap().selected_mailbox.as_ref().expect("just selected");
let uid_validity = self.config.read().unwrap().selected_mailbox.as_ref().expect("just selected").uid_validity; ensure!(mailbox.uid_validity.is_some(),
"Cannot get UIDVALIDITY for folder \"{}\".", folder,);
if uid_validity.is_none() { let mailbox = mailbox.unwrap();
error!( let current_uid_validity = mailbox.uid_validity.unwrap_or_default();
context,
"Cannot get UIDVALIDITY for folder \"{}\".",
folder.as_ref(),
);
return 0; if current_uid_validity != last_uid_validity {
} // first time this folder is selected or UIDVALIDITY has changed,
// init lastseenuid and save it to config
if .uid_validity.unwrap_or_default() != uid_validity {
// first time this folder is selected or UIDVALIDITY has changed, init lastseenuid and save it to config
if mailbox.exists == 0 { if mailbox.exists == 0 {
info!(context, "Folder \"{}\" is empty.", folder.as_ref()); info!(context, "Folder \"{}\" is empty.", folder);
// set lastseenuid=0 for empty folders. // set lastseenuid=0 for empty folders.
// id we do not do this here, we'll miss the first message // id we do not do this here, we'll miss the first message
// as we will get in here again and fetch from lastseenuid+1 then // as we will get in here again and fetch from lastseenuid+1 then
@@ -779,10 +746,10 @@ impl Imap {
self.set_config_last_seen_uid( self.set_config_last_seen_uid(
context, context,
&folder, &folder,
mailbox.uid_validity.unwrap_or_default(), current_uid_validity,
0, 0,
); );
return 0; return Ok(false);
} }
let list = if let Some(ref mut session) = &mut *self.session.lock().unwrap() { let list = if let Some(ref mut session) = &mut *self.session.lock().unwrap() {
@@ -790,57 +757,57 @@ impl Imap {
let set = format!("{}", mailbox.exists); let set = format!("{}", mailbox.exists);
match session.fetch(set, PREFETCH_FLAGS) { match session.fetch(set, PREFETCH_FLAGS) {
Ok(list) => list, Ok(list) => list,
Err(_err) => { Err(err) => {
self.should_reconnect.store(true, Ordering::Relaxed); self.should_reconnect.store(true, Ordering::Relaxed);
info!( bail!("IMAP FETCH {} {} failed on folder '{}': {}'",
context, set, PREFETCH_FLAGS, folder, err);
"No result returned for folder \"{}\".",
folder.as_ref()
);
return 0;
} }
} }
} else { } else {
return 0; unreachable!();
}; };
last_seen_uid = list[0].uid.unwrap_or_else(|| 0); last_seen_uid = list[0].uid.unwrap_or_default();
// if the UIDVALIDITY has _changed_, decrease lastseenuid by one to avoid gaps (well add 1 below // if the UIDVALIDITY has _changed_, decrease lastseenuid by one to avoid gaps (well add 1 below
if uid_validity > 0 && last_seen_uid > 1 { if last_uid_validity > 0 && last_seen_uid > 1 {
last_seen_uid -= 1; last_seen_uid -= 1;
} }
uid_validity = mailbox.uid_validity.unwrap_or_default();
self.set_config_last_seen_uid(context, &folder, uid_validity, last_seen_uid); self.set_config_last_seen_uid(context, &folder, uid_validity, last_seen_uid);
info!( info!(
context, context,
"lastseenuid initialized to {} for {}@{}", "lastseenuid initialized to {} for {}@{}",
last_seen_uid, last_seen_uid,
folder.as_ref(), folder,
uid_validity, uid_validity,
); );
Ok(true)
} }
*/
fn fetch_from_single_folder(&self, context: &Context, folder: &str) -> Result<(), Error> {
self.select_folder(context, &folder)?;
if self.folder_is_uidvalid_and_empty(&self, context: &Context, folder: &str)? {
return
let mut read_cnt = 0; let mut read_cnt = 0;
let mut read_errors = 0; let mut read_errors = 0;
let mut new_last_seen_uid = 0; let mut new_last_seen_uid = 0;
let set = format!("{}:*", last_seen_uid + 1);
let list = if let Some(ref mut session) = &mut *self.session.lock().unwrap() { let list = if let Some(ref mut session) = &mut *self.session.lock().unwrap() {
// fetch messages with larger UID than the last one seen // fetch messages with larger UID than the last one seen
// (`UID FETCH lastseenuid+1:*)`, see RFC 4549 // (`UID FETCH lastseenuid+1:*)`, see RFC 4549
let set = format!("{}:*", last_seen_uid + 1);
match session.uid_fetch(set, PREFETCH_FLAGS) { match session.uid_fetch(set, PREFETCH_FLAGS) {
Ok(list) => list, Ok(list) => list,
Err(err) => { Err(err) => {
warn!(context, "failed to fetch uids: {}", err); warn!(context, "failed to fetch uids: {}", err);
return 0; bail!("failed to fetch uids: {}", err)
} }
} }
} else { } else {
return 0; unreachable!();
}; };
// go through all mails in folder (this is typically _fast_ as we already have the whole list) // go through all mails in folder (this is typically _fast_ as we already have the whole list)
@@ -901,7 +868,7 @@ impl Imap {
); );
} }
read_cnt Ok(read_cnt)
} }
fn set_config_last_seen_uid( fn set_config_last_seen_uid(
@@ -985,19 +952,16 @@ impl Imap {
1 1
} }
pub fn idle(&self, context: &Context) { pub fn idle(&self, context: &Context) -> Result<(), Error> {
if !self.config.read().unwrap().can_idle { if !self.config.read().unwrap().can_idle {
return self.fake_idle(context); self.fake_idle(context);
} }
self.setup_handle_if_needed(context); self.setup_handle_if_needed(context)?;
let watch_folder = self.config.read().unwrap().watch_folder.clone(); let watch_folder = self.config.read().unwrap().watch_folder.clone();
if let Some(wfolder) = watch_folder { if let Some(wfolder) = watch_folder {
if self.select_folder(context, &wfolder) == 0 { self.select_folder(context, &wfolder)?;
warn!(context, "IMAP-IDLE not setup.",);
return self.fake_idle(context);
}
} }
let session = self.session.clone(); let session = self.session.clone();
@@ -1091,6 +1055,7 @@ impl Imap {
} }
*watch = false; *watch = false;
Ok(())
} }
fn fake_idle(&self, context: &Context) { fn fake_idle(&self, context: &Context) {
@@ -1152,7 +1117,7 @@ impl Imap {
let watch_folder = self.config.read().unwrap().watch_folder.clone(); let watch_folder = self.config.read().unwrap().watch_folder.clone();
if let Some(watch_folder) = watch_folder { if let Some(watch_folder) = watch_folder {
if 0 != self.fetch_from_single_folder(context, &watch_folder) { if self.fetch_from_single_folder(context, &watch_folder).is_ok() {
do_fake_idle = false; do_fake_idle = false;
} }
} }
@@ -1289,14 +1254,12 @@ impl Imap {
return Some(ImapResult::RetryLater); return Some(ImapResult::RetryLater);
} }
} }
if self.select_folder(context, &folder) == 0 { match self.select_folder(context, &folder) {
warn!( Ok(()) => None,
context, Err(err) => {
"Cannot select folder {} for preparing IMAP operation", folder warn!(context, "Cannot select folder {}: {}", folder, err);
); Some(ImapResult::RetryLater)
Some(ImapResult::RetryLater) }
} else {
None
} }
} }
@@ -1502,21 +1465,17 @@ impl Imap {
} }
} }
pub fn empty_folder(&self, context: &Context, folder: &str) { pub fn empty_folder(&self, context: &Context, folder: &str) -> Result<(), Error> {
info!(context, "emptying folder {}", folder); info!(context, "emptying folder {}", folder);
if folder.is_empty() || self.select_folder(context, &folder) == 0 { self.select_folder(context, &folder)?;
warn!(context, "Cannot select folder '{}' for emptying", folder);
return;
}
if !self.add_flag_finalized_with_set(context, SELECT_ALL, "\\Deleted") { ensure!(self.add_flag_finalized_with_set(context, SELECT_ALL, "\\Deleted"),
warn!(context, "Cannot empty folder {}", folder); "Could not set \\Deleted flags to empty {}", folder);
} else {
if self.expunge_folder(context) { ensure!(self.expunge_folder(context), "Could not expunge");
emit_event!(context, Event::ImapFolderEmptied(folder.to_string())); emit_event!(context, Event::ImapFolderEmptied(folder.to_string()));
} Ok(())
}
} }
} }

View File

@@ -291,19 +291,20 @@ impl Job {
} }
#[allow(non_snake_case)] #[allow(non_snake_case)]
fn do_DC_JOB_EMPTY_SERVER(&mut self, context: &Context) { fn do_DC_JOB_EMPTY_SERVER(&mut self, context: &Context) -> Result<(), Error> {
let inbox = context.inbox.read().unwrap(); let inbox = context.inbox.read().unwrap();
if self.foreign_id & DC_EMPTY_MVBOX > 0 { if self.foreign_id & DC_EMPTY_MVBOX > 0 {
if let Some(mvbox_folder) = context if let Some(mvbox_folder) = context
.sql .sql
.get_raw_config(context, "configured_mvbox_folder") .get_raw_config(context, "configured_mvbox_folder")
{ {
inbox.empty_folder(context, &mvbox_folder); inbox.empty_folder(context, &mvbox_folder)?;
} }
} }
if self.foreign_id & DC_EMPTY_INBOX > 0 { if self.foreign_id & DC_EMPTY_INBOX > 0 {
inbox.empty_folder(context, "INBOX"); inbox.empty_folder(context, "INBOX")?;
} }
Ok(())
} }
#[allow(non_snake_case)] #[allow(non_snake_case)]
@@ -793,7 +794,14 @@ fn job_perform(context: &Context, thread: Thread, probe_network: bool) {
warn!(context, "Unknown job id found"); warn!(context, "Unknown job id found");
} }
Action::SendMsgToSmtp => job.do_DC_JOB_SEND(context), Action::SendMsgToSmtp => job.do_DC_JOB_SEND(context),
Action::EmptyServer => job.do_DC_JOB_EMPTY_SERVER(context), Action::EmptyServer => {
match job.do_DC_JOB_EMPTY_SERVER(context) {
Ok(()) => {},
Err(err) => {
warn!(context, "emptying server folder(s) failed: {}", err);
}
}
}
Action::DeleteMsgOnImap => job.do_DC_JOB_DELETE_MSG_ON_IMAP(context), Action::DeleteMsgOnImap => job.do_DC_JOB_DELETE_MSG_ON_IMAP(context),
Action::MarkseenMsgOnImap => job.do_DC_JOB_MARKSEEN_MSG_ON_IMAP(context), Action::MarkseenMsgOnImap => job.do_DC_JOB_MARKSEEN_MSG_ON_IMAP(context),
Action::MarkseenMdnOnImap => job.do_DC_JOB_MARKSEEN_MDN_ON_IMAP(context), Action::MarkseenMdnOnImap => job.do_DC_JOB_MARKSEEN_MDN_ON_IMAP(context),