Files
beads/internal/storage/sqlite/gate_no_daemon_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

103 lines
3.1 KiB
Go

package sqlite
import (
"context"
"testing"
"time"
"github.com/steveyegge/beads/internal/types"
)
// TestGateFieldsPreservedAcrossConnections reproduces beads-70c4:
// Gate await fields should not be cleared when a new database connection
// is opened (simulating --no-daemon CLI access).
func TestGateFieldsPreservedAcrossConnections(t *testing.T) {
// Use a temporary file database (not :memory:) to simulate real-world scenario
dbPath := t.TempDir() + "/beads.db"
ctx := context.Background()
// Step 1: Create a database and add a gate with await fields
// (simulating daemon creating a gate)
store1, err := New(ctx, dbPath)
if err != nil {
t.Fatalf("failed to create first store: %v", err)
}
// Initialize the database with a prefix
if err := store1.SetConfig(ctx, "issue_prefix", "beads"); err != nil {
t.Fatalf("failed to set issue_prefix: %v", err)
}
// Configure custom types for Gas Town types (gate is not a core type)
if err := store1.SetConfig(ctx, "types.custom", "gate"); err != nil {
t.Fatalf("failed to set types.custom: %v", err)
}
gate := &types.Issue{
ID: "beads-test1",
Title: "Test Gate",
Status: types.StatusOpen,
Priority: 1,
IssueType: types.TypeGate,
Ephemeral: true,
AwaitType: "timer",
AwaitID: "5s",
Timeout: 5 * time.Second,
Waiters: []string{"system"},
CreatedAt: time.Now(),
UpdatedAt: time.Now(),
}
gate.ContentHash = gate.ComputeContentHash()
if err := store1.CreateIssue(ctx, gate, "daemon"); err != nil {
t.Fatalf("failed to create gate: %v", err)
}
// Verify gate was created with await fields
retrieved1, err := store1.GetIssue(ctx, gate.ID)
if err != nil || retrieved1 == nil {
t.Fatalf("failed to get gate from store1: %v", err)
}
if retrieved1.AwaitType != "timer" {
t.Errorf("store1: expected AwaitType=timer, got %q", retrieved1.AwaitType)
}
if retrieved1.AwaitID != "5s" {
t.Errorf("store1: expected AwaitID=5s, got %q", retrieved1.AwaitID)
}
// Close the first store (simulating daemon connection)
if err := store1.Close(); err != nil {
t.Fatalf("failed to close store1: %v", err)
}
// Step 2: Open a NEW connection to the same database
// (simulating `bd show --no-daemon` opening a new connection)
store2, err := New(ctx, dbPath)
if err != nil {
t.Fatalf("failed to create second store: %v", err)
}
defer store2.Close()
// Step 3: Read the gate from the new connection
// This should NOT clear the await fields
retrieved2, err := store2.GetIssue(ctx, gate.ID)
if err != nil || retrieved2 == nil {
t.Fatalf("failed to get gate from store2: %v", err)
}
// Verify await fields are PRESERVED
if retrieved2.AwaitType != "timer" {
t.Errorf("AwaitType was cleared! expected 'timer', got %q", retrieved2.AwaitType)
}
if retrieved2.AwaitID != "5s" {
t.Errorf("AwaitID was cleared! expected '5s', got %q", retrieved2.AwaitID)
}
if retrieved2.Timeout != 5*time.Second {
t.Errorf("Timeout was cleared! expected %v, got %v", 5*time.Second, retrieved2.Timeout)
}
if len(retrieved2.Waiters) != 1 || retrieved2.Waiters[0] != "system" {
t.Errorf("Waiters was cleared! expected [system], got %v", retrieved2.Waiters)
}
}