NeoMutt  2025-12-11-860-g80c9cc
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
newsrc.c File Reference

Read/parse/write an NNTP config file of subscribed newsgroups. More...

#include "config.h"
#include <dirent.h>
#include <errno.h>
#include <limits.h>
#include <stdbool.h>
#include <stdio.h>
#include <string.h>
#include <sys/stat.h>
#include <time.h>
#include <unistd.h>
#include "private.h"
#include "mutt/lib.h"
#include "config/lib.h"
#include "email/lib.h"
#include "core/lib.h"
#include "conn/lib.h"
#include "mutt.h"
#include "lib.h"
#include "bcache/lib.h"
#include "expando/lib.h"
#include "store/lib.h"
#include "adata.h"
#include "edata.h"
#include "expando_newsrc.h"
#include "mdata.h"
#include "module_data.h"
#include "mutt_logging.h"
#include "mutt_socket.h"
#include "muttlib.h"
#include "hcache/lib.h"
+ Include dependency graph for newsrc.c:

Go to the source code of this file.

Functions

static struct NntpMboxDatamdata_find (struct NntpAccountData *adata, const char *group)
 Find NntpMboxData for given newsgroup or add it.
 
void nntp_acache_free (struct NntpMboxData *mdata)
 Remove all temporarily cache files.
 
void nntp_newsrc_close (struct NntpAccountData *adata)
 Unlock and close .newsrc file.
 
void nntp_group_unread_stat (struct NntpMboxData *mdata)
 Count number of unread articles using .newsrc data.
 
int nntp_newsrc_parse (struct NntpAccountData *adata)
 Parse .newsrc file.
 
void nntp_newsrc_gen_entries (struct Mailbox *m)
 Generate array of .newsrc entries.
 
static int update_file (char *filename, char *buf)
 Update file with new contents.
 
int nntp_newsrc_update (struct NntpAccountData *adata)
 Update .newsrc file.
 
static void cache_expand (char *dst, size_t dstlen, struct ConnAccount *cac, const char *src)
 Make fully qualified cache file name.
 
void nntp_expand_path (char *buf, size_t buflen, struct ConnAccount *cac)
 Make fully qualified url from newsgroup name.
 
int nntp_add_group (char *line, void *data)
 Parse newsgroup.
 
static int active_get_cache (struct NntpAccountData *adata)
 Load list of all newsgroups from cache.
 
int nntp_active_save_cache (struct NntpAccountData *adata)
 Save list of all newsgroups to cache.
 
static void nntp_hcache_namer (const struct StoreOps *store_ops, const char *path, struct Buffer *dest)
 Compose hcache file names - Implements hcache_namer_t -.
 
struct HeaderCachenntp_hcache_open (struct NntpMboxData *mdata)
 Open newsgroup hcache.
 
void nntp_hcache_update (struct NntpMboxData *mdata, struct HeaderCache *hc)
 Remove stale cached headers.
 
static int nntp_bcache_delete (const char *id, struct BodyCache *bcache, void *data)
 Delete an entry from the message cache - Implements bcache_list_t -.
 
void nntp_bcache_update (struct NntpMboxData *mdata)
 Remove stale cached messages.
 
void nntp_delete_group_cache (struct NntpMboxData *mdata)
 Remove hcache and bcache of newsgroup.
 
void nntp_clear_cache (struct NntpAccountData *adata)
 Clear the NNTP cache.
 
static const char * nntp_get_field (enum ConnAccountField field, void *gf_data)
 Get connection login credentials - Implements ConnAccount::get_field() -.
 
struct NntpAccountDatanntp_select_server (struct Mailbox *m, const char *server, bool leave_lock)
 Open a connection to an NNTP server.
 
void nntp_article_status (struct Mailbox *m, struct Email *e, char *group, anum_t anum)
 Get status of articles from .newsrc.
 
struct NntpMboxDatamutt_newsgroup_subscribe (struct NntpAccountData *adata, char *group)
 Subscribe newsgroup.
 
struct NntpMboxDatamutt_newsgroup_unsubscribe (struct NntpAccountData *adata, char *group)
 Unsubscribe newsgroup.
 
struct NntpMboxDatamutt_newsgroup_catchup (struct Mailbox *m, struct NntpAccountData *adata, char *group)
 Catchup newsgroup.
 
struct NntpMboxDatamutt_newsgroup_uncatchup (struct Mailbox *m, struct NntpAccountData *adata, char *group)
 Uncatchup newsgroup.
 
void nntp_mailbox (struct Mailbox *m, char *buf, size_t buflen)
 Get first newsgroup with new messages.
 

Detailed Description

Read/parse/write an NNTP config file of subscribed newsgroups.

Authors
  • Richard Russon
  • Pietro Cerutti
  • Ian Zimmerman
  • Ramkumar Ramachandra
  • Tóth János

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 newsrc.c.

Function Documentation

◆ mdata_find()

static struct NntpMboxData * mdata_find ( struct NntpAccountData * adata,
const char * group )
static

Find NntpMboxData for given newsgroup or add it.

Parameters
adataNNTP server
groupNewsgroup
Return values
ptrNNTP data
NULLError

Definition at line 75 of file newsrc.c.

76{
78 if (mdata)
79 return mdata;
80
81 size_t len = strlen(group) + 1;
82 /* create NntpMboxData structure and add it to hash */
83 mdata = mutt_mem_calloc(1, sizeof(struct NntpMboxData) + len);
84 mdata->group = (char *) mdata + sizeof(struct NntpMboxData);
85 mutt_str_copy(mdata->group, group, len);
86 mdata->adata = adata;
87 mdata->deleted = true;
89
90 /* add NntpMboxData to list */
92 {
93 adata->groups_max *= 2;
95 }
97
98 return mdata;
99}
struct HashElem * mutt_hash_insert(struct HashTable *table, const char *strkey, void *data)
Add a new element to the Hash Table (with string keys)
Definition hash.c:337
void * mutt_hash_find(const struct HashTable *table, const char *strkey)
Find the HashElem data in a Hash Table element using a key.
Definition hash.c:364
void * mutt_mem_calloc(size_t nmemb, size_t size)
Allocate zeroed memory on the heap.
Definition memory.c:76
#define MUTT_MEM_REALLOC(pptr, n, type)
Definition memory.h:55
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:586
void * mdata
Driver specific data.
Definition mailbox.h:131
struct HashTable * groups_hash
Hash Table: "newsgroup" -> NntpMboxData.
Definition adata.h:62
struct NntpMboxData ** groups_list
List of newsgroups.
Definition adata.h:60
unsigned int groups_num
Number of newsgroups.
Definition adata.h:58
unsigned int groups_max
Maximum number of newsgroups.
Definition adata.h:59
NNTP-specific Mailbox data -.
Definition mdata.h:34
char * group
Name of newsgroup.
Definition mdata.h:35
struct NntpAccountData * adata
Account data.
Definition mdata.h:48
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_acache_free()

void nntp_acache_free ( struct NntpMboxData * mdata)

Remove all temporarily cache files.

Parameters
mdataNNTP Mailbox data

Definition at line 105 of file newsrc.c.

106{
107 for (int i = 0; i < NNTP_ACACHE_LEN; i++)
108 {
109 if (mdata->acache[i].path)
110 {
111 unlink(mdata->acache[i].path);
112 FREE(&mdata->acache[i].path);
113 }
114 }
115}
#define FREE(x)
Free memory and set the pointer to NULL.
Definition memory.h:68
#define NNTP_ACACHE_LEN
Definition lib.h:85
char * path
Cache path.
Definition lib.h:72
struct NntpAcache acache[NNTP_ACACHE_LEN]
Article cache.
Definition mdata.h:49
+ Here is the caller graph for this function:

