```Show HN: Posthorn,无需邮件服务器的自托管邮件服务```
Show HN: Posthorn, self-hosted mail without the mail server

原始链接: https://github.com/craigmccaskill/posthorn

Posthorn 是一个为自托管项目设计的统一出站邮件网关。它充当了您的自托管应用程序(如 Ghost、Gitea 或自定义 Cloudflare Workers)与您首选的事务性电子邮件提供商(Postmark、Resend、Mailgun 或 AWS SES)之间的桥梁。 您无需为技术栈中的每个服务配置和维护独立的邮件集成,只需将所有应用程序指向同一个 Posthorn 实例即可。它提供了三种灵活的接入方式: * **HTTP 表单:** 处理联系表单,并内置了蜜罐(honeypot)、速率限制和来源验证等安全功能。 * **HTTP API:** 通过 JSON 实现服务器到服务器的通信,具备幂等重试和授权功能。 * **SMTP 监听器:** 使遗留或受限应用程序能够通过 SMTP 发送邮件,Posthorn 会将其转换为针对您所选提供商的 API 调用。 Posthorn 并非邮件服务器;它通过利用可靠的第三方提供商,避免了管理送达率、SPF、DKIM 或 IP 信誉等复杂问题。它以单个 Go 二进制文件或 Docker 容器的形式发布,通过将凭证和配置集中到一个 TOML 文件中来简化您的基础设施。对于希望在无需维护自有邮件基础设施负担的情况下,获得稳健事务性邮件服务的运营者来说,它是理想的解决方案。

Posthorn 是一款开源、轻量级的电子邮件网关,旨在简化自托管应用的事务性邮件处理。该项目由 Craig McCaskill 创建,通过在应用程序与可靠的电子邮件服务之间提供一个集中式中继,解决了 VPS 提供商屏蔽默认 SMTP 端口的常见问题。 无论是作为 Go 二进制文件运行,还是以 10MB 的 Docker 容器部署,Posthorn 都能与 Postmark、Resend、Mailgun 和 Amazon SES 等服务提供商无缝集成。它作为一个功能多样的“中间人”,支持标准 SMTP、直接 JSON HTTP API 请求,甚至支持静态网站的 HTML 表单提交。为增强安全性,它还内置了蜜罐字段、来源验证和 IP 速率限制等功能。 Posthorn 旨在消除托管 Ghost、Gitea 或 Mastodon 等服务时,针对每个应用进行复杂配置(如手动配置 Postfix 中继)的必要性。该项目采用 Apache 2.0 许可证开源,完整文档请访问 posthorn.dev。
相关文章

原文

CI Docs License: Apache-2.0 Go version

The unified outbound mail layer for self-hosted projects. One gateway between every app you self-host and the transactional mail provider you've already picked. Three ingress shapes (HTTP form, HTTP API, SMTP), five transports (Postmark, Resend, Mailgun, AWS SES, outbound-SMTP relay), single Go binary, single TOML config.

Real-world stacks: Hugo + Comentario · Ghost · Gitea · Umami digest cron · Cloudflare Worker

Nobody wants to run a mail server in 2026. Self-hosted operators use Postmark, Resend, Mailgun, or AWS SES because they're cheap, they handle deliverability properly, and someone else worries about SPF / DKIM / DMARC / bounces / sender reputation.

But every app you self-host has to integrate with that service independently. Your contact form. Your Ghost blog's admin emails. Your Gitea magic links. Your Mastodon notifications. The Cloudflare Worker that fires a password-reset email when someone clicks the link. Each one needs its own copy of the API key, its own integration code, its own quirks around retry and bounce handling. The same outbound concern duplicated across your stack.

And on cloud hosts that block outbound SMTP — DigitalOcean, AWS Lightsail, Linode, Vultr — the SMTP-only apps don't work at all without a workaround.

Posthorn is the bridge. One container, one config, one set of credentials. Your apps point at Posthorn. Posthorn talks to your provider.

Where your app connects What Posthorn does
HTTP form (contact forms, signups, alert webhooks) Honeypot + Origin/Referer + rate limit + optional CSRF; templates the email; sends
HTTP API mode (workers, cron, payment handlers, internal services) Authorization: Bearer auth; JSON body; idempotent retries; per-request to_override for transactional sends
SMTP listener (Ghost, Gitea, Mastodon, Matrix, NextCloud, Authentik, anything that emits SMTP) AUTH PLAIN or client-cert; STARTTLS-required; sender + recipient allowlists; parses MIME; forwards via HTTP API transport

