Send an email.
2032{
2034 FILE *fp_tmp = NULL;
2035 struct Body *pbody = NULL;
2036 int i;
2037 bool free_clear_content = false;
2038
2039 struct Body *clear_content = NULL;
2040 char *pgpkeylist = NULL;
2041
2042 char *pgp_sign_as = NULL;
2043 char *smime_sign_as = NULL;
2044 const char *tag = NULL;
2045 char *err = NULL;
2046 char *finalpath = NULL;
2047 struct Email *e_cur = NULL;
2048
2051
2052 int rc = -1;
2053
2056 else
2058
2060
2062 {
2063
2064
2067 return rc;
2068
2071 }
2072
2074 {
2076 {
2079 }
2081 {
2082 const char *
const c_smime_sign_as =
cs_subset_string(sub,
"smime_sign_as");
2084 }
2085 }
2086
2087
2088
2089
2090
2091 if (!e_templ)
2092 {
2094
2096 {
2098 if (rc < 0)
2099 {
2101 goto cleanup;
2102 }
2103 flags = rc;
2104
2105
2107 {
2110 }
2111 else
2112 {
2115 }
2116 }
2117
2119 {
2124 if (!fp_tmp)
2125 {
2127 goto cleanup;
2128 }
2129 }
2130
2133 }
2134
2135
2137 {
2138
2140
2142 }
2143
2145 {
2146
2147
2149 {
2152 e_templ->
body = pbody;
2153
2155 const char *ctype =
mutt_str_dup(c_content_type ? c_content_type :
"text/plain");
2161
2162 if (tempfile)
2163 {
2168 }
2169 else
2170 {
2176 }
2177 }
2178 else
2179 {
2184 }
2185
2186 if (!fp_tmp)
2187 {
2191 goto cleanup;
2192 }
2193 }
2194
2196
2198 {
2199
2200
2201
2202
2203
2204
2206 {
2210 }
2212 }
2213
2214 const bool c_reply_with_xorig =
cs_subset_bool(sub,
"reply_with_xorig");
2216 {
2217
2218
2219
2220
2221
2222
2223
2224
2226 {
2230 }
2231 }
2232
2235
2236 const bool c_resume_draft_files =
cs_subset_bool(sub,
"resume_draft_files");
2239 {
2242 {
2243 goto cleanup;
2244 }
2245
2247 if (c_hdrs)
2249
2250
2252
2255
2257 {
2259 }
2260
2264 if (!(flags &
SEND_BATCH) && !(c_auto_edit && c_edit_headers) &&
2266 {
2268 goto cleanup;
2269 }
2270
2271
2272
2273
2274
2275
2277 if (killfrom)
2278 {
2280 }
2281
2283 {
2284
2286
2287
2288
2289
2291 }
2292
2293
2294
2296
2297
2298
2299
2300
2302
2303
2304 if (killfrom)
2305 {
2307
2311 }
2312
2313 if (c_hdrs)
2315
2317 {
2319 {
2321 goto cleanup;
2322 }
2323 }
2324
2327
2331 {
2333 }
2334
2335
2338 {
2339 goto cleanup;
2340 }
2341
2343 {
2345 }
2346 }
2347
2348
2349
2350
2351
2353 {
2356 {
2358 }
2359 }
2360
2361
2362
2364
2365
2366
2367 {
2370 {
2372 if (c_real_name)
2374 }
2375 }
2376
2379
2381 {
2382 struct stat st = { 0 };
2383 time_t mtime;
2388 if (mtime == (time_t) -1)
2389 {
2391 goto cleanup;
2392 }
2393
2395
2398
2399
2400
2401
2402
2403
2404
2405
2406
2408 (((flags &
SEND_FORWARD) == 0) || (c_edit_headers && c_auto_edit) ||
2410 {
2411
2417 {
2419 goto cleanup;
2420 }
2421 else if (c_edit_headers)
2422 {
2426 }
2427 else
2428 {
2431 {
2432 if (mtime != st.st_mtime)
2434 }
2435 else
2436 {
2438 }
2439 }
2440
2442 }
2443
2445 {
2447 {
2448
2449 if ((mtime == st.st_mtime) && !e_templ->
body->
next &&
2451 {
2453 goto cleanup;
2454 }
2455 }
2456 else
2457 {
2459 }
2460 }
2461 }
2462
2463
2464
2465
2466
2467
2468
2469
2470
2471
2472
2475 {
2476 bool c_autocrypt = false;
2477 bool c_autocrypt_reply = false;
2478
2479#ifdef USE_AUTOCRYPT
2482#endif
2483
2485 {
2487 }
2488 else
2489 {
2490 const bool c_crypt_auto_sign =
cs_subset_bool(sub,
"crypt_auto_sign");
2491 const bool c_crypt_auto_encrypt =
cs_subset_bool(sub,
"crypt_auto_encrypt");
2492 const bool c_crypt_reply_encrypt =
cs_subset_bool(sub,
"crypt_reply_encrypt");
2493 const bool c_crypt_reply_sign =
cs_subset_bool(sub,
"crypt_reply_sign");
2494 const bool c_crypt_reply_sign_encrypted =
cs_subset_bool(sub,
"crypt_reply_sign_encrypted");
2495
2496 if (c_crypt_auto_sign)
2498 if (c_crypt_auto_encrypt)
2506
2507 const bool c_crypt_opportunistic_encrypt =
cs_subset_bool(sub,
"crypt_opportunistic_encrypt");
2508
2511 {
2512 const bool c_pgp_auto_inline =
cs_subset_bool(sub,
"pgp_auto_inline");
2513 const bool c_pgp_reply_inline =
cs_subset_bool(sub,
"pgp_reply_inline");
2514
2515 if (c_pgp_auto_inline)
2519 }
2520 }
2521
2522 const bool c_crypt_opportunistic_encrypt =
cs_subset_bool(sub,
"crypt_opportunistic_encrypt");
2523
2524 if (e_templ->
security || c_crypt_opportunistic_encrypt)
2525 {
2526 const bool c_crypt_auto_pgp =
cs_subset_bool(sub,
"crypt_auto_pgp");
2527 const bool c_crypt_auto_smime =
cs_subset_bool(sub,
"crypt_auto_smime");
2528
2529
2530
2531
2532
2533
2534
2535
2536 if (e_cur)
2537 {
2540 {
2542 }
2545 {
2547 }
2548 }
2549
2550 const bool c_smime_is_default =
cs_subset_bool(sub,
"smime_is_default");
2551
2552
2553
2555 {
2557 {
2559 }
2561 {
2563 }
2565 {
2567 }
2568 }
2569 }
2570
2571
2572 if (c_crypt_opportunistic_encrypt)
2573 {
2574
2575
2576
2578 {
2581 }
2582 }
2583
2584
2587 }
2588
2589
2590
2591
2593 {
2594 mutt_error(
_(
"No crypto backend configured. Disabling message security setting."));
2596 }
2597
2598
2599
2600
2602
2605 {
2606
2608 if (killfrom)
2609 {
2611 }
2613 if (killfrom)
2614 {
2616 }
2617 }
2618
2620
2622
2624 {
2625 main_loop:
2626
2630 if (i == -1)
2631 {
2632
2635 else
2637 goto cleanup;
2638 }
2639 else if (i == 1)
2640 {
2642 goto main_loop;
2644 rc = 1;
2645 goto cleanup;
2646 }
2647 }
2648
2650 {
2654 {
2656 {
2657 puts(
_(
"No recipients specified"));
2658 goto cleanup;
2659 }
2660
2662 goto main_loop;
2663 }
2664 }
2665
2667 {
2671 goto cleanup;
2672 goto main_loop;
2673 }
2674
2676
2679 {
2680
2683 goto main_loop;
2684 }
2685
2687 {
2689 goto main_loop;
2690 }
2691
2693 {
2695 goto main_loop;
2696 }
2697
2699 {
2700 goto main_loop;
2701 }
2702
2703 const bool c_confirm_empty_to =
cs_subset_bool(sub,
"confirm_empty_to");
2706 {
2707 goto main_loop;
2708 }
2709
2712
2713
2714
2715
2716
2718
2719
2720
2721
2722
2723
2724 clear_content = NULL;
2725 free_clear_content = false;
2726
2728 {
2730 {
2731
2732 clear_content = e_templ->
body;
2733
2736 {
2739
2741
2743
2745 {
2747 rc = -1;
2748 goto cleanup;
2749 }
2750
2751 goto main_loop;
2752 }
2754 }
2755
2756
2757
2758
2759
2760
2761
2762
2763
2764 if (clear_content && (e_templ->
body != clear_content) &&
2765 (e_templ->
body->
parts != clear_content))
2766 free_clear_content = true;
2767 }
2768
2771
2773
2774 const bool c_fcc_before_send =
cs_subset_bool(sub,
"fcc_before_send");
2775 if (c_fcc_before_send)
2776 {
2777 if (
save_fcc(m, e_templ, fcc, clear_content, pgpkeylist, flags, &finalpath, sub) &&
2779 {
2780
2781 puts(
_(
"Fcc failed. Aborting sending."));
2782 goto cleanup;
2783 }
2784 }
2785
2787 if (i < 0)
2788 {
2790 {
2792 ;
2795 {
2796 if (e_templ->
body != clear_content)
2797 {
2799 e_templ->
body = clear_content;
2800 }
2801 }
2803 {
2807 {
2809 }
2810 }
2811
2820 goto main_loop;
2821 }
2822 else
2823 {
2824 puts(
_(
"Could not send the message"));
2825 goto cleanup;
2826 }
2827 }
2828
2829 if (!c_fcc_before_send)
2830 save_fcc(m, e_templ, fcc, clear_content, pgpkeylist, flags, &finalpath, sub);
2831
2833 {
2837#ifdef USE_NOTMUCH
2839 if (c_nm_record)
2841#endif
2843 }
2844
2847
2850
2851
2852
2854 {
2856 {
2857 struct Email **ep = NULL;
2859 {
2860 struct Email *e = *ep;
2862 }
2863 }
2864 }
2865
2866 rc = 0;
2867
2868cleanup:
2870
2872 {
2874 {
2877 }
2879 {
2881 FREE(&smime_sign_as);
2882 }
2883 }
2884
2888
2890 return rc;
2891}
int mutt_addrlist_count_recips(const struct AddressList *al)
Count the number of Addresses with valid recipients.
void mutt_expand_aliases_env(struct Envelope *env)
Expand aliases in all the fields of an Envelope.
struct Buffer * buf_new(const char *str)
Allocate a new Buffer.
char * buf_strdup(const struct Buffer *buf)
Copy a Buffer's string.
@ CMD_SEND_HOOK
:send-hook
@ CMD_SEND2_HOOK
:send2-hook
@ CMD_REPLY_HOOK
:reply-hook
#define MUTT_COMPOSE_NOFREEHEADER
Don't free the header when closing compose dialog.
@ MUTT_NNTP
'NNTP' (Usenet) Mailbox type
int crypt_get_keys(struct Email *e, char **keylist, bool oppenc_mode)
Check we have all the keys we need.
bool crypt_has_module_backend(SecurityFlags type)
Is there a crypto backend for a given type?
void mutt_edit_file(const char *editor, const char *file)
Let the user edit a file.
struct Body * mutt_body_new(void)
Create a new Body.
void mutt_parse_content_type(const char *s, struct Body *b)
Parse a content type.
bool mutt_parse_mailto(struct Envelope *env, char **body, const char *src)
Parse a mailto:// url.
struct Envelope * mutt_env_new(void)
Create a new Envelope.
void mutt_env_to_local(struct Envelope *env)
Convert an Envelope's Address fields to local format.
char * msgid_generate(void)
Generate a Message-ID.
time_t mutt_file_decrease_mtime(const char *fp, struct stat *st)
Decrease a file's modification time by 1 second.
void mutt_set_flag(struct Mailbox *m, struct Email *e, enum MessageType flag, bool bf, bool upd_mbox)
Set a flag on an email.
bool OptGui
(pseudo) when the gui (and curses) are started
int dlg_compose(struct Email *e, struct Buffer *fcc, uint8_t flags, struct ConfigSubset *sub)
Allow the user to edit the message envelope -.
#define mutt_debug(LEVEL,...)
void mutt_select_fcc(struct Buffer *path, struct Email *e)
Select the FCC path for an email.
@ LL_DEBUG5
Log at debug level 5.
@ LL_DEBUG1
Log at debug level 1.
@ DISP_INLINE
Content is inline.
@ MUTT_REPLIED
Messages that have been replied to.
bool mutt_edit_attachment(struct Body *b)
Edit an attachment.
void mutt_sleep(short s)
Sleep for a while.
void pretty_mailbox(struct Buffer *buf)
Shorten a mailbox path using '~' or '='.
bool mutt_needs_mailcap(struct Body *b)
Does this type need a mailcap entry do display.
@ SEC_NONE
No flags are set.
@ SEC_INLINE
Email has an inline signature.
@ SEC_AUTOCRYPT_OVERRIDE
(Autocrypt) Indicates manual set/unset of encryption
int nm_record_message(struct Mailbox *m, char *path, struct Email *e)
Add a message to the Notmuch database.
void mutt_param_set(struct ParameterList *pl, const char *attribute, const char *value)
Set a Parameter.
int mutt_num_postponed(struct Mailbox *m, bool force)
Return the number of postponed messages.
int mutt_get_postponed(struct Mailbox *m_cur, struct Email *hdr, struct Email **cur, struct Buffer *fcc)
Recall a postponed message.
enum QuadOption query_yesorno(const char *prompt, enum QuadOption def)
Ask the user a Yes/No question.
void mutt_rfc3676_space_stuff(struct Email *e)
Perform RFC3676 space stuffing on an Email.
static int postpone_message(struct Email *e_post, struct Email *e_cur, const char *fcc, SendFlags flags, struct ConfigSubset *sub)
Save an Email for another day.
static bool is_reply(struct Email *reply, struct Email *orig)
Is one email a reply to another?
static int save_fcc(struct Mailbox *m, struct Email *e, struct Buffer *fcc, struct Body *clear_content, char *pgpkeylist, SendFlags flags, char **finalpath, struct ConfigSubset *sub)
Save an Email to a 'sent mail' folder.
static int envelope_defaults(struct Envelope *env, struct EmailArray *ea, SendFlags flags, struct ConfigSubset *sub)
Fill in some defaults for a new email.
void mutt_fix_reply_recipients(struct Envelope *env, struct ConfigSubset *sub)
Remove duplicate recipients.
static int generate_body(FILE *fp_tmp, struct Email *e, SendFlags flags, struct Mailbox *m, struct EmailArray *ea, struct ConfigSubset *sub)
Create a new email body.
static void mutt_make_greeting(struct Email *e, FILE *fp_out, struct ConfigSubset *sub)
Add greetings string.
static int invoke_mta(struct Mailbox *m, struct Email *e, struct ConfigSubset *sub)
Send an email.
static void process_user_recips(struct Envelope *env)
Process the user headers.
static void process_user_header(struct Envelope *env)
Process the user headers.
static int edit_envelope(struct Envelope *en, SendFlags flags, struct ConfigSubset *sub)
Edit Envelope fields.
static bool abort_for_missing_attachments(const struct Body *b, struct ConfigSubset *sub)
Should we abort sending because of missing attachments?
static void set_reverse_name(struct AddressList *al, struct Envelope *env, struct ConfigSubset *sub)
Try to set the 'from' field from the recipients.
static void fix_end_of_file(const char *data)
Ensure a file ends with a linefeed.
static void append_signature(FILE *fp, struct ConfigSubset *sub)
Append a signature to an email.
@ SEND_CONSUMED_STDIN
stdin has been read; don't read it twice
@ SEND_NO_FREE_HEADER
Used by the -E flag.
@ SEND_CLI_CRYPTO
Enable message security in modes that by default don't enable it.
@ SEND_DRAFT_FILE
Used by the -H flag.
@ SEND_POSTPONED_FCC
Used by mutt_get_postponed() to signal that the Mutt-Fcc header field was present.
@ SEND_POSTPONED
Recall a postponed email.
void mutt_update_encoding(struct Body *b, struct ConfigSubset *sub)
Update the encoding type.
bool unlink
If true, filename should be unlink()ed before free()ing this structure.
bool use_disp
Content-Disposition uses filename= ?
unsigned int disposition
content-disposition, ContentDisposition
bool replied
Email has been replied to.
struct AddressList x_original_to
Email's 'X-Original-to'.
char * list_post
This stores a mailto URL, or nothing.
enum MailboxType type
Mailbox type.
void * mdata
Driver specific data.
NNTP-specific Mailbox data -.
int cs_subset_str_string_set(const struct ConfigSubset *sub, const char *name, const char *value, struct Buffer *err)
Set a config item by string.
#define buf_mktemp_draft(buf)