feat(docker): add Docker container for the agent (salvage #1841) (#3668)
Adds a complete Docker packaging for Hermes Agent:
- Dockerfile based on debian:13.4 with all deps
- Entrypoint that bootstraps .env, config.yaml, SOUL.md on first run
- CI workflow to build, test, and push to DockerHub
- Documentation for interactive, gateway, and upgrade workflows
Closes #850, #913.
Changes vs original PR:
- Removed pre-created legacy cache/platform dirs from entrypoint
(image_cache, audio_cache, pairing, whatsapp/session) — these are
now created on demand by the application using the consolidated
layout from get_hermes_dir()
- Moved docs from docs/docker.md to website/docs/user-guide/docker.md
and added to Docusaurus sidebar
Co-authored-by: benbarclay <benbarclay@users.noreply.github.com>
2026-03-28 22:21:48 -07:00
|
|
|
#!/bin/bash
|
2026-04-14 21:20:37 -07:00
|
|
|
# Docker/Podman entrypoint: bootstrap config files into the mounted volume, then run hermes.
|
feat(docker): add Docker container for the agent (salvage #1841) (#3668)
Adds a complete Docker packaging for Hermes Agent:
- Dockerfile based on debian:13.4 with all deps
- Entrypoint that bootstraps .env, config.yaml, SOUL.md on first run
- CI workflow to build, test, and push to DockerHub
- Documentation for interactive, gateway, and upgrade workflows
Closes #850, #913.
Changes vs original PR:
- Removed pre-created legacy cache/platform dirs from entrypoint
(image_cache, audio_cache, pairing, whatsapp/session) — these are
now created on demand by the application using the consolidated
layout from get_hermes_dir()
- Moved docs from docs/docker.md to website/docs/user-guide/docker.md
and added to Docusaurus sidebar
Co-authored-by: benbarclay <benbarclay@users.noreply.github.com>
2026-03-28 22:21:48 -07:00
|
|
|
set -e
|
|
|
|
|
|
2026-04-14 21:20:37 -07:00
|
|
|
HERMES_HOME="${HERMES_HOME:-/opt/data}"
|
feat(docker): add Docker container for the agent (salvage #1841) (#3668)
Adds a complete Docker packaging for Hermes Agent:
- Dockerfile based on debian:13.4 with all deps
- Entrypoint that bootstraps .env, config.yaml, SOUL.md on first run
- CI workflow to build, test, and push to DockerHub
- Documentation for interactive, gateway, and upgrade workflows
Closes #850, #913.
Changes vs original PR:
- Removed pre-created legacy cache/platform dirs from entrypoint
(image_cache, audio_cache, pairing, whatsapp/session) — these are
now created on demand by the application using the consolidated
layout from get_hermes_dir()
- Moved docs from docs/docker.md to website/docs/user-guide/docker.md
and added to Docusaurus sidebar
Co-authored-by: benbarclay <benbarclay@users.noreply.github.com>
2026-03-28 22:21:48 -07:00
|
|
|
INSTALL_DIR="/opt/hermes"
|
|
|
|
|
|
2026-04-12 00:26:08 -07:00
|
|
|
# --- Privilege dropping via gosu ---
|
2026-04-14 21:20:37 -07:00
|
|
|
# When started as root (the default for Docker, or fakeroot in rootless Podman),
|
|
|
|
|
# optionally remap the hermes user/group to match host-side ownership, fix volume
|
|
|
|
|
# permissions, then re-exec as hermes.
|
2026-04-12 00:26:08 -07:00
|
|
|
if [ "$(id -u)" = "0" ]; then
|
|
|
|
|
if [ -n "$HERMES_UID" ] && [ "$HERMES_UID" != "$(id -u hermes)" ]; then
|
|
|
|
|
echo "Changing hermes UID to $HERMES_UID"
|
|
|
|
|
usermod -u "$HERMES_UID" hermes
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
if [ -n "$HERMES_GID" ] && [ "$HERMES_GID" != "$(id -g hermes)" ]; then
|
|
|
|
|
echo "Changing hermes GID to $HERMES_GID"
|
2026-04-14 21:20:37 -07:00
|
|
|
# -o allows non-unique GID (e.g. macOS GID 20 "staff" may already exist
|
|
|
|
|
# as "dialout" in the Debian-based container image)
|
|
|
|
|
groupmod -o -g "$HERMES_GID" hermes 2>/dev/null || true
|
2026-04-12 00:26:08 -07:00
|
|
|
fi
|
|
|
|
|
|
2026-04-21 19:12:15 +08:00
|
|
|
# Fix ownership of the data volume. When HERMES_UID remaps the hermes user,
|
|
|
|
|
# files created by previous runs (under the old UID) become inaccessible.
|
|
|
|
|
# Always chown -R when UID was remapped; otherwise only if top-level is wrong.
|
2026-04-12 00:26:08 -07:00
|
|
|
actual_hermes_uid=$(id -u hermes)
|
2026-04-21 19:12:15 +08:00
|
|
|
needs_chown=false
|
|
|
|
|
if [ -n "$HERMES_UID" ] && [ "$HERMES_UID" != "10000" ]; then
|
|
|
|
|
needs_chown=true
|
|
|
|
|
elif [ "$(stat -c %u "$HERMES_HOME" 2>/dev/null)" != "$actual_hermes_uid" ]; then
|
|
|
|
|
needs_chown=true
|
|
|
|
|
fi
|
|
|
|
|
if [ "$needs_chown" = true ]; then
|
|
|
|
|
echo "Fixing ownership of $HERMES_HOME to hermes ($actual_hermes_uid)"
|
2026-04-14 21:20:37 -07:00
|
|
|
# In rootless Podman the container's "root" is mapped to an unprivileged
|
|
|
|
|
# host UID — chown will fail. That's fine: the volume is already owned
|
|
|
|
|
# by the mapped user on the host side.
|
|
|
|
|
chown -R hermes:hermes "$HERMES_HOME" 2>/dev/null || \
|
|
|
|
|
echo "Warning: chown failed (rootless container?) — continuing anyway"
|
2026-04-12 00:26:08 -07:00
|
|
|
fi
|
|
|
|
|
|
2026-04-26 08:27:39 -07:00
|
|
|
# Ensure config.yaml is readable by the hermes runtime user even if it was
|
|
|
|
|
# edited on the host after initial ownership setup. Must run here (as root)
|
|
|
|
|
# rather than after the gosu drop, otherwise a non-root caller like
|
|
|
|
|
# `docker run -u $(id -u):$(id -g)` hits "Operation not permitted" (#15865).
|
|
|
|
|
if [ -f "$HERMES_HOME/config.yaml" ]; then
|
|
|
|
|
chown hermes:hermes "$HERMES_HOME/config.yaml" 2>/dev/null || true
|
|
|
|
|
chmod 640 "$HERMES_HOME/config.yaml" 2>/dev/null || true
|
|
|
|
|
fi
|
|
|
|
|
|
2026-04-12 00:26:08 -07:00
|
|
|
echo "Dropping root privileges"
|
|
|
|
|
exec gosu hermes "$0" "$@"
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# --- Running as hermes from here ---
|
|
|
|
|
source "${INSTALL_DIR}/.venv/bin/activate"
|
|
|
|
|
|
feat(docker): add Docker container for the agent (salvage #1841) (#3668)
Adds a complete Docker packaging for Hermes Agent:
- Dockerfile based on debian:13.4 with all deps
- Entrypoint that bootstraps .env, config.yaml, SOUL.md on first run
- CI workflow to build, test, and push to DockerHub
- Documentation for interactive, gateway, and upgrade workflows
Closes #850, #913.
Changes vs original PR:
- Removed pre-created legacy cache/platform dirs from entrypoint
(image_cache, audio_cache, pairing, whatsapp/session) — these are
now created on demand by the application using the consolidated
layout from get_hermes_dir()
- Moved docs from docs/docker.md to website/docs/user-guide/docker.md
and added to Docusaurus sidebar
Co-authored-by: benbarclay <benbarclay@users.noreply.github.com>
2026-03-28 22:21:48 -07:00
|
|
|
# Create essential directory structure. Cache and platform directories
|
|
|
|
|
# (cache/images, cache/audio, platforms/whatsapp, etc.) are created on
|
|
|
|
|
# demand by the application — don't pre-create them here so new installs
|
|
|
|
|
# get the consolidated layout from get_hermes_dir().
|
2026-04-10 13:37:45 -07:00
|
|
|
# The "home/" subdirectory is a per-profile HOME for subprocesses (git,
|
|
|
|
|
# ssh, gh, npm …). Without it those tools write to /root which is
|
|
|
|
|
# ephemeral and shared across profiles. See issue #4426.
|
2026-04-10 15:11:20 -07:00
|
|
|
mkdir -p "$HERMES_HOME"/{cron,sessions,logs,hooks,memories,skills,skins,plans,workspace,home}
|
feat(docker): add Docker container for the agent (salvage #1841) (#3668)
Adds a complete Docker packaging for Hermes Agent:
- Dockerfile based on debian:13.4 with all deps
- Entrypoint that bootstraps .env, config.yaml, SOUL.md on first run
- CI workflow to build, test, and push to DockerHub
- Documentation for interactive, gateway, and upgrade workflows
Closes #850, #913.
Changes vs original PR:
- Removed pre-created legacy cache/platform dirs from entrypoint
(image_cache, audio_cache, pairing, whatsapp/session) — these are
now created on demand by the application using the consolidated
layout from get_hermes_dir()
- Moved docs from docs/docker.md to website/docs/user-guide/docker.md
and added to Docusaurus sidebar
Co-authored-by: benbarclay <benbarclay@users.noreply.github.com>
2026-03-28 22:21:48 -07:00
|
|
|
|
|
|
|
|
# .env
|
|
|
|
|
if [ ! -f "$HERMES_HOME/.env" ]; then
|
|
|
|
|
cp "$INSTALL_DIR/.env.example" "$HERMES_HOME/.env"
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# config.yaml
|
|
|
|
|
if [ ! -f "$HERMES_HOME/config.yaml" ]; then
|
|
|
|
|
cp "$INSTALL_DIR/cli-config.yaml.example" "$HERMES_HOME/config.yaml"
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# SOUL.md
|
|
|
|
|
if [ ! -f "$HERMES_HOME/SOUL.md" ]; then
|
|
|
|
|
cp "$INSTALL_DIR/docker/SOUL.md" "$HERMES_HOME/SOUL.md"
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
# Sync bundled skills (manifest-based so user edits are preserved)
|
|
|
|
|
if [ -d "$INSTALL_DIR/skills" ]; then
|
|
|
|
|
python3 "$INSTALL_DIR/tools/skills_sync.py"
|
|
|
|
|
fi
|
|
|
|
|
|
2026-05-04 15:37:27 +10:00
|
|
|
# Optionally start `hermes dashboard` as a side-process.
|
|
|
|
|
#
|
|
|
|
|
# Toggled by HERMES_DASHBOARD=1 (also accepts "true"/"yes", case-insensitive).
|
|
|
|
|
# Host/port/TUI can be overridden via:
|
|
|
|
|
# HERMES_DASHBOARD_HOST (default 0.0.0.0 — exposed outside the container)
|
|
|
|
|
# HERMES_DASHBOARD_PORT (default 9119, matches `hermes dashboard` default)
|
|
|
|
|
# HERMES_DASHBOARD_TUI (already honored by `hermes dashboard` itself)
|
|
|
|
|
#
|
|
|
|
|
# The dashboard is a long-lived server. We background it *before* the final
|
|
|
|
|
# `exec hermes "$@"` so the user's chosen foreground command (chat, gateway,
|
|
|
|
|
# sleep infinity, …) remains PID-of-interest for the container runtime. When
|
|
|
|
|
# the container stops the whole process tree is torn down, so no explicit
|
|
|
|
|
# cleanup is needed.
|
|
|
|
|
case "${HERMES_DASHBOARD:-}" in
|
|
|
|
|
1|true|TRUE|True|yes|YES|Yes)
|
|
|
|
|
dash_host="${HERMES_DASHBOARD_HOST:-0.0.0.0}"
|
|
|
|
|
dash_port="${HERMES_DASHBOARD_PORT:-9119}"
|
|
|
|
|
dash_args=(--host "$dash_host" --port "$dash_port" --no-open)
|
|
|
|
|
# Binding to anything other than localhost requires --insecure — the
|
|
|
|
|
# dashboard refuses otherwise because it exposes API keys. Inside a
|
|
|
|
|
# container this is the expected deployment (host reaches it via
|
|
|
|
|
# published port), so opt in automatically.
|
|
|
|
|
if [ "$dash_host" != "127.0.0.1" ] && [ "$dash_host" != "localhost" ]; then
|
|
|
|
|
dash_args+=(--insecure)
|
|
|
|
|
fi
|
|
|
|
|
echo "Starting hermes dashboard on ${dash_host}:${dash_port} (background)"
|
|
|
|
|
# Prefix dashboard output so it's distinguishable from the main
|
|
|
|
|
# process in `docker logs`. stdbuf keeps the pipe line-buffered.
|
|
|
|
|
(
|
|
|
|
|
stdbuf -oL -eL hermes dashboard "${dash_args[@]}" 2>&1 \
|
|
|
|
|
| sed -u 's/^/[dashboard] /'
|
|
|
|
|
) &
|
|
|
|
|
;;
|
|
|
|
|
esac
|
|
|
|
|
|
2026-04-21 08:29:33 +03:00
|
|
|
# Final exec: two supported invocation patterns.
|
|
|
|
|
#
|
|
|
|
|
# docker run <image> -> exec `hermes` with no args (legacy default)
|
|
|
|
|
# docker run <image> chat -q "..." -> exec `hermes chat -q "..."` (legacy wrap)
|
|
|
|
|
# docker run <image> sleep infinity -> exec `sleep infinity` directly
|
|
|
|
|
# docker run <image> bash -> exec `bash` directly
|
|
|
|
|
#
|
|
|
|
|
# If the first positional arg resolves to an executable on PATH, we assume the
|
|
|
|
|
# caller wants to run it directly (needed by the launcher which runs long-lived
|
|
|
|
|
# `sleep infinity` sandbox containers — see tools/environments/docker.py).
|
|
|
|
|
# Otherwise we treat the args as a hermes subcommand and wrap with `hermes`,
|
|
|
|
|
# preserving the documented `docker run <image> <subcommand>` behavior.
|
|
|
|
|
if [ $# -gt 0 ] && command -v "$1" >/dev/null 2>&1; then
|
|
|
|
|
exec "$@"
|
|
|
|
|
fi
|
feat(docker): add Docker container for the agent (salvage #1841) (#3668)
Adds a complete Docker packaging for Hermes Agent:
- Dockerfile based on debian:13.4 with all deps
- Entrypoint that bootstraps .env, config.yaml, SOUL.md on first run
- CI workflow to build, test, and push to DockerHub
- Documentation for interactive, gateway, and upgrade workflows
Closes #850, #913.
Changes vs original PR:
- Removed pre-created legacy cache/platform dirs from entrypoint
(image_cache, audio_cache, pairing, whatsapp/session) — these are
now created on demand by the application using the consolidated
layout from get_hermes_dir()
- Moved docs from docs/docker.md to website/docs/user-guide/docker.md
and added to Docusaurus sidebar
Co-authored-by: benbarclay <benbarclay@users.noreply.github.com>
2026-03-28 22:21:48 -07:00
|
|
|
exec hermes "$@"
|