]> code.delx.au - gnu-emacs/blobdiff - lisp/org/org-crypt.el
Spelling fixes.
[gnu-emacs] / lisp / org / org-crypt.el
index 0a6001bf62a2a829ab8c9f6016d40167c769792c..b260391a0dab3c66d4857b45a4e31263f3a8ddfe 100644 (file)
@@ -4,7 +4,7 @@
 
 ;; Emacs Lisp Archive Entry
 ;; Filename: org-crypt.el
-;; Version: 7.4
+;; Version: 7.7
 ;; Keywords: org-mode
 ;; Author: John Wiegley <johnw@gnu.org>
 ;; Maintainer: Peter Jones <pjones@pmade.com>
@@ -56,9 +56,6 @@
 ;; 4. To automatically encrypt all necessary entries when saving a
 ;;    file, call `org-crypt-use-before-save-magic' after loading
 ;;    org-crypt.el.
-;;
-;; TODO:
-;;   - Allow symmetric encryption as well
 
 ;;; Thanks:
 
 
 (defgroup org-crypt nil
   "Org Crypt"
-  :tag "Org Crypt" :group 'org)
+  :tag "Org Crypt" 
+  :group 'org)
 
 (defcustom org-crypt-tag-matcher "crypt"
   "The tag matcher used to find headings whose contents should be encrypted.
 
 See the \"Match syntax\" section of the org manual for more details."
-  :type 'string :group 'org-crypt)
+  :type 'string 
+  :group 'org-crypt)
 
-(defcustom org-crypt-key nil
+(defcustom org-crypt-key ""
   "The default key to use when encrypting the contents of a heading.
 
 This setting can also be overridden in the CRYPTKEY property."
-  :type 'string :group 'org-crypt)
+  :type 'string 
+  :group 'org-crypt)
+
+(defcustom org-crypt-disable-auto-save 'ask
+  "What org-decrypt should do if `auto-save-mode' is enabled.
+
+t        : Disable auto-save-mode for the current buffer
+           prior to decrypting an entry.
+
+nil      : Leave auto-save-mode enabled.
+           This may cause data to be written to disk unencrypted!
+
+'ask     : Ask user whether or not to disable auto-save-mode
+           for the current buffer.
+
+'encrypt : Leave auto-save-mode enabled for the current buffer,
+           but automatically re-encrypt all decrypted entries
+           *before* auto-saving.
+           NOTE: This only works for entries which have a tag
+           that matches `org-crypt-tag-matcher'."
+  :group 'org-crypt
+  :type '(choice (const :tag "Always"  t)
+                 (const :tag "Never"   nil)
+                 (const :tag "Ask"     ask)
+                 (const :tag "Encrypt" encrypt)))
 
 (defun org-crypt-key-for-heading ()
   "Return the encryption key for the current heading."
@@ -103,6 +126,15 @@ This setting can also be overridden in the CRYPTKEY property."
         (and (boundp 'epa-file-encrypt-to) epa-file-encrypt-to)
         (message "No crypt key set, using symmetric encryption."))))
 
+(defun org-encrypt-string (str crypt-key)
+  "Return STR encrypted with CRYPT-KEY."
+  ;; Text and key have to be identical, otherwise we re-crypt.
+  (if (and (string= crypt-key (get-text-property 0 'org-crypt-key str))
+          (string= (sha1 str) (get-text-property 0 'org-crypt-checksum str)))
+      (get-text-property 0 'org-crypt-text str)
+    (let ((epg-context (epg-make-context nil t t)))
+      (epg-encrypt-string epg-context str (epg-list-keys epg-context crypt-key)))))
+
 (defun org-encrypt-entry ()
   "Encrypt the content of the current headline."
   (interactive)
@@ -112,7 +144,7 @@ This setting can also be overridden in the CRYPTKEY property."
     (let ((start-heading (point)))
       (forward-line)
       (when (not (looking-at "-----BEGIN PGP MESSAGE-----"))
-        (let ((folded (org-invisible-p))
+        (let ((folded (outline-invisible-p))
               (epg-context (epg-make-context nil t t))
               (crypt-key (org-crypt-key-for-heading))
               (beg (point))
@@ -122,10 +154,7 @@ This setting can also be overridden in the CRYPTKEY property."
           (org-back-over-empty-lines)
           (setq end (point)
                 encrypted-text
-                (epg-encrypt-string
-                 epg-context
-                 (buffer-substring-no-properties beg end)
-                 (epg-list-keys epg-context crypt-key)))
+               (org-encrypt-string (buffer-substring beg end) crypt-key))
           (delete-region beg end)
           (insert encrypted-text)
           (when folded
@@ -136,27 +165,68 @@ This setting can also be overridden in the CRYPTKEY property."
 (defun org-decrypt-entry ()
   "Decrypt the content of the current headline."
   (interactive)
+
+  ; auto-save-mode may cause leakage, so check whether it's enabled.
+  (when buffer-auto-save-file-name
+    (cond
+     ((or
+       (eq org-crypt-disable-auto-save t)
+       (and
+        (eq org-crypt-disable-auto-save 'ask)
+        (y-or-n-p "org-decrypt: auto-save-mode may cause leakage. Disable it for current buffer? ")))
+      (message (concat "org-decrypt: Disabling auto-save-mode for " (or (buffer-file-name) (current-buffer))))
+      ; The argument to auto-save-mode has to be "-1", since
+      ; giving a "nil" argument toggles instead of disabling.
+      (auto-save-mode -1))
+     ((eq org-crypt-disable-auto-save nil)
+      (message "org-decrypt: Decrypting entry with auto-save-mode enabled. This may cause leakage."))
+     ((eq org-crypt-disable-auto-save 'encrypt)
+      (message "org-decrypt: Enabling re-encryption on auto-save.")
+      (add-hook 'auto-save-hook
+                (lambda ()
+                  (message "org-crypt: Re-encrypting all decrypted entries due to auto-save.")
+                  (org-encrypt-entries))
+                nil t))
+     (t nil)))
+
   (require 'epg)
   (unless (org-before-first-heading-p)
     (save-excursion
       (org-back-to-heading t)
-      (forward-line)
-      (when (looking-at "-----BEGIN PGP MESSAGE-----")
-       (let* ((beg (point))
-              (end (save-excursion
-                     (search-forward "-----END PGP MESSAGE-----")
-                     (forward-line)
-                     (point)))
-              (epg-context (epg-make-context nil t t))
-              (decrypted-text
-               (decode-coding-string
-                (epg-decrypt-string
-                 epg-context
-                 (buffer-substring-no-properties beg end))
-                'utf-8)))
-         (delete-region beg end)
-         (insert decrypted-text)
-         nil)))))
+      (let ((heading-point (point))
+           (heading-was-invisible-p
+            (save-excursion
+              (outline-end-of-heading)
+              (outline-invisible-p))))
+       (forward-line)
+       (when (looking-at "-----BEGIN PGP MESSAGE-----")
+         (let* ((end (save-excursion
+                       (search-forward "-----END PGP MESSAGE-----")
+                       (forward-line)
+                       (point)))
+                (epg-context (epg-make-context nil t t))
+                (encrypted-text (buffer-substring-no-properties (point) end))
+                (decrypted-text
+                 (decode-coding-string
+                  (epg-decrypt-string
+                   epg-context
+                   encrypted-text)
+                  'utf-8)))
+           ;; Delete region starting just before point, because the
+           ;; outline property starts at the \n of the heading.
+           (delete-region (1- (point)) end)
+           ;; Store a checksum of the decrypted and the encrypted
+           ;; text value. This allow to reuse the same encrypted text
+           ;; if the text does not change, and therefore avoid a
+           ;; re-encryption process.
+           (insert "\n" (propertize decrypted-text
+                                    'org-crypt-checksum (sha1 decrypted-text)
+                                    'org-crypt-key (org-crypt-key-for-heading)
+                                    'org-crypt-text encrypted-text))
+           (when heading-was-invisible-p
+             (goto-char heading-point)
+             (org-flag-subtree t))
+           nil))))))
 
 (defun org-encrypt-entries ()
   "Encrypt all top-level entries in the current buffer."
@@ -183,4 +253,5 @@ This setting can also be overridden in the CRYPTKEY property."
 (provide 'org-crypt)
 
 
+
 ;;; org-crypt.el ends here