2017-02-02 19:15:58 -05:00
|
|
|
;;; packages.el
|
|
|
|
|
2017-04-04 12:29:34 -04:00
|
|
|
(defvar doom--last-refresh nil)
|
|
|
|
|
2017-02-03 07:58:16 -05:00
|
|
|
;;;###autoload
|
|
|
|
(defun doom-refresh-packages ()
|
|
|
|
"Refresh ELPA packages."
|
2017-02-04 21:07:54 -05:00
|
|
|
(doom-initialize)
|
2017-02-19 06:59:55 -05:00
|
|
|
(let ((last-refresh (persistent-soft-fetch 'last-pkg-refresh "emacs")))
|
2017-04-04 12:29:34 -04:00
|
|
|
(when last-refresh
|
|
|
|
(setq doom--last-refresh last-refresh)))
|
|
|
|
(when (or (not doom--last-refresh)
|
|
|
|
(> (nth 1 (time-since doom--last-refresh)) 600))
|
|
|
|
(package-refresh-contents)
|
|
|
|
(persistent-soft-store
|
|
|
|
'last-pkg-refresh (setq doom--last-refresh (current-time))
|
|
|
|
"emacs")))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
2017-02-04 21:07:54 -05:00
|
|
|
(defun doom-package-backend (name)
|
|
|
|
"Get which backend the package NAME was installed with. Can either be elpa,
|
|
|
|
quelpa or nil (if not installed)."
|
2017-02-03 07:58:16 -05:00
|
|
|
(doom-initialize)
|
|
|
|
(unless (quelpa-setup-p)
|
|
|
|
(error "Could not initialize quelpa"))
|
2017-02-20 01:14:10 -05:00
|
|
|
(cond ((let ((plist (cdr (assq name doom-packages))))
|
|
|
|
(and (not (plist-get plist :pin))
|
|
|
|
(or (assq name quelpa-cache)
|
|
|
|
(plist-get plist :recipe))))
|
2017-02-04 21:07:54 -05:00
|
|
|
'quelpa)
|
|
|
|
((assq name package-alist)
|
|
|
|
'elpa)))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom-package-outdated-p (name)
|
|
|
|
"Determine whether NAME (a symbol) is outdated or not. If outdated, returns a
|
|
|
|
list, whose car is NAME, and cdr the current version list and latest version
|
|
|
|
list of the package."
|
2017-02-19 06:59:55 -05:00
|
|
|
(doom-initialize)
|
|
|
|
(when-let (pkg (assq name package-alist))
|
2017-02-11 00:46:42 -05:00
|
|
|
(let* ((old-version (package-desc-version (cadr pkg)))
|
|
|
|
(new-version
|
|
|
|
(pcase (doom-package-backend name)
|
|
|
|
('quelpa
|
|
|
|
(let ((recipe (assq name quelpa-cache))
|
2017-02-19 06:59:55 -05:00
|
|
|
(dir (expand-file-name (symbol-name name) quelpa-build-dir))
|
2017-02-11 00:46:42 -05:00
|
|
|
(inhibit-message t))
|
2017-02-19 06:59:55 -05:00
|
|
|
(if-let (ver (and (quelpa-setup-p) (quelpa-checkout recipe dir)))
|
2017-02-11 00:46:42 -05:00
|
|
|
(version-to-list ver)
|
|
|
|
old-version)))
|
|
|
|
('elpa
|
2017-02-19 06:59:55 -05:00
|
|
|
(doom-refresh-packages)
|
2017-02-11 00:46:42 -05:00
|
|
|
(let ((desc (cadr (assq name package-archive-contents))))
|
|
|
|
(when (package-desc-p desc)
|
|
|
|
(package-desc-version desc)))))))
|
|
|
|
(when (and (listp old-version) (listp new-version)
|
|
|
|
(version-list-< old-version new-version))
|
|
|
|
(list name old-version new-version)))))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom-get-packages (&optional backend)
|
|
|
|
"Retrieves a list of explicitly installed packages (i.e. non-dependencies).
|
|
|
|
Each element is a cons cell, whose car is the package symbol and whose cdr is
|
|
|
|
the quelpa recipe (if any).
|
|
|
|
|
|
|
|
BACKEND can be 'quelpa or 'elpa, and will instruct this function to return only
|
2017-02-19 06:59:55 -05:00
|
|
|
the packages relevant to that backend.
|
|
|
|
|
|
|
|
Warning: this function is expensive; it re-evaluates all of doom's config files.
|
|
|
|
Be careful not to use it in a loop."
|
2017-02-11 00:46:42 -05:00
|
|
|
(doom-initialize-packages t)
|
2017-02-03 07:58:16 -05:00
|
|
|
(unless (quelpa-setup-p)
|
|
|
|
(error "Could not initialize quelpa"))
|
2017-02-20 01:22:22 -05:00
|
|
|
(delq nil
|
|
|
|
(mapcar (lambda (pkgsym)
|
|
|
|
(or (assq pkgsym doom-packages)
|
|
|
|
(list (car (assq pkgsym package-alist)))))
|
2017-03-14 15:18:36 -04:00
|
|
|
(cl-delete-duplicates
|
|
|
|
(append doom-protected-packages (mapcar 'car doom-packages))))))
|
2017-02-19 06:59:55 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom-get-dependencies-for (name)
|
|
|
|
"Return a list of packages that depend on the package named NAME."
|
|
|
|
(doom-initialize)
|
|
|
|
(when-let (desc (cadr (assq name package-alist)))
|
|
|
|
(mapcar 'package-desc-name (package--used-elsewhere-p desc nil t))))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom-get-outdated-packages ()
|
2017-02-11 00:46:42 -05:00
|
|
|
"Return a list of packages that are out of date. Each element is a list,
|
|
|
|
containing (PACKAGE-SYMBOL OLD-VERSION-LIST NEW-VERSION-LIST).
|
2017-02-09 04:25:32 -05:00
|
|
|
|
|
|
|
Used by `doom/packages-update'."
|
2017-02-19 06:59:55 -05:00
|
|
|
(delq nil (mapcar 'doom-package-outdated-p (mapcar 'car (doom-get-packages)))))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
2017-02-02 19:15:58 -05:00
|
|
|
;;;###autoload
|
2017-02-03 07:58:16 -05:00
|
|
|
(defun doom-get-orphaned-packages ()
|
2017-02-09 04:25:32 -05:00
|
|
|
"Return a list of symbols representing packages that are no longer needed or
|
|
|
|
depended on.
|
|
|
|
|
|
|
|
Used by `doom/packages-autoremove'."
|
2017-02-11 00:46:42 -05:00
|
|
|
(doom-initialize-packages t)
|
2017-02-19 06:59:55 -05:00
|
|
|
(let ((package-selected-packages
|
|
|
|
(append (mapcar 'car doom-packages) doom-protected-packages)))
|
2017-02-03 19:20:47 -05:00
|
|
|
(package--removable-packages)))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
2017-02-11 06:01:03 -05:00
|
|
|
(defun doom-get-missing-packages ()
|
2017-02-19 06:59:55 -05:00
|
|
|
"Return a list of requested packages that aren't installed or built-in. Each
|
|
|
|
element is a list whose CAR is the package symbol, and whose CDR is a plist
|
2017-02-23 00:06:12 -05:00
|
|
|
taken from that package's `package!' declaration.
|
2017-02-09 04:25:32 -05:00
|
|
|
|
|
|
|
Used by `doom/packages-install'."
|
2017-02-19 06:59:55 -05:00
|
|
|
(cl-remove-if (lambda (pkgsym)
|
|
|
|
(or (assq (car pkgsym) package-alist)
|
|
|
|
(and (not (plist-get (assq (car pkgsym) doom-packages) :pin))
|
|
|
|
(assq (car pkgsym) package--builtins))))
|
|
|
|
(doom-get-packages)))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
2017-02-03 19:20:47 -05:00
|
|
|
;;;###autoload
|
2017-02-19 06:59:55 -05:00
|
|
|
(defun doom*package-delete (name &rest _)
|
2017-02-06 00:13:24 -05:00
|
|
|
"Update `quelpa-cache' upon a successful `package-delete'."
|
2017-02-03 19:20:47 -05:00
|
|
|
(when (and (not (package-installed-p name))
|
|
|
|
(quelpa-setup-p)
|
|
|
|
(assq name quelpa-cache))
|
|
|
|
(setq quelpa-cache (assq-delete-all name quelpa-cache))
|
|
|
|
(quelpa-save-cache)
|
2017-02-19 06:59:55 -05:00
|
|
|
(let ((path (expand-file-name (symbol-name name) quelpa-build-dir)))
|
|
|
|
(when (file-exists-p path)
|
2017-02-03 19:20:47 -05:00
|
|
|
(delete-directory path t)))))
|
|
|
|
|
2017-02-19 06:59:55 -05:00
|
|
|
;;; Private functions
|
|
|
|
(defsubst doom--sort-alpha (it other)
|
|
|
|
(string-lessp (symbol-name (car it))
|
|
|
|
(symbol-name (car other))))
|
|
|
|
|
2017-02-03 07:58:16 -05:00
|
|
|
|
|
|
|
;;
|
|
|
|
;; Main functions
|
|
|
|
;;
|
|
|
|
|
2017-02-04 21:07:54 -05:00
|
|
|
(defun doom-install-package (name &optional plist)
|
2017-02-03 07:58:16 -05:00
|
|
|
"Installs package NAME with optional quelpa RECIPE (see `quelpa-recipe' for an
|
|
|
|
example; the package name can be omitted)."
|
|
|
|
(doom-refresh-packages)
|
2017-02-11 00:46:42 -05:00
|
|
|
(doom-initialize-packages)
|
2017-02-03 07:58:16 -05:00
|
|
|
(when (package-installed-p name)
|
2017-02-04 21:07:54 -05:00
|
|
|
(error "%s is already installed, skipping" name))
|
2017-02-11 00:46:42 -05:00
|
|
|
(let ((inhibit-message (not doom-debug-mode))
|
|
|
|
(recipe (plist-get plist :recipe)))
|
|
|
|
(cond (recipe (quelpa recipe))
|
2017-02-06 01:22:54 -05:00
|
|
|
(t (package-install name))))
|
2017-03-05 14:18:35 -05:00
|
|
|
(cl-pushnew (cons name plist) doom-packages :test 'eq :key 'car)
|
2017-02-03 07:58:16 -05:00
|
|
|
(package-installed-p name))
|
|
|
|
|
|
|
|
(defun doom-update-package (name)
|
|
|
|
"Updates package NAME if it is out of date, using quelpa or package.el as
|
|
|
|
appropriate."
|
2017-02-19 06:59:55 -05:00
|
|
|
(doom-initialize)
|
2017-02-03 07:58:16 -05:00
|
|
|
(unless (package-installed-p name)
|
|
|
|
(error "%s isn't installed" name))
|
|
|
|
(when (doom-package-outdated-p name)
|
2017-02-06 01:22:54 -05:00
|
|
|
(let ((inhibit-message (not doom-debug-mode))
|
|
|
|
quelpa-modified-p)
|
2017-02-04 21:07:54 -05:00
|
|
|
(pcase (doom-package-backend name)
|
|
|
|
('quelpa
|
|
|
|
(let ((quelpa-upgrade-p t))
|
2017-02-11 00:46:42 -05:00
|
|
|
(quelpa (assq name quelpa-cache))
|
2017-02-04 21:07:54 -05:00
|
|
|
(setq quelpa-modified-p t)))
|
|
|
|
('elpa
|
|
|
|
(let ((desc (cadr (assq name package-alist)))
|
|
|
|
(archive (cadr (assq name package-archive-contents))))
|
|
|
|
(package-install-from-archive archive)
|
|
|
|
(delete-directory (package-desc-dir desc) t))))
|
|
|
|
(when quelpa-modified-p
|
|
|
|
(quelpa-save-cache)))
|
2017-02-03 07:58:16 -05:00
|
|
|
(version-list-=
|
|
|
|
(package-desc-version (cadr (assq name package-alist)))
|
|
|
|
(package-desc-version (cadr (assq name package-archive-contents))))))
|
|
|
|
|
2017-02-19 06:59:55 -05:00
|
|
|
(defun doom-delete-package (name &optional force-p)
|
2017-02-03 07:58:16 -05:00
|
|
|
"Uninstalls package NAME if it exists, and clears it from `quelpa-cache'."
|
|
|
|
(doom-initialize)
|
|
|
|
(unless (package-installed-p name)
|
|
|
|
(error "%s isn't installed" name))
|
2017-02-19 06:59:55 -05:00
|
|
|
(let ((desc (cadr (assq name package-alist)))
|
|
|
|
(inhibit-message t))
|
|
|
|
(package-delete desc force-p))
|
2017-02-03 07:58:16 -05:00
|
|
|
(not (package-installed-p name)))
|
|
|
|
|
|
|
|
|
|
|
|
;;
|
|
|
|
;; Interactive commands
|
|
|
|
;;
|
2017-02-02 19:15:58 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom/packages-install ()
|
2017-02-03 19:20:47 -05:00
|
|
|
"Interactive command for installing missing packages."
|
2017-02-02 19:15:58 -05:00
|
|
|
(interactive)
|
2017-02-11 06:01:03 -05:00
|
|
|
(let ((packages (doom-get-missing-packages)))
|
2017-02-03 07:58:16 -05:00
|
|
|
(cond ((not packages)
|
|
|
|
(message "No packages to install!"))
|
|
|
|
|
2017-02-19 06:59:55 -05:00
|
|
|
((not (or (getenv "YES")
|
|
|
|
(y-or-n-p
|
|
|
|
(format "%s packages will be installed:\n\n%s\n\nProceed?"
|
|
|
|
(length packages)
|
|
|
|
(mapconcat (lambda (pkg)
|
|
|
|
(format "+ %s (%s)"
|
|
|
|
(car pkg)
|
|
|
|
(if (plist-get (cdr pkg) :recipe)
|
|
|
|
"QUELPA"
|
|
|
|
"ELPA")))
|
|
|
|
(sort (cl-copy-list packages) 'doom--sort-alpha)
|
|
|
|
"\n")))))
|
2017-02-03 07:58:16 -05:00
|
|
|
(message "Aborted!"))
|
|
|
|
|
|
|
|
(t
|
2017-02-19 06:59:55 -05:00
|
|
|
(message "Installing %s packages" (length packages))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
|
|
|
(dolist (pkg packages)
|
|
|
|
(condition-case ex
|
2017-02-19 06:59:55 -05:00
|
|
|
(progn
|
|
|
|
(message "%s %s (%s)"
|
|
|
|
(cond ((package-installed-p (car pkg))
|
|
|
|
"Skipped (already installed)")
|
|
|
|
((doom-install-package (car pkg) (cdr pkg))
|
|
|
|
"Installed")
|
|
|
|
(t "Failed to install"))
|
2017-02-20 01:21:57 -05:00
|
|
|
(concat (symbol-name (car pkg))
|
|
|
|
(when (plist-member (cdr pkg) :pin)
|
|
|
|
(format " [pinned: %s]" (plist-get (cdr pkg) :pin))))
|
|
|
|
(pcase (doom-package-backend (car pkg))
|
|
|
|
('quelpa "QUELPA")
|
|
|
|
('elpa "ELPA"))))
|
2017-02-03 20:10:03 -05:00
|
|
|
(error
|
2017-02-19 06:59:55 -05:00
|
|
|
(message "Error (%s): %s" (car pkg) ex))))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
2017-03-04 00:26:37 -05:00
|
|
|
(message "Finished!")
|
2017-03-25 01:03:40 -04:00
|
|
|
(doom/reload)))))
|
2017-02-02 19:15:58 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
|
|
|
(defun doom/packages-update ()
|
2017-02-03 19:20:47 -05:00
|
|
|
"Interactive command for updating packages."
|
2017-02-02 19:15:58 -05:00
|
|
|
(interactive)
|
2017-02-19 18:02:40 -05:00
|
|
|
(let ((packages (sort (doom-get-outdated-packages) 'doom--sort-alpha)))
|
2017-02-03 07:58:16 -05:00
|
|
|
(cond ((not packages)
|
|
|
|
(message "Everything is up-to-date"))
|
|
|
|
|
2017-02-19 06:59:55 -05:00
|
|
|
((not (or (getenv "YES")
|
|
|
|
(y-or-n-p
|
|
|
|
(format "%s packages will be updated:\n\n%s\n\nProceed?"
|
|
|
|
(length packages)
|
|
|
|
(let ((max-len
|
|
|
|
(or (car (sort (mapcar (lambda (it) (length (symbol-name (car it)))) packages)
|
|
|
|
(lambda (it other) (> it other))))
|
|
|
|
10)))
|
|
|
|
(mapconcat
|
|
|
|
(lambda (pkg)
|
|
|
|
(format "+ %s %s -> %s"
|
|
|
|
(s-pad-right (+ max-len 2) " " (symbol-name (car pkg)))
|
2017-04-04 13:16:11 -04:00
|
|
|
(s-pad-right 14 " " (package-version-join (cadr pkg)))
|
|
|
|
(package-version-join (cl-caddr pkg))))
|
2017-02-19 06:59:55 -05:00
|
|
|
packages
|
|
|
|
"\n"))))))
|
2017-02-03 07:58:16 -05:00
|
|
|
(message "Aborted!"))
|
|
|
|
|
|
|
|
(t
|
|
|
|
(dolist (pkg packages)
|
|
|
|
(condition-case ex
|
2017-02-19 06:59:55 -05:00
|
|
|
(progn
|
|
|
|
(message "%s %s"
|
|
|
|
(if (doom-update-package (car pkg))
|
|
|
|
"Updated"
|
|
|
|
"Failed to update")
|
|
|
|
(car pkg)))
|
2017-02-03 20:10:03 -05:00
|
|
|
(error
|
2017-02-19 06:59:55 -05:00
|
|
|
(message "Error installing %s: %s" (car pkg) ex))))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
2017-03-04 00:26:37 -05:00
|
|
|
(message "Finished!")
|
2017-03-25 01:03:40 -04:00
|
|
|
(doom/reload)))))
|
2017-02-02 19:15:58 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
2017-02-03 07:58:16 -05:00
|
|
|
(defun doom/packages-autoremove ()
|
2017-02-03 19:20:47 -05:00
|
|
|
"Interactive command for auto-removing orphaned packages."
|
2017-02-02 19:15:58 -05:00
|
|
|
(interactive)
|
2017-02-03 07:58:16 -05:00
|
|
|
(let ((packages (doom-get-orphaned-packages)))
|
|
|
|
(cond ((not packages)
|
|
|
|
(message "No unused packages to remove"))
|
2017-02-02 19:15:58 -05:00
|
|
|
|
2017-02-19 06:59:55 -05:00
|
|
|
((not (or (getenv "YES")
|
|
|
|
(y-or-n-p
|
|
|
|
(format "%s packages will be deleted:\n\n%s\n\nProceed?"
|
|
|
|
(length packages)
|
|
|
|
(mapconcat (lambda (sym) (format "+ %s" (symbol-name sym)))
|
|
|
|
(sort (cl-copy-list packages) 'string-lessp)
|
|
|
|
"\n")))))
|
2017-02-03 07:58:16 -05:00
|
|
|
(message "Aborted!"))
|
2017-02-02 19:15:58 -05:00
|
|
|
|
|
|
|
(t
|
2017-02-03 07:58:16 -05:00
|
|
|
(dolist (pkg packages)
|
|
|
|
(condition-case ex
|
2017-02-19 06:59:55 -05:00
|
|
|
(message "%s %s"
|
|
|
|
(if (doom-delete-package pkg t)
|
2017-02-03 07:58:16 -05:00
|
|
|
"Deleted"
|
|
|
|
"Failed to delete")
|
|
|
|
pkg)
|
2017-02-03 20:10:03 -05:00
|
|
|
(error
|
2017-02-19 06:59:55 -05:00
|
|
|
(message "Error deleting %s: %s" pkg ex))))
|
2017-02-03 07:58:16 -05:00
|
|
|
|
2017-03-04 00:26:37 -05:00
|
|
|
(message "Finished!")
|
2017-03-25 01:03:40 -04:00
|
|
|
(doom/reload)))))
|
2017-02-11 00:46:42 -05:00
|
|
|
|
2017-02-02 19:15:58 -05:00
|
|
|
;;;###autoload
|
2017-02-04 21:07:54 -05:00
|
|
|
(defalias 'doom/install-package 'package-install)
|
2017-02-02 19:15:58 -05:00
|
|
|
|
|
|
|
;;;###autoload
|
2017-02-11 00:46:42 -05:00
|
|
|
(defun doom/delete-package (package)
|
|
|
|
"Prompts the user with a list of packages and deletes the selected package.
|
|
|
|
Use this interactively. Use `doom-delete-package' for direct calls."
|
2017-02-03 07:58:16 -05:00
|
|
|
(interactive
|
2017-02-20 21:30:41 -05:00
|
|
|
(progn
|
|
|
|
(doom-initialize)
|
|
|
|
(list (completing-read
|
|
|
|
"Delete package: "
|
|
|
|
(delq nil
|
|
|
|
(mapcar (lambda (p) (unless (package-built-in-p p) p))
|
|
|
|
(mapcar 'car package-alist)))
|
|
|
|
nil t))))
|
2017-02-03 07:58:16 -05:00
|
|
|
(if (package-installed-p package)
|
2017-02-11 00:46:42 -05:00
|
|
|
(if (y-or-n-p (format "%s will be deleted. Confirm?" package))
|
|
|
|
(message "%s %s"
|
|
|
|
(if (doom-delete-package package)
|
|
|
|
"Deleted"
|
|
|
|
"Failed to delete")
|
|
|
|
pkg)
|
|
|
|
(message "Aborted"))
|
2017-02-03 07:58:16 -05:00
|
|
|
(message "%s isn't installed" package)))
|
|
|
|
|
|
|
|
;;;###autoload
|
2017-02-11 00:46:42 -05:00
|
|
|
(defun doom/update-package (package)
|
|
|
|
"Prompts the user with a list of outdated packages and updates the selected
|
|
|
|
package. Use this interactively. Use `doom-update-package' for direct
|
|
|
|
calls."
|
2017-02-06 00:13:24 -05:00
|
|
|
(declare (interactive-only t))
|
2017-02-03 07:58:16 -05:00
|
|
|
(interactive
|
2017-02-11 00:46:42 -05:00
|
|
|
(let ((packages (doom-get-outdated-packages)))
|
|
|
|
(list
|
|
|
|
(if packages
|
2017-02-19 06:59:55 -05:00
|
|
|
(completing-read "Update package: " (mapcar 'symbol-name (mapcar 'car packages)))
|
2017-02-11 00:46:42 -05:00
|
|
|
(user-error "All packages are up-to-date")))))
|
2017-02-19 06:59:55 -05:00
|
|
|
(if-let (desc (doom-package-outdated-p (intern package)))
|
2017-02-11 00:46:42 -05:00
|
|
|
(if (y-or-n-p (format "%s will be updated from %s to %s. Update?"
|
|
|
|
(car desc)
|
2017-04-04 13:16:11 -04:00
|
|
|
(package-version-join (cadr desc))
|
|
|
|
(package-version-join (cl-caddr desc))))
|
2017-02-11 00:46:42 -05:00
|
|
|
(message "%s %s"
|
|
|
|
(if (doom-update-package package)
|
|
|
|
"Updated"
|
|
|
|
"Failed to update")
|
|
|
|
pkg)
|
|
|
|
(message "Aborted"))
|
2017-02-03 07:58:16 -05:00
|
|
|
(message "%s is up-to-date" package)))
|