detached
- Description
- A package to launch, and manage, detached processes
- Latest
- detached-0.10.1.tar (.sig), 2024-Mar-31, 330 KiB
- Maintainer
- detached.el Development <~niklaseklund/detached.el@lists.sr.ht>
- Atom feed
- detached.xml
- Website
- https://sr.ht/~niklaseklund/detached.el/
- Browse ELPA's repository
- CGit or Gitweb
- Badge
To install this package from Emacs, use package-install
or list-packages
.
Full description
detached.el
Introduction
detached.el
is a package to launch, and manage, detached processes. The idea is that these processes are detached from the Emacs process and the package can make Emacs seamlessly attach to these processes. This enables users to launch processes that can survive when Emacs itself is being shutdown. The package relies on the program dtach, in order to make this functionality possible.
Internally the package transforms the detached process into a detached-session
object, which essentially is a text-based representation of the process. All detached-session
objects are stored in a lisp file, and the output of all sessions are captured into individual logs.
The package supports integration with multiple packages, here is a list of the built-in packages that are supported:
shell
eshell
compile
org
dired
Features
Since a detached-session
contain all the output of the process as well as data such as, what command
was run, which directory
the process was launched etc, it opens up the possibility for the following features:
Unlimited scrollback:
All the output from adetached-session
is always accessibleRemote support:
Full support for running on remote hosts. SeeRemote support
section of the READMENotifications:
The package will monitor all detached sessions and notify when a session has finishedPost compilation
: The package will know the exit status of a session, which enables the package to post compile the output of a session to enable Emacs's built-in functionality of navigating between errors in the output.Annotations:
When selecting a session all are presented with a rich set of annotationsActions:
The package provides actions to act on a session:kill
a an active sessionrerun
a sessioncopy
the output of a sessiondiff
the output of two different sessions
Installation
The package is available on GNU ELPA and MELPA, and for users of the GNU Guix package manager there is a guix package.
Configuration
The prerequisite for detached.el
is that the user has the programs dtach
and tail
installed.
Use-package example
A minimal use-package
configuration.
emacs-lisp
(use-package detached
:init
(detached-init)
:bind (;; Replace `async-shell-command' with `detached-shell-command'
([remap async-shell-command] . detached-shell-command)
;; Replace `compile' with `detached-compile'
([remap compile] . detached-compile)
([remap recompile] . detached-compile-recompile)
;; Replace built in completion of sessions with `consult'
([remap detached-open-session] . detached-consult-session))
:custom ((detached-show-output-on-attach t)
(detached-terminal-data-command system-type)))
The users are required to call detached-init
. This function orchestrates the integration with all other internal and external packages that detached.el
supports. These are:
compile
dired
dired-rsync
embark
eshell
org
projectile
shell
vterm
All of the integration are configured to enable lazy-loading. Meaning that if you are not a user of projectile
that code that handles the integration will never load.
However if you do want to disable the integration with a specific package, or enable for a subset of the packages use the variables detached-init-allow-list
or detached-init-block-list
.
Usage
The idea is that users can choose to either:
- create
: a detached session and attach to it
- detach
: from a detached session
- attach
: to a detached session
In the integration of detached.el
with other packages these commands are always existent, with the exception for org-mode
.
To detach from a detached-session
in any of the modes, use the universal detached-detach-session
command. The keybinding for this command is defined by the detached-detach-key
variable, which by default has the value C-c C-d
.
General
To interact with a session the package provides the command detached-open-session
. This provides a convenient completion interface, enriched with annotations to provide useful information about the sessions. The detached-open-session
command is implemented as a do what I mean command. This results in detached.el
performing different actions depending on the state of a session. The actions can be configured based on the origin
of the session. The user can have one set of configurations for sessions started in shell
which is different from those started in compile
.
The actions are controlled by the customizable variables named detached-.*-session-action
. They come preconfigured but if you don't like the behavior of detached-open-session
these variables allows for tweaking the experience.
- If the session is
active
, call the sessionsattach
function - If the session is
inactive
call the sessionsview
function, which by default performs a post-compile on the session if its status isfailure
otherwise the sessions raw output is opened.
The package also provides additional commands to interact with a session.
Command (Keybinding) | Description |
---|---|
detached-view-session (v) | View a session's output |
detached-attach-session (a) | Attach to a session |
detached-diff-session (=) | Diff a session with another session |
detached-compile-session (c) | Open the session output in compilation mode |
detached-rerun-session (r) | Rerun a session |
detached-insert-session-command (i) | Insert the session's command at point |
detached-copy-session-command (w) | Copy the session's shell command |
detached-copy-session-output (W) | Copy the session's output |
detached-kill-session (k) | Kill an active session |
detached-delete-session (d) | Delete an inactive session |
These commands are available through the detached-action-map
. The user can bind the action map to a keybinding of choice. For example
emacs-lisp
(global-set-key (kbd "C-c d") detached-action-map)
Then upon invocation the user can choose an action, keybindings listed in the table above, and then choose a session to perform the action upon. For those using embark
this will not be necessary as detached-init
sets up integration with embark actions.
Shell command
The detached-shell-command
is for the Emacs users that are accustomed to running shell commands from M-x shell-command
or M-x async-shell-command
.
Shell
A minor mode named detached-shell-mode
is provided, and will be enabled in shell
. The commands that are implemented are:
Command | Description | Keybinding |
---|---|---|
detached-shell-send-input | Run command with detached | |
detached-shell-attach-session | Attach to a detached session | |
detached-detach-session | Detach from a detached session | detached-detach-key |
Eshell
A minor mode named detached-eshell-mode
is provided, and will be enabled in eshell
. The commands that are implemented are:
Command | Description | Keybinding |
---|---|---|
detached-eshell-send-input | Run command with detached | |
detached-eshell-attach-session | Attach to a detached session | |
detached-detach-session | Detach from a detached session | detached-detach-key |
Org babel
The package implements an additional header argument for ob-shell
. The header argument is :detached t
. When provided it will enable the code inside a src block to be run with detached.el
. Since org is not providing any live updates on the output the session is created with detached-sesion-mode
set to create
. This means that if you want to access the output of the session you do that the same way you would for any other type of session. The detached-org
works both with and without the :session
header argument.
```
+begin_src sh :detached t
cd ~/code
ls -la
+end_src
+RESULTS:
: [detached] ```
Compile
The package implements the commands detached-compile
and detached-compile-recompile
, which are thin wrappers around the original compile
and recompile
commands. The users should be able to use the former as replacements for the latter without noticing any difference except from the possibility to detach
.
Consult
The command detached-consult-session
is a replacement for detached-open-session
using the consult package. The difference is that the consult command provides multiple session sources, which is defined in the detached-consult-sources
variable. Users can customize which sources to use, as well as use individual sources in other consult
commands, such as consult-buffer
. The users can also narrow the list of sessions by entering a key. The list of supported keys are:
Type | Key |
---|---|
Active sessions | a |
Inactive sessions | i |
Successful sessions | s |
Failed sessions | f |
Hidden sessions | SPC |
Local host sessions | l |
Remote host sessions | r |
Current host sessions | c |
Customization
Customizable variables
The package provides the following customizable variables.
Name | Description |
---|---|
detached-session-directory | A host specific directory to store sessions in |
detached-db-directory | A localhost specific directory to store the database |
detached-dtach-program | Name or path to the dtach program |
detached-shell-program | Name or path to the shell that detached.el should use |
detached-timer-configuration | Configuration of the timer that runs on remote hosts |
detached-annotation-format | A list of annotations that should be present in completion |
detached-command-format | A configuration for displaying a session command |
detached-degraded-commands | A list of commands that should be run in degraded mode |
detached-notification-function | Specifies which function to issue notifications with |
detached-detach-key | Specifies which keybinding to use to detach from a session |
detached-shell-command-initial-input | Enables latest value in history to be used as initial input |
detached-filter-ansi-sequences | Specifies if detached.el will use ansi-color to filter out escape sequences |
detached-show-output-command | Specifies if detached.el should show the session's output when attaching to it |
Apart from those variables there is also the different action
variables, which can be configured differently depending on the origin of the session.
Name | Description |
---|---|
detached-shell-command-session-action | Actions for sessions launched with detached-shell-command |
detached-eshell-session-action | Actions for sessions launched with detached-eshell-send-input |
detached-shell-session-action | Actions for sessions launched with detached-shell-send-input |
detached-compile-session-action | Actions for sessions launched with detached-compile |
detached-org-session-action | Actions for sessions launched with detached-org |
Remote support
The detached.el
supports Connection Local Variables which allows the user to customize the variables used by detached.el
when running on a remote host. This example shows how the following variables are customized for all remote hosts.
``` emacs-lisp (connection-local-set-profile-variables 'remote-detached '((detached-shell-program . "/bin/bash") (detached-session-directory . "~/tmp") (detached-dtach-program . "/home/user/.local/bin/dtach")))
(connection-local-set-profiles '(:application tramp :protocol "ssh") 'remote-detached) ```
Completion annotations
Users can customize the appearance of annotations in detached-open-session
by modifying the detached-annotation-format
. The default annotation format is the following.
emacs-lisp
(defvar detached-annotation-format
`((:width 3 :function detached--state-str :face detached-state-face)
(:width 3 :function detached--status-str :face detached-failure-face)
(:width 10 :function detached--host-str :face detached-host-face)
(:width 40 :function detached--working-dir-str :face detached-working-dir-face)
(:width 30 :function detached--metadata-str :face detached-metadata-face)
(:width 10 :function detached--duration-str :face detached-duration-face)
(:width 8 :function detached--size-str :face detached-size-face)
(:width 12 :function detached--creation-str :face detached-creation-face))
"The format of the annotations.")
Show session context when attaching
By default the detached-show-session-context
is set to t. This means that part of the output from a session will be shown when attaching to a session. The number of lines of the context is determined by detached-session-context-lines
. The package uses tail
in order to display the context.
Metadata annotators
The user can configure any number of annotators to run upon creation of a session. The package comes with a function which captures the git branch name, if the session is started in a git repository.
This function can be added as an annotation function to the detached-metadata-annotators-alist
together with a symbol describing the property.
emacs-lisp
(setq detached-metadata-annotators-alist '((branch . detached--metadata-git-branch)))
Degraded commands
To be able to both attach to a dtach session as well as logging its output detached.el
relies on the usage of tee
. However it is possible that the user tries to run a command which involves a program that doesn't integrate well with tee. In those situations the output could be delayed until the session ends, which is not preferable.
For these situations detached.el
provides the detached-degraded-commands
variable. This is a list of regular expressions. Any command that matches any of the strings will be getting the property degraded
set to true.
emacs-lisp
(setq detached-degraded-commands '("^ls"))
Here a command beginning with ls
would from now on be considered degraded, hence detached
will use tail
to tail the sessions log instead of attaching to the dtach
process.
Colors in sessions
The package needs to use a trick to get programs programs such as git
or grep
to show color in their outputs. This is because these commands only use colors and ansi sequences if they are being run in a terminal, as opposed to a pipe. The package therefore has two different modes, either plain-text
or terminal-data
. The latter is now the default for all sessions. When in terminal-data
mode the script
tool is used to make programs run by detached.el
think they are inside of a full-featured terminal, and therefore can log their raw terminal data.
The drawback is that there can be commands which generates escape sequences that the package supports and will therefore mess up the output for some commands. If you detect such an incompatible command you can add a regexp that matches that command to the list detached-plain-text-commands
. By doing so detached.el
will be instructed to run those commands in plain-text mode.
The tool script
can have different options depending on version and operating system. The package requires the user to set the detached-terminal-data-command
in the package configuration and with connection-local-set-profiles
for remote sessions. It can either be set to a symbol or to a string.
Chained commands
With detached
there exist the possibility to use callback. This functionality makes it possible to create chained sessions, essentially starting a new session once a previous one is finished. Here is an example:
emacs-lisp
;; The detached commands are run in serial.
;; This is equivalent to run sleep 1 && ls && ls -la
(let* ((default-directory "/tmp")
(detached-session-action
`(,@detached-shell-command-session-action
:callback (lambda (session1)
(when (eq 'success (detached-session-status session1))
(let ((default-directory (detached-session-working-directory session1))
(detached-session-action
`(,@detached-shell-command-session-action
:callback (lambda (session2)
(when (eq 'success (detached-session-status session2))
(let ((default-directory (detached-session-working-directory session2)))
(detached-start-session "ls -la" t)))))))
(detached-start-session "ls" t)))))))
(detached-start-session "sleep 1" t))
Tips & Tricks
The detached.el
package integrates with core Emacs packages as well as 3rd party packages. Integration is orchestrated in the detached-init.el
. In this section you can find tips for integrations that are not supported in the package itself.
Alert
By default detached.el
uses the built in notifications
library to issue a notification. This solution uses dbus
but if that doesn't work for the user there is the possibility to set the detached-notification-function
to detached-state-transition-echo-message
to use the echo area instead. If that doesn't suffice there is the possibility to use the alert package to get a system notification instead.
emacs-lisp
(setq detached-notification-function #'detached-extra-alert-notification)
Versions
Information about larger changes that has been made between versions can be found in the CHANGELOG.org
Support
The detached.el
package should work on Linux
and macOS
. It is regularly tested on Ubuntu
and GNU Guix System
.
Contributions
The package is part of ELPA which means that if you want to contribute you must have a copyright assignment.
Acknowledgments
This package wouldn't have been were it is today without these contributors.
Code contributors
- rosetail
- protesilaos
- Stefan Monnier
- dpettersson
- Greg Pfeil
Idea contributors
- rosetail for all the great ideas and improvements to the package. Without those contributions
detached.el
would be a less sophisticated package. - Troy de Freitas for solving the problem of getting
detached.el
to work withfilenotify
on macOS. - Daniel Mendler for helping out in improving
detached.el
, among other things integration with other packages such asembark
andconsult
. - Ambrevar who indirectly contributed by inspiring me with his yes eshell is my main shell. It was through that I discovered his package-eshell-detach which got me into the idea of using
dtach
as a base for detached shell commands.
Old versions
detached-0.10.0.tar.lz | 2022-Nov-23 | 52.5 KiB |
detached-0.9.2.tar.lz | 2022-Nov-01 | 50.2 KiB |
detached-0.9.1.tar.lz | 2022-Sep-26 | 41.2 KiB |
detached-0.9.0.tar.lz | 2022-Sep-16 | 41.1 KiB |
detached-0.8.1.tar.lz | 2022-Sep-08 | 37.0 KiB |
detached-0.8.0.tar.lz | 2022-Sep-02 | 36.7 KiB |
detached-0.7.tar.lz | 2022-May-21 | 36.0 KiB |
News
1. Development
2. Version 0.10.1 (2022-11-25)
- Add fallback validation of sessions using timer-based solution. This should make sure that sessions are verified even if notify-watch fails to do it due to raise condition.
3. Version 0.10.0 (2022-11-23)
- Add
edit-and-run
command, which is convenient when a session command needs to be tweaked before re-running. - Fix bug caused by incorrect adaptation to detached-local-session
- Implement mean and std duration for sessions
- Make tail commands work on macOS
- Rewrite core code related to creating, starting, and attaching to sessions.
- New implementation of kill command. Instead of determining process IDs and sending termination signal to all of them detached attach to the session behind the scenes and sends a termination.
- Add new keymap
detached-session-map
which could be bound by the user to a relevant key to operate on sessions in the current buffer. The idea is to have a global keymap for that which should work in any major-mode supported by detached.
4. Version 0.9.2 (2022-11-01)
- Use
display-buffer
indetached-list
in order for users to more easily customize the display of the buffer. - Add concept of
filters
to quickly apply a set of narrow criteria. - Speed up initialization of sessions.
- Narrow criteria are persistent and applied when resuming detached list sessions.
- Add narrow criterion which searches through the output of all sessions in detached list which matches a regexp.
- Add database hook in order for detached list to update automatically when a session is updated.
- Add annotation property to session. This allows users to add a custom annotation to a session, see it as a possibility to document a session for the future.
- Implement mode-line indicator for criteria in
detached-list
- Add command for initializing a session directory
- Add integration with
nano-modeline
- Add
detached-describe-session
command - Add unique narrow criteria
5. Version 0.9.1 (2022-09-22)
- Fix bug with detaching in
shell
6. Version 0.9.0 (2022-09-16)
- Add
detached-list
user interface. It is an alternative user interface where it is easier to manage thedetached
sessions. Commands to mark, toggle, and narrow based on different criteria are available. - Rework session initialization and validation. This also improves the way Emacsen co-operates in term of updating sessions that become inactive. The new approach is described under
architecture
innotes.org
. - Improved handling of remote sessions. The package will only try to initialize sessions that are accessible when package loads. Other active sessions it will wait until a remote connection has been established before they are being watched.
7. Version 0.8.1 (2022-09-08)
- Adjust default location for sessions, from a directory under
temporary-file-directory
touser-emacs-directory
. This change is for security reasons in case the user shares the computer with other users. By default sessions will not be removed on reboots, the user can useM-x detached-delete-sessions
to clear old sessions manually. - Fix bug when initializing
detached-session-directory
, this worked incorrectly when users initializeddetached
while being on a remote host. - Add an
initial-mode
property to a session. This enables sessions that were created usingcreate
mode instead ofcreate-and-attach
to inherit this property when a session is being rerun.
8. Version 0.8.0 (2022-09-02)
- Change to semantic versioning
- Update expectations on
detached-terminal-data-command
command, user should now set it manually. This approach works better since it doesn't cause any conflict when remote host and local host operating system differs. - Improved handling of remote sessions. The package will not eagerly try to watch sessions, it waits until user requests or that it finds an open connection to the remote host.
- Detached tries to guess the
detached-terimal-data-command
based on operating system, if that's not possible the user needs to set it manually. Currently detection ofgnu/linux
anddarwin
is supported.
… …