use axum::extract::{Path, Query, State};
use axum::http::StatusCode;
use axum::response::IntoResponse;
use axum::{Extension, Json};
use serde::Deserialize;
use systemprompt_identifiers::{ContextId, TaskId, UserId};
use systemprompt_models::api::ApiError;
use systemprompt_agent::models::a2a::TaskState;
use systemprompt_agent::repository::task::TaskRepository;
use systemprompt_models::RequestContext;
use systemprompt_runtime::AppContext;
#[derive(Debug, Deserialize)]
pub struct TaskFilterParams {
status: Option<String>,
limit: Option<u32>,
}
pub async fn list_tasks_by_context(
Extension(_req_ctx): Extension<RequestContext>,
State(app_context): State<AppContext>,
Path(context_id): Path<String>,
) -> Result<impl IntoResponse, ApiError> {
tracing::debug!(context_id = %context_id, "Listing tasks");
let task_repo = TaskRepository::new(app_context.db_pool())
.map_err(|e| ApiError::internal_error(format!("Database error: {e}")))?;
let context_id_typed = ContextId::new(&context_id);
let tasks = task_repo
.list_tasks_by_context(&context_id_typed)
.await
.map_err(|e| {
tracing::error!(error = %e, "Failed to list tasks");
ApiError::internal_error("Failed to retrieve tasks")
})?;
tracing::debug!(context_id = %context_id, count = %tasks.len(), "Tasks listed");
Ok((StatusCode::OK, Json(tasks)))
}
pub async fn get_task(
Extension(_req_ctx): Extension<RequestContext>,
State(app_context): State<AppContext>,
Path(task_id): Path<String>,
) -> Result<impl IntoResponse, ApiError> {
tracing::debug!(task_id = %task_id, "Retrieving task");
let task_repo = TaskRepository::new(app_context.db_pool())
.map_err(|e| ApiError::internal_error(format!("Database error: {e}")))?;
let task_id_typed = TaskId::new(&task_id);
match task_repo.get_task(&task_id_typed).await {
Ok(Some(task)) => {
tracing::debug!("Task retrieved successfully");
Ok((StatusCode::OK, Json(task)).into_response())
},
Ok(None) => {
tracing::debug!("Task not found");
Err(ApiError::not_found(format!("Task '{}' not found", task_id)))
},
Err(e) => {
tracing::error!(error = %e, "Failed to retrieve task");
Err(ApiError::internal_error("Failed to retrieve task"))
},
}
}
pub async fn list_tasks_by_user(
Extension(req_ctx): Extension<RequestContext>,
State(app_context): State<AppContext>,
Query(params): Query<TaskFilterParams>,
) -> Result<impl IntoResponse, ApiError> {
let user_id = req_ctx.auth.user_id.as_str();
tracing::debug!(user_id = %user_id, "Listing tasks");
let task_repo = TaskRepository::new(app_context.db_pool())
.map_err(|e| ApiError::internal_error(format!("Database error: {e}")))?;
let task_state = params.status.as_ref().and_then(|s| match s.as_str() {
"submitted" => Some(TaskState::Submitted),
"working" => Some(TaskState::Working),
"input-required" => Some(TaskState::InputRequired),
"completed" => Some(TaskState::Completed),
"canceled" | "cancelled" => Some(TaskState::Canceled),
"failed" => Some(TaskState::Failed),
"rejected" => Some(TaskState::Rejected),
"auth-required" => Some(TaskState::AuthRequired),
_ => None,
});
let user_id_typed = UserId::new(user_id);
let mut tasks = task_repo
.get_tasks_by_user_id(&user_id_typed, params.limit.map(|l| l as i32), None)
.await
.map_err(|e| {
tracing::error!(error = %e, "Failed to list tasks");
ApiError::internal_error("Failed to retrieve tasks")
})?;
if let Some(state) = task_state {
tasks.retain(|t| t.status.state == state);
}
tracing::debug!(user_id = %user_id, count = %tasks.len(), "Tasks listed");
Ok((StatusCode::OK, Json(tasks)))
}
pub async fn get_messages_by_task(
Extension(_req_ctx): Extension<RequestContext>,
State(app_context): State<AppContext>,
Path(task_id): Path<String>,
) -> Result<impl IntoResponse, ApiError> {
tracing::debug!(task_id = %task_id, "Retrieving messages");
let task_repo = TaskRepository::new(app_context.db_pool())
.map_err(|e| ApiError::internal_error(format!("Database error: {e}")))?;
let task_id_typed = TaskId::new(&task_id);
let messages = task_repo
.get_messages_by_task(&task_id_typed)
.await
.map_err(|e| {
tracing::error!(error = %e, "Failed to retrieve messages");
ApiError::internal_error("Failed to retrieve messages")
})?;
tracing::debug!(task_id = %task_id, count = %messages.len(), "Messages retrieved");
Ok((StatusCode::OK, Json(messages)))
}
pub async fn delete_task(
Extension(_req_ctx): Extension<RequestContext>,
State(app_context): State<AppContext>,
Path(task_id): Path<String>,
) -> Result<impl IntoResponse, ApiError> {
tracing::debug!(task_id = %task_id, "Deleting task");
let task_repo = TaskRepository::new(app_context.db_pool())
.map_err(|e| ApiError::internal_error(format!("Database error: {e}")))?;
let task_id_typed = TaskId::new(&task_id);
task_repo.delete_task(&task_id_typed).await.map_err(|e| {
tracing::error!(error = %e, "Failed to delete task");
ApiError::internal_error("Failed to delete task")
})?;
tracing::debug!(task_id = %task_id, "Task deleted");
Ok(StatusCode::NO_CONTENT)
}