2017-01-16 23:15:48 -05:00
|
|
|
;;; core-packages.el
|
2017-02-06 00:12:44 -05:00
|
|
|
;;
|
2017-01-31 19:42:41 -05:00
|
|
|
;; Emacs package management is opinionated. Unfortunately, so am I. So I
|
2017-02-06 00:12:44 -05:00
|
|
|
;; combined `use-package', `quelpa' and package.el to manage my plugins.
|
2017-01-16 23:15:48 -05:00
|
|
|
;;
|
2017-01-28 02:02:16 -05:00
|
|
|
;; Why all the trouble? Because:
|
|
|
|
;; 1. Scriptability: I want my plugins managable from the command line (as well
|
2017-01-31 19:42:41 -05:00
|
|
|
;; as an `doom/packages-update' command within emacs to update my plugins
|
|
|
|
;; automatically, rather than through package.el's interface).
|
2017-01-28 02:02:16 -05:00
|
|
|
;; 2. Flexibility: I want to install packages from sources other than ELPA
|
|
|
|
;; repositories. Such as github or the Emacs wiki. Some plugins are out of
|
|
|
|
;; date through official channels, have changed hands unofficially, or simply
|
|
|
|
;; haven't been submitted to an ELPA repo yet.
|
2017-02-01 00:29:39 -05:00
|
|
|
;; 3. Stability: I don't want to worry that each time I use my package
|
|
|
|
;; manager something might inexplicably go wrong. This was the case with
|
|
|
|
;; Cask, which I used previously. package.el and quelpa appear to be much
|
2017-02-09 04:25:32 -05:00
|
|
|
;; faster and more stable.
|
2017-01-28 02:02:16 -05:00
|
|
|
;; 4. No external dependencies (e.g. Cask) for plugin management.
|
2017-02-06 00:13:24 -05:00
|
|
|
;;
|
2017-02-09 04:25:32 -05:00
|
|
|
;; Note: it should be safe to use *most* package.el functions directly, though I
|
|
|
|
;; wouldn't recommend you use `package-autoremove'. For complete certainty, I've provided safer DOOM vaiants:
|
|
|
|
;; `doom/install-package', `doom/delete-package' and `doom/update-packages'.
|
|
|
|
;;
|
|
|
|
;; As well as: `doom/packages-install', `doom/packages-update', and
|
|
|
|
;; `doom/packages-autoremove', which are called from the Makefile.
|
2017-02-06 00:13:24 -05:00
|
|
|
;;
|
|
|
|
;; See core/autoload/packages.el for more functions.
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-06 01:22:54 -05:00
|
|
|
(defvar doom-modules nil
|
2017-02-01 00:29:39 -05:00
|
|
|
"List of enabled modules; each element is a cons cell (MODULE . SUBMODULE),
|
|
|
|
where MODULE is the module's property symbol, e.g. :lang, and SUBMODULE is the
|
|
|
|
submodule symbol, e.g. 'evil.")
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defvar doom-packages nil
|
|
|
|
"A list of enabled packages.")
|
|
|
|
|
2017-02-03 07:58:16 -05:00
|
|
|
(defvar doom-protected-packages '(quelpa use-package dash f s)
|
2017-02-02 04:37:59 -05:00
|
|
|
"A list of packages that shouldn't be deleted.")
|
|
|
|
|
2017-02-03 07:58:16 -05:00
|
|
|
(defvar doom-init-p nil
|
2017-02-01 00:29:39 -05:00
|
|
|
"Non-nil if doom's package system has been initialized or not. It may not be
|
|
|
|
if you have byte-compiled your configuration (as intended).")
|
2017-01-31 04:31:14 -05:00
|
|
|
|
2017-02-03 07:58:16 -05:00
|
|
|
(defvar doom--base-load-path (append (list doom-core-dir
|
|
|
|
doom-modules-dir)
|
|
|
|
load-path)
|
2017-01-31 04:31:14 -05:00
|
|
|
"A backup of `load-path', used as a bare-bones foundation for
|
|
|
|
`doom/packages-reload' or `doom-initialize'.")
|
2017-01-16 23:15:48 -05:00
|
|
|
|
|
|
|
(setq load-prefer-newer nil
|
|
|
|
package--init-file-ensured t
|
|
|
|
package-user-dir (expand-file-name "elpa" doom-packages-dir)
|
|
|
|
package-enable-at-startup nil
|
|
|
|
package-archives
|
|
|
|
'(("gnu" . "http://elpa.gnu.org/packages/")
|
|
|
|
("melpa" . "http://melpa.org/packages/")
|
|
|
|
("org" . "http://orgmode.org/elpa/"))
|
|
|
|
|
|
|
|
use-package-always-defer t
|
2017-01-28 02:02:16 -05:00
|
|
|
use-package-always-ensure nil
|
2017-02-02 04:37:59 -05:00
|
|
|
use-package-expand-minimally t
|
2017-02-02 21:54:47 -05:00
|
|
|
use-package-debug nil
|
2017-01-31 04:31:14 -05:00
|
|
|
use-package-verbose doom-debug-mode
|
2017-02-06 01:23:24 -05:00
|
|
|
|
2017-01-16 23:15:48 -05:00
|
|
|
quelpa-checkout-melpa-p nil
|
|
|
|
quelpa-update-melpa-p nil
|
2017-01-31 18:59:58 -05:00
|
|
|
quelpa-dir (expand-file-name "quelpa" doom-packages-dir)
|
2017-02-06 01:23:24 -05:00
|
|
|
|
|
|
|
byte-compile-dynamic t
|
|
|
|
byte-compile-warnings '(not mapcar free-vars unresolved noruntime lexical make-local))
|
2017-01-16 23:15:48 -05:00
|
|
|
|
|
|
|
|
|
|
|
;;
|
2017-01-31 04:31:14 -05:00
|
|
|
;; Bootstrap function
|
2017-01-16 23:15:48 -05:00
|
|
|
;;
|
|
|
|
|
2017-02-04 02:55:05 -05:00
|
|
|
(autoload 'use-package "use-package" nil nil 'macro)
|
2017-02-06 00:13:24 -05:00
|
|
|
(advice-add 'package-delete :after 'doom*package-delete)
|
2017-02-04 02:55:05 -05:00
|
|
|
|
2017-02-09 04:22:08 -05:00
|
|
|
(defmacro @doom (&rest packages)
|
2017-01-31 18:47:59 -05:00
|
|
|
"DOOM Emacs bootstrap macro. List the modules to load. Benefits from
|
|
|
|
byte-compilation."
|
|
|
|
(let (mode)
|
|
|
|
(dolist (p packages)
|
2017-02-04 21:07:54 -05:00
|
|
|
(cond ((keywordp p)
|
2017-01-31 18:47:59 -05:00
|
|
|
(setq mode p))
|
|
|
|
((not mode)
|
2017-02-09 04:22:08 -05:00
|
|
|
(error "No namespace specified on `@doom' for %s" p))
|
2017-02-06 01:23:35 -05:00
|
|
|
((eq p '*)
|
|
|
|
(let ((mode-name (substring (symbol-name mode) 1)))
|
|
|
|
(--map (setq doom-modules (append doom-modules (list (cons mode (f-base it)))))
|
|
|
|
(f-directories (f-expand mode-name doom-modules-dir)))))
|
2017-01-31 18:47:59 -05:00
|
|
|
(t
|
2017-02-06 01:22:54 -05:00
|
|
|
(setq doom-modules (append doom-modules (list (cons mode p))))))))
|
|
|
|
(unless noninteractive
|
|
|
|
`(let (file-name-handler-alist)
|
2017-02-09 04:22:08 -05:00
|
|
|
,@(mapcar (lambda (pkg) `(@load ,(car pkg) ,(cdr pkg)))
|
2017-02-06 01:22:54 -05:00
|
|
|
doom-modules)
|
2017-02-04 21:07:54 -05:00
|
|
|
|
2017-02-06 00:13:24 -05:00
|
|
|
(when (display-graphic-p)
|
|
|
|
(require 'server)
|
|
|
|
(unless (server-running-p)
|
|
|
|
(server-start)))
|
2017-02-04 21:07:54 -05:00
|
|
|
|
2017-02-06 00:13:24 -05:00
|
|
|
;; Benchmark
|
|
|
|
(format "Loaded %s packages in %s"
|
|
|
|
(- (length load-path) (length doom--base-load-path))
|
|
|
|
(emacs-init-time)))))
|
2017-01-31 04:31:14 -05:00
|
|
|
|
|
|
|
(defun doom-initialize (&optional force-p)
|
2017-02-02 04:37:59 -05:00
|
|
|
"Initialize installed packages (using package.el) and ensure the core packages
|
|
|
|
are installed. If you byte compile core/core.el, calls to `package.el' are
|
|
|
|
avoided to speed up startup."
|
2017-02-03 07:58:16 -05:00
|
|
|
(unless (or doom-init-p force-p)
|
|
|
|
(setq load-path doom--base-load-path
|
2017-01-31 04:31:14 -05:00
|
|
|
package-activated-list nil)
|
2017-02-03 19:20:47 -05:00
|
|
|
(package-initialize t)
|
|
|
|
;; Sure, package-initialize fills the load-path, but it will error out on
|
|
|
|
;; missing packages. UNACCEPTAABBLLLE!
|
2017-02-06 00:13:24 -05:00
|
|
|
(setq load-path (append load-path (directory-files package-user-dir t "^[a-zA-Z0-9]" t)))
|
|
|
|
|
|
|
|
;; Ensure cache folder exists
|
|
|
|
(unless (file-exists-p doom-cache-dir)
|
|
|
|
(make-directory doom-cache-dir t))
|
2017-02-03 19:20:47 -05:00
|
|
|
|
2017-02-09 04:25:32 -05:00
|
|
|
;; Ensure core packages are installed
|
2017-02-02 04:37:59 -05:00
|
|
|
(unless (and (file-exists-p doom-packages-dir)
|
2017-02-03 07:58:16 -05:00
|
|
|
(require 'use-package nil t)
|
|
|
|
(require 'quelpa nil t))
|
2017-01-31 04:31:14 -05:00
|
|
|
(package-refresh-contents)
|
2017-02-03 07:58:16 -05:00
|
|
|
(condition-case ex
|
|
|
|
(mapc (lambda (pkg)
|
|
|
|
(package-install pkg)
|
|
|
|
(unless (package-installed-p pkg)
|
|
|
|
(error "Couldn't install %s" pkg)))
|
|
|
|
doom-protected-packages)
|
2017-02-03 20:10:03 -05:00
|
|
|
(error
|
2017-02-03 07:58:16 -05:00
|
|
|
(delete-directory doom-packages-dir t)
|
|
|
|
(error "There was an error initializing DOOM. Try running it again"))))
|
|
|
|
|
|
|
|
(require 'quelpa)
|
|
|
|
(require 'use-package)
|
|
|
|
;; Remove package management keywords, I'll deal with the myself
|
|
|
|
(mapc (lambda (keyword) (setq use-package-keywords (delq keyword use-package-keywords)))
|
|
|
|
'(:ensure :pin))
|
|
|
|
(setq doom-init-p t)))
|
|
|
|
|
2017-02-06 00:13:24 -05:00
|
|
|
(defun doom-initialize-autoloads (&optional force-p)
|
|
|
|
"Ensures that an autoloads file exists and is loaded."
|
|
|
|
(unless (or (featurep 'autoloads)
|
|
|
|
(load doom-autoload-file t t))
|
|
|
|
(doom/refresh-autoloads)
|
|
|
|
(unless (file-exists-p doom-autoload-file)
|
|
|
|
(error "Autoloads file couldn't be generated"))))
|
|
|
|
|
2017-01-31 04:31:14 -05:00
|
|
|
|
|
|
|
;;
|
|
|
|
;; Macros
|
|
|
|
;;
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-03 07:58:16 -05:00
|
|
|
(defvar __PACKAGE__ nil "The name of the current package.")
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-09 04:22:08 -05:00
|
|
|
(defalias '@use-package 'use-package
|
2017-02-06 00:13:24 -05:00
|
|
|
"A `use-package' alias. It exists so DOOM configs adhere to the naming
|
|
|
|
conventions of DOOM emacs. Note that packages are deferred by default.
|
|
|
|
|
2017-02-09 04:22:08 -05:00
|
|
|
By DOOM conventions, using this instead of `@package' means you are configuring
|
|
|
|
a package regardless of whether it's installed or not, while `@package' is used
|
2017-02-06 00:13:24 -05:00
|
|
|
to declare how to install/setup a package.")
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-09 04:22:08 -05:00
|
|
|
(defmacro @package (name &rest plist)
|
2017-02-06 00:13:24 -05:00
|
|
|
"Declares a package. This does not load nor install them explicitly.
|
2017-02-04 21:07:54 -05:00
|
|
|
|
2017-02-09 04:22:08 -05:00
|
|
|
If used in `doom-core-dir', this is a wrapper for `@use-package' (all packages
|
2017-02-04 21:07:54 -05:00
|
|
|
are deferred by default), and takes the same arguments as `use-package'.
|
|
|
|
|
2017-02-06 00:13:24 -05:00
|
|
|
If used outside of `doom-core-dir' (i.e. in packages.el files within modules),
|
2017-02-09 04:22:08 -05:00
|
|
|
this macro serves a purely declarative purpose and doesn't call `@use-package'.
|
2017-02-06 00:13:24 -05:00
|
|
|
These calls are parsed by `doom-read-packages' to build `doom-packages'.
|
2017-02-04 21:07:54 -05:00
|
|
|
|
|
|
|
Adds a few custom properties in either case:
|
2017-02-03 20:10:40 -05:00
|
|
|
|
|
|
|
:recipe RECIPE Takes a MELPA-style recipe (see `quelpa-recipe' for an
|
|
|
|
example); for packages to be installed from external
|
|
|
|
sources.
|
|
|
|
:pin ARCHIVE-NAME Instructs ELPA to only look for this package in
|
|
|
|
ARCHIVE-NAME. e.g. \"org\".
|
2017-02-04 21:07:54 -05:00
|
|
|
:needs FEATURE Don't install this package if FEATURE isn't available. Can be a
|
2017-02-06 01:24:00 -05:00
|
|
|
(:module . submodule) cons pair."
|
2017-01-16 23:15:48 -05:00
|
|
|
(declare (indent defun))
|
2017-02-04 21:07:54 -05:00
|
|
|
(mapc (lambda (key) (setq plist (use-package-plist-delete plist key)))
|
2017-02-06 01:24:00 -05:00
|
|
|
'(:recipe :pin :needs))
|
2017-02-06 00:13:24 -05:00
|
|
|
`(let ((__PACKAGE__ ',name))
|
2017-02-09 04:22:08 -05:00
|
|
|
(@use-package ,name ,@plist)))
|
2017-02-02 21:55:25 -05:00
|
|
|
|
2017-02-09 04:22:08 -05:00
|
|
|
(defmacro @load (module &optional submodule file)
|
2017-02-03 19:20:47 -05:00
|
|
|
"Load a module from `doom-modules-dir' when both MODULE and SUBMODULE is
|
|
|
|
provided (both symbols). If FILE is non-nil, append it to the resulting path. If
|
|
|
|
SUBMODULE is nil, MODULE is loaded relative to the current file (see `__DIR__').
|
|
|
|
When SUBMODULE is nil, FILE isn't used.
|
|
|
|
|
2017-01-28 02:02:16 -05:00
|
|
|
Examples:
|
2017-02-09 04:22:08 -05:00
|
|
|
(@load :lang emacs-lisp)
|
2017-02-01 00:29:39 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
Loads modules/lang/emacs-lisp/FILE.el (defaults to config.el).
|
2017-02-01 00:29:39 -05:00
|
|
|
|
2017-02-09 04:22:08 -05:00
|
|
|
(@load +local-module)
|
2017-01-31 04:31:14 -05:00
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
Loads +local-module.el relative to `__DIR__' or `doom-core-dir'."
|
2017-02-04 21:07:54 -05:00
|
|
|
(let (path file)
|
2017-01-31 04:31:14 -05:00
|
|
|
(cond ((null submodule)
|
2017-02-02 21:56:08 -05:00
|
|
|
(setq path __DIR__
|
2017-02-03 19:20:47 -05:00
|
|
|
file (concat (symbol-name module) ".el")))
|
2017-01-31 04:31:14 -05:00
|
|
|
(t
|
2017-02-03 19:20:47 -05:00
|
|
|
(cl-pushnew (cons module submodule)
|
2017-02-06 01:22:54 -05:00
|
|
|
doom-modules
|
2017-02-03 19:20:47 -05:00
|
|
|
:test (lambda (x y) (and (eq (car x) (car y))
|
|
|
|
(eq (cdr x) (cdr y)))))
|
|
|
|
(setq path (doom-module-path module submodule)
|
2017-02-02 04:37:59 -05:00
|
|
|
file (or file "config.el"))))
|
2017-01-31 04:31:14 -05:00
|
|
|
(setq path (f-slash path)
|
|
|
|
file (concat path file))
|
2017-02-02 04:37:59 -05:00
|
|
|
`(let ((__FILE__ ,file)
|
|
|
|
(__DIR__ ,path))
|
2017-02-04 21:07:54 -05:00
|
|
|
(load ,(f-no-ext file) nil (not doom-debug-mode)))))
|
2017-01-28 02:02:16 -05:00
|
|
|
|
2017-01-31 19:42:11 -05:00
|
|
|
(defun doom-module-path (module submodule &optional file)
|
2017-02-02 04:37:59 -05:00
|
|
|
"Get the full path to a module: e.g. :lang emacs-lisp maps to
|
2017-02-06 00:13:24 -05:00
|
|
|
~/.emacs.d/modules/lang/emacs-lisp/ and will append FILE if non-nil."
|
2017-02-02 19:17:02 -05:00
|
|
|
(setq module
|
|
|
|
(cond ((keywordp module) (substring (symbol-name module) 1))
|
2017-02-03 07:58:16 -05:00
|
|
|
((symbolp module) (symbol-name module))
|
2017-02-02 19:17:02 -05:00
|
|
|
((stringp module) module)
|
|
|
|
(t (error "Not a valid module name: %s" module))))
|
2017-02-02 04:37:59 -05:00
|
|
|
(when (symbolp submodule)
|
|
|
|
(setq submodule (symbol-name submodule)))
|
|
|
|
(f-expand (concat module "/" submodule "/" file)
|
2017-01-31 19:42:11 -05:00
|
|
|
doom-modules-dir))
|
|
|
|
|
2017-01-31 04:31:14 -05:00
|
|
|
|
|
|
|
;;
|
2017-02-03 07:58:16 -05:00
|
|
|
;; Commands
|
2017-01-31 04:31:14 -05:00
|
|
|
;;
|
|
|
|
|
2017-02-03 07:58:16 -05:00
|
|
|
(defun doom/reload ()
|
2017-02-06 01:22:54 -05:00
|
|
|
"Reload `load-path', `doom-modules' and `doom-packages' by
|
2017-02-09 04:22:08 -05:00
|
|
|
reinitializing doom and parsing config files for `@package' and `@doom' calls.
|
2017-02-06 00:12:44 -05:00
|
|
|
There are few reasons to use this."
|
2017-02-03 07:58:16 -05:00
|
|
|
(interactive)
|
|
|
|
(doom-initialize t)
|
2017-02-06 00:12:44 -05:00
|
|
|
(doom-read-packages t)
|
|
|
|
(doom-initialize-autoloads)
|
2017-02-03 07:58:16 -05:00
|
|
|
(message "Reloaded %s packages" (length package-alist)))
|
|
|
|
|
2017-02-02 04:37:59 -05:00
|
|
|
(defun doom/refresh-autoloads ()
|
|
|
|
"Refreshes the autoloads.el file, which tells Emacs where to find all the
|
2017-02-06 00:12:44 -05:00
|
|
|
autoloaded functions in the modules you use or among the core libraries, e.g.
|
|
|
|
core/autoload/*.el.
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-06 00:12:44 -05:00
|
|
|
In modules, checks modules/*/autoload.el and modules/*/autoload/*.el.
|
2017-02-03 19:20:47 -05:00
|
|
|
|
2017-02-06 00:12:44 -05:00
|
|
|
Rerun this whenever init.el is modified. You can also use `make autoloads` from
|
|
|
|
the commandline."
|
2017-01-16 23:15:48 -05:00
|
|
|
(interactive)
|
2017-02-06 00:12:44 -05:00
|
|
|
(let ((generated-autoload-file doom-autoload-file)
|
|
|
|
autoload-files)
|
|
|
|
(setq autoload-files
|
|
|
|
(append (-flatten (--map (let ((auto-dir (f-expand "autoload" it))
|
|
|
|
(auto-file (f-expand "autoload.el" it)))
|
2017-02-08 19:58:12 -05:00
|
|
|
(append (and (f-exists-p auto-file)
|
|
|
|
(list auto-file))
|
|
|
|
(and (f-directory-p auto-dir)
|
|
|
|
(f-glob "*.el" auto-dir))))
|
2017-02-04 21:07:54 -05:00
|
|
|
(--map (doom-module-path (car it) (cdr it))
|
2017-02-06 01:22:54 -05:00
|
|
|
doom-modules)))
|
2017-02-06 00:12:44 -05:00
|
|
|
(f-glob "autoload/*.el" doom-core-dir)))
|
2017-02-04 21:07:54 -05:00
|
|
|
(when (f-exists-p generated-autoload-file)
|
|
|
|
(f-delete generated-autoload-file)
|
|
|
|
(message "Deleted old autoloads.el"))
|
|
|
|
(dolist (file autoload-files)
|
|
|
|
(update-file-autoloads file)
|
|
|
|
(message "Scanned %s" (f-relative file doom-emacs-dir)))
|
|
|
|
(with-current-buffer (get-file-buffer generated-autoload-file)
|
|
|
|
(save-buffer)
|
|
|
|
(eval-buffer))
|
|
|
|
(message "Done!")))
|
2017-01-16 23:15:48 -05:00
|
|
|
|
2017-02-06 00:12:44 -05:00
|
|
|
(defun doom/byte-compile (&optional simple-p)
|
|
|
|
"Byte (re)compile the important files in your emacs configuration (init.el &
|
|
|
|
core/*.el). DOOM Emacs was designed to benefit from this.
|
2017-02-02 21:56:40 -05:00
|
|
|
|
2017-02-06 00:12:44 -05:00
|
|
|
If SIMPLE-P is nil, also byte-compile modules/*/*/*.el (except for packages.el).
|
|
|
|
There should be a measurable benefit from this, but it may take a while."
|
2017-01-16 23:15:48 -05:00
|
|
|
(interactive)
|
2017-02-06 00:12:44 -05:00
|
|
|
(let ((targets
|
|
|
|
(append (list (f-expand "init.el" doom-emacs-dir)
|
|
|
|
(f-expand "core.el" doom-core-dir))
|
|
|
|
(f-glob "core-*.el" doom-core-dir)
|
|
|
|
(unless simple-p
|
|
|
|
(-flatten
|
|
|
|
(--map (f--entries (doom-module-path (car it) (cdr it))
|
|
|
|
(and (f-ext-p it "el")
|
|
|
|
(or (string= (f-base it) "config")
|
|
|
|
(string-prefix-p "+" (f-base it))))
|
|
|
|
t)
|
2017-02-06 01:22:54 -05:00
|
|
|
doom-modules)))))
|
2017-02-03 07:58:16 -05:00
|
|
|
(n 0)
|
|
|
|
results)
|
2017-02-04 21:07:54 -05:00
|
|
|
(dolist (file targets)
|
|
|
|
(push (cons (f-relative file doom-emacs-dir)
|
|
|
|
(when (byte-recompile-file file nil 0)
|
|
|
|
(setq n (1+ n))
|
|
|
|
t))
|
|
|
|
results))
|
2017-01-31 19:44:31 -05:00
|
|
|
(when noninteractive
|
|
|
|
(when targets (message "\n"))
|
2017-02-04 21:07:54 -05:00
|
|
|
(message "Compiled %s files:\n%s" n
|
2017-02-02 04:37:59 -05:00
|
|
|
(mapconcat (lambda (file) (concat "+ " (if (cdr file) "SUCCESS" "FAIL") ": " (car file)))
|
|
|
|
(reverse results) "\n")))))
|
2017-01-16 23:15:48 -05:00
|
|
|
|
|
|
|
(provide 'core-packages)
|
|
|
|
;;; core-packages.el ends here
|