refactor(cli): restructure cmd packages into subdirectories
- Move CLI commands into subdirectories matching command hierarchy:
dev/, go/, php/, build/, ci/, sdk/, pkg/, vm/, docs/, setup/, doctor/, test/, ai/
- Create shared/ package for common styles and utilities
- Add new `core ai` root command with claude subcommand
- Update package declarations and imports across all files
- Create commands.go entry points for each package
- Remove GUI-related files (moved to core-gui repo)
This makes the filesystem structure match the CLI command structure,
improving context capture and code organization.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-29 18:02:43 +00:00
|
|
|
package dev
|
2026-01-27 21:08:51 +00:00
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"os"
|
|
|
|
|
"sort"
|
|
|
|
|
|
|
|
|
|
"github.com/charmbracelet/lipgloss"
|
2026-01-30 00:22:47 +00:00
|
|
|
"github.com/host-uk/core/cmd/shared"
|
refactor(cli): restructure cmd packages into subdirectories
- Move CLI commands into subdirectories matching command hierarchy:
dev/, go/, php/, build/, ci/, sdk/, pkg/, vm/, docs/, setup/, doctor/, test/, ai/
- Create shared/ package for common styles and utilities
- Add new `core ai` root command with claude subcommand
- Update package declarations and imports across all files
- Create commands.go entry points for each package
- Remove GUI-related files (moved to core-gui repo)
This makes the filesystem structure match the CLI command structure,
improving context capture and code organization.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-29 18:02:43 +00:00
|
|
|
"github.com/host-uk/core/pkg/repos"
|
2026-01-27 21:08:51 +00:00
|
|
|
"github.com/leaanthony/clir"
|
|
|
|
|
)
|
|
|
|
|
|
2026-01-30 00:22:47 +00:00
|
|
|
// Impact-specific styles
|
2026-01-27 21:08:51 +00:00
|
|
|
var (
|
|
|
|
|
impactDirectStyle = lipgloss.NewStyle().
|
|
|
|
|
Bold(true).
|
|
|
|
|
Foreground(lipgloss.Color("#ef4444")) // red-500
|
|
|
|
|
|
|
|
|
|
impactIndirectStyle = lipgloss.NewStyle().
|
|
|
|
|
Foreground(lipgloss.Color("#f59e0b")) // amber-500
|
|
|
|
|
|
|
|
|
|
impactSafeStyle = lipgloss.NewStyle().
|
|
|
|
|
Foreground(lipgloss.Color("#22c55e")) // green-500
|
|
|
|
|
)
|
|
|
|
|
|
2026-01-30 00:22:47 +00:00
|
|
|
// addImpactCommand adds the 'impact' command to the given parent command.
|
|
|
|
|
func addImpactCommand(parent *clir.Command) {
|
2026-01-27 21:08:51 +00:00
|
|
|
var registryPath string
|
|
|
|
|
|
|
|
|
|
impactCmd := parent.NewSubCommand("impact", "Show impact of changing a repo")
|
|
|
|
|
impactCmd.LongDescription("Analyzes the dependency graph to show which repos\n" +
|
|
|
|
|
"would be affected by changes to the specified repo.")
|
|
|
|
|
|
|
|
|
|
impactCmd.StringFlag("registry", "Path to repos.yaml (auto-detected if not specified)", ®istryPath)
|
|
|
|
|
|
|
|
|
|
impactCmd.Action(func() error {
|
|
|
|
|
args := os.Args[2:] // Skip "core" and "impact"
|
|
|
|
|
// Filter out flags
|
|
|
|
|
var repoName string
|
|
|
|
|
for _, arg := range args {
|
|
|
|
|
if arg[0] != '-' {
|
|
|
|
|
repoName = arg
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if repoName == "" {
|
|
|
|
|
return fmt.Errorf("usage: core impact <repo-name>")
|
|
|
|
|
}
|
|
|
|
|
return runImpact(registryPath, repoName)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func runImpact(registryPath string, repoName string) error {
|
|
|
|
|
// Find or use provided registry
|
|
|
|
|
var reg *repos.Registry
|
|
|
|
|
var err error
|
|
|
|
|
|
|
|
|
|
if registryPath != "" {
|
|
|
|
|
reg, err = repos.LoadRegistry(registryPath)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("failed to load registry: %w", err)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
registryPath, err = repos.FindRegistry()
|
|
|
|
|
if err == nil {
|
|
|
|
|
reg, err = repos.LoadRegistry(registryPath)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("failed to load registry: %w", err)
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
return fmt.Errorf("impact analysis requires repos.yaml with dependency information")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check repo exists
|
|
|
|
|
repo, exists := reg.Get(repoName)
|
|
|
|
|
if !exists {
|
|
|
|
|
return fmt.Errorf("repo '%s' not found in registry", repoName)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Build reverse dependency graph
|
|
|
|
|
dependents := buildDependentsGraph(reg)
|
|
|
|
|
|
|
|
|
|
// Find all affected repos (direct and transitive)
|
|
|
|
|
direct := dependents[repoName]
|
|
|
|
|
allAffected := findAllDependents(repoName, dependents)
|
|
|
|
|
|
|
|
|
|
// Separate direct vs indirect
|
|
|
|
|
directSet := make(map[string]bool)
|
|
|
|
|
for _, d := range direct {
|
|
|
|
|
directSet[d] = true
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var indirect []string
|
|
|
|
|
for _, a := range allAffected {
|
|
|
|
|
if !directSet[a] {
|
|
|
|
|
indirect = append(indirect, a)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Sort for consistent output
|
|
|
|
|
sort.Strings(direct)
|
|
|
|
|
sort.Strings(indirect)
|
|
|
|
|
|
|
|
|
|
// Print results
|
|
|
|
|
fmt.Println()
|
|
|
|
|
fmt.Printf("%s %s\n", dimStyle.Render("Impact analysis for"), repoNameStyle.Render(repoName))
|
|
|
|
|
if repo.Description != "" {
|
|
|
|
|
fmt.Printf("%s\n", dimStyle.Render(repo.Description))
|
|
|
|
|
}
|
|
|
|
|
fmt.Println()
|
|
|
|
|
|
|
|
|
|
if len(allAffected) == 0 {
|
2026-01-30 00:22:47 +00:00
|
|
|
fmt.Printf("%s No repos depend on %s\n", impactSafeStyle.Render("v"), repoName)
|
2026-01-27 21:08:51 +00:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Direct dependents
|
|
|
|
|
if len(direct) > 0 {
|
|
|
|
|
fmt.Printf("%s %d direct dependent(s):\n",
|
2026-01-30 00:22:47 +00:00
|
|
|
impactDirectStyle.Render("*"),
|
2026-01-27 21:08:51 +00:00
|
|
|
len(direct),
|
|
|
|
|
)
|
|
|
|
|
for _, d := range direct {
|
|
|
|
|
r, _ := reg.Get(d)
|
|
|
|
|
desc := ""
|
|
|
|
|
if r != nil && r.Description != "" {
|
2026-01-30 00:22:47 +00:00
|
|
|
desc = dimStyle.Render(" - " + shared.Truncate(r.Description, 40))
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
|
|
|
|
fmt.Printf(" %s%s\n", d, desc)
|
|
|
|
|
}
|
|
|
|
|
fmt.Println()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Indirect dependents
|
|
|
|
|
if len(indirect) > 0 {
|
|
|
|
|
fmt.Printf("%s %d transitive dependent(s):\n",
|
2026-01-30 00:22:47 +00:00
|
|
|
impactIndirectStyle.Render("o"),
|
2026-01-27 21:08:51 +00:00
|
|
|
len(indirect),
|
|
|
|
|
)
|
|
|
|
|
for _, d := range indirect {
|
|
|
|
|
r, _ := reg.Get(d)
|
|
|
|
|
desc := ""
|
|
|
|
|
if r != nil && r.Description != "" {
|
2026-01-30 00:22:47 +00:00
|
|
|
desc = dimStyle.Render(" - " + shared.Truncate(r.Description, 40))
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
|
|
|
|
fmt.Printf(" %s%s\n", d, desc)
|
|
|
|
|
}
|
|
|
|
|
fmt.Println()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Summary
|
|
|
|
|
fmt.Printf("%s Changes to %s affect %s\n",
|
|
|
|
|
dimStyle.Render("Summary:"),
|
|
|
|
|
repoNameStyle.Render(repoName),
|
|
|
|
|
impactDirectStyle.Render(fmt.Sprintf("%d/%d repos", len(allAffected), len(reg.Repos)-1)),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// buildDependentsGraph creates a reverse dependency map
|
|
|
|
|
// key = repo, value = repos that depend on it
|
|
|
|
|
func buildDependentsGraph(reg *repos.Registry) map[string][]string {
|
|
|
|
|
dependents := make(map[string][]string)
|
|
|
|
|
|
|
|
|
|
for name, repo := range reg.Repos {
|
|
|
|
|
for _, dep := range repo.DependsOn {
|
|
|
|
|
dependents[dep] = append(dependents[dep], name)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return dependents
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// findAllDependents recursively finds all repos that depend on the given repo
|
|
|
|
|
func findAllDependents(repoName string, dependents map[string][]string) []string {
|
|
|
|
|
visited := make(map[string]bool)
|
|
|
|
|
var result []string
|
|
|
|
|
|
|
|
|
|
var visit func(name string)
|
|
|
|
|
visit = func(name string) {
|
|
|
|
|
for _, dep := range dependents[name] {
|
|
|
|
|
if !visited[dep] {
|
|
|
|
|
visited[dep] = true
|
|
|
|
|
result = append(result, dep)
|
|
|
|
|
visit(dep) // Recurse for transitive deps
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
visit(repoName)
|
|
|
|
|
return result
|
|
|
|
|
}
|