1 ;;; mml2015.el --- MIME Security with Pretty Good Privacy (PGP)
2 ;; Copyright (C) 2000, 2001, 2002, 2003, 2004, 2005
3 ;; Free Software Foundation, Inc.
5 ;; Author: Shenghuo Zhu <zsh@cs.rochester.edu>
6 ;; Keywords: PGP MIME MML
8 ;; This file is part of GNU Emacs.
10 ;; GNU Emacs is free software; you can redistribute it and/or modify
11 ;; it under the terms of the GNU General Public License as published
12 ;; by the Free Software Foundation; either version 2, or (at your
13 ;; option) any later version.
15 ;; GNU Emacs is distributed in the hope that it will be useful, but
16 ;; WITHOUT ANY WARRANTY; without even the implied warranty of
17 ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 ;; General Public License for more details.
20 ;; You should have received a copy of the GNU General Public License
21 ;; along with GNU Emacs; see the file COPYING. If not, write to the
22 ;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
23 ;; Boston, MA 02110-1301, USA.
27 ;; RFC 2015 is updated by RFC 3156, this file should be compatible
32 (eval-when-compile (require 'cl))
37 (defvar mml2015-use (or
41 (and (fboundp 'pgg-sign-region)
46 (and (fboundp 'gpg-sign-detached)
50 (and (fboundp 'mc-encrypt-generic)
51 (fboundp 'mc-sign-generic)
52 (fboundp 'mc-cleanup-recipient-headers)
54 "The package used for PGP/MIME.")
56 ;; Something is not RFC2015.
57 (defvar mml2015-function-alist
58 '((mailcrypt mml2015-mailcrypt-sign
59 mml2015-mailcrypt-encrypt
60 mml2015-mailcrypt-verify
61 mml2015-mailcrypt-decrypt
62 mml2015-mailcrypt-clear-verify
63 mml2015-mailcrypt-clear-decrypt)
68 mml2015-gpg-clear-verify
69 mml2015-gpg-clear-decrypt)
74 mml2015-pgg-clear-verify
75 mml2015-pgg-clear-decrypt))
76 "Alist of PGP/MIME functions.")
78 (defvar mml2015-result-buffer nil)
80 (defcustom mml2015-unabbrev-trust-alist
81 '(("TRUST_UNDEFINED" . nil)
83 ("TRUST_MARGINAL" . t)
85 ("TRUST_ULTIMATE" . t))
86 "Map GnuPG trust output values to a boolean saying if you trust the key."
89 :type '(repeat (cons (regexp :tag "GnuPG output regexp")
90 (boolean :tag "Trust key"))))
95 (autoload 'mailcrypt-decrypt "mailcrypt")
96 (autoload 'mailcrypt-verify "mailcrypt")
97 (autoload 'mc-pgp-always-sign "mailcrypt")
98 (autoload 'mc-encrypt-generic "mc-toplev")
99 (autoload 'mc-cleanup-recipient-headers "mc-toplev")
100 (autoload 'mc-sign-generic "mc-toplev"))
103 (defvar mc-default-scheme)
106 (defvar mml2015-decrypt-function 'mailcrypt-decrypt)
107 (defvar mml2015-verify-function 'mailcrypt-verify)
109 (defun mml2015-format-error (err)
110 (if (stringp (cadr err))
112 (format "%S" (cdr err))))
114 (defun mml2015-mailcrypt-decrypt (handle ctl)
116 (let (child handles result)
117 (unless (setq child (mm-find-part-by-type
119 "application/octet-stream" nil t))
120 (mm-set-handle-multipart-parameter
121 mm-security-handle 'gnus-info "Corrupted")
122 (throw 'error handle))
124 (mm-insert-part child)
127 (funcall mml2015-decrypt-function)
129 (mm-set-handle-multipart-parameter
130 mm-security-handle 'gnus-details (mml2015-format-error err))
133 (mm-set-handle-multipart-parameter
134 mm-security-handle 'gnus-details "Quit.")
137 (mm-set-handle-multipart-parameter
138 mm-security-handle 'gnus-info "Failed")
139 (throw 'error handle))
140 (setq handles (mm-dissect-buffer t)))
141 (mm-destroy-parts handle)
142 (mm-set-handle-multipart-parameter
143 mm-security-handle 'gnus-info
145 (let ((sig (with-current-buffer mml2015-result-buffer
146 (mml2015-gpg-extract-signature-details))))
147 (concat ", Signer: " sig))))
148 (if (listp (car handles))
152 (defun mml2015-mailcrypt-clear-decrypt ()
156 (funcall mml2015-decrypt-function)
158 (mm-set-handle-multipart-parameter
159 mm-security-handle 'gnus-details (mml2015-format-error err))
162 (mm-set-handle-multipart-parameter
163 mm-security-handle 'gnus-details "Quit.")
166 (mm-set-handle-multipart-parameter
167 mm-security-handle 'gnus-info "OK")
168 (mm-set-handle-multipart-parameter
169 mm-security-handle 'gnus-info "Failed"))))
171 (defun mml2015-fix-micalg (alg)
173 ;; Mutt/1.2.5i has seen sending micalg=php-sha1
174 (upcase (if (string-match "^p[gh]p-" alg)
175 (substring alg (match-end 0))
178 (defun mml2015-mailcrypt-verify (handle ctl)
181 (unless (setq part (mm-find-raw-part-by-type
182 ctl (or (mm-handle-multipart-ctl-parameter
184 "application/pgp-signature")
186 (mm-set-handle-multipart-parameter
187 mm-security-handle 'gnus-info "Corrupted")
188 (throw 'error handle))
190 (insert "-----BEGIN PGP SIGNED MESSAGE-----\n")
191 (insert (format "Hash: %s\n\n"
192 (or (mml2015-fix-micalg
193 (mm-handle-multipart-ctl-parameter
197 (narrow-to-region (point) (point))
199 (goto-char (point-min))
201 (if (looking-at "^-")
204 (unless (setq part (mm-find-part-by-type
205 (cdr handle) "application/pgp-signature" nil t))
206 (mm-set-handle-multipart-parameter
207 mm-security-handle 'gnus-info "Corrupted")
208 (throw 'error handle))
210 (narrow-to-region (point) (point))
211 (mm-insert-part part)
212 (goto-char (point-min))
213 (if (re-search-forward "^-----BEGIN PGP [^-]+-----\r?$" nil t)
214 (replace-match "-----BEGIN PGP SIGNATURE-----" t t))
215 (if (re-search-forward "^-----END PGP [^-]+-----\r?$" nil t)
216 (replace-match "-----END PGP SIGNATURE-----" t t)))
217 (let ((mc-gpg-debug-buffer (get-buffer-create " *gnus gpg debug*")))
218 (unless (condition-case err
220 (funcall mml2015-verify-function)
221 (if (get-buffer " *mailcrypt stderr temp")
222 (mm-set-handle-multipart-parameter
223 mm-security-handle 'gnus-details
224 (with-current-buffer " *mailcrypt stderr temp"
226 (if (get-buffer " *mailcrypt stdout temp")
227 (kill-buffer " *mailcrypt stdout temp"))
228 (if (get-buffer " *mailcrypt stderr temp")
229 (kill-buffer " *mailcrypt stderr temp"))
230 (if (get-buffer " *mailcrypt status temp")
231 (kill-buffer " *mailcrypt status temp"))
232 (if (get-buffer mc-gpg-debug-buffer)
233 (kill-buffer mc-gpg-debug-buffer)))
235 (mm-set-handle-multipart-parameter
236 mm-security-handle 'gnus-details (mml2015-format-error err))
239 (mm-set-handle-multipart-parameter
240 mm-security-handle 'gnus-details "Quit.")
242 (mm-set-handle-multipart-parameter
243 mm-security-handle 'gnus-info "Failed")
244 (throw 'error handle))))
245 (mm-set-handle-multipart-parameter
246 mm-security-handle 'gnus-info "OK")
249 (defun mml2015-mailcrypt-clear-verify ()
250 (let ((mc-gpg-debug-buffer (get-buffer-create " *gnus gpg debug*")))
251 (if (condition-case err
253 (funcall mml2015-verify-function)
254 (if (get-buffer " *mailcrypt stderr temp")
255 (mm-set-handle-multipart-parameter
256 mm-security-handle 'gnus-details
257 (with-current-buffer " *mailcrypt stderr temp"
259 (if (get-buffer " *mailcrypt stdout temp")
260 (kill-buffer " *mailcrypt stdout temp"))
261 (if (get-buffer " *mailcrypt stderr temp")
262 (kill-buffer " *mailcrypt stderr temp"))
263 (if (get-buffer " *mailcrypt status temp")
264 (kill-buffer " *mailcrypt status temp"))
265 (if (get-buffer mc-gpg-debug-buffer)
266 (kill-buffer mc-gpg-debug-buffer)))
268 (mm-set-handle-multipart-parameter
269 mm-security-handle 'gnus-details (mml2015-format-error err))
272 (mm-set-handle-multipart-parameter
273 mm-security-handle 'gnus-details "Quit.")
275 (mm-set-handle-multipart-parameter
276 mm-security-handle 'gnus-info "OK")
277 (mm-set-handle-multipart-parameter
278 mm-security-handle 'gnus-info "Failed"))))
280 (defun mml2015-mailcrypt-sign (cont)
281 (mc-sign-generic (message-options-get 'message-sender)
283 (let ((boundary (mml-compute-boundary cont))
285 (goto-char (point-min))
286 (unless (re-search-forward "^-----BEGIN PGP SIGNED MESSAGE-----\r?$" nil t)
287 (error "Cannot find signed begin line"))
288 (goto-char (match-beginning 0))
290 (unless (looking-at "Hash:[ \t]*\\([a-zA-Z0-9]+\\)")
291 (error "Cannot not find PGP hash"))
292 (setq hash (match-string 1))
293 (unless (re-search-forward "^$" nil t)
294 (error "Cannot not find PGP message"))
296 (delete-region (point-min) (point))
297 (insert (format "Content-Type: multipart/signed; boundary=\"%s\";\n"
299 (insert (format "\tmicalg=pgp-%s; protocol=\"application/pgp-signature\"\n"
301 (insert (format "\n--%s\n" boundary))
303 (goto-char (point-max))
304 (unless (re-search-backward "^-----END PGP SIGNATURE-----\r?$" nil t)
305 (error "Cannot find signature part"))
306 (replace-match "-----END PGP MESSAGE-----" t t)
307 (goto-char (match-beginning 0))
308 (unless (re-search-backward "^-----BEGIN PGP SIGNATURE-----\r?$"
310 (error "Cannot find signature part"))
311 (replace-match "-----BEGIN PGP MESSAGE-----" t t)
312 (goto-char (match-beginning 0))
314 (narrow-to-region point (point))
316 (while (re-search-forward "^- -" nil t)
317 (replace-match "-" t t))
318 (goto-char (point-max)))
319 (insert (format "--%s\n" boundary))
320 (insert "Content-Type: application/pgp-signature\n\n")
321 (goto-char (point-max))
322 (insert (format "--%s--\n" boundary))
323 (goto-char (point-max))))
325 (defun mml2015-mailcrypt-encrypt (cont &optional sign)
326 (let ((mc-pgp-always-sign
327 (or mc-pgp-always-sign
329 (eq t (or (message-options-get 'message-sign-encrypt)
331 'message-sign-encrypt
332 (or (y-or-n-p "Sign the message? ")
335 (mm-with-unibyte-current-buffer
337 (or (message-options-get 'message-recipients)
338 (message-options-set 'message-recipients
339 (mc-cleanup-recipient-headers
340 (read-string "Recipients: "))))
342 (message-options-get 'message-sender))))
343 (goto-char (point-min))
344 (unless (looking-at "-----BEGIN PGP MESSAGE-----")
345 (error "Fail to encrypt the message"))
346 (let ((boundary (mml-compute-boundary cont)))
347 (insert (format "Content-Type: multipart/encrypted; boundary=\"%s\";\n"
349 (insert "\tprotocol=\"application/pgp-encrypted\"\n\n")
350 (insert (format "--%s\n" boundary))
351 (insert "Content-Type: application/pgp-encrypted\n\n")
352 (insert "Version: 1\n\n")
353 (insert (format "--%s\n" boundary))
354 (insert "Content-Type: application/octet-stream\n\n")
355 (goto-char (point-max))
356 (insert (format "--%s--\n" boundary))
357 (goto-char (point-max))))
362 (autoload 'gpg-decrypt "gpg")
363 (autoload 'gpg-verify "gpg")
364 (autoload 'gpg-verify-cleartext "gpg")
365 (autoload 'gpg-sign-detached "gpg")
366 (autoload 'gpg-sign-encrypt "gpg")
367 (autoload 'gpg-encrypt "gpg")
368 (autoload 'gpg-passphrase-read "gpg"))
370 (defun mml2015-gpg-passphrase ()
371 (or (message-options-get 'gpg-passphrase)
372 (message-options-set 'gpg-passphrase (gpg-passphrase-read))))
374 (defun mml2015-gpg-decrypt-1 ()
375 (let ((cipher (current-buffer)) plain result)
376 (if (with-temp-buffer
378 (gpg-decrypt cipher (setq plain (current-buffer))
379 mml2015-result-buffer nil)
380 (mm-set-handle-multipart-parameter
381 mm-security-handle 'gnus-details
382 (with-current-buffer mml2015-result-buffer
386 (insert-buffer-substring plain)
387 (goto-char (point-min))
388 (while (search-forward "\r\n" nil t)
389 (replace-match "\n" t t))))
391 ;; Some wrong with the return value, check plain text buffer.
392 (if (> (point-max) (point-min))
396 (defun mml2015-gpg-decrypt (handle ctl)
397 (let ((mml2015-decrypt-function 'mml2015-gpg-decrypt-1))
398 (mml2015-mailcrypt-decrypt handle ctl)))
400 (defun mml2015-gpg-clear-decrypt ()
402 (setq result (mml2015-gpg-decrypt-1))
404 (mm-set-handle-multipart-parameter
405 mm-security-handle 'gnus-info "OK")
406 (mm-set-handle-multipart-parameter
407 mm-security-handle 'gnus-info "Failed"))))
409 (defun mml2015-gpg-pretty-print-fpr (fingerprint)
411 (fpr-length (string-width fingerprint))
414 (setq fingerprint (string-to-list fingerprint))
416 (setq fpr-length (- fpr-length 4))
417 (setq slice (butlast fingerprint fpr-length))
418 (setq fingerprint (nthcdr 4 fingerprint))
419 (setq n-slice (1+ n-slice))
425 (otherwise (concat " " slice))))))
428 (defun mml2015-gpg-extract-signature-details ()
429 (goto-char (point-min))
430 (let* ((expired (re-search-forward
431 "^\\[GNUPG:\\] SIGEXPIRED$"
433 (signer (and (re-search-forward
434 "^\\[GNUPG:\\] GOODSIG \\([0-9A-Za-z]*\\) \\(.*\\)$"
436 (cons (match-string 1) (match-string 2))))
437 (fprint (and (re-search-forward
438 "^\\[GNUPG:\\] VALIDSIG \\([0-9a-zA-Z]*\\) "
441 (trust (and (re-search-forward
442 "^\\[GNUPG:\\] \\(TRUST_.*\\)$"
446 (cdr (assoc trust mml2015-unabbrev-trust-alist))))
447 (cond ((and signer fprint)
449 (unless trust-good-enough-p
450 (concat "\nUntrusted, Fingerprint: "
451 (mml2015-gpg-pretty-print-fpr fprint)))
453 (format "\nWARNING: Signature from expired key (%s)"
456 "^\\(gpg: \\)?Good signature from \"\\(.*\\)\"$" nil t)
459 "From unknown user"))))
461 (defun mml2015-gpg-verify (handle ctl)
463 (let (part message signature info-is-set-p)
464 (unless (setq part (mm-find-raw-part-by-type
465 ctl (or (mm-handle-multipart-ctl-parameter
467 "application/pgp-signature")
469 (mm-set-handle-multipart-parameter
470 mm-security-handle 'gnus-info "Corrupted")
471 (throw 'error handle))
473 (setq message (current-buffer))
475 ;; Convert <LF> to <CR><LF> in verify mode. Sign and
476 ;; clearsign use --textmode. The conversion is not necessary.
477 ;; In clearverify, the conversion is not necessary either.
478 (goto-char (point-min))
481 (unless (eq (char-before) ?\r)
486 (setq signature (current-buffer))
487 (unless (setq part (mm-find-part-by-type
488 (cdr handle) "application/pgp-signature" nil t))
489 (mm-set-handle-multipart-parameter
490 mm-security-handle 'gnus-info "Corrupted")
491 (throw 'error handle))
492 (mm-insert-part part)
493 (unless (condition-case err
495 (gpg-verify message signature mml2015-result-buffer)
496 (mm-set-handle-multipart-parameter
497 mm-security-handle 'gnus-details
498 (with-current-buffer mml2015-result-buffer
501 (mm-set-handle-multipart-parameter
502 mm-security-handle 'gnus-details (mml2015-format-error err))
503 (mm-set-handle-multipart-parameter
504 mm-security-handle 'gnus-info "Error.")
505 (setq info-is-set-p t)
508 (mm-set-handle-multipart-parameter
509 mm-security-handle 'gnus-details "Quit.")
510 (mm-set-handle-multipart-parameter
511 mm-security-handle 'gnus-info "Quit.")
512 (setq info-is-set-p t)
514 (unless info-is-set-p
515 (mm-set-handle-multipart-parameter
516 mm-security-handle 'gnus-info "Failed"))
517 (throw 'error handle)))
518 (mm-set-handle-multipart-parameter
519 mm-security-handle 'gnus-info
520 (with-current-buffer mml2015-result-buffer
521 (mml2015-gpg-extract-signature-details))))
524 (defun mml2015-gpg-clear-verify ()
525 (if (condition-case err
527 (gpg-verify-cleartext (current-buffer) mml2015-result-buffer)
528 (mm-set-handle-multipart-parameter
529 mm-security-handle 'gnus-details
530 (with-current-buffer mml2015-result-buffer
533 (mm-set-handle-multipart-parameter
534 mm-security-handle 'gnus-details (mml2015-format-error err))
537 (mm-set-handle-multipart-parameter
538 mm-security-handle 'gnus-details "Quit.")
540 (mm-set-handle-multipart-parameter
541 mm-security-handle 'gnus-info
542 (with-current-buffer mml2015-result-buffer
543 (mml2015-gpg-extract-signature-details)))
544 (mm-set-handle-multipart-parameter
545 mm-security-handle 'gnus-info "Failed")))
547 (defun mml2015-gpg-sign (cont)
548 (let ((boundary (mml-compute-boundary cont))
549 (text (current-buffer)) signature)
550 (goto-char (point-max))
554 (unless (gpg-sign-detached text (setq signature (current-buffer))
555 mml2015-result-buffer
557 (message-options-get 'message-sender)
558 t t) ; armor & textmode
559 (unless (> (point-max) (point-min))
560 (pop-to-buffer mml2015-result-buffer)
561 (error "Sign error")))
562 (goto-char (point-min))
563 (while (re-search-forward "\r+$" nil t)
564 (replace-match "" t t))
566 (goto-char (point-min))
567 (insert (format "Content-Type: multipart/signed; boundary=\"%s\";\n"
569 ;;; FIXME: what is the micalg?
570 (insert "\tmicalg=pgp-sha1; protocol=\"application/pgp-signature\"\n")
571 (insert (format "\n--%s\n" boundary))
572 (goto-char (point-max))
573 (insert (format "\n--%s\n" boundary))
574 (insert "Content-Type: application/pgp-signature\n\n")
575 (insert-buffer-substring signature)
576 (goto-char (point-max))
577 (insert (format "--%s--\n" boundary))
578 (goto-char (point-max)))))
580 (defun mml2015-gpg-encrypt (cont &optional sign)
581 (let ((boundary (mml-compute-boundary cont))
582 (text (current-buffer))
584 (mm-with-unibyte-current-buffer
586 ;; set up a function to call the correct gpg encrypt routine
587 ;; with the right arguments. (FIXME: this should be done
589 (flet ((gpg-encrypt-func
590 (sign plaintext ciphertext result recipients &optional
591 passphrase sign-with-key armor textmode)
594 plaintext ciphertext result recipients passphrase
595 sign-with-key armor textmode)
597 plaintext ciphertext result recipients passphrase
599 (unless (gpg-encrypt-func
600 sign ; passed in when using signencrypt
601 text (setq cipher (current-buffer))
602 mml2015-result-buffer
605 (message-options-get 'message-recipients)
606 (message-options-set 'message-recipients
607 (read-string "Recipients: ")))
610 (message-options-get 'message-sender)
611 t t) ; armor & textmode
612 (unless (> (point-max) (point-min))
613 (pop-to-buffer mml2015-result-buffer)
614 (error "Encrypt error"))))
615 (goto-char (point-min))
616 (while (re-search-forward "\r+$" nil t)
617 (replace-match "" t t))
619 (delete-region (point-min) (point-max))
620 (insert (format "Content-Type: multipart/encrypted; boundary=\"%s\";\n"
622 (insert "\tprotocol=\"application/pgp-encrypted\"\n\n")
623 (insert (format "--%s\n" boundary))
624 (insert "Content-Type: application/pgp-encrypted\n\n")
625 (insert "Version: 1\n\n")
626 (insert (format "--%s\n" boundary))
627 (insert "Content-Type: application/octet-stream\n\n")
628 (insert-buffer-substring cipher)
629 (goto-char (point-max))
630 (insert (format "--%s--\n" boundary))
631 (goto-char (point-max))))))
636 (defvar pgg-default-user-id)
637 (defvar pgg-errors-buffer)
638 (defvar pgg-output-buffer))
641 (autoload 'pgg-decrypt-region "pgg")
642 (autoload 'pgg-verify-region "pgg")
643 (autoload 'pgg-sign-region "pgg")
644 (autoload 'pgg-encrypt-region "pgg"))
646 (defun mml2015-pgg-decrypt (handle ctl)
648 (let ((pgg-errors-buffer mml2015-result-buffer)
649 child handles result decrypt-status)
650 (unless (setq child (mm-find-part-by-type
652 "application/octet-stream" nil t))
653 (mm-set-handle-multipart-parameter
654 mm-security-handle 'gnus-info "Corrupted")
655 (throw 'error handle))
657 (mm-insert-part child)
658 (if (condition-case err
660 (pgg-decrypt-region (point-min) (point-max))
662 (with-current-buffer mml2015-result-buffer
664 (mm-set-handle-multipart-parameter
665 mm-security-handle 'gnus-details
668 (mm-set-handle-multipart-parameter
669 mm-security-handle 'gnus-details (mml2015-format-error err))
672 (mm-set-handle-multipart-parameter
673 mm-security-handle 'gnus-details "Quit.")
675 (with-current-buffer pgg-output-buffer
676 (goto-char (point-min))
677 (while (search-forward "\r\n" nil t)
678 (replace-match "\n" t t))
679 (setq handles (mm-dissect-buffer t))
680 (mm-destroy-parts handle)
681 (mm-set-handle-multipart-parameter
682 mm-security-handle 'gnus-info "OK")
683 (mm-set-handle-multipart-parameter
684 mm-security-handle 'gnus-details
685 (concat decrypt-status
686 (when (stringp (car handles))
687 "\n" (mm-handle-multipart-ctl-parameter
688 handles 'gnus-details))))
689 (if (listp (car handles))
692 (mm-set-handle-multipart-parameter
693 mm-security-handle 'gnus-info "Failed")
694 (throw 'error handle))))))
696 (defun mml2015-pgg-clear-decrypt ()
697 (let ((pgg-errors-buffer mml2015-result-buffer))
699 (pgg-decrypt-region (point-min) (point-max))
700 (mm-set-handle-multipart-parameter
701 mm-security-handle 'gnus-details
702 (with-current-buffer mml2015-result-buffer
706 (insert-buffer-substring pgg-output-buffer)
707 (goto-char (point-min))
708 (while (search-forward "\r\n" nil t)
709 (replace-match "\n" t t))
710 (mm-set-handle-multipart-parameter
711 mm-security-handle 'gnus-info "OK"))
712 (mm-set-handle-multipart-parameter
713 mm-security-handle 'gnus-info "Failed"))))
715 (defun mml2015-pgg-verify (handle ctl)
716 (let ((pgg-errors-buffer mml2015-result-buffer)
717 signature-file part signature)
718 (if (or (null (setq part (mm-find-raw-part-by-type
719 ctl (or (mm-handle-multipart-ctl-parameter
721 "application/pgp-signature")
723 (null (setq signature (mm-find-part-by-type
724 (cdr handle) "application/pgp-signature" nil t))))
726 (mm-set-handle-multipart-parameter
727 mm-security-handle 'gnus-info "Corrupted")
731 ;; Convert <LF> to <CR><LF> in verify mode. Sign and
732 ;; clearsign use --textmode. The conversion is not necessary.
733 ;; In clearverify, the conversion is not necessary either.
734 (goto-char (point-min))
737 (unless (eq (char-before) ?\r)
741 (with-temp-file (setq signature-file (mm-make-temp-file "pgg"))
742 (mm-insert-part signature))
743 (if (condition-case err
745 (pgg-verify-region (point-min) (point-max)
747 (goto-char (point-min))
748 (while (search-forward "\r\n" nil t)
749 (replace-match "\n" t t))
750 (mm-set-handle-multipart-parameter
751 mm-security-handle 'gnus-details
752 (concat (with-current-buffer pgg-output-buffer
754 (with-current-buffer pgg-errors-buffer
757 (mm-set-handle-multipart-parameter
758 mm-security-handle 'gnus-details (mml2015-format-error err))
761 (mm-set-handle-multipart-parameter
762 mm-security-handle 'gnus-details "Quit.")
765 (delete-file signature-file)
766 (mm-set-handle-multipart-parameter
767 mm-security-handle 'gnus-info
768 (with-current-buffer pgg-errors-buffer
769 (mml2015-gpg-extract-signature-details))))
770 (delete-file signature-file)
771 (mm-set-handle-multipart-parameter
772 mm-security-handle 'gnus-info "Failed")))))
775 (defun mml2015-pgg-clear-verify ()
776 (let ((pgg-errors-buffer mml2015-result-buffer)
777 (text (buffer-string))
778 (coding-system buffer-file-coding-system))
779 (if (condition-case err
781 (mm-with-unibyte-buffer
782 (insert (encode-coding-string text coding-system))
783 (pgg-verify-region (point-min) (point-max) nil t))
784 (goto-char (point-min))
785 (while (search-forward "\r\n" nil t)
786 (replace-match "\n" t t))
787 (mm-set-handle-multipart-parameter
788 mm-security-handle 'gnus-details
789 (concat (with-current-buffer pgg-output-buffer
791 (with-current-buffer pgg-errors-buffer
794 (mm-set-handle-multipart-parameter
795 mm-security-handle 'gnus-details (mml2015-format-error err))
798 (mm-set-handle-multipart-parameter
799 mm-security-handle 'gnus-details "Quit.")
801 (mm-set-handle-multipart-parameter
802 mm-security-handle 'gnus-info
803 (with-current-buffer pgg-errors-buffer
804 (mml2015-gpg-extract-signature-details)))
805 (mm-set-handle-multipart-parameter
806 mm-security-handle 'gnus-info "Failed"))))
808 (defun mml2015-pgg-sign (cont)
809 (let ((pgg-errors-buffer mml2015-result-buffer)
810 (boundary (mml-compute-boundary cont))
811 (pgg-default-user-id (or (message-options-get 'mml-sender)
812 pgg-default-user-id)))
813 (unless (pgg-sign-region (point-min) (point-max))
814 (pop-to-buffer mml2015-result-buffer)
815 (error "Sign error"))
816 (goto-char (point-min))
817 (insert (format "Content-Type: multipart/signed; boundary=\"%s\";\n"
819 ;;; FIXME: what is the micalg?
820 (insert "\tmicalg=pgp-sha1; protocol=\"application/pgp-signature\"\n")
821 (insert (format "\n--%s\n" boundary))
822 (goto-char (point-max))
823 (insert (format "\n--%s\n" boundary))
824 (insert "Content-Type: application/pgp-signature\n\n")
825 (insert-buffer-substring pgg-output-buffer)
826 (goto-char (point-max))
827 (insert (format "--%s--\n" boundary))
828 (goto-char (point-max))))
830 (defun mml2015-pgg-encrypt (cont &optional sign)
831 (let ((pgg-errors-buffer mml2015-result-buffer)
832 (boundary (mml-compute-boundary cont)))
833 (unless (pgg-encrypt-region (point-min) (point-max)
836 (message-options-get 'message-recipients)
837 (message-options-set 'message-recipients
838 (read-string "Recipients: ")))
841 (pop-to-buffer mml2015-result-buffer)
842 (error "Encrypt error"))
843 (delete-region (point-min) (point-max))
844 (goto-char (point-min))
845 (insert (format "Content-Type: multipart/encrypted; boundary=\"%s\";\n"
847 (insert "\tprotocol=\"application/pgp-encrypted\"\n\n")
848 (insert (format "--%s\n" boundary))
849 (insert "Content-Type: application/pgp-encrypted\n\n")
850 (insert "Version: 1\n\n")
851 (insert (format "--%s\n" boundary))
852 (insert "Content-Type: application/octet-stream\n\n")
853 (insert-buffer-substring pgg-output-buffer)
854 (goto-char (point-max))
855 (insert (format "--%s--\n" boundary))
856 (goto-char (point-max))))
860 (defun mml2015-clean-buffer ()
861 (if (gnus-buffer-live-p mml2015-result-buffer)
862 (with-current-buffer mml2015-result-buffer
865 (setq mml2015-result-buffer
866 (gnus-get-buffer-create "*MML2015 Result*"))
869 (defsubst mml2015-clear-decrypt-function ()
870 (nth 6 (assq mml2015-use mml2015-function-alist)))
872 (defsubst mml2015-clear-verify-function ()
873 (nth 5 (assq mml2015-use mml2015-function-alist)))
876 (defun mml2015-decrypt (handle ctl)
877 (mml2015-clean-buffer)
878 (let ((func (nth 4 (assq mml2015-use mml2015-function-alist))))
880 (funcall func handle ctl)
884 (defun mml2015-decrypt-test (handle ctl)
888 (defun mml2015-verify (handle ctl)
889 (mml2015-clean-buffer)
890 (let ((func (nth 3 (assq mml2015-use mml2015-function-alist))))
892 (funcall func handle ctl)
896 (defun mml2015-verify-test (handle ctl)
900 (defun mml2015-encrypt (cont &optional sign)
901 (mml2015-clean-buffer)
902 (let ((func (nth 2 (assq mml2015-use mml2015-function-alist))))
904 (funcall func cont sign)
905 (error "Cannot find encrypt function"))))
908 (defun mml2015-sign (cont)
909 (mml2015-clean-buffer)
910 (let ((func (nth 1 (assq mml2015-use mml2015-function-alist))))
913 (error "Cannot find sign function"))))
916 (defun mml2015-self-encrypt ()
917 (mml2015-encrypt nil))
921 ;;; arch-tag: b04701d5-0b09-44d8-bed8-de901bf435f2
922 ;;; mml2015.el ends here