|
| 1 | +"""Context assembly endpoint. |
| 2 | +
|
| 3 | +Provides per-task context packaging via POST /api/v1/context/assemble. |
| 4 | +Uses semantic search + dependency graph + project rules to build a |
| 5 | +minimal, precise context package for AI coding assistants. |
| 6 | +""" |
| 7 | +import time |
| 8 | +from typing import Any |
| 9 | + |
| 10 | +from fastapi import APIRouter, Depends, HTTPException |
| 11 | +from pydantic import BaseModel, Field |
| 12 | + |
| 13 | +from dependencies import verify_repo_access |
| 14 | +from middleware.auth import AuthContext, require_auth |
| 15 | +from services.observability import ( |
| 16 | + add_breadcrumb, |
| 17 | + capture_exception, |
| 18 | + logger, |
| 19 | + metrics, |
| 20 | + set_operation_context, |
| 21 | +) |
| 22 | + |
| 23 | +router = APIRouter(tags=["context"]) |
| 24 | + |
| 25 | + |
| 26 | +class AssembleRequest(BaseModel): |
| 27 | + task: str = Field(..., min_length=3, max_length=1000) |
| 28 | + repo_id: str |
| 29 | + token_budget: int = Field(default=1500, ge=100, le=10000) |
| 30 | + |
| 31 | + |
| 32 | +@router.post("/context/assemble") |
| 33 | +async def assemble_context( |
| 34 | + request: AssembleRequest, |
| 35 | + auth: AuthContext = Depends(require_auth), |
| 36 | +) -> dict[str, Any]: |
| 37 | + """Assemble task-specific context from semantic search + deps + rules. |
| 38 | +
|
| 39 | + Returns a markdown context package sized to fit within token_budget, |
| 40 | + containing only the files, dependencies, and project rules relevant |
| 41 | + to the given task description. |
| 42 | + """ |
| 43 | + set_operation_context( |
| 44 | + "context_assemble", |
| 45 | + user_id=auth.user_id, |
| 46 | + repo_id=request.repo_id, |
| 47 | + ) |
| 48 | + add_breadcrumb("Context assembly requested", category="context", repo_id=request.repo_id) |
| 49 | + |
| 50 | + verify_repo_access(request.repo_id, auth.user_id) |
| 51 | + |
| 52 | + from dependencies import context_assembler |
| 53 | + |
| 54 | + start = time.time() |
| 55 | + try: |
| 56 | + result = await context_assembler.assemble( |
| 57 | + task=request.task, |
| 58 | + repo_id=request.repo_id, |
| 59 | + user_id=auth.user_id, |
| 60 | + token_budget=request.token_budget, |
| 61 | + ) |
| 62 | + |
| 63 | + elapsed = time.time() - start |
| 64 | + logger.info( |
| 65 | + "Context assembled", |
| 66 | + repo_id=request.repo_id, |
| 67 | + files=result["files_found"], |
| 68 | + tokens=result["tokens_used"], |
| 69 | + budget=request.token_budget, |
| 70 | + duration_ms=round(elapsed * 1000), |
| 71 | + ) |
| 72 | + metrics.timing("context_assemble_ms", elapsed * 1000) |
| 73 | + |
| 74 | + return result |
| 75 | + |
| 76 | + except HTTPException: |
| 77 | + raise |
| 78 | + except Exception as exc: |
| 79 | + capture_exception(exc, operation="context_assemble", repo_id=request.repo_id) |
| 80 | + logger.error("Context assembly failed: %s", exc) |
| 81 | + raise HTTPException(status_code=500, detail="Context assembly failed") |
0 commit comments