All three ingresses converge on one transport.Message and one outbound provider — pick from Postmark, Resend, Mailgun, AWS SES, or an outbound-SMTP relay.

To save you a wrong turn:

What it does Look at instead
Not a mail server No mailbox storage, no IMAP/JMAP, no DKIM key management, no MX target Stalwart, Mailcow, iRedMail
Not its own outbound infrastructure Posthorn relays through a provider you chose; it doesn't run its own SMTP fleet or manage IP reputation Postal, Hyvor Relay
Not a marketing email platform No list management, no segmentation, no campaign dashboard Listmonk
Not webmail / a mailbox UI No interface for reading mail Roundcube, Snappymail (with a mail server)

The wedge is the integration layer between your self-hosted apps and the transactional provider you've already picked.

posthorn.dev — getting started, configuration reference, deployment guides, feature deep-dives, security model, HTTP API reference, FAQ. Ten recipes covering contact forms, newsletter signups, multi-form sites, monitoring alerts, Cloudflare Workers, internal SMTP relay (Docker Compose), and full case studies for Hugo+Comentario, Ghost, Gitea, and self-hosted Umami digests.

For project history and the v1.0 spec, see spec/.

# docker-compose.yml
services:
  posthorn:
    image: ghcr.io/craigmccaskill/posthorn:latest
    restart: unless-stopped
    volumes:
      - ./posthorn.toml:/etc/posthorn/config.toml:ro
    environment:
      POSTMARK_API_KEY: ${POSTMARK_API_KEY}
    ports:
      - "127.0.0.1:8080:8080"   # bind to loopback; reverse-proxy from your front door
# posthorn.toml
[[endpoints]]
path = "/api/contact"
to = ["[email protected]"]
from = "Contact Form <[email protected]>"
honeypot = "_gotcha"
allowed_origins = ["https://example.com"]
required = ["name", "email", "message"]
subject = "Contact from {{.name}}"
body = """
From: {{.name}} <{{.email}}>

{{.message}}
"""
redirect_success = "/thank-you"

[endpoints.transport]
type = "postmark"

[endpoints.transport.settings]
api_key = "${env.POSTMARK_API_KEY}"

[endpoints.rate_limit]
count = 5
interval = "1m"

Reverse-proxy /api/contact from your front door (Caddy, nginx, Traefik) to http://posthorn:8080. Point your form's action at /api/contact. Done.

Full walkthrough: posthorn.dev/getting-started/quick-start.

API mode (server-to-server)

For Workers, cron jobs, internal services — anything that speaks JSON instead of forms:

[[endpoints]]
path = "/api/transactional"
to = ["[email protected]"]
from = "YourApp <[email protected]>"
auth = "api-key"
api_keys = ["${env.WORKER_KEY_PRIMARY}", "${env.WORKER_KEY_BACKUP}"]
required = ["subject_line", "message"]
subject = "{{.subject_line}}"
body = "{{.message}}"

[endpoints.transport]
type = "postmark"

[endpoints.transport.settings]
api_key = "${env.POSTMARK_API_KEY}"
curl -X POST https://posthorn.yourdomain.com/api/transactional \
  -H "Authorization: Bearer $WORKER_KEY_PRIMARY" \
  -H "Content-Type: application/json" \
  -H "Idempotency-Key: reset:user-123:$(date -u +%FT%H)" \
  --data '{
    "to_override": "[email protected]",
    "subject_line": "Reset your password",
    "message": "Click here: https://app.example.com/reset/abc"
  }'

Full walkthrough: posthorn.dev/recipes/cloudflare-worker.

SMTP listener (Ghost / Gitea / Mastodon / Authentik)

For apps that speak SMTP natively and can't be reconfigured to call an HTTP API:

[smtp_listener]
listen          = ":2525"
require_tls     = true
tls_cert        = "/etc/posthorn/cert.pem"
tls_key         = "/etc/posthorn/key.pem"
auth_required   = "smtp-auth"
allowed_senders = ["*@yourdomain.com"]
max_recipients_per_session = 10
max_message_size = "1MB"