◆ nntp_newsrc_close()

void nntp_newsrc_close ( struct NntpAccountData * adata)

Unlock and close .newsrc file.

Parameters
adataNNTP server

Definition at line 121 of file newsrc.c.

122{
123 if (!adata->fp_newsrc)
124 return;
125
126 mutt_debug(LL_DEBUG1, "Unlocking %s\n", adata->newsrc_file);
127 mutt_file_unlock(fileno(adata->fp_newsrc));
129}
int mutt_file_unlock(int fd)
Unlock a file previously locked by mutt_file_lock()
Definition file.c:1135
#define mutt_file_fclose(FP)
Definition file.h:139
#define mutt_debug(LEVEL,...)
Definition logging2.h:91
@ LL_DEBUG1
Log at debug level 1.
Definition logging2.h:45
FILE * fp_newsrc
Newsrc file handle.
Definition adata.h:50
char * newsrc_file
Newsrc file path.
Definition adata.h:51
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_group_unread_stat()

void nntp_group_unread_stat ( struct NntpMboxData * mdata)

Count number of unread articles using .newsrc data.

Parameters
mdataNNTP Mailbox data

Definition at line 135 of file newsrc.c.

136{
137 mdata->unread = 0;
138 if ((mdata->last_message == 0) ||
139 (mdata->first_message > mdata->last_message) || !mdata->newsrc_ent)
140 {
141 return;
142 }
143
144 mdata->unread = mdata->last_message - mdata->first_message + 1;
145 for (unsigned int i = 0; i < mdata->newsrc_len; i++)
146 {
147 anum_t first = mdata->newsrc_ent[i].first;
148 if (first < mdata->first_message)
149 first = mdata->first_message;
150 anum_t last = mdata->newsrc_ent[i].last;
151 if (last > mdata->last_message)
152 last = mdata->last_message;
153 if (first <= last)
154 mdata->unread -= last - first + 1;
155 }
156}
#define anum_t
Definition lib.h:63
anum_t last
Last article number in run.
Definition lib.h:81
anum_t first
First article number in run.
Definition lib.h:80
anum_t last_message
Last article number.
Definition mdata.h:38
struct NewsrcEntry * newsrc_ent
Newsrc entries.
Definition mdata.h:47
anum_t unread
Unread articles.
Definition mdata.h:41
unsigned int newsrc_len
Length of newsrc entry.
Definition mdata.h:46
anum_t first_message
First article number.
Definition mdata.h:37
+ Here is the caller graph for this function:

◆ nntp_newsrc_parse()

int nntp_newsrc_parse ( struct NntpAccountData * adata)

Parse .newsrc file.

Parameters
adataNNTP server
Return values
0Not changed
1Parsed
-1Error

Definition at line 165 of file newsrc.c.

166{
167 if (!adata)
168 return -1;
169
170 char *line = NULL;
171 struct stat st = { 0 };
172
173 if (adata->fp_newsrc)
174 {
175 /* if we already have a handle, close it and reopen */
177 }
178 else
179 {
180 /* if file doesn't exist, create it */
181 adata->fp_newsrc = mutt_file_fopen(adata->newsrc_file, "a");
183 }
184
185 /* open .newsrc */
186 adata->fp_newsrc = mutt_file_fopen(adata->newsrc_file, "r");
187 if (!adata->fp_newsrc)
188 {
189 mutt_perror("%s", adata->newsrc_file);
190 return -1;
191 }
192
193 /* lock it */
194 mutt_debug(LL_DEBUG1, "Locking %s\n", adata->newsrc_file);
195 if (mutt_file_lock(fileno(adata->fp_newsrc), false, true))
196 {
198 return -1;
199 }
200
201 if (stat(adata->newsrc_file, &st) != 0)
202 {
203 mutt_perror("%s", adata->newsrc_file);
204 nntp_newsrc_close(adata);
205 return -1;
206 }
207
208 if ((adata->size == st.st_size) && (adata->mtime == st.st_mtime))
209 return 0;
210
211 adata->size = st.st_size;
212 adata->mtime = st.st_mtime;
213 adata->newsrc_modified = true;
214 mutt_debug(LL_DEBUG1, "Parsing %s\n", adata->newsrc_file);
215
216 /* .newsrc has been externally modified or hasn't been loaded yet */
217 for (unsigned int i = 0; i < adata->groups_num; i++)
218 {
219 struct NntpMboxData *mdata = adata->groups_list[i];
220 if (!mdata)
221 continue;
222
223 mdata->subscribed = false;
224 mdata->newsrc_len = 0;
225 FREE(&mdata->newsrc_ent);
226 }
227
228 line = MUTT_MEM_MALLOC(st.st_size + 1, char);
229 while (st.st_size && fgets(line, st.st_size + 1, adata->fp_newsrc))
230 {
231 char *b = NULL, *h = NULL;
232 unsigned int j = 1;
233 bool subs = false;
234
235 /* find end of newsgroup name */
236 char *p = strpbrk(line, ":!");
237 if (!p)
238 continue;
239
240 /* ":" - subscribed, "!" - unsubscribed */
241 if (*p == ':')
242 subs = true;
243 *p++ = '\0';
244
245 /* get newsgroup data */
246 struct NntpMboxData *mdata = mdata_find(adata, line);
247 FREE(&mdata->newsrc_ent);
248
249 /* count number of entries */
250 b = p;
251 while (*b)
252 if (*b++ == ',')
253 j++;
254 mdata->newsrc_ent = MUTT_MEM_CALLOC(j, struct NewsrcEntry);
255 mdata->subscribed = subs;
256
257 /* parse entries */
258 j = 0;
259 while (p)
260 {
261 b = p;
262
263 /* find end of entry */
264 p = strchr(p, ',');
265 if (p)
266 *p++ = '\0';
267
268 /* first-last or single number */
269 h = strchr(b, '-');
270 if (h)
271 *h++ = '\0';
272 else
273 h = b;
274
275 if ((sscanf(b, ANUM_FMT, &mdata->newsrc_ent[j].first) == 1) &&
276 (sscanf(h, ANUM_FMT, &mdata->newsrc_ent[j].last) == 1))
277 {
278 j++;
279 }
280 }
281 if (j == 0)
282 {
283 mdata->newsrc_ent[j].first = 1;
284 mdata->newsrc_ent[j].last = 0;
285 j++;
286 }
287 if (mdata->last_message == 0)
288 mdata->last_message = mdata->newsrc_ent[j - 1].last;
289 mdata->newsrc_len = j;
290 MUTT_MEM_REALLOC(&mdata->newsrc_ent, j, struct NewsrcEntry);
292 mutt_debug(LL_DEBUG2, "%s\n", mdata->group);
293 }
294 FREE(&line);
295 return 1;
296}
int mutt_file_lock(int fd, bool excl, bool timeout)
(Try to) Lock a file using fcntl()
Definition file.c:1088
#define mutt_file_fopen(PATH, MODE)
Definition file.h:138
#define mutt_perror(...)
Definition logging2.h:95
@ LL_DEBUG2
Log at debug level 2.
Definition logging2.h:46
#define MUTT_MEM_CALLOC(n, type)
Definition memory.h:52
#define MUTT_MEM_MALLOC(n, type)
Definition memory.h:53
void nntp_newsrc_close(struct NntpAccountData *adata)
Unlock and close .newsrc file.
Definition newsrc.c:121
static struct NntpMboxData * mdata_find(struct NntpAccountData *adata, const char *group)
Find NntpMboxData for given newsgroup or add it.
Definition newsrc.c:75
void nntp_group_unread_stat(struct NntpMboxData *mdata)
Count number of unread articles using .newsrc data.
Definition newsrc.c:135
#define ANUM_FMT
Definition lib.h:64
An entry in a .newsrc (subscribed newsgroups)
Definition lib.h:79
bool newsrc_modified
Newsrc file was modified.
Definition adata.h:49
off_t size
Newsrc file size.
Definition adata.h:54
time_t mtime
Newsrc modification time.
Definition adata.h:55
bool subscribed
Subscribed to this newsgroup.
Definition mdata.h:42
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_newsrc_gen_entries()

