Files
Charon/backend/internal/testutil/db.go
akanealw eec8c28fb3
Some checks failed
Go Benchmark / Performance Regression Check (push) Has been cancelled
Cerberus Integration / Cerberus Security Stack Integration (push) Has been cancelled
Upload Coverage to Codecov / Backend Codecov Upload (push) Has been cancelled
Upload Coverage to Codecov / Frontend Codecov Upload (push) Has been cancelled
CodeQL - Analyze / CodeQL analysis (go) (push) Has been cancelled
CodeQL - Analyze / CodeQL analysis (javascript-typescript) (push) Has been cancelled
CrowdSec Integration / CrowdSec Bouncer Integration (push) Has been cancelled
Docker Build, Publish & Test / build-and-push (push) Has been cancelled
Quality Checks / Auth Route Protection Contract (push) Has been cancelled
Quality Checks / Codecov Trigger/Comment Parity Guard (push) Has been cancelled
Quality Checks / Backend (Go) (push) Has been cancelled
Quality Checks / Frontend (React) (push) Has been cancelled
Rate Limit integration / Rate Limiting Integration (push) Has been cancelled
Security Scan (PR) / Trivy Binary Scan (push) Has been cancelled
Supply Chain Verification (PR) / Verify Supply Chain (push) Has been cancelled
WAF integration / Coraza WAF Integration (push) Has been cancelled
Docker Build, Publish & Test / Security Scan PR Image (push) Has been cancelled
Repo Health Check / Repo health (push) Has been cancelled
History Rewrite Dry-Run / Dry-run preview for history rewrite (push) Has been cancelled
Prune Renovate Branches / prune (push) Has been cancelled
Renovate / renovate (push) Has been cancelled
Nightly Build & Package / sync-development-to-nightly (push) Has been cancelled
Nightly Build & Package / Trigger Nightly Validation Workflows (push) Has been cancelled
Nightly Build & Package / build-and-push-nightly (push) Has been cancelled
Nightly Build & Package / test-nightly-image (push) Has been cancelled
Nightly Build & Package / verify-nightly-supply-chain (push) Has been cancelled
Update GeoLite2 Checksum / update-checksum (push) Has been cancelled
Container Registry Prune / prune-ghcr (push) Has been cancelled
Container Registry Prune / prune-dockerhub (push) Has been cancelled
Container Registry Prune / summarize (push) Has been cancelled
Supply Chain Verification / Verify SBOM (push) Has been cancelled
Supply Chain Verification / Verify Release Artifacts (push) Has been cancelled
Supply Chain Verification / Verify Docker Image Supply Chain (push) Has been cancelled
Monitor Caddy Major Release / check-caddy-major (push) Has been cancelled
Weekly Nightly to Main Promotion / Verify Nightly Branch Health (push) Has been cancelled
Weekly Nightly to Main Promotion / Create Promotion PR (push) Has been cancelled
Weekly Nightly to Main Promotion / Trigger Missing Required Checks (push) Has been cancelled
Weekly Nightly to Main Promotion / Notify on Failure (push) Has been cancelled
Weekly Nightly to Main Promotion / Workflow Summary (push) Has been cancelled
changed perms
2026-04-22 18:19:14 +00:00

89 lines
2.7 KiB
Go
Executable File

package testutil
import (
"testing"
"gorm.io/gorm"
)
// WithTx runs a test function within a transaction that is always rolled back.
// This provides test isolation without the overhead of creating new databases.
//
// Usage Example:
//
// func TestSomething(t *testing.T) {
// sharedDB := setupSharedDB(t) // Create once per package
// testutil.WithTx(t, sharedDB, func(tx *gorm.DB) {
// // Use tx for all DB operations in this test
// tx.Create(&models.User{Name: "test"})
// // Transaction automatically rolled back at end
// })
// }
func WithTx(t *testing.T, db *gorm.DB, fn func(tx *gorm.DB)) {
t.Helper()
tx := db.Begin()
defer func() {
if r := recover(); r != nil {
tx.Rollback()
panic(r)
}
tx.Rollback()
}()
fn(tx)
}
// GetTestTx returns a transaction that will be rolled back when the test completes.
// This is useful for tests that need to pass the transaction to multiple functions.
//
// Usage Example:
//
// func TestSomething(t *testing.T) {
// t.Parallel() // Safe to run in parallel with transaction isolation
// sharedDB := getSharedDB(t)
// tx := testutil.GetTestTx(t, sharedDB)
// // Use tx for all DB operations
// tx.Create(&models.User{Name: "test"})
// // Transaction automatically rolled back via t.Cleanup()
// }
//
// Note: When using GetTestTx with t.Parallel(), ensure the shared DB is safe for
// concurrent access (e.g., using ?cache=shared for SQLite).
func GetTestTx(t *testing.T, db *gorm.DB) *gorm.DB {
t.Helper()
tx := db.Begin()
t.Cleanup(func() {
tx.Rollback()
})
return tx
}
// Best Practices for Transaction-Based Testing:
//
// 1. Create a shared DB once per test package (not per test):
// var sharedDB *gorm.DB
// func init() {
// db, _ := gorm.Open(sqlite.Open("file::memory:?cache=shared"), &gorm.Config{})
// db.AutoMigrate(&models.User{}, &models.Setting{})
// sharedDB = db
// }
//
// 2. Use transactions for test isolation:
// func TestUser(t *testing.T) {
// t.Parallel()
// tx := testutil.GetTestTx(t, sharedDB)
// // Test operations using tx
// }
//
// 3. When NOT to use transaction rollbacks:
// - Tests that need specific DB schemas per test
// - Tests that intentionally test transaction behavior
// - Tests that require nil DB values
// - Tests using in-memory :memory: (already fast enough)
// - Complex tests with custom setup/teardown logic
//
// 4. Benefits of transaction rollbacks:
// - Faster than creating new databases (especially for disk-based DBs)
// - Automatic cleanup (no manual teardown needed)
// - Enables safe use of t.Parallel() for concurrent test execution
// - Reduces disk I/O and memory usage in CI environments