package main import ( "context" "fmt" "os" "github.com/spf13/cobra" "github.com/steveyegge/beads/internal/storage/sqlite" "github.com/steveyegge/beads/internal/ui" "github.com/steveyegge/beads/internal/utils" ) var molBurnCmd = &cobra.Command{ Use: "burn ", Short: "Delete a wisp molecule without creating a digest", Long: `Burn a wisp molecule, deleting it without creating a digest. Unlike squash (which creates a permanent digest before deletion), burn completely removes the wisp with no trace. Use this for: - Abandoned patrol cycles - Crashed or failed workflows - Test/debug wisps you don't want to preserve The burn operation: 1. Verifies the molecule has Wisp=true (is ephemeral) 2. Deletes the molecule and all its wisp children 3. No digest is created (use 'bd mol squash' if you want a digest) CAUTION: This is a destructive operation. The wisp's data will be permanently lost. If you want to preserve a summary, use 'bd mol squash'. Example: bd mol burn bd-abc123 # Delete wisp with no trace bd mol burn bd-abc123 --dry-run # Preview what would be deleted bd mol burn bd-abc123 --force # Skip confirmation`, Args: cobra.ExactArgs(1), Run: runMolBurn, } // BurnResult holds the result of a burn operation type BurnResult struct { MoleculeID string `json:"molecule_id"` DeletedIDs []string `json:"deleted_ids"` DeletedCount int `json:"deleted_count"` } func runMolBurn(cmd *cobra.Command, args []string) { CheckReadonly("mol burn") ctx := rootCtx // mol burn requires direct store access if store == nil { if daemonClient != nil { fmt.Fprintf(os.Stderr, "Error: mol burn requires direct database access\n") fmt.Fprintf(os.Stderr, "Hint: use --no-daemon flag: bd --no-daemon mol burn %s ...\n", args[0]) } else { fmt.Fprintf(os.Stderr, "Error: no database connection\n") } os.Exit(1) } dryRun, _ := cmd.Flags().GetBool("dry-run") force, _ := cmd.Flags().GetBool("force") moleculeID := args[0] // Resolve molecule ID in main store resolvedID, err := utils.ResolvePartialID(ctx, store, moleculeID) if err != nil { fmt.Fprintf(os.Stderr, "Error resolving molecule ID %s: %v\n", moleculeID, err) os.Exit(1) } // Load the molecule rootIssue, err := store.GetIssue(ctx, resolvedID) if err != nil { fmt.Fprintf(os.Stderr, "Error loading molecule: %v\n", err) os.Exit(1) } // Verify it's a wisp if !rootIssue.Wisp { fmt.Fprintf(os.Stderr, "Error: molecule %s is not a wisp (Wisp=false)\n", resolvedID) fmt.Fprintf(os.Stderr, "Hint: mol burn only works with wisp molecules\n") fmt.Fprintf(os.Stderr, " Use 'bd delete' to remove non-wisp issues\n") os.Exit(1) } // Load the molecule subgraph subgraph, err := loadTemplateSubgraph(ctx, store, resolvedID) if err != nil { fmt.Fprintf(os.Stderr, "Error loading wisp molecule: %v\n", err) os.Exit(1) } // Collect wisp issue IDs to delete (only delete wisps, not regular children) var wispIDs []string for _, issue := range subgraph.Issues { if issue.Wisp { wispIDs = append(wispIDs, issue.ID) } } if len(wispIDs) == 0 { if jsonOutput { outputJSON(BurnResult{ MoleculeID: resolvedID, DeletedCount: 0, }) } else { fmt.Printf("No wisp issues found for molecule %s\n", resolvedID) } return } if dryRun { fmt.Printf("\nDry run: would burn wisp %s\n\n", resolvedID) fmt.Printf("Root: %s\n", subgraph.Root.Title) fmt.Printf("\nWisp issues to delete (%d total):\n", len(wispIDs)) for _, issue := range subgraph.Issues { if !issue.Wisp { continue } status := string(issue.Status) if issue.ID == subgraph.Root.ID { fmt.Printf(" - [%s] %s (%s) [ROOT]\n", status, issue.Title, issue.ID) } else { fmt.Printf(" - [%s] %s (%s)\n", status, issue.Title, issue.ID) } } fmt.Printf("\nNo digest will be created (use 'bd mol squash' to create one).\n") return } // Confirm unless --force if !force && !jsonOutput { fmt.Printf("About to burn wisp %s (%d issues)\n", resolvedID, len(wispIDs)) fmt.Printf("This will permanently delete all wisp data with no digest.\n") fmt.Printf("Use 'bd mol squash' instead if you want to preserve a summary.\n") fmt.Printf("\nContinue? [y/N] ") var response string _, _ = fmt.Scanln(&response) if response != "y" && response != "Y" { fmt.Println("Canceled.") return } } // Perform the burn result, err := burnWisps(ctx, store, wispIDs) if err != nil { fmt.Fprintf(os.Stderr, "Error burning wisp: %v\n", err) os.Exit(1) } result.MoleculeID = resolvedID // Schedule auto-flush markDirtyAndScheduleFlush() if jsonOutput { outputJSON(result) return } fmt.Printf("%s Burned wisp: %d issues deleted\n", ui.RenderPass("✓"), result.DeletedCount) fmt.Printf(" Wisp: %s\n", resolvedID) fmt.Printf(" No digest created.\n") } // burnWisps deletes all wisp issues without creating a digest func burnWisps(ctx context.Context, s interface{}, ids []string) (*BurnResult, error) { // Type assert to SQLite storage for delete access sqliteStore, ok := s.(*sqlite.SQLiteStorage) if !ok { return nil, fmt.Errorf("burn requires SQLite storage backend") } result := &BurnResult{ DeletedIDs: make([]string, 0, len(ids)), } for _, id := range ids { if err := sqliteStore.DeleteIssue(ctx, id); err != nil { // Log but continue - try to delete as many as possible fmt.Fprintf(os.Stderr, "Warning: failed to delete %s: %v\n", id, err) continue } result.DeletedIDs = append(result.DeletedIDs, id) result.DeletedCount++ } return result, nil } func init() { molBurnCmd.Flags().Bool("dry-run", false, "Preview what would be deleted") molBurnCmd.Flags().Bool("force", false, "Skip confirmation prompt") molCmd.AddCommand(molBurnCmd) }