Split all cmd/* packages for maintainability, following the pattern established in cmd/php. Each package now has: - Main file with styles (using cmd/shared) and Add*Commands function - Separate files for logical command groupings Packages refactored: - cmd/dev: 13 files (was 2779 lines in one file) - cmd/build: 5 files (was 913 lines) - cmd/setup: 6 files (was 961 lines) - cmd/go: 5 files (was 655 lines) - cmd/pkg: 5 files (was 634 lines) - cmd/vm: 4 files (was 717 lines) - cmd/ai: 5 files (was 800 lines) - cmd/docs: 5 files (was 379 lines) - cmd/doctor: 5 files (was 301 lines) - cmd/test: 3 files (was 429 lines) - cmd/ci: 5 files (was 272 lines) All packages now import shared styles from cmd/shared instead of redefining them locally. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
165 lines
4.6 KiB
Go
165 lines
4.6 KiB
Go
// setup_bootstrap.go implements bootstrap mode for new workspaces.
|
|
//
|
|
// Bootstrap mode is activated when no repos.yaml exists in the current
|
|
// directory or any parent. It clones core-devops first, then uses its
|
|
// repos.yaml to present the package wizard.
|
|
|
|
package setup
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"github.com/host-uk/core/pkg/repos"
|
|
)
|
|
|
|
// runSetupOrchestrator decides between registry mode and bootstrap mode.
|
|
func runSetupOrchestrator(registryPath, only string, dryRun, all bool, projectName string, runBuild bool) error {
|
|
ctx := context.Background()
|
|
|
|
// Try to find an existing registry
|
|
var foundRegistry string
|
|
var err error
|
|
|
|
if registryPath != "" {
|
|
foundRegistry = registryPath
|
|
} else {
|
|
foundRegistry, err = repos.FindRegistry()
|
|
}
|
|
|
|
// If registry exists, use registry mode
|
|
if err == nil && foundRegistry != "" {
|
|
return runRegistrySetup(ctx, foundRegistry, only, dryRun, all, runBuild)
|
|
}
|
|
|
|
// No registry found - enter bootstrap mode
|
|
return runBootstrap(ctx, only, dryRun, all, projectName, runBuild)
|
|
}
|
|
|
|
// runBootstrap handles the case where no repos.yaml exists.
|
|
func runBootstrap(ctx context.Context, only string, dryRun, all bool, projectName string, runBuild bool) error {
|
|
cwd, err := os.Getwd()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get working directory: %w", err)
|
|
}
|
|
|
|
fmt.Printf("%s Bootstrap mode (no repos.yaml found)\n", dimStyle.Render(">>"))
|
|
|
|
var targetDir string
|
|
|
|
// Check if current directory is empty
|
|
empty, err := isDirEmpty(cwd)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to check directory: %w", err)
|
|
}
|
|
|
|
if empty {
|
|
// Clone into current directory
|
|
targetDir = cwd
|
|
fmt.Printf("%s Cloning into current directory\n", dimStyle.Render(">>"))
|
|
} else {
|
|
// Directory has content - check if it's a git repo root
|
|
isRepo := isGitRepoRoot(cwd)
|
|
|
|
if isRepo && isTerminal() && !all {
|
|
// Offer choice: setup working directory or create package
|
|
choice, err := promptSetupChoice()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get choice: %w", err)
|
|
}
|
|
|
|
if choice == "setup" {
|
|
// Setup this working directory with .core/ config
|
|
return runRepoSetup(cwd, dryRun)
|
|
}
|
|
// Otherwise continue to "create package" flow
|
|
}
|
|
|
|
// Create package flow - need a project name
|
|
if projectName == "" {
|
|
if !isTerminal() || all {
|
|
projectName = defaultOrg
|
|
} else {
|
|
projectName, err = promptProjectName(defaultOrg)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to get project name: %w", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
targetDir = filepath.Join(cwd, projectName)
|
|
fmt.Printf("%s Creating project directory: %s\n", dimStyle.Render(">>"), projectName)
|
|
|
|
if !dryRun {
|
|
if err := os.MkdirAll(targetDir, 0755); err != nil {
|
|
return fmt.Errorf("failed to create directory: %w", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Clone core-devops first
|
|
devopsPath := filepath.Join(targetDir, devopsRepo)
|
|
if _, err := os.Stat(filepath.Join(devopsPath, ".git")); os.IsNotExist(err) {
|
|
fmt.Printf("%s Cloning %s...\n", dimStyle.Render(">>"), devopsRepo)
|
|
|
|
if !dryRun {
|
|
if err := gitClone(ctx, defaultOrg, devopsRepo, devopsPath); err != nil {
|
|
return fmt.Errorf("failed to clone %s: %w", devopsRepo, err)
|
|
}
|
|
fmt.Printf("%s %s cloned\n", successStyle.Render(">>"), devopsRepo)
|
|
} else {
|
|
fmt.Printf(" Would clone %s/%s to %s\n", defaultOrg, devopsRepo, devopsPath)
|
|
}
|
|
} else {
|
|
fmt.Printf("%s %s already exists\n", dimStyle.Render(">>"), devopsRepo)
|
|
}
|
|
|
|
// Load the repos.yaml from core-devops
|
|
registryPath := filepath.Join(devopsPath, devopsReposYaml)
|
|
|
|
if dryRun {
|
|
fmt.Printf("\n%s Would load registry from %s and present package wizard\n", dimStyle.Render(">>"), registryPath)
|
|
return nil
|
|
}
|
|
|
|
reg, err := repos.LoadRegistry(registryPath)
|
|
if err != nil {
|
|
return fmt.Errorf("failed to load registry from %s: %w", devopsRepo, err)
|
|
}
|
|
|
|
// Override base path to target directory
|
|
reg.BasePath = targetDir
|
|
|
|
// Now run the regular setup with the loaded registry
|
|
return runRegistrySetupWithReg(ctx, reg, registryPath, only, dryRun, all, runBuild)
|
|
}
|
|
|
|
// isGitRepoRoot returns true if the directory is a git repository root.
|
|
func isGitRepoRoot(path string) bool {
|
|
_, err := os.Stat(filepath.Join(path, ".git"))
|
|
return err == nil
|
|
}
|
|
|
|
// isDirEmpty returns true if the directory is empty or contains only hidden files.
|
|
func isDirEmpty(path string) (bool, error) {
|
|
entries, err := os.ReadDir(path)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
for _, e := range entries {
|
|
name := e.Name()
|
|
// Ignore common hidden/metadata files
|
|
if name == ".DS_Store" || name == ".git" || name == ".gitignore" {
|
|
continue
|
|
}
|
|
// Any other non-hidden file means directory is not empty
|
|
if len(name) > 0 && name[0] != '.' {
|
|
return false, nil
|
|
}
|
|
}
|
|
|
|
return true, nil
|
|
}
|