/*---------------------------------------------------------------------- new_mail() - check for new mail in the inbox Input: force -- flag indicating we should check for new mail no matter time_for_check_point -- 0: GoodTime, 1: BadTime, 2: VeryBadTime flags -- whether to q a new mail status message or defer the sort Result: returns -1 if there was no new mail. Otherwise returns the sorted message number of the smallest message number that was changed. That is the screen needs to be repainted from that message down. Limit frequency of checks because checks use some resources. That is they generate an IMAP packet or require locking the local mailbox. (Acutally the lock isn't required, a stat will do, but the current c-client mail code locks before it stats.) Returns >= 0 only if there is a change in the given mail stream. Otherwise this returns -1. On return the message counts in the pine structure are updated to reflect the current number of messages including any new mail and any expunging. --- */ long new_mail(int force_arg, CheckPointTime time_for_check_point, int flags) { static time_t last_check_point_call = 0; long since_last_input; time_t expunged_reaper_to, adj_idle_timeout, interval, adj; static int nexttime = 0; time_t now; long n, rv = 0, t_nm_count = 0, exp_count; MAILSTREAM *m; int force, i, started_on; int new_mail_was_announced = 0; int have_pinged_non_special = 0; int timeo; dprint((9, "new mail called (force=%d %s flags=0x%x)\n", force_arg, time_for_check_point == GoodTime ? "GoodTime" : time_for_check_point == BadTime ? "BadTime" : time_for_check_point == VeryBadTime ? "VeryBad" : time_for_check_point == DoItNow ? "DoItNow" : "?", flags)); force = force_arg; now = time(0); timeo = get_input_timeout(); if(time_for_check_point == GoodTime) adrbk_maintenance(); if(time_for_check_point == GoodTime || force_arg) folder_unseen_count_updater(UFU_ANNOUNCE | (force_arg ? UFU_FORCE : 0)); if(sp_need_to_rethread(ps_global->mail_stream)) force = 1; if(!force && sp_unsorted_newmail(ps_global->mail_stream)) force = !(flags & NM_DEFER_SORT); if(!ps_global->mail_stream || !(timeo || force || sp_a_locked_stream_changed())) return(-1); last_check_point_call = now; since_last_input = (long) now - (long) time_of_last_input(); /* * We have this for loop followed by the do-while so that we will prefer * to ping the active streams before we ping the inactive ones, in cases * where the pings or checks are taking a long time. */ for(i = 0; i < ps_global->s_pool.nstream; i++){ m = ps_global->s_pool.streams[i]; if(!m || m->halfopen || (m != ps_global->mail_stream && !(force_arg && sp_flagged(m, SP_LOCKED)))) continue; /* * This for() loop is only the current folder, unless the user * has forced the check, in which case it is all locked folders. */ /* * After some amount of inactivity on a stream, the server may * close the stream. Each protocol has its own idea of how much * inactivity should be allowed before the stream is closed. For * example, IMAP specifies that the server should not close the * stream unilaterally until at least 30 minutes of inactivity. * The GET_IDLETIMEOUT call gives us that time in minutes. We * want to be sure to keep the stream alive if it is about to die * due to inactivity. */ adj_idle_timeout = 60 * (long) mail_parameters(m,GET_IDLETIMEOUT,NULL); if(adj_idle_timeout <= 0) adj_idle_timeout = 600; adj = (adj_idle_timeout >= 50 * FUDGE) ? 5 * FUDGE : (adj_idle_timeout >= 30 * FUDGE) ? 3 * FUDGE : (adj_idle_timeout >= 20 * FUDGE) ? 2 * FUDGE : FUDGE; adj_idle_timeout = MAX(adj_idle_timeout - adj, 120); /* * Set interval to mail-check-interval unless * mail-check-interval-noncurrent is nonzero and this is not inbox * or current stream. */ if(ps_global->check_interval_for_noncurr > 0 && m != ps_global->mail_stream && !sp_flagged(m, SP_INBOX)) interval = ps_global->check_interval_for_noncurr; else interval = timeo; /* * We want to make sure that we notice expunges, but we don't have * to be fanatical about it. Every once in a while we'll do a ping * because we haven't had a command that notices expunges for a * while. It's also a larger number than interval so it gives us a * convenient interval to do slower pinging than interval if we * are busy. */ if(interval <= adj_idle_timeout) expunged_reaper_to = MIN(MAX(2*interval,180), adj_idle_timeout); else expunged_reaper_to = interval; /* * User may want to avoid new mail checking while composing. * In this case we will only do the keepalives. */ if(timeo == 0 || (flags & NM_FROM_COMPOSER && ((F_ON(F_QUELL_PINGS_COMPOSING, ps_global) && !sp_flagged(m, SP_INBOX)) || (F_ON(F_QUELL_PINGS_COMPOSING_INBOX, ps_global) && sp_flagged(m, SP_INBOX))))){ interval = expunged_reaper_to = 0; } dprint((9, "%s: force=%d interval=%ld exp_reap_to=%ld adj_idle_to=%ld\n", STREAMNAME(m), force_arg, (long) interval, (long) expunged_reaper_to, (long) adj_idle_timeout)); dprint((9, " since_last_ping=%ld since_last_reap=%ld\n", (long) (now - sp_last_ping(m)), (long) (now - sp_last_expunged_reaper(m)))); /* if check_point does a check it resets last_ping time */ if(force_arg || (timeo && expunged_reaper_to > 0)) (void) check_point(m, time_for_check_point, flags); /* * Remember that unless force_arg is set, this check is really * only for the current folder. This is usually going to fire * on the first test, which is the interval the user set. */ if(force_arg || (timeo && ((interval && (now - sp_last_ping(m) >= interval-1)) || (expunged_reaper_to && (now - sp_last_expunged_reaper(m)) >= expunged_reaper_to) || (now - sp_last_ping(m) >= adj_idle_timeout)))){ if((flags & NM_STATUS_MSG) && pith_opt_newmail_check_cue) (*pith_opt_newmail_check_cue)(TRUE); dprint((7, "Mail_Ping(%s): lastping=%ld er=%ld%s%s %s\n", STREAMNAME(m), (long) (now - sp_last_ping(m)), (long) (now - sp_last_expunged_reaper(m)), force_arg ? " [forced]" : interval && (now-sp_last_ping(m) >= interval-1) ? " [it's time]" : expunged_reaper_to && (now - sp_last_expunged_reaper(m) >= expunged_reaper_to) ? " [expunged reaper]" : " [keepalive]", m == ps_global->mail_stream ? " [current]" : "", debug_time(0,1))); /* * We're about to ping the stream. * If the stream is a #move Mail Drop there is a minimum time * between re-opens of the mail drop to check for new mail. * If the check is forced by the user, they want it to * happen now. We use knowledge of c-client internals to * make this happen. */ if(force_arg && m && m->snarf.name) m->snarf.time = 0; /*-- Ping the stream to check for new mail --*/ if(sp_dead_stream(m)){ dprint((6, "no check: stream is dead\n")); } else if(!pine_mail_ping(m)){ dprint((6, "ping failed: stream is dead\n")); sp_set_dead_stream(m, 1); } dprint((7, "Ping complete: %s\n", debug_time(0,1))); if((flags & NM_STATUS_MSG) && pith_opt_newmail_check_cue) (*pith_opt_newmail_check_cue)(FALSE); } } if(nexttime < 0 || nexttime >= ps_global->s_pool.nstream) nexttime = 0; i = started_on = nexttime; do { m = ps_global->s_pool.streams[i]; nexttime = i; if(ps_global->s_pool.nstream > 0) i = (i + 1) % ps_global->s_pool.nstream; /* * This do() loop handles all the other streams that weren't covered * in the for() loop above. */ if((!m || m->halfopen) || (m == ps_global->mail_stream || (force_arg && sp_flagged(m, SP_LOCKED)))){ nexttime = i; continue; } /* * If it is taking an extra long time to do the pings and checks, * think about skipping some of them. Always do at least one of * these non-special streams (if they are due to be pinged). * The nexttime variable keeps track of where we were so that we * don't always ping the first one in the list and then skip out. */ if((time(0) - now >= 5) && have_pinged_non_special){ dprint((7, "skipping pings due to delay: %s\n", debug_time(0,1))); break; } nexttime = i; have_pinged_non_special++; adj_idle_timeout = 60 * (long) mail_parameters(m,GET_IDLETIMEOUT,NULL); if(adj_idle_timeout <= 0) adj_idle_timeout = 600; adj = (adj_idle_timeout >= 50 * FUDGE) ? 5 * FUDGE : (adj_idle_timeout >= 30 * FUDGE) ? 3 * FUDGE : (adj_idle_timeout >= 20 * FUDGE) ? 2 * FUDGE : FUDGE; adj_idle_timeout = MAX(adj_idle_timeout - adj, 120); if(ps_global->check_interval_for_noncurr > 0 && m != ps_global->mail_stream && !sp_flagged(m, SP_INBOX)) interval = ps_global->check_interval_for_noncurr; else interval = timeo; if(interval <= adj_idle_timeout) expunged_reaper_to = MIN(MAX(2*interval,180), adj_idle_timeout); else expunged_reaper_to = interval; if(timeo == 0 || (flags & NM_FROM_COMPOSER && ((F_ON(F_QUELL_PINGS_COMPOSING, ps_global) && !sp_flagged(m, SP_INBOX)) || (F_ON(F_QUELL_PINGS_COMPOSING_INBOX, ps_global) && sp_flagged(m, SP_INBOX))))){ interval = expunged_reaper_to = 0; } dprint((9, "%s: force=%d interval=%ld exp_reap_to=%ld adj_idle_to=%ld\n", STREAMNAME(m), force_arg, (long) interval, (long) expunged_reaper_to, (long) adj_idle_timeout)); dprint((9, " since_last_ping=%ld since_last_reap=%ld since_last_input=%ld\n", (long) (now - sp_last_ping(m)), (long) (now - sp_last_expunged_reaper(m)), (long) since_last_input)); /* if check_point does a check it resets last_ping time */ if(force_arg || (timeo && expunged_reaper_to > 0)) (void) check_point(m, time_for_check_point, flags); /* * The check here is a little bit different from the current folder * check in the for() loop above. In particular, we defer our * pinging for awhile if the last input was recent (except for the * inbox!). We ping streams which are cached but not actively being * used (that is, the non-locked streams) at a slower rate. * If we don't use them for a long time we will eventually close them * (in maybe_kill_old_stream()) but we do it when we want to instead * of when the server wants us to by attempting to keep it alive here. * The other reason to ping the cached streams is that we only get * told the new mail in those streams is recent one time, the messages * that weren't handled here will no longer be recent next time * we open the folder. */ if((force_arg && sp_flagged(m, SP_LOCKED)) || (timeo && ((interval && sp_flagged(m, SP_LOCKED) && ((since_last_input >= 3 && (now-sp_last_ping(m) >= interval-1)) || (sp_flagged(m, SP_INBOX) && (now-sp_last_ping(m) >= interval-1)))) || (expunged_reaper_to && sp_flagged(m, SP_LOCKED) && (now-sp_last_expunged_reaper(m) >= expunged_reaper_to)) || (expunged_reaper_to && !sp_flagged(m, SP_LOCKED) && since_last_input >= 3 && (now-sp_last_ping(m) >= expunged_reaper_to)) || (now - sp_last_ping(m) >= adj_idle_timeout)))){ if((flags & NM_STATUS_MSG) && pith_opt_newmail_check_cue) (*pith_opt_newmail_check_cue)(TRUE); dprint((7, "Mail_Ping(%s): lastping=%ld er=%ld%s idle: %ld %s\n", STREAMNAME(m), (long) (now - sp_last_ping(m)), (long) (now - sp_last_expunged_reaper(m)), (force_arg && sp_flagged(m, SP_LOCKED)) ? " [forced]" : (interval && sp_flagged(m, SP_LOCKED) && ((since_last_input >= 3 && (now-sp_last_ping(m) >= interval-1)) || (sp_flagged(m, SP_INBOX) && (now-sp_last_ping(m) >= interval-1)))) ? " [it's time]" : (expunged_reaper_to && sp_flagged(m, SP_LOCKED) && (now-sp_last_expunged_reaper(m) >= expunged_reaper_to)) ? " [expunged reaper]" : (expunged_reaper_to && !sp_flagged(m, SP_LOCKED) && since_last_input >= 3 && (now-sp_last_ping(m) >= expunged_reaper_to)) ? " [slow ping]" : " [keepalive]", since_last_input, debug_time(0,1))); /* * We're about to ping the stream. * If the stream is a #move Mail Drop there is a minimum time * between re-opens of the mail drop to check for new mail. * If the check is forced by the user, they want it to * happen now. We use knowledge of c-client internals to * make this happen. */ if(force_arg && m && m->snarf.name) m->snarf.time = 0; /*-- Ping the stream to check for new mail --*/ if(sp_dead_stream(m)){ dprint((6, "no check: stream is dead\n")); } else if(!pine_mail_ping(m)){ dprint((6, "ping failed: stream is dead\n")); sp_set_dead_stream(m, 1); } dprint((7, "Ping complete: %s\n", debug_time(0,1))); if((flags & NM_STATUS_MSG) && pith_opt_newmail_check_cue) (*pith_opt_newmail_check_cue)(FALSE); } } while(i != started_on); /* * Current mail box state changed, could be additions or deletions. * Also check if we need to do sorting that has been deferred. * We handle the current stream separately from the rest in the * similar loop that follows this paragraph. */ m = ps_global->mail_stream; if(sp_mail_box_changed(m) || sp_unsorted_newmail(m) || sp_need_to_rethread(m)){ dprint((7, "Cur new mail, %s, new_mail_count: %ld expunge: %ld, max_msgno: %ld\n", (m && m->mailbox) ? m->mailbox : "?", sp_new_mail_count(m), sp_expunge_count(m), mn_get_total(sp_msgmap(m)))); new_mail_was_announced = 0; if(sp_mail_box_changed(m)) fixup_flags(m, sp_msgmap(m)); if(sp_new_mail_count(m)) process_filter_patterns(m, sp_msgmap(m), sp_new_mail_count(m)); /* worry about sorting */ if((sp_new_mail_count(m) > 0L || sp_unsorted_newmail(m) || sp_need_to_rethread(m)) && !((flags & NM_DEFER_SORT) || any_lflagged(sp_msgmap(m), MN_HIDE))) refresh_sort(m, sp_msgmap(m), (flags & NM_STATUS_MSG) ? SRT_VRB : SRT_NON); else if(sp_new_mail_count(m) > 0L) sp_set_unsorted_newmail(m, 1); if(sp_new_mail_count(m) > 0L){ sp_set_mail_since_cmd(m, sp_mail_since_cmd(m)+sp_new_mail_count(m)); rv += (t_nm_count = sp_new_mail_count(m)); sp_set_new_mail_count(m, 0L); if((flags & NM_STATUS_MSG) && pith_opt_newmail_announce){ for(n = m->nmsgs; n > 1L; n--) if(!get_lflag(m, NULL, n, MN_EXLD)) break; (*pith_opt_newmail_announce)(m, n, t_nm_count); if(n) new_mail_was_announced++; } } update_folder_unseen_by_stream(m, new_mail_was_announced ? UFU_NONE : UFU_ANNOUNCE); if(flags & NM_STATUS_MSG) sp_set_mail_box_changed(m, 0); } /* the rest of the streams */ for(i = 0; i < ps_global->s_pool.nstream; i++){ m = ps_global->s_pool.streams[i]; if(!m || m == ps_global->mail_stream) continue; if(sp_mail_box_changed(m)){ /*-- New mail for this stream, queue up the notification --*/ dprint((7, "New mail, %s, new_mail_count: %ld expunge: %ld, max_msgno: %ld\n", (m && m->mailbox) ? m->mailbox : "?", sp_new_mail_count(m), sp_expunge_count(m), mn_get_total(sp_msgmap(m)))); new_mail_was_announced = 0; fixup_flags(m, sp_msgmap(m)); if(sp_new_mail_count(m)) process_filter_patterns(m, sp_msgmap(m), sp_new_mail_count(m)); if(sp_new_mail_count(m) > 0){ sp_set_unsorted_newmail(m, 1); sp_set_mail_since_cmd(m, sp_mail_since_cmd(m) + sp_new_mail_count(m)); if(sp_flagged(m, SP_LOCKED) && sp_flagged(m, SP_USERFLDR)) rv += (t_nm_count = sp_new_mail_count(m)); sp_set_new_mail_count(m, 0L); /* messages only for user's streams */ if(flags & NM_STATUS_MSG && pith_opt_newmail_announce && sp_flagged(m, SP_LOCKED) && sp_flagged(m, SP_USERFLDR)){ for(n = m->nmsgs; n > 1L; n--) if(!get_lflag(m, NULL, n, MN_EXLD)) break; (*pith_opt_newmail_announce)(m, n, t_nm_count); if(n) new_mail_was_announced++; } } update_folder_unseen_by_stream(m, new_mail_was_announced ? UFU_NONE : UFU_ANNOUNCE); if(flags & NM_STATUS_MSG) sp_set_mail_box_changed(m, 0); } } /* so quit_screen can tell new mail from expunged mail */ exp_count = sp_expunge_count(ps_global->mail_stream); /* see if we want to kill any cached streams */ for(i = 0; i < ps_global->s_pool.nstream; i++){ m = ps_global->s_pool.streams[i]; if(sp_last_ping(m) >= now) maybe_kill_old_stream(m); } /* * This is here to prevent banging on the down arrow key (or holding * it down and repeating) at the end of the index from causing * a whole bunch of new mail checks. Last_nextitem_forcechk does get * set at the place it is tested in mailcmd.c, but this is here to * reset it to a little bit later in case it takes a second or more * to check for the mail. If we didn't do this, we'd just check every * keystroke as long as the checks took more than a second. */ if(force_arg) ps_global->last_nextitem_forcechk = time(0); dprint((6, "******** new mail returning %ld ********\n", rv ? rv : (exp_count ? 0 : -1))); return(rv ? rv : (exp_count ? 0 : -1)); }
/*---------------------------------------------------------------------- Accepts: stream -- mail stream to removed message references from msgs -- pointer to message manipulation struct flags MI_REFILTERING -- do includes appropriate for refiltering MI_STATECHGONLY -- when refiltering, maybe only re-include messages which have had state changes since they were originally filtered Returns 1 if any new messages are included (indicating that we need to re-sort) 0 if no new messages are included ----*/ int msgno_include(MAILSTREAM *stream, MSGNO_S *msgs, int flags) { long i, slop, old_total, old_size; int exbits, ret = 0; size_t len; MESSAGECACHE *mc; if(!msgs) return(ret); for(i = 1L; i <= stream->nmsgs; i++){ if(!msgno_exceptions(stream, i, "0", &exbits, FALSE)) exbits = 0; if((((flags & MI_REFILTERING) && (exbits & MSG_EX_FILTERED) && !(exbits & MSG_EX_FILED) && (!(flags & MI_STATECHGONLY) || (exbits & MSG_EX_STATECHG))) || (!(flags & MI_REFILTERING) && !(exbits & MSG_EX_FILTERED))) && get_lflag(stream, NULL, i, MN_EXLD)){ old_total = msgs->max_msgno; old_size = msgs->sort_size; slop = (msgs->max_msgno + 1L) % 64; msgs->sort_size = (msgs->max_msgno + 1L) + (64 - slop); len = (size_t) msgs->sort_size * sizeof(long); if(msgs->sort){ if(old_size != msgs->sort_size) fs_resize((void **)&(msgs->sort), len); } else msgs->sort = (long *)fs_get(len); ret = 1; msgs->sort[++msgs->max_msgno] = i; msgs->isort[i] = msgs->max_msgno; set_lflag(stream, msgs, msgs->max_msgno, MN_EXLD, 0); if(flags & MI_REFILTERING){ exbits &= ~(MSG_EX_FILTERED | MSG_EX_TESTED); msgno_exceptions(stream, i, "0", &exbits, TRUE); } if(old_total <= 0L){ /* if no previous messages, */ if(!msgs->select){ /* select the new message */ msgs->sel_size = 8L; len = (size_t)msgs->sel_size * sizeof(long); msgs->select = (long *)fs_get(len); } msgs->sel_cnt = 1L; msgs->sel_cur = 0L; msgs->select[0] = 1L; } } else if((flags & MI_REFILTERING) && (exbits & (MSG_EX_FILTERED | MSG_EX_TESTED)) && !(exbits & MSG_EX_FILED) && (!(exbits & MSG_EX_MANUNDEL) || ((mc = mail_elt(stream, i)) && mc->deleted)) && (!(flags & MI_STATECHGONLY) || (exbits & MSG_EX_STATECHG))){ /* * We get here if the message was filtered by a filter that * just changes status bits (it wasn't excluded), and now also * if the message was merely tested for filtering. It has also * not been manually undeleted. If it was manually undeleted, we * don't want to reprocess the filter, undoing the user's * manual undeleting. Of course, a new pine will re check this * message anyway, so the user had better be using this * manual undeleting only to temporarily save him or herself * from an expunge before Saving or printing or something. * Also, we want to still try filtering if the message has at * all been marked deleted, even if the there was any manual * undeleting, since this directly precedes an expunge, we want * to make sure the filter does the right thing before getting * rid of the message forever. */ exbits &= ~(MSG_EX_FILTERED | MSG_EX_TESTED); msgno_exceptions(stream, i, "0", &exbits, TRUE); } } return(ret); }