Move source from go/core/ to root, tests from go/tests/ to tests/. Module path dappco.re/go/core resolves cleanly — builds and tests pass. Co-Authored-By: Virgil <virgil@lethean.io>
169 lines
3.5 KiB
Go
169 lines
3.5 KiB
Go
// SPDX-License-Identifier: EUPL-1.2
|
|
|
|
// Cli is the CLI surface layer for the Core command tree.
|
|
// It reads commands from Core's registry and wires them to terminal I/O.
|
|
//
|
|
// Run the CLI:
|
|
//
|
|
// c := core.New(core.Options{{Key: "name", Value: "myapp"}})
|
|
// c.Command("deploy", handler)
|
|
// c.Cli().Run()
|
|
//
|
|
// The Cli resolves os.Args to a command path, parses flags,
|
|
// and calls the command's action with parsed options.
|
|
package core
|
|
|
|
import (
|
|
"io"
|
|
"os"
|
|
)
|
|
|
|
// Cli is the CLI surface for the Core command tree.
|
|
type Cli struct {
|
|
core *Core
|
|
output io.Writer
|
|
banner func(*Cli) string
|
|
}
|
|
|
|
// Print writes to the CLI output (defaults to os.Stdout).
|
|
//
|
|
// c.Cli().Print("hello %s", "world")
|
|
func (cl *Cli) Print(format string, args ...any) {
|
|
Print(cl.output, format, args...)
|
|
}
|
|
|
|
// SetOutput sets the CLI output writer.
|
|
//
|
|
// c.Cli().SetOutput(os.Stderr)
|
|
func (cl *Cli) SetOutput(w io.Writer) {
|
|
cl.output = w
|
|
}
|
|
|
|
// Run resolves os.Args to a command path and executes it.
|
|
//
|
|
// c.Cli().Run()
|
|
// c.Cli().Run("deploy", "to", "homelab")
|
|
func (cl *Cli) Run(args ...string) Result {
|
|
if len(args) == 0 {
|
|
args = os.Args[1:]
|
|
}
|
|
|
|
clean := FilterArgs(args)
|
|
|
|
if cl.core == nil || cl.core.commands == nil {
|
|
if cl.banner != nil {
|
|
cl.Print(cl.banner(cl))
|
|
}
|
|
return Result{}
|
|
}
|
|
|
|
cl.core.commands.mu.RLock()
|
|
cmdCount := len(cl.core.commands.commands)
|
|
cl.core.commands.mu.RUnlock()
|
|
|
|
if cmdCount == 0 {
|
|
if cl.banner != nil {
|
|
cl.Print(cl.banner(cl))
|
|
}
|
|
return Result{}
|
|
}
|
|
|
|
// Resolve command path from args
|
|
var cmd *Command
|
|
var remaining []string
|
|
|
|
cl.core.commands.mu.RLock()
|
|
for i := len(clean); i > 0; i-- {
|
|
path := JoinPath(clean[:i]...)
|
|
if c, ok := cl.core.commands.commands[path]; ok {
|
|
cmd = c
|
|
remaining = clean[i:]
|
|
break
|
|
}
|
|
}
|
|
cl.core.commands.mu.RUnlock()
|
|
|
|
if cmd == nil {
|
|
if cl.banner != nil {
|
|
cl.Print(cl.banner(cl))
|
|
}
|
|
cl.PrintHelp()
|
|
return Result{}
|
|
}
|
|
|
|
// Build options from remaining args
|
|
opts := Options{}
|
|
for _, arg := range remaining {
|
|
key, val, valid := ParseFlag(arg)
|
|
if valid {
|
|
if Contains(arg, "=") {
|
|
opts = append(opts, Option{Key: key, Value: val})
|
|
} else {
|
|
opts = append(opts, Option{Key: key, Value: true})
|
|
}
|
|
} else if !IsFlag(arg) {
|
|
opts = append(opts, Option{Key: "_arg", Value: arg})
|
|
}
|
|
}
|
|
|
|
if cmd.Action != nil {
|
|
return cmd.Run(opts)
|
|
}
|
|
if cmd.Lifecycle != nil {
|
|
return cmd.Start(opts)
|
|
}
|
|
return Result{E("core.Cli.Run", Concat("command \"", cmd.Path, "\" is not executable"), nil), false}
|
|
}
|
|
|
|
// PrintHelp prints available commands.
|
|
//
|
|
// c.Cli().PrintHelp()
|
|
func (cl *Cli) PrintHelp() {
|
|
if cl.core == nil || cl.core.commands == nil {
|
|
return
|
|
}
|
|
|
|
name := ""
|
|
if cl.core.app != nil {
|
|
name = cl.core.app.Name
|
|
}
|
|
if name != "" {
|
|
cl.Print("%s commands:", name)
|
|
} else {
|
|
cl.Print("Commands:")
|
|
}
|
|
|
|
cl.core.commands.mu.RLock()
|
|
defer cl.core.commands.mu.RUnlock()
|
|
|
|
for path, cmd := range cl.core.commands.commands {
|
|
if cmd.Hidden || (cmd.Action == nil && cmd.Lifecycle == nil) {
|
|
continue
|
|
}
|
|
tr := cl.core.I18n().Translate(cmd.I18nKey())
|
|
desc, _ := tr.Value.(string)
|
|
if desc == "" || desc == cmd.I18nKey() {
|
|
cl.Print(" %s", path)
|
|
} else {
|
|
cl.Print(" %-30s %s", path, desc)
|
|
}
|
|
}
|
|
}
|
|
|
|
// SetBanner sets the banner function.
|
|
//
|
|
// c.Cli().SetBanner(func(_ *core.Cli) string { return "My App v1.0" })
|
|
func (cl *Cli) SetBanner(fn func(*Cli) string) {
|
|
cl.banner = fn
|
|
}
|
|
|
|
// Banner returns the banner string.
|
|
func (cl *Cli) Banner() string {
|
|
if cl.banner != nil {
|
|
return cl.banner(cl)
|
|
}
|
|
if cl.core != nil && cl.core.app != nil && cl.core.app.Name != "" {
|
|
return cl.core.app.Name
|
|
}
|
|
return ""
|
|
}
|