237 lines
6.9 KiB
Go
237 lines
6.9 KiB
Go
package inbox
|
|
|
|
import (
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
)
|
|
|
|
// TestSendCreatesNewThread verifies send creates a pending thread with the initial task message.
|
|
func TestSendCreatesNewThread(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
dbPath := initCommandTestDB(t)
|
|
|
|
sendOut := runInboxCommand(
|
|
t,
|
|
"--db", dbPath,
|
|
"--json",
|
|
"send",
|
|
"--from", "leader",
|
|
"--to", "worker-a",
|
|
"--subject", "Implement feature X",
|
|
"--summary", "Add retry policy",
|
|
"--body", "Implement retry handling for the HTTP client.",
|
|
"--run", "run_blog_001",
|
|
"--task", "T1",
|
|
)
|
|
|
|
var sendResp map[string]any
|
|
mustDecodeJSON(t, sendOut, &sendResp)
|
|
|
|
if got := nestedString(t, sendResp, "data", "thread", "thread_id"); got == "" {
|
|
t.Fatalf("expected thread_id, got empty")
|
|
}
|
|
if got := nestedString(t, sendResp, "data", "thread", "status"); got != "pending" {
|
|
t.Fatalf("expected pending status, got %q", got)
|
|
}
|
|
if got := nestedString(t, sendResp, "data", "thread", "created_by"); got != "leader" {
|
|
t.Fatalf("expected created_by leader, got %q", got)
|
|
}
|
|
if got := nestedString(t, sendResp, "data", "thread", "assigned_to"); got != "worker-a" {
|
|
t.Fatalf("expected assigned_to worker-a, got %q", got)
|
|
}
|
|
if got := nestedString(t, sendResp, "data", "message", "kind"); got != "task" {
|
|
t.Fatalf("expected message kind task, got %q", got)
|
|
}
|
|
}
|
|
|
|
// TestSendAppendsMessageToExistingThread verifies send appends a message without creating a new thread.
|
|
func TestSendAppendsMessageToExistingThread(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
dbPath := initCommandTestDB(t)
|
|
threadID := sendPendingThread(t, dbPath, "leader", "worker-d", "Build editor", "Create editor v1")
|
|
|
|
appendOut := runInboxCommand(
|
|
t,
|
|
"--db", dbPath,
|
|
"--json",
|
|
"send",
|
|
"--from", "leader",
|
|
"--to", "worker-d",
|
|
"--thread", threadID,
|
|
"--summary", "Use a markdown editor",
|
|
"--body", "Prefer a textarea-based markdown editor for v1.",
|
|
)
|
|
|
|
var appendResp map[string]any
|
|
mustDecodeJSON(t, appendOut, &appendResp)
|
|
if got := nestedString(t, appendResp, "data", "thread", "thread_id"); got != threadID {
|
|
t.Fatalf("expected same thread_id %q, got %q", threadID, got)
|
|
}
|
|
if got := nestedString(t, appendResp, "data", "thread", "status"); got != "pending" {
|
|
t.Fatalf("expected thread status to stay pending, got %q", got)
|
|
}
|
|
|
|
showOut := runInboxCommand(
|
|
t,
|
|
"--db", dbPath,
|
|
"--json",
|
|
"show",
|
|
"--thread", threadID,
|
|
)
|
|
|
|
var showResp map[string]any
|
|
mustDecodeJSON(t, showOut, &showResp)
|
|
messages, ok := nestedValue(t, showResp, "data", "messages").([]any)
|
|
if !ok || len(messages) != 2 {
|
|
t.Fatalf("expected two messages after append, got %#v", nestedValue(t, showResp, "data", "messages"))
|
|
}
|
|
}
|
|
|
|
// TestSendReadsBodyFromBodyFile verifies send loads the message body from a body file.
|
|
func TestSendReadsBodyFromBodyFile(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tempDir := t.TempDir()
|
|
dbPath := filepath.Join(tempDir, "coord.db")
|
|
bodyPath := filepath.Join(tempDir, "task.md")
|
|
bodyContent := "Create the first editor screen.\nUse markdown syntax."
|
|
if err := os.WriteFile(bodyPath, []byte(bodyContent), 0o644); err != nil {
|
|
t.Fatalf("write body file: %v", err)
|
|
}
|
|
runInboxCommand(t, "--db", dbPath, "--json", "init")
|
|
|
|
sendOut := runInboxCommand(
|
|
t,
|
|
"--db", dbPath,
|
|
"--json",
|
|
"send",
|
|
"--from", "leader",
|
|
"--to", "worker-d",
|
|
"--subject", "Build admin editor",
|
|
"--summary", "Create the first editor screen",
|
|
"--body-file", bodyPath,
|
|
)
|
|
|
|
var sendResp map[string]any
|
|
mustDecodeJSON(t, sendOut, &sendResp)
|
|
threadID := nestedString(t, sendResp, "data", "thread", "thread_id")
|
|
|
|
showOut := runInboxCommand(
|
|
t,
|
|
"--db", dbPath,
|
|
"--json",
|
|
"show",
|
|
"--thread", threadID,
|
|
)
|
|
|
|
var showResp map[string]any
|
|
mustDecodeJSON(t, showOut, &showResp)
|
|
messages, ok := nestedValue(t, showResp, "data", "messages").([]any)
|
|
if !ok || len(messages) != 1 {
|
|
t.Fatalf("expected one message, got %#v", nestedValue(t, showResp, "data", "messages"))
|
|
}
|
|
message, ok := messages[0].(map[string]any)
|
|
if !ok {
|
|
t.Fatalf("expected message object, got %#v", messages[0])
|
|
}
|
|
if got, _ := message["body"].(string); got != bodyContent {
|
|
t.Fatalf("expected body %q, got %#v", bodyContent, message["body"])
|
|
}
|
|
}
|
|
|
|
// TestSendAttachesArtifactWithMetadata verifies send persists an artifact and its metadata on the message.
|
|
func TestSendAttachesArtifactWithMetadata(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tempDir := t.TempDir()
|
|
dbPath := filepath.Join(tempDir, "coord.db")
|
|
artifactPath := filepath.Join(tempDir, "task.md")
|
|
if err := os.WriteFile(artifactPath, []byte("task brief"), 0o644); err != nil {
|
|
t.Fatalf("write artifact file: %v", err)
|
|
}
|
|
runInboxCommand(t, "--db", dbPath, "--json", "init")
|
|
|
|
sendOut := runInboxCommand(
|
|
t,
|
|
"--db", dbPath,
|
|
"--json",
|
|
"send",
|
|
"--from", "leader",
|
|
"--to", "worker-d",
|
|
"--subject", "Build admin editor",
|
|
"--summary", "Create the first editor screen",
|
|
"--artifact", artifactPath,
|
|
"--artifact-kind", "brief",
|
|
"--artifact-metadata-json", `{"label":"task-brief"}`,
|
|
)
|
|
|
|
var sendResp map[string]any
|
|
mustDecodeJSON(t, sendOut, &sendResp)
|
|
|
|
artifacts, ok := nestedValue(t, sendResp, "data", "message", "artifacts").([]any)
|
|
if !ok || len(artifacts) != 1 {
|
|
t.Fatalf("expected one artifact, got %#v", nestedValue(t, sendResp, "data", "message", "artifacts"))
|
|
}
|
|
artifact, ok := artifacts[0].(map[string]any)
|
|
if !ok {
|
|
t.Fatalf("expected artifact object, got %#v", artifacts[0])
|
|
}
|
|
if got, _ := artifact["path"].(string); got != artifactPath {
|
|
t.Fatalf("expected artifact path %q, got %#v", artifactPath, artifact["path"])
|
|
}
|
|
if got, _ := artifact["kind"].(string); got != "brief" {
|
|
t.Fatalf("expected artifact kind brief, got %#v", artifact["kind"])
|
|
}
|
|
metadata, ok := artifact["metadata_json"].(map[string]any)
|
|
if !ok {
|
|
t.Fatalf("expected metadata_json object, got %#v", artifact["metadata_json"])
|
|
}
|
|
if got, _ := metadata["label"].(string); got != "task-brief" {
|
|
t.Fatalf("expected metadata_json.label task-brief, got %#v", metadata["label"])
|
|
}
|
|
}
|
|
|
|
// TestSendRejectsInvalidPayloadJSON verifies send returns invalid_input for malformed payload JSON.
|
|
func TestSendRejectsInvalidPayloadJSON(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
dbPath := initCommandTestDB(t)
|
|
stdout, _, exitCode := executeInboxCommand(
|
|
"--db", dbPath,
|
|
"--json",
|
|
"send",
|
|
"--from", "leader",
|
|
"--to", "worker-z",
|
|
"--subject", "Invalid payload json",
|
|
"--payload-json", "not-json",
|
|
)
|
|
if exitCode != 30 {
|
|
t.Fatalf("expected exit code 30, got %d", exitCode)
|
|
}
|
|
assertErrorJSON(t, stdout, "invalid_input")
|
|
}
|
|
|
|
// TestSendRejectsInvalidArtifactMetadataJSON verifies send returns invalid_input for malformed artifact metadata JSON.
|
|
func TestSendRejectsInvalidArtifactMetadataJSON(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
dbPath := initCommandTestDB(t)
|
|
stdout, _, exitCode := executeInboxCommand(
|
|
"--db", dbPath,
|
|
"--json",
|
|
"send",
|
|
"--from", "leader",
|
|
"--to", "worker-z",
|
|
"--subject", "Invalid artifact json",
|
|
"--artifact", "/tmp/report.md",
|
|
"--artifact-metadata-json", "not-json",
|
|
)
|
|
if exitCode != 30 {
|
|
t.Fatalf("expected exit code 30, got %d", exitCode)
|
|
}
|
|
assertErrorJSON(t, stdout, "invalid_input")
|
|
}
|