d64b0c052d
Add the data layer for WhatsApp-style accounts on the wallet model: the admin mints a single-use invitation link, the new user redeems it by publishing only its public keys, and the admin can hard-delete a user. - Invite type and lifecycle (invites.go): 32-byte crypto/rand hex token, 7-day default TTL, fail-closed expiry parsing. Methods CreateInvite/GetInvite/ ListInvites/ConsumeInvite/CancelInvite on both backends. ConsumeInvite is atomic and single-use: SQLite uses a transaction guarded by `used = 0`, the KV store uses a compare-and-swap on the entry revision (mark-first). Both burn the token on claim, so an already-registered key surfaces ErrUserExists with the invite spent — identical semantics across backends. - DeleteUser (users.go + jetstream_store.go): hard-delete of the allowlist row, distinct from RevokeUser's status flip. Room memberships of the ex-user are intentionally left inert (they can no longer authenticate); no partial cleanup. - Migration 003_invites.sql (root + embedded copy, byte-identical): additive `invites` table with audit columns, per db_migrations rules. - Store interface gains DeleteUser, CreateInvite, GetInvite, ListInvites, ConsumeInvite, CancelInvite. New UNIBUS_invites KV bucket. - Consistency fix: SQLite GetUser now maps sql.ErrNoRows to ErrNotFound, matching the KV backend and the storage-agnostic contract documented in store.go. - ValidateKexPubHex added alongside ValidateSignPubHex for /register key checks. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
399 lines
14 KiB
Go
399 lines
14 KiB
Go
// Package membership implements the authoritative control plane of unibus:
|
|
// room metadata, member directory, and per-epoch sealed room keys.
|
|
//
|
|
// The data plane (actual messages) is NATS; this package owns the SQLite-backed
|
|
// state that NATS cannot: who is in a room, what their public keys are, and the
|
|
// encrypted room key K each member needs to participate at a given epoch.
|
|
//
|
|
// Migrations are embedded and applied idempotently on Open. The embedded copy
|
|
// under pkg/membership/migrations mirrors the module-root migrations/ directory
|
|
// (kept in sync); both are additive-only per .claude/rules/db_migrations.md.
|
|
package membership
|
|
|
|
import (
|
|
"database/sql"
|
|
"embed"
|
|
"errors"
|
|
"fmt"
|
|
"io/fs"
|
|
"sort"
|
|
"strings"
|
|
"time"
|
|
|
|
// modernc.org/sqlite registers the pure-Go "sqlite" driver (no CGO).
|
|
_ "modernc.org/sqlite"
|
|
)
|
|
|
|
//go:embed migrations/*.sql
|
|
var migrationsFS embed.FS
|
|
|
|
// ErrNotFound is the store-agnostic "no such record" sentinel. Both backends
|
|
// (SQLite and JetStream KV) return it, wrapped, when a lookup misses, so callers
|
|
// distinguish "not invited / no key yet" from a genuine backend failure without
|
|
// depending on a specific driver's error (the SQLite store maps sql.ErrNoRows to
|
|
// it; the KV store maps a missing key to it). This is what lets the control
|
|
// plane stay storage-agnostic under the branch-by-abstraction of issue 0003b.
|
|
var ErrNotFound = errors.New("membership: not found")
|
|
|
|
// Member is a participant of a room with their published public keys.
|
|
type Member struct {
|
|
Endpoint string `json:"endpoint"`
|
|
Role string `json:"role"`
|
|
SignPub []byte `json:"sign_pub"`
|
|
KexPub []byte `json:"kex_pub"`
|
|
}
|
|
|
|
// RoomInfo is the metadata of a room.
|
|
type RoomInfo struct {
|
|
RoomID string
|
|
Subject string
|
|
Epoch int
|
|
Encrypt bool
|
|
Persist bool
|
|
SignMsgs bool
|
|
OwnerEndpoint string
|
|
}
|
|
|
|
// Store is the membership/key control-plane store: the authoritative source of
|
|
// room metadata, the member directory, per-epoch sealed room keys, and the bus
|
|
// user allowlist. It is an interface (branch-by-abstraction, issue 0003b) with
|
|
// two implementations: sqliteStore (the default, single-node, local SQLite) and
|
|
// jetstreamStore (rooms/members/keys/users on replicated JetStream KV, selected
|
|
// when the `decentralized` flag is on). Every lookup miss returns ErrNotFound
|
|
// (wrapped); every implementation MUST fail closed (IsAuthorized returns false
|
|
// on any backend error), so a KV quorum loss denies rather than admits.
|
|
type Store interface {
|
|
// Rooms / members / keys.
|
|
CreateRoom(info RoomInfo, ownerSignPub, ownerKexPub, ownerSealedKey []byte) error
|
|
GetRoom(roomID string) (RoomInfo, error)
|
|
AddMember(roomID string, m Member, epoch int, sealedKey []byte) error
|
|
GetMember(roomID, endpoint string) (Member, error)
|
|
ListMembers(roomID string) ([]Member, error)
|
|
ListRoomsForEndpoint(endpoint string) ([]RoomMembership, error)
|
|
GetSealedKey(roomID, endpoint string, epoch int) (int, []byte, error)
|
|
PutSealedKeys(roomID string, epoch int, keys map[string][]byte) error
|
|
BumpEpoch(roomID string, newEpoch int) error
|
|
RemoveMember(roomID, endpoint string) error
|
|
|
|
// Users (the bus allowlist).
|
|
AddUser(signPub, handle, role string) error
|
|
GetUser(signPub string) (User, error)
|
|
ListUsers() ([]User, error)
|
|
RevokeUser(signPub string) error
|
|
// DeleteUser hard-deletes a user (the purge counterpart of RevokeUser's
|
|
// status flip): the row is removed, not just flagged. The ex-user can no
|
|
// longer authenticate, so any room memberships they hold become inert.
|
|
DeleteUser(signPub string) error
|
|
IsAuthorized(signPub string) bool
|
|
HasAdmin() bool
|
|
|
|
// Invites (single-use registration tokens; the wallet-model join path).
|
|
// CreateInvite mints a token fixing handle+role; ConsumeInvite is the only
|
|
// path that adds to the allowlist without an admin signature (the bearer
|
|
// token is the authorization), spending the token exactly once.
|
|
CreateInvite(handle, role string, ttlSecs int) (Invite, error)
|
|
GetInvite(token string) (Invite, error)
|
|
ListInvites() ([]Invite, error)
|
|
ConsumeInvite(token, signPub, kexPub string) error
|
|
CancelInvite(token string) error
|
|
|
|
// Lifecycle.
|
|
Close() error
|
|
}
|
|
|
|
// sqliteStore is the SQLite-backed implementation of Store (the default,
|
|
// single-node backend). It stays the production default while the
|
|
// `decentralized` flag is off.
|
|
type sqliteStore struct {
|
|
db *sql.DB
|
|
}
|
|
|
|
// Open opens (creating if needed) the SQLite database at path, applies all
|
|
// embedded migrations idempotently, and returns it as a Store. It remains the
|
|
// default control-plane backend; the JetStream KV store is opened separately
|
|
// (OpenJetStream) when decentralization is enabled.
|
|
func Open(path string) (Store, error) {
|
|
return openSQLite(path)
|
|
}
|
|
|
|
// openSQLite is the concrete constructor, returning *sqliteStore so internal
|
|
// callers (e.g. the SQLite->KV migration) can use SQLite-specific helpers that
|
|
// are not part of the storage-agnostic Store interface.
|
|
func openSQLite(path string) (*sqliteStore, error) {
|
|
// _pragma busy_timeout avoids spurious "database is locked" under concurrent
|
|
// HTTP handlers; foreign_keys kept off — we manage referential integrity in code.
|
|
dsn := fmt.Sprintf("file:%s?_pragma=busy_timeout(5000)&_pragma=journal_mode(WAL)", path)
|
|
db, err := sql.Open("sqlite", dsn)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("membership: open db: %w", err)
|
|
}
|
|
if err := db.Ping(); err != nil {
|
|
db.Close()
|
|
return nil, fmt.Errorf("membership: ping db: %w", err)
|
|
}
|
|
s := &sqliteStore{db: db}
|
|
if err := s.applyMigrations(); err != nil {
|
|
db.Close()
|
|
return nil, err
|
|
}
|
|
return s, nil
|
|
}
|
|
|
|
// Close closes the underlying database.
|
|
func (s *sqliteStore) Close() error { return s.db.Close() }
|
|
|
|
// applyMigrations runs every embedded migration in lexical order, tolerating
|
|
// the "already applied" errors that SQLite's non-idempotent DDL produces.
|
|
func (s *sqliteStore) applyMigrations() error {
|
|
files, err := fs.Glob(migrationsFS, "migrations/*.sql")
|
|
if err != nil {
|
|
return fmt.Errorf("membership: glob migrations: %w", err)
|
|
}
|
|
sort.Strings(files)
|
|
for _, f := range files {
|
|
b, err := migrationsFS.ReadFile(f)
|
|
if err != nil {
|
|
return fmt.Errorf("membership: read %s: %w", f, err)
|
|
}
|
|
if _, err := s.db.Exec(string(b)); err != nil {
|
|
msg := err.Error()
|
|
if !strings.Contains(msg, "duplicate column") && !strings.Contains(msg, "already exists") {
|
|
return fmt.Errorf("membership: apply %s: %w", f, err)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func nowRFC3339() string { return time.Now().UTC().Format(time.RFC3339Nano) }
|
|
|
|
// CreateRoom inserts a room at epoch 1, registers the owner as a member with
|
|
// role "owner", and stores the owner's sealed key for epoch 1. Idempotent
|
|
// inserts are not used: a duplicate room_id returns an error.
|
|
func (s *sqliteStore) CreateRoom(info RoomInfo, ownerSignPub, ownerKexPub, ownerSealedKey []byte) error {
|
|
tx, err := s.db.Begin()
|
|
if err != nil {
|
|
return fmt.Errorf("membership: begin: %w", err)
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
now := nowRFC3339()
|
|
if _, err := tx.Exec(
|
|
`INSERT INTO rooms (room_id, subject, key_epoch, encrypt, persist, sign_msgs, owner_endpoint, created_at)
|
|
VALUES (?, ?, ?, ?, ?, ?, ?, ?)`,
|
|
info.RoomID, info.Subject, 1,
|
|
b2i(info.Encrypt), b2i(info.Persist), b2i(info.SignMsgs),
|
|
info.OwnerEndpoint, now,
|
|
); err != nil {
|
|
return fmt.Errorf("membership: insert room: %w", err)
|
|
}
|
|
|
|
if _, err := tx.Exec(
|
|
`INSERT INTO members (room_id, endpoint, role, joined_at, sign_pub, kex_pub)
|
|
VALUES (?, ?, 'owner', ?, ?, ?)`,
|
|
info.RoomID, info.OwnerEndpoint, now, ownerSignPub, ownerKexPub,
|
|
); err != nil {
|
|
return fmt.Errorf("membership: insert owner member: %w", err)
|
|
}
|
|
|
|
if info.Encrypt {
|
|
if _, err := tx.Exec(
|
|
`INSERT INTO room_keys (room_id, epoch, endpoint, sealed_key) VALUES (?, 1, ?, ?)`,
|
|
info.RoomID, info.OwnerEndpoint, ownerSealedKey,
|
|
); err != nil {
|
|
return fmt.Errorf("membership: insert owner key: %w", err)
|
|
}
|
|
}
|
|
|
|
return tx.Commit()
|
|
}
|
|
|
|
// GetRoom returns room metadata (including current epoch).
|
|
func (s *sqliteStore) GetRoom(roomID string) (RoomInfo, error) {
|
|
var info RoomInfo
|
|
var enc, per, sgn int
|
|
err := s.db.QueryRow(
|
|
`SELECT room_id, subject, key_epoch, encrypt, persist, sign_msgs, owner_endpoint
|
|
FROM rooms WHERE room_id = ?`, roomID,
|
|
).Scan(&info.RoomID, &info.Subject, &info.Epoch, &enc, &per, &sgn, &info.OwnerEndpoint)
|
|
if err != nil {
|
|
return RoomInfo{}, fmt.Errorf("membership: get room %q: %w", roomID, err)
|
|
}
|
|
info.Encrypt, info.Persist, info.SignMsgs = enc != 0, per != 0, sgn != 0
|
|
return info, nil
|
|
}
|
|
|
|
// AddMember inserts a member at the given role and stores their sealed key for
|
|
// the supplied epoch.
|
|
func (s *sqliteStore) AddMember(roomID string, m Member, epoch int, sealedKey []byte) error {
|
|
tx, err := s.db.Begin()
|
|
if err != nil {
|
|
return fmt.Errorf("membership: begin: %w", err)
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
now := nowRFC3339()
|
|
if _, err := tx.Exec(
|
|
`INSERT INTO members (room_id, endpoint, role, joined_at, sign_pub, kex_pub)
|
|
VALUES (?, ?, ?, ?, ?, ?)`,
|
|
roomID, m.Endpoint, m.Role, now, m.SignPub, m.KexPub,
|
|
); err != nil {
|
|
return fmt.Errorf("membership: insert member: %w", err)
|
|
}
|
|
if len(sealedKey) > 0 {
|
|
if _, err := tx.Exec(
|
|
`INSERT INTO room_keys (room_id, epoch, endpoint, sealed_key) VALUES (?, ?, ?, ?)`,
|
|
roomID, epoch, m.Endpoint, sealedKey,
|
|
); err != nil {
|
|
return fmt.Errorf("membership: insert member key: %w", err)
|
|
}
|
|
}
|
|
return tx.Commit()
|
|
}
|
|
|
|
// GetMember returns a single member of a room.
|
|
func (s *sqliteStore) GetMember(roomID, endpoint string) (Member, error) {
|
|
var m Member
|
|
err := s.db.QueryRow(
|
|
`SELECT endpoint, role, sign_pub, kex_pub FROM members WHERE room_id = ? AND endpoint = ?`,
|
|
roomID, endpoint,
|
|
).Scan(&m.Endpoint, &m.Role, &m.SignPub, &m.KexPub)
|
|
if err != nil {
|
|
return Member{}, fmt.Errorf("membership: get member %q/%q: %w", roomID, endpoint, err)
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
// ListMembers returns all members of a room ordered by endpoint.
|
|
func (s *sqliteStore) ListMembers(roomID string) ([]Member, error) {
|
|
rows, err := s.db.Query(
|
|
`SELECT endpoint, role, sign_pub, kex_pub FROM members WHERE room_id = ? ORDER BY endpoint`,
|
|
roomID,
|
|
)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("membership: list members %q: %w", roomID, err)
|
|
}
|
|
defer rows.Close()
|
|
|
|
var out []Member
|
|
for rows.Next() {
|
|
var m Member
|
|
if err := rows.Scan(&m.Endpoint, &m.Role, &m.SignPub, &m.KexPub); err != nil {
|
|
return nil, fmt.Errorf("membership: scan member: %w", err)
|
|
}
|
|
out = append(out, m)
|
|
}
|
|
return out, rows.Err()
|
|
}
|
|
|
|
// RoomMembership is a room an endpoint belongs to, with that endpoint's role.
|
|
// It is the per-endpoint view used for room discovery (a peer asking "which
|
|
// rooms am I in?") so a freshly-invited member can find and join its rooms.
|
|
type RoomMembership struct {
|
|
RoomInfo
|
|
Role string
|
|
}
|
|
|
|
// ListRoomsForEndpoint returns every room the given endpoint is a member of,
|
|
// with the room's current metadata and the endpoint's role, ordered by room id.
|
|
// An endpoint that is in no rooms yields an empty slice (not an error).
|
|
func (s *sqliteStore) ListRoomsForEndpoint(endpoint string) ([]RoomMembership, error) {
|
|
rows, err := s.db.Query(
|
|
`SELECT r.room_id, r.subject, r.key_epoch, r.encrypt, r.persist, r.sign_msgs, r.owner_endpoint, m.role
|
|
FROM members m JOIN rooms r ON r.room_id = m.room_id
|
|
WHERE m.endpoint = ? ORDER BY r.room_id`,
|
|
endpoint,
|
|
)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("membership: list rooms for endpoint %q: %w", endpoint, err)
|
|
}
|
|
defer rows.Close()
|
|
|
|
var out []RoomMembership
|
|
for rows.Next() {
|
|
var rm RoomMembership
|
|
var enc, per, sgn int
|
|
if err := rows.Scan(&rm.RoomID, &rm.Subject, &rm.Epoch, &enc, &per, &sgn, &rm.OwnerEndpoint, &rm.Role); err != nil {
|
|
return nil, fmt.Errorf("membership: scan room membership: %w", err)
|
|
}
|
|
rm.Encrypt, rm.Persist, rm.SignMsgs = enc != 0, per != 0, sgn != 0
|
|
out = append(out, rm)
|
|
}
|
|
return out, rows.Err()
|
|
}
|
|
|
|
// GetSealedKey returns the sealed room key for an endpoint at a given epoch.
|
|
// If epoch <= 0, the latest epoch for that endpoint is returned.
|
|
func (s *sqliteStore) GetSealedKey(roomID, endpoint string, epoch int) (int, []byte, error) {
|
|
var ep int
|
|
var sealed []byte
|
|
var err error
|
|
if epoch <= 0 {
|
|
err = s.db.QueryRow(
|
|
`SELECT epoch, sealed_key FROM room_keys
|
|
WHERE room_id = ? AND endpoint = ? ORDER BY epoch DESC LIMIT 1`,
|
|
roomID, endpoint,
|
|
).Scan(&ep, &sealed)
|
|
} else {
|
|
err = s.db.QueryRow(
|
|
`SELECT epoch, sealed_key FROM room_keys
|
|
WHERE room_id = ? AND endpoint = ? AND epoch = ?`,
|
|
roomID, endpoint, epoch,
|
|
).Scan(&ep, &sealed)
|
|
}
|
|
if err != nil {
|
|
// Map "no such row" to the store-agnostic sentinel so the control plane
|
|
// can tell "not invited / no key yet" (-> 403 with a helpful message) from
|
|
// a genuine backend failure, the same way the KV store will.
|
|
if errors.Is(err, sql.ErrNoRows) {
|
|
return 0, nil, fmt.Errorf("membership: get sealed key %q/%q@%d: %w", roomID, endpoint, epoch, ErrNotFound)
|
|
}
|
|
return 0, nil, fmt.Errorf("membership: get sealed key %q/%q@%d: %w", roomID, endpoint, epoch, err)
|
|
}
|
|
return ep, sealed, nil
|
|
}
|
|
|
|
// PutSealedKeys stores a batch of sealed keys for the given epoch (endpoint ->
|
|
// sealed bytes), upserting on conflict so a rekey can overwrite stale entries.
|
|
func (s *sqliteStore) PutSealedKeys(roomID string, epoch int, keys map[string][]byte) error {
|
|
tx, err := s.db.Begin()
|
|
if err != nil {
|
|
return fmt.Errorf("membership: begin: %w", err)
|
|
}
|
|
defer tx.Rollback()
|
|
for endpoint, sealed := range keys {
|
|
if _, err := tx.Exec(
|
|
`INSERT INTO room_keys (room_id, epoch, endpoint, sealed_key) VALUES (?, ?, ?, ?)
|
|
ON CONFLICT(room_id, epoch, endpoint) DO UPDATE SET sealed_key = excluded.sealed_key`,
|
|
roomID, epoch, endpoint, sealed,
|
|
); err != nil {
|
|
return fmt.Errorf("membership: put sealed key for %q: %w", endpoint, err)
|
|
}
|
|
}
|
|
return tx.Commit()
|
|
}
|
|
|
|
// BumpEpoch sets the room's current key_epoch to newEpoch.
|
|
func (s *sqliteStore) BumpEpoch(roomID string, newEpoch int) error {
|
|
if _, err := s.db.Exec(`UPDATE rooms SET key_epoch = ? WHERE room_id = ?`, newEpoch, roomID); err != nil {
|
|
return fmt.Errorf("membership: bump epoch %q->%d: %w", roomID, newEpoch, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// RemoveMember deletes a member from a room. Their sealed keys for past epochs
|
|
// are left intact (they encrypt only data that member could already read).
|
|
func (s *sqliteStore) RemoveMember(roomID, endpoint string) error {
|
|
if _, err := s.db.Exec(`DELETE FROM members WHERE room_id = ? AND endpoint = ?`, roomID, endpoint); err != nil {
|
|
return fmt.Errorf("membership: remove member %q/%q: %w", roomID, endpoint, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func b2i(b bool) int {
|
|
if b {
|
|
return 1
|
|
}
|
|
return 0
|
|
}
|