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 (
|
refactor(cli): move commands from cmd/ to pkg/ with self-registration
Implements defence in depth through build variants - only compiled code
exists in the binary. Commands now self-register via cli.RegisterCommands()
in their init() functions, mirroring the i18n.RegisterLocales() pattern.
Structure changes:
- cmd/{ai,build,ci,dev,docs,doctor,go,php,pkg,sdk,setup,test,vm}/ → pkg/*/cmd_*.go
- cmd/core_dev.go, cmd/core_ci.go → cmd/variants/{full,ci,php,minimal}.go
- Added pkg/cli/commands.go with RegisterCommands API
- Updated pkg/cli/runtime.go to attach registered commands
Build variants:
- go build → full (21MB, all 13 command groups)
- go build -tags ci → ci (18MB, build/ci/sdk/doctor)
- go build -tags php → php (14MB, php/doctor)
- go build -tags minimal → minimal (11MB, doctor only)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 21:55:55 +00:00
|
|
|
"errors"
|
2026-01-27 21:08:51 +00:00
|
|
|
"sort"
|
|
|
|
|
|
2026-01-30 10:32:05 +00:00
|
|
|
"github.com/host-uk/core/pkg/cli"
|
feat(i18n): add translation keys to all CLI commands
Replace hardcoded strings with i18n.T() calls across all cmd/* packages:
- ai, build, ci, dev, docs, doctor, go, php, pkg, sdk, setup, test, vm
Adds 500+ translation keys to en.json for command descriptions,
flag descriptions, labels, messages, and error strings.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 02:37:57 +00:00
|
|
|
"github.com/host-uk/core/pkg/i18n"
|
2026-02-04 18:03:54 +00:00
|
|
|
"github.com/host-uk/core/pkg/io"
|
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
|
|
|
)
|
|
|
|
|
|
2026-01-30 01:28:39 +00:00
|
|
|
// Impact-specific styles (aliases to shared)
|
2026-01-27 21:08:51 +00:00
|
|
|
var (
|
2026-01-30 10:32:05 +00:00
|
|
|
impactDirectStyle = cli.ErrorStyle
|
2026-01-31 23:36:43 +00:00
|
|
|
impactIndirectStyle = cli.WarningStyle
|
|
|
|
|
impactSafeStyle = cli.SuccessStyle
|
2026-01-27 21:08:51 +00:00
|
|
|
)
|
|
|
|
|
|
2026-01-30 00:47:54 +00:00
|
|
|
// Impact command flags
|
|
|
|
|
var impactRegistryPath string
|
2026-01-27 21:08:51 +00:00
|
|
|
|
2026-01-30 00:47:54 +00:00
|
|
|
// addImpactCommand adds the 'impact' command to the given parent command.
|
2026-01-31 11:39:19 +00:00
|
|
|
func addImpactCommand(parent *cli.Command) {
|
|
|
|
|
impactCmd := &cli.Command{
|
2026-01-30 00:47:54 +00:00
|
|
|
Use: "impact <repo-name>",
|
feat(i18n): add translation keys to all CLI commands
Replace hardcoded strings with i18n.T() calls across all cmd/* packages:
- ai, build, ci, dev, docs, doctor, go, php, pkg, sdk, setup, test, vm
Adds 500+ translation keys to en.json for command descriptions,
flag descriptions, labels, messages, and error strings.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 02:37:57 +00:00
|
|
|
Short: i18n.T("cmd.dev.impact.short"),
|
|
|
|
|
Long: i18n.T("cmd.dev.impact.long"),
|
2026-01-31 11:39:19 +00:00
|
|
|
Args: cli.ExactArgs(1),
|
|
|
|
|
RunE: func(cmd *cli.Command, args []string) error {
|
2026-01-30 00:47:54 +00:00
|
|
|
return runImpact(impactRegistryPath, args[0])
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
refactor(i18n): consolidate duplicate translation keys into common section
Add common.* keys for reusable translations:
- common.label.* - UI labels (error, done, status, version, etc.)
- common.status.* - status words (running, stopped, dirty, synced)
- common.error.* - error messages (failed, not_found, working_dir)
- common.flag.* - CLI flag descriptions (registry, verbose, etc.)
- common.count.* - count templates (failed, passed, skipped)
- common.result.* - result messages (all_passed, no_issues)
- common.progress.* - progress messages (running, checking)
- common.hint.* - help hints (install_with, fix_deps)
Update all cmd/* files to use common keys instead of duplicated
command-specific keys. Reduces translation maintenance burden
and ensures consistency across the CLI.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 11:32:25 +00:00
|
|
|
impactCmd.Flags().StringVar(&impactRegistryPath, "registry", "", i18n.T("common.flag.registry"))
|
2026-01-30 00:47:54 +00:00
|
|
|
|
|
|
|
|
parent.AddCommand(impactCmd)
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func runImpact(registryPath string, repoName string) error {
|
|
|
|
|
// Find or use provided registry
|
|
|
|
|
var reg *repos.Registry
|
|
|
|
|
var err error
|
|
|
|
|
|
|
|
|
|
if registryPath != "" {
|
2026-02-04 18:03:54 +00:00
|
|
|
reg, err = repos.LoadRegistry(io.Local, registryPath)
|
2026-01-27 21:08:51 +00:00
|
|
|
if err != nil {
|
2026-01-31 11:39:19 +00:00
|
|
|
return cli.Wrap(err, "failed to load registry")
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
|
|
|
|
} else {
|
2026-02-04 18:03:54 +00:00
|
|
|
registryPath, err = repos.FindRegistry(io.Local)
|
2026-01-27 21:08:51 +00:00
|
|
|
if err == nil {
|
2026-02-04 18:03:54 +00:00
|
|
|
reg, err = repos.LoadRegistry(io.Local, registryPath)
|
2026-01-27 21:08:51 +00:00
|
|
|
if err != nil {
|
2026-01-31 11:39:19 +00:00
|
|
|
return cli.Wrap(err, "failed to load registry")
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
|
|
|
|
} else {
|
refactor(cli): move commands from cmd/ to pkg/ with self-registration
Implements defence in depth through build variants - only compiled code
exists in the binary. Commands now self-register via cli.RegisterCommands()
in their init() functions, mirroring the i18n.RegisterLocales() pattern.
Structure changes:
- cmd/{ai,build,ci,dev,docs,doctor,go,php,pkg,sdk,setup,test,vm}/ → pkg/*/cmd_*.go
- cmd/core_dev.go, cmd/core_ci.go → cmd/variants/{full,ci,php,minimal}.go
- Added pkg/cli/commands.go with RegisterCommands API
- Updated pkg/cli/runtime.go to attach registered commands
Build variants:
- go build → full (21MB, all 13 command groups)
- go build -tags ci → ci (18MB, build/ci/sdk/doctor)
- go build -tags php → php (14MB, php/doctor)
- go build -tags minimal → minimal (11MB, doctor only)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 21:55:55 +00:00
|
|
|
return errors.New(i18n.T("cmd.dev.impact.requires_registry"))
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check repo exists
|
|
|
|
|
repo, exists := reg.Get(repoName)
|
|
|
|
|
if !exists {
|
refactor(cli): move commands from cmd/ to pkg/ with self-registration
Implements defence in depth through build variants - only compiled code
exists in the binary. Commands now self-register via cli.RegisterCommands()
in their init() functions, mirroring the i18n.RegisterLocales() pattern.
Structure changes:
- cmd/{ai,build,ci,dev,docs,doctor,go,php,pkg,sdk,setup,test,vm}/ → pkg/*/cmd_*.go
- cmd/core_dev.go, cmd/core_ci.go → cmd/variants/{full,ci,php,minimal}.go
- Added pkg/cli/commands.go with RegisterCommands API
- Updated pkg/cli/runtime.go to attach registered commands
Build variants:
- go build → full (21MB, all 13 command groups)
- go build -tags ci → ci (18MB, build/ci/sdk/doctor)
- go build -tags php → php (14MB, php/doctor)
- go build -tags minimal → minimal (11MB, doctor only)
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 21:55:55 +00:00
|
|
|
return errors.New(i18n.T("error.repo_not_found", map[string]interface{}{"Name": repoName}))
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 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
|
2026-01-31 23:36:43 +00:00
|
|
|
cli.Blank()
|
2026-01-31 11:39:19 +00:00
|
|
|
cli.Print("%s %s\n", dimStyle.Render(i18n.T("cmd.dev.impact.analysis_for")), repoNameStyle.Render(repoName))
|
2026-01-27 21:08:51 +00:00
|
|
|
if repo.Description != "" {
|
2026-01-31 11:39:19 +00:00
|
|
|
cli.Print("%s\n", dimStyle.Render(repo.Description))
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
2026-01-31 23:36:43 +00:00
|
|
|
cli.Blank()
|
2026-01-27 21:08:51 +00:00
|
|
|
|
|
|
|
|
if len(allAffected) == 0 {
|
2026-01-31 11:39:19 +00:00
|
|
|
cli.Print("%s %s\n", impactSafeStyle.Render("v"), i18n.T("cmd.dev.impact.no_dependents", map[string]interface{}{"Name": repoName}))
|
2026-01-27 21:08:51 +00:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Direct dependents
|
|
|
|
|
if len(direct) > 0 {
|
2026-01-31 11:39:19 +00:00
|
|
|
cli.Print("%s %s\n",
|
2026-01-30 00:22:47 +00:00
|
|
|
impactDirectStyle.Render("*"),
|
feat(i18n): add translation keys to all CLI commands
Replace hardcoded strings with i18n.T() calls across all cmd/* packages:
- ai, build, ci, dev, docs, doctor, go, php, pkg, sdk, setup, test, vm
Adds 500+ translation keys to en.json for command descriptions,
flag descriptions, labels, messages, and error strings.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 02:37:57 +00:00
|
|
|
i18n.T("cmd.dev.impact.direct_dependents", map[string]interface{}{"Count": len(direct)}),
|
2026-01-27 21:08:51 +00:00
|
|
|
)
|
|
|
|
|
for _, d := range direct {
|
|
|
|
|
r, _ := reg.Get(d)
|
|
|
|
|
desc := ""
|
|
|
|
|
if r != nil && r.Description != "" {
|
2026-01-30 10:32:05 +00:00
|
|
|
desc = dimStyle.Render(" - " + cli.Truncate(r.Description, 40))
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
2026-01-31 11:39:19 +00:00
|
|
|
cli.Print(" %s%s\n", d, desc)
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
2026-01-31 23:36:43 +00:00
|
|
|
cli.Blank()
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Indirect dependents
|
|
|
|
|
if len(indirect) > 0 {
|
2026-01-31 11:39:19 +00:00
|
|
|
cli.Print("%s %s\n",
|
2026-01-30 00:22:47 +00:00
|
|
|
impactIndirectStyle.Render("o"),
|
feat(i18n): add translation keys to all CLI commands
Replace hardcoded strings with i18n.T() calls across all cmd/* packages:
- ai, build, ci, dev, docs, doctor, go, php, pkg, sdk, setup, test, vm
Adds 500+ translation keys to en.json for command descriptions,
flag descriptions, labels, messages, and error strings.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 02:37:57 +00:00
|
|
|
i18n.T("cmd.dev.impact.transitive_dependents", map[string]interface{}{"Count": len(indirect)}),
|
2026-01-27 21:08:51 +00:00
|
|
|
)
|
|
|
|
|
for _, d := range indirect {
|
|
|
|
|
r, _ := reg.Get(d)
|
|
|
|
|
desc := ""
|
|
|
|
|
if r != nil && r.Description != "" {
|
2026-01-30 10:32:05 +00:00
|
|
|
desc = dimStyle.Render(" - " + cli.Truncate(r.Description, 40))
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
2026-01-31 11:39:19 +00:00
|
|
|
cli.Print(" %s%s\n", d, desc)
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
2026-01-31 23:36:43 +00:00
|
|
|
cli.Blank()
|
2026-01-27 21:08:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Summary
|
2026-01-31 11:39:19 +00:00
|
|
|
cli.Print("%s %s\n",
|
refactor(i18n): migrate all pkg/* to grammar engine
Replace verbose map-based translation calls with concise grammar
engine helpers across all command packages:
- i18n.T("common.label.xxx") → i18n.Label("xxx")
- i18n.T("common.error.failed", map) → i18n.T("i18n.fail.verb", subj)
- i18n.T("common.progress.running", map) → i18n.ProgressSubject()
- i18n.T("common.count.xxx", map) → i18n.T("i18n.count.xxx", n)
Packages updated: ai, ci, dev, docs, php, pkgcmd, sdk, setup, test, vm
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 23:09:45 +00:00
|
|
|
dimStyle.Render(i18n.Label("summary")),
|
feat(i18n): add translation keys to all CLI commands
Replace hardcoded strings with i18n.T() calls across all cmd/* packages:
- ai, build, ci, dev, docs, doctor, go, php, pkg, sdk, setup, test, vm
Adds 500+ translation keys to en.json for command descriptions,
flag descriptions, labels, messages, and error strings.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-30 02:37:57 +00:00
|
|
|
i18n.T("cmd.dev.impact.changes_affect", map[string]interface{}{
|
|
|
|
|
"Repo": repoNameStyle.Render(repoName),
|
|
|
|
|
"Affected": len(allAffected),
|
|
|
|
|
"Total": len(reg.Repos) - 1,
|
|
|
|
|
}),
|
2026-01-27 21:08:51 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|