void nntp_newsrc_gen_entries ( struct Mailbox * m)

Generate array of .newsrc entries.

Parameters
mMailbox

Definition at line 302 of file newsrc.c.

303{
304 if (!m)
305 return;
306
307 struct NntpMboxData *mdata = m->mdata;
308 anum_t last = 0, first = 1;
309 bool series;
310 unsigned int entries;
311
312 const enum EmailSortType c_sort = cs_subset_sort(NeoMutt->sub, "sort");
313 if (c_sort != EMAIL_SORT_UNSORTED)
314 {
317 }
318
319 entries = mdata->newsrc_len;
320 if (!entries)
321 {
322 entries = 5;
323 mdata->newsrc_ent = MUTT_MEM_CALLOC(entries, struct NewsrcEntry);
324 }
325
326 /* Set up to fake initial sequence from 1 to the article before the
327 * first article in our list */
328 mdata->newsrc_len = 0;
329 series = true;
330 for (int i = 0; i < m->msg_count; i++)
331 {
332 struct Email *e = m->emails[i];
333 if (!e)
334 break;
335
336 /* search for first unread */
337 if (series)
338 {
339 /* We don't actually check sequential order, since we mark
340 * "missing" entries as read/deleted */
341 last = nntp_edata_get(e)->article_num;
342 if ((last >= mdata->first_message) && !e->deleted && !e->read)
343 {
344 if (mdata->newsrc_len >= entries)
345 {
346 entries *= 2;
347 MUTT_MEM_REALLOC(&mdata->newsrc_ent, entries, struct NewsrcEntry);
348 }
349 mdata->newsrc_ent[mdata->newsrc_len].first = first;
350 mdata->newsrc_ent[mdata->newsrc_len].last = last - 1;
351 mdata->newsrc_len++;
352 series = false;
353 }
354 }
355 else
356 {
357 /* search for first read */
358 if (e->deleted || e->read)
359 {
360 first = last + 1;
361 series = true;
362 }
363 last = nntp_edata_get(e)->article_num;
364 }
365 }
366
367 if (series && (first <= mdata->last_loaded))
368 {
369 if (mdata->newsrc_len >= entries)
370 {
371 entries++;
372 MUTT_MEM_REALLOC(&mdata->newsrc_ent, entries, struct NewsrcEntry);
373 }
374 mdata->newsrc_ent[mdata->newsrc_len].first = first;
375 mdata->newsrc_ent[mdata->newsrc_len].last = mdata->last_loaded;
376 mdata->newsrc_len++;
377 }
378 MUTT_MEM_REALLOC(&mdata->newsrc_ent, mdata->newsrc_len, struct NewsrcEntry);
379
380 if (c_sort != EMAIL_SORT_UNSORTED)
381 {
382 cs_subset_str_native_set(NeoMutt->sub, "sort", c_sort, NULL);
384 }
385}
short cs_subset_sort(const struct ConfigSubset *sub, const char *name)
Get a sort config item by name.
Definition helpers.c:266
void mailbox_changed(struct Mailbox *m, enum NotifyMailbox action)
Notify observers of a change to a Mailbox.
Definition mailbox.c:232
@ NT_MAILBOX_RESORT
Email list needs resorting.
Definition mailbox.h:180
EmailSortType
Methods for sorting Emails.
Definition sort.h:53
@ EMAIL_SORT_UNSORTED
Sort by the order the messages appear in the mailbox.
Definition sort.h:64
struct NntpEmailData * nntp_edata_get(struct Email *e)
Get the private data for this Email.
Definition edata.c:60
The envelope/body of an email.
Definition email.h:39
bool read
Email is read.
Definition email.h:50
bool deleted
Email is deleted.
Definition email.h:78
int msg_count
Total number of messages.
Definition mailbox.h:87
struct Email ** emails
Array of Emails.
Definition mailbox.h:95
Container for Accounts, Notifications.
Definition neomutt.h:41
struct ConfigSubset * sub
Inherited config items.
Definition neomutt.h:49
anum_t article_num
NNTP article number.
Definition edata.h:36
anum_t last_loaded
Last loaded article.
Definition mdata.h:39
int cs_subset_str_native_set(const struct ConfigSubset *sub, const char *name, intptr_t value, struct Buffer *err)
Natively set the value of a string config item.
Definition subset.c:303
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ update_file()

static int update_file ( char * filename,
char * buf )
static

Update file with new contents.

Parameters
filenameFile to update
bufNew context
Return values
0Success
-1Failure

Definition at line 394 of file newsrc.c.

395{
396 FILE *fp = NULL;
397 char tempfile[PATH_MAX] = { 0 };
398 int rc = -1;
399
400 while (true)
401 {
402 snprintf(tempfile, sizeof(tempfile), "%s.tmp", filename);
403 fp = mutt_file_fopen(tempfile, "w");
404 if (!fp)
405 {
406 mutt_perror("%s", tempfile);
407 *tempfile = '\0';
408 break;
409 }
410 if (fputs(buf, fp) == EOF)
411 {
412 mutt_perror("%s", tempfile);
413 break;
414 }
415 if (mutt_file_fclose(&fp) == EOF)
416 {
417 mutt_perror("%s", tempfile);
418 fp = NULL;
419 break;
420 }
421 fp = NULL;
422 if (rename(tempfile, filename) < 0)
423 {
424 mutt_perror("%s", filename);
425 break;
426 }
427 *tempfile = '\0';
428 rc = 0;
429 break;
430 }
431 mutt_file_fclose(&fp);
432
433 if (*tempfile)
434 unlink(tempfile);
435 return rc;
436}
#define PATH_MAX
Definition mutt.h:49
+ Here is the caller graph for this function:

◆ nntp_newsrc_update()

int nntp_newsrc_update ( struct NntpAccountData * adata)

Update .newsrc file.

Parameters
adataNNTP server
Return values
0Success
-1Failure

Definition at line 444 of file newsrc.c.

445{
446 if (!adata)
447 return -1;
448
449 int rc = -1;
450
451 size_t buflen = 10240;
452 char *buf = MUTT_MEM_CALLOC(buflen, char);
453 size_t off = 0;
454
455 /* we will generate full newsrc here */
456 for (unsigned int i = 0; i < adata->groups_num; i++)
457 {
458 struct NntpMboxData *mdata = adata->groups_list[i];
459
460 if (!mdata || !mdata->newsrc_ent)
461 continue;
462
463 /* write newsgroup name */
464 if ((off + strlen(mdata->group) + 3) > buflen)
465 {
466 buflen *= 2;
467 MUTT_MEM_REALLOC(&buf, buflen, char);
468 }
469 snprintf(buf + off, buflen - off, "%s%c ", mdata->group, mdata->subscribed ? ':' : '!');
470 off += strlen(buf + off);
471
472 /* write entries */
473 for (unsigned int j = 0; j < mdata->newsrc_len; j++)
474 {
475 if ((off + 1024) > buflen)
476 {
477 buflen *= 2;
478 MUTT_MEM_REALLOC(&buf, buflen, char);
479 }
480 if (j)
481 buf[off++] = ',';
482 if (mdata->newsrc_ent[j].first == mdata->newsrc_ent[j].last)
483 {
484 snprintf(buf + off, buflen - off, ANUM_FMT, mdata->newsrc_ent[j].first);
485 }
486 else if (mdata->newsrc_ent[j].first < mdata->newsrc_ent[j].last)
487 {
488 snprintf(buf + off, buflen - off, ANUM_FMT "-" ANUM_FMT,
489 mdata->newsrc_ent[j].first, mdata->newsrc_ent[j].last);
490 }
491 off += strlen(buf + off);
492 }
493 buf[off++] = '\n';
494 }
495 buf[off] = '\0';
496
497 /* newrc being fully rewritten */
498 mutt_debug(LL_DEBUG1, "Updating %s\n", adata->newsrc_file);
499 if (adata->newsrc_file && (update_file(adata->newsrc_file, buf) == 0))
500 {
501 struct stat st = { 0 };
502
503 rc = stat(adata->newsrc_file, &st);
504 if (rc == 0)
505 {
506 adata->size = st.st_size;
507 adata->mtime = st.st_mtime;
508 }
509 else
510 {
511 mutt_perror("%s", adata->newsrc_file);
512 }
513 }
514 FREE(&buf);
515 return rc;
516}
static int update_file(char *filename, char *buf)
Update file with new contents.
Definition newsrc.c:394
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ cache_expand()

