I liked my piper idea, but I just used it so seldom. Instead, I feel like the ideas could be integrated into a data-focused function collection. The interface is actually more dynamic and I can use it without the "Piper" interface.
13 KiB
My Emacs Bootstrap
A literate programming file for bootstraping my Emacs Configuration.
Introduction
This file contains all the variable definitions and library loading for the other files in my project.
Straight Package Installer
I'm going to be installing everything using the straight.el for package installation and management. Before I can tangle these files, I need straight
to grab the latest org
, so the following initialization code is actually in initialize, but if you are reading this online, configuring straight
amounts to the following:
(defvar bootstrap-version)
(let ((bootstrap-file
(expand-file-name "straight/repos/straight.el/bootstrap.el" user-emacs-directory))
(bootstrap-version 5))
(unless (file-exists-p bootstrap-file)
(with-current-buffer
(url-retrieve-synchronously
"https://raw.githubusercontent.com/raxod502/straight.el/develop/install.el"
'silent 'inhibit-cookies)
(goto-char (point-max))
(eval-print-last-sexp)))
(load bootstrap-file nil 'nomessage))
To get the Straight project working with use-package
:
(straight-use-package 'use-package)
While the above code enables the :straight t
extension to use-package
, let's have that as the default:
(use-package straight
:custom (straight-use-package-by-default t
straight-default-vc 'git))
See the details in this essay.
OS Path and Native Compilation
Helper functions to allow code for specific operating systems:
(defun ha-running-on-macos? ()
"Return non-nil if running on Mac OS systems."
(equal system-type 'darwin))
(defun ha-running-on-linux? ()
"Return non-nil if running on Linux systems."
(equal system-type 'gnu/linux))
With the way I start Emacs, I may not have the PATH
I actually use (from the shell) available, so we'll force it (code taken from here):
(defun set-exec-path-from-shell ()
"Set up Emacs' `exec-path' and PATH environment variable to match
that used by the user's shell.
The MacOS, where GUI apps are not started from a shell, requires this."
(interactive)
(let* ((path-from-shell (shell-command-to-string "echo $PATH"))
(trimmed-path (replace-regexp-in-string (rx (zero-or-more space) eol)
"" path-from-shell))
(in-fish? (string-match (rx "fish" eol)
(shell-command-to-string "echo $SHELL")))
(separator (if in-fish? " " ":"))
(env-path (if in-fish? (replace-regexp-in-string " " ":" trimmed-path) trimmed-path)))
(message "PATH=%s" path-from-shell)
(setenv "PATH" env-path)
(setq exec-path (split-string trimmed-path separator))))
Clear up a Mac-specific issue that sometimes arises since I'm switching to native compilation project, as the Emacs.app
that I use doesn't have its bin
directory, e.g. Emacs.app/Contents/MacOS/bin
:
(when (ha-running-on-macos?)
(add-to-list 'exec-path "/usr/local/bin")
(add-to-list 'exec-path (concat invocation-directory "bin") t))
Getting tired off all the packages that I load spewing a bunch of warnings that I can't do anything about:
(when (and (fboundp 'native-comp-available-p)
(native-comp-available-p))
(setq native-comp-async-report-warnings-errors nil
native-comp-deferred-compilation t))
GNU Pretty Good Privacy
On Linux, GPG is pretty straight-forward, but on the Mac, I often have troubles doing:
brew install gpg
Next, on every reboot, start the agent:
/usr/local/Cellar/gnupg/2.3.6/bin/gpg-agent --daemon
Since brew link gpg
doesn’t always work, this helper function may find the executable:
(defun executable (path)
"Return PATH if executable, see `file-executable-p'."
(let ((epath (first (file-expand-wildcards path))))
(when (file-executable-p epath) epath)))
(use-package epa-file
:straight (:type built-in)
:custom (epg-gpg-program (or (executable "/usr/local/Cellar/gnupg/*/bin/gpg")
(executable "/usr/local/bin/gpg")
(executable "/usr/local/opt/gpg")
(executable "/usr/bin/pgp")))
:config (epa-file-enable))
Basic Libraries
The following packages come with Emacs, but seems like they still need loading:
(use-package cl-lib
:straight (:type built-in)
:init (defun first (elt) (car elt))
:commands (first))
(require 'subr-x)
Ugh. Why am I getting a missing first
function error? I define a simple implementation, that the CL library will overwrite … at some point.
While most libraries will take care of their dependencies, I want to install my dependent libraries, e.g, Magnar Sveen's Clojure-inspired dash.el project:
(use-package dash)
Sure this package is essentially syntactic sugar, and to help share my configuration, I attempt to use thread-last
instead of ->>
, but, I still like it.
The s.el project is a simpler string manipulation library that I (and other projects) use:
(use-package s)
Manipulate file paths with the f.el project:
(use-package f)
The shell-command function is useful, but having it split the output into a list is a helpful abstraction:
(defun shell-command-to-list (command)
"Return list of lines from running COMMAND in shell."
(thread-last command
shell-command-to-string
s-lines
(-map 's-trim)
(-remove 's-blank-str?)))
And let’s see the results:
(ert-deftest shell-command-to-list-test ()
(should (equal '("hello world")
(shell-command-to-list "echo hello world")))
;; We don't need blank lines:
(should (equal '("hello world" "goodbye for now")
(shell-command-to-list "echo '\n\nhello world\n\ngoodbye for now\n\n'"))
;; No output? Return null:
(should (null (shell-command-to-list "echo")))
;; No line should contain carriage returns:
(should (null (seq-filter
(lambda (line) (s-contains? "\n" line))
(shell-command-to-list "ls")))))
My Code Location
Much of my more complicated code comes from my website essays and other projects. The destination shows up here:
(add-to-list 'load-path (f-expand "~/.emacs.d/elisp"))
Hopefully, this will tie me over while I transition.
Emacs Server Control
I actually run two instances of Emacs on some systems, where one instance has all my work-related projects, perspectives, and packages installed (like LSP), and my personal instance has other packages running (like IRC and Mail). I need a function that can make that distinction, and based on that, it will set server-start
appropriately, so that emacsclient
can call into the correct one.
(defun ha-emacs-for-work? ()
"Return non-nil when the Emacs application's location matches as one for work.
Based on initially running the app with a `FOR_WORK' environment variable."
(and (f-dir? "~/work")
(getenv "FOR_WORK")))
And now start the server with an appropriate tag name:
(if (not (ha-emacs-for-work?))
(setq server-name "personal")
(setq server-name "work")
(when (ha-running-on-macos?)
(set-exec-path-from-shell)))
(server-start)
Load the Rest
The following loads the rest of my org-mode literate files. I add new filesas they are ready:
(defvar ha-hamacs-files (flatten-list
`("ha-private.org"
"ha-config.org"
,(when (display-graphic-p)
"ha-display.org")
"ha-org.org"
,(when (display-graphic-p)
"ha-org-word-processor.org")
"ha-org-clipboard.org"
"ha-capturing-notes.org"
"ha-agendas.org"
"ha-data.org"
"ha-passwords.org"
"ha-eshell.org"
"ha-remoting.org"
"ha-programming.org"
"ha-programming-elisp.org"
"ha-programming-python.org"
,(if (ha-emacs-for-work?)
'("ha-org-sprint.org" "ha-work.org")
;; Personal Editor
'("ha-org-journaling.org"
"ha-irc.org"
"ha-org-publishing.org"
"ha-email.org"
"ha-aux-apps.org"
"ha-feed-reader.org"))))
"List of org files that complete the hamacs project.")
The list of hamacs org-formatted files stored in ha-hamacs-files
is selectively short, and doesn’t include all files, for instance, certain languages that I’m learning aren’t automatically included. The function, ha-hamacs-files
will return the list loaded at startup, as well as with an optional parameter, return them all.
(defun ha-hamacs-files (&optional all)
"Return a list of my org files in my `hamacs' directory."
(if (not all)
ha-hamacs-files
(thread-last (rx ".org" string-end)
(directory-files "~/other/hamacs" nil)
(append ha-hamacs-files)
(--filter (not (string-match (rx "README") it)))
(-uniq))))
With this function, we can test/debug/reload any individual file, via:
(defun ha-hamacs-load (file)
"Load or reload an org-mode FILE containing literate Emacs configuration code."
(interactive (list (completing-read "Org file: " (ha-hamacs-files :all))))
(let ((full-file (f-join hamacs-source-dir file)))
(when (f-exists? full-file)
(org-babel-load-file full-file))))
And we can now reload all startup files:
(defun ha-hamacs-reload-all ()
"Reload our entire ecosystem of configuration files."
(interactive)
(dolist (file (ha-hamacs-files))
(unless (equal file "bootstrap.org")
(ha-hamacs-load file))))
And do it:
(ha-hamacs-reload-all)