Skip to main content

routa_core/rpc/methods/
tasks.rs

1//! RPC methods for task management.
2//!
3//! Methods:
4//! - `tasks.list`         — list tasks with optional filters
5//! - `tasks.get`          — get a single task by id
6//! - `tasks.create`       — create a new task
7//! - `tasks.delete`       — delete a task
8//! - `tasks.updateStatus` — update a task's status
9//! - `tasks.findReady`    — find tasks ready for execution
10//! - `tasks.listArtifacts` — list artifacts attached to a task
11//! - `tasks.provideArtifact` — attach an artifact to a task
12
13use chrono::Utc;
14use serde::{Deserialize, Serialize};
15use std::collections::{BTreeMap, BTreeSet};
16
17use crate::models::artifact::{Artifact, ArtifactStatus, ArtifactType};
18use crate::models::kanban::KanbanBoard;
19use crate::models::task::{
20    build_task_invest_validation, build_task_story_readiness, Task, TaskLaneSessionStatus,
21    TaskStatus,
22};
23use crate::rpc::error::RpcError;
24use crate::state::AppState;
25
26const KANBAN_HAPPY_PATH_COLUMN_ORDER: [&str; 5] = ["backlog", "todo", "dev", "review", "done"];
27
28#[derive(Debug, Serialize)]
29#[serde(rename_all = "camelCase")]
30pub struct TaskArtifactSummary {
31    pub total: usize,
32    pub by_type: BTreeMap<String, usize>,
33    pub required_satisfied: bool,
34    pub missing_required: Vec<String>,
35}
36
37#[derive(Debug, Serialize)]
38#[serde(rename_all = "camelCase")]
39pub struct TaskVerificationSummary {
40    pub has_verdict: bool,
41    #[serde(skip_serializing_if = "Option::is_none")]
42    pub verdict: Option<String>,
43    pub has_report: bool,
44}
45
46#[derive(Debug, Serialize)]
47#[serde(rename_all = "camelCase")]
48pub struct TaskCompletionSummary {
49    pub has_summary: bool,
50}
51
52#[derive(Debug, Serialize)]
53#[serde(rename_all = "camelCase")]
54pub struct TaskRunSummary {
55    pub total: usize,
56    pub latest_status: String,
57}
58
59#[derive(Debug, Serialize)]
60#[serde(rename_all = "camelCase")]
61pub struct TaskEvidenceSummary {
62    pub artifact: TaskArtifactSummary,
63    pub verification: TaskVerificationSummary,
64    pub completion: TaskCompletionSummary,
65    pub runs: TaskRunSummary,
66}
67
68// ---------------------------------------------------------------------------
69// tasks.list
70// ---------------------------------------------------------------------------
71
72#[derive(Debug, Deserialize)]
73#[serde(rename_all = "camelCase")]
74pub struct ListParams {
75    #[serde(default = "default_workspace_id")]
76    pub workspace_id: String,
77    pub session_id: Option<String>,
78    pub status: Option<String>,
79    pub assigned_to: Option<String>,
80}
81
82fn default_workspace_id() -> String {
83    "default".into()
84}
85
86#[derive(Debug, Serialize)]
87pub struct ListResult {
88    pub tasks: Vec<serde_json::Value>,
89}
90
91pub async fn list(state: &AppState, params: ListParams) -> Result<ListResult, RpcError> {
92    let tasks = if let Some(session_id) = &params.session_id {
93        // Filter by session_id takes priority
94        state.task_store.list_by_session(session_id).await?
95    } else if let Some(assignee) = &params.assigned_to {
96        state.task_store.list_by_assignee(assignee).await?
97    } else if let Some(status_str) = &params.status {
98        let status = TaskStatus::from_str(status_str)
99            .ok_or_else(|| RpcError::BadRequest(format!("Invalid status: {status_str}")))?;
100        state
101            .task_store
102            .list_by_status(&params.workspace_id, &status)
103            .await?
104    } else {
105        state
106            .task_store
107            .list_by_workspace(&params.workspace_id)
108            .await?
109    };
110
111    Ok(ListResult {
112        tasks: serialize_tasks_with_evidence(state, &tasks).await?,
113    })
114}
115
116// ---------------------------------------------------------------------------
117// tasks.get
118// ---------------------------------------------------------------------------
119
120#[derive(Debug, Deserialize)]
121#[serde(rename_all = "camelCase")]
122pub struct GetParams {
123    pub id: String,
124}
125
126pub async fn get(state: &AppState, params: GetParams) -> Result<serde_json::Value, RpcError> {
127    let task = state
128        .task_store
129        .get(&params.id)
130        .await?
131        .ok_or_else(|| RpcError::NotFound(format!("Task {} not found", params.id)))?;
132    serialize_task_with_evidence(state, &task).await
133}
134
135// ---------------------------------------------------------------------------
136// tasks.create
137// ---------------------------------------------------------------------------
138
139#[derive(Debug, Deserialize)]
140#[serde(rename_all = "camelCase")]
141pub struct CreateParams {
142    pub title: String,
143    pub objective: String,
144    #[serde(default = "default_workspace_id")]
145    pub workspace_id: String,
146    pub session_id: Option<String>,
147    pub scope: Option<String>,
148    pub acceptance_criteria: Option<Vec<String>>,
149    pub verification_commands: Option<Vec<String>>,
150    pub test_cases: Option<Vec<String>>,
151    pub dependencies: Option<Vec<String>>,
152    pub parallel_group: Option<String>,
153}
154
155#[derive(Debug, Serialize)]
156pub struct CreateResult {
157    pub task: serde_json::Value,
158}
159
160pub async fn create(state: &AppState, params: CreateParams) -> Result<CreateResult, RpcError> {
161    let task = Task::new(
162        uuid::Uuid::new_v4().to_string(),
163        params.title,
164        params.objective,
165        params.workspace_id,
166        params.session_id,
167        params.scope,
168        params.acceptance_criteria,
169        params.verification_commands,
170        params.test_cases,
171        params.dependencies,
172        params.parallel_group,
173    );
174
175    state.task_store.save(&task).await?;
176    Ok(CreateResult {
177        task: serialize_task_with_evidence(state, &task).await?,
178    })
179}
180
181// ---------------------------------------------------------------------------
182// tasks.delete
183// ---------------------------------------------------------------------------
184
185#[derive(Debug, Deserialize)]
186#[serde(rename_all = "camelCase")]
187pub struct DeleteParams {
188    pub id: String,
189}
190
191#[derive(Debug, Serialize)]
192pub struct DeleteResult {
193    pub deleted: bool,
194}
195
196pub async fn delete(state: &AppState, params: DeleteParams) -> Result<DeleteResult, RpcError> {
197    state.task_store.delete(&params.id).await?;
198    Ok(DeleteResult { deleted: true })
199}
200
201// ---------------------------------------------------------------------------
202// tasks.updateStatus
203// ---------------------------------------------------------------------------
204
205#[derive(Debug, Deserialize)]
206#[serde(rename_all = "camelCase")]
207pub struct UpdateStatusParams {
208    pub id: String,
209    pub status: String,
210}
211
212#[derive(Debug, Serialize)]
213pub struct UpdateStatusResult {
214    pub updated: bool,
215}
216
217pub async fn update_status(
218    state: &AppState,
219    params: UpdateStatusParams,
220) -> Result<UpdateStatusResult, RpcError> {
221    let status = TaskStatus::from_str(&params.status)
222        .ok_or_else(|| RpcError::BadRequest(format!("Invalid status: {}", params.status)))?;
223    state.task_store.update_status(&params.id, &status).await?;
224    Ok(UpdateStatusResult { updated: true })
225}
226
227// ---------------------------------------------------------------------------
228// tasks.findReady
229// ---------------------------------------------------------------------------
230
231#[derive(Debug, Deserialize)]
232#[serde(rename_all = "camelCase")]
233pub struct FindReadyParams {
234    #[serde(default = "default_workspace_id")]
235    pub workspace_id: String,
236}
237
238pub async fn find_ready(state: &AppState, params: FindReadyParams) -> Result<ListResult, RpcError> {
239    let tasks = state
240        .task_store
241        .find_ready_tasks(&params.workspace_id)
242        .await?;
243    Ok(ListResult {
244        tasks: serialize_tasks_with_evidence(state, &tasks).await?,
245    })
246}
247
248// ---------------------------------------------------------------------------
249// tasks.listArtifacts
250// ---------------------------------------------------------------------------
251
252#[derive(Debug, Deserialize)]
253#[serde(rename_all = "camelCase")]
254pub struct ListArtifactsParams {
255    pub task_id: String,
256    #[serde(rename = "type")]
257    pub artifact_type: Option<String>,
258}
259
260#[derive(Debug, Serialize)]
261pub struct ListArtifactsResult {
262    pub artifacts: Vec<Artifact>,
263}
264
265pub async fn list_artifacts(
266    state: &AppState,
267    params: ListArtifactsParams,
268) -> Result<ListArtifactsResult, RpcError> {
269    let artifacts = if let Some(artifact_type) = params.artifact_type.as_deref() {
270        let artifact_type = parse_artifact_type(artifact_type)?;
271        state
272            .artifact_store
273            .list_by_task_and_type(&params.task_id, &artifact_type)
274            .await?
275    } else {
276        state.artifact_store.list_by_task(&params.task_id).await?
277    };
278
279    Ok(ListArtifactsResult { artifacts })
280}
281
282// ---------------------------------------------------------------------------
283// tasks.provideArtifact
284// ---------------------------------------------------------------------------
285
286#[derive(Debug, Deserialize)]
287#[serde(rename_all = "camelCase")]
288pub struct ProvideArtifactParams {
289    pub task_id: String,
290    pub agent_id: String,
291    #[serde(rename = "type")]
292    pub artifact_type: String,
293    pub content: String,
294    pub context: Option<String>,
295    pub request_id: Option<String>,
296    pub metadata: Option<BTreeMap<String, String>>,
297}
298
299#[derive(Debug, Serialize)]
300pub struct ProvideArtifactResult {
301    pub artifact: Artifact,
302}
303
304pub async fn provide_artifact(
305    state: &AppState,
306    params: ProvideArtifactParams,
307) -> Result<ProvideArtifactResult, RpcError> {
308    let task = state
309        .task_store
310        .get(&params.task_id)
311        .await?
312        .ok_or_else(|| RpcError::NotFound(format!("Task {} not found", params.task_id)))?;
313
314    let agent_id = params.agent_id.trim();
315    if agent_id.is_empty() {
316        return Err(RpcError::BadRequest(
317            "agentId is required for artifact submission".to_string(),
318        ));
319    }
320
321    let content = params.content.trim();
322    if content.is_empty() {
323        return Err(RpcError::BadRequest(
324            "artifact content cannot be blank".to_string(),
325        ));
326    }
327
328    let artifact = Artifact {
329        id: uuid::Uuid::new_v4().to_string(),
330        artifact_type: parse_artifact_type(&params.artifact_type)?,
331        task_id: task.id,
332        workspace_id: task.workspace_id,
333        provided_by_agent_id: Some(agent_id.to_string()),
334        requested_by_agent_id: None,
335        request_id: params.request_id,
336        content: Some(content.to_string()),
337        context: params
338            .context
339            .as_deref()
340            .map(str::trim)
341            .filter(|value| !value.is_empty())
342            .map(str::to_string),
343        status: ArtifactStatus::Provided,
344        expires_at: None,
345        metadata: params.metadata,
346        created_at: Utc::now(),
347        updated_at: Utc::now(),
348    };
349
350    state.artifact_store.save(&artifact).await?;
351    Ok(ProvideArtifactResult { artifact })
352}
353
354fn parse_artifact_type(value: &str) -> Result<ArtifactType, RpcError> {
355    ArtifactType::from_str(value).ok_or_else(|| {
356        RpcError::BadRequest(format!(
357            "Invalid artifact type: {value}. Expected one of: screenshot, test_results, code_diff, logs"
358        ))
359    })
360}
361
362async fn serialize_tasks_with_evidence(
363    state: &AppState,
364    tasks: &[Task],
365) -> Result<Vec<serde_json::Value>, RpcError> {
366    let mut serialized = Vec::with_capacity(tasks.len());
367    for task in tasks {
368        serialized.push(serialize_task_with_evidence(state, task).await?);
369    }
370    Ok(serialized)
371}
372
373async fn serialize_task_with_evidence(
374    state: &AppState,
375    task: &Task,
376) -> Result<serde_json::Value, RpcError> {
377    let evidence_summary = build_task_evidence_summary(state, task).await?;
378    let board = match task.board_id.as_deref() {
379        Some(board_id) => state.kanban_store.get(board_id).await?,
380        None => None,
381    };
382    let story_readiness = build_task_story_readiness(
383        task,
384        &resolve_next_required_task_fields(board.as_ref(), task.column_id.as_deref()),
385    );
386    let invest_validation = build_task_invest_validation(task);
387    let mut task_value = serde_json::to_value(task)
388        .map_err(|error| RpcError::Internal(format!("Failed to serialize task: {error}")))?;
389    let task_object = task_value.as_object_mut().ok_or_else(|| {
390        RpcError::Internal("Task payload must serialize to a JSON object".to_string())
391    })?;
392    task_object.insert(
393        "artifactSummary".to_string(),
394        serde_json::to_value(&evidence_summary.artifact).map_err(|error| {
395            RpcError::Internal(format!(
396                "Failed to serialize task artifact summary: {error}"
397            ))
398        })?,
399    );
400    task_object.insert(
401        "evidenceSummary".to_string(),
402        serde_json::to_value(&evidence_summary).map_err(|error| {
403            RpcError::Internal(format!(
404                "Failed to serialize task evidence summary: {error}"
405            ))
406        })?,
407    );
408    task_object.insert(
409        "storyReadiness".to_string(),
410        serde_json::to_value(&story_readiness).map_err(|error| {
411            RpcError::Internal(format!(
412                "Failed to serialize task story readiness summary: {error}"
413            ))
414        })?,
415    );
416    task_object.insert(
417        "investValidation".to_string(),
418        serde_json::to_value(&invest_validation).map_err(|error| {
419            RpcError::Internal(format!(
420                "Failed to serialize task INVEST validation summary: {error}"
421            ))
422        })?,
423    );
424    Ok(task_value)
425}
426
427async fn build_task_evidence_summary(
428    state: &AppState,
429    task: &Task,
430) -> Result<TaskEvidenceSummary, RpcError> {
431    let artifacts = state.artifact_store.list_by_task(&task.id).await?;
432    let mut by_type = BTreeMap::new();
433    for artifact in &artifacts {
434        let key = artifact.artifact_type.as_str().to_string();
435        *by_type.entry(key).or_insert(0) += 1;
436    }
437
438    let board = match task.board_id.as_deref() {
439        Some(board_id) => state.kanban_store.get(board_id).await?,
440        None => None,
441    };
442    let required_artifacts =
443        resolve_next_required_artifacts(board.as_ref(), task.column_id.as_deref());
444    let present_artifacts = by_type.keys().cloned().collect::<BTreeSet<_>>();
445    let missing_required = required_artifacts
446        .into_iter()
447        .filter(|artifact| !present_artifacts.contains(artifact))
448        .collect::<Vec<_>>();
449
450    let latest_status = task
451        .lane_sessions
452        .last()
453        .map(|session| task_lane_session_status_as_str(&session.status).to_string())
454        .unwrap_or_else(|| {
455            if task.session_ids.is_empty() {
456                "idle".to_string()
457            } else {
458                "unknown".to_string()
459            }
460        });
461
462    Ok(TaskEvidenceSummary {
463        artifact: TaskArtifactSummary {
464            total: artifacts.len(),
465            by_type,
466            required_satisfied: missing_required.is_empty(),
467            missing_required,
468        },
469        verification: TaskVerificationSummary {
470            has_verdict: task.verification_verdict.is_some(),
471            verdict: task
472                .verification_verdict
473                .as_ref()
474                .map(|verdict| verdict.as_str().to_string()),
475            has_report: task
476                .verification_report
477                .as_ref()
478                .is_some_and(|report| !report.trim().is_empty()),
479        },
480        completion: TaskCompletionSummary {
481            has_summary: task
482                .completion_summary
483                .as_ref()
484                .is_some_and(|summary| !summary.trim().is_empty()),
485        },
486        runs: TaskRunSummary {
487            total: task.session_ids.len(),
488            latest_status,
489        },
490    })
491}
492
493fn resolve_next_required_artifacts(
494    board: Option<&KanbanBoard>,
495    current_column_id: Option<&str>,
496) -> Vec<String> {
497    let current_column_id = current_column_id.unwrap_or("backlog").to_ascii_lowercase();
498    let next_column_id = KANBAN_HAPPY_PATH_COLUMN_ORDER
499        .iter()
500        .position(|column_id| *column_id == current_column_id)
501        .and_then(|index| KANBAN_HAPPY_PATH_COLUMN_ORDER.get(index + 1))
502        .copied();
503    let Some(next_column_id) = next_column_id else {
504        return Vec::new();
505    };
506
507    board
508        .and_then(|board| {
509            board
510                .columns
511                .iter()
512                .find(|column| column.id == next_column_id)
513        })
514        .and_then(|column| column.automation.as_ref())
515        .and_then(|automation| automation.required_artifacts.clone())
516        .unwrap_or_default()
517}
518
519fn resolve_next_required_task_fields(
520    board: Option<&KanbanBoard>,
521    current_column_id: Option<&str>,
522) -> Vec<String> {
523    let current_column_id = current_column_id.unwrap_or("backlog").to_ascii_lowercase();
524    let next_column_id = KANBAN_HAPPY_PATH_COLUMN_ORDER
525        .iter()
526        .position(|column_id| *column_id == current_column_id)
527        .and_then(|index| KANBAN_HAPPY_PATH_COLUMN_ORDER.get(index + 1))
528        .copied();
529    let Some(next_column_id) = next_column_id else {
530        return Vec::new();
531    };
532
533    board
534        .and_then(|board| {
535            board
536                .columns
537                .iter()
538                .find(|column| column.id == next_column_id)
539        })
540        .and_then(|column| column.automation.as_ref())
541        .and_then(|automation| automation.required_task_fields.clone())
542        .unwrap_or_default()
543}
544
545fn task_lane_session_status_as_str(status: &TaskLaneSessionStatus) -> &'static str {
546    match status {
547        TaskLaneSessionStatus::Running => "running",
548        TaskLaneSessionStatus::Completed => "completed",
549        TaskLaneSessionStatus::Failed => "failed",
550        TaskLaneSessionStatus::TimedOut => "timed_out",
551        TaskLaneSessionStatus::Transitioned => "transitioned",
552    }
553}
554
555#[cfg(test)]
556mod tests {
557    use super::*;
558    use crate::models::kanban::KanbanColumnAutomation;
559    use crate::models::task::{TaskLaneSession, VerificationVerdict};
560    use crate::{AppState, AppStateInner, Database};
561    use std::sync::Arc;
562
563    async fn setup_state() -> AppState {
564        let db = Database::open_in_memory().expect("in-memory db should open");
565        let state: AppState = Arc::new(AppStateInner::new(db));
566        state
567            .workspace_store
568            .ensure_default()
569            .await
570            .expect("default workspace should exist");
571        state
572    }
573
574    #[tokio::test]
575    async fn provide_and_list_artifacts_roundtrip() {
576        let state = setup_state().await;
577        let created = create(
578            &state,
579            CreateParams {
580                title: "Artifact task".to_string(),
581                objective: "Store screenshot evidence".to_string(),
582                workspace_id: "default".to_string(),
583                session_id: None,
584                scope: None,
585                acceptance_criteria: None,
586                verification_commands: None,
587                test_cases: None,
588                dependencies: None,
589                parallel_group: None,
590            },
591        )
592        .await
593        .expect("task should be created");
594        let created_task_id = created.task["id"]
595            .as_str()
596            .expect("created task id")
597            .to_string();
598
599        let provided = provide_artifact(
600            &state,
601            ProvideArtifactParams {
602                task_id: created_task_id.clone(),
603                agent_id: "agent-1".to_string(),
604                artifact_type: "screenshot".to_string(),
605                content: "base64-content".to_string(),
606                context: Some("Verification screenshot".to_string()),
607                request_id: None,
608                metadata: None,
609            },
610        )
611        .await
612        .expect("artifact should be created");
613
614        assert_eq!(provided.artifact.artifact_type, ArtifactType::Screenshot);
615        assert_eq!(
616            provided.artifact.provided_by_agent_id.as_deref(),
617            Some("agent-1")
618        );
619
620        let listed = list_artifacts(
621            &state,
622            ListArtifactsParams {
623                task_id: created_task_id,
624                artifact_type: Some("screenshot".to_string()),
625            },
626        )
627        .await
628        .expect("artifacts should be listed");
629
630        assert_eq!(listed.artifacts.len(), 1);
631        assert_eq!(
632            listed.artifacts[0].context.as_deref(),
633            Some("Verification screenshot")
634        );
635    }
636
637    #[tokio::test]
638    async fn rpc_task_methods_include_evidence_summary() {
639        let state = setup_state().await;
640        let mut board = state
641            .kanban_store
642            .ensure_default_board("default")
643            .await
644            .expect("default board should exist");
645        let dev_column = board
646            .columns
647            .iter_mut()
648            .find(|column| column.id == "dev")
649            .expect("dev column");
650        dev_column.automation = Some(KanbanColumnAutomation {
651            enabled: true,
652            required_artifacts: Some(vec!["screenshot".to_string()]),
653            required_task_fields: Some(vec![
654                "scope".to_string(),
655                "acceptance_criteria".to_string(),
656                "verification_plan".to_string(),
657            ]),
658            ..Default::default()
659        });
660        state
661            .kanban_store
662            .update(&board)
663            .await
664            .expect("board should update");
665
666        let mut task = Task::new(
667            "task-rpc-1".to_string(),
668            "RPC evidence".to_string(),
669            "Return parity task payload".to_string(),
670            "default".to_string(),
671            None,
672            None,
673            None,
674            None,
675            None,
676            None,
677            None,
678        );
679        task.board_id = Some(board.id.clone());
680        task.column_id = Some("todo".to_string());
681        task.session_ids = vec!["session-1".to_string()];
682        task.lane_sessions = vec![TaskLaneSession {
683            session_id: "session-1".to_string(),
684            routa_agent_id: None,
685            column_id: Some("todo".to_string()),
686            column_name: Some("Todo".to_string()),
687            step_id: None,
688            step_index: None,
689            step_name: None,
690            provider: None,
691            role: None,
692            specialist_id: None,
693            specialist_name: None,
694            transport: None,
695            external_task_id: None,
696            context_id: None,
697            attempt: None,
698            loop_mode: None,
699            completion_requirement: None,
700            objective: None,
701            last_activity_at: None,
702            recovered_from_session_id: None,
703            recovery_reason: None,
704            status: TaskLaneSessionStatus::Running,
705            started_at: "2026-03-27T00:00:00Z".to_string(),
706            completed_at: None,
707        }];
708        task.completion_summary = Some("Done".to_string());
709        task.verification_verdict = Some(VerificationVerdict::Approved);
710        task.verification_report = Some("Verified".to_string());
711        state
712            .task_store
713            .save(&task)
714            .await
715            .expect("task should save");
716
717        let artifact = Artifact {
718            id: "artifact-rpc-1".to_string(),
719            artifact_type: ArtifactType::Screenshot,
720            task_id: task.id.clone(),
721            workspace_id: task.workspace_id.clone(),
722            provided_by_agent_id: Some("agent-1".to_string()),
723            requested_by_agent_id: None,
724            request_id: None,
725            content: Some("base64".to_string()),
726            context: None,
727            status: ArtifactStatus::Provided,
728            expires_at: None,
729            metadata: None,
730            created_at: Utc::now(),
731            updated_at: Utc::now(),
732        };
733        state
734            .artifact_store
735            .save(&artifact)
736            .await
737            .expect("artifact should save");
738
739        let get_value = get(
740            &state,
741            GetParams {
742                id: task.id.clone(),
743            },
744        )
745        .await
746        .expect("task should load");
747        assert_eq!(get_value["artifactSummary"]["total"], serde_json::json!(1));
748        assert_eq!(
749            get_value["evidenceSummary"]["artifact"]["requiredSatisfied"],
750            serde_json::json!(true)
751        );
752        assert_eq!(
753            get_value["evidenceSummary"]["verification"]["verdict"],
754            serde_json::json!("APPROVED")
755        );
756        assert_eq!(
757            get_value["evidenceSummary"]["runs"]["latestStatus"],
758            serde_json::json!("running")
759        );
760        assert_eq!(
761            get_value["storyReadiness"]["requiredTaskFields"],
762            serde_json::json!(["scope", "acceptance_criteria", "verification_plan"])
763        );
764        assert_eq!(
765            get_value["storyReadiness"]["ready"],
766            serde_json::json!(false)
767        );
768        assert_eq!(
769            get_value["investValidation"]["source"],
770            serde_json::json!("heuristic")
771        );
772
773        let listed = list(
774            &state,
775            ListParams {
776                workspace_id: "default".to_string(),
777                session_id: None,
778                status: None,
779                assigned_to: None,
780            },
781        )
782        .await
783        .expect("tasks should list");
784        assert_eq!(listed.tasks.len(), 1);
785        assert_eq!(
786            listed.tasks[0]["evidenceSummary"]["completion"]["hasSummary"],
787            serde_json::json!(true)
788        );
789        assert_eq!(
790            listed.tasks[0]["storyReadiness"]["ready"],
791            serde_json::json!(false)
792        );
793
794        let ready = find_ready(
795            &state,
796            FindReadyParams {
797                workspace_id: "default".to_string(),
798            },
799        )
800        .await
801        .expect("ready tasks should list");
802        assert_eq!(ready.tasks.len(), 1);
803        assert_eq!(
804            ready.tasks[0]["artifactSummary"]["byType"]["screenshot"],
805            serde_json::json!(1)
806        );
807        assert_eq!(
808            ready.tasks[0]["investValidation"]["source"],
809            serde_json::json!("heuristic")
810        );
811
812        let created = create(
813            &state,
814            CreateParams {
815                title: "Fresh task".to_string(),
816                objective: "No evidence yet".to_string(),
817                workspace_id: "default".to_string(),
818                session_id: None,
819                scope: None,
820                acceptance_criteria: None,
821                verification_commands: None,
822                test_cases: None,
823                dependencies: None,
824                parallel_group: None,
825            },
826        )
827        .await
828        .expect("task should create");
829        assert_eq!(
830            created.task["artifactSummary"]["total"],
831            serde_json::json!(0)
832        );
833        assert_eq!(
834            created.task["evidenceSummary"]["runs"]["latestStatus"],
835            serde_json::json!("idle")
836        );
837        assert_eq!(
838            created.task["storyReadiness"]["requiredTaskFields"],
839            serde_json::json!([])
840        );
841    }
842}