NeoMutt  2025-12-11-435-g4ac674
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
lib.h File Reference

POP network mailbox. More...

#include "core/lib.h"
+ Include dependency graph for lib.h:
+ This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Functions

void pop_fetch_mail (void)
 Fetch messages and save them in $spool_file.
 
enum MailboxType pop_path_probe (const char *path, const struct stat *st)
 Is this a POP Mailbox?
 

Variables

const struct MxOps MxPopOps
 POP Mailbox - Implements MxOps -.
 

Detailed Description

POP network mailbox.

Authors
  • Vsevolod Volkov
  • Richard Russon

This program 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 this program. If not, see http://www.gnu.org/licenses/.

Definition in file lib.h.

Function Documentation

◆ pop_fetch_mail()

void pop_fetch_mail ( void )

Fetch messages and save them in $spool_file.

Definition at line 517 of file pop.c.

518{
519 const char *const c_pop_host = cs_subset_string(NeoMutt->sub, "pop_host");
520 if (!c_pop_host)
521 {
522 mutt_error(_("POP host is not defined"));
523 return;
524 }
525
526 char buf[1024] = { 0 };
527 char msgbuf[128] = { 0 };
528 int last = 0, msgs = 0, bytes = 0, rset = 0, rc;
529 struct ConnAccount cac = { { 0 } };
530
531 char *p = MUTT_MEM_CALLOC(strlen(c_pop_host) + 7, char);
532 char *url = p;
533 if (url_check_scheme(c_pop_host) == U_UNKNOWN)
534 {
535 strcpy(url, "pop://");
536 p = strchr(url, '\0');
537 }
538 strcpy(p, c_pop_host);
539
540 rc = pop_parse_path(url, &cac);
541 FREE(&url);
542 if (rc)
543 {
544 mutt_error(_("%s is an invalid POP path"), c_pop_host);
545 return;
546 }
547
548 struct Connection *conn = mutt_conn_find(&cac);
549 if (!conn)
550 return;
551
553 adata->conn = conn;
554
555 if (pop_open_connection(adata) < 0)
556 {
557 pop_adata_free((void **) &adata);
558 return;
559 }
560
561 mutt_message(_("Checking for new messages..."));
562
563 /* find out how many messages are in the mailbox. */
564 mutt_str_copy(buf, "STAT\r\n", sizeof(buf));
565 rc = pop_query(adata, buf, sizeof(buf));
566 if (rc == -1)
567 goto fail;
568 if (rc == -2)
569 {
570 mutt_error("%s", adata->err_msg);
571 goto finish;
572 }
573
574 sscanf(buf, "+OK %d %d", &msgs, &bytes);
575
576 /* only get unread messages */
577 const bool c_pop_last = cs_subset_bool(NeoMutt->sub, "pop_last");
578 if ((msgs > 0) && c_pop_last)
579 {
580 mutt_str_copy(buf, "LAST\r\n", sizeof(buf));
581 rc = pop_query(adata, buf, sizeof(buf));
582 if (rc == -1)
583 goto fail;
584 if (rc == 0)
585 sscanf(buf, "+OK %d", &last);
586 }
587
588 if (msgs <= last)
589 {
590 mutt_message(_("No new mail in POP mailbox"));
591 goto finish;
592 }
593
594 const char *const c_spool_file = cs_subset_string(NeoMutt->sub, "spool_file");
595 struct Mailbox *m_spool = mx_path_resolve(c_spool_file);
596
597 if (!mx_mbox_open(m_spool, MUTT_OPEN_NO_FLAGS))
598 {
599 mailbox_free(&m_spool);
600 goto finish;
601 }
602 bool old_append = m_spool->append;
603 m_spool->append = true;
604
605 enum QuadOption delanswer = query_quadoption(_("Delete messages from server?"),
606 NeoMutt->sub, "pop_delete");
607
608 snprintf(msgbuf, sizeof(msgbuf),
609 ngettext("Reading new messages (%d byte)...",
610 "Reading new messages (%d bytes)...", bytes),
611 bytes);
612 mutt_message("%s", msgbuf);
613
614 for (int i = last + 1; i <= msgs; i++)
615 {
616 struct Message *msg = mx_msg_open_new(m_spool, NULL, MUTT_ADD_FROM);
617 if (msg)
618 {
619 snprintf(buf, sizeof(buf), "RETR %d\r\n", i);
620 rc = pop_fetch_data(adata, buf, NULL, fetch_message, msg->fp);
621 if (rc == -3)
622 rset = 1;
623
624 if ((rc == 0) && (mx_msg_commit(m_spool, msg) != 0))
625 {
626 rset = 1;
627 rc = -3;
628 }
629
630 mx_msg_close(m_spool, &msg);
631 }
632 else
633 {
634 rc = -3;
635 }
636
637 if ((rc == 0) && (delanswer == MUTT_YES))
638 {
639 /* delete the message on the server */
640 snprintf(buf, sizeof(buf), "DELE %d\r\n", i);
641 rc = pop_query(adata, buf, sizeof(buf));
642 }
643
644 if (rc == -1)
645 {
646 m_spool->append = old_append;
647 mx_mbox_close(m_spool);
648 goto fail;
649 }
650 if (rc == -2)
651 {
652 mutt_error("%s", adata->err_msg);
653 break;
654 }
655 if (rc == -3)
656 {
657 mutt_error(_("Error while writing mailbox"));
658 break;
659 }
660
661 /* L10N: The plural is picked by the second numerical argument, i.e.
662 the %d right before 'messages', i.e. the total number of messages. */
663 mutt_message(ngettext("%s [%d of %d message read]",
664 "%s [%d of %d messages read]", msgs - last),
665 msgbuf, i - last, msgs - last);
666 }
667
668 m_spool->append = old_append;
669 mx_mbox_close(m_spool);
670
671 if (rset)
672 {
673 /* make sure no messages get deleted */
674 mutt_str_copy(buf, "RSET\r\n", sizeof(buf));
675 if (pop_query(adata, buf, sizeof(buf)) == -1)
676 goto fail;
677 }
678
679finish:
680 /* exit gracefully */
681 mutt_str_copy(buf, "QUIT\r\n", sizeof(buf));
682 if (pop_query(adata, buf, sizeof(buf)) == -1)
683 goto fail;
684 mutt_socket_close(conn);
685 pop_adata_free((void **) &adata);
686 return;
687
688fail:
689 mutt_error(_("Server closed connection"));
690 mutt_socket_close(conn);
691 pop_adata_free((void **) &adata);
692}
const char * cs_subset_string(const struct ConfigSubset *sub, const char *name)
Get a string config item by name.
Definition helpers.c:291
bool cs_subset_bool(const struct ConfigSubset *sub, const char *name)
Get a boolean config item by name.
Definition helpers.c:47
void mailbox_free(struct Mailbox **ptr)
Free a Mailbox.
Definition mailbox.c:90
void pop_adata_free(void **ptr)
Free the private Account data - Implements Account::adata_free() -.
Definition adata.c:41
#define mutt_error(...)
Definition logging2.h:94
#define mutt_message(...)
Definition logging2.h:93
static int fetch_message(const char *line, void *data)
Parse a Message response - Implements pop_fetch_t -.
Definition pop.c:97
#define FREE(x)
Free memory and set the pointer to NULL.
Definition memory.h:68
#define MUTT_MEM_CALLOC(n, type)
Definition memory.h:52
#define _(a)
Definition message.h:28
size_t mutt_str_copy(char *dest, const char *src, size_t dsize)
Copy a string into a buffer (guaranteeing NUL-termination)
Definition string.c:583
struct Connection * mutt_conn_find(const struct ConnAccount *cac)
Find a connection from a list.
Definition mutt_socket.c:88
int mx_msg_close(struct Mailbox *m, struct Message **ptr)
Close a message.
Definition mx.c:1182
bool mx_mbox_open(struct Mailbox *m, OpenMailboxFlags flags)
Open a mailbox and parse it.
Definition mx.c:285
struct Message * mx_msg_open_new(struct Mailbox *m, const struct Email *e, MsgOpenFlags flags)
Open a new message.
Definition mx.c:1041
int mx_msg_commit(struct Mailbox *m, struct Message *msg)
Commit a message to a folder - Wrapper for MxOps::msg_commit()
Definition mx.c:1161
struct Mailbox * mx_path_resolve(const char *path)
Get a Mailbox for a path.
Definition mx.c:1647
enum MxStatus mx_mbox_close(struct Mailbox *m)
Save changes and close mailbox.
Definition mx.c:595
#define MUTT_ADD_FROM
add a From_ line
Definition mx.h:39
#define MUTT_OPEN_NO_FLAGS
No flags are set.
Definition mxapi.h:39
struct PopAccountData * pop_adata_new(void)
Create a new PopAccountData object.
Definition adata.c:63
int pop_open_connection(struct PopAccountData *adata)
Open connection and authenticate.
Definition lib.c:316
int pop_parse_path(const char *path, struct ConnAccount *cac)
Parse a POP mailbox name.
Definition lib.c:82
int pop_fetch_data(struct PopAccountData *adata, const char *query, struct Progress *progress, pop_fetch_t callback, void *data)
Read Headers with callback function.
Definition lib.c:511
#define pop_query(adata, buf, buflen)
Definition private.h:109
QuadOption
Possible values for a quad-option.
Definition quad.h:36
@ MUTT_YES
User answered 'Yes', or assume 'Yes'.
Definition quad.h:39
enum QuadOption query_quadoption(const char *prompt, struct ConfigSubset *sub, const char *name)
Ask the user a quad-question.
Definition question.c:378
int mutt_socket_close(struct Connection *conn)
Close a socket.
Definition socket.c:100
void * adata
Private data (for Mailbox backends)
Definition account.h:42
Login details for a remote server.
Definition connaccount.h:53
A mailbox.
Definition mailbox.h:78
bool append
Mailbox is opened in append mode.
Definition mailbox.h:108
A local copy of an email.
Definition message.h:34
FILE * fp
pointer to the message data
Definition message.h:35
Container for Accounts, Notifications.
Definition neomutt.h:41
struct ConfigSubset * sub
Inherited config items.
Definition neomutt.h:49
POP-specific Account data -.
Definition adata.h:37
char err_msg[POP_CMD_RESPONSE]
Last error message.
Definition adata.h:57
struct Connection * conn
Connection to POP server.
Definition adata.h:38
enum UrlScheme url_check_scheme(const char *str)
Check the protocol of a URL.
Definition url.c:226
@ U_UNKNOWN
Url wasn't recognised.
Definition url.h:35
+ Here is the call graph for this function:
+ Here is the caller graph for this function: