Skip to content

Every URL you’ve been using? The proxy makes it work.

The Hoody Proxy is the gateway between the outside world and your containers. It handles HTTPS certificates, routes requests to the right service, manages permissions, and preserves real client IPs. All automatically. Every program you run gets HTTPS, HTTP/2, and HTTP/3 (QUIC) — out of the box. No configuration. No cert rotation. No Let’s Encrypt dance.

You will never think about a certificate again in your life. It just works.


  1. Automatic HTTPS — Wildcard TLS certificates for every container URL. No Let’s Encrypt setup, no cert rotation, no DNS challenge.

  2. URL routing — Parses {projectId}-{containerId}-{service}-{instance}.{serverName}.containers.hoody.icu and routes to the correct service inside the correct container.

  3. Permission enforcement — Authentication (JWT, password, IP whitelist, bearer token) checked before any request reaches your container.

  4. Real client IP — Uses netfilter hooks to preserve the real client IP address. Your application sees the actual visitor, not a proxy address.

  5. Protocol support — HTTP/1.1, HTTP/2, HTTP/3 (QUIC), and WebSocket upgrades. Real-time terminals and displays work seamlessly.


When you hit:

https://abc123-def456-terminal-1.node-us-1.containers.hoody.icu/api/v1/terminal/execute

The proxy:

  1. Extracts abc123 (project), def456 (container), terminal-1 (service + instance)
  2. Looks up node-us-1 to find the physical server
  3. Routes to terminal service instance 1 inside container def456
  4. Forwards the request with real client IP preserved

All in milliseconds. No configuration on your part.


Tired of sharing a URL stuffed with Project and Container IDs? Create an alias — a memorable name (3-61 chars, a-z, 0-9, hyphens) that resolves to a short, clean URL:

Terminal window
# Create a proxy alias
hoody proxy create \
--container-id $CONTAINER_ID \
--program "exec" \
--alias "my-api" \
--index 1

By default, container URLs are accessible by anyone who has the URL. The URL itself is unguessable (48+ characters of hex), which provides a baseline of security.

When you’re ready to lock things down:

Terminal window
# Require password authentication
hoody containers proxy permissions replace -c $CONTAINER_ID \
--project $PROJECT_ID \
--groups auth='{"type": "password", "password": "my-secret"}' \
--permissions auth='{"terminal": true, "files": true, "display": true}'
# Restrict to specific IP addresses
hoody containers proxy permissions replace -c $CONTAINER_ID \
--project $PROJECT_ID \
--groups office='{"type": "ip", "range": "203.0.113.10/32"}' \
--permissions office='{"terminal": true, "files": true, "display": true}'

One permissions document declares reusable auth groups (password, JWT, IP, token) and grants per-program access to them. Add a program in the permissions.<group>.<program> map to let that group in; anything not granted stays at the default policy. Configure once, apply it across whichever services need protection.


URLs are unguessable. Sharing requires knowing the URL. This means:

  • Development: No auth friction. Just build.
  • Collaboration: Share the URL. Everyone’s in.
  • Production: Add authentication when you’re ready.

No premature security configuration slowing you down. No “I can’t access the dev environment” tickets.

The proxy is invisible when you don’t need it, and bulletproof when you do.

Next: Your First API →