Minor cleanup on eshell and remote terminals

This commit is contained in:
Howard Abrams 2025-03-20 10:34:29 -07:00
parent 9bb159c087
commit a1118e0c2e
2 changed files with 63 additions and 33 deletions

View file

@ -43,6 +43,9 @@ I like =debug-on-error=, but not in Eshell, so I set this up when entering Eshel
(setq mode-line-format nil)) (setq mode-line-format nil))
#+end_src #+end_src
** Directory Notification ** Directory Notification
:PROPERTIES:
:ID: AD963BB5-6346-46C0-A364-4C8AEBA29882
:END:
While most people put the “current directory” in their shell prompt, Ive always found that pretty distracting, difficult when copy/pasting commands, and cuts out on the length of my commands (which might be a good thing, tbh). While most people put the “current directory” in their shell prompt, Ive always found that pretty distracting, difficult when copy/pasting commands, and cuts out on the length of my commands (which might be a good thing, tbh).
I use the =header-line= for this. I use the =header-line= for this.
@ -155,7 +158,7 @@ Keep in mind that the predicates are somewhat finicky. For instance, the followi
mv $f $f(:r).txt mv $f $f(:r).txt
#+end_src #+end_src
But you could call this: Instead, call this:
#+begin_src sh #+begin_src sh
mv $f $f(:s/org$/txt/) mv $f $f(:s/org$/txt/)
@ -1247,23 +1250,6 @@ I use =ex= to refer to both =en= / =ec=. Use cases:
The =-c= option can be combined with the /command/, but I dont want it to grab the last output, as I think I would just like to send text to the notebook as after thoughts. If the option to =-c= is blank, perhaps it just calls the capture template to allow me to enter voluminous content. The =-c= option can be combined with the /command/, but I dont want it to grab the last output, as I think I would just like to send text to the notebook as after thoughts. If the option to =-c= is blank, perhaps it just calls the capture template to allow me to enter voluminous content.
This requires capture templates that dont do any formatting. I will reused =c c= from [[file:ha-capturing-notes.org::*General Notes][capturing-notes]] code, and create other templates under =e= prefix:
#+begin_src emacs-lisp
;; (setq org-capture-templates nil)
(add-to-list 'org-capture-templates
'("e" "Engineering Notebook"))
(add-to-list 'org-capture-templates
'("ee" "Notes and Commentary" plain
(file+olp+datetree org-default-notes-file "General Notes")
"%i" :empty-lines 1 :tree-type month :unnarrowed t))
(add-to-list 'org-capture-templates
'("ef" "Piped-in Contents" plain
(file+olp+datetree org-default-notes-file "General Notes")
"%i" :immediate-finish t :empty-lines 1 :tree-type month))
#+end_src
#+begin_src emacs-lisp #+begin_src emacs-lisp
(defun ha-eshell-engineering-notebook (capture-template args) (defun ha-eshell-engineering-notebook (capture-template args)
"Capture commands and output from Eshell into an Engineering Notebook. "Capture commands and output from Eshell into an Engineering Notebook.
@ -1328,7 +1314,33 @@ This requires capture templates that dont do any formatting. I will reused =c
;; Return output from the command, or nothing if there wasn't anything: ;; Return output from the command, or nothing if there wasn't anything:
(or output ""))) (or output "")))
#+end_src #+end_src
And now we have a =en= and a =ec= version:
This requires capture templates that dont do any formatting. I will reused =c c= from [[file:ha-capturing-notes.org::*General Notes][capturing-notes]] code, and create other templates under =e= prefix:
#+begin_src emacs-lisp
(add-to-list 'org-capture-templates '("e" "Engineering Notebook"))
(add-to-list 'org-capture-templates
'("ee" "Notes and Commentary" plain
(file+olp+datetree org-default-notes-file "General Notes")
"%i" :empty-lines 1 :tree-type month :unnarrowed t))
(add-to-list 'org-capture-templates
'("ef" "Piped-in Contents" plain
(file+olp+datetree org-default-notes-file "General Notes")
"%i" :immediate-finish t :empty-lines 1 :tree-type month))
#+end_src
The =cap= function simply calls [[help-org-capture][org-capture]] with [[info:org#Template elements][a template]]:
#+begin_src emacs-lisp
(defun eshell/cap (&rest args)
"Call `org-capture' with the `ee' template to enter text into the engineering notebook."
(org-capture nil "ee"))
#+end_src
The =en= and =ec= functions, since they do /pre-processing/ before calling =org-capture= calls some helper functions:
#+begin_src emacs-lisp #+begin_src emacs-lisp
(defun eshell/en (&rest args) (defun eshell/en (&rest args)
"Call `ha-eshell-engineering-notebook' to \"General Notes\"." "Call `ha-eshell-engineering-notebook' to \"General Notes\"."
@ -1341,12 +1353,7 @@ And now we have a =en= and a =ec= version:
(ha-eshell-engineering-notebook "cc" args)) (ha-eshell-engineering-notebook "cc" args))
#+end_src #+end_src
This function simply calls [[help-org-capture][org-capture]] with [[info:org#Template elements][a template]]: In order to have /devices/, for instance =some-command > e=, we define some functions:
#+begin_src emacs-lisp
(defun eshell/cap (&rest args)
"Call `org-capture' with the `ee' template to enter text into the engineering notebook."
(org-capture nil "ee"))
#+end_src
#+begin_src emacs-lisp #+begin_src emacs-lisp
(defun ha-eshell-target-engineering-notebook (output) (defun ha-eshell-target-engineering-notebook (output)
@ -1357,7 +1364,9 @@ This function simply calls [[help-org-capture][org-capture]] with [[info:org#Tem
"Write OUTPUT into the current clocked in task via `org-capture'." "Write OUTPUT into the current clocked in task via `org-capture'."
(ha-eshell-engineering-capture "cc" nil nil output)) (ha-eshell-engineering-capture "cc" nil nil output))
#+end_src #+end_src
And finally, add our new functions to [[elisp(describe-variable 'eshell-virtual-targets)][eshell-virtual-targets]]:
And add these functions to [[elisp(describe-variable 'eshell-virtual-targets)][eshell-virtual-targets]], after =eshell= has been loaded:
#+begin_src emacs-lisp #+begin_src emacs-lisp
(with-eval-after-load "eshell" (with-eval-after-load "eshell"
(add-to-list 'eshell-virtual-targets '("/dev/e" ha-eshell-target-engineering-notebook nil)) (add-to-list 'eshell-virtual-targets '("/dev/e" ha-eshell-target-engineering-notebook nil))
@ -1505,8 +1514,11 @@ Now tie it all together with a prompt function can color each of the prompts com
(setq-default eshell-prompt-function #'eshell/eshell-local-prompt-function) (setq-default eshell-prompt-function #'eshell/eshell-local-prompt-function)
#+end_src #+end_src
Here is the result: Here is the result:
[[http://imgur.com/nkpwII0.png]] [[file:screenshots/eshell-special-prompt.png]]
To be honest, this sort of prompt is garish and distracting. A better approach is to have that information in either the mode line, or a [[Directory Notification][header-line]].
** Simple Prompt with Mode Line ** Simple Prompt with Mode Line
To achieve more /screen estate/, leave your prompt simple: To achieve more /screen estate/, leave your prompt simple:
#+begin_src emacs-lisp #+begin_src emacs-lisp
@ -1615,6 +1627,17 @@ And lets bind it:
#+begin_src emacs-lisp #+begin_src emacs-lisp
(bind-key "C-!" 'eshell-here) (bind-key "C-!" 'eshell-here)
#+end_src #+end_src
Obviously, I usually want to open a shell in the root of the current project.
#+BEGIN_SRC emacs-lisp
(defun eshell-project ()
"Call `eshell-there' with the current project root."
(interactive)
(eshell-there (or (project-root (project-current))
default-directory)))
#+END_SRC
** Shell Over There ** 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: Would be nice to be able to run an eshell session and use Tramp to connect to the remote host in one fell swoop:
#+begin_src emacs-lisp #+begin_src emacs-lisp

View file

@ -162,6 +162,7 @@ While not as fast as [[https://github.com/akermu/emacs-libvterm][vterm]], the [[
("terminfo/65" "terminfo/65/*") ("terminfo/65" "terminfo/65/*")
("integration" "integration/*") ("integration" "integration/*")
(:exclude ".dir-locals.el" "*-tests.el"))) (:exclude ".dir-locals.el" "*-tests.el")))
:commands (eat eat-make eat-project)
:bind (:map eat-semi-char-mode-map :bind (:map eat-semi-char-mode-map
("C-c C-t" . ha-eat-narrow-to-shell-prompt-dwim)) ("C-c C-t" . ha-eat-narrow-to-shell-prompt-dwim))
:config :config
@ -475,7 +476,7 @@ We'll give =openstack= CLI a =--format json= option to make it easier for parsin
#+begin_src emacs-lisp #+begin_src emacs-lisp
(defun ha-ssh-overcloud-query-for-hosts () (defun ha-ssh-overcloud-query-for-hosts ()
"If the overcloud cache hasn't be populated, ask the user if we want to run the command." "If the overcloud cache hasn't be populated, ask the user if we want to run the command."
(when (not ha-ssh-favorite-hostnames) (unless ha-ssh-favorite-hostnames
(when (y-or-n-p "Cache of Overcloud hosts aren't populated. Retrieve hosts?") (when (y-or-n-p "Cache of Overcloud hosts aren't populated. Retrieve hosts?")
(call-interactively 'ha-ssh-overcloud-cache-populate)))) (call-interactively 'ha-ssh-overcloud-cache-populate))))
@ -488,7 +489,7 @@ We'll do the work of getting the /server list/ with this function:
(defun ha-ssh-overcloud-cache-populate (cluster) (defun ha-ssh-overcloud-cache-populate (cluster)
"Given an `os-cloud' entry, stores all available hostnames. "Given an `os-cloud' entry, stores all available hostnames.
Calls `ha-ssh-add-favorite-host' for each host found." Calls `ha-ssh-add-favorite-host' for each host found."
(interactive (list (completing-read "Cluster: " '(devprod501 devprod502)))) (interactive (list (completing-read "Cluster: " '(devprod501 devprod502 devprod502-yawxway))))
(message "Calling the `openstack' command...this will take a while. Grab a coffee, eh?") (message "Calling the `openstack' command...this will take a while. Grab a coffee, eh?")
(let* ((command (format "openstack --os-cloud %s server list --no-name-lookup -f json" cluster)) (let* ((command (format "openstack --os-cloud %s server list --no-name-lookup -f json" cluster))
(json-data (thread-last command (json-data (thread-last command
@ -496,10 +497,12 @@ We'll do the work of getting the /server list/ with this function:
(json-read-from-string)))) (json-read-from-string))))
(dolist (entry (seq--into-list json-data)) (dolist (entry (seq--into-list json-data))
(ha-ssh-add-favorite-host (alist-get 'Name entry) (ha-ssh-add-favorite-host (alist-get 'Name entry)
(thread-last entry (or
(alist-get 'Networks) (thread-last entry
(alist-get 'cedev13) (alist-get 'Networks)
(seq-first)))) (alist-get 'cedev13)
(seq-first))
(alist-get 'Name entry))))
(message "Call to `openstack' complete. Found %d hosts." (length json-data)))) (message "Call to `openstack' complete. Found %d hosts." (length json-data))))
#+end_src #+end_src
@ -554,3 +557,7 @@ Before you can build this on a new system, make sure that you put the cursor ove
#+options: num:nil toc:t todo:nil tasks:nil tags:nil date:nil #+options: num:nil toc:t todo:nil tasks:nil tags:nil date:nil
#+options: skip:nil author:nil email:nil creator:nil timestamp:nil #+options: skip:nil author:nil email:nil creator:nil timestamp:nil
#+infojs_opt: view:nil toc:t ltoc:t mouse:underline buttons:0 path:http://orgmode.org/org-info.js #+infojs_opt: view:nil toc:t ltoc:t mouse:underline buttons:0 path:http://orgmode.org/org-info.js
# Local Variables:
# eval: (org-next-visible-heading 1)
# End: