Add async gates - coordination primitives for agents to wait on external events like CI completion, PR merges, timers, or human approval. Changes: - Add 'gate' issue type to types.go with gate-specific fields: - AwaitType: condition type (gh:run, gh:pr, timer, human, mail) - AwaitID: condition identifier - Timeout: max wait duration - Waiters: mail addresses to notify when gate clears - Add SQLite migration 027_gate_columns for new fields - Update all SQLite storage queries to handle gate fields - Add bd gate commands: create, show, list, close, wait - All commands support --json output and --no-daemon mode Closes: bd-2v0f, bd-lz49, bd-u66e 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
159 lines
7.5 KiB
Go
159 lines
7.5 KiB
Go
// Package sqlite - database migrations
|
|
package sqlite
|
|
|
|
import (
|
|
"database/sql"
|
|
"fmt"
|
|
|
|
"github.com/steveyegge/beads/internal/storage/sqlite/migrations"
|
|
)
|
|
|
|
// Migration represents a single database migration
|
|
type Migration struct {
|
|
Name string
|
|
Func func(*sql.DB) error
|
|
}
|
|
|
|
// migrations is the ordered list of all migrations to run
|
|
// Migrations are run in order during database initialization
|
|
var migrationsList = []Migration{
|
|
{"dirty_issues_table", migrations.MigrateDirtyIssuesTable},
|
|
{"external_ref_column", migrations.MigrateExternalRefColumn},
|
|
{"composite_indexes", migrations.MigrateCompositeIndexes},
|
|
{"closed_at_constraint", migrations.MigrateClosedAtConstraint},
|
|
{"compaction_columns", migrations.MigrateCompactionColumns},
|
|
{"snapshots_table", migrations.MigrateSnapshotsTable},
|
|
{"compaction_config", migrations.MigrateCompactionConfig},
|
|
{"compacted_at_commit_column", migrations.MigrateCompactedAtCommitColumn},
|
|
{"export_hashes_table", migrations.MigrateExportHashesTable},
|
|
{"content_hash_column", migrations.MigrateContentHashColumn},
|
|
{"external_ref_unique", migrations.MigrateExternalRefUnique},
|
|
{"source_repo_column", migrations.MigrateSourceRepoColumn},
|
|
{"repo_mtimes_table", migrations.MigrateRepoMtimesTable},
|
|
{"child_counters_table", migrations.MigrateChildCountersTable},
|
|
{"blocked_issues_cache", migrations.MigrateBlockedIssuesCache},
|
|
{"orphan_detection", migrations.MigrateOrphanDetection},
|
|
{"close_reason_column", migrations.MigrateCloseReasonColumn},
|
|
{"tombstone_columns", migrations.MigrateTombstoneColumns},
|
|
{"messaging_fields", migrations.MigrateMessagingFields},
|
|
{"edge_consolidation", migrations.MigrateEdgeConsolidation},
|
|
{"migrate_edge_fields", migrations.MigrateEdgeFields},
|
|
{"drop_edge_columns", migrations.MigrateDropEdgeColumns},
|
|
{"pinned_column", migrations.MigratePinnedColumn},
|
|
{"is_template_column", migrations.MigrateIsTemplateColumn},
|
|
{"remove_depends_on_fk", migrations.MigrateRemoveDependsOnFK},
|
|
{"additional_indexes", migrations.MigrateAdditionalIndexes},
|
|
{"gate_columns", migrations.MigrateGateColumns},
|
|
}
|
|
|
|
// MigrationInfo contains metadata about a migration for inspection
|
|
type MigrationInfo struct {
|
|
Name string `json:"name"`
|
|
Description string `json:"description"`
|
|
}
|
|
|
|
// ListMigrations returns list of all registered migrations with descriptions
|
|
// Note: This returns ALL registered migrations, not just pending ones (all are idempotent)
|
|
func ListMigrations() []MigrationInfo {
|
|
result := make([]MigrationInfo, len(migrationsList))
|
|
for i, m := range migrationsList {
|
|
result[i] = MigrationInfo{
|
|
Name: m.Name,
|
|
Description: getMigrationDescription(m.Name),
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
// getMigrationDescription returns a human-readable description for a migration
|
|
func getMigrationDescription(name string) string {
|
|
descriptions := map[string]string{
|
|
"dirty_issues_table": "Adds dirty_issues table for auto-export tracking",
|
|
"external_ref_column": "Adds external_ref column to issues table",
|
|
"composite_indexes": "Adds composite indexes for better query performance",
|
|
"closed_at_constraint": "Adds constraint ensuring closed issues have closed_at timestamp",
|
|
"compaction_columns": "Adds compaction tracking columns (compacted_at, compacted_at_commit)",
|
|
"snapshots_table": "Adds snapshots table for issue history",
|
|
"compaction_config": "Adds config entries for compaction",
|
|
"compacted_at_commit_column": "Adds compacted_at_commit to snapshots table",
|
|
"export_hashes_table": "Adds export_hashes table for idempotent exports",
|
|
"content_hash_column": "Adds content_hash column for collision resolution",
|
|
"external_ref_unique": "Adds UNIQUE constraint on external_ref column",
|
|
"source_repo_column": "Adds source_repo column for multi-repo support",
|
|
"repo_mtimes_table": "Adds repo_mtimes table for multi-repo hydration caching",
|
|
"child_counters_table": "Adds child_counters table for hierarchical ID generation with ON DELETE CASCADE",
|
|
"blocked_issues_cache": "Adds blocked_issues_cache table for GetReadyWork performance optimization (bd-5qim)",
|
|
"orphan_detection": "Detects orphaned child issues and logs them for user action (bd-3852)",
|
|
"close_reason_column": "Adds close_reason column to issues table for storing closure explanations (bd-uyu)",
|
|
"tombstone_columns": "Adds tombstone columns (deleted_at, deleted_by, delete_reason, original_type) for inline soft-delete (bd-vw8)",
|
|
"messaging_fields": "Adds messaging fields (sender, ephemeral, replies_to, relates_to, duplicate_of, superseded_by) for inter-agent communication (bd-kwro)",
|
|
"edge_consolidation": "Adds metadata and thread_id columns to dependencies table for edge schema consolidation (Decision 004)",
|
|
"migrate_edge_fields": "Migrates existing issue fields (replies_to, relates_to, duplicate_of, superseded_by) to dependency edges (Decision 004 Phase 3)",
|
|
"drop_edge_columns": "Drops deprecated edge columns (replies_to, relates_to, duplicate_of, superseded_by) from issues table (Decision 004 Phase 4)",
|
|
"pinned_column": "Adds pinned column for persistent context markers (bd-7h5)",
|
|
"is_template_column": "Adds is_template column for template molecules (beads-1ra)",
|
|
"remove_depends_on_fk": "Removes FK constraint on depends_on_id to allow external references (bd-zmmy)",
|
|
"additional_indexes": "Adds performance optimization indexes for common query patterns (bd-h0we)",
|
|
"gate_columns": "Adds gate columns (await_type, await_id, timeout_ns, waiters) for async coordination (bd-udsi)",
|
|
}
|
|
|
|
if desc, ok := descriptions[name]; ok {
|
|
return desc
|
|
}
|
|
return "Unknown migration"
|
|
}
|
|
|
|
// RunMigrations executes all registered migrations in order with invariant checking.
|
|
// Uses EXCLUSIVE transaction to prevent race conditions when multiple processes
|
|
// open the database simultaneously (GH#720).
|
|
func RunMigrations(db *sql.DB) error {
|
|
// Disable foreign keys BEFORE starting the transaction.
|
|
// PRAGMA foreign_keys must be called when no transaction is active (SQLite limitation).
|
|
// Some migrations (022, 025) drop/recreate tables and need foreign keys off
|
|
// to prevent ON DELETE CASCADE from deleting related data.
|
|
_, err := db.Exec("PRAGMA foreign_keys = OFF")
|
|
if err != nil {
|
|
return fmt.Errorf("failed to disable foreign keys for migrations: %w", err)
|
|
}
|
|
defer func() { _, _ = db.Exec("PRAGMA foreign_keys = ON") }()
|
|
|
|
// Acquire EXCLUSIVE lock to serialize migrations across processes.
|
|
// Without this, parallel processes can race on check-then-modify operations
|
|
// (e.g., checking if a column exists then adding it), causing "duplicate column" errors.
|
|
_, err = db.Exec("BEGIN EXCLUSIVE")
|
|
if err != nil {
|
|
return fmt.Errorf("failed to acquire exclusive lock for migrations: %w", err)
|
|
}
|
|
|
|
// Ensure we release the lock on any exit path
|
|
committed := false
|
|
defer func() {
|
|
if !committed {
|
|
_, _ = db.Exec("ROLLBACK")
|
|
}
|
|
}()
|
|
|
|
snapshot, err := captureSnapshot(db)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to capture pre-migration snapshot: %w", err)
|
|
}
|
|
|
|
for _, migration := range migrationsList {
|
|
if err := migration.Func(db); err != nil {
|
|
return fmt.Errorf("migration %s failed: %w", migration.Name, err)
|
|
}
|
|
}
|
|
|
|
if err := verifyInvariants(db, snapshot); err != nil {
|
|
return fmt.Errorf("post-migration validation failed: %w", err)
|
|
}
|
|
|
|
// Commit the transaction
|
|
if _, err := db.Exec("COMMIT"); err != nil {
|
|
return fmt.Errorf("failed to commit migrations: %w", err)
|
|
}
|
|
committed = true
|
|
|
|
return nil
|
|
}
|