2004-09-04 13:13:48 +00:00
|
|
|
;;; pgg-gpg.el --- GnuPG support for PGG.
|
|
|
|
|
2013-01-01 09:11:05 +00:00
|
|
|
;; Copyright (C) 1999-2000, 2002-2013 Free Software Foundation, Inc.
|
2004-09-04 13:13:48 +00:00
|
|
|
|
|
|
|
;; Author: Daiki Ueno <ueno@unixuser.org>
|
2010-08-29 16:17:13 +00:00
|
|
|
;; Symmetric encryption and gpg-agent support added by:
|
2006-04-07 21:06:12 +00:00
|
|
|
;; Sascha Wilde <wilde@sha-bang.de>
|
2004-09-04 13:13:48 +00:00
|
|
|
;; Created: 1999/10/28
|
|
|
|
;; Keywords: PGP, OpenPGP, GnuPG
|
2010-08-29 16:17:13 +00:00
|
|
|
;; Package: pgg
|
2010-12-22 09:23:58 +00:00
|
|
|
;; Obsolete-since: 24.1
|
2004-09-04 13:13:48 +00:00
|
|
|
|
|
|
|
;; This file is part of GNU Emacs.
|
|
|
|
|
2008-05-06 08:06:51 +00:00
|
|
|
;; GNU Emacs is free software: you can redistribute it and/or modify
|
2004-09-04 13:13:48 +00:00
|
|
|
;; it under the terms of the GNU General Public License as published by
|
2008-05-06 08:06:51 +00:00
|
|
|
;; the Free Software Foundation, either version 3 of the License, or
|
|
|
|
;; (at your option) any later version.
|
2004-09-04 13:13:48 +00:00
|
|
|
|
|
|
|
;; GNU Emacs 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
|
2008-05-06 08:06:51 +00:00
|
|
|
;; along with GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
|
2004-09-04 13:13:48 +00:00
|
|
|
|
|
|
|
;;; Code:
|
|
|
|
|
|
|
|
(eval-when-compile
|
2013-05-24 03:47:23 +00:00
|
|
|
(require 'cl))
|
|
|
|
|
|
|
|
(require 'pgg)
|
2004-09-04 13:13:48 +00:00
|
|
|
|
|
|
|
(defgroup pgg-gpg ()
|
2005-07-04 00:56:54 +00:00
|
|
|
"GnuPG interface."
|
2004-09-04 13:13:48 +00:00
|
|
|
:group 'pgg)
|
|
|
|
|
|
|
|
(defcustom pgg-gpg-program "gpg"
|
|
|
|
"The GnuPG executable."
|
|
|
|
:group 'pgg-gpg
|
|
|
|
:type 'string)
|
|
|
|
|
|
|
|
(defcustom pgg-gpg-extra-args nil
|
|
|
|
"Extra arguments for every GnuPG invocation."
|
|
|
|
:group 'pgg-gpg
|
|
|
|
:type '(repeat (string :tag "Argument")))
|
|
|
|
|
|
|
|
(defcustom pgg-gpg-recipient-argument "--recipient"
|
|
|
|
"GnuPG option to specify recipient."
|
|
|
|
:group 'pgg-gpg
|
|
|
|
:type '(choice (const :tag "New `--recipient' option" "--recipient")
|
|
|
|
(const :tag "Old `--remote-user' option" "--remote-user")))
|
|
|
|
|
2006-12-17 23:18:36 +00:00
|
|
|
(defcustom pgg-gpg-use-agent t
|
2006-04-07 21:06:12 +00:00
|
|
|
"Whether to use gnupg agent for key caching."
|
|
|
|
:group 'pgg-gpg
|
|
|
|
:type 'boolean)
|
|
|
|
|
2004-09-04 13:13:48 +00:00
|
|
|
(defvar pgg-gpg-user-id nil
|
|
|
|
"GnuPG ID of your default identity.")
|
|
|
|
|
2006-04-06 18:01:16 +00:00
|
|
|
(defun pgg-gpg-process-region (start end passphrase program args)
|
2007-02-17 20:39:55 +00:00
|
|
|
(let* ((use-agent (and (null passphrase) (pgg-gpg-use-agent-p)))
|
2006-04-07 21:06:12 +00:00
|
|
|
(output-file-name (pgg-make-temp-file "pgg-output"))
|
2004-09-04 13:13:48 +00:00
|
|
|
(args
|
2006-04-06 18:01:16 +00:00
|
|
|
`("--status-fd" "2"
|
2006-04-07 21:06:12 +00:00
|
|
|
,@(if use-agent '("--use-agent")
|
|
|
|
(if passphrase '("--passphrase-fd" "0")))
|
2006-04-06 18:01:16 +00:00
|
|
|
"--yes" ; overwrite
|
|
|
|
"--output" ,output-file-name
|
|
|
|
,@pgg-gpg-extra-args ,@args))
|
|
|
|
(output-buffer pgg-output-buffer)
|
|
|
|
(errors-buffer pgg-errors-buffer)
|
2006-03-27 09:36:18 +00:00
|
|
|
(orig-mode (default-file-modes))
|
2006-04-06 18:01:16 +00:00
|
|
|
(process-connection-type nil)
|
2006-09-05 08:17:35 +00:00
|
|
|
(inhibit-redisplay t)
|
|
|
|
process status exit-status
|
|
|
|
passphrase-with-newline
|
|
|
|
encoded-passphrase-with-new-line)
|
2006-04-06 18:01:16 +00:00
|
|
|
(with-current-buffer (get-buffer-create errors-buffer)
|
|
|
|
(buffer-disable-undo)
|
|
|
|
(erase-buffer))
|
2004-09-04 13:13:48 +00:00
|
|
|
(unwind-protect
|
|
|
|
(progn
|
|
|
|
(set-default-file-modes 448)
|
2006-09-04 06:27:27 +00:00
|
|
|
(let ((coding-system-for-write 'binary))
|
|
|
|
(setq process
|
|
|
|
(apply #'start-process "*GnuPG*" errors-buffer
|
|
|
|
program args)))
|
|
|
|
(set-process-sentinel process #'ignore)
|
|
|
|
(when passphrase
|
2006-09-05 08:17:35 +00:00
|
|
|
(setq passphrase-with-newline (concat passphrase "\n"))
|
2006-09-07 05:58:54 +00:00
|
|
|
(if pgg-passphrase-coding-system
|
2006-09-05 08:17:35 +00:00
|
|
|
(progn
|
|
|
|
(setq encoded-passphrase-with-new-line
|
Merge from gnus--rel--5.10
Patches applied:
* gnus--rel--5.10 (patch 179-183)
- Update from CVS
2006-12-25 Daiki Ueno <ueno@unixuser.org>
* lisp/pgg-def.el (pgg-passphrase-coding-system): Default to nil instead of
locale-coding-system.
* lisp/pgg-gpg.el (pgg-gpg-process-region): Encode passphrase with eol-type
LF.
2006-12-29 Jouni K. Sepp,Ad(Bnen <jks@iki.fi>
* lisp/gnus/nnimap.el (nnimap-expunge-search-string): Mention
nnimap-search-uids-not-since-is-evil in docstring.
2006-12-28 Reiner Steib <Reiner.Steib@gmx.de>
* lisp/gnus/spam.el: Revert to make-obsolete-variable because
define-obsolete-variable-alias is not supported in Emacs 21.
2006-12-28 Daiki Ueno <ueno@unixuser.org>
* lisp/gnus/gnus-sum.el (gnus-summary-next-article): Make sure we are in the
summary buffer.
2006-12-27 Reiner Steib <Reiner.Steib@gmx.de>
* lisp/gnus/spam.el (spam-ifile-path, spam-ifile-database-path)
(spam-bogofilter-path): Use define-obsolete-variable-alias instead of
make-obsolete-variable.
2006-12-26 Reiner Steib <Reiner.Steib@gmx.de>
* lisp/gnus/message.el (message-make-fqdn): Fix comment.
(message-bogus-system-names): Add ".local".
* lisp/gnus/spam.el (spam-ifile-path, spam-ifile-program)
(spam-ifile-database-path, spam-ifile-database)
(spam-bogofilter-path, spam-bogofilter-program): Rename variables.
Don't use "path" inappropriately.
(spam-spamoracle-database, spam-get-ifile-database-parameter): Fix doc
strings.
(spam-check-ifile, spam-ifile-register-with-ifile)
(spam-check-bogofilter, spam-bogofilter-register-with-bogofilter): Use
new variable names.
* lisp/gnus/gnus-art.el (gnus-treat-display-x-face, gnus-treat-display-face)
(gnus-treat-display-smileys): Simplify using
gnus-image-type-available-p.
* lisp/gnus/gnus-ems.el (gnus-image-type-available-p): Use display-images-p if
available.
2006-12-22 Katsumi Yamaoka <yamaoka@jpl.org>
* lisp/gnus/nnrss.el (nnrss-fetch): Replace buffer's contents with the decoded
one after turning on the buffer's multibyteness instead of decoding
them directly in the unibyte buffer that causes unexpected conversion
in Emacs 23 (unicode).
2006-12-29 Reiner Steib <Reiner.Steib@gmx.de>
* man/gnus.texi (Customizing Articles): Add index entries for all
gnus-treat-* variables.
2006-12-29 Jouni K. Sepp,Ad(Bnen <jks@iki.fi>
* man/gnus.texi (IMAP): Fix incorrect explanation of
nnimap-search-uids-not-since-is-evil in documentation for
nnimap-expunge-search-string.
2006-12-27 Reiner Steib <Reiner.Steib@gmx.de>
* man/gnus.texi (ifile spam filtering): Rename spam-ifile-database-path to
spam-ifile-database.
2006-12-26 Reiner Steib <Reiner.Steib@gmx.de>
* man/gnus.texi (Spam Package Configuration Examples): Don't encourage to
rebind C-s.
2006-12-26 Jouni K. Sepp,Ad(Bnen <jks@iki.fi>
* man/gnus.texi (Group Parameters, Group Maintenance, Topic Commands)
(Mail Group Commands, Expiring Mail, IMAP): Add index entries for
"expiring mail".
(IMAP): Document nnimap-search-uids-not-since-is-evil and
nnimap-nov-is-evil.
Revision: emacs@sv.gnu.org/emacs--devo--0--patch-576
2006-12-30 15:34:42 +00:00
|
|
|
(encode-coding-string
|
|
|
|
passphrase-with-newline
|
|
|
|
(coding-system-change-eol-conversion
|
|
|
|
pgg-passphrase-coding-system 'unix)))
|
2006-09-05 08:17:35 +00:00
|
|
|
(pgg-clear-string passphrase-with-newline))
|
|
|
|
(setq encoded-passphrase-with-new-line passphrase-with-newline
|
|
|
|
passphrase-with-newline nil))
|
|
|
|
(process-send-string process encoded-passphrase-with-new-line))
|
2006-09-04 06:27:27 +00:00
|
|
|
(process-send-region process start end)
|
|
|
|
(process-send-eof process)
|
|
|
|
(while (eq 'run (process-status process))
|
|
|
|
(accept-process-output process 5))
|
2008-08-02 19:02:42 +00:00
|
|
|
;; Accept any remaining pending output coming after the
|
|
|
|
;; status change.
|
|
|
|
(accept-process-output process 5)
|
2006-09-04 06:27:27 +00:00
|
|
|
(setq status (process-status process)
|
|
|
|
exit-status (process-exit-status process))
|
|
|
|
(delete-process process)
|
2006-04-06 18:01:16 +00:00
|
|
|
(with-current-buffer (get-buffer-create output-buffer)
|
|
|
|
(buffer-disable-undo)
|
|
|
|
(erase-buffer)
|
|
|
|
(if (file-exists-p output-file-name)
|
2006-04-04 10:34:03 +00:00
|
|
|
(let ((coding-system-for-read (if pgg-text-mode
|
|
|
|
'raw-text
|
|
|
|
'binary)))
|
2006-04-06 18:01:16 +00:00
|
|
|
(insert-file-contents output-file-name)))
|
|
|
|
(set-buffer errors-buffer)
|
2006-09-04 06:27:27 +00:00
|
|
|
(if (memq status '(stop signal))
|
|
|
|
(error "%s exited abnormally: '%s'" program exit-status))
|
|
|
|
(if (= 127 exit-status)
|
|
|
|
(error "%s could not be found" program))))
|
2006-09-05 08:17:35 +00:00
|
|
|
(if passphrase-with-newline
|
|
|
|
(pgg-clear-string passphrase-with-newline))
|
|
|
|
(if encoded-passphrase-with-new-line
|
|
|
|
(pgg-clear-string encoded-passphrase-with-new-line))
|
2006-09-04 06:27:27 +00:00
|
|
|
(if (and process (eq 'run (process-status process)))
|
|
|
|
(interrupt-process process))
|
2006-04-06 18:01:16 +00:00
|
|
|
(if (file-exists-p output-file-name)
|
|
|
|
(delete-file output-file-name))
|
|
|
|
(set-default-file-modes orig-mode))))
|
|
|
|
|
|
|
|
(defun pgg-gpg-possibly-cache-passphrase (passphrase &optional key notruncate)
|
2006-04-07 21:06:12 +00:00
|
|
|
(if (and passphrase
|
|
|
|
pgg-cache-passphrase
|
2006-04-06 18:01:16 +00:00
|
|
|
(progn
|
|
|
|
(goto-char (point-min))
|
|
|
|
(re-search-forward "^\\[GNUPG:] \\(GOOD_PASSPHRASE\\>\\)\\|\\(SIG_CREATED\\)" nil t)))
|
|
|
|
(pgg-add-passphrase-to-cache
|
|
|
|
(or key
|
|
|
|
(progn
|
|
|
|
(goto-char (point-min))
|
|
|
|
(if (re-search-forward
|
|
|
|
"^\\[GNUPG:] NEED_PASSPHRASE\\(_PIN\\)? \\w+ ?\\w*" nil t)
|
|
|
|
(substring (match-string 0) -8))))
|
|
|
|
passphrase
|
|
|
|
notruncate)))
|
|
|
|
|
|
|
|
(defvar pgg-gpg-all-secret-keys 'unknown)
|
|
|
|
|
|
|
|
(defun pgg-gpg-lookup-all-secret-keys ()
|
|
|
|
"Return all secret keys present in secret key ring."
|
|
|
|
(when (eq pgg-gpg-all-secret-keys 'unknown)
|
|
|
|
(setq pgg-gpg-all-secret-keys '())
|
|
|
|
(let ((args (list "--with-colons" "--no-greeting" "--batch"
|
|
|
|
"--list-secret-keys")))
|
|
|
|
(with-temp-buffer
|
|
|
|
(apply #'call-process pgg-gpg-program nil t nil args)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(while (re-search-forward
|
|
|
|
"^\\(sec\\|pub\\):[^:]*:[^:]*:[^:]*:\\([^:]*\\)" nil t)
|
|
|
|
(push (substring (match-string 2) 8)
|
|
|
|
pgg-gpg-all-secret-keys)))))
|
|
|
|
pgg-gpg-all-secret-keys)
|
2004-09-04 13:13:48 +00:00
|
|
|
|
|
|
|
(defun pgg-gpg-lookup-key (string &optional type)
|
|
|
|
"Search keys associated with STRING."
|
|
|
|
(let ((args (list "--with-colons" "--no-greeting" "--batch"
|
|
|
|
(if type "--list-secret-keys" "--list-keys")
|
|
|
|
string)))
|
|
|
|
(with-temp-buffer
|
|
|
|
(apply #'call-process pgg-gpg-program nil t nil args)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(if (re-search-forward "^\\(sec\\|pub\\):[^:]*:[^:]*:[^:]*:\\([^:]*\\)"
|
|
|
|
nil t)
|
|
|
|
(substring (match-string 2) 8)))))
|
|
|
|
|
2006-04-06 18:01:16 +00:00
|
|
|
(defun pgg-gpg-lookup-key-owner (string &optional all)
|
|
|
|
"Search keys associated with STRING and return owner of identified key.
|
|
|
|
|
|
|
|
The value may be just the bare key id, or it may be a combination of the
|
|
|
|
user name associated with the key and the key id, with the key id enclosed
|
|
|
|
in \"<...>\" angle brackets.
|
|
|
|
|
|
|
|
Optional ALL non-nil means search all keys, including secret keys."
|
|
|
|
(let ((args (list "--with-colons" "--no-greeting" "--batch"
|
|
|
|
(if all "--list-secret-keys" "--list-keys")
|
|
|
|
string))
|
2009-06-18 17:47:48 +00:00
|
|
|
(key-regexp (concat "^\\(sec\\|pub\\|uid\\)"
|
2006-04-06 18:01:16 +00:00
|
|
|
":[^:]*:[^:]*:[^:]*:\\([^:]*\\):[^:]*"
|
2009-06-18 17:47:48 +00:00
|
|
|
":[^:]*:[^:]*:[^:]*:\\([^:]+\\):")))
|
2006-04-06 18:01:16 +00:00
|
|
|
(with-temp-buffer
|
|
|
|
(apply #'call-process pgg-gpg-program nil t nil args)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(if (re-search-forward key-regexp
|
|
|
|
nil t)
|
|
|
|
(match-string 3)))))
|
|
|
|
|
|
|
|
(defun pgg-gpg-key-id-from-key-owner (key-owner)
|
|
|
|
(cond ((not key-owner) nil)
|
|
|
|
;; Extract bare key id from outermost paired angle brackets, if any:
|
|
|
|
((string-match "[^<]*<\\(.+\\)>[^>]*" key-owner)
|
|
|
|
(substring key-owner (match-beginning 1)(match-end 1)))
|
|
|
|
(key-owner)))
|
|
|
|
|
2005-10-29 11:31:08 +00:00
|
|
|
(defun pgg-gpg-encrypt-region (start end recipients &optional sign passphrase)
|
2004-09-04 13:13:48 +00:00
|
|
|
"Encrypt the current region between START and END.
|
2005-10-29 11:31:08 +00:00
|
|
|
|
2006-04-06 18:01:16 +00:00
|
|
|
If optional argument SIGN is non-nil, do a combined sign and encrypt.
|
|
|
|
|
|
|
|
If optional PASSPHRASE is not specified, it will be obtained from the
|
|
|
|
passphrase cache or user."
|
2004-09-04 13:13:48 +00:00
|
|
|
(let* ((pgg-gpg-user-id (or pgg-gpg-user-id pgg-default-user-id))
|
2006-04-06 18:01:16 +00:00
|
|
|
(passphrase (or passphrase
|
2006-04-07 21:06:12 +00:00
|
|
|
(when (and sign (not (pgg-gpg-use-agent-p)))
|
2006-04-06 18:01:16 +00:00
|
|
|
(pgg-read-passphrase
|
|
|
|
(format "GnuPG passphrase for %s: "
|
|
|
|
pgg-gpg-user-id)
|
|
|
|
pgg-gpg-user-id))))
|
2004-09-04 13:13:48 +00:00
|
|
|
(args
|
|
|
|
(append
|
2006-04-06 18:01:16 +00:00
|
|
|
(list "--batch" "--armor" "--always-trust" "--encrypt")
|
|
|
|
(if pgg-text-mode (list "--textmode"))
|
2004-09-04 13:13:48 +00:00
|
|
|
(if sign (list "--sign" "--local-user" pgg-gpg-user-id))
|
2007-02-24 00:07:19 +00:00
|
|
|
(if (or recipients pgg-encrypt-for-me)
|
2004-09-04 13:13:48 +00:00
|
|
|
(apply #'nconc
|
|
|
|
(mapcar (lambda (rcpt)
|
|
|
|
(list pgg-gpg-recipient-argument rcpt))
|
|
|
|
(append recipients
|
|
|
|
(if pgg-encrypt-for-me
|
2006-04-06 18:01:16 +00:00
|
|
|
(list pgg-gpg-user-id)))))))))
|
|
|
|
(pgg-gpg-process-region start end passphrase pgg-gpg-program args)
|
|
|
|
(when sign
|
|
|
|
(with-current-buffer pgg-errors-buffer
|
|
|
|
;; Possibly cache passphrase under, e.g. "jas", for future sign.
|
|
|
|
(pgg-gpg-possibly-cache-passphrase passphrase pgg-gpg-user-id)
|
|
|
|
;; Possibly cache passphrase under, e.g. B565716F, for future decrypt.
|
|
|
|
(pgg-gpg-possibly-cache-passphrase passphrase)))
|
|
|
|
(pgg-process-when-success)))
|
2004-09-04 13:13:48 +00:00
|
|
|
|
2005-10-29 11:31:08 +00:00
|
|
|
(defun pgg-gpg-encrypt-symmetric-region (start end &optional passphrase)
|
2006-04-06 18:01:16 +00:00
|
|
|
"Encrypt the current region between START and END with symmetric cipher.
|
|
|
|
|
|
|
|
If optional PASSPHRASE is not specified, it will be obtained from the
|
|
|
|
passphrase cache or user."
|
|
|
|
(let* ((passphrase (or passphrase
|
2006-04-07 21:06:12 +00:00
|
|
|
(when (not (pgg-gpg-use-agent-p))
|
|
|
|
(pgg-read-passphrase
|
|
|
|
"GnuPG passphrase for symmetric encryption: "))))
|
2006-04-06 18:01:16 +00:00
|
|
|
(args
|
|
|
|
(append (list "--batch" "--armor" "--symmetric" )
|
|
|
|
(if pgg-text-mode (list "--textmode")))))
|
|
|
|
(pgg-gpg-process-region start end passphrase pgg-gpg-program args)
|
|
|
|
(pgg-process-when-success)))
|
2005-10-29 11:31:08 +00:00
|
|
|
|
|
|
|
(defun pgg-gpg-decrypt-region (start end &optional passphrase)
|
2006-04-06 18:01:16 +00:00
|
|
|
"Decrypt the current region between START and END.
|
|
|
|
|
|
|
|
If optional PASSPHRASE is not specified, it will be obtained from the
|
|
|
|
passphrase cache or user."
|
|
|
|
(let* ((current-buffer (current-buffer))
|
|
|
|
(message-keys (with-temp-buffer
|
|
|
|
(insert-buffer-substring current-buffer)
|
|
|
|
(pgg-decode-armor-region (point-min) (point-max))))
|
|
|
|
(secret-keys (pgg-gpg-lookup-all-secret-keys))
|
|
|
|
;; XXX the user is stuck if they need to use the passphrase for
|
|
|
|
;; any but the first secret key for which the message is
|
|
|
|
;; encrypted. ideally, we would incrementally give them a
|
|
|
|
;; chance with subsequent keys each time they fail with one.
|
|
|
|
(key (pgg-gpg-select-matching-key message-keys secret-keys))
|
|
|
|
(key-owner (and key (pgg-gpg-lookup-key-owner key t)))
|
|
|
|
(key-id (pgg-gpg-key-id-from-key-owner key-owner))
|
|
|
|
(pgg-gpg-user-id (or key-id key
|
|
|
|
pgg-gpg-user-id pgg-default-user-id))
|
|
|
|
(passphrase (or passphrase
|
2006-04-07 21:06:12 +00:00
|
|
|
(when (not (pgg-gpg-use-agent-p))
|
|
|
|
(pgg-read-passphrase
|
|
|
|
(format (if (pgg-gpg-symmetric-key-p message-keys)
|
|
|
|
"Passphrase for symmetric decryption: "
|
|
|
|
"GnuPG passphrase for %s: ")
|
|
|
|
(or key-owner "??"))
|
|
|
|
pgg-gpg-user-id))))
|
2006-04-06 18:01:16 +00:00
|
|
|
(args '("--batch" "--decrypt")))
|
|
|
|
(pgg-gpg-process-region start end passphrase pgg-gpg-program args)
|
|
|
|
(with-current-buffer pgg-errors-buffer
|
|
|
|
(pgg-gpg-possibly-cache-passphrase passphrase pgg-gpg-user-id)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(re-search-forward "^\\[GNUPG:] DECRYPTION_OKAY\\>" nil t))))
|
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun pgg-gpg-symmetric-key-p (message-keys)
|
|
|
|
"True if decoded armor MESSAGE-KEYS has symmetric encryption indicator."
|
|
|
|
(let (result)
|
|
|
|
(dolist (key message-keys result)
|
|
|
|
(when (and (eq (car key) 3)
|
|
|
|
(member '(symmetric-key-algorithm) key))
|
|
|
|
(setq result key)))))
|
|
|
|
|
|
|
|
(defun pgg-gpg-select-matching-key (message-keys secret-keys)
|
|
|
|
"Choose a key from MESSAGE-KEYS that matches one of the keys in SECRET-KEYS."
|
|
|
|
(loop for message-key in message-keys
|
|
|
|
for message-key-id = (and (equal (car message-key) 1)
|
|
|
|
(cdr (assq 'key-identifier
|
|
|
|
(cdr message-key))))
|
|
|
|
for key = (and message-key-id (pgg-lookup-key message-key-id 'encrypt))
|
|
|
|
when (and key (member key secret-keys)) return key))
|
2004-09-04 13:13:48 +00:00
|
|
|
|
2005-10-29 11:31:08 +00:00
|
|
|
(defun pgg-gpg-sign-region (start end &optional cleartext passphrase)
|
2004-09-04 13:13:48 +00:00
|
|
|
"Make detached signature from text between START and END."
|
|
|
|
(let* ((pgg-gpg-user-id (or pgg-gpg-user-id pgg-default-user-id))
|
2006-04-06 18:01:16 +00:00
|
|
|
(passphrase (or passphrase
|
2006-04-07 21:06:12 +00:00
|
|
|
(when (not (pgg-gpg-use-agent-p))
|
|
|
|
(pgg-read-passphrase
|
|
|
|
(format "GnuPG passphrase for %s: "
|
|
|
|
pgg-gpg-user-id)
|
|
|
|
pgg-gpg-user-id))))
|
2004-09-04 13:13:48 +00:00
|
|
|
(args
|
2006-02-10 05:08:29 +00:00
|
|
|
(append (list (if cleartext "--clearsign" "--detach-sign")
|
2006-04-06 18:01:16 +00:00
|
|
|
"--armor" "--batch" "--verbose"
|
2006-02-10 05:08:29 +00:00
|
|
|
"--local-user" pgg-gpg-user-id)
|
2006-04-06 18:01:16 +00:00
|
|
|
(if pgg-text-mode (list "--textmode"))))
|
|
|
|
(inhibit-read-only t)
|
|
|
|
buffer-read-only)
|
|
|
|
(pgg-gpg-process-region start end passphrase pgg-gpg-program args)
|
|
|
|
(with-current-buffer pgg-errors-buffer
|
|
|
|
;; Possibly cache passphrase under, e.g. "jas", for future sign.
|
|
|
|
(pgg-gpg-possibly-cache-passphrase passphrase pgg-gpg-user-id)
|
|
|
|
;; Possibly cache passphrase under, e.g. B565716F, for future decrypt.
|
|
|
|
(pgg-gpg-possibly-cache-passphrase passphrase))
|
|
|
|
(pgg-process-when-success)))
|
2004-09-04 13:13:48 +00:00
|
|
|
|
|
|
|
(defun pgg-gpg-verify-region (start end &optional signature)
|
|
|
|
"Verify region between START and END as the detached signature SIGNATURE."
|
2006-04-06 18:01:16 +00:00
|
|
|
(let ((args '("--batch" "--verify")))
|
2004-09-04 13:13:48 +00:00
|
|
|
(when (stringp signature)
|
|
|
|
(setq args (append args (list signature))))
|
2006-04-06 18:01:16 +00:00
|
|
|
(setq args (append args '("-")))
|
|
|
|
(pgg-gpg-process-region start end nil pgg-gpg-program args)
|
|
|
|
(with-current-buffer pgg-errors-buffer
|
|
|
|
(goto-char (point-min))
|
|
|
|
(while (re-search-forward "^gpg: \\(.*\\)\n" nil t)
|
|
|
|
(with-current-buffer pgg-output-buffer
|
|
|
|
(insert-buffer-substring pgg-errors-buffer
|
|
|
|
(match-beginning 1) (match-end 0)))
|
|
|
|
(delete-region (match-beginning 0) (match-end 0)))
|
|
|
|
(goto-char (point-min))
|
|
|
|
(re-search-forward "^\\[GNUPG:] GOODSIG\\>" nil t))))
|
2004-09-04 13:13:48 +00:00
|
|
|
|
|
|
|
(defun pgg-gpg-insert-key ()
|
|
|
|
"Insert public key at point."
|
|
|
|
(let* ((pgg-gpg-user-id (or pgg-gpg-user-id pgg-default-user-id))
|
2006-04-06 18:01:16 +00:00
|
|
|
(args (list "--batch" "--export" "--armor"
|
|
|
|
pgg-gpg-user-id)))
|
|
|
|
(pgg-gpg-process-region (point)(point) nil pgg-gpg-program args)
|
2004-09-04 13:13:48 +00:00
|
|
|
(insert-buffer-substring pgg-output-buffer)))
|
|
|
|
|
|
|
|
(defun pgg-gpg-snarf-keys-region (start end)
|
|
|
|
"Add all public keys in region between START and END to the keyring."
|
2006-04-06 18:01:16 +00:00
|
|
|
(let ((args '("--import" "--batch" "-")) status)
|
|
|
|
(pgg-gpg-process-region start end nil pgg-gpg-program args)
|
|
|
|
(set-buffer pgg-errors-buffer)
|
|
|
|
(goto-char (point-min))
|
|
|
|
(when (re-search-forward "^\\[GNUPG:] IMPORT_RES\\>" nil t)
|
|
|
|
(setq status (buffer-substring (match-end 0)
|
|
|
|
(progn (end-of-line)(point)))
|
|
|
|
status (vconcat (mapcar #'string-to-number (split-string status))))
|
|
|
|
(erase-buffer)
|
|
|
|
(insert (format "Imported %d key(s).
|
|
|
|
\tArmor contains %d key(s) [%d bad, %d old].\n"
|
|
|
|
(+ (aref status 2)
|
|
|
|
(aref status 10))
|
|
|
|
(aref status 0)
|
|
|
|
(aref status 1)
|
|
|
|
(+ (aref status 4)
|
|
|
|
(aref status 11)))
|
|
|
|
(if (zerop (aref status 9))
|
|
|
|
""
|
|
|
|
"\tSecret keys are imported.\n")))
|
|
|
|
(append-to-buffer pgg-output-buffer (point-min)(point-max))
|
|
|
|
(pgg-process-when-success)))
|
2006-03-22 16:09:16 +00:00
|
|
|
|
2006-04-07 21:06:12 +00:00
|
|
|
(defun pgg-gpg-update-agent ()
|
2011-11-15 07:55:13 +00:00
|
|
|
"Try to connect to gpg-agent and send UPDATESTARTUPTTY."
|
2006-04-07 21:06:12 +00:00
|
|
|
(if (fboundp 'make-network-process)
|
|
|
|
(let* ((agent-info (getenv "GPG_AGENT_INFO"))
|
|
|
|
(socket (and agent-info
|
|
|
|
(string-match "^\\([^:]*\\)" agent-info)
|
|
|
|
(match-string 1 agent-info)))
|
|
|
|
(conn (and socket
|
|
|
|
(make-network-process :name "gpg-agent-process"
|
|
|
|
:host 'local :family 'local
|
|
|
|
:service socket))))
|
|
|
|
(when (and conn (eq (process-status conn) 'open))
|
|
|
|
(process-send-string conn "UPDATESTARTUPTTY\n")
|
|
|
|
(delete-process conn)
|
|
|
|
t))
|
|
|
|
;; We can't check, so assume gpg-agent is up.
|
|
|
|
t))
|
|
|
|
|
|
|
|
(defun pgg-gpg-use-agent-p ()
|
|
|
|
"Return t if `pgg-gpg-use-agent' is t and gpg-agent is available."
|
|
|
|
(and pgg-gpg-use-agent (pgg-gpg-update-agent)))
|
|
|
|
|
2004-09-04 13:13:48 +00:00
|
|
|
(provide 'pgg-gpg)
|
|
|
|
|
|
|
|
;;; pgg-gpg.el ends here
|