deltachat/imap/
scan_folders.rs1use std::collections::BTreeMap;
2
3use anyhow::{Context as _, Result};
4
5use super::{get_folder_meaning_by_attrs, get_folder_meaning_by_name};
6use crate::config::Config;
7use crate::imap::{Imap, session::Session};
8use crate::log::LogExt;
9use crate::tools::{self, time_elapsed};
10use crate::{context::Context, imap::FolderMeaning};
11
12impl Imap {
13 pub(crate) async fn scan_folders(
15 &mut self,
16 context: &Context,
17 session: &mut Session,
18 ) -> Result<bool> {
19 {
21 let mut last_scan = session.last_full_folder_scan.lock().await;
22 if let Some(last_scan) = *last_scan {
23 let elapsed_secs = time_elapsed(&last_scan).as_secs();
24 let debounce_secs = context
25 .get_config_u64(Config::ScanAllFoldersDebounceSecs)
26 .await?;
27
28 if elapsed_secs < debounce_secs {
29 return Ok(false);
30 }
31 }
32
33 last_scan.replace(tools::Time::now());
38 }
39 info!(context, "Starting full folder scan");
40
41 let folders = session.list_folders().await?;
42 let watched_folders = get_watched_folders(context).await?;
43
44 let mut folder_configs = BTreeMap::new();
45 let mut folder_names = Vec::new();
46
47 for folder in folders {
48 let folder_meaning = get_folder_meaning_by_attrs(folder.attributes());
49 if folder_meaning == FolderMeaning::Virtual {
50 continue;
55 }
56 folder_names.push(folder.name().to_string());
57 let folder_name_meaning = get_folder_meaning_by_name(folder.name());
58
59 if let Some(config) = folder_meaning.to_config() {
60 folder_configs.insert(config, folder.name().to_string());
62 } else if let Some(config) = folder_name_meaning.to_config() {
63 folder_configs
65 .entry(config)
66 .or_insert_with(|| folder.name().to_string());
67 }
68
69 let folder_meaning = match folder_meaning {
70 FolderMeaning::Unknown => folder_name_meaning,
71 _ => folder_meaning,
72 };
73
74 if !watched_folders.contains(&folder.name().to_string())
76 && folder_meaning != FolderMeaning::Trash
77 && folder_meaning != FolderMeaning::Unknown
78 {
79 self.fetch_move_delete(context, session, folder.name(), folder_meaning)
80 .await
81 .context("Can't fetch new msgs in scanned folder")
82 .log_err(context)
83 .ok();
84 }
85 }
86
87 let conf = Config::ConfiguredTrashFolder;
89 let val = folder_configs.get(&conf).map(|s| s.as_str());
90 let interrupt = val.is_some() && context.get_config(conf).await?.is_none();
91 context.set_config_internal(conf, val).await?;
92 if interrupt {
93 context.scheduler.interrupt_oboxes().await;
96 }
97
98 info!(context, "Found folders: {folder_names:?}.");
99 Ok(true)
100 }
101}
102
103pub(crate) async fn get_watched_folder_configs(context: &Context) -> Result<Vec<Config>> {
104 let mut res = vec![Config::ConfiguredInboxFolder];
105 if context.should_watch_mvbox().await? {
106 res.push(Config::ConfiguredMvboxFolder);
107 }
108 Ok(res)
109}
110
111pub(crate) async fn get_watched_folders(context: &Context) -> Result<Vec<String>> {
112 let mut res = Vec::new();
113 for folder_config in get_watched_folder_configs(context).await? {
114 if let Some(folder) = context.get_config(folder_config).await? {
115 res.push(folder);
116 }
117 }
118 Ok(res)
119}