1use 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#[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) = ¶ms.session_id {
93 state.task_store.list_by_session(session_id).await?
95 } else if let Some(assignee) = ¶ms.assigned_to {
96 state.task_store.list_by_assignee(assignee).await?
97 } else if let Some(status_str) = ¶ms.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(¶ms.workspace_id, &status)
103 .await?
104 } else {
105 state
106 .task_store
107 .list_by_workspace(¶ms.workspace_id)
108 .await?
109 };
110
111 Ok(ListResult {
112 tasks: serialize_tasks_with_evidence(state, &tasks).await?,
113 })
114}
115
116#[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(¶ms.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#[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#[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(¶ms.id).await?;
198 Ok(DeleteResult { deleted: true })
199}
200
201#[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(¶ms.status)
222 .ok_or_else(|| RpcError::BadRequest(format!("Invalid status: {}", params.status)))?;
223 state.task_store.update_status(¶ms.id, &status).await?;
224 Ok(UpdateStatusResult { updated: true })
225}
226
227#[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(¶ms.workspace_id)
242 .await?;
243 Ok(ListResult {
244 tasks: serialize_tasks_with_evidence(state, &tasks).await?,
245 })
246}
247
248#[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(¶ms.task_id, &artifact_type)
274 .await?
275 } else {
276 state.artifact_store.list_by_task(¶ms.task_id).await?
277 };
278
279 Ok(ListArtifactsResult { artifacts })
280}
281
282#[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(¶ms.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(¶ms.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}