Analysis shows main_test.go is NOT a good candidate for shared DB pattern due to global state manipulation and integration test characteristics. Changes: - Added MAIN_TEST_REFACTOR_NOTES.md documenting findings - Fixed unused import in duplicates_test.go (from recent pull) Key findings: - 18 tests with 14 newTestStore() calls - Tests manipulate global state (autoFlushEnabled, isDirty, etc.) - Tests simulate workflows (flush, import) not just CRUD - Shared DB causes deadlocks between flush ops and cleanup - Integration tests need process-level isolation Recommendation: Leave as-is or use Option 2 (grouped tests without shared DB). Focus P2 efforts on integrity_test.go instead. 🤖 Generated with Claude Code Co-Authored-By: Claude <noreply@anthropic.com>
5.5 KiB
main_test.go Refactoring Notes (bd-1rh follow-up)
Status: DEFERRED - Needs Different Approach
Summary
Attempted to refactor main_test.go (18 tests, 14 newTestStore() calls) to use shared DB pattern like P1 files. Discovered fundamental incompatibility with shared DB approach due to global state manipulation and integration test characteristics.
What We Tried
- Created
TestAutoFlushSuiteandTestAutoImportSuitewith shared DB - Converted 18 individual tests to subtests
- Reduced from 14 DB setups to 2
Problems Encountered
1. Deadlock Issue
- Tests call
flushToJSONL()which accesses the database - Test cleanup (from
newTestStore()) tries to close the database - Results in database lock contention and test timeouts
- Stack trace shows:
database/sql.(*DB).Close()waiting whileflushToJSONL()is accessing DB
2. Global State Manipulation
These tests heavily manipulate package-level globals:
autoFlushEnabled,isDirty,flushTimerstore,storeActive,storeMutexdbPath(used to compute JSONL path dynamically)flushFailureCount,lastFlushError
3. Integration Test Characteristics
- Tests simulate end-to-end flush/import workflows
- Tests capture stderr to verify error messages
- Tests manipulate filesystem state directly
- Tests create directories to force error conditions
Key Differences from P1 Tests
| Aspect | P1 Tests (create, dep, etc.) | main_test.go |
|---|---|---|
| DB Usage | Pure DB operations | Global state + DB + filesystem |
| Isolation | Data-level only | Requires process-level isolation |
| Cleanup | Simple | Complex (timers, goroutines, mutexes) |
| Pattern | CRUD operations | Workflow simulation |
Why Shared DB Doesn't Work
-
jsonlPathis computed dynamically fromdbPathviafindJSONLPath()- Not a global variable like in old tests
- Changes to
dbPathaffect where JSONL files are written/read
-
Tests need to control exact JSONL paths for:
- Creating files to force errors (making JSONL a directory)
- Verifying files were/weren't created
- Touching files to simulate git pull scenarios
-
Concurrent access issues:
- Background flush operations may trigger during test cleanup
- Global mutexes protect state but cause deadlocks with shared DB
What Tests Actually Do
Auto-Flush Tests (9 tests)
- Test global state flags (
isDirty,autoFlushEnabled) - Test timer management (
flushTimer) - Test concurrency (goroutines calling
markDirtyAndScheduleFlush()) - Simulate program exit (PersistentPostRun behavior)
- Force error conditions by making JSONL path a directory
Auto-Import Tests (9 tests)
- Test JSONL -> DB sync when JSONL is newer
- Test merge conflict detection (literal
<<<<<<<markers in file) - Test JSON-encoded conflict markers (false positive prevention)
- Test status transition invariants (closed_at management)
- Manipulate file timestamps with
os.Chtimes()
Recommended Approach
Option 1: Leave As-Is (RECOMMENDED)
Rationale: These are integration tests, not unit tests. The overhead of 14 DB setups is acceptable for:
- Tests that manipulate global state
- Tests that simulate complex workflows
- Tests that are relatively fast already (~0.5s each)
Expected speedup: Minimal (2-3x at most) vs. complexity cost
Option 2: Refactor Without Shared DB
Changes:
- Keep individual test functions (not suite)
- Reduce DB setups by reusing test stores within related test groups
- Add helpers to reset global state between tests
- Document which tests can share vs. need isolation
Example:
func TestAutoFlushGroup(t *testing.T) {
tmpDir := t.TempDir()
testDB := filepath.Join(tmpDir, "test.db")
testStore := newTestStore(t, testDB)
// Helper to reset state
resetState := func() {
autoFlushEnabled = true
isDirty = false
if flushTimer != nil {
flushTimer.Stop()
flushTimer = nil
}
}
t.Run("DirtyMarking", func(t *testing.T) {
resetState()
// test...
})
t.Run("Disabled", func(t *testing.T) {
resetState()
// test...
})
}
Option 3: Mock/Stub Approach
Changes:
- Introduce interfaces for
flushToJSONLandautoImportIfNewer - Mock the filesystem operations
- Test state transitions without actual DB/filesystem
Trade-offs: More refactoring, loses integration test value
Files Modified (Reverted)
cmd/bd/main_test.go- Reverted to originalcmd/bd/duplicates_test.go- Fixed unused import (kept fix)
Lessons Learned
-
Not all tests benefit from shared DB pattern
- Integration tests need isolation
- Global state manipulation requires careful handling
-
P1 test pattern assumes:
- Pure DB operations
- No global state
- Data-level isolation sufficient
-
Test classification matters:
- Unit tests: Share DB ✓
- Integration tests: Need isolation ✓
- Workflow tests: Need full process isolation ✓
Next Steps
- Document in TEST_SUITE_AUDIT.md that main_test.go is P2 but NOT a good candidate for shared DB pattern
- Update audit classification: Move main_test.go to "Special Cases" category
- Focus P2 efforts on
integrity_test.goandexport_import_test.goinstead
References
- Original issue: bd-1rh (Phase 2 test suite optimization)
- Pattern source:
label_test.go, P1 refactored files - Related: bd-159 (test config issues), bd-270 (merge conflict detection)