;;; core-modules.el --- module & package management system -*- lexical-binding: t; -*- (defvar doom-init-modules-p nil "Non-nil if `doom-initialize-modules' has run.") (defvar doom-modules () "A hash table of enabled modules. Set by `doom-initialize-modules'.") (defvar doom-modules-dirs (list (expand-file-name "modules/" doom-private-dir) doom-modules-dir) "A list of module root directories. Order determines priority.") (defconst doom-obsolete-modules '((:feature (version-control (:emacs vc) (:ui vc-gutter)) (spellcheck (:tools flyspell)) (syntax-checker (:tools flycheck))) (:tools (rotate-text (:editor rotate-text))) (:emacs (electric-indent (:emacs electric)) (hideshow (:editor fold))) (:ui (doom-modeline (:ui modeline))) (:ui (fci (:ui fill-column)))) "An alist of deprecated modules, mapping deprecated modules to an optional new location (which will create an alias). Each CAR and CDR is a (CATEGORY . MODULES). E.g. ((:emacs . electric-indent) . (:emacs electric)) ((:feature . version-control) (:emacs vc) (:ui . vc-gutter)) A warning will be put out if these deprecated modules are used.") (defvar doom--current-module nil) (defvar doom--current-flags nil) ;; ;;; Custom hooks (defvar doom-before-init-modules-hook nil "A list of hooks to run before Doom's modules' config.el files are loaded, but after their init.el files are loaded.") (defvar doom-init-modules-hook nil "A list of hooks to run after Doom's modules' config.el files have loaded, but before the user's private module.") (defvaralias 'doom-after-init-modules-hook 'after-init-hook) (define-obsolete-variable-alias 'doom-post-init-hook 'doom-init-modules-hook "2.1.0") (define-obsolete-variable-alias 'doom-init-hook 'doom-before-init-modules-hook "2.1.0") ;; ;;; Bootstrap API (defun doom-initialize-modules (&optional force-p) "Loads the init.el in `doom-private-dir' and sets up hooks for a healthy session of Dooming. Will noop if used more than once, unless FORCE-P is non-nil." (when (and (or force-p (not doom-init-modules-p)) (not (setq doom-modules nil)) (load! "init" doom-private-dir t)) (setq doom-init-modules-p t) (maphash (lambda (key plist) (let ((doom--current-module key) (doom--current-flags (plist-get plist :flags))) (load! "init" (plist-get plist :path) t))) doom-modules) (run-hook-wrapped 'doom-before-init-modules-hook #'doom-try-run-hook) (unless noninteractive (maphash (lambda (key plist) (let ((doom--current-module key) (doom--current-flags (plist-get plist :flags))) (load! "config" (plist-get plist :path) t))) doom-modules) (run-hook-wrapped 'doom-init-modules-hook #'doom-try-run-hook) (load! "config" doom-private-dir t) (unless custom-file (setq custom-file (concat doom-local-dir "custom.el"))) (when (stringp custom-file) (load custom-file t t t))))) ;; ;;; Module API (defun doom-module-p (category module) "Returns t if CATEGORY MODULE is enabled (ie. present in `doom-modules')." (declare (pure t) (side-effect-free t)) (and (hash-table-p doom-modules) (gethash (cons category module) doom-modules) t)) (defun doom-module-get (category module &optional property) "Returns the plist for CATEGORY MODULE. Gets PROPERTY, specifically, if set." (declare (pure t) (side-effect-free t)) (when-let* ((plist (gethash (cons category module) doom-modules))) (if property (plist-get plist property) plist))) (defun doom-module-put (category module &rest plist) "Set a PROPERTY for CATEGORY MODULE to VALUE. PLIST should be additional pairs of PROPERTY and VALUEs. \(fn CATEGORY MODULE PROPERTY VALUE &rest [PROPERTY VALUE [...]])" (if-let* ((old-plist (doom-module-get category module))) (progn (when plist (when (cl-oddp (length plist)) (signal 'wrong-number-of-arguments (list (length plist)))) (while plist (plist-put old-plist (pop plist) (pop plist)))) (puthash (cons category module) old-plist doom-modules)) (puthash (cons category module) plist doom-modules))) (defun doom-module-set (category module &rest plist) "Enables a module by adding it to `doom-modules'. CATEGORY is a keyword, module is a symbol, PLIST is a plist that accepts the following properties: :flags [SYMBOL LIST] list of enabled category flags :path [STRING] path to category root directory Example: (doom-module-set :lang 'haskell :flags '(+intero))" (puthash (cons category module) plist doom-modules)) (defun doom-module-path (category module &optional file) "Like `expand-file-name', but expands FILE relative to CATEGORY (keywordp) and MODULE (symbol). If the category isn't enabled this will always return nil. For finding disabled modules use `doom-module-locate-path'." (let ((path (doom-module-get category module :path)) file-name-handler-alist) (if file (expand-file-name file path) path))) (defun doom-module-locate-path (category &optional module file) "Searches `doom-modules-dirs' to find the path to a module. CATEGORY is a keyword (e.g. :lang) and MODULE is a symbol (e.g. 'python). FILE is a string that will be appended to the resulting path. If no path exists, this returns nil, otherwise an absolute path. This doesn't require modules to be enabled. For enabled modules us `doom-module-path'." (when (keywordp category) (setq category (doom-keyword-name category))) (when (and module (symbolp module)) (setq module (symbol-name module))) (cl-loop with file-name-handler-alist = nil for default-directory in doom-modules-dirs for path = (concat category "/" module "/" file) if (file-exists-p path) return (expand-file-name path))) (defun doom-module-from-path (&optional path) "Returns a cons cell (CATEGORY . MODULE) derived from PATH (a file path)." (or doom--current-module (let* (file-name-handler-alist (path (or path (FILE!)))) (save-match-data (setq path (file-truename path)) (when (string-match "/modules/\\([^/]+\\)/\\([^/]+\\)\\(?:/.*\\)?$" path) (when-let* ((category (match-string 1 path)) (module (match-string 2 path))) (cons (doom-keyword-intern category) (intern module)))))))) (defun doom-module-load-path (&optional all-p) "Return a list of absolute file paths to activated modules. If ALL-P is non-nil, return paths of possible modules, activated or otherwise." (declare (pure t) (side-effect-free t)) (append (if all-p (doom-files-in doom-modules-dirs :type 'dirs :mindepth 1 :depth 1 :full t) (cl-loop for plist being the hash-values of (doom-modules) collect (plist-get plist :path))) (list doom-private-dir))) (defun doom-modules (&optional refresh-p) "Minimally initialize `doom-modules' (a hash table) and return it." (or (unless refresh-p doom-modules) (let ((noninteractive t) doom-modules doom-init-modules-p) (load! "init" doom-private-dir t) (or doom-modules (make-hash-table :test 'equal :size 20 :rehash-threshold 1.0))))) ;; ;;; Use-package modifications (autoload 'use-package "use-package-core" nil nil t) (setq use-package-compute-statistics doom-debug-mode use-package-verbose doom-debug-mode use-package-minimum-reported-time (if doom-debug-mode 0 0.1) use-package-expand-minimally (not noninteractive)) ;; Adds two new keywords to `use-package' (and consequently, `def-package!') to ;; expand its lazy-loading capabilities. They are: ;; ;; :after-call SYMBOL|LIST ;; :defer-incrementally SYMBOL|LIST|t ;; ;; Check out `def-package!'s documentation for more about these two. (defvar doom--deferred-packages-alist '(t)) (after! use-package-core ;; :ensure and :pin don't work well with Doom, so we forcibly remove them. (dolist (keyword '(:ensure :pin)) (setq use-package-keywords (delq keyword use-package-keywords))) ;; Insert new deferring keywords (dolist (keyword '(:defer-incrementally :after-call)) (add-to-list 'use-package-deferring-keywords keyword nil #'eq) (setq use-package-keywords (use-package-list-insert keyword use-package-keywords :after))) (defalias 'use-package-normalize/:defer-incrementally 'use-package-normalize-symlist) (defun use-package-handler/:defer-incrementally (name _keyword targets rest state) (use-package-concat `((doom-load-packages-incrementally ',(if (equal targets '(t)) (list name) (append targets (list name))))) (use-package-process-keywords name rest state))) (defalias 'use-package-normalize/:after-call 'use-package-normalize-symlist) (defun use-package-handler/:after-call (name _keyword hooks rest state) (if (plist-get state :demand) (use-package-process-keywords name rest state) (let ((fn (intern (format "doom|transient-hook--load-%s" name)))) (use-package-concat `((fset ',fn (lambda (&rest _) (doom-log "Loading deferred package %s from %s" ',name ',fn) (condition-case e (require ',name) ((debug error) (message "Failed to load deferred package %s: %s" ',name e))) (when-let* ((deferral-list (assq ',name doom--deferred-packages-alist))) (dolist (hook (cdr deferral-list)) (if (functionp hook) (advice-remove hook #',fn) (remove-hook hook #',fn))) (setq doom--deferred-packages-alist (delq deferral-list doom--deferred-packages-alist)))))) (let (forms) (dolist (hook hooks forms) (push (if (functionp hook) `(advice-add #',hook :before #',fn) `(add-hook ',hook #',fn)) forms))) `((unless (assq ',name doom--deferred-packages-alist) (push '(,name) doom--deferred-packages-alist)) (nconc (assq ',name doom--deferred-packages-alist) '(,@hooks))) (use-package-process-keywords name rest state)))))) ;; ;;; Module config macros (defmacro doom! (&rest modules) "Bootstraps DOOM Emacs and its modules. The bootstrap process involves making sure the essential directories exist, core packages are installed, `doom-autoload-file' is loaded, `doom-packages-file' cache exists (and is loaded) and, finally, loads your private init.el (which should contain your `doom!' block). If the cache exists, much of this function isn't run, which substantially reduces startup time. The overall load order of Doom is as follows: ~/.emacs.d/init.el ~/.emacs.d/core/core.el $DOOMDIR/init.el {$DOOMDIR,~/.emacs.d}/modules/*/*/init.el `doom-before-init-modules-hook' {$DOOMDIR,~/.emacs.d}/modules/*/*/config.el `doom-init-modules-hook' $DOOMDIR/config.el `doom-after-init-modules-hook' `after-init-hook' `emacs-startup-hook' `window-setup-hook' Module load order is determined by your `doom!' block. See `doom-modules-dirs' for a list of all recognized module trees. Order defines precedence (from most to least)." (unless doom-modules (setq doom-modules (make-hash-table :test 'equal :size (if modules (length modules) 150) :rehash-threshold 1.0))) (let (category m) (while modules (setq m (pop modules)) (cond ((keywordp m) (setq category m)) ((not category) (error "No module category specified for %s" m)) ((catch 'doom-modules (let* ((module (if (listp m) (car m) m)) (flags (if (listp m) (cdr m)))) (when-let* ((obsolete (assq category doom-obsolete-modules)) (new (assq module obsolete))) (let ((newkeys (cdr new))) (if (null newkeys) (message "Warning: the %s module is deprecated" key) (message "Warning: the %s module is deprecated. Use %s instead." (list category module) newkeys) (push category modules) (dolist (key newkeys) (setq modules (append key modules))) (throw 'doom-modules t)))) (if-let* ((path (doom-module-locate-path category module))) (doom-module-set category module :flags flags :path path) (message "Warning: couldn't find the %s %s module" category module)))))))) `(setq doom-modules ',doom-modules)) (defvar doom-disabled-packages) (defmacro def-package! (name &rest plist) "This is a thin wrapper around `use-package'. It is ignored if the NAME package is disabled. Supports two special properties over `use-package': :after-call SYMBOL|LIST Takes a symbol or list of symbols representing functions or hook variables. The first time any of these functions or hooks are executed, the package is loaded. e.g. (def-package! projectile :after-call (pre-command-hook after-find-file dired-before-readin-hook) ...) :defer-incrementally SYMBOL|LIST|t Takes a symbol or list of symbols representing packages that will be loaded incrementally at startup before this one. This is helpful for large packages like magit or org, which load a lot of dependencies on first load. This lets you load them piece-meal during idle periods, so that when you finally do need the package, it'll load quicker. e.g. NAME is implicitly added if this property is present and non-nil. No need to specify it. A value of `t' implies NAME, e.g. (def-package! x ;; This is equivalent to :defer-incrementally (x) :defer-incrementally t ...)" (unless (or (memq name doom-disabled-packages) ;; At compile-time, use-package will forcibly load its package to ;; prevent compile-time errors. However, Doom users can ;; intentionally disable packages, resulting if file-missing ;; package errors, so we preform this check at compile time: (and (bound-and-true-p byte-compile-current-file) (not (locate-library (symbol-name name))))) `(use-package ,name ,@plist))) (defmacro def-package-hook! (package when &rest body) "Reconfigures a package's `def-package!' block. Only use this macro in a module's init.el file. Under the hood, this uses use-package's `use-package-inject-hooks'. PACKAGE is a symbol; the package's name. WHEN should be one of the following: :pre-init :post-init :pre-config :post-config WARNING: If :pre-init or :pre-config hooks return nil, the original `def-package!''s :init/:config block (respectively) is overwritten, so remember to have them return non-nil (or exploit that to overwrite Doom's config)." (declare (indent defun)) (doom--assert-stage-p 'init #'package!) (unless (memq when '(:pre-init :post-init :pre-config :post-config)) (error "'%s' isn't a valid hook for def-package-hook!" when)) `(progn (setq use-package-inject-hooks t) (add-hook! ',(intern (format "use-package--%s--%s-hook" package (substring (symbol-name when) 1))) ,@body))) (defmacro require! (category module &rest plist) "Loads the module specified by CATEGORY (a keyword) and MODULE (a symbol)." `(let ((module-path (doom-module-locate-path ,category ',module))) (doom-module-set ,category ',module ,@(when plist (let ((old-plist (doom-module-get category module))) (unless (plist-member plist :flags) (plist-put plist :flags (plist-get old-plist :flags))) (unless (plist-member plist :path) (plist-put plist :path (or (plist-get old-plist :path) (doom-module-locate-path category module))))) plist)) (if (directory-name-p module-path) (condition-case-unless-debug ex (let ((doom--current-module ',(cons category module))) (load! "init" module-path :noerror) (let ((doom--stage 'config)) (load! "config" module-path :noerror))) ('error (lwarn 'doom-modules :error "%s in '%s %s' -> %s" (car ex) ,category ',module (error-message-string ex)))) (warn 'doom-modules :warning "Couldn't find module '%s %s'" ,category ',module)))) (defmacro featurep! (category &optional module flag) "Returns t if CATEGORY MODULE is enabled. If FLAG is provided, returns t if CATEGORY MODULE has FLAG enabled. (featurep! :config default) Module FLAGs are set in your config's `doom!' block, typically in ~/.emacs.d/init.el. Like so: :config (default +flag1 -flag2) When this macro is used from inside a module, CATEGORY and MODULE can be omitted. eg. (featurep! +flag1)" (and (cond (flag (memq flag (doom-module-get category module :flags))) (module (doom-module-p category module)) (doom--current-flags (memq category doom--current-flags)) ((let ((module-pair (or doom--current-module (doom-module-from-path (FILE!))))) (unless module-pair (error "featurep! couldn't detect what module its in! (in %s)" (FILE!))) (memq category (doom-module-get (car module-pair) (cdr module-pair) :flags))))) t)) (provide 'core-modules) ;;; core-modules.el ends here