using ClaudeDo.Worker.Prime; using ClaudeDo.Worker.Runner; using ClaudeDo.Worker.Tests.Infrastructure; using Microsoft.Extensions.Logging.Abstractions; namespace ClaudeDo.Worker.Tests.Prime; public class PrimeRunnerTests : IDisposable { private readonly DbFixture _db = new(); public void Dispose() => _db.Dispose(); private sealed class FakeClock : IPrimeClock { public DateTimeOffset Now { get; set; } = new DateTimeOffset(2026, 6, 3, 8, 0, 0, TimeSpan.FromHours(2)); } private sealed class FakeClaudeProcess : IClaudeProcess { private readonly TimeSpan _delay; private readonly int _exitCode; private readonly string[] _emitLines; private readonly string? _result; public FakeClaudeProcess(TimeSpan delay = default, int exitCode = 0, string[]? emitLines = null, string? result = null) { _delay = delay; _exitCode = exitCode; _emitLines = emitLines ?? []; _result = result; } public async Task RunAsync( string arguments, string prompt, string workingDirectory, Func onStdoutLine, CancellationToken ct) { if (_delay > TimeSpan.Zero) await Task.Delay(_delay, ct); foreach (var line in _emitLines) await onStdoutLine(line); return new RunResult { ExitCode = _exitCode, ResultMarkdown = _result ?? (_exitCode == 0 ? "ok" : null) }; } } private sealed class RecordingPrimeBroadcaster : IPrimeBroadcaster { public int StartedCount { get; private set; } public List Lines { get; } = []; public List FinishedResults { get; } = []; public Task PrimeFiredAsync(Guid scheduleId, bool success, string message, DateTimeOffset firedAt) => Task.CompletedTask; public Task PrepStartedAsync() { StartedCount++; return Task.CompletedTask; } public Task PrepLineAsync(string line) { Lines.Add(line); return Task.CompletedTask; } public Task PrepFinishedAsync(bool success) { FinishedResults.Add(success); return Task.CompletedTask; } } private PrimeRunner NewRunner(TimeSpan claudeDelay = default, int exitCode = 0) => NewRunner(new FakeClaudeProcess(claudeDelay, exitCode), new RecordingPrimeBroadcaster()); private PrimeRunner NewRunner(FakeClaudeProcess claude, IPrimeBroadcaster broadcaster) => new PrimeRunner( claude, _db.CreateFactory(), new FakeClock(), NullLogger.Instance, broadcaster); private static PrimeScheduleDto DefaultSchedule() => new(Guid.Empty, 0, TimeSpan.Zero, true, null, null); [Fact] public async Task FireAsync_returns_success_when_claude_exits_zero() { var runner = NewRunner(); var outcome = await runner.FireAsync(DefaultSchedule(), CancellationToken.None); Assert.True(outcome.Success); Assert.Equal("Daily prep complete", outcome.Message); } [Fact] public async Task FireAsync_returns_failure_when_claude_exits_nonzero() { var runner = NewRunner(exitCode: 1); var outcome = await runner.FireAsync(DefaultSchedule(), CancellationToken.None); Assert.False(outcome.Success); Assert.Contains("exit code", outcome.Message, StringComparison.OrdinalIgnoreCase); } [Fact] public async Task FireAsync_returns_already_running_when_gate_held() { var runner = NewRunner(claudeDelay: TimeSpan.FromSeconds(2)); var schedule = DefaultSchedule(); var first = runner.FireAsync(schedule, CancellationToken.None); var second = await runner.FireAsync(schedule, CancellationToken.None); Assert.False(second.Success); Assert.Contains("already running", second.Message, StringComparison.OrdinalIgnoreCase); await first; } [Fact] public async Task FireAsync_streams_started_lines_and_finished() { var broadcaster = new RecordingPrimeBroadcaster(); var claude = new FakeClaudeProcess(emitLines: ["{\"a\":1}", "{\"b\":2}"], exitCode: 0, result: "ok"); var runner = NewRunner(claude, broadcaster); var schedule = new PrimeScheduleDto(Guid.Empty, 0, TimeSpan.Zero, true, null, null); var outcome = await runner.FireAsync(schedule, CancellationToken.None); Assert.True(outcome.Success); Assert.Equal(1, broadcaster.StartedCount); Assert.Equal(new[] { "{\"a\":1}", "{\"b\":2}" }, broadcaster.Lines); Assert.Single(broadcaster.FinishedResults); Assert.True(broadcaster.FinishedResults[0]); } }