freshdock
A modern Docker container auto-updater, built in Rust as a successor to Watchtower.
Status
Pre-alpha. Active planning, no usable code yet.
The current 0.0.1 release on crates.io exists only to reserve the name. The first working version is targeted for the read-only spike (Phase 1 in the roadmap). If you want to know when there is something to try, star or watch this repo.
If you need a working tool today, take a look at Cup (Rust, checker only), What's Up Docker (Go, full-featured), or Tugtainer (Go, with web UI).
Why freshdock exists
Watchtower — the de-facto Docker container auto-updater for years — was archived by its maintainers on 17 December 2025. Beyond being abandoned, the codebase ships an outdated embedded Docker SDK (API 1.25) that is incompatible with Docker Engine 29+ (which requires API ≥ 1.44).
The community has produced several forks and alternatives, but none combine the things that matter for a small homelab:
- Modern Docker API support that survives the next few engine releases.
- Updates that are safe — meaning a broken new image rolls back automatically instead of leaving you with a dead container.
- A footprint that does not require pulling a 100+ MB image to manage your other containers.
freshdock targets exactly that gap.
Three things freshdock will do differently
- Modern Docker API. Tested against Docker 24.x through 29+, with auto-negotiated API versions via bollard.
- Health-gated rollback. A container is only considered successfully updated when its healthcheck returns healthy (or a configurable grace period passes for containers without one). If the new container fails to come up, freshdock stops it, restores the previous container, and notifies you. No more waking up to a silent breakage.
- Single static binary. Target footprint: ≤ 10 MB binary, ≤ 30 MB resident memory at idle. No JVM, no Go runtime, no 100 MB image to manage your homelab.
Planned features
Update modes (per container, via Docker labels)
| Mode | Behaviour |
|---|---|
live |
Poll registry frequently; pull and recreate on every new digest. |
nightly |
Check once per day at a configurable time (default 04:00 local). |
weekly |
Check once per week. |
monthly |
Check on a configured day of the month. |
watch |
Detect updates and notify only — never pull or restart. |
off |
Ignore the container entirely. |
A single deployment can mix modes freely. Container A can live-update, container B can be nightly, container C can be watch-only.
Example:
services:
myapp:
image: ghcr.io/example/myapp:latest
labels:
- "freshdock.enable=true"
- "freshdock.mode=nightly"
- "freshdock.notify=true"
Notifications (v1 scope)
Webhook, Discord, Telegram, and SMTP email. Triggers for: update available (watch mode), update succeeded, update failed (with rollback status).
Registry support (v1 scope)
Docker Hub, GitHub Container Registry (GHCR), Quay.io, lscr.io, and any OCI-compliant registry with bearer-token auth. ECR/GCR/ACR coming post-v1.
Compatibility targets
| Platform | Status |
|---|---|
| Plain Docker (24.x – 29+) | Primary target. |
| Docker Desktop (Linux, macOS, Windows) | Supported. |
| Portainer (CE and Business) | Supported via the same Docker socket. |
| Podman 4+ | Supported via Bollard's auto-discovery. |
| Compose-based UIs (Dockge, Komodo, etc.) | Containers are updated individually; compose files are not edited. |
| Kubernetes / Swarm | Out of scope — use platform-native mechanisms. |
Roadmap
The full plan, including phased milestones and architecture, lives in PLAN.md.
Short version:
- Phase 0 (current) — Name reservation, repo scaffolding, CI.
- Phase 1 — Read-only spike: list containers, check digests, print update status.
- Phase 2 — Single container recreate cycle.
- Phase 3 — Health-gating and rollback (the quality bar for v1.0).
- Phase 4 — Scheduling and per-container modes.
- Phase 5 — Multi-registry auth.
- Phase 6 — Notifications.
- Phase 7 — Polish, documentation, v1.0 release.
Estimated total time to v1.0: roughly 12 weeks of part-time work.
Installation
Not yet. Come back after Phase 1.
When freshdock is ready, installation will be the standard options for a Rust binary:
# Cargo (after first usable release)
# Docker (after first usable release)
Contributing
Contributions are welcome once Phase 1 lands. Until then the codebase is a single println! and there is little to do.
If you want to help shape the project before code exists:
- Open an issue with feedback on PLAN.md.
- Suggest registries, notification targets, or label conventions you'd want supported.
- Share what broke for you in Watchtower or its forks — pain points are the best feature requests.
License
freshdock is licensed under the Apache License 2.0.
Acknowledgements
- Watchtower for being the standard for ten years and shaping the design space.
- bollard for the Rust Docker SDK that makes any of this possible.