A high-performance TRAMP backend for Emacs that uses a binary RPC server instead of parsing shell command output.
Traditional TRAMP works by piping shell commands over SSH and parsing their output. This approach is robust but slow, especially for operations that require many round-trips (like directory listings or VC operations).
TRAMP-RPC replaces this with a lightweight Rust server that runs on the remote host. Emacs communicates with it using MessagePack-RPC over SSH, resulting in significantly faster file operations.
| Aspect | Original TRAMP | TRAMP-RPC |
|---|---|---|
| Communication | Shell commands + parsing | MessagePack-RPC protocol |
| Latency | Multiple round-trips | Single round-trip |
| Batching | Not supported | Multiple ops per request |
| Shell dependency | Required on remote | Not needed |
| Binary required | None | ~1MB Rust server |
- Fast file operations via binary RPC protocol (2-57x faster than shell-based TRAMP)
- Async process support (
make-process,start-file-process) - Full VC mode integration (git, etc.)
- Automatic binary deployment (download or build from source)
- Support for Linux and macOS (x86_64 and aarch64)
- Batch/pipelined requests for reduced round-trip latency
- PTY support for terminal emulators (vterm, eat)
- Emacs 30.1 or later
msgpack.elpackage (installed automatically from MELPA)- SSH access to remote hosts
- Remote host running Linux or macOS (x86_64 or aarch64)
(use-package tramp-rpc
:ensure t)
- Clone this repository:
git clone https://github.com/ArthurHeymans/emacs-tramp-rpc.git - Add to your Emacs init file:
(add-to-list 'load-path "/path/to/emacs-tramp-rpc/lisp") (require 'tramp-rpc)
Access remote files using the rpc method:
/rpc:user@host:/path/to/file
On first connection, the server binary is automatically obtained and deployed:
- Download from GitHub Releases (fastest, ~1MB download)
- Build from source if Rust is installed and download fails
The binary is cached locally in ~/.emacs.d/tramp-rpc/ and deployed to ~/.cache/tramp-rpc/ on the remote host.
| Command | Description |
|---|---|
M-x tramp-rpc-deploy-status |
Show binary deployment status |
M-x tramp-rpc-deploy-clear-cache |
Clear local binary cache |
M-x tramp-rpc-deploy-remove-binary |
Remove binary from remote |
┌─────────────┐ SSH/MessagePack-RPC ┌──────────────────┐ │ Emacs │ ◄────────────────────► │ tramp-rpc-server │ │ (tramp-rpc) │ │ (Rust) │ └─────────────┘ └──────────────────┘
The server exposes RPC methods for:
| Category | Methods |
|---|---|
| File | file.stat, file.read, file.write, file.copy, … |
| Directory | dir.list, dir.create, dir.remove, ~dir.completions~ |
| Process | process.run, process.start, process.read, … |
| PTY | process.start_pty, process.read_pty, process.resize_pty, … |
| System | system.info, system.getenv, system.statvfs |
User connects via /rpc:host:/path
│
▼
Remote already has binary? ──yes──► Done
│ no
▼
Check local cache (~/.emacs.d/tramp-rpc/VERSION/ARCH/)
│
├─ Found ──────────────────► Transfer to remote
│
▼
Download from GitHub Releases
│
├─ Success ────────────────► Cache locally, transfer to remote
│
▼
Build with cargo (if Rust installed)
│
├─ Success ────────────────► Cache locally, transfer to remote
│
▼
Error with helpful instructions
| Platform | Architecture | Status |
|---|---|---|
| Linux | x86_64 | ✓ |
| Linux | aarch64 | ✓ |
| macOS | x86_64 | ✓ |
| macOS (Apple Silicon) | aarch64 | ✓ |
If automatic deployment fails, download manually from GitHub Releases and extract to:
~/.emacs.d/tramp-rpc/VERSION/ARCH/tramp-rpc-server
For example:
~/.emacs.d/tramp-rpc/0.1.0/x86_64-linux/tramp-rpc-server
# Build for current platform
nix build
# Cross-compile for specific target
nix build .#tramp-rpc-server-x86_64-linux
nix build .#tramp-rpc-server-aarch64-linux
# Development shell with all tools
nix develop
cd server
cargo build --release
The binary will be at target/release/tramp-rpc-server.
# Install target
rustup target add aarch64-unknown-linux-gnu
# Build (requires appropriate linker)
cargo build --release --target aarch64-unknown-linux-gnu
;; Prefer building from source over downloading (default: nil)
(setq tramp-rpc-deploy-prefer-build t)
;; Local cache directory (default: ~/.emacs.d/tramp-rpc/)
(setq tramp-rpc-deploy-local-cache-directory "~/.cache/tramp-rpc-binaries")
;; Remote installation directory (default: ~/.cache/tramp-rpc)
(setq tramp-rpc-deploy-remote-directory "~/.local/bin/tramp-rpc")
;; Disable automatic deployment (default: t)
(setq tramp-rpc-deploy-auto-deploy nil)
;; Use different GitHub repo for downloads
(setq tramp-rpc-deploy-github-repo "myuser/my-fork")
;; Download timeout in seconds (default: 120)
(setq tramp-rpc-deploy-download-timeout 60)
Run M-x tramp-rpc-deploy-status to see:
- Current version
- Local architecture
- Whether Rust/cargo is available
- Cached binaries
- Download URLs
If you experience issues with diff-hl in dired buffers on remote hosts:
(setq diff-hl-disable-on-remote t)
The server binary is deployed using standard SSH (sshx method). Ensure you can connect to the remote host with:
ssh -o BatchMode=yes user@host echo success
If GitHub downloads fail (corporate firewall, etc.), you can:
- Install Rust and let tramp-rpc build locally:
curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh
- Download manually and place in the cache directory (see above)
- Pre-deploy to remote hosts using your own method
TRAMP-RPC uses MessagePack-RPC over stdin/stdout with length-prefixed binary framing.
TRAMP-RPC originally used JSON-RPC with newline-delimited messages. This was changed to MessagePack-RPC for several reasons:
| Aspect | JSON-RPC (old) | MessagePack-RPC (current) |
|---|---|---|
| Binary data | Base64 encoded (~33% overhead) | Native binary type (no overhead) |
| Message framing | Newline-delimited | Length-prefixed binary |
| Non-UTF8 filenames | Required escaping/encoding | Native binary support |
| Boolean false | JSON false |
MessagePack false (0xc2) |
| Message size | Larger (text format) | ~33% smaller (binary format) |
The switch eliminates encoding overhead for file transfers and fixes edge cases with non-UTF8 filenames that are valid on Unix filesystems.
MessagePack is a binary serialization format that provides:
- Native binary data support (no base64 encoding needed for file content)
- ~33% smaller messages compared to JSON
- Faster serialization/deserialization
- Proper distinction between null and false values
Each message is prefixed with a 4-byte big-endian length:
Request (conceptual structure):
((version . "2.0")
(id . 1)
(method . "file.stat")
(params . ((path . "/etc/passwd"))))
Response (conceptual structure):
((version . "2.0")
(id . 1)
(result . ((type . "file")
(size . 2847)
(mode . 420))))
File content, paths, and process I/O are transmitted as raw binary (MessagePack bin type), eliminating encoding overhead and ensuring correct handling of:
- Non-UTF8 filenames
- Binary file content
- Arbitrary byte sequences in process output
TRAMP-RPC significantly outperforms traditional TRAMP for most operations:
| Operation | Speedup vs SSH |
|---|---|
| file-exists | 13.7x |
| file-write | 56.6x |
| directory-files | 10.5x |
| copy-file | 8.7x |
| git commands | 2-5x |
Batch operations provide additional 3-6x speedup by combining multiple requests into a single round-trip.
For detailed benchmarks and an in-depth technical comparison with original TRAMP, see Technical Comparison.
TRAMP-RPC includes a comprehensive test suite using Emacs ERT (Emacs Lisp Regression Testing).
| Category | Tests | Requirements |
|---|---|---|
| Protocol | 6 | None (pure Elisp) |
| Conversion | 2 | None (pure Elisp) |
| Server Integration | 4 | RPC server binary |
| Remote File Ops | 18 | SSH + RPC server |
./test/run-tests.sh --protocol
Or directly with Emacs:
emacs -Q --batch -l test/tramp-rpc-mock-tests.el \
--eval "(ert-run-tests-batch-and-exit \"^tramp-rpc-mock-test-protocol\")"
./test/run-tests.sh --mock
This runs protocol tests plus server integration tests that communicate directly with the RPC server (no SSH needed).
TRAMP_RPC_TEST_HOST=your-remote-host ./test/run-tests.sh --remote
Or:
emacs -Q --batch \
-l test/tramp-rpc-tests.el \
--eval "(setq tramp-rpc-test-host \"your-remote-host\")" \
--eval "(ert-run-tests-batch-and-exit \"^tramp-rpc-test\")"
test/tramp-rpc-tests.el– Full ERT test suite for remote operationstest/tramp-rpc-mock-tests.el– CI-compatible tests (no SSH required)test/run-tests.sh– Test runner script
The GitHub Actions workflow runs:
- Protocol Tests – MessagePack-RPC encoding/decoding (no server)
- Server Integration Tests – Direct server communication (Rust binary)
- Full Test Suite – (main branch only) Complete tests via SSH to localhost
This project is licensed under the GNU General Public License v3.0 or later – see the LICENSE file for details.
Contributions welcome! Please ensure code passes cargo clippy and cargo test before submitting.
For Emacs Lisp changes, also run the test suite:
./test/run-tests.sh --mock