First runnable kez-chat-server binary plus its docker-compose deploy
recipe. Implements steps 2-3 of the document.md sequenced plan; the
rust-lib refactor (step 1) is deferred — chat-server path-deps on
rust/crates/kez-core for now, which works and matches what
rust-sig-server already does.
What's in this commit:
kez-core (1-line change)
- New public `verify_envelope<T>(payload, signature)` helper that
dispatches Schnorr / Ed25519 / future suites by signature.alg.
Used by chat-server's registration verifier; downstream value
beyond chat-server too.
kez-chat-server (new crate)
- src/main.rs: tokio + axum + tracing entry; clap config; graceful
Ctrl-C shutdown.
- src/lib.rs: re-exports so tests can drive the same router.
- src/config.rs: env/flag config (bind, db, server, sig_server_url,
web_dir) with defaults sane for both dev and prod.
- src/error.rs: typed ApiError → structured JSON responses with
stable error codes.
- src/store.rs: SQLite-backed handle registry, UNIQUE on both
(handle) and (primary_id); race-safe via SQL primary key.
- src/handles.rs: username validation (length, charset, reserved
list, must start with letter/digit).
- src/registration.rs: SignedRegistration envelope sharing KEZ's
JCS canonical-bytes pattern; signature verification via the new
kez-core helper; replay protection via ±5-minute clock skew check.
- src/api.rs: all six routes in one file —
GET /v1/healthz
GET /v1/u/:handle
POST /v1/register
GET /.well-known/webfinger
POST /internal/nats/auth (501 stub for v0.1; wired up in v0.2)
GET / (placeholder HTML; ServeDir when web/dist exists)
tests/http.rs — 13 integration tests
- Stands up the real router on a random port; uses reqwest.
- Coverage: healthz, lookup-404, full register→lookup round-trip,
duplicate-handle conflict, wrong-server rejection, reserved-name
rejection, tampered-signature rejection, stale-timestamp rejection,
WebFinger success + wrong-server-404, placeholder SPA renders,
NATS callout 501, JCS determinism sanity.
deploy/
- Dockerfile: multi-stage build (rust:1.86-slim → debian:bookworm-slim).
Build context is repo root so the path dep on kez-core resolves.
Runtime image ~50 MB; runs as non-root uid 10001.
- Dockerfile.sig-server: same pattern for the existing
rust-sig-server, so the stack builds from one git pull.
- docker-compose.yml: three services (chat-server + nats + sig-server)
with named volumes for persistence. Ports: 6969 (chat HTTP),
4222/8443/8222 (NATS native/ws/monitoring), 7878 (sig-server).
- nats.conf: WebSocket on 8443 for the browser SPA, JetStream
enabled, auth_callout pointing at chat-server's
/internal/nats/auth endpoint (issuer nkey is a placeholder — must
be replaced with a real one before going live).
README.md
- Documents all endpoints with example bodies.
- Quick-start for both local dev and full Docker compose.
- Honest list of what's in v0.1 vs what's still stubbed.
Smoke-tested running on 127.0.0.1:6969:
GET /v1/healthz → {"server":"kez.lat","status":"ok","version":"0.1.0"}
GET / → placeholder HTML rendering
GET /v1/u/ghost → 404
POST /internal/nats/auth → 501 with "wired up in v0.2"
cargo test → 13 passed.
cargo build --release → 19.6s, clean.
170 lines
5.2 KiB
Markdown
170 lines
5.2 KiB
Markdown
# kez-chat-server
|
|
|
|
Home server for the kez-chat application. One Rust binary that hosts:
|
|
|
|
- Handle registry (`POST /v1/register`, `GET /v1/u/:handle`)
|
|
- WebFinger discovery (`GET /.well-known/webfinger`)
|
|
- NATS auth callout endpoint (`POST /internal/nats/auth`) — stub in v0.1
|
|
- Static SPA serving (`GET /`) — placeholder until the Svelte build lands
|
|
- Healthz (`GET /v1/healthz`)
|
|
|
|
Designed in [`document.md`](document.md). Spec for the underlying KEZ
|
|
identity layer in [`../SPEC.md`](../SPEC.md).
|
|
|
|
## What's in v0.1 (this is the scaffold)
|
|
|
|
✅ HTTP API end-to-end
|
|
✅ Ed25519-signed handle registration with replay protection
|
|
✅ SQLite-backed registry with uniqueness on both handle and primary key
|
|
✅ WebFinger endpoint
|
|
✅ Placeholder SPA at `/`
|
|
✅ docker-compose for full stack (chat + nats + sig-server)
|
|
✅ Multi-stage Dockerfiles
|
|
✅ 13 integration tests against a live router
|
|
|
|
⚠️ NATS auth callout returns 501 — wired up in v0.2
|
|
⚠️ Svelte SPA build pipeline not yet in place — placeholder HTML for now
|
|
⚠️ TLS terminated upstream (no cert handling in this binary)
|
|
|
|
## Quick start (local development)
|
|
|
|
```sh
|
|
# Run from source
|
|
cargo run -- --bind 127.0.0.1:6969 --db ./kez-chat.db --server kez.lat
|
|
|
|
# Or install once
|
|
cargo install --path .
|
|
kez-chat-server --bind 127.0.0.1:6969 --server kez.lat
|
|
```
|
|
|
|
Configuration via flags or env vars:
|
|
|
|
| Flag | Env | Default |
|
|
|---|---|---|
|
|
| `--bind` | `KEZ_CHAT_BIND` | `0.0.0.0:6969` |
|
|
| `--db` | `KEZ_CHAT_DB` | `kez-chat.db` |
|
|
| `--server` | `KEZ_CHAT_SERVER` | `kez.lat` |
|
|
| `--sig-server-url` | `KEZ_CHAT_SIG_SERVER_URL` | `http://localhost:7878` |
|
|
| `--web-dir` | `KEZ_CHAT_WEB_DIR` | _(unset → placeholder page)_ |
|
|
|
|
Logging: `RUST_LOG=debug,hyper=info` etc.
|
|
|
|
## Quick start (Docker compose, full stack)
|
|
|
|
```sh
|
|
cd deploy
|
|
docker compose up -d --build
|
|
```
|
|
|
|
Brings up three services:
|
|
|
|
| Service | Port(s) | What it does |
|
|
|---|---|---|
|
|
| `chat-server` | 6969 | HTTP API + SPA |
|
|
| `nats` | 4222 (native), 8443 (WebSocket), 8222 (monitoring) | Dumb broker, JetStream enabled |
|
|
| `sig-server` | 7878 | Sigchain storage (the existing `rust-sig-server`) |
|
|
|
|
Then point a reverse proxy / Cloudflare tunnel at `localhost:6969`.
|
|
|
|
## Testing
|
|
|
|
```sh
|
|
cargo test # 13 integration tests (real server, real HTTP)
|
|
```
|
|
|
|
The tests stand up the router on a random local port and exercise it
|
|
via `reqwest`. No mocks. They cover: healthz, lookup, registration
|
|
(success + duplicate + wrong-server + reserved-name + tampered-sig +
|
|
stale-timestamp), WebFinger, the placeholder SPA, and the NATS auth
|
|
callout stub.
|
|
|
|
## Endpoints in detail
|
|
|
|
### `GET /v1/healthz`
|
|
|
|
```json
|
|
{ "status": "ok", "server": "kez.lat", "version": "0.1.0" }
|
|
```
|
|
|
|
### `GET /v1/u/:handle`
|
|
|
|
Returns:
|
|
|
|
```json
|
|
{
|
|
"handle": "tudisco",
|
|
"fqhn": "tudisco@kez.lat",
|
|
"primary": "ed25519:2152f8d19b...",
|
|
"sigchain_url": "https://sig.kez.lat/v1/sigchains/ed25519/2152f8d19b...",
|
|
"registered_at": "2026-05-25T03:00:00Z"
|
|
}
|
|
```
|
|
|
|
Returns 404 if the handle isn't registered.
|
|
|
|
### `POST /v1/register`
|
|
|
|
Request body — a signed registration envelope:
|
|
|
|
```json
|
|
{
|
|
"kez": "handle_registration",
|
|
"payload": {
|
|
"type": "kez.chat.handle_registration",
|
|
"version": 1,
|
|
"handle": "tudisco",
|
|
"primary": "ed25519:2152f8d19b...",
|
|
"server": "kez.lat",
|
|
"created_at": "2026-05-25T03:00:00Z"
|
|
},
|
|
"signature": {
|
|
"alg": "ed25519-sha512-jcs",
|
|
"key": "ed25519:2152f8d19b...",
|
|
"sig": "<128-char-hex>"
|
|
}
|
|
}
|
|
```
|
|
|
|
Server validates:
|
|
1. Envelope tag is `"handle_registration"`
|
|
2. Payload type is `"kez.chat.handle_registration"`, version 1
|
|
3. `signature.key` equals `payload.primary`
|
|
4. Signature verifies against the primary key (Ed25519 only for chat)
|
|
5. `payload.server` matches this server's configured domain
|
|
6. `payload.handle` passes validation (length 3-32, `a-z0-9_-`,
|
|
starts with letter/digit, not in reserved list)
|
|
7. `payload.created_at` is within 5 minutes of server time
|
|
|
|
On success: `201 Created` with the same body as `GET /v1/u/:handle`.
|
|
|
|
### `GET /.well-known/webfinger?resource=acct:user@server`
|
|
|
|
Standard fediverse-style discovery. Returns the user's KEZ identity
|
|
info as a WebFinger JRD. Used by other servers (federated lookup,
|
|
future) and by tools like fediverse browsers.
|
|
|
|
### `POST /internal/nats/auth`
|
|
|
|
NATS auth callout endpoint. **Stub in v0.1** — returns 501. The real
|
|
implementation (v0.2) will: parse the NATS auth request JWT, extract
|
|
the connecting client's nkey, look up the corresponding handle, sign
|
|
a response permitting `kez.inbox.<pubkey>.>` subjects.
|
|
|
|
## Deployment notes
|
|
|
|
- The `chat-server` Docker image is built from the **repo root** as
|
|
context (so it can copy `rust/crates/kez-core` for the path dep).
|
|
`docker-compose.yml` sets this correctly.
|
|
- The `sig-server` is the existing [`../rust-sig-server`](../rust-sig-server/)
|
|
binary, built into a separate image via `Dockerfile.sig-server`.
|
|
- NATS config (`nats.conf`) has WebSocket enabled on port 8443 so the
|
|
browser SPA can connect via `nats.ws`. The `issuer` field in
|
|
`auth_callout` is a placeholder — generate a real nkey and replace
|
|
before going to production.
|
|
- TLS is **not** handled by this binary. Put a reverse proxy (Caddy,
|
|
nginx, Cloudflare tunnel) in front for HTTPS.
|
|
|
|
## License
|
|
|
|
Dual-licensed under MIT or Apache-2.0.
|