static void cache_expand ( char * dst,
size_t dstlen,
struct ConnAccount * cac,
const char * src )
static

Make fully qualified cache file name.

Parameters
dstBuffer for filename
dstlenLength of buffer
cacAccount
srcPath to add to the URL

Definition at line 525 of file newsrc.c.

526{
527 char file[PATH_MAX] = { 0 };
528
529 /* server subdirectory */
530 struct Url url = { 0 };
531 account_to_url(cac, &url);
532 url.path = mutt_str_dup(src);
533 url_tostring(&url, file, sizeof(file), U_PATH);
534 FREE(&url.path);
535
536 /* remove trailing slash */
537 char *c = file + strlen(file) - 1;
538 if (*c == '/')
539 *c = '\0';
540
541 struct Buffer *tmp = buf_pool_get();
542 buf_addstr(tmp, file);
543 mutt_encode_path(tmp, file);
544
545 const char *const c_news_cache_dir = cs_subset_path(NeoMutt->sub, "news_cache_dir");
546 snprintf(dst, dstlen, "%s/%s", c_news_cache_dir, buf_string(tmp));
547
548 buf_pool_release(&tmp);
549}
size_t buf_addstr(struct Buffer *buf, const char *s)
Add a string to a Buffer.
Definition buffer.c:226
static const char * buf_string(const struct Buffer *buf)
Convert a buffer to a const char * "string".
Definition buffer.h:96
const char * cs_subset_path(const struct ConfigSubset *sub, const char *name)
Get a path config item by name.
Definition helpers.c:168
char * mutt_str_dup(const char *str)
Copy a string, safely.
Definition string.c:257
void account_to_url(struct ConnAccount *cac, struct Url *url)
Fill URL with info from account.
void mutt_encode_path(struct Buffer *buf, const char *src)
Convert a path to 'us-ascii'.
Definition muttlib.c:803
struct Buffer * buf_pool_get(void)
Get a Buffer from the pool.
Definition pool.c:91
void buf_pool_release(struct Buffer **ptr)
Return a Buffer to the pool.
Definition pool.c:111
String manipulation buffer.
Definition buffer.h:36
A parsed URL proto://user:password@host:port/path?a=1&b=2
Definition url.h:69
char * src
Raw URL string.
Definition url.h:77
char * path
Path.
Definition url.h:75
int url_tostring(const struct Url *url, char *dest, size_t len, uint8_t flags)
Output the URL string for a given Url object.
Definition url.c:426
#define U_PATH
Path is included in URL.
Definition url.h:50
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_expand_path()

void nntp_expand_path ( char * buf,
size_t buflen,
struct ConnAccount * cac )

Make fully qualified url from newsgroup name.

Parameters
bufBuffer for the result
buflenLength of buffer
cacAccount to serialise

Definition at line 557 of file newsrc.c.

558{
559 struct Url url = { 0 };
560
561 account_to_url(cac, &url);
562 url.path = mutt_str_dup(buf);
563 url_tostring(&url, buf, buflen, U_NO_FLAGS);
564 FREE(&url.path);
565}
#define U_NO_FLAGS
No flags are set for URL parsing.
Definition url.h:49
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_add_group()

int nntp_add_group ( char * line,
void * data )

Parse newsgroup.

Parameters
lineString to parse
dataNNTP data
Return values
0Always

Definition at line 573 of file newsrc.c.

574{
575 struct NntpAccountData *adata = data;
576 struct NntpMboxData *mdata = NULL;
577 char group[1024] = { 0 };
578 char desc[8192] = { 0 };
579 char mod = '\0';
580 anum_t first = 0, last = 0;
581
582 if (!adata || !line)
583 return 0;
584
585 /* These sscanf limits must match the sizes of the group and desc arrays */
586 if (sscanf(line, "%1023s " ANUM_FMT " " ANUM_FMT " %c %8191[^\n]", group,
587 &last, &first, &mod, desc) < 4)
588 {
589 mutt_debug(LL_DEBUG2, "Can't parse server line: %s\n", line);
590 return 0;
591 }
592
594 mdata->deleted = false;
595 mdata->first_message = first;
596 mdata->last_message = last;
597 mdata->allowed = (mod == 'y') || (mod == 'm');
598 mutt_str_replace(&mdata->desc, desc);
599 if (mdata->newsrc_ent || (mdata->last_cached != 0))
601 else if (mdata->last_message && (mdata->first_message <= mdata->last_message))
602 mdata->unread = mdata->last_message - mdata->first_message + 1;
603 else
604 mdata->unread = 0;
605 return 0;
606}
char * mutt_str_replace(char **p, const char *s)
Replace one string with another.
Definition string.c:284
void * adata
Private data (for Mailbox backends)
Definition account.h:42
NNTP-specific Account data -.
Definition adata.h:36
char * desc
Description of newsgroup.
Definition mdata.h:36
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ active_get_cache()

static int active_get_cache ( struct NntpAccountData * adata)
static

Load list of all newsgroups from cache.

Parameters
adataNNTP server
Return values
0Success
-1Failure

Definition at line 614 of file newsrc.c.

615{
616 char buf[8192] = { 0 };
617 char file[4096] = { 0 };
618 time_t t = 0;
619
620 cache_expand(file, sizeof(file), &adata->conn->account, ".active");
621 mutt_debug(LL_DEBUG1, "Parsing %s\n", file);
622 FILE *fp = mutt_file_fopen(file, "r");
623 if (!fp)
624 return -1;
625
626 if (!fgets(buf, sizeof(buf), fp) || (sscanf(buf, "%jd%4095s", &t, file) != 1) || (t == 0))
627 {
628 mutt_file_fclose(&fp);
629 return -1;
630 }
631 adata->newgroups_time = t;
632
633 mutt_message(_("Loading list of groups from cache..."));
634 while (fgets(buf, sizeof(buf), fp))
635 nntp_add_group(buf, adata);
636 nntp_add_group(NULL, NULL);
637 mutt_file_fclose(&fp);
639 return 0;
640}
#define mutt_message(...)
Definition logging2.h:93
#define _(a)
Definition message.h:28
void mutt_clear_error(void)
Clear the message line (bottom line of screen)
int nntp_add_group(char *line, void *data)
Parse newsgroup.
Definition newsrc.c:573
static void cache_expand(char *dst, size_t dstlen, struct ConnAccount *cac, const char *src)
Make fully qualified cache file name.
Definition newsrc.c:525
struct ConnAccount account
Account details: username, password, etc.
Definition connection.h:49
time_t newgroups_time
Last newgroups request time.
Definition adata.h:56
struct Connection * conn
Connection to NNTP Server.
Definition adata.h:63
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_active_save_cache()

