1. ArgumentList (fix injection): ClaudeArgsBuilder.Build() now returns IReadOnlyList<string>; ClaudeProcess populates ProcessStartInfo.ArgumentList instead of Arguments, so values like system prompts are never shell-split. DailyPrepPrompt, RefinePrompt, and WeekReportService migrated similarly. All IClaudeProcess fakes updated. 2. ContinueAsync exception guard: wrap RunOnceAsync in try/catch matching the RunAsync pattern so an unexpected exception never leaves the task stuck in Running status. 3. Planning chain cascade: OnChildFinishedAsync now calls CancelAsync on the immediate blocked successor when a child fails or is cancelled, triggering a recursive cascade that clears the entire remaining chain instead of leaving it wedged. 4. FailAsync guard: restrict valid source states to Running and Queued; WaitingForReview -> Failed is now rejected, preventing an invalid transition that could corrupt the review workflow. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
229 lines
7.9 KiB
C#
229 lines
7.9 KiB
C#
using ClaudeDo.Worker.Runner;
|
|
|
|
namespace ClaudeDo.Worker.Tests.Runner;
|
|
|
|
public sealed class ClaudeArgsBuilderTests
|
|
{
|
|
private readonly ClaudeArgsBuilder _builder = new();
|
|
|
|
[Fact]
|
|
public void Default_Config_Produces_Base_Args()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig(null, null, null, null));
|
|
Assert.Contains("-p", args);
|
|
Assert.Contains("--output-format", args);
|
|
Assert.Contains("stream-json", args);
|
|
Assert.Contains("--verbose", args);
|
|
Assert.Contains("--permission-mode", args);
|
|
Assert.Contains("auto", args);
|
|
Assert.DoesNotContain("--dangerously-skip-permissions", args);
|
|
Assert.Contains("--json-schema", args);
|
|
Assert.DoesNotContain("--model", args);
|
|
Assert.DoesNotContain("--append-system-prompt", args);
|
|
Assert.DoesNotContain("--agents", args);
|
|
Assert.DoesNotContain("--resume", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void Model_Adds_Model_Flag()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig("sonnet-4-6", null, null, null));
|
|
Assert.Contains("--model", args);
|
|
Assert.Contains("sonnet-4-6", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void SystemPrompt_Adds_Append_System_Prompt_Flag()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig(null, "Be concise.", null, null));
|
|
Assert.Contains("--append-system-prompt", args);
|
|
Assert.Contains("Be concise.", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void AgentPath_Adds_Agents_Flag_As_Json()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig(null, null, "/path/to/agent.md", null));
|
|
Assert.Contains("--agents", args);
|
|
Assert.Contains(args, a => a.Contains("/path/to/agent.md"));
|
|
}
|
|
|
|
[Fact]
|
|
public void ResumeSessionId_Adds_Resume_Flag()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig(null, null, null, "sess-abc-123"));
|
|
Assert.Contains("--resume", args);
|
|
Assert.Contains("sess-abc-123", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void All_Options_Set_Includes_All_Flags()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig("opus-4-6", "Be thorough.", "/agents/dev.md", "sess-xyz"));
|
|
Assert.Contains("--model", args);
|
|
Assert.Contains("opus-4-6", args);
|
|
Assert.Contains("--append-system-prompt", args);
|
|
Assert.Contains("--agents", args);
|
|
Assert.Contains("--resume", args);
|
|
Assert.Contains("sess-xyz", args);
|
|
Assert.Contains("--json-schema", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void SystemPrompt_With_Quotes_Is_Passed_Verbatim()
|
|
{
|
|
var prompt = """Don't say "hello".""";
|
|
var args = _builder.Build(new ClaudeRunConfig(null, prompt, null, null));
|
|
Assert.Contains("--append-system-prompt", args);
|
|
Assert.Contains(prompt, args);
|
|
}
|
|
|
|
[Fact]
|
|
public void SystemPrompt_With_Newline_Is_Passed_As_Single_Element()
|
|
{
|
|
var prompt = "line1\nline2";
|
|
var args = _builder.Build(new ClaudeRunConfig(
|
|
Model: null,
|
|
SystemPrompt: prompt,
|
|
AgentPath: null,
|
|
ResumeSessionId: null));
|
|
|
|
var list = args.ToList();
|
|
var idx = list.IndexOf("--append-system-prompt");
|
|
Assert.True(idx >= 0);
|
|
Assert.Equal(prompt, list[idx + 1]);
|
|
}
|
|
|
|
[Fact]
|
|
public void SystemPrompt_With_Tab_Is_Passed_As_Single_Element()
|
|
{
|
|
var prompt = "col1\tcol2";
|
|
var args = _builder.Build(new ClaudeRunConfig(
|
|
Model: null,
|
|
SystemPrompt: prompt,
|
|
AgentPath: null,
|
|
ResumeSessionId: null));
|
|
|
|
Assert.Contains(prompt, args);
|
|
}
|
|
|
|
[Fact]
|
|
public void MaxTurns_Adds_Flag_When_Positive()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig(null, null, null, null, MaxTurns: 25));
|
|
Assert.Contains("--max-turns", args);
|
|
Assert.Contains("25", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void MaxTurns_Omitted_When_Null_Or_ZeroOrLess()
|
|
{
|
|
var a = _builder.Build(new ClaudeRunConfig(null, null, null, null, MaxTurns: null));
|
|
var b = _builder.Build(new ClaudeRunConfig(null, null, null, null, MaxTurns: 0));
|
|
Assert.DoesNotContain("--max-turns", a);
|
|
Assert.DoesNotContain("--max-turns", b);
|
|
}
|
|
|
|
[Fact]
|
|
public void PermissionMode_bypass_Maps_To_Auto()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig(null, null, null, null, PermissionMode: "bypassPermissions"));
|
|
Assert.Contains("--permission-mode", args);
|
|
Assert.Contains("auto", args);
|
|
Assert.DoesNotContain("--dangerously-skip-permissions", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void PermissionMode_acceptEdits_Emits_PermissionMode_Flag()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig(null, null, null, null, PermissionMode: "acceptEdits"));
|
|
Assert.Contains("--permission-mode", args);
|
|
Assert.Contains("acceptEdits", args);
|
|
Assert.DoesNotContain("--dangerously-skip-permissions", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void PermissionMode_Null_Defaults_To_Auto()
|
|
{
|
|
var args = _builder.Build(new ClaudeRunConfig(null, null, null, null));
|
|
Assert.Contains("--permission-mode", args);
|
|
Assert.Contains("auto", args);
|
|
Assert.DoesNotContain("--dangerously-skip-permissions", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void Build_emits_mcpConfig_and_allowedTools_when_set()
|
|
{
|
|
var args = new ClaudeArgsBuilder().Build(new ClaudeRunConfig(
|
|
Model: null, SystemPrompt: null, AgentPath: null, ResumeSessionId: null,
|
|
McpConfigPath: "C:\\tmp\\t_mcp.json",
|
|
AllowedTools: "mcp__claudedo_run__SuggestImprovement"));
|
|
Assert.Contains("--mcp-config", args);
|
|
Assert.Contains(args, a => a.Contains("t_mcp.json"));
|
|
Assert.Contains("--allowedTools", args);
|
|
Assert.Contains("mcp__claudedo_run__SuggestImprovement", args);
|
|
}
|
|
|
|
[Fact]
|
|
public void SystemPrompt_ContainingDangerousFlag_IsPassedAsLiteral_NotAsFlag()
|
|
{
|
|
// If the system prompt contains "--dangerously-skip-permissions", it must arrive
|
|
// as the VALUE of --append-system-prompt, not as a standalone CLI flag.
|
|
const string dangerousPrompt = "--dangerously-skip-permissions";
|
|
var args = _builder.Build(new ClaudeRunConfig(null, dangerousPrompt, null, null));
|
|
|
|
var list = args.ToList();
|
|
var flagIdx = list.IndexOf("--append-system-prompt");
|
|
Assert.True(flagIdx >= 0, "--append-system-prompt flag must be present");
|
|
// The dangerous string sits immediately after the flag as its value.
|
|
Assert.Equal(dangerousPrompt, list[flagIdx + 1]);
|
|
// It does NOT appear as a separate element (i.e., not treated as its own flag).
|
|
Assert.Equal(1, list.Count(a => a == dangerousPrompt));
|
|
}
|
|
}
|
|
|
|
public sealed class MergeInstructionsTests
|
|
{
|
|
[Fact]
|
|
public void All_Empty_Returns_Empty()
|
|
{
|
|
var s = ClaudeDo.Worker.Runner.TaskRunner.MergeInstructions(null, null, null);
|
|
Assert.Equal("", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void Only_Global_Returns_Global()
|
|
{
|
|
var s = ClaudeDo.Worker.Runner.TaskRunner.MergeInstructions("global rule", null, null);
|
|
Assert.Equal("global rule", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void Only_Task_Returns_Task()
|
|
{
|
|
var s = ClaudeDo.Worker.Runner.TaskRunner.MergeInstructions(null, null, "task rule");
|
|
Assert.Equal("task rule", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void Global_And_Task_Are_Prepended_With_Separator()
|
|
{
|
|
var s = ClaudeDo.Worker.Runner.TaskRunner.MergeInstructions("global rule", null, "task rule");
|
|
Assert.Equal("global rule\n\ntask rule", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void All_Three_Are_Joined_In_Order()
|
|
{
|
|
var s = ClaudeDo.Worker.Runner.TaskRunner.MergeInstructions("G", "L", "T");
|
|
Assert.Equal("G\n\nL\n\nT", s);
|
|
}
|
|
|
|
[Fact]
|
|
public void Whitespace_Only_Parts_Are_Skipped()
|
|
{
|
|
var s = ClaudeDo.Worker.Runner.TaskRunner.MergeInstructions(" ", "L", "\t\n");
|
|
Assert.Equal("L", s);
|
|
}
|
|
}
|