Files
gastown/internal/doctor/lifecycle_check.go
Ryan Snodgrass e1f2bb8b4b feat(ui): import comprehensive UX system from beads
Import beads' UX design system into gastown:

- Add internal/ui/ package with Ayu theme colors and semantic styling
  - styles.go: AdaptiveColor definitions for light/dark mode
  - terminal.go: TTY detection, NO_COLOR/CLICOLOR support
  - markdown.go: Glamour rendering with agent mode bypass
  - pager.go: Smart paging with GT_PAGER support

- Add colorized help output (internal/cmd/help.go)
  - Group headers in accent color
  - Command names styled for scannability
  - Flag types and defaults muted

- Add gt thanks command (internal/cmd/thanks.go)
  - Contributor display with same logic as bd thanks
  - Styled with Ayu theme colors

- Update gt doctor to match bd doctor UX
  - Category grouping (Core, Infrastructure, Rig, Patrol, etc.)
  - Semantic icons (✓ ⚠ ✖) with Ayu colors
  - Tree connectors for detail lines
  - Summary line with pass/warn/fail counts
  - Warnings section at end with numbered issues

- Migrate existing styles to use ui package
  - internal/style/style.go uses ui.ColorPass etc.
  - internal/tui/feed/styles.go uses ui package colors

Co-Authored-By: SageOx <ox@sageox.ai>
2026-01-09 22:46:06 -08:00

114 lines
2.8 KiB
Go

package doctor
import (
"encoding/json"
"fmt"
"os/exec"
"strings"
)
// LifecycleHygieneCheck detects and cleans up stale lifecycle state.
// This can happen when lifecycle messages weren't properly deleted after processing.
type LifecycleHygieneCheck struct {
FixableCheck
staleMessages []staleMessage
}
type staleMessage struct {
ID string
Subject string
From string
}
// NewLifecycleHygieneCheck creates a new lifecycle hygiene check.
func NewLifecycleHygieneCheck() *LifecycleHygieneCheck {
return &LifecycleHygieneCheck{
FixableCheck: FixableCheck{
BaseCheck: BaseCheck{
CheckName: "lifecycle-hygiene",
CheckDescription: "Check for stale lifecycle messages",
CheckCategory: CategoryConfig,
},
},
}
}
// Run checks for stale lifecycle state.
func (c *LifecycleHygieneCheck) Run(ctx *CheckContext) *CheckResult {
c.staleMessages = nil
// Check for stale lifecycle messages in deacon inbox
staleCount := c.checkDeaconInbox(ctx)
if staleCount == 0 {
return &CheckResult{
Name: c.Name(),
Status: StatusOK,
Message: "No stale lifecycle messages found",
}
}
return &CheckResult{
Name: c.Name(),
Status: StatusWarning,
Message: fmt.Sprintf("Found %d stale lifecycle message(s) in deacon inbox", staleCount),
FixHint: "Run 'gt doctor --fix' to clean up",
}
}
// checkDeaconInbox looks for stale lifecycle messages.
func (c *LifecycleHygieneCheck) checkDeaconInbox(ctx *CheckContext) int {
// Get deacon inbox via gt mail
cmd := exec.Command("gt", "mail", "inbox", "--identity", "deacon/", "--json")
cmd.Dir = ctx.TownRoot
output, err := cmd.Output()
if err != nil {
return 0 // Can't check, assume OK
}
if len(output) == 0 || string(output) == "[]" || string(output) == "[]\n" {
return 0
}
var messages []struct {
ID string `json:"id"`
From string `json:"from"`
Subject string `json:"subject"`
}
if err := json.Unmarshal(output, &messages); err != nil {
return 0
}
// Look for lifecycle messages
for _, msg := range messages {
if strings.HasPrefix(strings.ToLower(msg.Subject), "lifecycle:") {
c.staleMessages = append(c.staleMessages, staleMessage{
ID: msg.ID,
Subject: msg.Subject,
From: msg.From,
})
}
}
return len(c.staleMessages)
}
// Fix cleans up stale lifecycle messages.
func (c *LifecycleHygieneCheck) Fix(ctx *CheckContext) error {
var errors []string
// Delete stale lifecycle messages
for _, msg := range c.staleMessages {
cmd := exec.Command("gt", "mail", "delete", msg.ID) //nolint:gosec // G204: msg.ID is from internal state, not user input
cmd.Dir = ctx.TownRoot
if err := cmd.Run(); err != nil {
errors = append(errors, fmt.Sprintf("failed to delete message %s: %v", msg.ID, err))
}
}
if len(errors) > 0 {
return fmt.Errorf("%s", strings.Join(errors, "; "))
}
return nil
}