Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
"""
|
|
|
|
|
Kin Web API — FastAPI backend reading ~/.kin/kin.db via core.models.
|
|
|
|
|
Run: uvicorn web.api:app --reload --port 8420
|
|
|
|
|
"""
|
|
|
|
|
|
2026-03-16 06:59:46 +02:00
|
|
|
import logging
|
|
|
|
|
import shutil
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
import subprocess
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
import sys
|
|
|
|
|
from pathlib import Path
|
|
|
|
|
|
|
|
|
|
# Ensure project root on sys.path
|
|
|
|
|
sys.path.insert(0, str(Path(__file__).parent.parent))
|
|
|
|
|
|
|
|
|
|
from fastapi import FastAPI, HTTPException, Query
|
|
|
|
|
from fastapi.middleware.cors import CORSMiddleware
|
2026-03-15 17:11:38 +02:00
|
|
|
from fastapi.responses import JSONResponse, FileResponse
|
|
|
|
|
from fastapi.staticfiles import StaticFiles
|
2026-03-16 09:43:26 +02:00
|
|
|
from pydantic import BaseModel, model_validator
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
|
|
|
|
|
from core.db import init_db
|
|
|
|
|
from core import models
|
2026-03-16 08:21:13 +02:00
|
|
|
from core.models import VALID_COMPLETION_MODES, TASK_CATEGORIES
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
from agents.bootstrap import (
|
|
|
|
|
detect_tech_stack, detect_modules, extract_decisions_from_claude_md,
|
|
|
|
|
find_vault_root, scan_obsidian, save_to_db,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
DB_PATH = Path.home() / ".kin" / "kin.db"
|
|
|
|
|
|
2026-03-16 06:59:46 +02:00
|
|
|
_logger = logging.getLogger("kin")
|
|
|
|
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Startup: verify claude CLI is available in PATH
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
def _check_claude_available() -> None:
|
|
|
|
|
"""Warn at startup if claude CLI cannot be found in PATH.
|
|
|
|
|
|
|
|
|
|
launchctl daemons run with a stripped environment and may not see
|
|
|
|
|
/opt/homebrew/bin where claude is typically installed.
|
|
|
|
|
See Decision #28.
|
|
|
|
|
"""
|
|
|
|
|
from agents.runner import _build_claude_env # avoid circular import at module level
|
|
|
|
|
env = _build_claude_env()
|
|
|
|
|
claude_path = shutil.which("claude", path=env["PATH"])
|
|
|
|
|
if claude_path:
|
|
|
|
|
_logger.info("claude CLI found: %s", claude_path)
|
|
|
|
|
else:
|
|
|
|
|
_logger.warning(
|
|
|
|
|
"WARNING: claude CLI not found in PATH (%s). "
|
|
|
|
|
"Agent pipelines will fail with returncode 127. "
|
|
|
|
|
"Fix: add /opt/homebrew/bin to EnvironmentVariables.PATH in "
|
|
|
|
|
"~/Library/LaunchAgents/com.kin.api.plist and reload with: "
|
|
|
|
|
"launchctl unload ~/Library/LaunchAgents/com.kin.api.plist && "
|
|
|
|
|
"launchctl load ~/Library/LaunchAgents/com.kin.api.plist",
|
|
|
|
|
env.get("PATH", ""),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def _check_git_available() -> None:
|
|
|
|
|
"""Warn at startup if git cannot be found in PATH.
|
|
|
|
|
|
|
|
|
|
launchctl daemons run with a stripped environment and may not see
|
|
|
|
|
git in the standard directories. See Decision #28.
|
|
|
|
|
"""
|
|
|
|
|
from agents.runner import _build_claude_env # avoid circular import at module level
|
|
|
|
|
env = _build_claude_env()
|
|
|
|
|
git_path = shutil.which("git", path=env["PATH"])
|
|
|
|
|
if git_path:
|
|
|
|
|
_logger.info("git found: %s", git_path)
|
|
|
|
|
else:
|
|
|
|
|
_logger.warning(
|
|
|
|
|
"WARNING: git not found in PATH (%s). "
|
|
|
|
|
"Autocommit will fail silently. "
|
|
|
|
|
"Fix: add git directory to EnvironmentVariables.PATH in "
|
|
|
|
|
"~/Library/LaunchAgents/com.kin.api.plist and reload with: "
|
|
|
|
|
"launchctl unload ~/Library/LaunchAgents/com.kin.api.plist && "
|
|
|
|
|
"launchctl load ~/Library/LaunchAgents/com.kin.api.plist",
|
|
|
|
|
env.get("PATH", ""),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
_check_claude_available()
|
|
|
|
|
_check_git_available()
|
|
|
|
|
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
app = FastAPI(title="Kin API", version="0.1.0")
|
|
|
|
|
|
|
|
|
|
app.add_middleware(
|
|
|
|
|
CORSMiddleware,
|
2026-03-15 17:11:38 +02:00
|
|
|
allow_origins=["*"],
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
allow_methods=["*"],
|
|
|
|
|
allow_headers=["*"],
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_conn():
|
|
|
|
|
return init_db(DB_PATH)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Projects
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
@app.get("/api/projects")
|
|
|
|
|
def list_projects(status: str | None = None):
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
summary = models.get_project_summary(conn)
|
|
|
|
|
if status:
|
|
|
|
|
summary = [s for s in summary if s["status"] == status]
|
|
|
|
|
conn.close()
|
|
|
|
|
return summary
|
|
|
|
|
|
|
|
|
|
|
2026-03-16 09:13:34 +02:00
|
|
|
class NewProjectCreate(BaseModel):
|
|
|
|
|
id: str
|
|
|
|
|
name: str
|
2026-03-16 09:47:56 +02:00
|
|
|
path: str | None = None
|
2026-03-16 09:13:34 +02:00
|
|
|
description: str
|
|
|
|
|
roles: list[str]
|
|
|
|
|
tech_stack: list[str] | None = None
|
|
|
|
|
priority: int = 5
|
|
|
|
|
language: str = "ru"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/projects/new")
|
|
|
|
|
def new_project_with_phases(body: NewProjectCreate):
|
|
|
|
|
"""Create project + sequential research phases (KIN-059)."""
|
|
|
|
|
from core.phases import create_project_with_phases, validate_roles
|
|
|
|
|
clean_roles = validate_roles(body.roles)
|
|
|
|
|
if not clean_roles:
|
|
|
|
|
raise HTTPException(400, "At least one research role must be selected (excluding architect)")
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
if models.get_project(conn, body.id):
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(409, f"Project '{body.id}' already exists")
|
|
|
|
|
try:
|
|
|
|
|
result = create_project_with_phases(
|
|
|
|
|
conn, body.id, body.name, body.path,
|
|
|
|
|
description=body.description,
|
|
|
|
|
selected_roles=clean_roles,
|
|
|
|
|
tech_stack=body.tech_stack,
|
|
|
|
|
priority=body.priority,
|
|
|
|
|
language=body.language,
|
|
|
|
|
)
|
|
|
|
|
except ValueError as e:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(400, str(e))
|
|
|
|
|
conn.close()
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
@app.get("/api/projects/{project_id}")
|
|
|
|
|
def get_project(project_id: str):
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, project_id)
|
|
|
|
|
if not p:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Project '{project_id}' not found")
|
|
|
|
|
tasks = models.list_tasks(conn, project_id=project_id)
|
|
|
|
|
mods = models.get_modules(conn, project_id)
|
|
|
|
|
decisions = models.get_decisions(conn, project_id)
|
|
|
|
|
conn.close()
|
|
|
|
|
return {**p, "tasks": tasks, "modules": mods, "decisions": decisions}
|
|
|
|
|
|
|
|
|
|
|
2026-03-16 09:13:34 +02:00
|
|
|
VALID_PROJECT_TYPES = {"development", "operations", "research"}
|
|
|
|
|
|
|
|
|
|
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
class ProjectCreate(BaseModel):
|
|
|
|
|
id: str
|
|
|
|
|
name: str
|
2026-03-16 09:47:56 +02:00
|
|
|
path: str | None = None
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
tech_stack: list[str] | None = None
|
|
|
|
|
status: str = "active"
|
|
|
|
|
priority: int = 5
|
2026-03-16 09:13:34 +02:00
|
|
|
project_type: str = "development"
|
|
|
|
|
ssh_host: str | None = None
|
|
|
|
|
ssh_user: str | None = None
|
|
|
|
|
ssh_key_path: str | None = None
|
|
|
|
|
ssh_proxy_jump: str | None = None
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
|
2026-03-16 09:43:26 +02:00
|
|
|
@model_validator(mode="after")
|
2026-03-16 09:47:56 +02:00
|
|
|
def validate_fields(self) -> "ProjectCreate":
|
2026-03-16 09:43:26 +02:00
|
|
|
if self.project_type == "operations" and not self.ssh_host:
|
|
|
|
|
raise ValueError("ssh_host is required for operations projects")
|
2026-03-16 09:47:56 +02:00
|
|
|
if self.project_type != "operations" and not self.path:
|
|
|
|
|
raise ValueError("path is required for non-operations projects")
|
2026-03-16 09:43:26 +02:00
|
|
|
return self
|
|
|
|
|
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
|
2026-03-15 20:02:01 +02:00
|
|
|
class ProjectPatch(BaseModel):
|
2026-03-16 07:06:34 +02:00
|
|
|
execution_mode: str | None = None
|
|
|
|
|
autocommit_enabled: bool | None = None
|
2026-03-16 07:14:32 +02:00
|
|
|
obsidian_vault_path: str | None = None
|
2026-03-16 08:21:13 +02:00
|
|
|
deploy_command: str | None = None
|
2026-03-16 09:13:34 +02:00
|
|
|
project_type: str | None = None
|
|
|
|
|
ssh_host: str | None = None
|
|
|
|
|
ssh_user: str | None = None
|
|
|
|
|
ssh_key_path: str | None = None
|
|
|
|
|
ssh_proxy_jump: str | None = None
|
2026-03-15 20:02:01 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.patch("/api/projects/{project_id}")
|
|
|
|
|
def patch_project(project_id: str, body: ProjectPatch):
|
2026-03-16 09:13:34 +02:00
|
|
|
has_any = any([
|
|
|
|
|
body.execution_mode, body.autocommit_enabled is not None,
|
|
|
|
|
body.obsidian_vault_path, body.deploy_command is not None,
|
|
|
|
|
body.project_type, body.ssh_host is not None,
|
|
|
|
|
body.ssh_user is not None, body.ssh_key_path is not None,
|
|
|
|
|
body.ssh_proxy_jump is not None,
|
|
|
|
|
])
|
|
|
|
|
if not has_any:
|
|
|
|
|
raise HTTPException(400, "Nothing to update.")
|
2026-03-16 07:06:34 +02:00
|
|
|
if body.execution_mode is not None and body.execution_mode not in VALID_EXECUTION_MODES:
|
2026-03-15 20:02:01 +02:00
|
|
|
raise HTTPException(400, f"Invalid execution_mode '{body.execution_mode}'. Must be one of: {', '.join(VALID_EXECUTION_MODES)}")
|
2026-03-16 09:13:34 +02:00
|
|
|
if body.project_type is not None and body.project_type not in VALID_PROJECT_TYPES:
|
|
|
|
|
raise HTTPException(400, f"Invalid project_type '{body.project_type}'. Must be one of: {', '.join(VALID_PROJECT_TYPES)}")
|
2026-03-15 20:02:01 +02:00
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, project_id)
|
|
|
|
|
if not p:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Project '{project_id}' not found")
|
2026-03-16 07:06:34 +02:00
|
|
|
fields = {}
|
|
|
|
|
if body.execution_mode is not None:
|
|
|
|
|
fields["execution_mode"] = body.execution_mode
|
|
|
|
|
if body.autocommit_enabled is not None:
|
|
|
|
|
fields["autocommit_enabled"] = int(body.autocommit_enabled)
|
2026-03-16 07:14:32 +02:00
|
|
|
if body.obsidian_vault_path is not None:
|
|
|
|
|
fields["obsidian_vault_path"] = body.obsidian_vault_path
|
2026-03-16 08:21:13 +02:00
|
|
|
if body.deploy_command is not None:
|
|
|
|
|
# Empty string = sentinel for clearing (decision #68)
|
|
|
|
|
fields["deploy_command"] = None if body.deploy_command == "" else body.deploy_command
|
2026-03-16 09:13:34 +02:00
|
|
|
if body.project_type is not None:
|
|
|
|
|
fields["project_type"] = body.project_type
|
|
|
|
|
if body.ssh_host is not None:
|
|
|
|
|
fields["ssh_host"] = body.ssh_host
|
|
|
|
|
if body.ssh_user is not None:
|
|
|
|
|
fields["ssh_user"] = body.ssh_user
|
|
|
|
|
if body.ssh_key_path is not None:
|
|
|
|
|
fields["ssh_key_path"] = body.ssh_key_path
|
|
|
|
|
if body.ssh_proxy_jump is not None:
|
|
|
|
|
fields["ssh_proxy_jump"] = body.ssh_proxy_jump
|
2026-03-16 07:06:34 +02:00
|
|
|
models.update_project(conn, project_id, **fields)
|
2026-03-15 20:02:01 +02:00
|
|
|
p = models.get_project(conn, project_id)
|
|
|
|
|
conn.close()
|
|
|
|
|
return p
|
|
|
|
|
|
|
|
|
|
|
2026-03-16 07:14:32 +02:00
|
|
|
@app.post("/api/projects/{project_id}/sync/obsidian")
|
|
|
|
|
def sync_obsidian_endpoint(project_id: str):
|
|
|
|
|
"""Запускает двусторонний Obsidian sync для проекта."""
|
|
|
|
|
from core.obsidian_sync import sync_obsidian
|
|
|
|
|
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, project_id)
|
|
|
|
|
if not p:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Project '{project_id}' not found")
|
|
|
|
|
if not p.get("obsidian_vault_path"):
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(400, "obsidian_vault_path not set for this project")
|
|
|
|
|
result = sync_obsidian(conn, project_id)
|
|
|
|
|
conn.close()
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
2026-03-16 08:21:13 +02:00
|
|
|
@app.post("/api/projects/{project_id}/deploy")
|
|
|
|
|
def deploy_project(project_id: str):
|
|
|
|
|
"""Execute deploy_command for a project. Returns stdout/stderr/exit_code.
|
|
|
|
|
|
|
|
|
|
# WARNING: shell=True — deploy_command is admin-only, set in Settings by the project owner.
|
|
|
|
|
"""
|
|
|
|
|
import time
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, project_id)
|
|
|
|
|
conn.close()
|
|
|
|
|
if not p:
|
|
|
|
|
raise HTTPException(404, f"Project '{project_id}' not found")
|
|
|
|
|
deploy_command = p.get("deploy_command")
|
|
|
|
|
if not deploy_command:
|
|
|
|
|
raise HTTPException(400, "deploy_command not set for this project")
|
|
|
|
|
cwd = p.get("path") or None
|
|
|
|
|
start = time.monotonic()
|
|
|
|
|
try:
|
|
|
|
|
result = subprocess.run(
|
|
|
|
|
deploy_command,
|
|
|
|
|
shell=True, # WARNING: shell=True — command is admin-only
|
|
|
|
|
cwd=cwd,
|
|
|
|
|
capture_output=True,
|
|
|
|
|
text=True,
|
|
|
|
|
timeout=60,
|
|
|
|
|
)
|
|
|
|
|
except subprocess.TimeoutExpired:
|
|
|
|
|
raise HTTPException(504, "Deploy command timed out after 60 seconds")
|
|
|
|
|
except Exception as e:
|
|
|
|
|
raise HTTPException(500, f"Deploy failed: {e}")
|
|
|
|
|
duration = round(time.monotonic() - start, 2)
|
|
|
|
|
return {
|
|
|
|
|
"success": result.returncode == 0,
|
|
|
|
|
"exit_code": result.returncode,
|
|
|
|
|
"stdout": result.stdout,
|
|
|
|
|
"stderr": result.stderr,
|
|
|
|
|
"duration_seconds": duration,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
@app.post("/api/projects")
|
|
|
|
|
def create_project(body: ProjectCreate):
|
2026-03-16 09:13:34 +02:00
|
|
|
if body.project_type not in VALID_PROJECT_TYPES:
|
|
|
|
|
raise HTTPException(400, f"Invalid project_type '{body.project_type}'. Must be one of: {', '.join(VALID_PROJECT_TYPES)}")
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
conn = get_conn()
|
|
|
|
|
if models.get_project(conn, body.id):
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(409, f"Project '{body.id}' already exists")
|
|
|
|
|
p = models.create_project(
|
|
|
|
|
conn, body.id, body.name, body.path,
|
|
|
|
|
tech_stack=body.tech_stack, status=body.status, priority=body.priority,
|
2026-03-16 09:13:34 +02:00
|
|
|
project_type=body.project_type,
|
|
|
|
|
ssh_host=body.ssh_host,
|
|
|
|
|
ssh_user=body.ssh_user,
|
|
|
|
|
ssh_key_path=body.ssh_key_path,
|
|
|
|
|
ssh_proxy_jump=body.ssh_proxy_jump,
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
)
|
|
|
|
|
conn.close()
|
|
|
|
|
return p
|
|
|
|
|
|
|
|
|
|
|
2026-03-16 09:13:34 +02:00
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Phases (KIN-059)
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
@app.get("/api/projects/{project_id}/phases")
|
|
|
|
|
def get_project_phases(project_id: str):
|
|
|
|
|
"""List research phases for a project, with task data joined."""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, project_id)
|
|
|
|
|
if not p:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Project '{project_id}' not found")
|
|
|
|
|
phases = models.list_phases(conn, project_id)
|
|
|
|
|
result = []
|
|
|
|
|
for phase in phases:
|
|
|
|
|
task = models.get_task(conn, phase["task_id"]) if phase.get("task_id") else None
|
|
|
|
|
result.append({**phase, "task": task})
|
|
|
|
|
conn.close()
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class PhaseApprove(BaseModel):
|
|
|
|
|
comment: str | None = None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class PhaseReject(BaseModel):
|
|
|
|
|
reason: str
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class PhaseRevise(BaseModel):
|
|
|
|
|
comment: str
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/phases/{phase_id}/approve")
|
|
|
|
|
def approve_phase(phase_id: int, body: PhaseApprove | None = None):
|
|
|
|
|
"""Approve a research phase and activate the next one."""
|
|
|
|
|
from core.phases import approve_phase as _approve
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
phase = models.get_phase(conn, phase_id)
|
|
|
|
|
if not phase:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Phase {phase_id} not found")
|
|
|
|
|
try:
|
|
|
|
|
result = _approve(conn, phase_id)
|
|
|
|
|
except ValueError as e:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(400, str(e))
|
2026-03-16 09:43:26 +02:00
|
|
|
# Mark the phase's task as done for consistency
|
|
|
|
|
if phase.get("task_id"):
|
|
|
|
|
models.update_task(conn, phase["task_id"], status="done")
|
2026-03-16 09:13:34 +02:00
|
|
|
conn.close()
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/phases/{phase_id}/reject")
|
|
|
|
|
def reject_phase(phase_id: int, body: PhaseReject):
|
|
|
|
|
"""Reject a research phase."""
|
|
|
|
|
from core.phases import reject_phase as _reject
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
phase = models.get_phase(conn, phase_id)
|
|
|
|
|
if not phase:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Phase {phase_id} not found")
|
|
|
|
|
try:
|
|
|
|
|
result = _reject(conn, phase_id, body.reason)
|
|
|
|
|
except ValueError as e:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(400, str(e))
|
|
|
|
|
conn.close()
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/phases/{phase_id}/revise")
|
|
|
|
|
def revise_phase(phase_id: int, body: PhaseRevise):
|
|
|
|
|
"""Request revision for a research phase."""
|
|
|
|
|
from core.phases import revise_phase as _revise
|
|
|
|
|
if not body.comment.strip():
|
|
|
|
|
raise HTTPException(400, "comment is required")
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
phase = models.get_phase(conn, phase_id)
|
|
|
|
|
if not phase:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Phase {phase_id} not found")
|
|
|
|
|
try:
|
|
|
|
|
result = _revise(conn, phase_id, body.comment)
|
|
|
|
|
except ValueError as e:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(400, str(e))
|
|
|
|
|
conn.close()
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
kin: KIN-059 Workflow new_project с выбором команды. При создании нового проекта через GUI или CLI директор описывает проект свободным текстом и выбирает галочками какие этапы research нужны: ☐ Business analyst (бизнес-модель, аудитория, монетизация) ☐ Market researcher (конкуренты, ниша, отзывы, сильные/слабые стороны) ☐ Legal researcher (юрисдикция, лицензии, KYC/AML, GDPR) ☐ Tech researcher (API, ограничения, стоимость, альтернативы) ☐ UX designer (анализ UX конкурентов, user journey, wireframes) ☐ Marketer (стратегия продвижения, SEO, conversion-паттерны) ☐ Architect (blueprint на основе одобренных research'ей) — всегда последний Architect включается автоматически если выбран хотя бы один researcher. Каждый выбранный этап — отдельная задача на review. Директор одобряет, отклоняет, или просит доисследовать (Revise). Следующий этап только после approve предыдущего. GUI: форма 'New Project' с описанием + чекбоксы ролей + кнопка 'Start Research'. CLI: kin new-project 'описание' --roles 'business,market,tech,architect'
2026-03-16 09:30:00 +02:00
|
|
|
@app.post("/api/projects/{project_id}/phases/start")
|
|
|
|
|
def start_project_phase(project_id: str):
|
|
|
|
|
"""Launch agent for the current active/revising phase in background. Returns 202.
|
|
|
|
|
|
|
|
|
|
Finds the first phase with status 'active' or 'revising', sets its task to
|
|
|
|
|
in_progress, spawns a background subprocess (same as /api/tasks/{id}/run),
|
|
|
|
|
and returns immediately so the HTTP request doesn't block on agent execution.
|
|
|
|
|
"""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, project_id)
|
|
|
|
|
if not p:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Project '{project_id}' not found")
|
|
|
|
|
|
|
|
|
|
phases = models.list_phases(conn, project_id)
|
|
|
|
|
active_phase = next(
|
|
|
|
|
(ph for ph in phases if ph["status"] in ("active", "revising")), None
|
|
|
|
|
)
|
|
|
|
|
if not active_phase:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"No active or revising phase for project '{project_id}'")
|
|
|
|
|
|
|
|
|
|
task_id = active_phase.get("task_id")
|
|
|
|
|
if not task_id:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(400, f"Phase {active_phase['id']} has no task assigned")
|
|
|
|
|
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
|
|
|
|
|
|
|
|
|
models.update_task(conn, task_id, status="in_progress")
|
|
|
|
|
conn.close()
|
|
|
|
|
|
|
|
|
|
kin_root = Path(__file__).parent.parent
|
|
|
|
|
cmd = [sys.executable, "-m", "cli.main", "--db", str(DB_PATH), "run", task_id]
|
|
|
|
|
cmd.append("--allow-write")
|
|
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
env = os.environ.copy()
|
|
|
|
|
env["KIN_NONINTERACTIVE"] = "1"
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
proc = subprocess.Popen(
|
|
|
|
|
cmd,
|
|
|
|
|
cwd=str(kin_root),
|
|
|
|
|
stdout=subprocess.DEVNULL,
|
|
|
|
|
stderr=subprocess.DEVNULL,
|
|
|
|
|
stdin=subprocess.DEVNULL,
|
|
|
|
|
env=env,
|
|
|
|
|
)
|
|
|
|
|
_logger.info("Phase agent started for task %s (phase %d), pid=%d",
|
|
|
|
|
task_id, active_phase["id"], proc.pid)
|
|
|
|
|
except Exception as e:
|
|
|
|
|
raise HTTPException(500, f"Failed to start phase agent: {e}")
|
|
|
|
|
|
|
|
|
|
return JSONResponse(
|
|
|
|
|
{"status": "started", "phase_id": active_phase["id"], "task_id": task_id},
|
|
|
|
|
status_code=202,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Tasks
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
@app.get("/api/tasks/{task_id}")
|
|
|
|
|
def get_task(task_id: str):
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
conn.close()
|
|
|
|
|
if not t:
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
|
|
|
|
return t
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class TaskCreate(BaseModel):
|
|
|
|
|
project_id: str
|
|
|
|
|
title: str
|
|
|
|
|
priority: int = 5
|
|
|
|
|
route_type: str | None = None
|
2026-03-16 08:21:13 +02:00
|
|
|
category: str | None = None
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/tasks")
|
|
|
|
|
def create_task(body: TaskCreate):
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, body.project_id)
|
|
|
|
|
if not p:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Project '{body.project_id}' not found")
|
2026-03-16 08:21:13 +02:00
|
|
|
category = None
|
|
|
|
|
if body.category:
|
|
|
|
|
category = body.category.upper()
|
|
|
|
|
if category not in TASK_CATEGORIES:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(400, f"Invalid category '{category}'. Must be one of: {', '.join(TASK_CATEGORIES)}")
|
|
|
|
|
task_id = models.next_task_id(conn, body.project_id, category=category)
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
brief = {"route_type": body.route_type} if body.route_type else None
|
|
|
|
|
t = models.create_task(conn, task_id, body.project_id, body.title,
|
2026-03-16 08:21:13 +02:00
|
|
|
priority=body.priority, brief=brief, category=category)
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
conn.close()
|
|
|
|
|
return t
|
|
|
|
|
|
|
|
|
|
|
2026-03-16 07:13:32 +02:00
|
|
|
VALID_ROUTE_TYPES = {"debug", "feature", "refactor", "hotfix"}
|
|
|
|
|
|
|
|
|
|
|
2026-03-15 18:17:57 +02:00
|
|
|
class TaskPatch(BaseModel):
|
2026-03-15 20:02:01 +02:00
|
|
|
status: str | None = None
|
|
|
|
|
execution_mode: str | None = None
|
2026-03-16 07:13:32 +02:00
|
|
|
priority: int | None = None
|
|
|
|
|
route_type: str | None = None
|
|
|
|
|
title: str | None = None
|
|
|
|
|
brief_text: str | None = None
|
2026-03-15 18:17:57 +02:00
|
|
|
|
|
|
|
|
|
2026-03-15 23:22:49 +02:00
|
|
|
VALID_STATUSES = set(models.VALID_TASK_STATUSES)
|
2026-03-16 06:59:46 +02:00
|
|
|
VALID_EXECUTION_MODES = VALID_COMPLETION_MODES
|
2026-03-15 18:17:57 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.patch("/api/tasks/{task_id}")
|
|
|
|
|
def patch_task(task_id: str, body: TaskPatch):
|
2026-03-15 20:02:01 +02:00
|
|
|
if body.status is not None and body.status not in VALID_STATUSES:
|
2026-03-15 18:17:57 +02:00
|
|
|
raise HTTPException(400, f"Invalid status '{body.status}'. Must be one of: {', '.join(VALID_STATUSES)}")
|
2026-03-15 20:02:01 +02:00
|
|
|
if body.execution_mode is not None and body.execution_mode not in VALID_EXECUTION_MODES:
|
|
|
|
|
raise HTTPException(400, f"Invalid execution_mode '{body.execution_mode}'. Must be one of: {', '.join(VALID_EXECUTION_MODES)}")
|
2026-03-16 07:13:32 +02:00
|
|
|
if body.priority is not None and not (1 <= body.priority <= 10):
|
|
|
|
|
raise HTTPException(400, "priority must be between 1 and 10")
|
|
|
|
|
if body.route_type is not None and body.route_type and body.route_type not in VALID_ROUTE_TYPES:
|
|
|
|
|
raise HTTPException(400, f"Invalid route_type '{body.route_type}'. Must be one of: {', '.join(sorted(VALID_ROUTE_TYPES))} or empty string to clear")
|
|
|
|
|
if body.title is not None and not body.title.strip():
|
|
|
|
|
raise HTTPException(400, "title must not be empty")
|
|
|
|
|
all_none = all(v is None for v in [body.status, body.execution_mode, body.priority, body.route_type, body.title, body.brief_text])
|
|
|
|
|
if all_none:
|
|
|
|
|
raise HTTPException(400, "Nothing to update.")
|
2026-03-15 18:17:57 +02:00
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
2026-03-15 20:02:01 +02:00
|
|
|
fields = {}
|
|
|
|
|
if body.status is not None:
|
|
|
|
|
fields["status"] = body.status
|
|
|
|
|
if body.execution_mode is not None:
|
|
|
|
|
fields["execution_mode"] = body.execution_mode
|
2026-03-16 07:13:32 +02:00
|
|
|
if body.priority is not None:
|
|
|
|
|
fields["priority"] = body.priority
|
|
|
|
|
if body.title is not None:
|
|
|
|
|
fields["title"] = body.title.strip()
|
|
|
|
|
if body.route_type is not None or body.brief_text is not None:
|
|
|
|
|
current_brief = t.get("brief") or {}
|
|
|
|
|
if isinstance(current_brief, str):
|
|
|
|
|
current_brief = {"text": current_brief}
|
|
|
|
|
if body.route_type is not None:
|
|
|
|
|
if body.route_type:
|
|
|
|
|
current_brief = {**current_brief, "route_type": body.route_type}
|
|
|
|
|
else:
|
|
|
|
|
current_brief = {k: v for k, v in current_brief.items() if k != "route_type"}
|
|
|
|
|
if body.brief_text is not None:
|
|
|
|
|
current_brief = {**current_brief, "text": body.brief_text}
|
|
|
|
|
fields["brief"] = current_brief if current_brief else None
|
2026-03-15 20:02:01 +02:00
|
|
|
models.update_task(conn, task_id, **fields)
|
2026-03-15 18:17:57 +02:00
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
conn.close()
|
|
|
|
|
return t
|
|
|
|
|
|
|
|
|
|
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
@app.get("/api/tasks/{task_id}/pipeline")
|
|
|
|
|
def get_task_pipeline(task_id: str):
|
|
|
|
|
"""Get agent_logs for a task (pipeline steps)."""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
|
|
|
|
rows = conn.execute(
|
|
|
|
|
"""SELECT id, agent_role, action, output_summary, success,
|
|
|
|
|
duration_seconds, tokens_used, model, cost_usd, created_at
|
|
|
|
|
FROM agent_logs WHERE task_id = ? ORDER BY created_at""",
|
|
|
|
|
(task_id,),
|
|
|
|
|
).fetchall()
|
|
|
|
|
steps = [dict(r) for r in rows]
|
|
|
|
|
conn.close()
|
|
|
|
|
return steps
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.get("/api/tasks/{task_id}/full")
|
|
|
|
|
def get_task_full(task_id: str):
|
|
|
|
|
"""Task + pipeline steps + related decisions."""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
|
|
|
|
rows = conn.execute(
|
|
|
|
|
"""SELECT id, agent_role, action, output_summary, success,
|
|
|
|
|
duration_seconds, tokens_used, model, cost_usd, created_at
|
|
|
|
|
FROM agent_logs WHERE task_id = ? ORDER BY created_at""",
|
|
|
|
|
(task_id,),
|
|
|
|
|
).fetchall()
|
|
|
|
|
steps = [dict(r) for r in rows]
|
|
|
|
|
decisions = models.get_decisions(conn, t["project_id"])
|
|
|
|
|
# Filter to decisions linked to this task
|
|
|
|
|
task_decisions = [d for d in decisions if d.get("task_id") == task_id]
|
2026-03-16 08:21:13 +02:00
|
|
|
p = models.get_project(conn, t["project_id"])
|
|
|
|
|
project_deploy_command = p.get("deploy_command") if p else None
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
conn.close()
|
2026-03-16 08:21:13 +02:00
|
|
|
return {**t, "pipeline_steps": steps, "related_decisions": task_decisions, "project_deploy_command": project_deploy_command}
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class TaskApprove(BaseModel):
|
|
|
|
|
decision_title: str | None = None
|
|
|
|
|
decision_description: str | None = None
|
|
|
|
|
decision_type: str = "decision"
|
Add follow-up task generation on approve
When approving a task, PM agent analyzes pipeline output and creates
follow-up tasks automatically (e.g. security audit → 8 fix tasks).
core/followup.py:
generate_followups() — collects pipeline output, runs followup agent,
parses JSON task list, creates tasks with parent_task_id linkage.
Handles: bare arrays, {tasks:[...]} wrappers, invalid JSON, empty.
agents/prompts/followup.md — PM prompt for analyzing results and
creating actionable follow-up tasks with priority from severity.
CLI: kin approve <task_id> [--followup] [--decision "text"]
API: POST /api/tasks/{id}/approve {create_followups: true}
Returns {status, decision, followup_tasks: [...]}
Frontend (TaskDetail approve modal):
- Checkbox "Create follow-up tasks" (default ON)
- Loading state during generation
- Results view: list of created tasks with links to /task/:id
ProjectView: tasks show "from VDOL-001" for follow-ups.
13 new tests (followup), 125 total, all passing.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 15:02:58 +02:00
|
|
|
create_followups: bool = False
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/tasks/{task_id}/approve")
|
|
|
|
|
def approve_task(task_id: str, body: TaskApprove | None = None):
|
Add follow-up task generation on approve
When approving a task, PM agent analyzes pipeline output and creates
follow-up tasks automatically (e.g. security audit → 8 fix tasks).
core/followup.py:
generate_followups() — collects pipeline output, runs followup agent,
parses JSON task list, creates tasks with parent_task_id linkage.
Handles: bare arrays, {tasks:[...]} wrappers, invalid JSON, empty.
agents/prompts/followup.md — PM prompt for analyzing results and
creating actionable follow-up tasks with priority from severity.
CLI: kin approve <task_id> [--followup] [--decision "text"]
API: POST /api/tasks/{id}/approve {create_followups: true}
Returns {status, decision, followup_tasks: [...]}
Frontend (TaskDetail approve modal):
- Checkbox "Create follow-up tasks" (default ON)
- Loading state during generation
- Results view: list of created tasks with links to /task/:id
ProjectView: tasks show "from VDOL-001" for follow-ups.
13 new tests (followup), 125 total, all passing.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 15:02:58 +02:00
|
|
|
"""Approve a task: set status=done, optionally add decision and create follow-ups."""
|
|
|
|
|
from core.followup import generate_followups
|
|
|
|
|
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
|
|
|
|
models.update_task(conn, task_id, status="done")
|
2026-03-15 23:22:49 +02:00
|
|
|
try:
|
|
|
|
|
from core.hooks import run_hooks as _run_hooks
|
|
|
|
|
task_modules = models.get_modules(conn, t["project_id"])
|
|
|
|
|
_run_hooks(conn, t["project_id"], task_id,
|
|
|
|
|
event="task_done", task_modules=task_modules)
|
|
|
|
|
except Exception:
|
|
|
|
|
pass
|
2026-03-16 09:43:26 +02:00
|
|
|
|
|
|
|
|
# Advance phase state machine if this task belongs to an active phase
|
|
|
|
|
phase_result = None
|
|
|
|
|
phase_row = conn.execute(
|
|
|
|
|
"SELECT * FROM project_phases WHERE task_id = ?", (task_id,)
|
|
|
|
|
).fetchone()
|
|
|
|
|
if phase_row:
|
|
|
|
|
phase = dict(phase_row)
|
|
|
|
|
if phase.get("status") == "active":
|
|
|
|
|
from core.phases import approve_phase as _approve_phase
|
|
|
|
|
try:
|
|
|
|
|
phase_result = _approve_phase(conn, phase["id"])
|
|
|
|
|
except ValueError:
|
|
|
|
|
pass
|
|
|
|
|
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
decision = None
|
|
|
|
|
if body and body.decision_title:
|
|
|
|
|
decision = models.add_decision(
|
|
|
|
|
conn, t["project_id"], body.decision_type,
|
|
|
|
|
body.decision_title, body.decision_description or body.decision_title,
|
|
|
|
|
task_id=task_id,
|
|
|
|
|
)
|
Add follow-up task generation on approve
When approving a task, PM agent analyzes pipeline output and creates
follow-up tasks automatically (e.g. security audit → 8 fix tasks).
core/followup.py:
generate_followups() — collects pipeline output, runs followup agent,
parses JSON task list, creates tasks with parent_task_id linkage.
Handles: bare arrays, {tasks:[...]} wrappers, invalid JSON, empty.
agents/prompts/followup.md — PM prompt for analyzing results and
creating actionable follow-up tasks with priority from severity.
CLI: kin approve <task_id> [--followup] [--decision "text"]
API: POST /api/tasks/{id}/approve {create_followups: true}
Returns {status, decision, followup_tasks: [...]}
Frontend (TaskDetail approve modal):
- Checkbox "Create follow-up tasks" (default ON)
- Loading state during generation
- Results view: list of created tasks with links to /task/:id
ProjectView: tasks show "from VDOL-001" for follow-ups.
13 new tests (followup), 125 total, all passing.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 15:02:58 +02:00
|
|
|
followup_tasks = []
|
2026-03-15 15:16:48 +02:00
|
|
|
pending_actions = []
|
Add follow-up task generation on approve
When approving a task, PM agent analyzes pipeline output and creates
follow-up tasks automatically (e.g. security audit → 8 fix tasks).
core/followup.py:
generate_followups() — collects pipeline output, runs followup agent,
parses JSON task list, creates tasks with parent_task_id linkage.
Handles: bare arrays, {tasks:[...]} wrappers, invalid JSON, empty.
agents/prompts/followup.md — PM prompt for analyzing results and
creating actionable follow-up tasks with priority from severity.
CLI: kin approve <task_id> [--followup] [--decision "text"]
API: POST /api/tasks/{id}/approve {create_followups: true}
Returns {status, decision, followup_tasks: [...]}
Frontend (TaskDetail approve modal):
- Checkbox "Create follow-up tasks" (default ON)
- Loading state during generation
- Results view: list of created tasks with links to /task/:id
ProjectView: tasks show "from VDOL-001" for follow-ups.
13 new tests (followup), 125 total, all passing.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 15:02:58 +02:00
|
|
|
if body and body.create_followups:
|
2026-03-15 15:16:48 +02:00
|
|
|
result = generate_followups(conn, task_id)
|
|
|
|
|
followup_tasks = result["created"]
|
|
|
|
|
pending_actions = result["pending_actions"]
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
conn.close()
|
Add follow-up task generation on approve
When approving a task, PM agent analyzes pipeline output and creates
follow-up tasks automatically (e.g. security audit → 8 fix tasks).
core/followup.py:
generate_followups() — collects pipeline output, runs followup agent,
parses JSON task list, creates tasks with parent_task_id linkage.
Handles: bare arrays, {tasks:[...]} wrappers, invalid JSON, empty.
agents/prompts/followup.md — PM prompt for analyzing results and
creating actionable follow-up tasks with priority from severity.
CLI: kin approve <task_id> [--followup] [--decision "text"]
API: POST /api/tasks/{id}/approve {create_followups: true}
Returns {status, decision, followup_tasks: [...]}
Frontend (TaskDetail approve modal):
- Checkbox "Create follow-up tasks" (default ON)
- Loading state during generation
- Results view: list of created tasks with links to /task/:id
ProjectView: tasks show "from VDOL-001" for follow-ups.
13 new tests (followup), 125 total, all passing.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 15:02:58 +02:00
|
|
|
return {
|
|
|
|
|
"status": "done",
|
|
|
|
|
"decision": decision,
|
|
|
|
|
"followup_tasks": followup_tasks,
|
2026-03-15 15:16:48 +02:00
|
|
|
"needs_decision": len(pending_actions) > 0,
|
|
|
|
|
"pending_actions": pending_actions,
|
2026-03-16 09:43:26 +02:00
|
|
|
"phase": phase_result,
|
Add follow-up task generation on approve
When approving a task, PM agent analyzes pipeline output and creates
follow-up tasks automatically (e.g. security audit → 8 fix tasks).
core/followup.py:
generate_followups() — collects pipeline output, runs followup agent,
parses JSON task list, creates tasks with parent_task_id linkage.
Handles: bare arrays, {tasks:[...]} wrappers, invalid JSON, empty.
agents/prompts/followup.md — PM prompt for analyzing results and
creating actionable follow-up tasks with priority from severity.
CLI: kin approve <task_id> [--followup] [--decision "text"]
API: POST /api/tasks/{id}/approve {create_followups: true}
Returns {status, decision, followup_tasks: [...]}
Frontend (TaskDetail approve modal):
- Checkbox "Create follow-up tasks" (default ON)
- Loading state during generation
- Results view: list of created tasks with links to /task/:id
ProjectView: tasks show "from VDOL-001" for follow-ups.
13 new tests (followup), 125 total, all passing.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 15:02:58 +02:00
|
|
|
}
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
|
|
|
|
|
|
2026-03-15 15:16:48 +02:00
|
|
|
class ResolveAction(BaseModel):
|
|
|
|
|
action: dict
|
|
|
|
|
choice: str # "rerun" | "manual_task" | "skip"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/tasks/{task_id}/resolve")
|
|
|
|
|
def resolve_action(task_id: str, body: ResolveAction):
|
|
|
|
|
"""Resolve a pending permission action from follow-up generation."""
|
|
|
|
|
from core.followup import resolve_pending_action
|
|
|
|
|
|
|
|
|
|
if body.choice not in ("rerun", "manual_task", "skip"):
|
|
|
|
|
raise HTTPException(400, f"Invalid choice: {body.choice}")
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
|
|
|
|
result = resolve_pending_action(conn, task_id, body.action, body.choice)
|
|
|
|
|
conn.close()
|
|
|
|
|
return {"choice": body.choice, "result": result}
|
|
|
|
|
|
|
|
|
|
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
class TaskReject(BaseModel):
|
|
|
|
|
reason: str
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/tasks/{task_id}/reject")
|
|
|
|
|
def reject_task(task_id: str, body: TaskReject):
|
|
|
|
|
"""Reject a task: set status=pending with reason in review field."""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
|
|
|
|
models.update_task(conn, task_id, status="pending", review={"rejected": body.reason})
|
|
|
|
|
conn.close()
|
|
|
|
|
return {"status": "pending", "reason": body.reason}
|
|
|
|
|
|
|
|
|
|
|
2026-03-16 07:13:32 +02:00
|
|
|
class TaskRevise(BaseModel):
|
|
|
|
|
comment: str
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/tasks/{task_id}/revise")
|
|
|
|
|
def revise_task(task_id: str, body: TaskRevise):
|
|
|
|
|
"""Revise a task: return to in_progress with director's comment for the agent."""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
|
|
|
|
models.update_task(conn, task_id, status="in_progress", revise_comment=body.comment)
|
|
|
|
|
conn.close()
|
|
|
|
|
return {"status": "in_progress", "comment": body.comment}
|
|
|
|
|
|
|
|
|
|
|
2026-03-15 15:29:05 +02:00
|
|
|
@app.get("/api/tasks/{task_id}/running")
|
|
|
|
|
def is_task_running(task_id: str):
|
|
|
|
|
"""Check if task has an active (running) pipeline."""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
|
|
|
|
row = conn.execute(
|
|
|
|
|
"SELECT id, status FROM pipelines WHERE task_id = ? ORDER BY created_at DESC LIMIT 1",
|
|
|
|
|
(task_id,),
|
|
|
|
|
).fetchone()
|
|
|
|
|
conn.close()
|
|
|
|
|
if row and row["status"] == "running":
|
|
|
|
|
return {"running": True, "pipeline_id": row["id"]}
|
|
|
|
|
return {"running": False}
|
|
|
|
|
|
|
|
|
|
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
@app.post("/api/tasks/{task_id}/run")
|
2026-03-15 23:22:49 +02:00
|
|
|
def run_task(task_id: str):
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
"""Launch pipeline for a task in background. Returns 202."""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
t = models.get_task(conn, task_id)
|
|
|
|
|
if not t:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Task '{task_id}' not found")
|
2026-03-15 15:29:05 +02:00
|
|
|
# Set task to in_progress immediately so UI updates
|
|
|
|
|
models.update_task(conn, task_id, status="in_progress")
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
conn.close()
|
|
|
|
|
# Launch kin run in background subprocess
|
|
|
|
|
kin_root = Path(__file__).parent.parent
|
2026-03-15 17:35:08 +02:00
|
|
|
cmd = [sys.executable, "-m", "cli.main", "--db", str(DB_PATH),
|
|
|
|
|
"run", task_id]
|
2026-03-15 23:22:49 +02:00
|
|
|
cmd.append("--allow-write") # always required: subprocess runs non-interactively (stdin=DEVNULL)
|
2026-03-15 17:35:08 +02:00
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
env = os.environ.copy()
|
|
|
|
|
env["KIN_NONINTERACTIVE"] = "1"
|
|
|
|
|
|
2026-03-15 15:29:05 +02:00
|
|
|
try:
|
|
|
|
|
proc = subprocess.Popen(
|
2026-03-15 17:35:08 +02:00
|
|
|
cmd,
|
2026-03-15 15:29:05 +02:00
|
|
|
cwd=str(kin_root),
|
|
|
|
|
stdout=subprocess.DEVNULL,
|
2026-03-16 06:59:46 +02:00
|
|
|
stderr=subprocess.DEVNULL,
|
2026-03-15 17:35:08 +02:00
|
|
|
stdin=subprocess.DEVNULL,
|
|
|
|
|
env=env,
|
2026-03-15 15:29:05 +02:00
|
|
|
)
|
|
|
|
|
import logging
|
|
|
|
|
logging.getLogger("kin").info(f"Pipeline started for {task_id}, pid={proc.pid}")
|
|
|
|
|
except Exception as e:
|
|
|
|
|
raise HTTPException(500, f"Failed to start pipeline: {e}")
|
Add task detail view, pipeline visualization, approve/reject workflow
API (web/api.py) — 5 new endpoints:
GET /api/tasks/{id}/pipeline — agent_logs as pipeline steps
GET /api/tasks/{id}/full — task + steps + related decisions
POST /api/tasks/{id}/approve — mark done, optionally add decision
POST /api/tasks/{id}/reject — return to pending with reason
POST /api/tasks/{id}/run — launch pipeline in background (202)
Frontend:
TaskDetail (/task/:id) — full task page with:
- Pipeline graph: role cards with icons, arrows, status colors
- Click step → expand output (pre-formatted, JSON detected)
- Action bar: Approve (with optional decision), Reject, Run Pipeline
- Polling for live pipeline updates
Dashboard: review_tasks badge ("awaiting review" in yellow)
ProjectView: task rows are now clickable links to /task/:id
Runner: output_summary no longer truncated (full output for GUI).
Models: get_project_summary includes review_tasks count.
13 new API tests, 105 total, all passing. Frontend builds clean.
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 14:32:29 +02:00
|
|
|
return JSONResponse({"status": "started", "task_id": task_id}, status_code=202)
|
|
|
|
|
|
|
|
|
|
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Decisions
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
@app.get("/api/decisions")
|
|
|
|
|
def list_decisions(
|
|
|
|
|
project: str = Query(...),
|
|
|
|
|
category: str | None = None,
|
|
|
|
|
tag: list[str] | None = Query(None),
|
|
|
|
|
type: list[str] | None = Query(None),
|
|
|
|
|
):
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
decisions = models.get_decisions(
|
|
|
|
|
conn, project, category=category, tags=tag, types=type,
|
|
|
|
|
)
|
|
|
|
|
conn.close()
|
|
|
|
|
return decisions
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class DecisionCreate(BaseModel):
|
|
|
|
|
project_id: str
|
|
|
|
|
type: str
|
|
|
|
|
title: str
|
|
|
|
|
description: str
|
|
|
|
|
category: str | None = None
|
|
|
|
|
tags: list[str] | None = None
|
|
|
|
|
task_id: str | None = None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/decisions")
|
|
|
|
|
def create_decision(body: DecisionCreate):
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, body.project_id)
|
|
|
|
|
if not p:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Project '{body.project_id}' not found")
|
|
|
|
|
d = models.add_decision(
|
|
|
|
|
conn, body.project_id, body.type, body.title, body.description,
|
|
|
|
|
category=body.category, tags=body.tags, task_id=body.task_id,
|
|
|
|
|
)
|
|
|
|
|
conn.close()
|
|
|
|
|
return d
|
|
|
|
|
|
|
|
|
|
|
2026-03-15 23:22:49 +02:00
|
|
|
@app.delete("/api/projects/{project_id}/decisions/{decision_id}")
|
|
|
|
|
def delete_decision(project_id: str, decision_id: int):
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
decision = models.get_decision(conn, decision_id)
|
|
|
|
|
if not decision or decision["project_id"] != project_id:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Decision #{decision_id} not found")
|
|
|
|
|
models.delete_decision(conn, decision_id)
|
|
|
|
|
conn.close()
|
|
|
|
|
return {"deleted": decision_id}
|
|
|
|
|
|
|
|
|
|
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Cost
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
@app.get("/api/cost")
|
|
|
|
|
def cost_summary(days: int = 7):
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
costs = models.get_cost_summary(conn, days=days)
|
|
|
|
|
conn.close()
|
|
|
|
|
return costs
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Support
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
@app.get("/api/support/tickets")
|
|
|
|
|
def list_tickets(project: str | None = None, status: str | None = None):
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
tickets = models.list_tickets(conn, project_id=project, status=status)
|
|
|
|
|
conn.close()
|
|
|
|
|
return tickets
|
|
|
|
|
|
|
|
|
|
|
2026-03-15 17:44:16 +02:00
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Audit
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
@app.post("/api/projects/{project_id}/audit")
|
|
|
|
|
def audit_project(project_id: str):
|
|
|
|
|
"""Run backlog audit — check which pending tasks are already done."""
|
|
|
|
|
from agents.runner import run_audit
|
|
|
|
|
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, project_id)
|
|
|
|
|
if not p:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Project '{project_id}' not found")
|
2026-03-15 18:00:39 +02:00
|
|
|
result = run_audit(conn, project_id, noninteractive=True, auto_apply=False)
|
2026-03-15 17:44:16 +02:00
|
|
|
conn.close()
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class AuditApply(BaseModel):
|
|
|
|
|
task_ids: list[str]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/projects/{project_id}/audit/apply")
|
|
|
|
|
def audit_apply(project_id: str, body: AuditApply):
|
|
|
|
|
"""Mark tasks as done after audit confirmation."""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
p = models.get_project(conn, project_id)
|
|
|
|
|
if not p:
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(404, f"Project '{project_id}' not found")
|
|
|
|
|
updated = []
|
|
|
|
|
for tid in body.task_ids:
|
|
|
|
|
t = models.get_task(conn, tid)
|
|
|
|
|
if t and t["project_id"] == project_id:
|
|
|
|
|
models.update_task(conn, tid, status="done")
|
|
|
|
|
updated.append(tid)
|
|
|
|
|
conn.close()
|
|
|
|
|
return {"updated": updated, "count": len(updated)}
|
|
|
|
|
|
|
|
|
|
|
Add web GUI: FastAPI API + Vue 3 frontend with dark theme
API (web/api.py):
GET /api/projects, /api/projects/{id}, /api/tasks/{id}
GET /api/decisions?project=X, /api/cost?days=7, /api/support/tickets
POST /api/projects, /api/tasks, /api/decisions, /api/bootstrap
CORS for localhost:5173, all queries via models.py
Frontend (web/frontend/):
Vue 3 + TypeScript + Vite + Tailwind CSS v3
Dashboard: project cards with task counters, cost, status badges
ProjectView: tabs for Tasks/Decisions/Modules with filters
Modals: Add Project, Add Task, Add Decision, Bootstrap
Dark theme, monospace font, minimal clean design
Startup:
API: cd web && uvicorn api:app --reload --port 8420
Web: cd web/frontend && npm install && npm run dev
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-15 13:50:15 +02:00
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Bootstrap
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
class BootstrapRequest(BaseModel):
|
|
|
|
|
path: str
|
|
|
|
|
id: str
|
|
|
|
|
name: str
|
|
|
|
|
vault_path: str | None = None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.post("/api/bootstrap")
|
|
|
|
|
def bootstrap(body: BootstrapRequest):
|
|
|
|
|
project_path = Path(body.path).expanduser().resolve()
|
|
|
|
|
if not project_path.is_dir():
|
|
|
|
|
raise HTTPException(400, f"Path '{body.path}' is not a directory")
|
|
|
|
|
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
if models.get_project(conn, body.id):
|
|
|
|
|
conn.close()
|
|
|
|
|
raise HTTPException(409, f"Project '{body.id}' already exists")
|
|
|
|
|
|
|
|
|
|
tech_stack = detect_tech_stack(project_path)
|
|
|
|
|
modules = detect_modules(project_path)
|
|
|
|
|
decisions = extract_decisions_from_claude_md(project_path, body.id, body.name)
|
|
|
|
|
|
|
|
|
|
obsidian = None
|
|
|
|
|
vault_root = find_vault_root(Path(body.vault_path) if body.vault_path else None)
|
|
|
|
|
if vault_root:
|
|
|
|
|
dir_name = project_path.name
|
|
|
|
|
obs = scan_obsidian(vault_root, body.id, body.name, dir_name)
|
|
|
|
|
if obs["tasks"] or obs["decisions"]:
|
|
|
|
|
obsidian = obs
|
|
|
|
|
|
|
|
|
|
save_to_db(conn, body.id, body.name, str(project_path),
|
|
|
|
|
tech_stack, modules, decisions, obsidian)
|
|
|
|
|
p = models.get_project(conn, body.id)
|
|
|
|
|
conn.close()
|
|
|
|
|
return {
|
|
|
|
|
"project": p,
|
|
|
|
|
"modules_count": len(modules),
|
|
|
|
|
"decisions_count": len(decisions) + len((obsidian or {}).get("decisions", [])),
|
|
|
|
|
"tasks_count": len((obsidian or {}).get("tasks", [])),
|
|
|
|
|
}
|
2026-03-15 17:11:38 +02:00
|
|
|
|
|
|
|
|
|
2026-03-16 09:13:34 +02:00
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# Notifications (escalations from blocked agents)
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
@app.get("/api/notifications")
|
|
|
|
|
def get_notifications(project_id: str | None = None):
|
|
|
|
|
"""Return tasks with status='blocked' as escalation notifications.
|
|
|
|
|
|
|
|
|
|
Each item includes task details, the agent role that blocked it,
|
2026-03-16 09:43:26 +02:00
|
|
|
the reason, the pipeline step, and whether a Telegram alert was sent.
|
|
|
|
|
Intended for GUI polling (5s interval).
|
2026-03-16 09:13:34 +02:00
|
|
|
"""
|
|
|
|
|
conn = get_conn()
|
|
|
|
|
query = "SELECT * FROM tasks WHERE status = 'blocked'"
|
|
|
|
|
params: list = []
|
|
|
|
|
if project_id:
|
|
|
|
|
query += " AND project_id = ?"
|
|
|
|
|
params.append(project_id)
|
|
|
|
|
query += " ORDER BY blocked_at DESC, updated_at DESC"
|
|
|
|
|
rows = conn.execute(query, params).fetchall()
|
|
|
|
|
conn.close()
|
|
|
|
|
|
|
|
|
|
notifications = []
|
|
|
|
|
for row in rows:
|
|
|
|
|
t = dict(row)
|
|
|
|
|
notifications.append({
|
|
|
|
|
"task_id": t["id"],
|
|
|
|
|
"project_id": t["project_id"],
|
|
|
|
|
"title": t.get("title"),
|
|
|
|
|
"agent_role": t.get("blocked_agent_role"),
|
|
|
|
|
"reason": t.get("blocked_reason"),
|
|
|
|
|
"pipeline_step": t.get("blocked_pipeline_step"),
|
|
|
|
|
"blocked_at": t.get("blocked_at") or t.get("updated_at"),
|
2026-03-16 09:43:26 +02:00
|
|
|
"telegram_sent": bool(t.get("telegram_sent")),
|
2026-03-16 09:13:34 +02:00
|
|
|
})
|
|
|
|
|
return notifications
|
|
|
|
|
|
|
|
|
|
|
2026-03-15 17:11:38 +02:00
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
# SPA static files (AFTER all /api/ routes)
|
|
|
|
|
# ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
DIST = Path(__file__).parent / "frontend" / "dist"
|
|
|
|
|
|
|
|
|
|
app.mount("/assets", StaticFiles(directory=str(DIST / "assets")), name="assets")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@app.get("/{path:path}")
|
|
|
|
|
async def serve_spa(path: str):
|
|
|
|
|
file = DIST / path
|
|
|
|
|
if file.exists() and file.is_file():
|
|
|
|
|
return FileResponse(file)
|
|
|
|
|
return FileResponse(DIST / "index.html")
|