My attempt at making a gnarly banner function more readable by creating lexical functions with `cl-flet` to pass to `mapconcat` was thwarted! Who would have thunk that `mapconcat' couldn't access a symbol reference to a function created by `cl-flet*'!?
40 KiB
Expanding Eshell
A literate programming file for configuring the Emacs Shell.
Introduction
While I like vterm for logging into remote systems, I find Emacs’ shell, eshell
, an interesting alternative.
If you find the documentation lacking, I documented most features, and you might find the following helpful.
Navigation and Keys
Along with the regular Emacs keybindings, Eshell comes with some interesting features:
M-RET
gives you a prompt, even when you are running another command. Sinceeshell
passes all input to subprocesses, there is no automatic input queueing as there is with other shells.C-c C-t
truncates the buffer if it grows too large.C-c C-r
will move point to the beginning of the output of the last command. With a prefix argument,eshell
narrows to view its output.C-c C-o
will delete the output from the last command.C-c C-f
will move forward a complete shell argument.C-c C-b
will move backward a complete shell argument.
Control-D Double Duty
Used to C-d
exiting from a shell? Want it to keep working, but still allow deleting a character? We can have it both (thanks to wasamasa):
(defun ha-eshell-quit-or-delete-char (arg)
"The `C-d' sequence closes window or deletes a character."
(interactive "p")
(if (and (eolp) (looking-back eshell-prompt-regexp))
(progn
(eshell-life-is-too-much) ; Why not? (eshell/exit)
(ignore-errors
(delete-window)))
(delete-forward-char arg)))
Pager Setup
If any program wants to pause the output through the $PAGER
variable, well, we don't need that:
(setenv "PAGER" "cat")
Aliases
Gotta have some shell aliases, right? We have three ways of doing that. First, enter them into an eshell
session:
alias less 'view-file $1'
Note that you need single quotes, and more than one argument doesn’t work with aliases. To resolve that, we need to write a function.
Second, you can create/populate the alias file, ~/.emacs.d/eshell/alias
… as long as you don’t use those single quotes:
alias ll ls -l $*
alias clear recenter 0
alias d dired $1
alias e find-file $1
alias less view-file $1
alias more view-file $1
alias find echo 'Please use fd instead.'
Which happens when you type those commands into an eshell
.
Third, you want control, write a function to define the aliases:
(defun ha-eshell-add-aliases ()
"Call `eshell/alias' to define my aliases."
(eshell/alias "e" "find-file $1")
(eshell/alias "d" "dired $1")
(eshell/alias "gd" "magit-diff-unstaged")
(eshell/alias "gds" "magit-diff-staged")
;; The 'ls' executable requires the Gnu version on the Mac
(let ((ls (if (file-exists-p "/usr/local/bin/gls")
"/usr/local/bin/gls"
"/bin/ls")))
(eshell/alias "ll" (concat ls " -AlohG --color=always"))))
Predicate Filters and Modifiers
The T
predicate filter allows me to limit file results that have internal org-mode
tags. For instance, eshell
will send files that have a #+TAGS:
header with a mac
label to the grep
function:
$ grep brew *.org(T'mac')
As described in this essay, to extend Eshell, we need a two-part function:
- Parse the Eshell buffer to look for the parameter (and move the point past the parameter).
- A predicate function that takes a file as a parameter.
For the first step, we have our function called as it helps parse the text. Based on what it sees, it returns the predicate function used to filter the files:
(defun eshell-org-file-tags ()
"Parse the eshell text at point.
Looks for parameters surrounded in single quotes. Returns a
function that takes a FILE and returns nil if the file given to
it doesn't contain the org-mode #+TAGS: entry specified."
;; Step 1. Parse the eshell buffer for our tag between quotes
;; Make sure to move point to the end of the match:
(if (looking-at (rx "'" (group (one-or-more (not (or ")" "'"))))"'"))
(let* ((tag (match-string 1))
(reg (rx line-start
"#+" (optional "file") "tags:"
(one-or-more space)
(zero-or-more any)
(literal tag) word-end)))
(goto-char (match-end 0))
;; Step 2. Return the predicate function:
;; Careful when accessing the `reg' variable.
`(lambda (file)
(with-temp-buffer
(insert-file-contents file)
(re-search-forward ,reg nil t 1))))
(error "The `T' predicate takes an org-mode tag value in single quotes.")))
Then we need add that function to the eshell-predicate-alist
as the T
tag:
(defun ha-eshell-add-predicates ()
"A hook to add a `eshell-org-file-tags' predicate filter to eshell."
(add-to-list 'eshell-predicate-alist '(?T . (eshell-org-file-tags))))
Note: We can’t add it to the list until after we start our first eshell session, so we add it to the eshell-pred-load-hook
.
Eshell Functions
Any function that begins with eshell/
is available as a command (with the remaining letters) Once I had a function eshell/f
as a replacement for find
, but the fd project is better.
Since eshell
is an Emacs shell, I try to think how to use Emacs buffers in a shell-focused workflow. For instance, use view-file
instead of less
, as it will show a file with syntax coloring, and typing q
returns to your shell session.
Buffer Cat
Why not be able to read a buffer and use it as the start of a pipeline?
(defun eshell/bcat (&rest args)
(mapconcat (lambda (buffer-name)
(when (bufferp buffer-name)
(save-window-excursion
(switch-to-buffer buffer-name)
(buffer-substring-no-properties (point-min) (point-max)))))
args "\n"))
Perhaps we should add this feature to eshell’s version of cat.
Piper
My piper project seems to like a good match with eshell. For instance, typing piper
in eshell with a file or a command, and the output from that goes into a Piper buffer, where standard Emacs commands can filter, sort or otherwise alter that output. Then, closing it and calling piper
in eshell without arguments outputs that buffer … to use as part of a pipe or something.
Map
While I like eshell’s for
loop well enough (if I can remember the syntax), as in:
for file in *.org {
chmod a+x $file
}
I like the idea of using a map structure, as in:
map chmod a+x *.org
How would this work without special syntax? Well, eshell sends the *.org
as a list of files, which we could use as the delimiter. The downside is that we want to list the files, we need to actually list the files, as in:
map chmod a+x (list "a.org" "c.org")
Pretty ugly, but that isn’t my use-case. I could introduce syntax like:
map chmod a+x :: *.org b.txt
But what if the file isn’t the last element? Well, I could replace a keyword, _
, with the filename when encountered.
Here is my initial function. After separating the arguments into two groups (split on the ::
string), we iterate over the file elements, creating a form that includes the filename.
(defun eshell/map (&rest args)
"Execute a command sequence over a collection of file elements.
Separate the sequence and the elements with a `::' string.
For instance:
map chmod a+x _ :: *.org b.txt
The function substitutes the `_' sequence a single filename element,
and if not specified, it appends the file name to the command."
(seq-let (forms elements) (--split-when (equal it "::") args)
(dolist (element (-flatten (-concat elements)))
(let* ((form (if (-contains? forms "_")
(-replace "_" element forms)
(-snoc forms element)))
(cmd (car form))
(args (cdr form)))
(eshell-named-command cmd args)))))
The eshell-named-command takes the command separately from the arguments, so we use car
and cdr
on the form.
Git
I used to have a number g
-prefixed aliases to call git-related commands, but now, I call Magit instead. My gst
command is an alias to magit-status
, but using the alias
doesn't pull in the current working directory, so I make it a function, instead:
(defun eshell/gst (&rest args)
(magit-status (pop args) nil)
(eshell/echo)) ;; The echo command suppresses output
Editing Files
The e
is an alias to find-file (which takes one argument), we define a special function to open each argument in a different window. We define a helper function for dealing with more than one argument. It takes two functions, where we call the first function on the first argument, and call the second function on each of the rest.
(defun eshell-fn-on-files (fun1 fun2 args)
(unless (null args)
(let ((filenames (thread-last args
(reverse)
(-flatten)
(-map 'file-expand-wildcards)
(-flatten))))
(apply fun1 (car filenames))
(when (cdr filenames)
(-map fun2 (cdr filenames))))))
This allows us to replace some of our aliases with functions:
(defun eshell/e (&rest args)
"Edit one or more files in current window."
(eshell-fn-on-files 'find-file 'find-file-other-window args))
(defun eshell/ee (&rest args)
"Edit one or more files in another window."
(eshell-fn-on-files 'find-file-other-window 'find-file-other-window args))
We’ll leave the e
alias to replace the eshell
buffer window.
Last Results
The shell-underscore project looks pretty cool, where the _
character represents a filename with the contents of the previous command (you know, like if you were planning on it, you’d tee
at the end of every command). An interesting idea that I could duplicate.
While diving into the eshell
source code, I noticed the special variables, $$
and $_
sometimes contains the output of the last command. For instance:
$ echo "hello world" hello world $ echo $$ hello world
What I would like is something like this to work:
$ ls *.org(U) a.org b.org f.org $ rg "foobar" $$
The problem may be between calling Emacs functions versus external commands, as the echo
works, but the call to ls
doesn’t:
$ ls *.org(U) b.txt a.org b.org f.org b.txt $ echo Nam $$ ("Nam" nil)
I over-write that special variables to behave as expected:
- A hook runs after every command
- It copies the previous command’s output to a ring (so that I can get the last as well as the fifth one)
- Create a replacement function for
$$
to read from my history ring
Let’s first make a ring that stores the output:
(defvar ha-eshell-output (make-ring 10)
"A ring (looped list) storing history of eshell command output.")
The following function does the work of saving the output of the last command. We can get this because after every command, eshell updates two variables, eshell-last-input-end (the start of the output), and eshell-last-output-start (the end of the output):
(defun ha-eshell-store-last-output ()
"Store the output from the last eshell command.
Called after every command by connecting to the `eshell-post-command-hook'."
(let ((output
(buffer-substring-no-properties eshell-last-input-end eshell-last-output-start)))
(ring-insert ha-eshell-output output)))
Now we save this output after every command by adding it to the eshell-post-command-hook:
(add-hook 'eshell-post-command-hook 'ha-eshell-store-last-output)
Next, this function returns values from the history ring. I feel the need to have different ways of returning the output data.
Unlike the behavior of the original shell (and most of its descendents, like bash
), eshell
doesn’t automatically split on whitespace. For instance, echo
called this way:
$ echo a b *.txt ("a" "b" ("b.txt" "date today.txt"))
Given a list of three elements: a
, b
, and a list of all files in the current directory with an .org
extension. An interesting side-effect is that spaces in filenames are often okay. If I specify and argument of text
, it should return the command’s output as a string, but if I give it, list
, it should contain the same information, but separated by spaces, into a list. For instance, if we are passing the output from ls
to grep
, we would use this format.
Like the shell-underscore
project mentioned earlier, I can access the output stored from a file when given a file
argument (the output will hold this temporary filename).
(defun eshell/output (&rest args)
"Return an eshell command output from its history.
The first argument is the index into the historical past, where
`0' is the most recent, `1' is the next oldest, etc.
The second argument represents the returned output:
,* `text' :: as a string
,* `list' :: as a list of elements separated by whitespace
,* `file' :: as a filename that contains the output
If the first argument is not a number, it assumes the format
to be `:text'.
"
(let (frmt element)
(cond
((> (length args) 1) (setq frmt (cadr args)
element (car args)))
((= (length args) 0) (setq frmt "text"
element 0))
((numberp (car args)) (setq frmt "text"
element (car args)))
((= (length args) 1) (setq frmt (car args)
element 0)))
(if-let ((results (ring-ref ha-eshell-output (or element 0))))
(cl-case (string-to-char frmt)
(?l (split-string results))
(?f (ha-eshell-store-file-output results))
(otherwise (s-trim results)))
"")))
(defun ha-eshell-store-file-output (results)
"Writes the string, RESULTS, to a temporary file and returns that file name."
(let ((filename (make-temp-file "ha-eshell-")))
(with-temp-file filename
(insert results))
filename))
How would this function work in practice?
$ ls a.org b.txt c.org date today.txt ever $ output a.org b.txt c.org date today.txt ever $ echo { output list } ("a.org" "b.txt" "c.org" "date" "today.txt" "ever")
Notice how commands between { … }
are eshell
commands, otherwise, if I replace the braces with parens, I would have to write eshell/output
. Let’s try the history feature:
$ echo "oldest" oldest $ echo "old" old $ echo "recent" recent $ echo "newest" newest $ echo { output 2 } old
Eshell has a feature where special variables (stored in eshell-variable-aliases-list), can be a function. The $$
holds text-formatted output, and $_
contains list-formatted output, and $OUTPUT
can be the output stored in a file.
(with-eval-after-load "eshell"
(defvar eshell-variable-aliases-list nil "Autoloading this eshell-defined variable")
(add-to-list 'eshell-variable-aliases-list '("$" ha-eshell-output-text))
(add-to-list 'eshell-variable-aliases-list '("_" ha-eshell-output-list))
(add-to-list 'eshell-variable-aliases-list '("OUTPUT" ha-eshell-output-file)))
Without this change, the $$
variable calls eshell-last-command-result, where I believe my version (with history) may work more reliably. I define these helper functions:
(defun ha-eshell-output (format-type indices)
"Wrapper around `eshell/output' for the `eshell-variable-aliases-list'."
(if indices
(eshell/output (string-to-number (caar indices)) format-type)
(eshell/output 0 format-type)))
(defun ha-eshell-output-text (&optional indices &rest ignored)
"A _text_ wrapper around `eshell/output' for the `eshell-variable-aliases-list'."
(ha-eshell-output "text" indices))
(defun ha-eshell-output-list (&optional indices &rest ignored)
"A _list_ wrapper around `eshell/output' for the `eshell-variable-aliases-list'."
(ha-eshell-output "list" indices))
(defun ha-eshell-output-file (&optional indices &rest ignored)
"A _file_ wrapper around `eshell/output' for the `eshell-variable-aliases-list'."
(ha-eshell-output "file" indices))
How would this look? Something like:
$ echo a a $ echo b b $ echo c c $ echo $$ c $ echo $$[2] b
The final trick is being able to count backwards and remember they are always shifting. I guess if I wanted to remember the output for more than one command, I could do:
$ ls *.org(U) b.txt a.org b.txt $ chmod o+w $_ $ rg Nam $_[1] a.org 8:Nam vestibulum accumsan nisl. b.txt 1:Nam euismod tellus id erat. 7:Name three animals that start with C
Wanna see something cool about Eshell? Let’s swirl Lisp and Shell commands:
$ rg (rx line-start "Nam ") $_[2] b.txt 1:Nam euismod tellus id erat. a.org 8:Nam vestibulum accumsan nisl.
Special Prompt
Following these instructions, we build a better prompt with the Git branch in it (Of course, it matches my Bash prompt). First, we need a function that returns a string with the Git branch in it, e.g. ":master"
(defun curr-dir-git-branch-string (pwd)
"Returns current git branch as a string, or the empty string if
PWD is not in a git repo (or the git command is not found)."
(interactive)
(when (and (not (file-remote-p pwd))
(eshell-search-path "git")
(locate-dominating-file pwd ".git"))
(let* ((git-url (shell-command-to-string "git config --get remote.origin.url"))
(git-repo (file-name-base (s-trim git-url)))
(git-output (shell-command-to-string (concat "git rev-parse --abbrev-ref HEAD")))
(git-branch (s-trim git-output))
(git-icon "\xe0a0")
(git-icon2 (propertize "\xf020" 'face `(:family "octicons"))))
(concat git-repo " " git-icon2 " " git-branch))))
The function takes the current directory passed in via pwd
and replaces the $HOME
part with a tilde. I'm sure this function already exists in the eshell source, but I didn't find it…
(defun pwd-replace-home (pwd)
"Replace home in PWD with tilde (~) character."
(interactive)
(let* ((home (expand-file-name (getenv "HOME")))
(home-len (length home)))
(if (and
(>= (length pwd) home-len)
(equal home (substring pwd 0 home-len)))
(concat "~" (substring pwd home-len))
pwd)))
Make the directory name be shorter… by replacing all directory names with its first names. We leave the last two to be the full names. Why yes, I did steal this.
(defun pwd-shorten-dirs (pwd)
"Shorten all directory names in PWD except the last two."
(let ((p-lst (split-string pwd "/")))
(if (> (length p-lst) 2)
(concat
(mapconcat (lambda (elm) (if (zerop (length elm)) ""
(substring elm 0 1)))
(butlast p-lst 2)
"/")
"/"
(mapconcat (lambda (elm) elm)
(last p-lst 2)
"/"))
pwd))) ;; Otherwise, we return the PWD
Break up the directory into a "parent" and a "base":
(defun split-directory-prompt (directory)
(if (string-match-p ".*/.*" directory)
(list (file-name-directory directory) (file-name-base directory))
(list "" directory)))
Using virtual environments for certain languages is helpful to know, since I change them based on the directory.
(defun ruby-prompt ()
"Returns a string (may be empty) based on the current Ruby Virtual Environment."
(let* ((executable "~/.rvm/bin/rvm-prompt")
(command (concat executable "v g")))
(when (file-exists-p executable)
(let* ((results (shell-command-to-string executable))
(cleaned (string-trim results))
(gem (propertize "\xe92b" 'face `(:family "alltheicons"))))
(when (and cleaned (not (equal cleaned "")))
(s-replace "ruby-" gem cleaned))))))
(defun python-prompt ()
"Returns a string (may be empty) based on the current Python
Virtual Environment. Assuming I've called the M-x command:
`pyenv-mode-set'."
(when (fboundp #'pyenv-mode-version)
(let ((venv (pyenv-mode-version)))
(when venv
(concat
(propertize "\xe928" 'face `(:family "alltheicons"))
(pyenv-mode-version))))))
Now tie it all together with a prompt function can color each of the prompts components.
(defun eshell/eshell-local-prompt-function ()
"A prompt for eshell that works locally (in that it assumes it
could run certain commands) to make a prettier, more-helpful
local prompt."
(interactive)
(let* ((pwd (eshell/pwd))
(directory (split-directory-prompt
(pwd-shorten-dirs
(pwd-replace-home pwd))))
(parent (car directory))
(name (cadr directory))
(branch (curr-dir-git-branch-string pwd))
(ruby (when (not (file-remote-p pwd)) (ruby-prompt)))
(python (when (not (file-remote-p pwd)) (python-prompt)))
(dark-env (eq 'dark (frame-parameter nil 'background-mode)))
(for-bars `(:weight bold))
(for-parent (if dark-env `(:foreground "dark orange") `(:foreground "blue")))
(for-dir (if dark-env `(:foreground "orange" :weight bold)
`(:foreground "blue" :weight bold)))
(for-git `(:foreground "green"))
(for-ruby `(:foreground "red"))
(for-python `(:foreground "#5555FF")))
(concat
(propertize "⟣─ " 'face for-bars)
(propertize parent 'face for-parent)
(propertize name 'face for-dir)
(when branch
(concat (propertize " ── " 'face for-bars)
(propertize branch 'face for-git)))
;; (when ruby
;; (concat (propertize " ── " 'face for-bars)
;; (propertize ruby 'face for-ruby)))
;; (when python
;; (concat (propertize " ── " 'face for-bars)
;; (propertize python 'face for-python)))
(propertize "\n" 'face for-bars)
(propertize (if (= (user-uid) 0) " #" " $") 'face `(:weight ultra-bold))
;; (propertize " └→" 'face (if (= (user-uid) 0) `(:weight ultra-bold :foreground "red") `(:weight ultra-bold)))
(propertize " " 'face `(:weight bold)))))
(setq-default eshell-prompt-function #'eshell/eshell-local-prompt-function)
Here is the result:
Fringe Status
The eshell-fringe-status project shows a color-coded icon of the previous command run (green for success, red for error). Doesn’t work reliably, but the fringe is inconspicuous. Seems to me, that if would be useful to rejuggle those fringe markers so that the marker matched the command entered (instead of seeing a red mark, and needing to scroll back to seethe command that made the error). Still…
(use-package eshell-fringe-status
:hook (eshell-mode . eshell-fringe-status-mode))
Opening Banner
Whenever I open a shell, I instinctively type ls
… so why not do that automatically?
(defun ha-eshell-banner ()
"Return a string containing the files in the current directory."
(let* ((non-hidden (rx string-start
(not (any "." "#"))
(one-or-more any)
(not "~")
string-end))
(files (directory-files default-directory nil non-hidden))
(longest (reduce (lambda (longest file) (max longest (length file)))
files :initial-value 1))
(padded (format "%%-%ds " longest))
(width (window-total-width))
(columns (/ width (+ longest 3))))
(cl-flet* ((process-file
(file)
(let ((image-rx (rx "." (or "png" "jpg" "jpeg" "tif" "wav") string-end))
(code-rx (rx "." (or "el" "py" "rb") string-end))
(docs-rx (rx "." (or "org" "md") string-end)))
(format padded (cond
((string-match image-rx file) (propertize file 'face '(:foreground "light pink")))
((string-match code-rx file) (propertize file 'face '(:foreground "DarkSeaGreen1")))
((file-directory-p file) (propertize file 'face '(:foreground "SteelBlue")))
(t file)))))
;; This nasty little function was an attempt to make
;; things readable, but who would have thunk that
;; `mapconcat' couldn't access a symbol reference to a
;; function created by `cl-flet*'!?
(process-files
(table)
(mapconcat (lambda (line) (mapconcat
(lambda (file) (process-file file))
line
"• "))
table
"\n")))
(concat (process-files (seq-partition files columns)) "\n\n"))))
Shell Windows
Now that I often need to quickly pop into remote systems to run a shell or commands, I create helper functions to create those buffer windows. Each begin with eshell-
:
Shell There
The basis for opening an shell depends on the location. After that, we make the window smaller, give the buffer a good name, as well as immediately display the files with ls
(since I instinctively just do that … every time).
(defun eshell-there (parent)
"Open an eshell session in a PARENT directory.
The window is smaller and named after this directory."
(let* ((name (thread-first parent
(split-string "/" t)
(last)
(car)))
(height (/ (window-total-height) 3))
(default-directory parent))
(split-window-vertically (- height))
(setq eshell-buffer-name (format "*eshell: %s*" name))
(eshell)))
Shell Here
This version of the eshell
is based on the current buffer’s parent directory:
(defun eshell-here ()
"Opens up a new shell in the directory of the current buffer.
The eshell is renamed to match that directory to make multiple
eshell windows easier."
(interactive)
(eshell-there (if (buffer-file-name)
(file-name-directory (buffer-file-name))
default-directory)))
And let’s bind it:
(bind-key "C-!" 'eshell-here)
Shell for a Project
I usually want the eshell
to start in the project’s root, using projectile-project-root:
(defun eshell-project ()
"Open a new shell in the project root directory, in a smaller window."
(interactive)
(eshell-there (projectile-project-root)))
And we can attach this function to the projectile
menu:
(ha-leader "p s" '("shell" . eshell-project))
Shell Over There
Would be nice to be able to run an eshell session and use Tramp to connect to the remote host in one fell swoop:
(defun eshell-remote (host)
"Creates an eshell session that uses Tramp to automatically
connect to a remote system, HOST. The hostname can be either the
IP address, or FQDN, and can specify the user account, as in
root@blah.com. HOST can also be a complete Tramp reference."
(interactive "sHost: ")
(let ((destination-path
(cond
((string-match-p "^/" host) host)
((string-match-p (ha-eshell-host-regexp 'full) host)
(string-match (ha-eshell-host-regexp 'full) host) ;; Why!?
(let* ((user1 (match-string 2 host))
(host1 (match-string 3 host))
(user2 (match-string 6 host))
(host2 (match-string 7 host)))
(if host1
(ha-eshell-host->tramp user1 host1)
(ha-eshell-host->tramp user2 host2))))
(t (format "/%s:" host)))))
(eshell-there destination-path)))
Shell Here to There
Since I have Org files that contains tables of system to remotely connect to, I figured I should have a little function that can jump to a host found listed anywhere on the line.
The regular expression associated with IP addresses, hostnames, user accounts (of the form, jenkins@my.build.server
, or even full Tramp references, is a bit…uhm, hairy. And since I want to reuse these, I will hide them in a function:
(defun ha-eshell-host-regexp (regexp)
"Returns a particular regular expression based on symbol, REGEXP"
(let* ((user-regexp "\\(\\([[:alnum:]._-]+\\)@\\)?")
(tramp-regexp "\\b/ssh:[:graph:]+")
(ip-char "[[:digit:]]")
(ip-plus-period (concat ip-char "+" "\\."))
(ip-regexp (concat "\\(\\(" ip-plus-period "\\)\\{3\\}" ip-char "+\\)"))
(host-char "[[:alpha:][:digit:]-]")
(host-plus-period (concat host-char "+" "\\."))
(host-regexp (concat "\\(\\(" host-plus-period "\\)+" host-char "+\\)"))
(horrific-regexp (concat "\\b"
user-regexp ip-regexp
"\\|"
user-regexp host-regexp
"\\b")))
(cond
((eq regexp 'tramp) tramp-regexp)
((eq regexp 'host) host-regexp)
((eq regexp 'full) horrific-regexp))))
The function to scan a line for hostname patterns uses different function calls that what I could use for eshell-there
, so let's save-excursion
and hunt around:
(defun ha-eshell-scan-for-hostnames ()
"Helper function to scan the current line for any hostnames, IP
or Tramp references. This returns a tuple of the username (if
found) and the hostname.
If a Tramp reference is found, the username part of the tuple
will be `nil'."
(save-excursion
(goto-char (line-beginning-position))
(if (search-forward-regexp (ha-eshell-host-regexp 'tramp) (line-end-position) t)
(cons nil (buffer-substring-no-properties (match-beginning 0) (match-end 0)))
;; Returns the text associated with match expression, NUM or `nil' if no match was found.
(cl-flet ((ha-eshell-get-expression (num) (if-let ((first (match-beginning num))
(end (match-end num)))
(buffer-substring-no-properties first end))))
(search-forward-regexp (ha-eshell-host-regexp 'full) (line-end-position))
;; Until this is completely robust, let's keep this debugging code here:
;; (message (mapconcat (lambda (tup) (if-let ((s (car tup))
;; (e (cadr tup)))
;; (buffer-substring-no-properties s e)
;; "null"))
;; (-partition 2 (match-data t)) " -- "))
(let ((user1 (ha-eshell-get-expression 2))
(host1 (ha-eshell-get-expression 3))
(user2 (ha-eshell-get-expression 6))
(host2 (ha-eshell-get-expression 7)))
(if host1
(cons user1 host1)
(cons user2 host2)))))))
Tramp reference can be long when attempting to connect as another user account using the pipe symbol.
(defun ha-eshell-host->tramp (username hostname &optional prefer-root)
"Return a TRAMP reference based on a USERNAME and HOSTNAME
that refers to any host or IP address."
(cond ((string-match-p "^/" host)
host)
((or (and prefer-root (not username)) (equal username "root"))
(format "/ssh:%s|sudo:%s:" hostname hostname))
((or (null username) (equal username user-login-name))
(format "/ssh:%s:" hostname))
(t
(format "/ssh:%s|sudo:%s|sudo@%s:%s:" hostname hostname username hostname))))
This function pulls it all together:
(defun eshell-here-on-line (p)
"Search the current line for an IP address or hostname, and call the `eshell-here' function.
Call with PREFIX to connect with the `root' useraccount, via
`sudo'."
(interactive "p")
(destructuring-bind (user host) (ha-eshell-scan-for-hostnames)
(let ((destination (ha-eshell-host->tramp user host (> p 1))))
(message "Connecting to: %s" destination)
(eshell-there destination))))
Better Command Line History
On this discussion a little gem for using IDO to search back through the history, instead of M-R
to prompt for the history.
(defun eshell-insert-history ()
"Displays the eshell history to select and insert back into your eshell."
(interactive)
(insert (completing-read "Eshell history: "
(delete-dups
(ring-elements eshell-history-ring)))))
Command on the File Buffer
Sometimes you need to change something about the current file you are editing…like the permissions or even execute it. Hitting Command-1
will prompt for a shell command string and then append the current file to it and execute it.
(defun execute-command-on-file-buffer (cmd)
"Executes a shell command, CMD, on the current buffer's file.
Appends the filename to the command if not specified, so:
chmod a+x
Works as expected. We replace the special variable `$$' with the
filename of the buffer. Note that `eshell-command' executes this
command, so eshell modifiers are available, for instance:
mv $$ $$(:r).txt
Will rename the current file to now have a .txt extension.
See `eshell-display-modifier-help' for details on that."
(interactive "sCommand to execute: ")
(let* ((file-name (buffer-file-name))
(full-cmd (cond ((string-match (rx "$$") cmd)
(replace-regexp-in-string (rx "$$") file-name cmd))
((string-match (rx (literal file-name)) cmd)
cmd)
(t
(concat cmd " " file-name)))))
(message "Executing: %s" full-cmd)
(eshell-command full-cmd)))
Configuration
Here is where we associate all the functions and their hooks with eshell
, through the magic of use-package
.
(use-package eshell
:straight (:type built-in)
:custom (eshell-banner-message '(ha-eshell-banner))
:init
(setq eshell-error-if-no-glob t
;; This jumps back to the prompt:
eshell-scroll-to-bottom-on-input 'all
eshell-hist-ignoredups t
eshell-save-history-on-exit t
;; Since eshell starts fast, let's dismiss it on exit:
eshell-kill-on-exit t
eshell-destroy-buffer-when-process-dies t
;; Can you remember the parameter differences between the
;; executables `chmod' and `find' and their Emacs counterpart?
;; Me neither, so this makes it act a bit more shell-like:
eshell-prefer-lisp-functions nil)
:hook ((eshell-pred-load . ha-eshell-add-predicates))
:bind (("M-!" . execute-command-on-file-buffer)
("s-1" . execute-command-on-file-buffer)
:map eshell-mode-map
("M-R" . eshell-insert-history)
("C-d" . ha-eshell-quit-or-delete-char)))
Note that the default list to eshell-visual-commands is good enough.
Add leader commands to call my defined functions:
(ha-leader
"!" '("eshell cmd" . execute-command-on-file-buffer)
"a e" '(:ignore t :which-key "eshell")
"a e e" '("new eshell" . eshell-here)
"a e r" '("remote" . eshell-remote)
"a e p" '("project" . eshell-project)
"a e g" '("at point" . eshell-here-on-line)
"a e !" '("exec on file-buffer" . execute-command-on-file-buffer))
No, i’m not sure why use-package
has an issue with both :hook
, :bind
and :config
directives in sequence.