int nntp_active_save_cache ( struct NntpAccountData * adata)

Save list of all newsgroups to cache.

Parameters
adataNNTP server
Return values
0Success
-1Failure

Definition at line 648 of file newsrc.c.

649{
650 if (!adata->cacheable)
651 return 0;
652
653 size_t buflen = 10240;
654 char *buf = MUTT_MEM_CALLOC(buflen, char);
655 snprintf(buf, buflen, "%lu\n", (unsigned long) adata->newgroups_time);
656 size_t off = strlen(buf);
657
658 for (unsigned int i = 0; i < adata->groups_num; i++)
659 {
660 struct NntpMboxData *mdata = adata->groups_list[i];
661
662 if (!mdata || mdata->deleted)
663 continue;
664
665 if ((off + strlen(mdata->group) + (mdata->desc ? strlen(mdata->desc) : 0) + 50) > buflen)
666 {
667 buflen *= 2;
668 MUTT_MEM_REALLOC(&buf, buflen, char);
669 }
670 snprintf(buf + off, buflen - off, "%s " ANUM_FMT " " ANUM_FMT " %c%s%s\n",
671 mdata->group, mdata->last_message, mdata->first_message,
672 mdata->allowed ? 'y' : 'n', mdata->desc ? " " : "",
673 mdata->desc ? mdata->desc : "");
674 off += strlen(buf + off);
675 }
676
677 char file[PATH_MAX] = { 0 };
678 cache_expand(file, sizeof(file), &adata->conn->account, ".active");
679 mutt_debug(LL_DEBUG1, "Updating %s\n", file);
680 int rc = update_file(file, buf);
681 FREE(&buf);
682 return rc;
683}
bool cacheable
Can be cached.
Definition adata.h:48
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_hcache_open()

struct HeaderCache * nntp_hcache_open ( struct NntpMboxData * mdata)

Open newsgroup hcache.

Parameters
mdataNNTP Mailbox data
Return values
ptrHeader cache
NULLError

Definition at line 709 of file newsrc.c.

710{
711 struct Url url = { 0 };
712 char file[PATH_MAX] = { 0 };
713
714 const bool c_save_unsubscribed = cs_subset_bool(NeoMutt->sub, "save_unsubscribed");
715 if (!mdata->adata || !mdata->adata->cacheable || !mdata->adata->conn || !mdata->group ||
716 !(mdata->newsrc_ent || mdata->subscribed || c_save_unsubscribed))
717 {
718 return NULL;
719 }
720
721 account_to_url(&mdata->adata->conn->account, &url);
722 url.path = mdata->group;
723 url_tostring(&url, file, sizeof(file), U_PATH);
724 const char *const c_news_cache_dir = cs_subset_path(NeoMutt->sub, "news_cache_dir");
725 return hcache_open(c_news_cache_dir, file, nntp_hcache_namer, true);
726}
bool cs_subset_bool(const struct ConfigSubset *sub, const char *name)
Get a boolean config item by name.
Definition helpers.c:47
static void nntp_hcache_namer(const struct StoreOps *store_ops, const char *path, struct Buffer *dest)
Compose hcache file names - Implements hcache_namer_t -.
Definition newsrc.c:689
struct HeaderCache * hcache_open(const char *path, const char *folder, hcache_namer_t namer, bool create)
Multiplexor for StoreOps::open.
Definition hcache.c:477
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_hcache_update()

void nntp_hcache_update ( struct NntpMboxData * mdata,
struct HeaderCache * hc )

Remove stale cached headers.

Parameters
mdataNNTP Mailbox data
hcHeader cache

Definition at line 733 of file newsrc.c.

734{
735 if (!hc)
736 return;
737
738 char buf[32] = { 0 };
739 bool old = false;
740 anum_t first = 0, last = 0;
741
742 /* fetch previous values of first and last */
743 char *hdata = hcache_fetch_raw_str(hc, "index", 5);
744 if (hdata)
745 {
746 mutt_debug(LL_DEBUG2, "hcache_fetch_email index: %s\n", hdata);
747 if (sscanf(hdata, ANUM_FMT " " ANUM_FMT, &first, &last) == 2)
748 {
749 old = true;
750 mdata->last_cached = last;
751
752 /* clean removed headers from cache */
753 for (anum_t current = first; current <= last; current++)
754 {
755 if ((current >= mdata->first_message) && (current <= mdata->last_message))
756 continue;
757
758 snprintf(buf, sizeof(buf), ANUM_FMT, current);
759 mutt_debug(LL_DEBUG2, "hcache_delete_email %s\n", buf);
760 hcache_delete_email(hc, buf, strlen(buf));
761 }
762 }
763 FREE(&hdata);
764 }
765
766 /* store current values of first and last */
767 if (!old || (mdata->first_message != first) || (mdata->last_message != last))
768 {
769 snprintf(buf, sizeof(buf), ANUM_FMT " " ANUM_FMT, mdata->first_message,
770 mdata->last_message);
771 mutt_debug(LL_DEBUG2, "hcache_store_email index: %s\n", buf);
772 hcache_store_raw(hc, "index", 5, buf, strlen(buf) + 1);
773 }
774}
int hcache_delete_email(struct HeaderCache *hc, const char *key, size_t keylen)
Multiplexor for StoreOps::delete_record.
Definition hcache.c:752
char * hcache_fetch_raw_str(struct HeaderCache *hc, const char *key, size_t keylen)
Fetch a string from the cache.
Definition hcache.c:662
int hcache_store_raw(struct HeaderCache *hc, const char *key, size_t keylen, void *data, size_t dlen)
Store a key / data pair.
Definition hcache.c:737
anum_t last_cached
Last cached article.
Definition mdata.h:40
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_bcache_update()

void nntp_bcache_update ( struct NntpMboxData * mdata)

Remove stale cached messages.

Parameters
mdataNNTP Mailbox data

Definition at line 801 of file newsrc.c.

802{
804}
int mutt_bcache_list(struct BodyCache *bcache, bcache_list_t want_id, void *data)
Find matching entries in the Body Cache.
Definition bcache.c:339
static int nntp_bcache_delete(const char *id, struct BodyCache *bcache, void *data)
Delete an entry from the message cache - Implements bcache_list_t -.
Definition newsrc.c:781
struct BodyCache * bcache
Body cache.
Definition mdata.h:50
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_delete_group_cache()

void nntp_delete_group_cache ( struct NntpMboxData * mdata)

Remove hcache and bcache of newsgroup.

Parameters
mdataNNTP Mailbox data

Definition at line 810 of file newsrc.c.

811{
812 if (!mdata || !mdata->adata || !mdata->adata->cacheable)
813 return;
814
815#ifdef USE_HCACHE
816 const char *const c_header_cache_backend = cs_subset_string(NeoMutt->sub, "header_cache_backend");
817 const struct StoreOps *store_ops = store_get_backend_ops(c_header_cache_backend);
818 if (store_ops)
819 {
820 struct Buffer *file = buf_pool_get();
821 nntp_hcache_namer(store_ops, mdata->group, file);
822 cache_expand(file->data, file->dsize, &mdata->adata->conn->account, buf_string(file));
823 unlink(buf_string(file));
824 mdata->last_cached = 0;
825 mutt_debug(LL_DEBUG2, "%s\n", buf_string(file));
826 buf_pool_release(&file);
827 }
828#endif
829
830 if (!mdata->bcache)
831 {
832 mdata->bcache = mutt_bcache_open(&mdata->adata->conn->account, mdata->group);
833 }
834 if (mdata->bcache)
835 {
836 mutt_debug(LL_DEBUG2, "%s/*\n", mdata->group);
838 mutt_bcache_close(&mdata->bcache);
839 }
840}
struct BodyCache * mutt_bcache_open(struct ConnAccount *account, const char *mailbox)
Open an Email-Body Cache.
Definition bcache.c:146
void mutt_bcache_close(struct BodyCache **ptr)
Close an Email-Body Cache.
Definition bcache.c:167
const char * cs_subset_string(const struct ConfigSubset *sub, const char *name)
Get a string config item by name.
Definition helpers.c:291
const struct StoreOps * store_get_backend_ops(const char *str)
Get the API functions for an store backend.
Definition store.c:88
size_t dsize
Length of data.
Definition buffer.h:39
char * data
Pointer to data.
Definition buffer.h:37
Definition lib.h:66
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_clear_cache()

