Add new cache library (persistent-soft wrapper)

This commit is contained in:
Henrik Lissner 2018-01-06 17:00:14 -05:00
parent fcd87f6f69
commit 0042a56d02
No known key found for this signature in database
GPG key ID: 5F6C0EA160557395
4 changed files with 104 additions and 16 deletions

97
core/autoload/cache.el Normal file
View file

@ -0,0 +1,97 @@
;;; ../core/autoload/cache.el -*- lexical-binding: t; -*-
;; This little library thinly wraps around persistent-soft (which is a pcache
;; wrapper, how about that). It has three purposes:
;;
;; + To encapsulate the cache backend (persistent-soft/pcache in this case), in
;; case it needs to change.
;; + To provide `doom-cache-persist': a mechanism for easily persisting
;; variables across Emacs sessions.
;; + To lazy-load persistent-soft until it is really needed.
;;
;; Like persistent-soft, caches assume a 2-tier structure, where all caches are
;; namespaced by location.
(require 'persistent-soft)
(defvar doom-cache-alists ()
"An alist of alists, containing lists of variables for the doom cache library
to persist across Emacs sessions.")
(defvar doom-cache-location 'doom
"The default location for cache files. This symbol is translated into a file
name under `pcache-directory' (by default a subdirectory under
`doom-cache-dir'). One file may contain multiple cache entries.")
(defun doom|save-persistent-cache ()
"Hook to run when an Emacs session is killed. Saves all persisted variables
listed in `doom-cache-alists' to files."
(dolist (alist doom-cache-alists)
(cl-loop with key = (car alist)
for var in (cdr alist)
if (symbol-value var)
do (doom-cache-set var it nil key))))
(add-hook 'kill-emacs-hook #'doom|save-persistent-cache)
;;
;; Library
;;
;;;###autoload
(defmacro with-cache! (location &rest body)
"Runs BODY with a different default `doom-cache-location'."
(declare (indent defun))
`(let ((doom-cache-location ',location))
,@body))
;;;###autoload
(defun doom-cache-persist (location variables)
"Persist VARIABLES (list of symbols) in LOCATION (symbol).
This populates these variables with cached values, if one exists, and saves them
to file when Emacs quits.
Warning: this is incompatible with buffer-local variables."
(dolist (var variables)
(when (doom-cache-exists var location)
(set var (doom-cache-get var location))))
(map-put doom-cache-alists location
(append variables (cdr (assq location doom-cache-alists)))))
;;;###autoload
(defun doom-cache-desist (location &optional variables)
"Unregisters VARIABLES (list of symbols) in LOCATION (symbol) from
`doom-cache-alists', thus preventing them from being saved between sessions.
Does not affect the actual variables themselves or their values."
(if variables
(map-put doom-cache-alists location
(cl-set-difference (cdr (assq location doom-cache-alists))
variables))
(map-delete doom-cache-alists location)))
;;;###autoload
(defun doom-cache-get (key &optional location)
"Retrieve KEY from LOCATION (defaults to `doom-cache-location'), if it exists
and hasn't expired."
(persistent-soft-fetch
key (symbol-name (or location doom-cache-location))))
;;;###autoload
(defun doom-cache-set (key value &optional ttl location)
"Set KEY to VALUE in the cache. TTL is the time (in seconds) until this cache
entry expires. LOCATION is the super-key to store this cache item under; the
default is `doom-cache-location'. "
(persistent-soft-store
key value
(symbol-name (or location doom-cache-location)) ttl))
;;;###autoload
(defun doom-cache-exists (key &optional location)
"Returns t if KEY exists at LOCATION (defaults to `doom-cache-location')."
(persistent-soft-exists-p key (or location doom-cache-location)))
;;;###autoload
(defun doom-cache-clear (&optional location)
"Clear a cache LOCATION (defaults to `doom-cache-location')."
(persistent-soft-flush (or location doom-cache-location)))

View file

@ -1,22 +1,22 @@
;;; core/autoload/packages.el -*- lexical-binding: t; -*- ;;; core/autoload/packages.el -*- lexical-binding: t; -*-
(load! cache)
(require 'use-package) (require 'use-package)
(require 'quelpa) (require 'quelpa)
(require 'async)
(defvar doom--last-refresh nil)
;;;###autoload ;;;###autoload
(defun doom-refresh-packages (&optional force-p) (defun doom-refresh-packages (&optional force-p)
"Refresh ELPA packages." "Refresh ELPA packages."
(when force-p (when force-p
(doom-refresh-clear-cache)) (doom-refresh-clear-cache))
(unless (or (persistent-soft-fetch 'last-pkg-refresh "emacs") (unless (or (doom-cache-get 'last-pkg-refresh)
doom--refreshed-p) doom--refreshed-p)
(condition-case-unless-debug ex (condition-case-unless-debug ex
(progn (progn
(message "Refreshing package archives") (message "Refreshing package archives")
(package-refresh-contents) (package-refresh-contents)
(persistent-soft-store 'last-pkg-refresh t "emacs" 900)) (doom-cache-set 'last-pkg-refresh t 900))
('error ('error
(doom-refresh-clear-cache) (doom-refresh-clear-cache)
(message "Failed to refresh packages: (%s) %s" (message "Failed to refresh packages: (%s) %s"
@ -26,7 +26,7 @@
(defun doom-refresh-clear-cache () (defun doom-refresh-clear-cache ()
"Clear the cache for `doom-refresh-packages'." "Clear the cache for `doom-refresh-packages'."
(setq doom--refreshed-p nil) (setq doom--refreshed-p nil)
(persistent-soft-store 'last-pkg-refresh nil "emacs")) (doom-cache-set 'last-pkg-refresh nil))
;;;###autoload ;;;###autoload
(defun doom-package-backend (name &optional noerror) (defun doom-package-backend (name &optional noerror)

View file

@ -2,7 +2,6 @@
(require 'subr-x) (require 'subr-x)
(load "async-autoloads" nil t) (load "async-autoloads" nil t)
(load "persistent-soft-autoloads" nil t)
(dolist (sym '(json-read json-read-file json-read-from-string json-encode)) (dolist (sym '(json-read json-read-file json-read-from-string json-encode))
(autoload sym "json")) (autoload sym "json"))
(eval-and-compile (eval-and-compile

View file

@ -16,16 +16,8 @@
(set! :evil-state 'prodigy-mode 'emacs) (set! :evil-state 'prodigy-mode 'emacs)
;; Make services, etc persistent between Emacs sessions ;; Make services, etc persistent between Emacs sessions
(setq prodigy-services (persistent-soft-fetch 'prodigy-services "prodigy") (doom-cache-persist
prodigy-tags (persistent-soft-fetch 'prodigy-tags "prodigy") :prodigy '(prodigy-services prodigy-tags prodigy-filters))
prodigy-filters (persistent-soft-fetch 'prodigy-filters "prodigy"))
(defun +services|save ()
"Save all services, tags and filters to files."
(+services/cleanup)
(cl-loop for sym in '(prodigy-services prodigy-tags prodigy-filters)
do (persistent-soft-store sym (symbol-value sym) "prodigy")))
(add-hook 'kill-emacs-hook #'+services|save)
(defun +services*prodigy-services (orig-fn &rest args) (defun +services*prodigy-services (orig-fn &rest args)
"Adds a new :project property to prodigy services, which hides the service "Adds a new :project property to prodigy services, which hides the service