Files
beads/cmd/bd/mol_ready_gated_test.go
beads/crew/emma 4f0f5744a6 feat(types): remove Gas Town types from core built-in types
Core beads built-in types now only include work types:
- bug, feature, task, epic, chore

Gas Town types (molecule, gate, convoy, merge-request, slot, agent,
role, rig, event, message) are now "well-known custom types":
- Constants still exist for code convenience
- Require types.custom configuration for validation
- bd types command shows core types and configured custom types

Changes:
- types.go: Separate core work types from well-known custom types
- IsValid(): Only accepts core work types
- bd types: Updated to show core types and custom types from config
- memory.go: Use ValidateWithCustom for custom type support
- multirepo.go: Only check core types as built-in
- Updated all tests to configure custom types

This allows Gas Town (and other projects) to define their own types
via config while keeping beads core focused on work tracking.

Closes: bd-find4

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-17 05:07:11 -08:00

447 lines
13 KiB
Go

package main
import (
"context"
"fmt"
"os"
"path/filepath"
"testing"
"time"
"github.com/steveyegge/beads/internal/storage/sqlite"
"github.com/steveyegge/beads/internal/types"
)
// setupGatedTestDB creates a temporary file-based test database
func setupGatedTestDB(t *testing.T) (*sqlite.SQLiteStorage, func()) {
t.Helper()
tmpDir, err := os.MkdirTemp("", "bd-test-gated-*")
if err != nil {
t.Fatalf("Failed to create temp dir: %v", err)
}
testDB := filepath.Join(tmpDir, "test.db")
store, err := sqlite.New(context.Background(), testDB)
if err != nil {
os.RemoveAll(tmpDir)
t.Fatalf("Failed to create test database: %v", err)
}
// Set issue_prefix (required for beads)
ctx := context.Background()
if err := store.SetConfig(ctx, "issue_prefix", "test"); err != nil {
store.Close()
os.RemoveAll(tmpDir)
t.Fatalf("Failed to set issue_prefix: %v", err)
}
// Configure Gas Town custom types for test compatibility (bd-find4)
if err := store.SetConfig(ctx, "types.custom", "molecule,gate,convoy,merge-request,slot,agent,role,rig,event,message"); err != nil {
store.Close()
os.RemoveAll(tmpDir)
t.Fatalf("Failed to set types.custom: %v", err)
}
cleanup := func() {
store.Close()
os.RemoveAll(tmpDir)
}
return store, cleanup
}
// =============================================================================
// mol ready --gated Tests (bd-lhalq: Gate-resume discovery)
// =============================================================================
// TestFindGateReadyMolecules_NoGates tests finding gate-ready molecules when no gates exist
func TestFindGateReadyMolecules_NoGates(t *testing.T) {
ctx := context.Background()
store, cleanup := setupGatedTestDB(t)
defer cleanup()
// Create a regular molecule (no gates)
mol := &types.Issue{
ID: "test-mol-001",
Title: "Test Molecule",
IssueType: types.TypeEpic,
Status: types.StatusInProgress,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
step := &types.Issue{
ID: "test-mol-001.step1",
Title: "Step 1",
IssueType: types.TypeTask,
Status: types.StatusOpen,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
if err := store.CreateIssue(ctx, mol, "test"); err != nil {
t.Fatalf("Failed to create molecule: %v", err)
}
if err := store.CreateIssue(ctx, step, "test"); err != nil {
t.Fatalf("Failed to create step: %v", err)
}
// Add parent-child relationship
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: step.ID,
DependsOnID: mol.ID,
Type: types.DepParentChild,
}, "test"); err != nil {
t.Fatalf("Failed to add dependency: %v", err)
}
// Find gate-ready molecules
molecules, err := findGateReadyMolecules(ctx, store)
if err != nil {
t.Fatalf("findGateReadyMolecules failed: %v", err)
}
if len(molecules) != 0 {
t.Errorf("Expected 0 gate-ready molecules, got %d", len(molecules))
}
}
// TestFindGateReadyMolecules_ClosedGate tests finding molecules with closed gates
func TestFindGateReadyMolecules_ClosedGate(t *testing.T) {
ctx := context.Background()
store, cleanup := setupGatedTestDB(t)
defer cleanup()
// Create molecule structure:
// mol-001
// └── gate-await-ci (closed)
// └── step1 (blocked by gate-await-ci, should become ready)
mol := &types.Issue{
ID: "test-mol-002",
Title: "Test Molecule with Gate",
IssueType: types.TypeEpic,
Status: types.StatusInProgress,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
gate := &types.Issue{
ID: "test-mol-002.gate-await-ci",
Title: "Gate: gh:run ci-workflow",
IssueType: types.TypeGate,
Status: types.StatusClosed, // Gate has closed
AwaitType: "gh:run",
AwaitID: "ci-workflow",
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
step := &types.Issue{
ID: "test-mol-002.step1",
Title: "Deploy after CI",
IssueType: types.TypeTask,
Status: types.StatusOpen,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
if err := store.CreateIssue(ctx, mol, "test"); err != nil {
t.Fatalf("Failed to create molecule: %v", err)
}
if err := store.CreateIssue(ctx, gate, "test"); err != nil {
t.Fatalf("Failed to create gate: %v", err)
}
if err := store.CreateIssue(ctx, step, "test"); err != nil {
t.Fatalf("Failed to create step: %v", err)
}
// Add parent-child relationships
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: gate.ID,
DependsOnID: mol.ID,
Type: types.DepParentChild,
}, "test"); err != nil {
t.Fatalf("Failed to add gate parent-child: %v", err)
}
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: step.ID,
DependsOnID: mol.ID,
Type: types.DepParentChild,
}, "test"); err != nil {
t.Fatalf("Failed to add step parent-child: %v", err)
}
// Add blocking dependency: step depends on gate (gate blocks step)
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: step.ID,
DependsOnID: gate.ID,
Type: types.DepBlocks,
}, "test"); err != nil {
t.Fatalf("Failed to add blocking dependency: %v", err)
}
// Find gate-ready molecules
molecules, err := findGateReadyMolecules(ctx, store)
if err != nil {
t.Fatalf("findGateReadyMolecules failed: %v", err)
}
if len(molecules) != 1 {
t.Errorf("Expected 1 gate-ready molecule, got %d", len(molecules))
return
}
if molecules[0].MoleculeID != mol.ID {
t.Errorf("Expected molecule ID %s, got %s", mol.ID, molecules[0].MoleculeID)
}
if molecules[0].ClosedGate == nil {
t.Error("Expected closed gate to be set")
} else if molecules[0].ClosedGate.ID != gate.ID {
t.Errorf("Expected closed gate ID %s, got %s", gate.ID, molecules[0].ClosedGate.ID)
}
if molecules[0].ReadyStep == nil {
t.Error("Expected ready step to be set")
} else if molecules[0].ReadyStep.ID != step.ID {
t.Errorf("Expected ready step ID %s, got %s", step.ID, molecules[0].ReadyStep.ID)
}
}
// TestFindGateReadyMolecules_OpenGate tests that open gates don't trigger ready
func TestFindGateReadyMolecules_OpenGate(t *testing.T) {
ctx := context.Background()
store, cleanup := setupGatedTestDB(t)
defer cleanup()
// Create molecule with OPEN gate
mol := &types.Issue{
ID: "test-mol-003",
Title: "Test Molecule with Open Gate",
IssueType: types.TypeEpic,
Status: types.StatusInProgress,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
gate := &types.Issue{
ID: "test-mol-003.gate-await-ci",
Title: "Gate: gh:run ci-workflow",
IssueType: types.TypeGate,
Status: types.StatusOpen, // Gate is still open
AwaitType: "gh:run",
AwaitID: "ci-workflow",
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
step := &types.Issue{
ID: "test-mol-003.step1",
Title: "Deploy after CI",
IssueType: types.TypeTask,
Status: types.StatusOpen,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
if err := store.CreateIssue(ctx, mol, "test"); err != nil {
t.Fatalf("Failed to create molecule: %v", err)
}
if err := store.CreateIssue(ctx, gate, "test"); err != nil {
t.Fatalf("Failed to create gate: %v", err)
}
if err := store.CreateIssue(ctx, step, "test"); err != nil {
t.Fatalf("Failed to create step: %v", err)
}
// Add parent-child relationships
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: gate.ID,
DependsOnID: mol.ID,
Type: types.DepParentChild,
}, "test"); err != nil {
t.Fatalf("Failed to add gate parent-child: %v", err)
}
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: step.ID,
DependsOnID: mol.ID,
Type: types.DepParentChild,
}, "test"); err != nil {
t.Fatalf("Failed to add step parent-child: %v", err)
}
// Add blocking dependency: step depends on gate
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: step.ID,
DependsOnID: gate.ID,
Type: types.DepBlocks,
}, "test"); err != nil {
t.Fatalf("Failed to add blocking dependency: %v", err)
}
// Find gate-ready molecules
molecules, err := findGateReadyMolecules(ctx, store)
if err != nil {
t.Fatalf("findGateReadyMolecules failed: %v", err)
}
if len(molecules) != 0 {
t.Errorf("Expected 0 gate-ready molecules (gate is open), got %d", len(molecules))
}
}
// TestFindGateReadyMolecules_HookedMolecule tests that hooked molecules are filtered out
func TestFindGateReadyMolecules_HookedMolecule(t *testing.T) {
ctx := context.Background()
store, cleanup := setupGatedTestDB(t)
defer cleanup()
// Create molecule with closed gate, but molecule is hooked
mol := &types.Issue{
ID: "test-mol-004",
Title: "Test Hooked Molecule",
IssueType: types.TypeEpic,
Status: types.StatusHooked, // Already hooked by an agent
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
gate := &types.Issue{
ID: "test-mol-004.gate-await-ci",
Title: "Gate: gh:run ci-workflow",
IssueType: types.TypeGate,
Status: types.StatusClosed,
AwaitType: "gh:run",
AwaitID: "ci-workflow",
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
step := &types.Issue{
ID: "test-mol-004.step1",
Title: "Deploy after CI",
IssueType: types.TypeTask,
Status: types.StatusOpen,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
if err := store.CreateIssue(ctx, mol, "test"); err != nil {
t.Fatalf("Failed to create molecule: %v", err)
}
if err := store.CreateIssue(ctx, gate, "test"); err != nil {
t.Fatalf("Failed to create gate: %v", err)
}
if err := store.CreateIssue(ctx, step, "test"); err != nil {
t.Fatalf("Failed to create step: %v", err)
}
// Add parent-child relationships
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: gate.ID,
DependsOnID: mol.ID,
Type: types.DepParentChild,
}, "test"); err != nil {
t.Fatalf("Failed to add gate parent-child: %v", err)
}
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: step.ID,
DependsOnID: mol.ID,
Type: types.DepParentChild,
}, "test"); err != nil {
t.Fatalf("Failed to add step parent-child: %v", err)
}
// Add blocking dependency
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: step.ID,
DependsOnID: gate.ID,
Type: types.DepBlocks,
}, "test"); err != nil {
t.Fatalf("Failed to add blocking dependency: %v", err)
}
// Find gate-ready molecules
molecules, err := findGateReadyMolecules(ctx, store)
if err != nil {
t.Fatalf("findGateReadyMolecules failed: %v", err)
}
if len(molecules) != 0 {
t.Errorf("Expected 0 gate-ready molecules (molecule is hooked), got %d", len(molecules))
}
}
// TestFindGateReadyMolecules_MultipleGates tests handling multiple closed gates
func TestFindGateReadyMolecules_MultipleGates(t *testing.T) {
ctx := context.Background()
store, cleanup := setupGatedTestDB(t)
defer cleanup()
// Create two molecules, each with a closed gate
for i := 1; i <= 2; i++ {
molID := fmt.Sprintf("test-multi-%d", i)
mol := &types.Issue{
ID: molID,
Title: fmt.Sprintf("Multi Gate Mol %d", i),
IssueType: types.TypeEpic,
Status: types.StatusInProgress,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
gate := &types.Issue{
ID: fmt.Sprintf("%s.gate", molID),
Title: "Gate: gh:run",
IssueType: types.TypeGate,
Status: types.StatusClosed,
AwaitType: "gh:run",
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
step := &types.Issue{
ID: fmt.Sprintf("%s.step1", molID),
Title: "Step 1",
IssueType: types.TypeTask,
Status: types.StatusOpen,
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
if err := store.CreateIssue(ctx, mol, "test"); err != nil {
t.Fatalf("Failed to create molecule %d: %v", i, err)
}
if err := store.CreateIssue(ctx, gate, "test"); err != nil {
t.Fatalf("Failed to create gate %d: %v", i, err)
}
if err := store.CreateIssue(ctx, step, "test"); err != nil {
t.Fatalf("Failed to create step %d: %v", i, err)
}
// Add dependencies
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: gate.ID,
DependsOnID: mol.ID,
Type: types.DepParentChild,
}, "test"); err != nil {
t.Fatalf("Failed to add gate parent-child %d: %v", i, err)
}
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: step.ID,
DependsOnID: mol.ID,
Type: types.DepParentChild,
}, "test"); err != nil {
t.Fatalf("Failed to add step parent-child %d: %v", i, err)
}
if err := store.AddDependency(ctx, &types.Dependency{
IssueID: step.ID,
DependsOnID: gate.ID,
Type: types.DepBlocks,
}, "test"); err != nil {
t.Fatalf("Failed to add blocking dep %d: %v", i, err)
}
}
// Find gate-ready molecules
molecules, err := findGateReadyMolecules(ctx, store)
if err != nil {
t.Fatalf("findGateReadyMolecules failed: %v", err)
}
if len(molecules) != 2 {
t.Errorf("Expected 2 gate-ready molecules, got %d", len(molecules))
}
}