void nntp_clear_cache ( struct NntpAccountData * adata)

Clear the NNTP cache.

Parameters
adataNNTP server

Remove hcache and bcache of all unexistent and unsubscribed newsgroups

Definition at line 848 of file newsrc.c.

849{
850 if (!adata || !adata->cacheable)
851 return;
852
853 struct dirent *de = NULL;
854 DIR *dir = NULL;
855 struct Buffer *cache = buf_pool_get();
856 struct Buffer *file = buf_pool_get();
857
858 cache_expand(cache->data, cache->dsize, &adata->conn->account, NULL);
859 buf_fix_dptr(cache);
861 if (!dir)
862 goto done;
863
864 buf_addch(cache, '/');
865 const bool c_save_unsubscribed = cs_subset_bool(NeoMutt->sub, "save_unsubscribed");
866
867 while ((de = readdir(dir)))
868 {
869 char *group = de->d_name;
870 if (mutt_str_equal(group, ".") || mutt_str_equal(group, ".."))
871 continue;
872
873 buf_printf(file, "%s%s", buf_string(cache), group);
874 struct stat st = { 0 };
875 if (stat(buf_string(file), &st) != 0)
876 continue;
877
878#ifdef USE_HCACHE
879 if (S_ISREG(st.st_mode))
880 {
881 char *ext = group + strlen(group) - 7;
882 if ((strlen(group) < 8) || !mutt_str_equal(ext, ".hcache"))
883 continue;
884 *ext = '\0';
885 }
886 else
887#endif
888 if (!S_ISDIR(st.st_mode))
889 continue;
890
891 struct NntpMboxData tmp_mdata = { 0 };
893 if (!mdata)
894 {
895 mdata = &tmp_mdata;
896 mdata->adata = adata;
897 mdata->group = group;
898 mdata->bcache = NULL;
899 }
900 else if (mdata->newsrc_ent || mdata->subscribed || c_save_unsubscribed)
901 {
902 continue;
903 }
904
906 if (S_ISDIR(st.st_mode))
907 {
908 rmdir(buf_string(file));
909 mutt_debug(LL_DEBUG2, "%s\n", buf_string(file));
910 }
911 }
912 closedir(dir);
913
914done:
915 buf_pool_release(&cache);
916 buf_pool_release(&file);
917}
int buf_printf(struct Buffer *buf, const char *fmt,...)
Format a string overwriting a Buffer.
Definition buffer.c:161
void buf_fix_dptr(struct Buffer *buf)
Move the dptr to end of the Buffer.
Definition buffer.c:182
size_t buf_addch(struct Buffer *buf, char c)
Add a single character to a Buffer.
Definition buffer.c:241
DIR * mutt_file_opendir(const char *path, enum MuttOpenDirMode mode)
Open a directory.
Definition file.c:535
@ MUTT_OPENDIR_NONE
Plain opendir()
Definition file.h:63
bool mutt_str_equal(const char *a, const char *b)
Compare two strings.
Definition string.c:665
void nntp_delete_group_cache(struct NntpMboxData *mdata)
Remove hcache and bcache of newsgroup.
Definition newsrc.c:810
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_select_server()

struct NntpAccountData * nntp_select_server ( struct Mailbox * m,
const char * server,
bool leave_lock )

Open a connection to an NNTP server.

Parameters
mMailbox
serverServer URL
leave_lockLeave the server locked?
Return values
ptrNNTP server
NULLError

Automatically loads a newsrc into memory, if necessary. Checks the size/mtime of a newsrc file, if it doesn't match, load again. Hmm, if a system has broken mtimes, this might mean the file is reloaded every time, which we'd have to fix.

See also
$newsrc

Definition at line 953 of file newsrc.c.

