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

Header cache multiplexor. More...

#include <stdbool.h>
#include <stddef.h>
#include <stdint.h>
#include "compress/lib.h"
#include "store/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.

Data Structures

struct  HeaderCache
 Header Cache. More...
 
struct  HCacheEntry
 Wrapper for Email retrieved from the header cache. More...
 

Macros

#define hcache_fetch_raw_obj(hc, key, keylen, dst)
 

Typedefs

typedef void(* hcache_namer_t) (const struct StoreOps *store_ops, const char *path, struct Buffer *dest)
 

Functions

struct HeaderCachehcache_open (const char *path, const char *folder, hcache_namer_t namer, bool create)
 Open the connection to the header cache.
 
void hcache_close (struct HeaderCache **ptr)
 Close the connection to the header cache.
 
int hcache_store_email (struct HeaderCache *hc, const char *key, size_t keylen, struct Email *e, uint32_t uidvalidity)
 Store a Header along with a validity datum.
 
struct HCacheEntry hcache_fetch_email (struct HeaderCache *hc, const char *key, size_t keylen, uint32_t uidvalidity)
 Fetch and validate a message's header from the cache.
 
char * hcache_fetch_raw_str (struct HeaderCache *hc, const char *key, size_t keylen)
 Fetch a string from the cache.
 
bool hcache_fetch_raw_obj_full (struct HeaderCache *hc, const char *key, size_t keylen, void *dst, size_t dstlen)
 Fetch a message's header from the cache into a destination object.
 
int hcache_store_raw (struct HeaderCache *hc, const char *key, size_t keylen, void *data, size_t dlen)
 Store a key / data pair.
 
int hcache_delete_email (struct HeaderCache *hc, const char *key, size_t keylen)
 Delete a key / data pair.
 
int hcache_delete_raw (struct HeaderCache *hc, const char *key, size_t keylen)
 Delete a key / data pair.
 

Detailed Description

Header cache multiplexor.

Authors
  • Richard Russon
  • Pietro Cerutti
  • Tino Reichardt

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.

Macro Definition Documentation

◆ hcache_fetch_raw_obj

#define hcache_fetch_raw_obj ( hc,
key,
keylen,
dst )
Value:
hcache_fetch_raw_obj_full(hc, key, keylen, dst, sizeof(*dst))
bool hcache_fetch_raw_obj_full(struct HeaderCache *hc, const char *key, size_t keylen, void *dst, size_t dstlen)
Fetch a message's header from the cache into a destination object.
Definition hcache.c:630

Definition at line 168 of file lib.h.

Typedef Documentation

◆ hcache_namer_t

typedef void(* hcache_namer_t) (const struct StoreOps *store_ops, const char *path, struct Buffer *dest)

Definition at line 119 of file lib.h.

Function Documentation

◆ hcache_open()

struct HeaderCache * hcache_open ( const char * path,
const char * folder,
hcache_namer_t namer,
bool create )

Open the connection to the header cache.

Parameters
pathLocation of the header cache (often as specified by the user)
folderName of the folder containing the messages
namerOptional (might be NULL) client-specific function to form the final name of the hcache database file.
createCreate the file if it's not there?
Return values
ptrSuccess, struct HeaderCache struct
NULLOtherwise

Open the connection to the header cache.

Definition at line 477 of file hcache.c.

