Files
beads/cmd/bd/sync_modes_test.go
Peter Chanthamynavong 2cbf3a5497 fix: Validate sync-branch at config-time and runtime (closes #1166) (#1168)
* fix(config): validate sync-branch at config time

Add sync-branch validation to validateYamlConfigValue() to reject
main/master at config time, preventing the validation bypass in GH#1166.

- Add case for sync-branch and sync.branch keys
- Inline validation logic to avoid import cycle (config <-> syncbranch)
- Add unit tests for rejection (main/master) and acceptance (valid names)

Part of: #1166

* fix(sync): add runtime guard for sync-branch == current-branch

Add dynamic runtime check before worktree operations to catch cases
where sync-branch matches the current branch. This provides defense
in depth for manual YAML edits, pre-fix configs, or non-main/master
branch names (trunk, develop, production, etc.).

- Check IsSyncBranchSameAsCurrent() after hasSyncBranchConfig is set
- Position check BEFORE worktree entry (CWD changes inside worktree)
- Add integration test TestSync_FailsWhenOnSyncBranch

Part of: #1166

* docs: note main/master restriction in sync-branch FAQ

Clarifies that git worktrees cannot checkout the same branch in
multiple locations, so main/master cannot be used as sync branch.
2026-01-19 10:11:06 -08:00

855 lines
30 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")
}
// TestSync_FailsWhenOnSyncBranch verifies that bd sync fails gracefully
// when the current branch matches the configured sync branch.
// This is the runtime guard for GH#1166 - prevents bd sync from attempting
// to create a worktree for a branch that's already checked out.
//
// The issue: If sync.branch = "main" and user is on "main", bd sync would
// try to create a worktree for "main" which fails because it's already checked out.
// Worse, some code paths could commit all staged files instead of just .beads/.
func TestSync_FailsWhenOnSyncBranch(t *testing.T) {
ctx := context.Background()
tmpDir, cleanup := setupGitRepo(t)
defer cleanup()
// Get current branch name (should be "main" from setupGitRepo)
currentBranch, err := syncbranch.GetCurrentBranch(ctx)
if err != nil {
t.Fatalf("failed to get current branch: %v", err)
}
t.Logf("Current branch: %s", currentBranch)
// Test 1: IsSyncBranchSameAsCurrent returns true when branch matches
if !syncbranch.IsSyncBranchSameAsCurrent(ctx, currentBranch) {
t.Error("IsSyncBranchSameAsCurrent should return true when sync branch matches current branch")
}
t.Log("✓ IsSyncBranchSameAsCurrent correctly detects same-branch condition")
// Test 2: IsSyncBranchSameAsCurrent returns false for different branch
if syncbranch.IsSyncBranchSameAsCurrent(ctx, "beads-sync") {
t.Error("IsSyncBranchSameAsCurrent should return false for different branch name")
}
t.Log("✓ IsSyncBranchSameAsCurrent correctly allows different branch names")
// Test 3: Setup sync.branch config to match current branch (the problematic state)
beadsDir := filepath.Join(tmpDir, ".beads")
if err := os.MkdirAll(beadsDir, 0755); err != nil {
t.Fatalf("failed to create .beads dir: %v", err)
}
dbPath := filepath.Join(beadsDir, "beads.db")
testStore, err := sqlite.New(ctx, dbPath)
if err != nil {
t.Fatalf("failed to create store: %v", err)
}
defer testStore.Close()
// Configure sync.branch to match current branch (this is the bug condition)
if err := testStore.SetConfig(ctx, "sync.branch", currentBranch); err != nil {
t.Fatalf("failed to set sync.branch: %v", err)
}
t.Logf("✓ Configured sync.branch = %s (matches current branch)", currentBranch)
// Verify the config is stored correctly
syncBranchValue, err := testStore.GetConfig(ctx, "sync.branch")
if err != nil {
t.Fatalf("failed to get sync.branch config: %v", err)
}
if syncBranchValue != currentBranch {
t.Errorf("sync.branch config = %q, want %q", syncBranchValue, currentBranch)
}
// Test 4: The runtime guard logic (same as in sync.go)
// This simulates what happens in the sync command
syncBranchName := syncBranchValue
hasSyncBranchConfig := syncBranchName != ""
if hasSyncBranchConfig {
if syncbranch.IsSyncBranchSameAsCurrent(ctx, syncBranchName) {
// This is the expected behavior - we caught the misconfiguration
t.Log("✓ Runtime guard correctly detected sync-branch = current-branch condition")
} else {
t.Error("FAIL: Runtime guard did not detect sync-branch = current-branch condition")
}
} else {
t.Error("FAIL: sync.branch config should be set")
}
t.Log("✓ TestSync_FailsWhenOnSyncBranch PASSED")
}