954{
955 char file[PATH_MAX] = { 0 };
956 int rc;
957 struct ConnAccount cac = { { 0 } };
958 struct NntpAccountData *adata = NULL;
959 struct Connection *conn = NULL;
960
961 if (!server || (*server == '\0'))
962 {
963 mutt_error(_("No news server defined"));
964 return NULL;
965 }
966
967 /* create account from news server url */
968 cac.flags = 0;
969 cac.port = NNTP_PORT;
971 cac.service = "nntp";
973
974 snprintf(file, sizeof(file), "%s%s", strstr(server, "://") ? "" : "news://", server);
975 struct Url *url = url_parse(file);
976 if (!url || (url->path && *url->path) ||
977 !((url->scheme == U_NNTP) || (url->scheme == U_NNTPS)) || !url->host ||
978 (account_from_url(&cac, url) < 0))
979 {
980 url_free(&url);
981 mutt_error(_("%s is an invalid news server specification"), server);
982 return NULL;
983 }
984 if (url->scheme == U_NNTPS)
985 {
986 cac.flags |= MUTT_ACCT_SSL;
987 cac.port = NNTP_SSL_PORT;
988 }
989 url_free(&url);
990
991 // If nntp_user and nntp_pass are specified in the config, use them to find the connection
992 const char *user = NULL;
993 const char *pass = NULL;
994 if ((user = cac.get_field(MUTT_CA_USER, NULL)))
995 {
996 mutt_str_copy(cac.user, user, sizeof(cac.user));
997 cac.flags |= MUTT_ACCT_USER;
998 }
999 if ((pass = cac.get_field(MUTT_CA_PASS, NULL)))
1000 {
1001 mutt_str_copy(cac.pass, pass, sizeof(cac.pass));
1002 cac.flags |= MUTT_ACCT_PASS;
1003 }
1004
1005 /* find connection by account */
1006 conn = mutt_conn_find(&cac);
1007 if (!conn)
1008 return NULL;
1009 if (!(conn->account.flags & MUTT_ACCT_USER) && cac.flags & MUTT_ACCT_USER)
1010 {
1011 conn->account.flags |= MUTT_ACCT_USER;
1012 conn->account.user[0] = '\0';
1013 }
1014
1015 /* new news server */
1016 adata = nntp_adata_new(conn);
1017
1018 rc = nntp_open_connection(adata);
1019
1020 /* try to create cache directory and enable caching */
1021 adata->cacheable = false;
1022 const char *const c_news_cache_dir = cs_subset_path(NeoMutt->sub, "news_cache_dir");
1023 if ((rc >= 0) && c_news_cache_dir)
1024 {
1025 cache_expand(file, sizeof(file), &conn->account, NULL);
1026 if (mutt_file_mkdir(file, S_IRWXU) < 0)
1027 {
1028 mutt_error(_("Can't create %s: %s"), file, strerror(errno));
1029 }
1030 adata->cacheable = true;
1031 }
1032
1033 /* load .newsrc */
1034 if (rc >= 0)
1035 {
1036 const struct Expando *c_newsrc = cs_subset_expando(NeoMutt->sub, "newsrc");
1037 struct Buffer *buf = buf_pool_get();
1039 buf->dsize, NeoMutt->env, buf);
1040 expand_path(buf, false);
1041 adata->newsrc_file = buf_strdup(buf);
1042 buf_pool_release(&buf);
1043 rc = nntp_newsrc_parse(adata);
1044 }
1045 if (rc >= 0)
1046 {
1047 /* try to load list of newsgroups from cache */
1048 if (adata->cacheable && (active_get_cache(adata) == 0))
1049 {
1050 rc = nntp_check_new_groups(m, adata);
1051 }
1052 else
1053 {
1054 /* load list of newsgroups from server */
1055 rc = nntp_active_fetch(adata, false);
1056 }
1057 }
1058
1059 if (rc >= 0)
1060 nntp_clear_cache(adata);
1061
1062#ifdef USE_HCACHE
1063 /* check cache files */
1064 if ((rc >= 0) && adata->cacheable)
1065 {
1066 struct dirent *de = NULL;
1067 DIR *dir = mutt_file_opendir(file, MUTT_OPENDIR_NONE);
1068
1069 if (dir)
1070 {
1071 while ((de = readdir(dir)))
1072 {
1073 struct HeaderCache *hc = NULL;
1074 char *group = de->d_name;
1075
1076 char *p = group + strlen(group) - 7;
1077 if ((strlen(group) < 8) || !mutt_str_equal(p, ".hcache"))
1078 continue;
1079 *p = '\0';
1081 if (!mdata)
1082 continue;
1083
1084 hc = nntp_hcache_open(mdata);
1085 if (!hc)
1086 continue;
1087
1088 /* fetch previous values of first and last */
1089 char *hdata = hcache_fetch_raw_str(hc, "index", 5);
1090 if (hdata)
1091 {
1092 anum_t first = 0, last = 0;
1093
1094 if (sscanf(hdata, ANUM_FMT " " ANUM_FMT, &first, &last) == 2)
1095 {
1096 if (mdata->deleted)
1097 {
1098 mdata->first_message = first;
1099 mdata->last_message = last;
1100 }
1101 if ((last >= mdata->first_message) && (last <= mdata->last_message))
1102 {
1103 mdata->last_cached = last;
1104 mutt_debug(LL_DEBUG2, "%s last_cached=" ANUM_FMT "\n", mdata->group, last);
1105 }
1106 }
1107 FREE(&hdata);
1108 }
1109 hcache_close(&hc);
1110 }
1111 closedir(dir);
1112 }
1113 }
1114#endif
1115
1116 if ((rc < 0) || !leave_lock)
1118
1119 if (rc < 0)
1120 {
1125 FREE(&adata);
1126 mutt_socket_close(conn);
1127 FREE(&conn);
1128 return NULL;
1129 }
1130
1131 return adata;
1132}
char * buf_strdup(const struct Buffer *buf)
Copy a Buffer's string.
Definition buffer.c:571
const struct Expando * cs_subset_expando(const struct ConfigSubset *sub, const char *name)
Get an Expando config item by name.
@ MUTT_CA_USER
User name.
Definition connaccount.h:36
@ MUTT_CA_PASS
Password.
Definition connaccount.h:37
#define MUTT_ACCT_SSL
Account uses SSL/TLS.
Definition connaccount.h:47
#define MUTT_ACCT_PASS
Password field has been set.
Definition connaccount.h:46
#define MUTT_ACCT_USER
User field has been set.
Definition connaccount.h:44
int expando_filter(const struct Expando *exp, const struct ExpandoRenderCallback *erc, void *data, MuttFormatFlags flags, int max_cols, char **env_list, struct Buffer *buf)
Render an Expando and run the result through a filter.
Definition filter.c:139
const struct ExpandoRenderCallback NntpRenderCallbacks[]
Callbacks for Newsrc Expandos.
int mutt_file_mkdir(const char *path, mode_t mode)
Recursively create directories.
Definition file.c:844
static const char * nntp_get_field(enum ConnAccountField field, void *gf_data)
Get connection login credentials - Implements ConnAccount::get_field() -.
Definition newsrc.c:922
#define mutt_error(...)
Definition logging2.h:94
void mutt_hash_free(struct HashTable **ptr)
Free a hash table.
Definition hash.c:459
void hcache_close(struct HeaderCache **ptr)
Multiplexor for StoreOps::close.
Definition hcache.c:549
int account_from_url(struct ConnAccount *cac, const struct Url *url)
Fill ConnAccount with information from url.
@ MUTT_ACCT_TYPE_NNTP
Nntp (Usenet) Account.
struct Connection * mutt_conn_find(const struct ConnAccount *cac)
Find a connection from a list.
Definition mutt_socket.c:88
void expand_path(struct Buffer *buf, bool regex)
Create the canonical path.
Definition muttlib.c:122
struct HeaderCache * nntp_hcache_open(struct NntpMboxData *mdata)
Open newsgroup hcache.
Definition newsrc.c:709
void nntp_clear_cache(struct NntpAccountData *adata)
Clear the NNTP cache.
Definition newsrc.c:848
int nntp_newsrc_parse(struct NntpAccountData *adata)
Parse .newsrc file.
Definition newsrc.c:165
static int active_get_cache(struct NntpAccountData *adata)
Load list of all newsgroups from cache.
Definition newsrc.c:614
struct NntpAccountData * nntp_adata_new(struct Connection *conn)
Allocate and initialise a new NntpAccountData structure.
Definition adata.c:65
int nntp_active_fetch(struct NntpAccountData *adata, bool mark_new)
Fetch list of all newsgroups from server.
Definition nntp.c:2037
#define NNTP_SSL_PORT
Definition private.h:37
#define NNTP_PORT
Definition private.h:36
int nntp_check_new_groups(struct Mailbox *m, struct NntpAccountData *adata)
Check for new groups/articles in subscribed groups.
Definition nntp.c:2105
int nntp_open_connection(struct NntpAccountData *adata)
Connect to server, authenticate and get capabilities.
Definition nntp.c:1765
#define MUTT_FORMAT_NO_FLAGS
No flags are set.
Definition render.h:33
int mutt_socket_close(struct Connection *conn)
Close a socket.
Definition socket.c:100
Login details for a remote server.
Definition connaccount.h:53
char user[128]
Username.
Definition connaccount.h:56
char pass[256]
Password.
Definition connaccount.h:57
const char * service
Name of the service, e.g. "imap".
Definition connaccount.h:61
const char *(* get_field)(enum ConnAccountField field, void *gf_data)
Definition connaccount.h:70
unsigned char type
Connection type, e.g. MUTT_ACCT_TYPE_IMAP.
Definition connaccount.h:59
MuttAccountFlags flags
Which fields are initialised, e.g. MUTT_ACCT_USER.
Definition connaccount.h:60
unsigned short port
Port to connect to.
Definition connaccount.h:58
Parsed Expando trees.
Definition expando.h:41
Header Cache.
Definition lib.h:87
char ** env
Private copy of the environment variables.
Definition neomutt.h:57
char * authenticators
Authenticators list.
Definition adata.h:52
char * user
Username.
Definition url.h:71
char * host
Host.
Definition url.h:73
char * pass
Password.
Definition url.h:72
enum UrlScheme scheme
Scheme, e.g. U_SMTPS.
Definition url.h:70
struct Url * url_parse(const char *src)
Fill in Url.
Definition url.c:242
void url_free(struct Url **ptr)
Free the contents of a URL.
Definition url.c:124
@ U_NNTPS
Url is nntps://.
Definition url.h:42
@ U_NNTP
Url is nntp://.
Definition url.h:41
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_article_status()

void nntp_article_status ( struct Mailbox * m,
struct Email * e,
char * group,
anum_t anum )

Get status of articles from .newsrc.

Parameters
mMailbox
eEmail
groupNewsgroup
anumArticle number

Full status flags are not supported by nntp, but we can fake some of them: Read = a read message number is in the .newsrc New = not read and not cached Old = not read but cached

