Files
claudetools/api/routers/coord_locks.py
Mike Swanson 73573800b0 feat: coord API — no-auth, DB softfail 503, agent tracking protocol
- coord routers: removed JWT auth requirement (internal-only endpoints)
- error_handler: SQLAlchemy OperationalError/DisconnectionError → 503
  with Retry-After: 30 header instead of 500
- /health: live DB probe (SELECT 1) instead of static response
- CLAUDE.md: "Live State Tracking" section with full agent protocol
  for all projects — session start, lock claim/release, component
  state updates, softfail + local queue catch-up
- COORDINATION_PROTOCOL.md: softfail/catch-up section + server-side
  503 behavior documented

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-12 08:45:33 -07:00

76 lines
2.6 KiB
Python

"""Coordination session locks router."""
from uuid import UUID
from fastapi import APIRouter, Depends, Query, status
from sqlalchemy.orm import Session
from api.database import get_db
from api.schemas.coord_session_lock import CoordSessionLockCreate, CoordSessionLockResponse
from api.services import coord_lock_service
router = APIRouter()
@router.get("", response_model=dict, status_code=status.HTTP_200_OK)
def list_active_locks(
project_key: str | None = Query(default=None),
session_id: str | None = Query(default=None),
skip: int = Query(default=0, ge=0),
limit: int = Query(default=100, ge=1, le=1000),
db: Session = Depends(get_db),
):
"""List currently active locks with optional filters."""
locks, total = coord_lock_service.get_active_locks(
db, project_key=project_key, session_id=session_id, skip=skip, limit=limit
)
return {
"total": total,
"skip": skip,
"limit": limit,
"locks": [CoordSessionLockResponse.model_validate(l) for l in locks],
}
@router.get("/check", response_model=dict, status_code=status.HTTP_200_OK)
def check_resource_locked(
project_key: str = Query(...),
resource: str = Query(...),
db: Session = Depends(get_db),
):
"""Check whether a resource is currently locked."""
lock = coord_lock_service.check_resource_locked(db, project_key, resource)
if lock:
return {"locked": True, "lock": CoordSessionLockResponse.model_validate(lock)}
return {"locked": False}
@router.post("", response_model=CoordSessionLockResponse, status_code=status.HTTP_201_CREATED)
def claim_lock(
data: CoordSessionLockCreate,
db: Session = Depends(get_db),
):
"""Claim a resource lock for a session."""
lock = coord_lock_service.claim_lock(db, data)
return CoordSessionLockResponse.model_validate(lock)
@router.delete("", response_model=dict, status_code=status.HTTP_200_OK)
def release_all_session_locks(
session_id: str = Query(..., description="Release all active locks held by this session"),
db: Session = Depends(get_db),
):
"""Release all active locks for a session (call on session end)."""
return coord_lock_service.release_all_session_locks(db, session_id)
@router.delete("/{lock_id}", response_model=CoordSessionLockResponse, status_code=status.HTTP_200_OK)
def release_lock(
lock_id: UUID,
session_id: str = Query(..., description="Must match the session that claimed the lock"),
db: Session = Depends(get_db),
):
"""Release a specific lock by ID."""
lock = coord_lock_service.release_lock(db, lock_id, session_id)
return CoordSessionLockResponse.model_validate(lock)