ht - headless terminal
ht (short for headless terminal) is a command line program that wraps an arbitrary other binary (e.g. bash, vim, etc.) with a VT100 style terminal interface--i.e. a pseudoterminal client (PTY) plus terminal server--and allows easy programmatic access to the input and output of that terminal (via JSON over stdin/stdout). ht is built in rust and works on MacOS and Linux.
Use Cases & Motivation
ht is useful for programmatically interacting with terminals, which is important for programs that depend heavily on the Terminal as UI. It is useful for testing and for getting AI agents to interact with terminals the way humans do.
As the original motiving use case was, I was trying to use LLM agents for coding, and needed something like a headless browser but for terminals.
Terminals are one of the oldest and most prolific UI frameworks in all of computing. And they are stateful so, for example, when you use an editor in your terminal, the terminal has to manage state about the cursor location. Without ht, an agent struggles to manage this state directly; with ht, an agent can just observe the terminal like a human does.
Installing
Download and use the latest binary for your architecture.
Building
Building from source requires the Rust compiler (1.74 or later), and the Cargo package manager. If they are not available via your system package manager then use rustup.
To download the source code, build the binary, and install it in
$HOME/.cargo/bin run:
cargo install --git https://github.com/andyk/ht
Then, ensure $HOME/.cargo/bin is in your shell's $PATH.
Alternatively, you can manually download the source code and build the binary with:
git clone https://github.com/andyk/ht
cd ht
cargo build --release
This produces the binary in release mode (--release) at
target/release/ht. There are no other build artifacts so you can just
copy the binary to a directory in your $PATH.
Usage
Run ht to start interactive bash shell running in a PTY (pseudo-terminal).
To launch a different program (a different shell, another program) run ht <command> <args...>. For example:
ht fish- starts fish shellht nano- starts nano editorht nano /etc/fstab- starts nano editor with /etc/fstab opened
Another way to run a specific program, e.g. nano, is to launch ht without a
command, i.e. use bash by default, and start nano from bash by sending nano\r
("nano" followed by "return" control character) to the process input. See input
command below.
Default size of the virtual terminal window is 120x40 (cols by rows), which can
be changed with --size argument. For example: ht --size 80x24. The window
size can also be dynamically changed - see resize command below.
Run ht -h or ht --help to see all available options.
API
Communication with ht is performed via stdin, stdout and stderr.
ht uses simple JSON-based protocol for sending commands to its stdin. Each
command must be sent on a separate line and be a JSON object having "type"
field set to one the supported commands (below).
ht sends responses (where applicable) to its stdout, as JSON-encoded objects.
Diagnostic messages (notices, errors) are printed to stderr.
input
input command allows sending arbitrary input to a process running in the
virtual terminal as if the input was typed on a keyboard.
{ "type": "input", "payload": "ls\r" }
To send control characters (e.g., ctrl-c), include in your input json the ascii control character that the keyboard combo would generate (e.g., \x03 for ctrl-c). So to send control-c to the terminal, send the following JSON message to ht:
{ "type": "input", "payload": "\x03" }
This command doesn't produce any output on stdout.
getView
getView command allows obtaining a textual view of a terminal window.
{ "type": "getView" }
This command responds with the current view on stdout. The view is a multi-line string, where each line represents a terminal row.
{ "view": "[user@host dir]$ \n \n..." }
resize
resize command allows resizing the virtual terminal window dynamically by
specifying new width (cols) and height (rows).
{ "type": "resize", "cols": 80, "rows": 24 }
This command doesn't produce any output on stdout.
Testing on command line
ht is aimed at programmatic use given its JSON-based API, however one can play with it by just launching it in a normal desktop terminal emulator and typing in JSON-encoded commands from keyboard and observing the output on stdout.
rlwrap can be used to wrap stdin in a readline based editable prompt, which also provides history (up/down arrows).
To use rlwrap with ht:
rlwrap ht [ht-args...]
Design doc
Here is the original design doc we used to drive the project development.
Python and Typescript libs
Here are some experimental versions of a simple Python and Typescript libraries that wrap ht: htlib.py and a htlib.ts.
TODO: either pull those into this repo or fork them into their own htlib repo.
Possible future work
Would be great to update the interface to return the view with additional color and style information (text color, background, bold/italic/etc) also in a simple JSON format (so no dealing with color-related escape sequence either), and the frontend could render this using HTML (e.g. with styled pre/span tags, similar to how asciinema-player does it) or with SVG.
License
All code is licensed under the Apache License, Version 2.0. See LICENSE file for details.