66df66a200
Added a link to a related blog post for more context.
266 lines
8.7 KiB
Markdown
266 lines
8.7 KiB
Markdown
# webterm
|
|
|
|

|
|
|
|
Serve terminal sessions over the web with a simple CLI command. [Blog post](https://taoofmac.com/space/notes/2026/01/25/2030#seizing-the-means-of-production)
|
|
|
|
> **Credit and Inspiration:** This project was originally based on the genius [web](https://github.com/Textualize/web) package, which uses `xterm.js`. It has been rewritten to use a [ghostty-web](https://github.com/coder/ghostty-web)'s WebAssembly-based terminal emulator, which provides better performance and native theme support.
|
|
|
|
It is, for the moment, temporarily based on a [patched version of ghostty-web](https://github.com/rcarmo/ghostty-web), because the current version has bugs and feature gaps that I needed to fill.
|
|
|
|
Coupled with [`agentbox`](https://github.com/rcarmo/agentbox), you can use it to keep track of several containerized AI coding agents, since it provides an easy way to expose terminal sessions via HTTP/WebSocket with automatic reconnection support:
|
|
|
|

|
|
|
|
## Features
|
|
|
|
- **Web-based terminal** - Access your terminal from any browser
|
|
- **Mobile support** - Works on iOS Safari and Android with on-screen keyboard
|
|
- **Session reconnection** - Refresh the page and reconnect to the same session
|
|
- **Full terminal emulation** - Colors, cursor, and ANSI codes work correctly
|
|
- **Customizable themes** - 9 built-in themes (monokai, dracula, nord, etc.)
|
|
- **Custom fonts** - Configure terminal font family and size
|
|
- **Scrollback history** - Scroll back through terminal output (configurable)
|
|
- **Auto-sizing** - Terminal automatically resizes to fit the browser window
|
|
- **Live screenshots** - Dashboard shows real-time SVG screenshots of terminals
|
|
- **CPU sparklines** - Dashboard displays 30-minute CPU history for Docker containers
|
|
- **SSE updates** - Real-time screenshot updates via Server-Sent Events
|
|
- **Simple CLI** - One command to start serving
|
|
|
|
## Non-Features
|
|
|
|
- **No Authentication** - this is meant to be used inside a dedicated container, and you should set up an authenticating reverse proxy like `authelia`
|
|
- **No Encryption (TLS/HTTPS)** - again, this is meant to be fronted by something like `traefik` or `caddy`
|
|
|
|
## Installation
|
|
|
|
Install directly from GitHub:
|
|
|
|
```bash
|
|
pip install git+https://github.com/rcarmo/webterm.git
|
|
```
|
|
|
|
## Quick Start
|
|
|
|
### Serve a Terminal
|
|
|
|
Serve your default shell:
|
|
|
|
```bash
|
|
webterm
|
|
```
|
|
|
|
Serve a specific command:
|
|
|
|
```bash
|
|
webterm htop
|
|
```
|
|
|
|
### Options
|
|
|
|
Specify host and port:
|
|
|
|
```bash
|
|
webterm --host 0.0.0.0 --port 8080 bash
|
|
```
|
|
|
|
Customize theme and font:
|
|
|
|
```bash
|
|
webterm --theme dracula --font-size 18
|
|
webterm --theme nord --font-family "JetBrains Mono, monospace"
|
|
```
|
|
|
|
Available themes: `xterm` (default), `monokai`, `dark`, `light`, `dracula`, `catppuccin`, `nord`, `gruvbox`, `solarized`, `tokyo`.
|
|
|
|
Then open http://localhost:8080 in your browser.
|
|
|
|
## Session Dashboard
|
|
|
|
You can serve a dashboard with multiple terminal tiles driven by a YAML manifest:
|
|
|
|
```yaml
|
|
- name: My Service
|
|
slug: my-service
|
|
command: docker logs -f my-service
|
|
```
|
|
|
|
Run with:
|
|
|
|
```bash
|
|
webterm --landing-manifest landing.yaml
|
|
```
|
|
|
|
### Docker Watch Mode
|
|
|
|
Watch for Docker containers with the `webterm-command` label and dynamically add/remove terminal sessions:
|
|
|
|
```bash
|
|
webterm --docker-watch
|
|
```
|
|
|
|
When a container starts with the label, it automatically appears in the dashboard. When it stops, it's removed. Label values:
|
|
|
|
- `webterm-command: auto` (or empty) - Opens a PTY via Docker exec API (override with `WEBTERM_DOCKER_AUTO_COMMAND`)
|
|
- `webterm-command: <command>` - Runs the specified command
|
|
- `webterm-theme: <theme>` - Sets the terminal theme for that container (xterm, monokai, dark, light, dracula, catppuccin, nord, gruvbox, solarized, tokyo)
|
|
|
|
Example docker-compose.yaml:
|
|
|
|
```yaml
|
|
services:
|
|
myapp:
|
|
image: myapp:latest
|
|
labels:
|
|
webterm-command: auto # Opens bash in container
|
|
webterm-theme: monokai
|
|
|
|
logs:
|
|
image: myapp:latest
|
|
labels:
|
|
webterm-command: docker logs -f myapp # Shows logs
|
|
webterm-theme: nord
|
|
```
|
|
|
|
**Requires**: Docker socket access (`-v /var/run/docker.sock:/var/run/docker.sock`)
|
|
|
|
### Docker Compose Integration
|
|
|
|
Point to a docker-compose file; services with the label `webterm-command` become tiles (and `webterm-theme` applies there too):
|
|
|
|
```yaml
|
|
services:
|
|
db:
|
|
image: postgres
|
|
labels:
|
|
webterm-command: docker exec -it db psql
|
|
webterm-theme: gruvbox
|
|
```
|
|
|
|
Start with:
|
|
|
|
```bash
|
|
webterm --compose-manifest compose.yaml
|
|
```
|
|
|
|
In compose mode, the dashboard displays **CPU sparklines** showing 30 minutes of container CPU usage history (requires access to Docker socket at `/var/run/docker.sock`).
|
|
|
|
### Dashboard Features
|
|
|
|
- **Live screenshots** - Terminal thumbnails update in real-time via SSE when activity occurs
|
|
- **Dynamic updates** - In docker-watch mode, tiles appear/disappear as containers start/stop
|
|
- **CPU sparklines** - Mini charts showing container CPU usage (compose mode only)
|
|
- **Tab reuse** - Clicking the same tile reopens the existing browser tab
|
|
- **Auto-focus** - Terminals automatically receive keyboard focus on load
|
|
|
|
## CLI Reference
|
|
|
|
```
|
|
Usage: webterm [OPTIONS] [COMMAND]
|
|
|
|
Serve a terminal over HTTP/WebSocket.
|
|
|
|
COMMAND: Shell command to run in terminal (default: $SHELL)
|
|
|
|
Options:
|
|
-H, --host TEXT Host to bind to [default: 0.0.0.0]
|
|
-p, --port INTEGER Port to bind to [default: 8080]
|
|
-L, --landing-manifest PATH YAML manifest describing landing page tiles
|
|
(slug/name/command).
|
|
-C, --compose-manifest PATH Docker compose YAML; services with label
|
|
"webterm-command" become landing tiles.
|
|
-D, --docker-watch Watch Docker for containers with
|
|
"webterm-command" label (dynamic mode).
|
|
-t, --theme TEXT Terminal color theme [default: xterm]
|
|
Options: xterm, monokai, dark, light, dracula,
|
|
catppuccin, nord, gruvbox, solarized, tokyo
|
|
-f, --font-family TEXT Terminal font family (CSS font stack)
|
|
-s, --font-size INTEGER Terminal font size in pixels [default: 16]
|
|
--version Show the version and exit.
|
|
--help Show this message and exit.
|
|
```
|
|
|
|
## API Endpoints
|
|
|
|
| Endpoint | Description |
|
|
|----------|-------------|
|
|
| `/` | Dashboard (with manifest/docker-watch) or terminal view |
|
|
| `/ws/{route_key}` | WebSocket for terminal I/O |
|
|
| `/screenshot.svg?route_key=...` | SVG screenshot of terminal |
|
|
| `/cpu-sparkline.svg?container=...` | CPU sparkline SVG (compose mode) |
|
|
| `/tiles` | JSON list of current tiles (for dynamic dashboards) |
|
|
| `/events` | SSE stream for activity notifications |
|
|
| `/health` | Health check endpoint |
|
|
|
|
## Development
|
|
|
|
### Setup (Makefile-first)
|
|
|
|
```bash
|
|
git clone https://github.com/rcarmo/webterm.git
|
|
cd webterm
|
|
|
|
# Install with dev dependencies via Makefile
|
|
make install-dev
|
|
```
|
|
|
|
### Common tasks (use Makefile)
|
|
|
|
- Lint: `make lint`
|
|
- Format: `make format`
|
|
- Tests: `make test`
|
|
- Coverage (fail_under=78): `make coverage`
|
|
- Full check (lint + coverage): `make check`
|
|
- Bump patch version: `make bump-patch`
|
|
|
|
### Frontend Development
|
|
|
|
The terminal UI is built with a [patched version of ghostty-web](https://github.com/rcarmo/ghostty-web), which provides Ghostty's VT100 parser via WebAssembly with native theme/palette support. This replaces the original xterm.js dependency used in earlier versions.
|
|
|
|
Key improvements over xterm.js:
|
|
- Native theme colors passed directly to WASM (no runtime color remapping)
|
|
- Smaller bundle size (~0.67 MB vs ~1.16 MB)
|
|
- IME input support for CJK languages
|
|
- Better Unicode and complex script rendering
|
|
|
|
The pre-built bundle is committed to the repo, so users can `pip install` without needing Node.js.
|
|
|
|
To rebuild the frontend after modifying `terminal.ts`:
|
|
|
|
```bash
|
|
# Requires Bun (https://bun.sh)
|
|
bun install
|
|
bun run build
|
|
# Or simply:
|
|
make bundle
|
|
```
|
|
|
|
For development with auto-rebuild:
|
|
|
|
```bash
|
|
make bundle-watch
|
|
```
|
|
|
|
### Notes
|
|
|
|
- WebSocket protocol (browser ↔ server) is JSON: `["stdin", data]`, `["resize", {"width": w, "height": h}]`, `["ping", data]`.
|
|
- Frontend source is in `src/webterm/static/js/terminal.ts`.
|
|
- Screenshots use [pyte](https://github.com/selectel/pyte) for ANSI interpretation and custom SVG rendering.
|
|
- CPU stats are read directly from Docker socket using asyncio (no additional dependencies).
|
|
|
|
## Requirements
|
|
|
|
- Python 3.9+
|
|
- Bun
|
|
- Linux or macOS
|
|
|
|
## License
|
|
|
|
MIT License - see [LICENSE](LICENSE) for details.
|
|
|
|
## Related Projects
|
|
|
|
- [ghostty-web](https://github.com/rcarmo/ghostty-web) - Patched Ghostty terminal for the web (vendored fork with theme support)
|
|
- [ghostty-web upstream](https://github.com/coder/ghostty-web) - Original Ghostty terminal for the web
|
|
- [pyte](https://github.com/selectel/pyte) - PYTE terminal emulator (used for SVG screenshots)
|