]> git.eshelyaron.com Git - emacs.git/commitdiff
Load themes using `load', querying if theme file not known safe.
authorChong Yidong <cyd@stupidchicken.com>
Sat, 8 Jan 2011 19:19:55 +0000 (14:19 -0500)
committerChong Yidong <cyd@stupidchicken.com>
Sat, 8 Jan 2011 19:19:55 +0000 (14:19 -0500)
* custom.el (custom-safe-theme-files): New defcustom.
(custom-theme-load-confirm): New function.
(load-theme): Load theme using `load', confirming with
custom-theme-load-confirm if necessary.

etc/NEWS
lisp/ChangeLog
lisp/custom.el

index eaacfac4d49dce5fe9f3c0bed9589a05088e39a8..0fd74252601e395f915641f3eecc5d773db9f297 100644 (file)
--- a/etc/NEWS
+++ b/etc/NEWS
@@ -219,6 +219,10 @@ Emacs no longer looks for custom themes in `load-path'.  The default
 is to search in `custom-theme-directory', followed by a built-in theme
 directory named "themes/" in `data-directory'.
 
+*** New option `custom-safe-theme-files' lists known-safe theme files.
+If a theme is not in this list, Emacs queries before loading it.
+The default value treats all themes included in Emacs as safe.
+
 ** The user option `remote-file-name-inhibit-cache' controls whether
 the remote file-name cache is used for read access.
 
index 0f84b977d56cb5a3cbe8d04815eed5c5aef17dbe..86f9e8fa0b9f5a8046e6ab95c08853d67463d17e 100644 (file)
@@ -1,5 +1,10 @@
 2011-01-08  Chong Yidong  <cyd@stupidchicken.com>
 
+       * custom.el (custom-safe-theme-files): New defcustom.
+       (custom-theme-load-confirm): New function.
+       (load-theme): Load theme using `load', confirming with
+       custom-theme-load-confirm if necessary.
+
        * subr.el (read-char-choice): New function, factored out from
        dired-query and hack-local-variables-confirm.
 
index bcb78e46a3c8942cecd5d1bdc3f641babf7f34ab..f984d132dd13feb4859dcccca478e1d67d4637e7 100644 (file)
@@ -1105,15 +1105,26 @@ property `theme-feature' (which is usually a symbol created by
     (let ((custom-enabling-themes t))
       (enable-theme 'user))))
 
+(defcustom custom-safe-theme-files '(default)
+  "List of theme files that are considered safe to load.
+Each list element should be either an absolute file name, or the
+symbol `default', which stands for the built-in Emacs theme
+directory (a directory named \"themes\" in `data-directory'."
+  :type '(repeat
+         (choice file (const :tag "Built-in theme directory" default)))
+  :group 'customize
+  :version "24.1")
+
 (defvar safe-functions) ; From unsafep.el
 
 (defun load-theme (theme &optional no-enable)
   "Load a theme's settings from its file.
 Normally, this also enables the theme; use `disable-theme' to
 disable it.  If optional arg NO-ENABLE is non-nil, don't enable
-the theme."
-  ;; Note we do no check for validity of the theme here.
-  ;; This allows to pull in themes by a file-name convention
+the theme.
+
+A theme file is named THEME-theme.el, where THEME is the theme name,
+in one of the directories specified by `custom-theme-load-path'."
   (interactive
    (list
     (intern (completing-read "Load custom theme: "
@@ -1132,35 +1143,71 @@ the theme."
                         '("" "c"))))
     (unless fn
       (error "Unable to find theme file for `%s'." theme))
-    ;; Instead of simply loading the theme file, read it manually.
-    (with-temp-buffer
-      (insert-file-contents fn)
-      (require 'unsafep)
-      (let ((custom--inhibit-theme-enable no-enable)
-           (safe-functions (append '(custom-theme-set-variables
-                                     custom-theme-set-faces)
-                                   safe-functions))
-           form scar)
-       (while (setq form (let ((read-circle nil))
-                           (condition-case nil
-                               (read (current-buffer))
-                             (end-of-file nil))))
-         (cond
-          ;; Check `deftheme' expressions.
-          ((eq (setq scar (car form)) 'deftheme)
-           (unless (eq (cadr form) theme)
-             (error "Incorrect theme name in `deftheme'"))
-           (and (symbolp (nth 1 form))
-                (stringp (nth 2 form))
-                (eval (list scar (nth 1 form) (nth 2 form)))))
-          ;; Check `provide-theme' expressions.
-          ((and (eq scar 'provide-theme)
-                (equal (cadr form) `(quote ,theme))
-                (= (length form) 2))
-           (eval form))
-          ;; All other expressions need to be safe.
-          ((not (unsafep form))
-           (eval form))))))))
+    ;; Check file safety.
+    (when (or (and (memq 'default custom-safe-theme-files)
+                  (equal (file-name-directory fn)
+                         (expand-file-name "themes/" data-directory)))
+             (member fn custom-safe-theme-files)
+             ;; If the file is not in the builtin theme directory or
+             ;; in `custom-safe-theme-files', check it with unsafep.
+             (with-temp-buffer
+               (require 'unsafep)
+               (insert-file-contents fn)
+               (let ((safe-functions (append '(provide-theme deftheme
+                                               custom-theme-set-variables
+                                               custom-theme-set-faces)
+                                             safe-functions))
+                     unsafep form)
+                 (while (and (setq form (condition-case nil
+                                            (let ((read-circle nil))
+                                              (read (current-buffer)))
+                                          (end-of-file nil)))
+                             (null (setq unsafep (unsafep form)))))
+                 (or (null unsafep)
+                     (custom-theme-load-confirm fn)))))
+      (let ((custom--inhibit-theme-enable no-enable))
+       (load fn)))))
+
+(defun custom-theme-load-confirm (filename)
+  (if noninteractive
+      nil
+    (let ((existing-buffer (find-buffer-visiting filename))
+         (exit-chars '(?y ?n ?\s ?\C-g))
+         prompt char)
+      (save-window-excursion
+       (if existing-buffer
+           (pop-to-buffer existing-buffer)
+         (find-file filename))
+       (unwind-protect
+           (progn
+             (setq prompt
+                   (format "This theme is not guaranteed to be safe.  Really load? %s"
+                           (if (< (line-number-at-pos (point-max))
+                                  (window-body-height))
+                               "(y or n) "
+                             (push ?\C-v exit-chars)
+                             "Type y or n, or C-v to scroll: ")))
+             (goto-char (point-min))
+             (while (null char)
+               (setq char (read-char-choice prompt exit-chars t))
+               (when (eq char ?\C-v)
+                 (condition-case nil
+                     (scroll-up)
+                   (error (goto-char (point-min))))
+                 (setq char nil)))
+             (when (memq char '(?\s ?y))
+               (push filename custom-safe-theme-files)
+               ;; Offer to save to `custom-safe-theme-files'.
+               (and (or custom-file user-init-file)
+                    (y-or-n-p "Treat %s as safe for future loads? "
+                              (file-name-nondirectory filename))
+                    (let ((coding-system-for-read nil))
+                      (customize-save-variable
+                       'custom-safe-theme-files
+                       custom-safe-theme-files)))
+               t))
+         ;; Unwind form.
+         (unless existing-buffer (kill-buffer)))))))
 
 (defun custom-theme-name-valid-p (name)
   "Return t if NAME is a valid name for a Custom theme, nil otherwise.