Files
ClaudeDo/tests/ClaudeDo.Worker.Tests/Runner/GitServiceMergeTests.cs

133 lines
5.1 KiB
C#

using ClaudeDo.Data.Git;
using ClaudeDo.Worker.Tests.Infrastructure;
namespace ClaudeDo.Worker.Tests.Runner;
public class GitServiceMergeTests : IDisposable
{
private readonly List<GitRepoFixture> _repos = new();
private GitRepoFixture NewRepo()
{
var r = new GitRepoFixture();
_repos.Add(r);
return r;
}
public void Dispose()
{
foreach (var r in _repos) try { r.Dispose(); } catch { }
}
[Fact]
public async Task GetCurrentBranchAsync_FreshRepo_ReturnsDefaultBranch()
{
if (!GitRepoFixture.IsGitAvailable()) return;
var repo = NewRepo();
var git = new GitService();
var branch = await git.GetCurrentBranchAsync(repo.RepoDir);
Assert.False(string.IsNullOrWhiteSpace(branch));
// Default branch is either "main" or "master" depending on git config.
Assert.True(branch == "main" || branch == "master",
$"Expected main or master, got '{branch}'");
}
[Fact]
public async Task ListLocalBranchesAsync_AfterCreatingSecondBranch_ReturnsBoth()
{
if (!GitRepoFixture.IsGitAvailable()) return;
var repo = NewRepo();
GitRepoFixture.RunGit(repo.RepoDir, "branch", "feature/x");
var git = new GitService();
var branches = await git.ListLocalBranchesAsync(repo.RepoDir);
Assert.Contains("feature/x", branches);
Assert.True(branches.Any(b => b == "main" || b == "master"));
}
[Fact]
public async Task IsMidMergeAsync_FreshRepo_ReturnsFalse()
{
if (!GitRepoFixture.IsGitAvailable()) return;
var repo = NewRepo();
var git = new GitService();
Assert.False(await git.IsMidMergeAsync(repo.RepoDir));
}
[Fact]
public async Task IsMidMergeAsync_MergeHeadPresent_ReturnsTrue()
{
if (!GitRepoFixture.IsGitAvailable()) return;
var repo = NewRepo();
// Simulate a mid-merge state by dropping a MERGE_HEAD file.
var mergeHead = Path.Combine(repo.RepoDir, ".git", "MERGE_HEAD");
File.WriteAllText(mergeHead, "0000000000000000000000000000000000000000\n");
var git = new GitService();
Assert.True(await git.IsMidMergeAsync(repo.RepoDir));
}
[Fact]
public async Task MergeNoFfAsync_DivergedNonConflicting_ReturnsZero_AndCreatesMergeCommit()
{
if (!GitRepoFixture.IsGitAvailable()) return;
var repo = NewRepo();
// Create a feature branch with one new file.
GitRepoFixture.RunGit(repo.RepoDir, "checkout", "-b", "feature/merge");
File.WriteAllText(Path.Combine(repo.RepoDir, "feature.txt"), "hello\n");
GitRepoFixture.RunGit(repo.RepoDir, "add", "-A");
GitRepoFixture.RunGit(repo.RepoDir, "commit", "-m", "feat: add feature.txt");
// Back to default and add a non-overlapping file so history diverges.
string defaultBranch;
try { defaultBranch = GitRepoFixture.RunGit(repo.RepoDir, "symbolic-ref", "--short", "refs/remotes/origin/HEAD").Trim().Replace("origin/", ""); }
catch { defaultBranch = "main"; }
if (string.IsNullOrEmpty(defaultBranch)) defaultBranch = "main";
try { GitRepoFixture.RunGit(repo.RepoDir, "checkout", defaultBranch); }
catch { GitRepoFixture.RunGit(repo.RepoDir, "checkout", "master"); defaultBranch = "master"; }
File.WriteAllText(Path.Combine(repo.RepoDir, "other.txt"), "other\n");
GitRepoFixture.RunGit(repo.RepoDir, "add", "-A");
GitRepoFixture.RunGit(repo.RepoDir, "commit", "-m", "chore: add other.txt");
var git = new GitService();
var (exitCode, stderr) = await git.MergeNoFfAsync(repo.RepoDir, "feature/merge", "Merge feature/merge");
Assert.Equal(0, exitCode);
// Confirm merge commit exists (two parents on HEAD).
var parents = GitRepoFixture.RunGit(repo.RepoDir, "rev-list", "--parents", "-n", "1", "HEAD").Trim();
Assert.True(parents.Split(' ').Length >= 3, $"Expected merge commit (3 tokens), got: '{parents}'");
}
[Fact]
public async Task MergeNoFfAsync_Conflict_ReturnsNonZero()
{
if (!GitRepoFixture.IsGitAvailable()) return;
var repo = NewRepo();
// Both branches modify README.md — guaranteed conflict.
GitRepoFixture.RunGit(repo.RepoDir, "checkout", "-b", "feature/conflict");
File.WriteAllText(Path.Combine(repo.RepoDir, "README.md"), "# feature side\n");
GitRepoFixture.RunGit(repo.RepoDir, "add", "-A");
GitRepoFixture.RunGit(repo.RepoDir, "commit", "-m", "feat: feature edit");
string defaultBranch = "main";
try { GitRepoFixture.RunGit(repo.RepoDir, "checkout", "main"); }
catch { GitRepoFixture.RunGit(repo.RepoDir, "checkout", "master"); defaultBranch = "master"; }
File.WriteAllText(Path.Combine(repo.RepoDir, "README.md"), "# main side\n");
GitRepoFixture.RunGit(repo.RepoDir, "add", "-A");
GitRepoFixture.RunGit(repo.RepoDir, "commit", "-m", "chore: main edit");
var git = new GitService();
var (exitCode, _) = await git.MergeNoFfAsync(repo.RepoDir, "feature/conflict", "merge");
Assert.NotEqual(0, exitCode);
}
}