hack-house/hh/smoke.sh
leetcrypt ff5186a9d3 feat(hh): graceful shutdown, crypt default theme, neutralize branding, share-prep
- Graceful shutdown: Ctrl+C quits in chat (interrupts PTY while driving),
  RAII TermGuard + panic hook + SIGTERM/SIGHUP always restore the terminal
- Default theme is now "crypt" (neutral monochrome); theme sigil mirrored in
  chat/roster/help so the pentagram only renders under the "church" theme
- Neutralize inverted-pentagram branding across CLI, scripts, docs, and Cargo
  metadata (kept only in themes/church.toml + the render-time placeholder)
- Rewrite root README around hack-house; add bootstrap.sh, SECURITY.md,
  CODE_OF_CONDUCT.md, CHANGELOG.md, and issue/PR templates
- .gitignore cleanup; stop tracking .venv

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-05-31 23:23:19 -07:00

78 lines
3.3 KiB
Bash
Executable File

#!/usr/bin/env bash
# hack-house smoke test
# Exercises the full use-case path end to end against a live server:
# rust unit tests → SRP self-test → boot server → rust client handshake +
# round-trip → cross-language (python decrypts what the rust client sent).
# Run from anywhere: hh/smoke.sh
set -uo pipefail
HERE="$(cd "$(dirname "$0")" && pwd)" # .../hh
ROOT="$(cd "$HERE/.." && pwd)" # repo root
PY="$ROOT/.venv/bin/python"
BIN="$HERE/target/debug/hack-house"
PORT="${PORT:-4199}"
PW="${PW:-labtest}"
fail() { echo "✖ SMOKE FAIL: $1"; exit 1; }
echo "── 1/5 rust unit tests (srp vectors + fernet interop) ──"
( cd "$HERE" && cargo test --quiet ) || fail "cargo test"
echo "── 2/5 build + SRP self-test (Rust SRP ≡ Python srp) ──"
( cd "$HERE" && cargo build --quiet ) || fail "cargo build"
"$BIN" selftest | grep -q "selftest passed" || fail "selftest"
echo "── 3/5 boot server on :$PORT ──"
"$PY" "$ROOT/cmd_chat.py" serve 127.0.0.1 "$PORT" --password "$PW" --no-tls \
>/tmp/hh-smoke-srv.log 2>&1 &
SRV=$!
trap 'kill $SRV 2>/dev/null' EXIT
for _ in $(seq 1 20); do
curl -s --max-time 2 "http://127.0.0.1:$PORT/health" 2>/dev/null | grep -q '"status":"ok"' && break
sleep 1
done
curl -s "http://127.0.0.1:$PORT/health" 2>/dev/null | grep -q '"status":"ok"' || fail "server did not come up"
echo "── 4/5 rust client: SRP auth + encrypted round-trip ──"
"$BIN" handshake 127.0.0.1 "$PORT" smoke-rust --password "$PW" --no-tls \
| tee /tmp/hh-smoke-rust.log | grep -q "round-trip ✓" || fail "rust handshake/round-trip"
echo "── 5/5 cross-language: python decrypts the rust-sent message ──"
"$PY" - "$PORT" "$PW" <<'PYEOF' || fail "python cross-language read"
import sys, asyncio, base64, json
import srp, requests, websockets
from cryptography.fernet import Fernet
from cryptography.hazmat.primitives.kdf.hkdf import HKDF
from cryptography.hazmat.primitives import hashes
srp.rfc5054_enable()
port, pw = sys.argv[1], sys.argv[2].encode()
base, wsb = f"http://127.0.0.1:{port}", f"ws://127.0.0.1:{port}"
async def main():
usr = srp.User(b"chat", pw, hash_alg=srp.SHA256)
_, A = usr.start_authentication()
r = requests.post(f"{base}/srp/init",
json={"username": "smoke-py", "A": base64.b64encode(A).decode()}).json()
uid = r["user_id"]
B, salt, rs = (base64.b64decode(r[k]) for k in ("B", "salt", "room_salt"))
M = usr.process_challenge(salt, B)
v = requests.post(f"{base}/srp/verify",
json={"user_id": uid, "username": "smoke-py",
"M": base64.b64encode(M).decode()}).json()
room = Fernet(base64.urlsafe_b64encode(
HKDF(algorithm=hashes.SHA256(), length=32, salt=rs,
info=b"cmd-chat-room-key").derive(pw)))
async with websockets.connect(f"{wsb}/ws/chat?user_id={uid}&ws_token={v['ws_token']}") as ws:
data = json.loads(await asyncio.wait_for(ws.recv(), 5))
msgs = [room.decrypt(m["text"].encode()).decode()
for m in data.get("messages", []) if m.get("text")]
hits = [m for m in msgs if "house is open" in m]
assert hits, f"rust-sent message not found/decryptable: {msgs!r}"
print(" ✓ python decrypted rust-sent message:", hits)
asyncio.run(main())
PYEOF
echo
echo "✓ SMOKE PASS — crypto · SRP · fernet · cross-language relay all green"