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

Signing/encryption multiplexor. More...

#include <stdbool.h>
+ Include dependency graph for crypt.h:
+ This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Functions

void crypt_convert_to_7bit (struct Body *b)
 Convert an email to 7bit encoding.
 
void crypt_current_time (struct State *state, const char *app_name)
 Print the current time.
 
const char * crypt_get_fingerprint_or_id (const char *p, const char **pphint, const char **ppl, const char **pps)
 Get the fingerprint or long key ID.
 
bool crypt_is_numerical_keyid (const char *s)
 Is this a numerical keyid.
 
int crypt_write_signed (struct Body *b, struct State *state, const char *tempfile)
 Write the message body/part.
 

Detailed Description

Signing/encryption multiplexor.

Authors
  • Richard Russon
  • Pietro Cerutti

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 crypt.h.

Function Documentation

◆ crypt_convert_to_7bit()

void crypt_convert_to_7bit ( struct Body * b)

Convert an email to 7bit encoding.

Parameters
bBody of email to convert

Definition at line 814 of file crypt.c.

815{
816 if (!WithCrypto)
817 return;
818
819 const bool c_pgp_strict_enc = cs_subset_bool(NeoMutt->sub, "pgp_strict_enc");
820 while (b)
821 {
822 if (b->type == TYPE_MULTIPART)
823 {
824 if (b->encoding != ENC_7BIT)
825 {
826 b->encoding = ENC_7BIT;
828 }
829 else if (((WithCrypto & APPLICATION_PGP) != 0) && c_pgp_strict_enc)
830 {
832 }
833 }
834 else if ((b->type == TYPE_MESSAGE) && !mutt_istr_equal(b->subtype, "delivery-status"))
835 {
836 if (b->encoding != ENC_7BIT)
838 }
839 else if (b->encoding == ENC_8BIT)
840 {
842 }
843 else if (b->encoding == ENC_BINARY)
844 {
845 b->encoding = ENC_BASE64;
846 }
847 else if (b->content && (b->encoding != ENC_BASE64) &&
848 (b->content->from || (b->content->space && c_pgp_strict_enc)))
849 {
851 }
852 b = b->next;
853 }
854}
bool cs_subset_bool(const struct ConfigSubset *sub, const char *name)
Get a boolean config item by name.
Definition helpers.c:47
void crypt_convert_to_7bit(struct Body *b)
Convert an email to 7bit encoding.
Definition crypt.c:814
@ ENC_7BIT
7-bit text
Definition mime.h:49
@ ENC_BINARY
Binary.
Definition mime.h:53
@ ENC_BASE64
Base-64 encoded text.
Definition mime.h:52
@ ENC_8BIT
8-bit text
Definition mime.h:50
@ ENC_QUOTED_PRINTABLE
Quoted-printable text.
Definition mime.h:51
@ TYPE_MESSAGE
Type: 'message/*'.
Definition mime.h:35
@ TYPE_MULTIPART
Type: 'multipart/*'.
Definition mime.h:37
bool mutt_istr_equal(const char *a, const char *b)
Compare two strings, ignoring case.
Definition string.c:677
#define APPLICATION_PGP
Use PGP to encrypt/sign.
Definition lib.h:106
#define WithCrypto
Definition lib.h:132
void mutt_message_to_7bit(struct Body *b, FILE *fp, struct ConfigSubset *sub)
Convert an email's MIME parts to 7-bit.
Definition sendlib.c:258
struct Body * parts
parts of a multipart or message/rfc822
Definition body.h:73
struct Content * content
Detailed info about the content of the attachment.
Definition body.h:70
struct Body * next
next attachment in the list
Definition body.h:72
char * subtype
content-type subtype
Definition body.h:61
unsigned int encoding
content-transfer-encoding, ContentEncoding
Definition body.h:41
unsigned int type
content-type primary type, ContentType
Definition body.h:40
bool space
Whitespace at the end of lines?
Definition content.h:42
bool from
Has a line beginning with "From "?
Definition content.h:44
Container for Accounts, Notifications.
Definition neomutt.h:41
struct ConfigSubset * sub
Inherited config items.
Definition neomutt.h:49
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ crypt_current_time()

void crypt_current_time ( struct State * state,
const char * app_name )

Print the current time.

Parameters
stateState to use
app_nameApp name, e.g. "PGP"

print the current time to avoid spoofing of the signature output

Definition at line 64 of file crypt.c.

65{
66 char p[256] = { 0 };
67 char tmp[512] = { 0 };
68
69 if (!WithCrypto)
70 return;
71
72 const bool c_crypt_timestamp = cs_subset_bool(NeoMutt->sub, "crypt_timestamp");
73 if (c_crypt_timestamp)
74 {
75 mutt_date_localtime_format(p, sizeof(p), _(" (current time: %c)"), mutt_date_now());
76 }
77 else
78 {
79 *p = '\0';
80 }
81
82 snprintf(tmp, sizeof(tmp), _("[-- %s output follows%s --]\n"), NONULL(app_name), p);
83 state_attach_puts(state, tmp);
84}
size_t mutt_date_localtime_format(char *buf, size_t buflen, const char *format, time_t t)
Format localtime.
Definition date.c:952
time_t mutt_date_now(void)
Return the number of seconds since the Unix epoch.
Definition date.c:457
#define _(a)
Definition message.h:28
void state_attach_puts(struct State *state, const char *t)
Write a string to the state.
Definition state.c:104
#define NONULL(x)
Definition string2.h:44
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ crypt_get_fingerprint_or_id()

const char * crypt_get_fingerprint_or_id ( const char * p,
const char ** pphint,
const char ** ppl,
const char ** pps )

Get the fingerprint or long key ID.

Parameters
[in]pString to examine
[out]pphintStart of string to be passed to pgp_add_string_to_hints() or crypt_add_string_to_hints()
[out]pplStart of long key ID if detected, else NULL
[out]ppsStart of short key ID if detected, else NULL
Return values
ptrCopy of fingerprint, if any, stripped of all spaces. Must be FREE'd by caller
NULLOtherwise

Obtain pointers to fingerprint or short or long key ID, if any.

Upon return, at most one of return, *ppl and *pps pointers is non-NULL, indicating the longest fingerprint or ID found, if any.

Definition at line 1395 of file crypt.c.

1397{
1398 const char *ps = NULL, *pl = NULL, *phint = NULL;
1399 char *pfcopy = NULL, *s1 = NULL, *s2 = NULL;
1400 char c;
1401 int isid;
1402 size_t hexdigits;
1403
1404 /* User input may be partial name, fingerprint or short or long key ID,
1405 * independent of `$pgp_long_ids`.
1406 * Fingerprint without spaces is 40 hex digits (SHA-1) or 32 hex digits (MD5).
1407 * Strip leading "0x" for key ID detection and prepare pl and ps to indicate
1408 * if an ID was found and to simplify logic in the key loop's inner
1409 * condition of the caller. */
1410
1411 char *pf = mutt_str_skip_whitespace(p);
1412 if (mutt_istr_startswith(pf, "0x"))
1413 pf += 2;
1414
1415 /* Check if a fingerprint is given, must be hex digits only, blanks
1416 * separating groups of 4 hex digits are allowed. Also pre-check for ID. */
1417 isid = 2; /* unknown */
1418 hexdigits = 0;
1419 s1 = pf;
1420 do
1421 {
1422 c = *(s1++);
1423 if ((('0' <= c) && (c <= '9')) || (('A' <= c) && (c <= 'F')) ||
1424 (('a' <= c) && (c <= 'f')))
1425 {
1426 hexdigits++;
1427 if (isid == 2)
1428 isid = 1; /* it is an ID so far */
1429 }
1430 else if (c)
1431 {
1432 isid = 0; /* not an ID */
1433 if ((c == ' ') && ((hexdigits % 4) == 0))
1434 ; /* skip blank before or after 4 hex digits */
1435 else
1436 break; /* any other character or position */
1437 }
1438 } while (c);
1439
1440 /* If at end of input, check for correct fingerprint length and copy if. */
1441 pfcopy = (!c && ((hexdigits == 40) || (hexdigits == 32)) ? mutt_str_dup(pf) : NULL);
1442
1443 if (pfcopy)
1444 {
1445 /* Use pfcopy to strip all spaces from fingerprint and as hint. */
1446 s1 = pfcopy;
1447 s2 = pfcopy;
1448 do
1449 {
1450 *(s1++) = *(s2 = mutt_str_skip_whitespace(s2));
1451 } while (*(s2++));
1452
1453 phint = pfcopy;
1454 ps = NULL;
1455 pl = NULL;
1456 }
1457 else
1458 {
1459 phint = p;
1460 ps = NULL;
1461 pl = NULL;
1462 if (isid == 1)
1463 {
1464 if (mutt_str_len(pf) == 16)
1465 pl = pf; /* long key ID */
1466 else if (mutt_str_len(pf) == 8)
1467 ps = pf; /* short key ID */
1468 }
1469 }
1470
1471 *pphint = phint;
1472 *ppl = pl;
1473 *pps = ps;
1474 return pfcopy;
1475}
char * mutt_str_dup(const char *str)
Copy a string, safely.
Definition string.c:257
char * mutt_str_skip_whitespace(const char *p)
Find the first non-whitespace character in a string.
Definition string.c:556
size_t mutt_str_len(const char *a)
Calculate the length of a string, safely.
Definition string.c:503
size_t mutt_istr_startswith(const char *str, const char *prefix)
Check whether a string starts with a prefix, ignoring case.
Definition string.c:246
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ crypt_is_numerical_keyid()

bool crypt_is_numerical_keyid ( const char * s)

Is this a numerical keyid.

Parameters
sKey to test
Return values
trueKeyid is numeric

Check if a crypt-hook value is a key id.

Definition at line 1484 of file crypt.c.

1485{
1486 /* or should we require the "0x"? */
1487 if (mutt_strn_equal(s, "0x", 2))
1488 s += 2;
1489 if (strlen(s) % 8)
1490 return false;
1491 while (*s)
1492 if (!strchr("0123456789ABCDEFabcdef", *s++))
1493 return false;
1494
1495 return true;
1496}
bool mutt_strn_equal(const char *a, const char *b, size_t num)
Check for equality of two strings (to a maximum), safely.
Definition string.c:429
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ crypt_write_signed()

int crypt_write_signed ( struct Body * b,
struct State * state,
const char * tempfile )

Write the message body/part.

Parameters
bBody to write
stateState to use
tempfileFile to write to
Return values
0Success
-1Error

Body/part A described by state state to the given TEMPFILE.

Definition at line 764 of file crypt.c.

765{
766 if (!WithCrypto)
767 return -1;
768
769 FILE *fp = mutt_file_fopen(tempfile, "w");
770 if (!fp)
771 {
772 mutt_perror("%s", tempfile);
773 return -1;
774 }
775
776 if (!mutt_file_seek(state->fp_in, b->hdr_offset, SEEK_SET))
777 {
778 mutt_file_fclose(&fp);
779 return -1;
780 }
781 size_t bytes = b->length + b->offset - b->hdr_offset;
782 bool hadcr = false;
783 while (bytes > 0)
784 {
785 const int c = fgetc(state->fp_in);
786 if (c == EOF)
787 break;
788
789 bytes--;
790
791 if (c == '\r')
792 {
793 hadcr = true;
794 }
795 else
796 {
797 if ((c == '\n') && !hadcr)
798 fputc('\r', fp);
799
800 hadcr = false;
801 }
802
803 fputc(c, fp);
804 }
805 mutt_file_fclose(&fp);
806
807 return 0;
808}
bool mutt_file_seek(FILE *fp, LOFF_T offset, int whence)
Wrapper for fseeko with error handling.
Definition file.c:648
#define mutt_file_fclose(FP)
Definition file.h:144
#define mutt_file_fopen(PATH, MODE)
Definition file.h:143
#define mutt_perror(...)
Definition logging2.h:95
LOFF_T offset
offset where the actual data begins
Definition body.h:52
LOFF_T length
length (in bytes) of attachment
Definition body.h:53
long hdr_offset
Offset in stream where the headers begin.
Definition body.h:81
FILE * fp_in
File to read from.
Definition state.h:55
+ Here is the call graph for this function:
+ Here is the caller graph for this function: