2026-02-04 02:28:51 +05:30
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
|
|
from datetime import datetime
|
|
|
|
|
|
2026-02-04 14:58:14 +05:30
|
|
|
from fastapi import APIRouter, Depends, HTTPException, status
|
2026-02-04 02:28:51 +05:30
|
|
|
from sqlmodel import Session, select
|
|
|
|
|
|
2026-02-04 14:58:14 +05:30
|
|
|
from app.api.deps import (
|
|
|
|
|
ActorContext,
|
|
|
|
|
get_board_or_404,
|
|
|
|
|
get_task_or_404,
|
|
|
|
|
require_admin_auth,
|
|
|
|
|
require_admin_or_agent,
|
|
|
|
|
)
|
2026-02-04 03:57:19 +05:30
|
|
|
from app.core.auth import AuthContext
|
2026-02-04 02:28:51 +05:30
|
|
|
from app.db.session import get_session
|
|
|
|
|
from app.models.boards import Board
|
|
|
|
|
from app.models.tasks import Task
|
|
|
|
|
from app.schemas.tasks import TaskCreate, TaskRead, TaskUpdate
|
2026-02-04 03:57:19 +05:30
|
|
|
from app.services.activity_log import record_activity
|
2026-02-04 02:28:51 +05:30
|
|
|
|
|
|
|
|
router = APIRouter(prefix="/boards/{board_id}/tasks", tags=["tasks"])
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.get("", response_model=list[TaskRead])
|
|
|
|
|
def list_tasks(
|
2026-02-04 03:57:19 +05:30
|
|
|
board: Board = Depends(get_board_or_404),
|
2026-02-04 02:28:51 +05:30
|
|
|
session: Session = Depends(get_session),
|
2026-02-04 14:58:14 +05:30
|
|
|
actor: ActorContext = Depends(require_admin_or_agent),
|
2026-02-04 02:28:51 +05:30
|
|
|
) -> list[Task]:
|
|
|
|
|
return list(session.exec(select(Task).where(Task.board_id == board.id)))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.post("", response_model=TaskRead)
|
|
|
|
|
def create_task(
|
|
|
|
|
payload: TaskCreate,
|
2026-02-04 03:57:19 +05:30
|
|
|
board: Board = Depends(get_board_or_404),
|
2026-02-04 02:28:51 +05:30
|
|
|
session: Session = Depends(get_session),
|
2026-02-04 03:57:19 +05:30
|
|
|
auth: AuthContext = Depends(require_admin_auth),
|
2026-02-04 02:28:51 +05:30
|
|
|
) -> Task:
|
|
|
|
|
task = Task.model_validate(payload)
|
|
|
|
|
task.board_id = board.id
|
|
|
|
|
if task.created_by_user_id is None and auth.user is not None:
|
|
|
|
|
task.created_by_user_id = auth.user.id
|
|
|
|
|
session.add(task)
|
|
|
|
|
session.commit()
|
|
|
|
|
session.refresh(task)
|
|
|
|
|
|
2026-02-04 03:57:19 +05:30
|
|
|
record_activity(
|
|
|
|
|
session,
|
2026-02-04 02:28:51 +05:30
|
|
|
event_type="task.created",
|
|
|
|
|
task_id=task.id,
|
|
|
|
|
message=f"Task created: {task.title}.",
|
|
|
|
|
)
|
|
|
|
|
session.commit()
|
|
|
|
|
return task
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.patch("/{task_id}", response_model=TaskRead)
|
|
|
|
|
def update_task(
|
|
|
|
|
payload: TaskUpdate,
|
2026-02-04 03:57:19 +05:30
|
|
|
task: Task = Depends(get_task_or_404),
|
2026-02-04 02:28:51 +05:30
|
|
|
session: Session = Depends(get_session),
|
2026-02-04 14:58:14 +05:30
|
|
|
actor: ActorContext = Depends(require_admin_or_agent),
|
2026-02-04 02:28:51 +05:30
|
|
|
) -> Task:
|
|
|
|
|
previous_status = task.status
|
|
|
|
|
updates = payload.model_dump(exclude_unset=True)
|
2026-02-04 14:58:14 +05:30
|
|
|
if actor.actor_type == "agent":
|
|
|
|
|
allowed_fields = {"status"}
|
|
|
|
|
if not set(updates).issubset(allowed_fields):
|
|
|
|
|
raise HTTPException(status_code=status.HTTP_403_FORBIDDEN)
|
2026-02-04 02:28:51 +05:30
|
|
|
for key, value in updates.items():
|
|
|
|
|
setattr(task, key, value)
|
|
|
|
|
task.updated_at = datetime.utcnow()
|
|
|
|
|
|
|
|
|
|
session.add(task)
|
|
|
|
|
session.commit()
|
|
|
|
|
session.refresh(task)
|
|
|
|
|
|
|
|
|
|
if "status" in updates and task.status != previous_status:
|
|
|
|
|
event_type = "task.status_changed"
|
|
|
|
|
message = f"Task moved to {task.status}: {task.title}."
|
|
|
|
|
else:
|
|
|
|
|
event_type = "task.updated"
|
|
|
|
|
message = f"Task updated: {task.title}."
|
2026-02-04 14:58:14 +05:30
|
|
|
record_activity(
|
|
|
|
|
session,
|
|
|
|
|
event_type=event_type,
|
|
|
|
|
task_id=task.id,
|
|
|
|
|
message=message,
|
|
|
|
|
agent_id=actor.agent.id if actor.actor_type == "agent" and actor.agent else None,
|
|
|
|
|
)
|
2026-02-04 02:28:51 +05:30
|
|
|
session.commit()
|
|
|
|
|
return task
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@router.delete("/{task_id}")
|
|
|
|
|
def delete_task(
|
|
|
|
|
session: Session = Depends(get_session),
|
2026-02-04 03:57:19 +05:30
|
|
|
task: Task = Depends(get_task_or_404),
|
|
|
|
|
auth: AuthContext = Depends(require_admin_auth),
|
2026-02-04 02:28:51 +05:30
|
|
|
) -> dict[str, bool]:
|
2026-02-04 03:57:19 +05:30
|
|
|
session.delete(task)
|
|
|
|
|
session.commit()
|
2026-02-04 02:28:51 +05:30
|
|
|
return {"ok": True}
|