[[smtp_listener.smtp_users]]
username = "ghost"
password = "${env.GHOST_SMTP_PASSWORD}"

[smtp_listener.transport]
type = "postmark"

[smtp_listener.transport.settings]
api_key = "${env.POSTMARK_API_KEY}"

Point Ghost (or any app's SMTP config) at posthorn.yourdomain.com:2525 with the username/password above. Posthorn parses the MIME, builds a transport.Message, forwards via Postmark.

Full doc: posthorn.dev/features/smtp-ingress.

Transport Best for Auth Body
Postmark Transactional email, strong deliverability defaults X-Postmark-Server-Token JSON
Resend Modern HTTP API, developer-friendly dashboard Authorization: Bearer JSON
Mailgun Higher-volume transactional, US + EU regions HTTP Basic multipart/form-data
AWS SES AWS-native deployments, cheapest at volume AWS SigV4 (bespoke) JSON
Outbound SMTP Any STARTTLS-capable relay (Mailtrap, your Postfix smarthost, etc.) AUTH PLAIN SMTP DATA

Switching providers is a TOML edit — every transport implements the same Transport interface. See posthorn.dev/configuration/transports for per-provider config.

Before pointing real traffic at Posthorn:

  1. DNS — SPF, DKIM, and DMARC records on your sending domain. Without these your mail goes to spam. See posthorn.dev/security/dns.
  2. Reverse proxy — Posthorn does not terminate TLS. Run it behind Caddy, nginx, or Traefik. See posthorn.dev/deployment/reverse-proxy.
  3. allowed_origins (form-mode endpoints) — set this to lock submissions to your domain. Without it, anyone can POST to your endpoint.
  4. rate_limit — set a tight bucket per endpoint (5/minute is a sensible default for a public contact form; API mode rate-limits per matched key).
  5. trusted_proxies — if behind a reverse proxy, list its CIDR (or use the cloudflare named preset) so the rate limiter sees the real client IP.
  6. /healthz and /metrics — auto-registered on the same listener. Wire your Docker healthcheck or Prometheus scrape to these.

The full operator checklist is on posthorn.dev.

Block Detail
Form ingress Form-encoded + multipart bodies; honeypot, Origin/Referer fail-closed, rate limit, optional CSRF tokens
API mode auth = "api-key" with Bearer tokens (constant-time compare); JSON content type; idempotency keys (24h, in-memory LRU); per-request to_override
Transports Postmark, Resend, Mailgun, AWS SES (bespoke SigV4), outbound-SMTP relay
SMTP listener TCP listener with AUTH PLAIN / client-cert, STARTTLS-required, sender + recipient allowlists, size cap, MIME → transport.Message
Operations /healthz, /metrics (Prometheus exposition), dry-run mode, IP-stripping, named trusted_proxies presets (Cloudflare)
Failure handling 1 retry on transient/5xx (1s), 1 retry on 429 (5s), 10s hard timeout
Logging Structured JSON; UUIDv4 submission IDs and SMTP session IDs; transport_message_id in submission_sent
Deployment Single Go binary, multi-arch distroless Docker image at ghcr.io/craigmccaskill/posthorn

Three external Go dependencies in the whole module: TOML parser, UUID library, LRU cache. Every transport is bespoke — no vendor SDK in transport code.

v2 — platform maturity. SQLite submission log, retry queue across restarts, suppression list (auto on hard bounces), durable idempotency, lifecycle event callbacks via HMAC-signed webhook, RFC 8058 one-click unsubscribe, file attachments, HTML body, multiple outputs per endpoint (email + webhook + log fan-out), multi-tenant SMTP routing.

v3 — speculative. Admin UI, proof-of-work spam challenge, PGP encryption. Depends on community traction.

Full trajectory: posthorn.dev/roadmap.

Requires Go 1.25+.

git clone https://github.com/craigmccaskill/posthorn
cd posthorn/core
go build -o /tmp/posthorn ./cmd/posthorn
/tmp/posthorn version

The v1.0 specification is in spec/ (brief, PRD, architecture). The architecture doc at spec/03-architecture.md is the source of truth for design questions.

Security issues: see SECURITY.md — do not open public issues for security disclosures.

Apache-2.0. See LICENSE.

联系我们 contact @ memedata.com