feat: Add bd info --whats-new for agent version awareness (bd-eiz9)

- Add --whats-new flag to bd info command
- Display agent-relevant changes from last 3 versions
- Support both human-readable and JSON output
- Add version changelog data structure with workflow-impacting changes
- Add comprehensive tests for version changes structure
- Update AGENTS.md with whats-new documentation

Helps agents efficiently learn what changed between bd versions without
re-reading full documentation. Addresses weekly major version releases
requiring workflow adaptations.

Closes bd-eiz9

Amp-Thread-ID: https://ampcode.com/threads/T-5fe7e93d-7398-41c5-94bf-e914f2b331dd
Co-authored-by: Amp <amp@ampcode.com>
This commit is contained in:
Steve Yegge
2025-11-08 02:28:42 -08:00
parent 1da414fa34
commit a03737c89a
3 changed files with 192 additions and 1 deletions

View File

@@ -23,13 +23,23 @@ or daemon connection. It shows:
- If using daemon: socket path, health status, version
- Database statistics (issue count)
- Schema information (with --schema flag)
- What's new in recent versions (with --whats-new flag)
Examples:
bd info
bd info --json
bd info --schema --json`,
bd info --schema --json
bd info --whats-new
bd info --whats-new --json`,
Run: func(cmd *cobra.Command, args []string) {
schemaFlag, _ := cmd.Flags().GetBool("schema")
whatsNewFlag, _ := cmd.Flags().GetBool("whats-new")
// Handle --whats-new flag
if whatsNewFlag {
showWhatsNew()
return
}
// Get database path (absolute)
absDBPath, err := filepath.Abs(dbPath)
@@ -239,8 +249,91 @@ func extractPrefix(issueID string) string {
return ""
}
// VersionChange represents agent-relevant changes for a specific version
type VersionChange struct {
Version string `json:"version"`
Date string `json:"date"`
Changes []string `json:"changes"`
}
// versionChanges contains agent-actionable changes for recent versions
var versionChanges = []VersionChange{
{
Version: "0.22.1",
Date: "2025-11-06",
Changes: []string{
"Native `bd merge` command vendored from beads-merge - no external binary needed",
"`bd info` detects outdated git hooks - warns if version mismatch",
"Multi-workspace deletion tracking fixed - deletions now propagate correctly",
"Hash ID recognition improved - recognizes Base36 IDs without a-f letters",
"Import/export deadlock fixed - no hanging when daemon running",
},
},
{
Version: "0.22.0",
Date: "2025-11-05",
Changes: []string{
"Intelligent merge driver auto-configured - eliminates most JSONL conflicts",
"Onboarding wizards: `bd init --contributor` and `bd init --team`",
"New `bd migrate-issues` command - migrate issues between repos with dependencies",
"`bd show` displays blocker status - 'Blocked by N open issues' or 'Ready to work'",
"SearchIssues N+1 query fixed - batch-loads labels for better performance",
"Sync validation prevents infinite dirty loop - verifies JSONL export",
},
},
{
Version: "0.21.0",
Date: "2025-11-04",
Changes: []string{
"Hash-based IDs eliminate collisions - remove ID coordination workarounds",
"Event-driven daemon mode (opt-in) - set BEADS_DAEMON_MODE=events",
"Agent Mail integration - real-time multi-agent coordination (<100ms latency)",
"`bd duplicates --auto-merge` - automated duplicate detection and merging",
"Hierarchical children for epics - dotted IDs (bd-abc.1, bd-abc.2) up to 3 levels",
"`--discovered-from` inline syntax - create with dependency in one command",
},
},
}
// showWhatsNew displays agent-relevant changes from recent versions
func showWhatsNew() {
currentVersion := Version // from version.go
if jsonOutput {
outputJSON(map[string]interface{}{
"current_version": currentVersion,
"recent_changes": versionChanges,
})
return
}
// Human-readable output
fmt.Printf("\n🆕 What's New in bd (Current: v%s)\n", currentVersion)
fmt.Println("=" + strings.Repeat("=", 60))
fmt.Println()
for _, vc := range versionChanges {
// Highlight if this is the current version
versionMarker := ""
if vc.Version == currentVersion {
versionMarker = " ← current"
}
fmt.Printf("## v%s (%s)%s\n\n", vc.Version, vc.Date, versionMarker)
for _, change := range vc.Changes {
fmt.Printf(" • %s\n", change)
}
fmt.Println()
}
fmt.Println("💡 Tip: Use `bd info --whats-new --json` for machine-readable output")
fmt.Println()
}
func init() {
infoCmd.Flags().Bool("schema", false, "Include schema information in output")
infoCmd.Flags().Bool("whats-new", false, "Show agent-relevant changes from recent versions")
infoCmd.Flags().BoolVar(&jsonOutput, "json", false, "Output in JSON format")
rootCmd.AddCommand(infoCmd)
}

View File

@@ -1,6 +1,7 @@
package main
import (
"encoding/json"
"testing"
)
@@ -11,3 +12,83 @@ func TestInfoCommand(t *testing.T) {
func TestInfoWithNoDaemon(t *testing.T) {
t.Skip("Manual test - bd info --no-daemon command is working, see manual testing")
}
func TestVersionChangesStructure(t *testing.T) {
// Verify versionChanges is properly structured
if len(versionChanges) == 0 {
t.Fatal("versionChanges should not be empty")
}
for i, vc := range versionChanges {
if vc.Version == "" {
t.Errorf("versionChanges[%d] has empty Version", i)
}
if vc.Date == "" {
t.Errorf("versionChanges[%d] has empty Date", i)
}
if len(vc.Changes) == 0 {
t.Errorf("versionChanges[%d] has no changes", i)
}
// Verify version format (should be like "0.22.1")
if len(vc.Version) < 5 {
t.Errorf("versionChanges[%d] has invalid Version format: %s", i, vc.Version)
}
// Verify date format (should be like "2025-11-06")
if len(vc.Date) != 10 {
t.Errorf("versionChanges[%d] has invalid Date format: %s", i, vc.Date)
}
// Verify each change is non-empty
for j, change := range vc.Changes {
if change == "" {
t.Errorf("versionChanges[%d].Changes[%d] is empty", i, j)
}
}
}
}
func TestVersionChangesJSON(t *testing.T) {
// Test that versionChanges can be marshaled to JSON
data, err := json.Marshal(versionChanges)
if err != nil {
t.Fatalf("Failed to marshal versionChanges to JSON: %v", err)
}
// Test that it can be unmarshaled back
var unmarshaled []VersionChange
err = json.Unmarshal(data, &unmarshaled)
if err != nil {
t.Fatalf("Failed to unmarshal versionChanges from JSON: %v", err)
}
// Verify structure is preserved
if len(unmarshaled) != len(versionChanges) {
t.Errorf("Unmarshaled length %d != original length %d", len(unmarshaled), len(versionChanges))
}
// Spot check first entry
if len(unmarshaled) > 0 && len(versionChanges) > 0 {
if unmarshaled[0].Version != versionChanges[0].Version {
t.Errorf("Version mismatch: %s != %s", unmarshaled[0].Version, versionChanges[0].Version)
}
if len(unmarshaled[0].Changes) != len(versionChanges[0].Changes) {
t.Errorf("Changes count mismatch: %d != %d", len(unmarshaled[0].Changes), len(versionChanges[0].Changes))
}
}
}
func TestVersionChangesCoverage(t *testing.T) {
// Ensure we have at least 3 recent versions documented
if len(versionChanges) < 3 {
t.Errorf("Should document at least 3 recent versions, found %d", len(versionChanges))
}
// Ensure each version has meaningful changes (at least 3 bullet points)
for i, vc := range versionChanges {
if len(vc.Changes) < 3 {
t.Errorf("versionChanges[%d] (v%s) should have at least 3 changes, found %d", i, vc.Version, len(vc.Changes))
}
}
}