We provide some tools thereafter described, aiming at reducing the amount of windows, and ease to some extent Acme's scriptability, pushing lib/acme.rc a bit further.
This is perhaps the most useful, at least to me. Most others tools were developed
mainly in order to create Run
. It aims at implementing a typical
"build/run" feature, more precisely, it:
- Puts (save) all opened files (see
XPutall
); - Opens if necessary an output buffer/window (by default, it
targets the
+Buffer
window, described in the next subsection), and clearing its content (seeClear
) if necessary - Executes (see
Exec
,To
) the program related to the current buffer (there's an ad-hoc long list ofif/else
inRun
to determine what to do); - Redirecting stderr/stdout to the output buffer along the way, eventually tweaking stderr's output by piping it to a program (this for instance allows to rewrite some error message to be more acme(1)/plumber(4) compliant).
Of course, CLI arguments can be provided to the program to be executed,
for instance: Run -- -o foo -x bar baz
will feed the program
-o foo -x bar baz
as CLI arguments.
A convenient -m <target>
option will execute make <target>
in
the first directory containing a Makefile
, starting from the current
directory, and going up in the tree until it reaches /
. For instance, when:
- middle clicking
Run -m tests
on the tagline of a file/home/user/gits/project/dir/subdir/foo.go
; - assuming a
/home/user/gits/project/Makefile
exists; - will run
make tests
in/home/user/gits/project/
(of course, surrounded by all previous boilerplate: saving files, opening output buffer, etc.)
In practice, I've found that having a single +Errors window, with automatic content flushing, is more suitable than spanning one +Errors window per directory, with no flushing.
NOTE: acme's default seem however most reasonable: better be verbose, and let the user filter as needed.
By convention, we called this window the +Buffer window, being the only window whose name suffixed by +Buffer. Its access is automatically managed via the Getids command, described thereafter.
This window's tagline can be used for running global commands, such as the session-related ones, also described later on.
Getids is a key script, on which most other scripts are built upon. It allows to easily target:
- the +Buffer window (creating it when necessary);
- multiple windows (X//-like looping);
- current window (the default).
It outputs a list of window IDs, one per line, from either:
- a number (window ID);
- an awk(1) regular expression;
- -a for automatically targeting the +Buffer window;
- nothing (current window).
As an example, a script like Do that runs a command from a window's tagline, thus allowing to run/chain acme(1) commands, has an optional argument that is forwarded to Getids:
# Clean current window
(tagline)$ Do 'Edit ,d'
# Clean the content of all +Errors windows
(sh|tagline)$ Do 'Edit ,d' '\+Errors$'
# Write is used to store data in special buffer's files (9P),
# similar to lib/acme.rc's winwrite().
#
# The following writes and closes all windows whose name contain
# "/project42/", cleaning the +Buffer window's content thereafter,
# creating it if necessary.
(sh|tagline)$ Write ctl put '\/project42\/' && \
Write ctl delete '\/project42\/' && \
Do 'Edit ,d' -a
Again aiming at reducing windows proliferation, those three small scripts load/dump current windows' states:
# Put all files and creates a dump file in
# $HOME/acme.dumps/project42.dump with current state
(sh|tagline)$ XPutall && XDump project42 # optional .dump extension
# Restore state from $HOME/acme.dumps/projects42.dump,
# closing all opened windows beforehand
(sh|tagline)$ XLoad project42.dump # optional .dump extension
# Dump current state to $HOME/acme.dumps/project42.dump and
# load session $HOME/acme.dumps/website.dump
(sh|tagline)$ Switch project42 website # optional .dump extension
NAME
+
SYNOPSYS
+ [-h]
DESCRIPTION
+ indents text from stdin by prepending a tab to
every line; resulting output on stdout.
NAME
++
SYNOPSYS
++ [-h]
DESCRIPTION
++ indents text from stdin by prepending 2 tabs to
every line; resulting output on stdout.
NAME
+++
SYNOPSYS
+++ [-h]
DESCRIPTION
+++ indents text from stdin by prepending 3 tabs to
every line; resulting output on stdout.
NAME
-
SYNOPSYS
- [-h]
DESCRIPTION
- unindents text from stdin by removing a heading tab from
every line; resulting output on stdout.
NAME
--
SYNOPSYS
-- [-h]
DESCRIPTION
-- unindents text from stdin by removing 2 heading tabs from
every line; resulting output on stdout.
NAME
---
SYNOPSYS
--- [-h]
DESCRIPTION
--- unindents text from stdin by removing 3 heading tab from
every line; resulting output on stdout.
NAME
Acme
SYNOPSYS
Acme [-h]
Acme [args...]
DESCRIPTION
Acme is a basic wrapper to launch acme(1). It starts
the plumber if needed, set a few convenient environment
variable, and will periodically create a
/home/mb/acme.dumps/autodump.dump using "XDump autodump".
NAME
Auto
SYNOPSYS
Auto [-h]
Auto [-k] [cmd]
Auto [-n seconds] [-r] [cmd]
DESCRIPTION
Auto will periodically run the given command from the tagline
where Auto was started.
-n allows to specify a running period (default: 10 seconds)
-k will kill either all automatic jobs associated with the concerned
window, or all the ones associated with the given command.
-r will first kill all automatic jobs associated with the given
command before (re)starting it
Unless for -k, where we look for all commands, cmd defaults
to "Run".
EXAMPLES
Running 'Auto -n 5 -r "Run -- -k"' from a window will:
- kill all "Run -- -k" jobs associated to $winid;
- run "Run -k" from the tagline every 5 seconds.
Running 'Auto' from a window will:
- Execute 'Run' from that tagline every 10 seconds.
NAME
Clear
SYNOPSYS
Clear [-h]
Clear [-a|id|pattern]
DESCRIPTION
Clear removes all content from the windows pointed by -a|id|pattern,
which is forwarded to Getids, thus defaulting to $winid (current
window).
NAME
Delall
SYNOPSYS
Delall [-h]
Delall
DESCRIPTION
Delall deletes all existing windows.
NAME
Do
SYNOPSYS
Do [-h]
Do <cmd> [-a|id|pattern [-k]]
DESCRIPTION
Do runs a command <cmd> in the taglines of the windows pointed by
[-a|id|pattern], which is forwarded to Getids, thus defaulting to
$winid (current window).
Similar to https://github.com/mkhl/cmd/blob/master/acme/acmeeval/main.go,
but written in sh(1), orthogonal with Getids's behavior (+Buffer
management, and X// like looping on filenames patterns.
If -a is augmented with a -k, we will keep the original directory
of the +Buffer window. Use case is the autodump in the Acme
script, which systematically reset the directory, and thus messes
up regularly e.g. the "clickable links" of the compiler output.
EXAMPLES
$ Do 'Edit ,d' -a
Clear '+Buffer' window's body
NAME
Exec
SYNOPSYS
Exec [-h]
Exec [-t [-a|id|pattern]] [-e [-a|id|pattern]]<cmd>
[-o [-a|id|pattern]] [-p outp] [-q errp] [-r both] <cmd>
DESCRIPTION
Exec runs a command, sending stdout/stderr to acme buffers,
defaulting to +Buffer.
The buffers are automatically cleaned before hand.
-t selects stdout buffer, forwarding [-a|id|pattern] to Getids
-t selects stderr buffer, forwarding [-a|id|pattern] to Getids
-o is a shortcut to set both -t and -e.
-p and -q allows to pipe stdout/stderr through the given shell
commands, before forwarding the result to To, which will be in
charge of writing the content to an acme(1) buffer.
-r allows to set both outp and errp at once.
NAME
Getall
SYNOPSYS
Getall [-h]
Getall
DESCRIPTION
Getall updates all windows from filesystem (runs Get everywhere).
NAME
Getfn
SYNOPSYS
Getfn [-h]
Getfn [-a|id|pattern]
DESCRIPTION
Getfn prints the filenames of windows pointed by [-a|id|pattern],
which is forwarded to Getids, thus defaulting to $winid
(current window).
This is a nice shortcut to avoid callers to handle multi-lines tag.
NAME
Getids
SYNOPSYS
Getids [-h]
Getids [-a|id|pattern]
DESCRIPTION
This is a key script; goals are:
- to transform a pattern to a list of acme buffer IDs;
- allow easy targetting the special '+Buffer' window;
- to default to current window ($winid).
When called with no arguments, simply display $winid, failing
with an error code if it is not set.
When called with -a, look for a +Buffer window, creating it
if needed, and display its id.
When called with a number, assume an existing windo id, and
display it.
Otherwise, assume argument is an awk(1) pattern, and display
all matching buffers IDs, one per line.
NAME
Godoc
SYNOPSYS
Godoc [-h]
Godoc [args...]
DESCRIPTION
Godoc executes go doc with the given arguments. This
is a shortcut for:
$ Exec go doc [args...]
NAME
Goimports
SYNOPSYS
Goimports [-h]
Goimports
DESCRIPTION
Goimports will run 'goimports -w' on every opened
'.go' file, and update ("get") the buffers.
NAME
Hook
SYNOPSYS
Hook [-h]
DESCRIPTION
Hook registers a single hook $HOME/acme.hook to
all existing buffers, and to all buffers about
to future buffer.
win(1) buffers are systematically ignored.
It also registers a pid in $HOME/acme.hook.pid for Unhook.
BUGS
For now at least, the idea is to have a central script
acting as a hook, that could eventually delegate things
to other scripts in case there's a need for finer filtering.
NAME
Iswin
SYNOPSYS
Iswin [-h]
Iswin <id>
DESCRIPTION
Iswin test if the buffer pointed by the given id seems to correspond
to a win(1) buffer.
It exists with status 0 if it's the case, otherwise with exit
status 1, following ordinary sh(1) conventions.
NAME
Mv
SYNOPSYS
Mv [-h]
Mv <to> [-a|id|pattern]
DESCRIPTION
Move (mv(1)) current file, or the one pointed by [-a|id|pattern],
to a new one, on the filesystem that is, and updating acme's
buffer's name accordingly.
If 'to' doesn't start with a leading '/', renaming is performed
relatively instead of absolutely.
If the current buffer points to a file (by opposition to a directory),
and 'to' ends with a '/', then the file pointed by the current buffer
is moved to the directory pointed by 'to'.
If 'to' describes a missing path, intermediate directories
are created (mkdir(1) -p); existing file is overwritten,
following mv(1)' semantic.
EXAMPLE
In each of the example below, the source filename is
to be interpreted as the current buffer's name.
Renaming "/home/foo/bar.c" to "/home/foo/baz.c":
Mv /home/foo/baz.c
Mv baz.c
Renaming "/tmp/bar/" to "/tmp/foo/":
Mv foo
Mv foo/
Renaming "foo.c" to "foo/bar/foo.c":
Mv foo/bar/
Renaming "foo.c" to "foo/bar":
Mv foo/bar
BUGS:
We may want to allow regexp(7) with capture, so as to allow
renaming multiple files/buffers at once.
We require realpath(1) to be available to clean up the path,
e.g. in case of 'Mv ../../'.
NAME
Open
SYNOPSYS
Open [-h]
Open [-m] [-n] [-p] [-u] [-x] [-g] [-q] <name> [pattern]
Open [-r] [name]
DESCRIPTION
Open given file with acme, creating it as an empty file
if necessary.
If -m is specified, move back to / and open the first file
that match given name. Note that this mimick Run's -m behavior,
hence the option's name.
If -n is specified, nor the file nor its parent directory will
be created on disk/loaded from disk.
If -p is specified, print opened files' acme windows' IDs,
one per line.
If -u is specified, Open will only open the file if no buffer
matches the given pattern (unique). If no pattern is specified,
the name will be used as a pattern instead. If there's already
a corresponding buffer opened, then some of it will be
made visible by writing show its acme//ctl.
-x automatically triggers a chmod +x on the file.
With -g, the name is interpreted as a pattern
to be grep(1)'d in $HOME/acme.files: the first existing
file matching the pattern is opened. If no match, exit
with failure. -g implies -u.
By default, Open will focus on the buffer being opened
(see acme(4), 'ctl' file, 'show' option): -q disables this
behavior.
-r registers a filename to $HOME/acme.files. If no name is
specified, then Open tries to collect one from a current
acme buffer, if any.
Paths starting with '/' are considered absolutely, otherwise,
relatively.
Nonexistent paths are created (mkdir(1) -p).
Opened filenames are systematically added to $HOME/acme.files,
which is cleaned via sort(1) -u, and only contains absolute
paths.
EXAMPLES
The following will create a '+Buffer' window if there's no
window with filename suffixed by +Buffer; this will be
considered a "virtual" file, not tied to an on-disk file:
Open -n -u '+Buffer' '\+Buffer$'
NAME
Rdoc
SYNOPSYS
Rdoc [-h]
Rdoc [cmd]
DESCRIPTION
Rdoc executes help(cmd) within R, via Exec. Conceptually,
this is a shortcut for (there's some extra-boilerplate):
$ Exec "echo 'help($cmd)' | Rscript /dev/stdin"
BUG
Relies on a sed(1) with the [:print:] character class for
cleanup.
NAME
Read
SYNOPSYS
Read [-h]
Read [-a|id|pattern]
DESCRIPTION
Read reads the body of the buffer pointed by [-a|id|pattern]
and prints it to stdout.
EXAMPLES
Display '+Buffer's content on stdout:
Read -a
Look for a buffer suffixed by '+Files'' and displays its
content on stdout:
Read '\+Files'
NAME
Rename
SYNOPSYS
Rename [-h]
Rename [-d dot] <from> <to> [-a|id|pattern]
DESCRIPTION
Rename renames an identifier in current selection of the
windows pointed by [-a|id|pattern], forwarded to Getids,
thus defaulting to $winid (default window), using the
famous sam(1) idiom described in sam_tut.pdf.
By default, renaming is performed on the whole file (dot=,);
the -d option allows to narrow the scope by setting the
dot.
EXAMPLE
Rename identifiers from "fs" to "ifs" on current window:
Rename fs ifs
Same, but perform renaming on current selection only:
Rename -d '' fs ifs
Rename -d . fs ifs
Same, but on all opened C files/headers:
Rename -d . fs ifs '\.[ch]$'
NAME
Rm
SYNOPSYS
Rm [-h]
Rm [-a|id|pattern]
DESCRIPTION
Rm removes buffers pointed by [-a|id|pattern], forwarded
to Getids, thus defaulting to $winid (current window),
and related files/directories from filesystems.
NAME
Run
SYNOPSYS
Run [-h]
Run [-t [-a|id|pattern]] [-e [-a|id|pattern]] [-m] [args]
Run [-o [-a|id|pattern]] [-m] [args]
DESCRIPTION
Run is a smart script running a command automagically inferred
from file/filename, redirecting its stdout/stderr to the buffers
pointed by -t/-e/-o, after having written all buffers.
User is expected to complete/adjust this script following
his need. In particular, the $errp variable allows to
pipe stderr to a command before sending to the error buffer
pointed by -e/-o: this can be used for instance to adjust
error messages to match plumbing rules.
The buffers are automatically cleaned before hand.
-t selects stdout buffer, forwarding [-a|id|pattern] to Getids
-t selects stderr buffer, forwarding [-a|id|pattern] to Getids
-o is a shortcut to set both -t and -e.
-m will force the use of a Makefile by going up to / from current
buffer's directory, until a Makefile is found.
As a convenience, the $ARGS environment variable is feed to the
program called to execute/compile the target. By comparison, the
[args] are forwarded to the executed/compiled target. This allows
specifying ad-hoc options.
EXAMPLES
1. By default, on a buffer pointing to a Makefile, 'Run tests' will:
- Put all buffers
- Run 'make tests', redirecting stdin/stdout to '+Buffer', creating
it if necessary, and cleaning it beforehand.
2. If executing 'Run -m tests' on a file /home/user/project/lib/lib.go,
and if a /home/user/project/Makefile file exist, will use that
makefile in a 'make tests'.
3. With the default setup, 'Run -- -m x' on a .c file will compile
the C file (assuming single file project) and execute the resulting
binary with '-m x' as CLI arguments.
4. 'ARGS=--show-trace Run' on a .nix file will attempt to execute the
.nix file, launching the nix interpreter with '--show-trace'.
./bin/See [-n]
NAME
Switch
SYNOPSYS
Switch [-h]
Switch <from> <to>
Switch -r <to> <from>
DESCRIPTION
Switch dumps current state to from and loads the state described
by to.
NAME
To
SYNOPSYS
To [-h]
To [-c] [-a|id|pattern]
DESCRIPTION
To redirects stdin/stdout to the first window pointed by
[-a|id|pattern]. We default to -a (+Buffer file).
If there's no buffer to write to, act as a cat(1).
If -c is specified, then the output buffer is cleared.
EXAMPLES
# Display stat(1)'s output for current buffer in +Buffer,
# creating it if necessary.
(tagline)$ stat $% | To
NAME
Unhook
SYNOPSYS
Unhook [-h]
DESCRIPTION
Unhook "unregisters" $HOME/acme.hook by killing all the
processes in the group of the process referenced in
$HOME/acme.hook.pid.
BUGS
This is crude.
For now, this is only useful because regular file dumping
mechanism is broken once someone listens to an event file.
So we need to unhook (before rehooking) each time we want
to dump. See XDump.
NAME
Write
SYNOPSYS
Write [-h]
Write <file> <data> [-a|id|pattern]
DESCRIPTION
Write writes data to buffer's special file (ctl/addr/body/data, etc.),
for all files pointed by [-a|id|pattern], forwarded to Getids,
thus defaulting to $winid (current window).
EXAMPLES
# Deletes all existing windows.
(sh|tagline)$ Write ctl delete '.*'
# Appends "hello" to current window's body
(sh|tagline)$ Write body hello
# Rename the current buffer (<data> is processed by printf(1))
(sh|tagline)$ Write ctl "name /path/to/newname\n"
NAME
XDel
SYNOPSYS
XDel [-h]
XDel [-a|id|pattern]
DESCRIPTION
XDel stores the content of the buffers pointed by [-a|id|pattern],
and then proceed to close the buffer, while storing
their filenames to the '+Files' buffer if necessary. The
'+Files' buffer is created is necessary.
NAME
XDump
SYNOPSYS
XDump [-h]
XDump <session> [-k]
DESCRIPTION
XDump dumps current state to <session>. <session> is automatically
appended a .dump suffix if needed, and is stored as a file in
$HOME/acme.dumps/
If -k is provided, it'll be forwarded to Do: the directory
name of the +Buffer window won't be altered.
NAME
XLoad
SYNOPSYS
XLoad [-h]
XLoad <session>
DESCRIPTION
XLoad loads the state described by <session>. <session> is automatically
appended a .dump suffix, and is looked as a file in $HOME/acme.dumps/.
We nuke all existing windows before hand, as Acme's Load just
add windows.
NAME
XPut
SYNOPSYS
XPut [-h]
XPut [-a|id|pattern]
DESCRIPTION
XPut stores files pointed by [-a|id|pattern], forwarded to Getids,
thus defaulting to $winid (current window).
EXAMPLES
# Writes all windows
(sh|tagline)$ XPut '.*'
NAME
XPutall
SYNOPSYS
XPutall [-h]
XPutall [-a|id|pattern]
DESCRIPTION
XPutall stores all buffers (runs Put everywhere).