479{
480 if (!path || (path[0] == '\0'))
481 return NULL;
482
484 if (mod_data->hcache_ver == 0x0)
485 mod_data->hcache_ver = generate_hcachever();
486
487 struct HeaderCache *hc = hcache_new();
488
490 hc->crc = mod_data->hcache_ver;
491
492 const char *const c_header_cache_backend = cs_subset_string(NeoMutt->sub, "header_cache_backend");
493 hc->store_ops = store_get_backend_ops(c_header_cache_backend);
494 if (!hc->store_ops)
495 {
496 hcache_free(&hc);
497 return NULL;
498 }
499
500#ifdef USE_HCACHE_COMPRESSION
501 const char *const c_header_cache_compress_method = cs_subset_string(NeoMutt->sub, "header_cache_compress_method");
502 if (c_header_cache_compress_method)
503 {
504 hc->compr_ops = compress_get_ops(c_header_cache_compress_method);
505
506 const short c_header_cache_compress_level = cs_subset_number(NeoMutt->sub, "header_cache_compress_level");
507 hc->compr_handle = hc->compr_ops->open(c_header_cache_compress_level);
508 if (!hc->compr_handle)
509 {
510 hcache_free(&hc);
511 return NULL;
512 }
513
514 /* remember the buffer of database backend */
515 mutt_debug(LL_DEBUG3, "Header cache will use %s compression\n",
516 hc->compr_ops->name);
517 }
518#endif
519
520 struct Buffer *hcpath = buf_pool_get();
521 hcache_per_folder(hc, hcpath, path, namer);
522
523 hc->store_handle = hc->store_ops->open(buf_string(hcpath), create);
524 if (!hc->store_handle)
525 {
526 /* remove a possibly incompatible version */
527 if (unlink(buf_string(hcpath)) == 0)
528 {
529 hc->store_handle = hc->store_ops->open(buf_string(hcpath), create);
530 }
531 }
532
533 if (!hc->store_handle)
534 {
535 if (hc->compr_ops)
536 {
537 hc->compr_ops->close(&hc->compr_handle);
538 }
539 hcache_free(&hc);
540 }
541
542 buf_pool_release(&hcpath);
543 return hc;
544}
static const char * buf_string(const struct Buffer *buf)
Convert a buffer to a const char * "string".
Definition buffer.h:96
const struct ComprOps * compress_get_ops(const char *compr)
Get the API functions for a compress backend.
Definition compress.c:78
const char * cs_subset_string(const struct ConfigSubset *sub, const char *name)
Get a string config item by name.
Definition helpers.c:291
short cs_subset_number(const struct ConfigSubset *sub, const char *name)
Get a number config item by name.
Definition helpers.c:143
#define mutt_debug(LEVEL,...)
Definition logging2.h:91
static void hcache_free(struct HeaderCache **ptr)
Free a header cache.
Definition hcache.c:106
static struct HeaderCache * hcache_new(void)
Create a new header cache.
Definition hcache.c:121
static unsigned int generate_hcachever(void)
Calculate hcache version from dynamic configuration.
Definition hcache.c:435
static void hcache_per_folder(struct HeaderCache *hc, struct Buffer *hcpath, const char *path, hcache_namer_t namer)
Generate the hcache pathname.
Definition hcache.c:359
static char * get_foldername(const char *folder)
Where should the cache be stored?
Definition hcache.c:412
@ LL_DEBUG3
Log at debug level 3.
Definition logging2.h:47
@ MODULE_ID_HCACHE
ModuleHcache, Email Header Cache
Definition module_api.h:67
void * neomutt_get_module_data(struct NeoMutt *n, enum ModuleId id)
Get the private data for a Module.
Definition neomutt.c:663
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
const struct StoreOps * store_get_backend_ops(const char *str)
Get the API functions for an store backend.
Definition store.c:88
String manipulation buffer.
Definition buffer.h:36
ComprHandle *(* open)(short level)
Definition lib.h:79
const char * name
Compression name.
Definition lib.h:66
void(* close)(ComprHandle **ptr)
Definition lib.h:124
Hcache private Module data.
Definition module_data.h:30
unsigned int hcache_ver
Header Cache version.
Definition module_data.h:32
Header Cache.
Definition lib.h:87
ComprHandle * compr_handle
Compression handle.
Definition lib.h:93
unsigned int crc
CRC of the cache entry.
Definition lib.h:89
char * folder
Folder name.
Definition lib.h:88
const struct StoreOps * store_ops
Store backend.
Definition lib.h:90
StoreHandle * store_handle
Store handle.
Definition lib.h:91
const struct ComprOps * compr_ops
Compression backend.
Definition lib.h:92
Container for Accounts, Notifications.
Definition neomutt.h:41
struct ConfigSubset * sub
Inherited config items.
Definition neomutt.h:49
StoreHandle *(* open)(const char *path, bool create)
Definition lib.h:83
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ hcache_close()

