setupGitRepo now creates an empty issues.jsonl for RepoContext, which triggers 'JSONL changed since import' validation in tests that set up their own .beads/ directory. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
776 lines
27 KiB
Go
776 lines
27 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/steveyegge/beads/internal/beads"
|
|
"github.com/steveyegge/beads/internal/git"
|
|
"github.com/steveyegge/beads/internal/storage/sqlite"
|
|
"github.com/steveyegge/beads/internal/syncbranch"
|
|
"github.com/steveyegge/beads/internal/types"
|
|
)
|
|
|
|
// TestSyncBranchModeWithPullFirst verifies that sync-branch mode config storage
|
|
// and retrieval works correctly. The pull-first sync gates on this config.
|
|
// This addresses Steve's review concern about --sync-branch regression.
|
|
func TestSyncBranchModeWithPullFirst(t *testing.T) {
|
|
ctx := context.Background()
|
|
tmpDir, cleanup := setupGitRepo(t)
|
|
defer cleanup()
|
|
|
|
// Setup: Create beads directory with database
|
|
beadsDir := filepath.Join(tmpDir, ".beads")
|
|
if err := os.MkdirAll(beadsDir, 0755); err != nil {
|
|
t.Fatalf("mkdir failed: %v", err)
|
|
}
|
|
|
|
// Create store and configure sync.branch
|
|
testDBPath := filepath.Join(beadsDir, "beads.db")
|
|
testStore, err := sqlite.New(ctx, testDBPath)
|
|
if err != nil {
|
|
t.Fatalf("failed to create store: %v", err)
|
|
}
|
|
defer testStore.Close()
|
|
|
|
// Set issue prefix (required)
|
|
if err := testStore.SetConfig(ctx, "issue_prefix", "test"); err != nil {
|
|
t.Fatalf("failed to set issue_prefix: %v", err)
|
|
}
|
|
|
|
// Configure sync.branch
|
|
if err := testStore.SetConfig(ctx, "sync.branch", "beads-metadata"); err != nil {
|
|
t.Fatalf("failed to set sync.branch: %v", err)
|
|
}
|
|
|
|
// Create the sync branch in git
|
|
if err := exec.Command("git", "branch", "beads-metadata").Run(); err != nil {
|
|
t.Fatalf("failed to create sync branch: %v", err)
|
|
}
|
|
|
|
// Create issues.jsonl with a test issue
|
|
jsonlPath := filepath.Join(beadsDir, "issues.jsonl")
|
|
issueContent := `{"id":"test-1","title":"Test Issue","status":"open","issue_type":"task","priority":2,"created_at":"2025-01-01T00:00:00Z","updated_at":"2025-01-01T00:00:00Z"}`
|
|
if err := os.WriteFile(jsonlPath, []byte(issueContent+"\n"), 0644); err != nil {
|
|
t.Fatalf("write JSONL failed: %v", err)
|
|
}
|
|
|
|
// Test 1: Verify sync.branch config is stored and retrievable
|
|
// This is what the pull-first sync checks at lines 181-189 in sync.go
|
|
syncBranch, err := testStore.GetConfig(ctx, "sync.branch")
|
|
if err != nil {
|
|
t.Fatalf("failed to get sync.branch config: %v", err)
|
|
}
|
|
if syncBranch != "beads-metadata" {
|
|
t.Errorf("sync.branch = %q, want %q", syncBranch, "beads-metadata")
|
|
}
|
|
t.Logf("✓ Sync-branch config correctly stored: %s", syncBranch)
|
|
|
|
// Test 2: Verify the git branch exists
|
|
checkCmd := exec.Command("git", "show-ref", "--verify", "--quiet", "refs/heads/beads-metadata")
|
|
if err := checkCmd.Run(); err != nil {
|
|
t.Error("expected beads-metadata branch to exist")
|
|
}
|
|
t.Log("✓ Git sync branch exists")
|
|
|
|
// Test 3: Verify the DB config key can be read directly by syncbranch package
|
|
// Note: syncbranch.Get() also checks config.yaml and env var, which may override
|
|
// the DB config in the beads repo test environment. We verify DB storage works.
|
|
dbValue, err := testStore.GetConfig(ctx, syncbranch.ConfigKey)
|
|
if err != nil {
|
|
t.Fatalf("failed to read %s from store: %v", syncbranch.ConfigKey, err)
|
|
}
|
|
if dbValue != "beads-metadata" {
|
|
t.Errorf("store.GetConfig(%s) = %q, want %q", syncbranch.ConfigKey, dbValue, "beads-metadata")
|
|
}
|
|
t.Logf("✓ sync.branch config key correctly stored: %s", dbValue)
|
|
|
|
// Key assertion: The sync-branch detection mechanism works
|
|
// When sync.branch is configured, doPullFirstSync gates on it (sync.go:181-189)
|
|
// and the daemon handles sync-branch commits (daemon_sync_branch.go)
|
|
}
|
|
|
|
// TestExternalBeadsDirWithPullFirst verifies that external BEADS_DIR mode
|
|
// is correctly detected and the commit/pull functions work.
|
|
// This addresses Steve's review concern about external beads dir regression.
|
|
func TestExternalBeadsDirWithPullFirst(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
// Setup: Create main project repo
|
|
mainDir, cleanupMain := setupGitRepo(t)
|
|
defer cleanupMain()
|
|
|
|
// Setup: Create separate external beads repo
|
|
// Resolve symlinks to avoid macOS /var -> /private/var issues
|
|
externalDir, err := filepath.EvalSymlinks(t.TempDir())
|
|
if err != nil {
|
|
t.Fatalf("eval symlinks failed: %v", err)
|
|
}
|
|
|
|
// Initialize external repo
|
|
if err := exec.Command("git", "-C", externalDir, "init", "--initial-branch=main").Run(); err != nil {
|
|
t.Fatalf("git init (external) failed: %v", err)
|
|
}
|
|
_ = exec.Command("git", "-C", externalDir, "config", "user.email", "test@test.com").Run()
|
|
_ = exec.Command("git", "-C", externalDir, "config", "user.name", "Test User").Run()
|
|
|
|
// Create initial commit in external repo
|
|
if err := os.WriteFile(filepath.Join(externalDir, "README.md"), []byte("External beads repo"), 0644); err != nil {
|
|
t.Fatalf("write README failed: %v", err)
|
|
}
|
|
_ = exec.Command("git", "-C", externalDir, "add", ".").Run()
|
|
if err := exec.Command("git", "-C", externalDir, "commit", "-m", "initial").Run(); err != nil {
|
|
t.Fatalf("external initial commit failed: %v", err)
|
|
}
|
|
|
|
// Create .beads directory in external repo
|
|
externalBeadsDir := filepath.Join(externalDir, ".beads")
|
|
if err := os.MkdirAll(externalBeadsDir, 0755); err != nil {
|
|
t.Fatalf("mkdir external beads failed: %v", err)
|
|
}
|
|
|
|
// Create issues.jsonl in external beads
|
|
jsonlPath := filepath.Join(externalBeadsDir, "issues.jsonl")
|
|
issueContent := `{"id":"ext-1","title":"External Issue","status":"open","issue_type":"task","priority":2,"created_at":"2025-01-01T00:00:00Z","updated_at":"2025-01-01T00:00:00Z"}`
|
|
if err := os.WriteFile(jsonlPath, []byte(issueContent+"\n"), 0644); err != nil {
|
|
t.Fatalf("write external JSONL failed: %v", err)
|
|
}
|
|
|
|
// Commit initial beads files
|
|
_ = exec.Command("git", "-C", externalDir, "add", ".beads").Run()
|
|
_ = exec.Command("git", "-C", externalDir, "commit", "-m", "add beads").Run()
|
|
|
|
// Change back to main repo (simulating user's project)
|
|
if err := os.Chdir(mainDir); err != nil {
|
|
t.Fatalf("chdir to main failed: %v", err)
|
|
}
|
|
|
|
// Test 1: isExternalBeadsDir should detect external repo
|
|
if !isExternalBeadsDir(ctx, externalBeadsDir) {
|
|
t.Error("isExternalBeadsDir should return true for external beads dir")
|
|
}
|
|
t.Log("✓ External beads dir correctly detected")
|
|
|
|
// Test 2: Verify the external beads functions exist and are callable
|
|
// The actual commit test requires more complex setup due to path resolution
|
|
// The key verification is that detection works (Test 1)
|
|
// and the functions are present (verified by compilation)
|
|
|
|
// Test 3: pullFromExternalBeadsRepo should not error (no remote)
|
|
// This tests the function handles no-remote gracefully
|
|
err = pullFromExternalBeadsRepo(ctx, externalBeadsDir)
|
|
if err != nil {
|
|
t.Errorf("pullFromExternalBeadsRepo should handle no-remote: %v", err)
|
|
}
|
|
t.Log("✓ Pull from external beads repo handled no-remote correctly")
|
|
|
|
// Test 4: Verify getRepoRootFromPath works for external dir
|
|
repoRoot, err := getRepoRootFromPath(ctx, externalBeadsDir)
|
|
if err != nil {
|
|
t.Fatalf("getRepoRootFromPath failed: %v", err)
|
|
}
|
|
// Should return the external repo root
|
|
resolvedExternal, _ := filepath.EvalSymlinks(externalDir)
|
|
if repoRoot != resolvedExternal {
|
|
t.Errorf("getRepoRootFromPath = %q, want %q", repoRoot, resolvedExternal)
|
|
}
|
|
t.Logf("✓ getRepoRootFromPath correctly identifies external repo: %s", repoRoot)
|
|
}
|
|
|
|
// TestMergeIssuesWithBaseState verifies the 3-way merge algorithm
|
|
// that underpins pull-first sync works correctly with base state.
|
|
// This is the core algorithm that prevents data loss (#911).
|
|
func TestMergeIssuesWithBaseState(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
baseTime := time.Date(2025, 1, 1, 0, 0, 0, 0, time.UTC)
|
|
localTime := time.Date(2025, 1, 2, 0, 0, 0, 0, time.UTC)
|
|
remoteTime := time.Date(2025, 1, 3, 0, 0, 0, 0, time.UTC)
|
|
|
|
tests := []struct {
|
|
name string
|
|
base []*beads.Issue
|
|
local []*beads.Issue
|
|
remote []*beads.Issue
|
|
wantCount int
|
|
wantConflicts int
|
|
wantStrategy map[string]string
|
|
wantTitles map[string]string // id -> expected title
|
|
}{
|
|
{
|
|
name: "only remote changed",
|
|
base: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Original", UpdatedAt: baseTime},
|
|
},
|
|
local: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Original", UpdatedAt: baseTime},
|
|
},
|
|
remote: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Remote Edit", UpdatedAt: remoteTime},
|
|
},
|
|
wantCount: 1,
|
|
wantConflicts: 0,
|
|
wantStrategy: map[string]string{"bd-1": StrategyRemote},
|
|
wantTitles: map[string]string{"bd-1": "Remote Edit"},
|
|
},
|
|
{
|
|
name: "only local changed",
|
|
base: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Original", UpdatedAt: baseTime},
|
|
},
|
|
local: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Local Edit", UpdatedAt: localTime},
|
|
},
|
|
remote: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Original", UpdatedAt: baseTime},
|
|
},
|
|
wantCount: 1,
|
|
wantConflicts: 0,
|
|
wantStrategy: map[string]string{"bd-1": StrategyLocal},
|
|
wantTitles: map[string]string{"bd-1": "Local Edit"},
|
|
},
|
|
{
|
|
name: "true conflict - remote wins LWW",
|
|
base: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Original", UpdatedAt: baseTime},
|
|
},
|
|
local: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Local Edit", UpdatedAt: localTime},
|
|
},
|
|
remote: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Remote Edit", UpdatedAt: remoteTime},
|
|
},
|
|
wantCount: 1,
|
|
wantConflicts: 1,
|
|
wantStrategy: map[string]string{"bd-1": StrategyMerged},
|
|
wantTitles: map[string]string{"bd-1": "Remote Edit"}, // Remote wins (later timestamp)
|
|
},
|
|
{
|
|
name: "new issue from remote",
|
|
base: []*beads.Issue{},
|
|
local: []*beads.Issue{},
|
|
remote: []*beads.Issue{
|
|
{ID: "bd-1", Title: "New Remote Issue", UpdatedAt: remoteTime},
|
|
},
|
|
wantCount: 1,
|
|
wantConflicts: 0,
|
|
wantStrategy: map[string]string{"bd-1": StrategyRemote},
|
|
wantTitles: map[string]string{"bd-1": "New Remote Issue"},
|
|
},
|
|
{
|
|
name: "new issue from local",
|
|
base: []*beads.Issue{},
|
|
local: []*beads.Issue{
|
|
{ID: "bd-1", Title: "New Local Issue", UpdatedAt: localTime},
|
|
},
|
|
remote: []*beads.Issue{},
|
|
wantCount: 1,
|
|
wantConflicts: 0,
|
|
wantStrategy: map[string]string{"bd-1": StrategyLocal},
|
|
wantTitles: map[string]string{"bd-1": "New Local Issue"},
|
|
},
|
|
{
|
|
name: "both made identical change",
|
|
base: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Original", UpdatedAt: baseTime},
|
|
},
|
|
local: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Same Edit", UpdatedAt: localTime},
|
|
},
|
|
remote: []*beads.Issue{
|
|
{ID: "bd-1", Title: "Same Edit", UpdatedAt: localTime},
|
|
},
|
|
wantCount: 1,
|
|
wantConflicts: 0,
|
|
wantStrategy: map[string]string{"bd-1": StrategySame},
|
|
wantTitles: map[string]string{"bd-1": "Same Edit"},
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
tt := tt
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
result := MergeIssues(tt.base, tt.local, tt.remote)
|
|
|
|
if len(result.Merged) != tt.wantCount {
|
|
t.Errorf("got %d merged issues, want %d", len(result.Merged), tt.wantCount)
|
|
}
|
|
|
|
if result.Conflicts != tt.wantConflicts {
|
|
t.Errorf("got %d conflicts, want %d", result.Conflicts, tt.wantConflicts)
|
|
}
|
|
|
|
for id, wantStrategy := range tt.wantStrategy {
|
|
if result.Strategy[id] != wantStrategy {
|
|
t.Errorf("strategy[%s] = %q, want %q", id, result.Strategy[id], wantStrategy)
|
|
}
|
|
}
|
|
|
|
for _, issue := range result.Merged {
|
|
if wantTitle, ok := tt.wantTitles[issue.ID]; ok {
|
|
if issue.Title != wantTitle {
|
|
t.Errorf("title[%s] = %q, want %q", issue.ID, issue.Title, wantTitle)
|
|
}
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestUpgradeFromOldSync verifies that existing projects safely upgrade to pull-first.
|
|
// When sync_base.jsonl doesn't exist (first sync after upgrade), the merge should:
|
|
// 1. Keep issues that only exist locally
|
|
// 2. Keep issues that only exist remotely
|
|
// 3. Merge issues that exist in both (using LWW for scalars, union for sets)
|
|
// This is critical for production safety.
|
|
func TestUpgradeFromOldSync(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
localTime := time.Date(2025, 1, 2, 0, 0, 0, 0, time.UTC)
|
|
remoteTime := time.Date(2025, 1, 3, 0, 0, 0, 0, time.UTC)
|
|
|
|
// Simulate upgrade scenario: base=nil (no sync_base.jsonl)
|
|
// Local has 2 issues, remote has 2 issues (1 overlap)
|
|
local := []*beads.Issue{
|
|
{ID: "bd-1", Title: "Shared Issue Local", Labels: []string{"local-label"}, UpdatedAt: localTime},
|
|
{ID: "bd-2", Title: "Local Only Issue", UpdatedAt: localTime},
|
|
}
|
|
remote := []*beads.Issue{
|
|
{ID: "bd-1", Title: "Shared Issue Remote", Labels: []string{"remote-label"}, UpdatedAt: remoteTime},
|
|
{ID: "bd-3", Title: "Remote Only Issue", UpdatedAt: remoteTime},
|
|
}
|
|
|
|
// Key: base is nil (simulating upgrade from old sync)
|
|
result := MergeIssues(nil, local, remote)
|
|
|
|
// Should have 3 issues total
|
|
if len(result.Merged) != 3 {
|
|
t.Fatalf("expected 3 merged issues, got %d", len(result.Merged))
|
|
}
|
|
|
|
// Build map for easier assertions
|
|
byID := make(map[string]*beads.Issue)
|
|
for _, issue := range result.Merged {
|
|
byID[issue.ID] = issue
|
|
}
|
|
|
|
// bd-1: Shared issue should be merged (remote wins LWW, labels union)
|
|
if issue, ok := byID["bd-1"]; ok {
|
|
// Remote wins LWW (later timestamp)
|
|
if issue.Title != "Shared Issue Remote" {
|
|
t.Errorf("bd-1 title = %q, want 'Shared Issue Remote' (LWW)", issue.Title)
|
|
}
|
|
// Labels should be union
|
|
if len(issue.Labels) != 2 {
|
|
t.Errorf("bd-1 labels = %v, want union of local and remote labels", issue.Labels)
|
|
}
|
|
if result.Strategy["bd-1"] != StrategyMerged {
|
|
t.Errorf("bd-1 strategy = %q, want %q", result.Strategy["bd-1"], StrategyMerged)
|
|
}
|
|
} else {
|
|
t.Error("bd-1 should exist in merged result")
|
|
}
|
|
|
|
// bd-2: Local only should be kept
|
|
if issue, ok := byID["bd-2"]; ok {
|
|
if issue.Title != "Local Only Issue" {
|
|
t.Errorf("bd-2 title = %q, want 'Local Only Issue'", issue.Title)
|
|
}
|
|
if result.Strategy["bd-2"] != StrategyLocal {
|
|
t.Errorf("bd-2 strategy = %q, want %q", result.Strategy["bd-2"], StrategyLocal)
|
|
}
|
|
} else {
|
|
t.Error("bd-2 should exist in merged result (local only)")
|
|
}
|
|
|
|
// bd-3: Remote only should be kept
|
|
if issue, ok := byID["bd-3"]; ok {
|
|
if issue.Title != "Remote Only Issue" {
|
|
t.Errorf("bd-3 title = %q, want 'Remote Only Issue'", issue.Title)
|
|
}
|
|
if result.Strategy["bd-3"] != StrategyRemote {
|
|
t.Errorf("bd-3 strategy = %q, want %q", result.Strategy["bd-3"], StrategyRemote)
|
|
}
|
|
} else {
|
|
t.Error("bd-3 should exist in merged result (remote only)")
|
|
}
|
|
|
|
t.Log("✓ Upgrade from old sync safely merges all issues")
|
|
}
|
|
|
|
// TestLabelUnionMerge verifies that labels use union merge (no data loss).
|
|
// This is the field-level resolution Steve asked about.
|
|
func TestLabelUnionMerge(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
baseTime := time.Date(2025, 1, 1, 0, 0, 0, 0, time.UTC)
|
|
localTime := time.Date(2025, 1, 2, 0, 0, 0, 0, time.UTC)
|
|
remoteTime := time.Date(2025, 1, 3, 0, 0, 0, 0, time.UTC)
|
|
|
|
base := []*beads.Issue{
|
|
{ID: "bd-1", Title: "Issue", Labels: []string{"bug"}, UpdatedAt: baseTime},
|
|
}
|
|
local := []*beads.Issue{
|
|
{ID: "bd-1", Title: "Issue", Labels: []string{"bug", "local-label"}, UpdatedAt: localTime},
|
|
}
|
|
remote := []*beads.Issue{
|
|
{ID: "bd-1", Title: "Issue", Labels: []string{"bug", "remote-label"}, UpdatedAt: remoteTime},
|
|
}
|
|
|
|
result := MergeIssues(base, local, remote)
|
|
|
|
if len(result.Merged) != 1 {
|
|
t.Fatalf("expected 1 merged issue, got %d", len(result.Merged))
|
|
}
|
|
|
|
// Labels should be union of both: bug, local-label, remote-label
|
|
labels := result.Merged[0].Labels
|
|
expectedLabels := map[string]bool{"bug": true, "local-label": true, "remote-label": true}
|
|
|
|
if len(labels) != 3 {
|
|
t.Errorf("expected 3 labels, got %d: %v", len(labels), labels)
|
|
}
|
|
|
|
for _, label := range labels {
|
|
if !expectedLabels[label] {
|
|
t.Errorf("unexpected label: %s", label)
|
|
}
|
|
}
|
|
|
|
t.Logf("✓ Labels correctly union-merged: %v", labels)
|
|
}
|
|
|
|
// setupBareRemoteWithClones creates a bare repo (simulating GitHub) and two clones
|
|
// for multi-machine E2E testing. Each clone has its own .beads directory for isolation.
|
|
//
|
|
// Returns:
|
|
// - remoteDir: path to bare repo (the "remote")
|
|
// - machineA: path to first clone
|
|
// - machineB: path to second clone
|
|
// - cleanup: function to call in defer
|
|
func setupBareRemoteWithClones(t *testing.T) (remoteDir, machineA, machineB string, cleanup func()) {
|
|
t.Helper()
|
|
|
|
// Create bare repo (acts as "GitHub")
|
|
remoteDir = t.TempDir()
|
|
// Resolve symlinks to avoid macOS /var -> /private/var issues
|
|
remoteDir, _ = filepath.EvalSymlinks(remoteDir)
|
|
cmd := exec.Command("git", "init", "--bare", "-b", "main")
|
|
cmd.Dir = remoteDir
|
|
if err := cmd.Run(); err != nil {
|
|
t.Fatalf("failed to init bare repo: %v", err)
|
|
}
|
|
|
|
// Clone for Machine A
|
|
machineA = t.TempDir()
|
|
machineA, _ = filepath.EvalSymlinks(machineA)
|
|
if err := exec.Command("git", "clone", remoteDir, machineA).Run(); err != nil {
|
|
t.Fatalf("failed to clone for machineA: %v", err)
|
|
}
|
|
// Configure git user in Machine A
|
|
_ = exec.Command("git", "-C", machineA, "config", "user.email", "machineA@test.com").Run()
|
|
_ = exec.Command("git", "-C", machineA, "config", "user.name", "Machine A").Run()
|
|
|
|
// Clone for Machine B
|
|
machineB = t.TempDir()
|
|
machineB, _ = filepath.EvalSymlinks(machineB)
|
|
if err := exec.Command("git", "clone", remoteDir, machineB).Run(); err != nil {
|
|
t.Fatalf("failed to clone for machineB: %v", err)
|
|
}
|
|
// Configure git user in Machine B
|
|
_ = exec.Command("git", "-C", machineB, "config", "user.email", "machineB@test.com").Run()
|
|
_ = exec.Command("git", "-C", machineB, "config", "user.name", "Machine B").Run()
|
|
|
|
// Initial commit from Machine A (bare repos need at least one commit)
|
|
readmePath := filepath.Join(machineA, "README.md")
|
|
if err := os.WriteFile(readmePath, []byte("# Test Repo\n"), 0644); err != nil {
|
|
t.Fatalf("failed to write README: %v", err)
|
|
}
|
|
_ = exec.Command("git", "-C", machineA, "add", ".").Run()
|
|
if err := exec.Command("git", "-C", machineA, "commit", "-m", "initial").Run(); err != nil {
|
|
t.Fatalf("failed to create initial commit: %v", err)
|
|
}
|
|
if err := exec.Command("git", "-C", machineA, "push", "-u", "origin", "main").Run(); err != nil {
|
|
t.Fatalf("failed to push initial commit: %v", err)
|
|
}
|
|
|
|
// Machine B fetches and checks out main
|
|
_ = exec.Command("git", "-C", machineB, "fetch", "origin").Run()
|
|
_ = exec.Command("git", "-C", machineB, "checkout", "main").Run()
|
|
|
|
cleanup = func() {
|
|
git.ResetCaches() // Prevent cache pollution between tests
|
|
}
|
|
|
|
return remoteDir, machineA, machineB, cleanup
|
|
}
|
|
|
|
// withBeadsDir runs a function with BEADS_DIR set to the specified directory's .beads subdirectory.
|
|
// This provides database isolation for multi-machine tests.
|
|
func withBeadsDir(t *testing.T, dir string, fn func()) {
|
|
t.Helper()
|
|
|
|
origBeadsDir := os.Getenv("BEADS_DIR")
|
|
beadsDir := filepath.Join(dir, ".beads")
|
|
|
|
// Create .beads directory if it doesn't exist
|
|
if err := os.MkdirAll(beadsDir, 0755); err != nil {
|
|
t.Fatalf("failed to create .beads dir: %v", err)
|
|
}
|
|
|
|
os.Setenv("BEADS_DIR", beadsDir)
|
|
defer func() {
|
|
if origBeadsDir != "" {
|
|
os.Setenv("BEADS_DIR", origBeadsDir)
|
|
} else {
|
|
os.Unsetenv("BEADS_DIR")
|
|
}
|
|
}()
|
|
|
|
fn()
|
|
}
|
|
|
|
// TestSyncBranchE2E tests the full sync-branch flow with concurrent changes from
|
|
// two machines using a real bare repo. This is an end-to-end regression test for PR#918.
|
|
//
|
|
// Flow:
|
|
// 1. Machine A creates bd-1, commits to sync branch, pushes to bare remote
|
|
// 2. Machine B creates bd-2, commits to sync branch, pushes to bare remote
|
|
// 3. Machine A pulls from sync branch - should merge both issues
|
|
// 4. Verify both issues present after merge
|
|
func TestSyncBranchE2E(t *testing.T) {
|
|
ctx := context.Background()
|
|
|
|
// Setup: Create bare remote with two clones
|
|
_, machineA, machineB, cleanup := setupBareRemoteWithClones(t)
|
|
defer cleanup()
|
|
|
|
syncBranch := "beads-sync"
|
|
|
|
// Machine A: Create .beads directory and bd-1
|
|
beadsDirA := filepath.Join(machineA, ".beads")
|
|
if err := os.MkdirAll(beadsDirA, 0755); err != nil {
|
|
t.Fatalf("failed to create .beads dir for A: %v", err)
|
|
}
|
|
jsonlPathA := filepath.Join(beadsDirA, "issues.jsonl")
|
|
issue1 := `{"id":"bd-1","title":"Issue from Machine A","status":"open","issue_type":"task","priority":2,"created_at":"2025-01-01T00:00:00Z","updated_at":"2025-01-01T00:00:00Z"}`
|
|
if err := os.WriteFile(jsonlPathA, []byte(issue1+"\n"), 0644); err != nil {
|
|
t.Fatalf("write JSONL failed for A: %v", err)
|
|
}
|
|
|
|
// Machine A: Commit to sync branch using the worktree-based API (push=true)
|
|
withBeadsDir(t, machineA, func() {
|
|
commitResult, err := syncbranch.CommitToSyncBranch(ctx, machineA, syncBranch, jsonlPathA, true)
|
|
if err != nil {
|
|
t.Fatalf("CommitToSyncBranch failed for A: %v", err)
|
|
}
|
|
if !commitResult.Committed {
|
|
t.Fatal("expected commit to succeed for Machine A's issue")
|
|
}
|
|
t.Log("Machine A committed and pushed bd-1 to sync branch")
|
|
})
|
|
|
|
// Machine B: Create .beads directory and bd-2
|
|
beadsDirB := filepath.Join(machineB, ".beads")
|
|
if err := os.MkdirAll(beadsDirB, 0755); err != nil {
|
|
t.Fatalf("failed to create .beads dir for B: %v", err)
|
|
}
|
|
jsonlPathB := filepath.Join(beadsDirB, "issues.jsonl")
|
|
issue2 := `{"id":"bd-2","title":"Issue from Machine B","status":"open","issue_type":"task","priority":2,"created_at":"2025-01-02T00:00:00Z","updated_at":"2025-01-02T00:00:00Z"}`
|
|
if err := os.WriteFile(jsonlPathB, []byte(issue2+"\n"), 0644); err != nil {
|
|
t.Fatalf("write JSONL failed for B: %v", err)
|
|
}
|
|
|
|
// Machine B: Pull first to get A's changes, then commit and push
|
|
withBeadsDir(t, machineB, func() {
|
|
// Pull from remote first (gets Machine A's bd-1)
|
|
pullResult, err := syncbranch.PullFromSyncBranch(ctx, machineB, syncBranch, jsonlPathB, false)
|
|
if err != nil {
|
|
t.Logf("Initial pull for B returned error (may be expected): %v", err)
|
|
}
|
|
if pullResult != nil && pullResult.Pulled {
|
|
t.Log("Machine B pulled existing sync branch content")
|
|
}
|
|
|
|
// Re-read and append bd-2 to maintain bd-1 from pull
|
|
existingContent, _ := os.ReadFile(jsonlPathB)
|
|
if !strings.Contains(string(existingContent), "bd-2") {
|
|
// Append bd-2 if not already present
|
|
if len(existingContent) > 0 && !strings.HasSuffix(string(existingContent), "\n") {
|
|
existingContent = append(existingContent, '\n')
|
|
}
|
|
newContent := string(existingContent) + issue2 + "\n"
|
|
if err := os.WriteFile(jsonlPathB, []byte(newContent), 0644); err != nil {
|
|
t.Fatalf("failed to append bd-2: %v", err)
|
|
}
|
|
}
|
|
|
|
// Commit and push bd-2
|
|
commitResult, err := syncbranch.CommitToSyncBranch(ctx, machineB, syncBranch, jsonlPathB, true)
|
|
if err != nil {
|
|
t.Fatalf("CommitToSyncBranch failed for B: %v", err)
|
|
}
|
|
if !commitResult.Committed {
|
|
t.Log("Machine B had no new changes to commit (bd-2 may already be in sync)")
|
|
} else {
|
|
t.Log("Machine B committed and pushed bd-2 to sync branch")
|
|
}
|
|
})
|
|
|
|
// Machine A: Pull from sync branch - should merge both issues
|
|
withBeadsDir(t, machineA, func() {
|
|
pullResult, err := syncbranch.PullFromSyncBranch(ctx, machineA, syncBranch, jsonlPathA, false)
|
|
if err != nil {
|
|
t.Logf("PullFromSyncBranch for A returned error (may be expected): %v", err)
|
|
}
|
|
if pullResult != nil {
|
|
t.Logf("Pull result for A: Pulled=%v, Merged=%v, FastForwarded=%v",
|
|
pullResult.Pulled, pullResult.Merged, pullResult.FastForwarded)
|
|
}
|
|
})
|
|
|
|
// Verify: Both issues should be present in Machine A's JSONL after merge
|
|
content, err := os.ReadFile(jsonlPathA)
|
|
if err != nil {
|
|
t.Fatalf("failed to read merged JSONL: %v", err)
|
|
}
|
|
|
|
contentStr := string(content)
|
|
hasIssue1 := strings.Contains(contentStr, "bd-1") || strings.Contains(contentStr, "Machine A")
|
|
hasIssue2 := strings.Contains(contentStr, "bd-2") || strings.Contains(contentStr, "Machine B")
|
|
|
|
if hasIssue1 {
|
|
t.Log("Issue bd-1 from Machine A preserved")
|
|
} else {
|
|
t.Error("FAIL: bd-1 from Machine A missing after merge")
|
|
}
|
|
|
|
if hasIssue2 {
|
|
t.Log("Issue bd-2 from Machine B merged")
|
|
} else {
|
|
t.Error("FAIL: bd-2 from Machine B missing after merge")
|
|
}
|
|
|
|
if hasIssue1 && hasIssue2 {
|
|
t.Log("Sync-branch E2E test PASSED: both issues present after merge")
|
|
}
|
|
}
|
|
|
|
// TestExportOnlySync tests the --no-pull mode (export-only sync).
|
|
// This mode skips pulling from remote and only exports local changes.
|
|
//
|
|
// Use case: "I just want to push my local changes, don't merge anything"
|
|
//
|
|
// Flow:
|
|
// 1. Create local issue in database
|
|
// 2. Run export-only sync (doExportOnlySync)
|
|
// 3. Verify issue is exported to JSONL
|
|
// 4. Verify changes are committed
|
|
func TestExportOnlySync(t *testing.T) {
|
|
ctx := context.Background()
|
|
tmpDir, cleanup := setupGitRepo(t)
|
|
defer cleanup()
|
|
|
|
beadsDir := filepath.Join(tmpDir, ".beads")
|
|
jsonlPath := filepath.Join(beadsDir, "issues.jsonl")
|
|
|
|
// Remove pre-existing issues.jsonl from setupGitRepo (we want a clean slate)
|
|
_ = os.Remove(jsonlPath)
|
|
|
|
// Setup: Create .beads directory
|
|
if err := os.MkdirAll(beadsDir, 0755); err != nil {
|
|
t.Fatalf("failed to create .beads dir: %v", err)
|
|
}
|
|
|
|
// Create a database with a test issue
|
|
dbPath := filepath.Join(beadsDir, "beads.db")
|
|
testStore, err := sqlite.New(ctx, dbPath)
|
|
if err != nil {
|
|
t.Fatalf("failed to create store: %v", err)
|
|
}
|
|
|
|
// Set issue prefix (required for export)
|
|
if err := testStore.SetConfig(ctx, "issue_prefix", "export-test"); err != nil {
|
|
t.Fatalf("failed to set issue_prefix: %v", err)
|
|
}
|
|
|
|
// Create a test issue in the database
|
|
testIssue := &types.Issue{
|
|
ID: "export-test-1",
|
|
Title: "Export Only Test Issue",
|
|
Status: "open",
|
|
IssueType: "task",
|
|
Priority: 2,
|
|
CreatedAt: time.Now(),
|
|
UpdatedAt: time.Now(),
|
|
}
|
|
if err := testStore.CreateIssue(ctx, testIssue, "test"); err != nil {
|
|
t.Fatalf("failed to create test issue: %v", err)
|
|
}
|
|
testStore.Close()
|
|
t.Log("✓ Created test issue in database")
|
|
|
|
// Initialize the global store for doExportOnlySync
|
|
// This simulates what `bd sync --no-pull` does
|
|
store, err = sqlite.New(ctx, dbPath)
|
|
if err != nil {
|
|
t.Fatalf("failed to open store: %v", err)
|
|
}
|
|
defer func() {
|
|
store.Close()
|
|
store = nil
|
|
}()
|
|
|
|
// Run export-only sync (--no-pull mode)
|
|
// noPush=true to avoid needing a real remote in tests
|
|
if err := doExportOnlySync(ctx, jsonlPath, true, "bd sync: export test"); err != nil {
|
|
t.Fatalf("doExportOnlySync failed: %v", err)
|
|
}
|
|
t.Log("✓ Export-only sync completed")
|
|
|
|
// Verify: JSONL file should exist with our issue
|
|
content, err := os.ReadFile(jsonlPath)
|
|
if err != nil {
|
|
t.Fatalf("failed to read JSONL: %v", err)
|
|
}
|
|
|
|
contentStr := string(content)
|
|
if !strings.Contains(contentStr, "export-test-1") {
|
|
t.Errorf("JSONL should contain issue ID 'export-test-1', got: %s", contentStr)
|
|
}
|
|
if !strings.Contains(contentStr, "Export Only Test Issue") {
|
|
t.Errorf("JSONL should contain issue title, got: %s", contentStr)
|
|
}
|
|
t.Log("✓ Issue correctly exported to JSONL")
|
|
|
|
// Verify: Changes should be committed
|
|
output, err := exec.Command("git", "log", "-1", "--pretty=format:%s").Output()
|
|
if err != nil {
|
|
t.Fatalf("git log failed: %v", err)
|
|
}
|
|
commitMsg := string(output)
|
|
if !strings.Contains(commitMsg, "bd sync") {
|
|
t.Errorf("expected commit message to contain 'bd sync', got: %s", commitMsg)
|
|
}
|
|
t.Log("✓ Changes committed with correct message")
|
|
|
|
// Verify: issues.jsonl should be tracked and committed (no modifications)
|
|
// Note: Database files (.db, .db-wal, .db-shm) and .sync.lock remain untracked
|
|
// as expected - only JSONL is committed to git
|
|
status, err := exec.Command("git", "status", "--porcelain", jsonlPath).Output()
|
|
if err != nil {
|
|
t.Fatalf("git status failed: %v", err)
|
|
}
|
|
if len(status) > 0 {
|
|
t.Errorf("expected issues.jsonl to be committed, got: %s", status)
|
|
}
|
|
t.Log("✓ issues.jsonl is committed after export-only sync")
|
|
}
|