Git repos backed by your own storage. Push to S3, R2, Tigris, MinIO, Postgres, SFTP, NFS, or local disk.
Project description
Trunks
The most powerful open-source POSIX-compatible, Git-native filesystem.
Trunks turns any backend into a Git-compatible remote. Point it at S3, R2, Tigris, GCS, Azure Blob, MinIO, Postgres, SFTP, a fileshare, or local disk. You get branches, commits, refs, push, pull, the whole protocol. No Git server. No service to operate. No control plane. No repo copy per workspace.
Applications write normal files. Developers run normal Git.
git remote add origin trunks://primary/my-app
git push -u origin main
That git push invokes git-remote-trunks, writes commit objects, and advances the branch ref straight into your configured storage. No GitHub. No Git server in between.
Install
pip install trunks
npm install @layerbrain/trunks
Quick Start
trunks storage add --name primary --backend local --path /tmp/trunks-store
mkdir my-app
cd my-app
trunks mount --repo my-app
echo "hello" > README.md
git add README.md
git commit -m "init"
git push
That's it. trunks mount initializes .git, wires origin -> trunks://primary/my-app, and sets the current branch to push there. Another machine can run git clone trunks://primary/my-app.
S3 here is whatever you have: GCS, Azure Blob, R2, Tigris, MinIO, Postgres, SFTP, fileshare, local disk. Trunks makes each one act like a Git remote. No Trunks server in the middle. No Git server in the middle.
How It Works
Trunks stores repos as Git-shaped objects in your backend. Real Git owns .git; Trunks only participates when Git invokes git-remote-trunks for a trunks:// remote.
s3://company-trunks/trunks/my-app.trunk/
├── objects/ blobs, trees, commits (sha-keyed, immutable)
├── refs/heads/ branch pointers (compare-and-swap)
└── journals/ crash recovery
- When a process writes a file, Trunks hashes the bytes into objects.
- When Git pushes, the remote helper uploads missing objects.
- When the push completes, Trunks advances the branch ref with compare-and-swap.
That's the whole protocol. Two writers can't clobber each other. Crashes don't corrupt state. Two machines sync by pointing at the same prefix.
Why It Exists
Modern software moves a lot of files across laptops, CI, sandboxes, servers, and storage buckets. Without version history, every write is just another blob that can overwrite the last one.
Git solves history. But Git expects a hosted server, a clone per agent, and has no native story for large or many repos.
Trunks keeps Git's commit objects and refs, drops the server, and writes straight to storage you already use. The bucket is the remote. So is the database, or the SFTP host. Work happens on branches, files are saved as commits, and diffs stay reviewable like any normal Git workflow.
What You Get
- Real files. Tools read and write with
cat,vim,grep,npm,python. The SDKs are a convenience, not a requirement. - Real Git.
git log,git diff,git blame,git push, andgit clone trunks://...operate through Git's standard remote-helper protocol. - Real concurrency. Branches are pointers. Different branches never collide. Same branch is one CAS. One writer wins, the others retry.
- Real backends. S3, R2, Tigris, GCS, Azure Blob, MinIO, Postgres, SFTP, fileshare, local disk. Each one passes the same multi-commit, branch, merge, and CAS-conflict contract test.
- Real scale. Virtual mode mounts a 100GB repo without materializing it.
- Real portability. Same commands on a laptop, EC2, Lambda, Cloudflare Worker, Modal sandbox, Daytona, CI runner.
Mount Modes
trunks mount --repo my-app --path ./my-app
trunks mount --repo my-app --path ./my-app --watch
trunks mount --repo my-app --path ./my-app --storage primary
trunks mount --repo big-repo --path ./big-repo --mode virtual
Default is plain files. --watch keeps a journal so a crash doesn't lose work. --mode virtual sparsely materializes huge repos.
trunks mount also bootstraps git: it runs git init --initial-branch=main if .git is missing, adds origin -> trunks://<storage>/<repo> if no origin is configured, and sets the current branch upstream so plain git push works. With one primary storage profile this is automatic; pass --storage <name> if you have several. Pass --no-git to skip the bootstrap.
Branches Are Pointers
One branch per task. Creating one is a single ref write. No copy.
trunks branch create --name feature/auth --from main
trunks branch switch --name feature/auth
trunks shell
git add .
git commit -m "update auth"
git push
Two writers on different branches don't collide. Two writers on the same branch race a compare-and-swap. One wins. The other retries.
Git Without GitHub
git remote add origin trunks://primary/my-app
git push -u origin main
git clone trunks://primary/my-app ./my-app-copy
trunks://primary/my-app resolves the configured primary storage profile and the my-app repo namespace. Same commits. Same refs. Your storage.
Mirrors
Add one primary storage profile and any number of explicit mirrors:
trunks storage add primary --backend s3 --bucket company-trunks
trunks storage add backup --mirror --backend s3 --bucket company-trunks-backup
git push origin main
Pushes to trunks://primary/<repo> write through a strict multi-backend path. The primary ref update is CAS-fenced, and configured mirrors receive the same objects, branch refs, and push trigger refs. Inline URLs such as trunks+s3://bucket/path target only that one backend.
Python
Use the Python SDK when your application wants files and commits without shelling out.
from trunks import Trunk
with Trunk(name="my-app") as trunk:
trunk.write("task.md", b"Fix auth\n")
trunk.commit(message="update auth")
trunk.push()
async with Trunk(name="my-app") as trunk:
await trunk.write("task.md", b"Fix auth\n")
await trunk.commit(message="update auth")
await trunk.push()
Node
import { Trunks } from "@layerbrain/trunks";
const trunks = new Trunks();
const fs = await trunks.mount({ repo: "my-app", path: "./my-app", watch: true });
await fs.write("task.md", "Fix auth\n");
await fs.checkpoint("update auth");
await fs.push();
Resource API
CLI, Python, and Node share the same resource API.
trunks repo create --name my-app --backend s3://company-trunks --json
trunks branch list --json --limit 20 --offset 0
client = Trunks(cwd="./my-app")
client.branches.create(name="feature/auth", from_ref="main")
const trunks = new Trunks();
await trunks.branches.create({ name: "feature/auth", from: "main" });
List calls return the same envelope everywhere:
{
"object": "list",
"data": [],
"limit": 20,
"offset": 0,
"total_count": 0,
"has_more": false
}
Where To Next
| Want to do | Page |
|---|---|
| Walk through your first repo | Tutorial |
| See every command | CLI reference |
| Use Python | Python SDK |
| Use Node | Node SDK |
| Resource shapes | Resources |
| Pick a backend | Backends |
| Use Trunks with agent frameworks | Agents |
| Run CI on your own infrastructure | Trunks Actions |
| Understand the bytes | Architecture |
Backend guides:
- S3-compatible storage
- Azure Blob Storage
- Google Cloud Storage
- Postgres
- SFTP
- Local disk and fileshares
Examples:
Actions:
MIT licensed.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file trunks-1.2.13.tar.gz.
File metadata
- Download URL: trunks-1.2.13.tar.gz
- Upload date:
- Size: 208.6 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5ae0fac963b31cdba8335fe1f29070af9fac6e3babaf3b153aaf81e06d90f6ab
|
|
| MD5 |
45fba954531fd371f31d3f69bd2e82fc
|
|
| BLAKE2b-256 |
2124c04fe5927e787a4411162c8c192fcabe748e1015ce9aea1f3893382f5a3e
|
File details
Details for the file trunks-1.2.13-py3-none-any.whl.
File metadata
- Download URL: trunks-1.2.13-py3-none-any.whl
- Upload date:
- Size: 222.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.13
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3becd896c362b8a8a08dd0674632ba76c4f420f3d819026015987bd3cac0f06e
|
|
| MD5 |
cc657f6c28e8d951a8f74c800be7c64c
|
|
| BLAKE2b-256 |
e706a3ba9d4ffce9249647c37af404edcd10857eccd09778a1141000e62e6563
|