bring-hermes
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@bring-hermesadd milk to my shopping list"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
bring-hermes
An MCP server that exposes the Bring! shopping list API as tools — so an assistant ("hermes") can read your lists and add items or whole recipes (with the correct quantity) to Bring!.
It is served over the MCP Streamable HTTP transport, protected by an API key, and built to run on Docker and Kubernetes (TLS terminated at the Ingress).
Built on the actively maintained bring-api
Python library (the one behind Home Assistant's Bring! integration), which
provides batch updates, automatic token refresh, and a recipe-URL parser.
What it can do
Tool | Description |
| List all lists on the account with their UUIDs. |
| Show the items currently on a list (name + quantity). |
| Add a single item; |
| Add all ingredients of a recipe in one call, with optional servings scaling. |
| Parse a recipe URL via Bring! and add its ingredients. |
| Remove an item from a list. |
| Mark an item as bought (moves it to "recently used"). |
Quantities: Bring! stores an item's amount in a free-text
specificationfield. Pass amounts like"500 g","2", or"1 EL"in thequantityfield — that is exactly what shows up under the item in the app.
Servings scaling
add_recipe (and import_recipe_from_url) accept base_servings and
target_servings. When both are given, the leading number of each quantity is
scaled by target / base (units and text are preserved):
"500 g" + base=4, target=6 -> "750 g"
"2" + base=4, target=6 -> "3"
"1 EL" + base=4, target=2 -> "0.5 EL"
"Salz" (no number) -> "Salz" (unchanged)Related MCP server: Unofficial AnyList MCP Server
Configuration
All configuration is via environment variables (see .env.example):
Variable | Required | Default | Description |
| ✅ | — | Bring! account email. |
| ✅ | — | Bring! account password. |
| ✅ | — | One or more API keys (comma-separated) clients must present. |
| — | first list | Default list (name or UUID) when a tool call omits one. |
| — |
| Bind address. |
| — |
| Bind port. |
| — |
| Path the MCP endpoint is served at. |
| — |
| Plain-JSON Streamable HTTP responses; |
| — |
| Python log level. |
Run locally (Docker Compose)
cp .env.example .env
# edit .env: set BRING_EMAIL, BRING_PASSWORD, and a strong MCP_API_KEY
docker compose up --buildThe server listens on http://localhost:8080 with the MCP endpoint at
http://localhost:8080/mcp. Health endpoints (/healthz, /readyz) are
unauthenticated; everything else requires the API key.
Quick check:
curl -s http://localhost:8080/healthz # -> ok
curl -s -o /dev/null -w "%{http_code}\n" http://localhost:8080/mcp # -> 401 (no key)Run without Docker (uv)
uv sync
uv run bring-hermes # reads the same env vars (e.g. via `set -a; . ./.env`)Connecting an MCP client
Point a Streamable-HTTP-capable MCP client at the /mcp URL and send the API
key in the Authorization header:
{
"mcpServers": {
"bring": {
"type": "http",
"url": "https://bring-hermes.example.com/mcp",
"headers": {
"Authorization": "Bearer <MCP_API_KEY>"
}
}
}
}(X-API-Key: <MCP_API_KEY> is accepted as an alternative to the bearer header.)
Deploy to Kubernetes (Helm)
A Helm chart lives in helm/bring-hermes/ — see its
README for the full values reference. TLS is
terminated at the Ingress (cert-manager + Let's Encrypt in the example), or you
can expose it via the Gateway API instead — set httpRoute.enabled=true with a
parentRefs Gateway (see the chart README).
# Credentials as a managed Secret (recommended)
kubectl create namespace bring-hermes
kubectl -n bring-hermes create secret generic bring-hermes-credentials \
--from-literal=BRING_EMAIL='you@example.com' \
--from-literal=BRING_PASSWORD='your-bring-password' \
--from-literal=MCP_API_KEY="$(openssl rand -hex 32)"
helm install bring-hermes ./helm/bring-hermes \
--namespace bring-hermes \
--set bring.existingSecret=bring-hermes-credentials \
--set image.repository=ghcr.io/scramb/bring--mcp \
--set image.tag=0.2.0 \
--set ingress.enabled=true \
--set ingress.hosts[0].host=bring-hermes.example.com(Or pass bring.email / bring.password / bring.apiKey directly and let the
chart create the Secret — fine for testing, not for production.)
Notes:
Stateless MCP transport → safe to run multiple replicas behind a normal Service (no session affinity required).
Probes:
/healthzis liveness;/readyzreturns503until the first successful Bring! login, then200.Responses: plain-JSON Streamable HTTP by default (no SSE), so no special proxy buffering/timeout tuning is required at the Ingress.
Hardening: runs as non-root with a read-only root filesystem and all capabilities dropped.
Releases (GitHub Actions → GHCR)
Pushing a version tag triggers .github/workflows/release.yml,
which publishes both artifacts to GHCR with the same version as the git tag
(a leading v is stripped so the value is valid SemVer for Helm):
git tag v0.2.0
git push origin v0.2.0
# -> image: ghcr.io/<owner>/bring-hermes:0.2.0 (+ :latest)
# -> chart: oci://ghcr.io/<owner>/charts/bring-hermes version 0.2.0The chart is packaged with --version/--app-version set to that version, and
its image.tag defaults to the chart appVersion — so image tag and chart
version always match.
Build the image manually
docker build -t ghcr.io/your-org/bring-hermes:0.1.0 .
docker push ghcr.io/your-org/bring-hermes:0.1.0How it works
One shared
Bringclient (single login) is created in the app's lifespan and reused across requests; the underlying library refreshes the access token automatically, and an expired-token error triggers a one-shot re-login.The FastMCP Streamable HTTP session manager is nested inside that lifespan, so the login is not re-run per request (which would otherwise happen in stateless mode).
API-key auth is a small ASGI middleware in front of the MCP mount; health endpoints are exempt so Kubernetes probes work without credentials.
The Streamable HTTP transport replies with plain JSON by default (
MCP_JSON_RESPONSE=true); set it tofalseto get SSE-framed responses for streaming clients.
Security
Treat
MCP_API_KEYas a secret; use a long random value and rotate it by setting a comma-separated list and removing the old key once clients migrate.Always serve behind TLS (the Ingress). The API key is a bearer credential — never expose
/mcpover plain HTTP in production.The Bring! credentials grant full access to your shopping lists; scope and store them like any other secret.
Credits
Huge thanks to the authors of bring-api,
Cyrill Raccaud (@miaucl) and Manfred
Dennerlein Rodelo — the actively maintained Python Bring! client that also
powers Home Assistant's Bring! integration. This server is a thin MCP wrapper
around their work; all the real Bring! API heavy lifting (batch updates,
automatic token refresh, recipe parsing) is theirs. 💙
If you find this useful, please go star miaucl/bring-api.
License
MIT. bring-api is MIT-licensed by its respective authors. Bring! is a
trademark of its respective owners; this project is unofficial and not
affiliated with Bring! Labs AG.
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/scramb/bring--mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server