Definition at line 1146 of file newsrc.c.

1147{
1148 struct NntpMboxData *mdata = m->mdata;
1149
1150 if (group)
1151 mdata = mutt_hash_find(mdata->adata->groups_hash, group);
1152
1153 if (!mdata)
1154 return;
1155
1156 for (unsigned int i = 0; i < mdata->newsrc_len; i++)
1157 {
1158 if ((anum >= mdata->newsrc_ent[i].first) && (anum <= mdata->newsrc_ent[i].last))
1159 {
1160 /* can't use mutt_set_flag() because mview_update() didn't get called yet */
1161 e->read = true;
1162 return;
1163 }
1164 }
1165
1166 /* article was not cached yet, it's new */
1167 if (anum > mdata->last_cached)
1168 return;
1169
1170 /* article isn't read but cached, it's old */
1171 const bool c_mark_old = cs_subset_bool(NeoMutt->sub, "mark_old");
1172 if (c_mark_old)
1173 e->old = true;
1174}
bool old
Email is seen, but unread.
Definition email.h:49
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_newsgroup_subscribe()

struct NntpMboxData * mutt_newsgroup_subscribe ( struct NntpAccountData * adata,
char * group )

Subscribe newsgroup.

Parameters
adataNNTP server
groupNewsgroup
Return values
ptrNNTP data
NULLError

Definition at line 1183 of file newsrc.c.

1184{
1185 if (!adata || !adata->groups_hash || !group || (*group == '\0'))
1186 return NULL;
1187
1189 mdata->subscribed = true;
1190 if (!mdata->newsrc_ent)
1191 {
1192 mdata->newsrc_ent = MUTT_MEM_CALLOC(1, struct NewsrcEntry);
1193 mdata->newsrc_len = 1;
1194 mdata->newsrc_ent[0].first = 1;
1195 mdata->newsrc_ent[0].last = 0;
1196 }
1197 return mdata;
1198}
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_newsgroup_unsubscribe()

struct NntpMboxData * mutt_newsgroup_unsubscribe ( struct NntpAccountData * adata,
char * group )

Unsubscribe newsgroup.

Parameters
adataNNTP server
groupNewsgroup
Return values
ptrNNTP data
NULLError

Definition at line 1207 of file newsrc.c.

1208{
1209 if (!adata || !adata->groups_hash || !group || (*group == '\0'))
1210 return NULL;
1211
1213 if (!mdata)
1214 return NULL;
1215
1216 mdata->subscribed = false;
1217 const bool c_save_unsubscribed = cs_subset_bool(NeoMutt->sub, "save_unsubscribed");
1218 if (!c_save_unsubscribed)
1219 {
1220 mdata->newsrc_len = 0;
1221 FREE(&mdata->newsrc_ent);
1222 }
1223 return mdata;
1224}
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_newsgroup_catchup()

struct NntpMboxData * mutt_newsgroup_catchup ( struct Mailbox * m,
struct NntpAccountData * adata,
char * group )

Catchup newsgroup.

Parameters
mMailbox
adataNNTP server
groupNewsgroup
Return values
ptrNNTP data
NULLError

Definition at line 1234 of file newsrc.c.

1236{
1237 if (!adata || !adata->groups_hash || !group || (*group == '\0'))
1238 return NULL;
1239
1241 if (!mdata)
1242 return NULL;
1243
1244 if (mdata->newsrc_ent)
1245 {
1246 MUTT_MEM_REALLOC(&mdata->newsrc_ent, 1, struct NewsrcEntry);
1247 mdata->newsrc_len = 1;
1248 mdata->newsrc_ent[0].first = 1;
1249 mdata->newsrc_ent[0].last = mdata->last_message;
1250 }
1251 mdata->unread = 0;
1252 if (m && (m->mdata == mdata))
1253 {
1254 for (unsigned int i = 0; i < m->msg_count; i++)
1255 {
1256 struct Email *e = m->emails[i];
1257 if (!e)
1258 break;
1259 mutt_set_flag(m, e, MUTT_READ, true, true);
1260 }
1261 }
1262 return mdata;
1263}
void mutt_set_flag(struct Mailbox *m, struct Email *e, enum MessageType flag, bool bf, bool upd_mbox)
Set a flag on an email.
Definition flags.c:54
@ MUTT_READ
Messages that have been read.
Definition mutt.h:92
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ mutt_newsgroup_uncatchup()

struct NntpMboxData * mutt_newsgroup_uncatchup ( struct Mailbox * m,
struct NntpAccountData * adata,
char * group )

Uncatchup newsgroup.

Parameters
mMailbox
adataNNTP server
groupNewsgroup
Return values
ptrNNTP data
NULLError

Definition at line 1273 of file newsrc.c.

1275{
1276 if (!adata || !adata->groups_hash || !group || (*group == '\0'))
1277 return NULL;
1278
1280 if (!mdata)
1281 return NULL;
1282
1283 if (mdata->newsrc_ent)
1284 {
1285 MUTT_MEM_REALLOC(&mdata->newsrc_ent, 1, struct NewsrcEntry);
1286 mdata->newsrc_len = 1;
1287 mdata->newsrc_ent[0].first = 1;
1288 mdata->newsrc_ent[0].last = mdata->first_message - 1;
1289 }
1290 if (m && (m->mdata == mdata))
1291 {
1292 mdata->unread = m->msg_count;
1293 for (unsigned int i = 0; i < m->msg_count; i++)
1294 {
1295 struct Email *e = m->emails[i];
1296 if (!e)
1297 break;
1298 mutt_set_flag(m, e, MUTT_READ, false, true);
1299 }
1300 }
1301 else
1302 {
1303 mdata->unread = mdata->last_message;
1304 if (mdata->newsrc_ent)
1305 mdata->unread -= mdata->newsrc_ent[0].last;
1306 }
1307 return mdata;
1308}
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ nntp_mailbox()

void nntp_mailbox ( struct Mailbox * m,
char * buf,
size_t buflen )

Get first newsgroup with new messages.

Parameters
mMailbox
bufBuffer for result
buflenLength of buffer

Definition at line 1316 of file newsrc.c.

1317{
1318 if (!m)
1319 return;
1320
1322 for (unsigned int i = 0; i < mod_data->current_news_srv->groups_num; i++)
1323 {
1324 struct NntpMboxData *mdata = mod_data->current_news_srv->groups_list[i];
1325
1326 if (!mdata || !mdata->subscribed || !mdata->unread)
1327 continue;
1328
1329 if ((m->type == MUTT_NNTP) &&
1330 mutt_str_equal(mdata->group, ((struct NntpMboxData *) m->mdata)->group))
1331 {
1332 unsigned int unread = 0;
1333
1334 for (unsigned int j = 0; j < m->msg_count; j++)
1335 {
1336 struct Email *e = m->emails[j];
1337 if (!e)
1338 break;
1339 if (!e->read && !e->deleted)
1340 unread++;
1341 }
1342 if (unread == 0)
1343 continue;
1344 }
1345 mutt_str_copy(buf, mdata->group, buflen);
1346 break;
1347 }
1348}
@ MUTT_NNTP
'NNTP' (Usenet) Mailbox type
Definition mailbox.h:48
@ MODULE_ID_NNTP
ModuleNntp, Nntp
Definition module_api.h:81
void * neomutt_get_module_data(struct NeoMutt *n, enum ModuleId id)
Get the private data for a Module.
Definition neomutt.c:663
enum MailboxType type
Mailbox type.
Definition mailbox.h:101
Nntp private Module data.
Definition module_data.h:30
struct NntpAccountData * current_news_srv
Current NNTP news server.
Definition module_data.h:32
+ Here is the call graph for this function:
+ Here is the caller graph for this function: