NeoMutt  2025-12-11-911-gd8d604
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
smime_build_smime_entity()

Encrypt the email body to all recipients. More...

+ Collaboration diagram for smime_build_smime_entity():

Functions

struct Bodysmime_gpgme_build_smime_entity (struct Body *b, char *keylist)
 Encrypt the email body to all recipients - Implements CryptModuleSpecs::smime_build_smime_entity() -.
 
struct Bodysmime_class_build_smime_entity (struct Body *b, char *certlist)
 Encrypt the email body to all recipients - Implements CryptModuleSpecs::smime_build_smime_entity() -.
 

Detailed Description

Encrypt the email body to all recipients.

Parameters
bBody of email
certlistList of key fingerprints (space separated)
Return values
ptrNew S/MIME encrypted Body
NULLError

Function Documentation

◆ smime_gpgme_build_smime_entity()

struct Body * smime_gpgme_build_smime_entity ( struct Body * b,
char * keylist )

Encrypt the email body to all recipients - Implements CryptModuleSpecs::smime_build_smime_entity() -.

Definition at line 1094 of file crypt_gpgme.c.

1095{
1096 /* OpenSSL converts line endings to crlf when encrypting. Some clients
1097 * depend on this for signed+encrypted messages: they do not convert line
1098 * endings between decrypting and checking the signature. */
1099 gpgme_data_t plaintext = body_to_data_object(b, true);
1100 if (!plaintext)
1101 return NULL;
1102
1103 char *outfile = encrypt_gpgme_object(plaintext, keylist, true, false, NULL);
1104 gpgme_data_release(plaintext);
1105 if (!outfile)
1106 return NULL;
1107
1108 struct Body *b_enc = mutt_body_new();
1109 b_enc->type = TYPE_APPLICATION;
1110 b_enc->subtype = mutt_str_dup("pkcs7-mime");
1111 mutt_param_set(&b_enc->parameter, "name", "smime.p7m");
1112 mutt_param_set(&b_enc->parameter, "smime-type", "enveloped-data");
1113 b_enc->encoding = ENC_BASE64; /* The output of OpenSSL SHOULD be binary */
1114 b_enc->use_disp = true;
1115 b_enc->disposition = DISP_ATTACH;
1116 b_enc->d_filename = mutt_str_dup("smime.p7m");
1117 b_enc->filename = outfile;
1118 b_enc->unlink = true; /* delete after sending the message */
1119 b_enc->parts = 0;
1120 b_enc->next = 0;
1121
1122 return b_enc;
1123}
static gpgme_data_t body_to_data_object(struct Body *b, bool convert)
Create GPGME object from the mail body.
static char * encrypt_gpgme_object(gpgme_data_t plaintext, char *keylist, bool use_smime, bool combined_signed, const struct AddressList *from)
Encrypt the GPGPME data object.
struct Body * mutt_body_new(void)
Create a new Body.
Definition body.c:44
@ ENC_BASE64
Base-64 encoded text.
Definition mime.h:52
@ TYPE_APPLICATION
Type: 'application/*'.
Definition mime.h:33
@ DISP_ATTACH
Content is attached.
Definition mime.h:63
char * mutt_str_dup(const char *str)
Copy a string, safely.
Definition string.c:257
void mutt_param_set(struct ParameterList *pl, const char *attribute, const char *value)
Set a Parameter.
Definition parameter.c:111
The body of an email.
Definition body.h:36
char * d_filename
filename to be used for the content-disposition header If NULL, filename is used instead.
Definition body.h:56
struct Body * parts
parts of a multipart or message/rfc822
Definition body.h:73
bool unlink
If true, filename should be unlink()ed before free()ing this structure.
Definition body.h:68
struct ParameterList parameter
Parameters of the content-type.
Definition body.h:63
bool use_disp
Content-Disposition uses filename= ?
Definition body.h:47
unsigned int disposition
content-disposition, ContentDisposition
Definition body.h:42
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
char * filename
When sending a message, this is the file to which this structure refers.
Definition body.h:59
+ Here is the call graph for this function:

◆ smime_class_build_smime_entity()

struct Body * smime_class_build_smime_entity ( struct Body * b,
char * certlist )

Encrypt the email body to all recipients - Implements CryptModuleSpecs::smime_build_smime_entity() -.

Definition at line 1186 of file smime.c.

1187{
1188 char buf[1024] = { 0 };
1189 char certfile[PATH_MAX] = { 0 };
1190 char *cert_end = NULL;
1191 FILE *fp_smime_in = NULL, *fp_smime_err = NULL, *fp_out = NULL, *fp_tmp = NULL;
1192 struct Body *b_enc = NULL;
1193 bool err = false;
1194 int empty, off;
1195 pid_t pid;
1196
1197 struct Buffer *tempfile = buf_pool_get();
1198 struct Buffer *smime_infile = buf_pool_get();
1199
1200 buf_mktemp(tempfile);
1201 fp_out = mutt_file_fopen(buf_string(tempfile), "w+");
1202 if (!fp_out)
1203 {
1204 mutt_perror("%s", buf_string(tempfile));
1205 goto cleanup;
1206 }
1207
1208 fp_smime_err = mutt_file_mkstemp();
1209 if (!fp_smime_err)
1210 {
1211 mutt_perror(_("Can't create temporary file"));
1212 goto cleanup;
1213 }
1214
1215 buf_mktemp(smime_infile);
1216 fp_tmp = mutt_file_fopen(buf_string(smime_infile), "w+");
1217 if (!fp_tmp)
1218 {
1219 mutt_perror("%s", buf_string(smime_infile));
1220 goto cleanup;
1221 }
1222
1223 *certfile = '\0';
1224 for (char *cert_start = certlist; cert_start; cert_start = cert_end)
1225 {
1226 cert_end = strchr(cert_start, ' ');
1227 if (cert_end)
1228 *cert_end = '\0';
1229 if (*cert_start)
1230 {
1231 off = mutt_str_len(certfile);
1232 const char *const c_smime_certificates = cs_subset_path(NeoMutt->sub, "smime_certificates");
1233 snprintf(certfile + off, sizeof(certfile) - off, "%s%s/%s",
1234 (off != 0) ? " " : "", NONULL(c_smime_certificates), cert_start);
1235 }
1236 if (cert_end)
1237 *cert_end++ = ' ';
1238 }
1239
1240 /* write a MIME entity */
1241 mutt_write_mime_header(b, fp_tmp, NeoMutt->sub);
1242 fputc('\n', fp_tmp);
1243 mutt_write_mime_body(b, fp_tmp, NeoMutt->sub);
1244 mutt_file_fclose(&fp_tmp);
1245
1246 pid = smime_invoke_encrypt(&fp_smime_in, NULL, NULL, -1, fileno(fp_out),
1247 fileno(fp_smime_err), buf_string(smime_infile), certfile);
1248 if (pid == -1)
1249 {
1250 mutt_file_unlink(buf_string(smime_infile));
1251 goto cleanup;
1252 }
1253
1254 mutt_file_fclose(&fp_smime_in);
1255
1256 filter_wait(pid);
1257 mutt_file_unlink(buf_string(smime_infile));
1258
1259 fflush(fp_out);
1260 rewind(fp_out);
1261 empty = (fgetc(fp_out) == EOF);
1262 mutt_file_fclose(&fp_out);
1263
1264 fflush(fp_smime_err);
1265 rewind(fp_smime_err);
1266 while (fgets(buf, sizeof(buf) - 1, fp_smime_err))
1267 {
1268 err = true;
1269 fputs(buf, stdout);
1270 }
1271 mutt_file_fclose(&fp_smime_err);
1272
1273 /* pause if there is any error output from SMIME */
1274 if (err)
1276
1277 if (empty)
1278 {
1279 /* fatal error while trying to encrypt message */
1280 if (!err)
1281 mutt_any_key_to_continue(_("No output from OpenSSL..."));
1282 mutt_file_unlink(buf_string(tempfile));
1283 goto cleanup;
1284 }
1285
1286 b_enc = mutt_body_new();
1287 b_enc->type = TYPE_APPLICATION;
1288 b_enc->subtype = mutt_str_dup("pkcs7-mime");
1289 mutt_param_set(&b_enc->parameter, "name", "smime.p7m");
1290 mutt_param_set(&b_enc->parameter, "smime-type", "enveloped-data");
1291 b_enc->encoding = ENC_BASE64; /* The output of OpenSSL SHOULD be binary */
1292 b_enc->use_disp = true;
1293 b_enc->disposition = DISP_ATTACH;
1294 b_enc->d_filename = mutt_str_dup("smime.p7m");
1295 b_enc->filename = buf_strdup(tempfile);
1296 b_enc->unlink = true; /* delete after sending the message */
1297 b_enc->parts = NULL;
1298 b_enc->next = NULL;
1299
1300cleanup:
1301 if (fp_out)
1302 {
1303 mutt_file_fclose(&fp_out);
1304 mutt_file_unlink(buf_string(tempfile));
1305 }
1306 mutt_file_fclose(&fp_smime_err);
1307 if (fp_tmp)
1308 {
1309 mutt_file_fclose(&fp_tmp);
1310 mutt_file_unlink(buf_string(smime_infile));
1311 }
1312 buf_pool_release(&tempfile);
1313 buf_pool_release(&smime_infile);
1314
1315 return b_enc;
1316}
char * buf_strdup(const struct Buffer *buf)
Copy a Buffer's string.
Definition buffer.c:571
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
int mutt_any_key_to_continue(const char *s)
Prompt the user to 'press any key' and wait.
Definition curs_lib.c:175
void mutt_file_unlink(const char *s)
Delete a file, carefully.
Definition file.c:156
#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
int filter_wait(pid_t pid)
Wait for the exit of a process and return its status.
Definition filter.c:228
#define _(a)
Definition message.h:28
size_t mutt_str_len(const char *a)
Calculate the length of a string, safely.
Definition string.c:503
#define PATH_MAX
Definition mutt.h:49
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
int mutt_write_mime_body(struct Body *b, FILE *fp, struct ConfigSubset *sub)
Write a MIME part.
Definition body.c:302
int mutt_write_mime_header(struct Body *b, FILE *fp, struct ConfigSubset *sub)
Create a MIME header.
Definition header.c:757
static pid_t smime_invoke_encrypt(FILE **fp_smime_in, FILE **fp_smime_out, FILE **fp_smime_err, int fp_smime_infd, int fp_smime_outfd, int fp_smime_errfd, const char *fname, const char *uids)
Use SMIME to encrypt a file.
Definition smime.c:1142
#define NONULL(x)
Definition string2.h:44
String manipulation buffer.
Definition buffer.h:36
Container for Accounts, Notifications.
Definition neomutt.h:41
struct ConfigSubset * sub
Inherited config items.
Definition neomutt.h:49
#define buf_mktemp(buf)
Definition tmp.h:33
#define mutt_file_mkstemp()
Definition tmp.h:36
+ Here is the call graph for this function: