Crypto + tokens
- app/core/security.py: Argon2id PasswordHasher (time_cost=2, memory_cost=
64 MiB, parallelism=2) + opaque-token SHA-256 helpers (raw token shown
once, only the hash lives in the DB).
- app/core/jwt_tokens.py: HS256, claims iss/sub/type/jti/iat/exp. Access
1h, refresh 30d.
Services
- services/auth.py: login, refresh with token rotation + reuse-detection
chain revoke, logout (idempotent), change_password (forces logout-all).
- services/invitations.py: create, preview, accept, revoke. Default 7d TTL.
- services/bootstrap.py: seeds the 3 system groups (admin/redteam/blueteam),
consumes the install token, attaches the first user to admin.
- core/install_token.py: mints, persists in settings, marks consumed,
regenerate hook for /diag/reset.
API
- POST /setup (consume install token, create 1st admin) + GET /setup
(status).
- POST /auth/{login,refresh,logout,change-password} + GET /auth/me.
- POST /invitations + GET /invitations + GET /invitations/preview/<token> +
POST /invitations/accept/<token> + POST /invitations/<id>/revoke.
- POST /diag/reset: test-only kill switch (truncate auth tables + mint
fresh install token). Allowed in dev too (with WARNING log) so the e2e
suite can run against a make-up stack; production locked out.
Middleware
- @require_auth populates g.current_user (snapshot dataclass, session
closed before request handler runs).
- @require_perm(*codes): atomic perm union check; admin group bypasses.
Perm catalogue lands in M3, scaffolding here.
- flask-limiter: 10/min/IP on /auth/login & /auth/refresh, 5/min on
/auth/change-password & /setup, 10–20/min on invitation endpoints.
Disabled in APP_ENV=test.
CLI
- flask --app app.cli metamorph print-install-token [--force]
- flask --app app.cli metamorph seed-mitre (M4 placeholder)
Refresh cookie metamorph_refresh: HttpOnly + Secure (localhost is a secure
context for modern browsers) + SameSite=Strict + Path=/api/v1/auth/.
Email validation: app.api._validation.Email permissive RFC-shape regex so
internal TLDs (.local/.corp/.test) are accepted — pydantic.EmailStr's
deliverability check is too strict for red-team labs.
Frontend
- lib/{api,auth}.ts: access token in module memory, refresh cookie,
automatic 401-retry via /auth/refresh, useAuth() hook.
- components/{Layout,RequireAuth}.tsx + ui/{TextField,Alert}.tsx.
- pages/{Login,Setup,Register,Profile}.
Testing
- tests/test_auth_flow.py: 15 integration tests (24 backend total).
- e2e/tests/m2-auth.spec.ts: 8 Playwright tests (20 e2e total).
- tasks/testing-m2.md.
DoD: make test-api → 24 passed, make e2e → 20 passed; spec-reviewer pass
applied (Secure unconditional, refresh limit 10/min/IP).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
189 lines
5.5 KiB
Python
189 lines
5.5 KiB
Python
"""Invitation flow: admin issues a one-shot URL token, invitee accepts.
|
|
|
|
The raw token is shown to the admin once (returned by `create_invitation`)
|
|
and never persisted — only its SHA-256 lives in the DB. Pre-assigned groups
|
|
are attached at creation and applied at acceptance.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import uuid
|
|
from dataclasses import dataclass
|
|
from datetime import datetime, timedelta, timezone
|
|
from typing import Iterable
|
|
|
|
from sqlalchemy import select
|
|
|
|
from app.core.security import (
|
|
generate_opaque_token,
|
|
hash_opaque_token,
|
|
hash_password,
|
|
)
|
|
from app.db.session import session_scope
|
|
from app.models.auth import Group, Invitation, InvitationGroup, User, UserGroup
|
|
|
|
INVITATION_TTL = timedelta(days=7)
|
|
|
|
|
|
class InvitationError(Exception):
|
|
pass
|
|
|
|
|
|
class InvitationExpired(InvitationError):
|
|
pass
|
|
|
|
|
|
class InvitationConsumed(InvitationError):
|
|
pass
|
|
|
|
|
|
class InvitationRevoked(InvitationError):
|
|
pass
|
|
|
|
|
|
@dataclass
|
|
class InvitationCreated:
|
|
invitation_id: uuid.UUID
|
|
raw_token: str
|
|
expires_at: datetime
|
|
|
|
|
|
@dataclass
|
|
class InvitationPreview:
|
|
email_hint: str | None
|
|
expires_at: datetime
|
|
groups: list[str]
|
|
is_valid: bool
|
|
reason: str | None
|
|
|
|
|
|
def _now() -> datetime:
|
|
return datetime.now(tz=timezone.utc)
|
|
|
|
|
|
def create_invitation(
|
|
*,
|
|
created_by_user_id: uuid.UUID,
|
|
email_hint: str | None,
|
|
group_ids: Iterable[uuid.UUID] = (),
|
|
ttl: timedelta = INVITATION_TTL,
|
|
) -> InvitationCreated:
|
|
raw = generate_opaque_token()
|
|
expires_at = _now() + ttl
|
|
with session_scope() as s:
|
|
inv = Invitation(
|
|
token_hash=hash_opaque_token(raw),
|
|
email_hint=email_hint.strip().lower() if email_hint else None,
|
|
created_by_user_id=created_by_user_id,
|
|
expires_at=expires_at,
|
|
)
|
|
s.add(inv)
|
|
s.flush()
|
|
for gid in group_ids:
|
|
s.add(InvitationGroup(invitation_id=inv.id, group_id=gid))
|
|
return InvitationCreated(
|
|
invitation_id=inv.id,
|
|
raw_token=raw,
|
|
expires_at=expires_at,
|
|
)
|
|
|
|
|
|
def _load_by_token(s, raw_token: str) -> Invitation | None:
|
|
return s.scalar(
|
|
select(Invitation).where(Invitation.token_hash == hash_opaque_token(raw_token))
|
|
)
|
|
|
|
|
|
def preview(raw_token: str) -> InvitationPreview:
|
|
with session_scope() as s:
|
|
inv = _load_by_token(s, raw_token)
|
|
if inv is None:
|
|
return InvitationPreview(None, _now(), [], False, "not_found")
|
|
groups = [g.name for g in inv.pre_assigned_groups]
|
|
if inv.revoked_at is not None:
|
|
return InvitationPreview(inv.email_hint, inv.expires_at, groups, False, "revoked")
|
|
if inv.consumed_at is not None:
|
|
return InvitationPreview(inv.email_hint, inv.expires_at, groups, False, "consumed")
|
|
if inv.expires_at <= _now():
|
|
return InvitationPreview(inv.email_hint, inv.expires_at, groups, False, "expired")
|
|
return InvitationPreview(inv.email_hint, inv.expires_at, groups, True, None)
|
|
|
|
|
|
def accept(raw_token: str, *, email: str, password: str, display_name: str | None) -> uuid.UUID:
|
|
"""Create the user, attach pre-assigned groups, mark invitation consumed."""
|
|
if len(password) < 8:
|
|
raise ValueError("password must be at least 8 characters")
|
|
|
|
email_norm = email.strip().lower()
|
|
with session_scope() as s:
|
|
inv = _load_by_token(s, raw_token)
|
|
if inv is None:
|
|
raise InvitationError("invitation not found")
|
|
if inv.revoked_at is not None:
|
|
raise InvitationRevoked("invitation revoked")
|
|
if inv.consumed_at is not None:
|
|
raise InvitationConsumed("invitation already consumed")
|
|
if inv.expires_at <= _now():
|
|
raise InvitationExpired("invitation expired")
|
|
|
|
# Email must not be already in use among active users.
|
|
existing = s.scalar(
|
|
select(User).where(User.email == email_norm, User.deleted_at.is_(None))
|
|
)
|
|
if existing is not None:
|
|
raise ValueError("email already in use")
|
|
|
|
user = User(
|
|
email=email_norm,
|
|
display_name=(display_name or "").strip() or None,
|
|
password_hash=hash_password(password),
|
|
)
|
|
s.add(user)
|
|
s.flush()
|
|
|
|
for grp in inv.pre_assigned_groups:
|
|
s.add(UserGroup(user_id=user.id, group_id=grp.id))
|
|
|
|
inv.consumed_at = _now()
|
|
inv.consumed_by_user_id = user.id
|
|
return user.id
|
|
|
|
|
|
def revoke(invitation_id: uuid.UUID) -> bool:
|
|
with session_scope() as s:
|
|
inv = s.get(Invitation, invitation_id)
|
|
if inv is None:
|
|
return False
|
|
if inv.revoked_at is not None or inv.consumed_at is not None:
|
|
return False
|
|
inv.revoked_at = _now()
|
|
return True
|
|
|
|
|
|
def list_active(*, limit: int = 100) -> list[Invitation]:
|
|
with session_scope() as s:
|
|
rows = s.scalars(
|
|
select(Invitation)
|
|
.where(
|
|
Invitation.consumed_at.is_(None),
|
|
Invitation.revoked_at.is_(None),
|
|
Invitation.expires_at > _now(),
|
|
)
|
|
.order_by(Invitation.created_at.desc())
|
|
.limit(limit)
|
|
).all()
|
|
# detach so caller can read after session closes
|
|
for r in rows:
|
|
s.expunge(r)
|
|
for g in r.pre_assigned_groups:
|
|
s.expunge(g)
|
|
return list(rows)
|
|
|
|
|
|
def find_group_id_by_name(name: str) -> uuid.UUID | None:
|
|
with session_scope() as s:
|
|
gid = s.scalar(
|
|
select(Group.id).where(Group.name == name, Group.deleted_at.is_(None))
|
|
)
|
|
return gid
|