void hcache_close ( struct HeaderCache ** ptr)

Close the connection to the header cache.

Parameters
ptrPointer to the struct HeaderCache structure got by hcache_open()
Note
The pointer will be set to NULL

Close the connection to the header cache.

Definition at line 549 of file hcache.c.

550{
551 if (!ptr || !*ptr)
552 return;
553
554 struct HeaderCache *hc = *ptr;
555
556#ifdef USE_HCACHE_COMPRESSION
557 if (hc->compr_ops)
558 hc->compr_ops->close(&hc->compr_handle);
559#endif
560
561 hc->store_ops->close(&hc->store_handle);
562
563 hcache_free(ptr);
564}
void(* close)(StoreHandle **ptr)
Definition lib.h:144
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ hcache_store_email()

int hcache_store_email ( struct HeaderCache * hc,
const char * key,
size_t keylen,
struct Email * e,
uint32_t uidvalidity )

Store a Header along with a validity datum.

Parameters
hcPointer to the struct HeaderCache structure got by hcache_open()
keyMessage identification string
keylenLength of the key string
eEmail to store
uidvalidityIMAP-specific UIDVALIDITY value, or 0 to use the current time
Return values
0Success
numGeneric or backend-specific error code otherwise

Store a Header along with a validity datum.

Definition at line 683 of file hcache.c.

685{
686 if (!hc)
687 return -1;
688
689 int dlen = 0;
690 char *data = dump_email(hc, e, &dlen, uidvalidity);
691
692#ifdef USE_HCACHE_COMPRESSION
693 if (hc->compr_ops)
694 {
695 /* We don't compress uidvalidity and the crc, so we can check them before
696 * decompressing on fetch(). */
697 size_t hlen = header_size();
698
699 /* data / dlen gets ptr to compressed data here */
700 size_t clen = dlen;
701 void *cdata = hc->compr_ops->compress(hc->compr_handle, data + hlen, dlen - hlen, &clen);
702 if (!cdata)
703 {
704 FREE(&data);
705 return -1;
706 }
707
708 char *whole = MUTT_MEM_MALLOC(hlen + clen, char);
709 memcpy(whole, data, hlen);
710 memcpy(whole + hlen, cdata, clen);
711
712 FREE(&data);
713
714 data = whole;
715 dlen = hlen + clen;
716 }
717#endif
718
719 struct RealKey *rk = realkey(hc, key, keylen, true);
720 int rc = hc->store_ops->store(hc->store_handle, rk->key, rk->keylen, data, dlen);
721
722 FREE(&data);
723
724 return rc;
725}
static void * dump_email(struct HeaderCache *hc, const struct Email *e, int *off, uint32_t uidvalidity)
Serialise an Email object.
Definition hcache.c:230
static struct RealKey * realkey(struct HeaderCache *hc, const char *key, size_t keylen, bool compress)
Compute the real key used in the backend, taking into account the compression method.
Definition hcache.c:78
static size_t header_size(void)
Compute the size of the header with uuid validity and crc.
Definition hcache.c:130
#define FREE(x)
Free memory and set the pointer to NULL.
Definition memory.h:68
#define MUTT_MEM_MALLOC(n, type)
Definition memory.h:53
void *(* compress)(ComprHandle *handle, const char *data, size_t dlen, size_t *clen)
Definition lib.h:96
Hcache key name (including compression method)
Definition hcache.c:65
char key[1024]
Key name.
Definition hcache.c:66
size_t keylen
Length of key.
Definition hcache.c:67
int(* store)(StoreHandle *store, const char *key, size_t klen, void *value, size_t vlen)
Definition lib.h:122
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ hcache_fetch_email()

