2479{
2480 int needpass = -1;
2481 bool pgp_keyblock = false;
2482 bool clearsign = false;
2483 long bytes;
2484 LOFF_T last_pos;
2485 LOFF_T block_begin;
2486 LOFF_T block_end;
2487 char buf[8192] = { 0 };
2488 FILE *fp_out = NULL;
2489
2490 gpgme_error_t err = GPG_ERR_NO_ERROR;
2491 gpgme_data_t armored_data = NULL;
2492
2493 bool maybe_goodsig = true;
2494 bool have_any_sigs = false;
2495
2496 char body_charset[256] = { 0 };
2497 char *gpgcharset = NULL;
2498
2500
2501
2502
2504 mutt_str_copy(body_charset,
"iso-8859-1",
sizeof(body_charset));
2505
2507 {
2508 return -1;
2509 }
2511
2512 for (bytes = b->
length; bytes > 0;)
2513 {
2514
2515 block_begin = last_pos;
2516
2517 if (!fgets(buf,
sizeof(buf), state->
fp_in))
2518 break;
2519
2520 LOFF_T offset = ftello(state->
fp_in);
2521 if (offset < 0)
2522 {
2524 offset = 0;
2525 }
2526 bytes -= (offset - last_pos);
2527 last_pos = offset;
2528
2530 if (plen != 0)
2531 {
2532 needpass = 0;
2533 clearsign = false;
2534 pgp_keyblock = false;
2535
2537 {
2538 needpass = 1;
2539 }
2541 {
2542 clearsign = true;
2543 }
2545 {
2546 pgp_keyblock = true;
2547 }
2548 else
2549 {
2550
2554 continue;
2555 }
2556
2557
2558 while ((bytes > 0) && (fgets(buf,
sizeof(buf) - 1, state->
fp_in) != NULL))
2559 {
2560 offset = ftello(state->
fp_in);
2561 if (offset < 0)
2562 {
2564 offset = 0;
2565 }
2566 bytes -= (offset - last_pos);
2567 last_pos = offset;
2568
2569 if (needpass &&
mutt_str_equal(
"-----END PGP MESSAGE-----\n", buf))
2570 {
2571 break;
2572 }
2573
2574 if (!needpass && (
mutt_str_equal(
"-----END PGP SIGNATURE-----\n", buf) ||
2576 {
2577 break;
2578 }
2579
2580
2582 {
2583 size_t l = 0;
2586 if ((l =
mutt_str_len(gpgcharset)) > 0 && gpgcharset[l - 1] ==
'\n')
2587 gpgcharset[l - 1] = 0;
2590 }
2591 }
2592 block_end = ftello(state->
fp_in);
2593 if (block_end < 0)
2594 {
2596 block_end = 0;
2597 }
2598
2599 have_any_sigs = (have_any_sigs || (clearsign && (state->
flags &
STATE_VERIFY)));
2600
2601
2603 fseeko(state->
fp_in, block_end, 0);
2604
2605
2606 if (pgp_keyblock)
2607 {
2609 }
2611 {
2614
2615 if (clearsign)
2616 {
2617 err = gpgme_op_verify(ctx, armored_data, NULL, plaintext);
2618 }
2619 else
2620 {
2621 err = gpgme_op_decrypt_verify(ctx, armored_data, plaintext);
2622 if (gpg_err_code(err) == GPG_ERR_NO_DATA)
2623 {
2624
2625 gpgme_data_seek(armored_data, 0, SEEK_SET);
2626
2627 gpgme_data_release(plaintext);
2629 err = gpgme_op_verify(ctx, armored_data, NULL, plaintext);
2630 }
2631 }
2633
2634 gpgme_decrypt_result_t result = gpgme_op_decrypt_result(ctx);
2637
2638 if (err != GPG_ERR_NO_ERROR)
2639 {
2640 char errbuf[200] = { 0 };
2641
2642 snprintf(errbuf, sizeof(errbuf) - 1,
2643 _(
"Error: decryption/verification failed: %s\n"), gpgme_strerror(err));
2645 }
2646 else
2647 {
2648
2649
2651
2652 bool sig_stat = false;
2653 char *tmpfname = NULL;
2654
2655
2656 gpgme_verify_result_t verify_result = gpgme_op_verify_result(ctx);
2657 if (verify_result->signatures)
2658 sig_stat = true;
2659
2660 have_any_sigs = false;
2661 maybe_goodsig = false;
2663 {
2664 int res, idx;
2665 bool anybad = false;
2666
2668 have_any_sigs = true;
2670 {
2671 if (res == 1)
2672 anybad = true;
2673 }
2674 if (!anybad && idx)
2675 maybe_goodsig = true;
2676
2678 }
2679
2681 if (tmpfname)
2682 {
2683 unlink(tmpfname);
2685 }
2686 else
2687 {
2689 state_puts(state,
_(
"Error: copy data failed\n"));
2690 }
2691 }
2692 gpgme_data_release(plaintext);
2693 gpgme_release(ctx);
2694 }
2695
2696
2697
2698
2700 {
2701 if (needpass)
2703 else if (pgp_keyblock)
2705 else
2707 }
2708
2709 if (clearsign)
2710 {
2712 }
2713 else if (fp_out)
2714 {
2715 int c;
2716 char *expected_charset = gpgcharset && *gpgcharset ? gpgcharset : "utf-8";
2717 rewind(fp_out);
2721 {
2723 if ((c ==
'\n') && state->
prefix)
2725 }
2727 }
2728
2730 {
2732 if (needpass)
2734 else if (pgp_keyblock)
2736 else
2738 }
2739
2740
2741 gpgme_data_release(armored_data);
2743 }
2744 else
2745 {
2746
2747
2751 }
2752 }
2754
2755 b->
goodsig = (maybe_goodsig && have_any_sigs);
2756
2757 if (needpass == -1)
2758 {
2759 state_attach_puts(state,
_(
"[-- Error: could not find beginning of PGP message --]\n\n"));
2760 return 1;
2761 }
2763
2764 return err;
2765}
const char * cc_charset(void)
Get the cached value of $charset.
static void show_encryption_info(struct State *state, gpgme_decrypt_result_t result)
Show encryption information.
static gpgme_data_t create_gpgme_data(void)
Create a new GPGME data object.
gpgme_ctx_t create_gpgme_context(bool for_smime)
Create a new GPGME context.
#define SIGNED_MESSAGE(_y)
static gpgme_data_t file_to_data_object(FILE *fp, long offset, size_t length)
Create GPGME data object from file.
#define PUBLIC_KEY_BLOCK(_y)
static char * data_object_to_tempfile(gpgme_data_t data, FILE **fp_ret)
Copy a data object to a temporary file.
static void redraw_if_needed(gpgme_ctx_t ctx)
Accommodate for a redraw if needed.
static int pgp_gpgme_extract_keys(gpgme_data_t keydata, FILE **fp)
Write PGP keys to a file.
static void copy_clearsigned(gpgme_data_t data, struct State *state, char *charset)
Copy a clearsigned message.
static int show_one_sig_status(gpgme_ctx_t ctx, int idx, struct State *state)
Show information about one signature.
char * mutt_body_get_charset(struct Body *b, char *buf, size_t buflen)
Get a body's character set.
bool mutt_file_seek(FILE *fp, LOFF_T offset, int whence)
Wrapper for fseeko with error handling.
#define mutt_file_fclose(FP)
#define mutt_message(...)
#define mutt_debug(LEVEL,...)
@ LL_DEBUG2
Log at debug level 2.
@ LL_DEBUG1
Log at debug level 1.
#define FREE(x)
Free memory and set the pointer to NULL.
bool mutt_ch_check_charset(const char *cs, bool strict)
Does iconv understand a character set?
int mutt_ch_fgetconv(struct FgetConv *fc)
Convert a file's character set.
struct FgetConv * mutt_ch_fgetconv_open(FILE *fp, const char *from, const char *to, uint8_t flags)
Prepare a file for charset conversion.
void mutt_ch_fgetconv_close(struct FgetConv **ptr)
Close an fgetconv handle.
#define MUTT_ICONV_HOOK_FROM
apply charset-hooks to fromcode
void state_attach_puts(struct State *state, const char *t)
Write a string to the state.
#define state_puts(STATE, STR)
#define state_putc(STATE, STR)
@ STATE_VERIFY
Perform signature verification.
@ STATE_DISPLAY
Output is displayed to the user.
char * mutt_str_dup(const char *str)
Copy a string, safely.
bool mutt_str_equal(const char *a, const char *b)
Compare two strings.
bool mutt_strn_equal(const char *a, const char *b, size_t num)
Check for equality of two strings (to a maximum), safely.
size_t mutt_str_startswith(const char *str, const char *prefix)
Check whether a string starts with a prefix.
size_t mutt_str_len(const char *a)
Calculate the length of a string, safely.
size_t mutt_str_copy(char *dest, const char *src, size_t dsize)
Copy a string into a buffer (guaranteeing NUL-termination)
char * mutt_str_replace(char **p, const char *s)
Replace one string with another.
LOFF_T offset
offset where the actual data begins
LOFF_T length
length (in bytes) of attachment
bool goodsig
Good cryptographic signature.
Cursor for converting a file's encoding.
StateFlags flags
Flags, e.g. STATE_DISPLAY.
FILE * fp_in
File to read from.
const char * prefix
String to add to the beginning of each output line.