struct HCacheEntry hcache_fetch_email ( struct HeaderCache * hc,
const char * key,
size_t keylen,
uint32_t uidvalidity )

Fetch and validate a message's header from the cache.

Parameters
hcPointer to the struct HeaderCache structure got by hcache_open()
keyMessage identification string
keylenLength of the string pointed to by key
uidvalidityOnly restore if it matches the stored uidvalidity
Return values
objHCacheEntry containing an Email, empty on failure
Note
This function performs a check on the validity of the data found by comparing it with the crc value of the struct HeaderCache structure.

Fetch and validate a message's header from the cache.

Definition at line 569 of file hcache.c.

571{
572 struct HCacheEntry hce = { 0 };
573 if (!hc)
574 return hce;
575
576 size_t dlen = 0;
577 struct RealKey *rk = realkey(hc, key, keylen, true);
578 void *data = hc->store_ops->fetch(hc->store_handle, rk->key, rk->keylen, &dlen);
579 void *to_free = data;
580 if (!data)
581 {
582 goto end;
583 }
584
585 /* restore uidvalidity and crc */
586 size_t hlen = header_size();
587 if (hlen > dlen)
588 {
589 goto end;
590 }
591 int off = 0;
592 serial_restore_uint32_t(&hce.uidvalidity, data, &off);
593 serial_restore_int(&hce.crc, data, &off);
594 ASSERT((size_t) off == hlen);
595 if ((hce.crc != hc->crc) || ((uidvalidity != 0) && (uidvalidity != hce.uidvalidity)))
596 {
597 goto end;
598 }
599
600#ifdef USE_HCACHE_COMPRESSION
601 if (hc->compr_ops)
602 {
603 void *dblob = hc->compr_ops->decompress(hc->compr_handle,
604 (char *) data + hlen, dlen - hlen);
605 if (!dblob)
606 {
607 goto end;
608 }
609 data = (char *) dblob - hlen; /* restore skips uidvalidity and crc */
610 }
611#endif
612
613 hce.email = restore_email(data);
614
615end:
616 free_raw(hc, &to_free);
617 return hce;
618}
static struct Email * restore_email(const unsigned char *d)
Restore an Email from data retrieved from the cache.
Definition hcache.c:270
static void free_raw(struct HeaderCache *hc, void **data)
Multiplexor for StoreOps::free.
Definition hcache.c:426
void serial_restore_int(unsigned int *i, const unsigned char *d, int *off)
Unpack an integer from a binary blob.
Definition serialize.c:121
void serial_restore_uint32_t(uint32_t *s, const unsigned char *d, int *off)
Unpack an uint32_t from a binary blob.
Definition serialize.c:133
#define ASSERT(COND)
Definition signal2.h:59
void *(* decompress)(ComprHandle *handle, const char *cbuf, size_t clen)
Definition lib.h:112
Wrapper for Email retrieved from the header cache.
Definition lib.h:100
uint32_t uidvalidity
IMAP-specific UIDVALIDITY.
Definition lib.h:101
struct Email * email
Retrieved email.
Definition lib.h:103
unsigned int crc
CRC of Email/Body/etc structs.
Definition lib.h:102
void *(* fetch)(StoreHandle *store, const char *key, size_t klen, size_t *vlen)
Definition lib.h:97
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ hcache_fetch_raw_str()

char * hcache_fetch_raw_str ( struct HeaderCache * hc,
const char * key,
size_t keylen )

Fetch a string from the cache.

Parameters
[in]hcPointer to the struct HeaderCache structure got by hcache_open()
[in]keyMessage identification string
[in]keylenLength of the string pointed to by key
Return values
ptrSuccess, the data if found
NULLOtherwise

Definition at line 662 of file hcache.c.

663{
664 if (!hc)
665 return NULL;
666
667 char *res = NULL;
668 size_t dlen = 0;
669
670 struct RealKey *rk = realkey(hc, key, keylen, false);
671 void *data = hc->store_ops->fetch(hc->store_handle, rk->key, rk->keylen, &dlen);
672 if (data)
673 {
674 res = mutt_strn_dup(data, dlen);
675 free_raw(hc, &data);
676 }
677 return res;
678}
char * mutt_strn_dup(const char *begin, size_t len)
Duplicate a sub-string.
Definition string.c:384
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ hcache_fetch_raw_obj_full()

bool hcache_fetch_raw_obj_full ( struct HeaderCache * hc,
const char * key,
size_t keylen,
void * dst,
size_t dstlen )

Fetch a message's header from the cache into a destination object.

Parameters
[in]hcPointer to the struct HeaderCache structure got by hcache_open()
[in]keyMessage identification string
[in]keylenLength of the string pointed to by key
[out]dstPointer to the destination object
[in]dstlenSize of the destination object
Return values
trueSuccess, the data was found and the length matches
falseOtherwise

Definition at line 630 of file hcache.c.

632{
633 if (!hc)
634 return false;
635
636 bool rc = true;
637 size_t srclen = 0;
638
639 struct RealKey *rk = realkey(hc, key, keylen, false);
640 void *src = hc->store_ops->fetch(hc->store_handle, rk->key, rk->keylen, &srclen);
641
642 if (src && (srclen == dstlen))
643 {
644 memcpy(dst, src, dstlen);
645 }
646 else
647 {
648 rc = false;
649 }
650 free_raw(hc, &src);
651 return rc;
652}
+ Here is the call graph for this function:

◆ hcache_store_raw()

int hcache_store_raw ( struct HeaderCache * hc,
const char * key,
size_t keylen,
void * data,
size_t dlen )

Store a key / data pair.

Parameters
hcPointer to the struct HeaderCache structure got by hcache_open()
keyMessage identification string
keylenLength of the string pointed to by key
dataPayload to associate with key
dlenLength of the buffer pointed to by the data parameter
Return values
0Success
numGeneric or backend-specific error code otherwise

Definition at line 737 of file hcache.c.

739{
740 if (!hc)
741 return -1;
742
743 struct RealKey *rk = realkey(hc, key, keylen, false);
744 int rc = hc->store_ops->store(hc->store_handle, rk->key, rk->keylen, data, dlen);
745
746 return rc;
747}
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ hcache_delete_email()

int hcache_delete_email ( struct HeaderCache * hc,
const char * key,
size_t keylen )

Delete a key / data pair.

Parameters
hcPointer to the struct HeaderCache structure got by hcache_open()
keyMessage identification string
keylenLength of the string pointed to by key
Return values
0Success
numGeneric or backend-specific error code otherwise

Delete a key / data pair.

Definition at line 752 of file hcache.c.

753{
754 if (!hc)
755 return -1;
756
757 struct RealKey *rk = realkey(hc, key, keylen, true);
758
759 return hc->store_ops->delete_record(hc->store_handle, rk->key, rk->keylen);
760}
int(* delete_record)(StoreHandle *store, const char *key, size_t klen)
Definition lib.h:135
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ hcache_delete_raw()

int hcache_delete_raw ( struct HeaderCache * hc,
const char * key,
size_t keylen )

Delete a key / data pair.

Parameters
hcPointer to the struct HeaderCache structure got by hcache_open()
keyMessage identification string
keylenLength of the string pointed to by key
Return values
0Success
numGeneric or backend-specific error code otherwise

Delete a key / data pair.

Definition at line 765 of file hcache.c.

766{
767 if (!hc)
768 return -1;
769
770 struct RealKey *rk = realkey(hc, key, keylen, false);
771
772 return hc->store_ops->delete_record(hc->store_handle, rk->key, rk->keylen);
773}
+ Here is the call graph for this function:
+ Here is the caller graph for this function: