mirror of
https://github.com/Tencent/WeKnora.git
synced 2026-06-04 13:30:32 +08:00
cli/acceptance/contract/:
envelope_test.go — 16 envelope golden cases (9 commands × {success/error
variants}; 3 cases dropped with rationale: doctor.success
non-offline has unstable timing detail; auth_login.* needs
stdin/keyring scaffold deferred to v0.2; context_use.error
needs leaf-local --json deferred to follow-up)
errorcodes_test.go — single-direction AST scan of cli/cmd/ extracting first
arg of cmdutil.NewError / cmdutil.Wrapf calls;
ClassifyHTTPError dynamic-classify bridged via
cmdutil.ClassifyHTTPErrorOutputs() per spec §4.3.
testdata/envelopes/ — 16 JSON golden files
helpers_test.go (PR-6 scaffold) extended:
runCmd now wires cobra Out/Err sinks (version uses c.OutOrStdout) AND
replicates cmd.Execute()'s error-envelope path so error-case goldens are
populated. Without this, every error scenario's golden was 0 bytes.
cli/cmd/root.go: mapCobraError → MapCobraError, wantsJSONOutput → WantsJSONOutput
(exported so the contract test helper can replicate Execute()'s
envelope-printing path without calling Execute() itself).
root_test.go updated to use new exported names.
.github/dependabot.yml (新增):gomod /cli + github-actions weekly,gh-style
ignore semver-major to avoid noise. Open-source
dependency safety,independent of release cadence.
v0.1 不发布到任何分发平台 (release infra 推迟到发布窗口 milestone)。
238 lines
8.9 KiB
Go
238 lines
8.9 KiB
Go
// Package cmd holds the cobra command tree. main.go calls Execute().
|
|
//
|
|
// v0.0 shipped: version / auth / search.
|
|
// v0.1 adds: whoami / doctor / kb (list + get) / context (use).
|
|
package cmd
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"github.com/Tencent/WeKnora/cli/cmd/auth"
|
|
contextcmd "github.com/Tencent/WeKnora/cli/cmd/context"
|
|
"github.com/Tencent/WeKnora/cli/cmd/doctor"
|
|
"github.com/Tencent/WeKnora/cli/cmd/kb"
|
|
"github.com/Tencent/WeKnora/cli/cmd/search"
|
|
"github.com/Tencent/WeKnora/cli/cmd/whoami"
|
|
"github.com/Tencent/WeKnora/cli/internal/agent"
|
|
"github.com/Tencent/WeKnora/cli/internal/build"
|
|
"github.com/Tencent/WeKnora/cli/internal/cmdutil"
|
|
"github.com/Tencent/WeKnora/cli/internal/format"
|
|
"github.com/Tencent/WeKnora/cli/internal/iostreams"
|
|
)
|
|
|
|
// Execute is the entry point invoked by main(). Returns the process exit code.
|
|
func Execute() int {
|
|
root := NewRootCmd(cmdutil.New())
|
|
// ExecuteC returns the actually-invoked leaf (or root when invocation
|
|
// failed before dispatch); we use it to honor the leaf's --json and
|
|
// inherited --format without walking the tree ourselves.
|
|
cmd, err := root.ExecuteC()
|
|
if err == nil {
|
|
return 0
|
|
}
|
|
err = MapCobraError(err)
|
|
if agent.ShouldUseAgentMode(cmd) || WantsJSONOutput(cmd) {
|
|
cmdutil.PrintErrorEnvelope(iostreams.IO.Out, err)
|
|
} else {
|
|
cmdutil.PrintError(iostreams.IO.Err, err)
|
|
}
|
|
return cmdutil.ExitCode(err)
|
|
}
|
|
|
|
// WantsJSONOutput reports whether cmd was invoked with --json, so error
|
|
// output matches the success format. Persistent flags inherit automatically
|
|
// via cmd.Flags().
|
|
//
|
|
// Falls back to scanning os.Args when cobra never reached the leaf — e.g.
|
|
// unknown subcommand or unknown flag at root level. Without this, `weknora
|
|
// bogus --json` would emit a human stderr line instead of the envelope the
|
|
// agent asked for.
|
|
//
|
|
// Exported so the acceptance/contract test helper can replicate Execute()'s
|
|
// envelope-printing path without having to call os.Exit-bound Execute() itself.
|
|
func WantsJSONOutput(cmd *cobra.Command) bool {
|
|
if v, err := cmd.Flags().GetBool("json"); err == nil && v {
|
|
return true
|
|
}
|
|
return argsRequestJSON(os.Args[1:])
|
|
}
|
|
|
|
// argsRequestJSON scans a flag-only slice for --json in the forms pflag
|
|
// accepts. Used as a fallback when cobra short-circuits before flag parsing
|
|
// (unknown command / unknown flag at root). Mirrors only the subset of pflag
|
|
// bool parsing relevant here — `--json=false` is treated as not-JSON,
|
|
// matching pflag.
|
|
func argsRequestJSON(args []string) bool {
|
|
for _, a := range args {
|
|
switch {
|
|
case a == "--json":
|
|
return true
|
|
case strings.HasPrefix(a, "--json="):
|
|
if isPflagTruthy(strings.TrimPrefix(a, "--json=")) {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// isPflagTruthy mirrors pflag's bool parsing for "--flag=<v>" tokens.
|
|
// pflag delegates to strconv.ParseBool, which accepts 1/t/T/TRUE/true/True
|
|
// as truthy and 0/f/F/FALSE/false/False as falsy. Anything else errors.
|
|
func isPflagTruthy(v string) bool {
|
|
b, err := strconv.ParseBool(v)
|
|
return err == nil && b
|
|
}
|
|
|
|
// MapCobraError tags the textually-emitted cobra errors as cmdutil.FlagError
|
|
// so they exit 2 like other user invocation mistakes. SetFlagErrorFunc handles
|
|
// flag parse errors at parse time; this catches positional/Args validation
|
|
// errors and unknown subcommands that propagate as plain errors.
|
|
//
|
|
// Pinned to cobra v1.10 message formats (cobra/args.go: ExactArgs / NoArgs;
|
|
// cobra/command.go: required-flag / unknown-command). TestMapCobraError_PinnedPrefixes
|
|
// guards against a silent break on cobra bumps.
|
|
//
|
|
// Exported so the acceptance/contract test helper can reuse the mapping when
|
|
// replicating Execute()'s error-envelope path in-process.
|
|
func MapCobraError(err error) error {
|
|
if err == nil {
|
|
return nil
|
|
}
|
|
msg := err.Error()
|
|
for _, prefix := range cobraFlagErrorPrefixes {
|
|
if strings.HasPrefix(msg, prefix) {
|
|
return cmdutil.NewFlagError(err)
|
|
}
|
|
}
|
|
return err
|
|
}
|
|
|
|
// cobraFlagErrorPrefixes lists the text prefixes cobra uses for invocation
|
|
// problems we want to surface as exit 2. Pinned per cobra v1.10.
|
|
var cobraFlagErrorPrefixes = []string{
|
|
"unknown command ",
|
|
"required flag(s)",
|
|
"accepts ", // ExactArgs / RangeArgs / etc. — `accepts N arg(s), received M`
|
|
"requires at least", // MinimumNArgs
|
|
"requires at most", // MaximumNArgs
|
|
"unknown flag",
|
|
"invalid argument", // pflag type-coercion failure (e.g. --top-k=foo)
|
|
}
|
|
|
|
// NewRootCmd builds the cobra tree. Splitting it from Execute() lets tests
|
|
// drive the tree directly with their own factory. Exported (PR-7) so the
|
|
// acceptance/contract suite can construct the tree in-process.
|
|
func NewRootCmd(f *cmdutil.Factory) *cobra.Command {
|
|
v, commit, date := build.Info()
|
|
cmd := &cobra.Command{
|
|
Use: "weknora",
|
|
Short: "WeKnora CLI — RAG knowledge base from your terminal",
|
|
Long: `WeKnora CLI lets you authenticate, browse knowledge bases, and run
|
|
hybrid searches against a WeKnora server from your shell or an AI agent.`,
|
|
Example: ` weknora auth login --host=https://kb.example.com # one-time setup
|
|
weknora kb list # list knowledge bases
|
|
weknora kb get <id> # show one
|
|
weknora search "your question" --kb=<id> # hybrid retrieval
|
|
weknora doctor --json # health check (agent-readable)`,
|
|
SilenceUsage: true,
|
|
SilenceErrors: true,
|
|
// Version makes cobra auto-register a `--version` global flag that
|
|
// prints this string. Mainstream CLIs (gh / kubectl / aws / gcloud)
|
|
// all accept both `--version` and a `version` subcommand; the
|
|
// subcommand still owns the richer `--json` envelope output.
|
|
Version: fmt.Sprintf("%s (commit %s, built %s)", v, commit, date),
|
|
PersistentPreRun: func(c *cobra.Command, args []string) {
|
|
agent.ApplyAgentSugar(c)
|
|
// Propagate the global --context flag into the Factory for this
|
|
// invocation only. Spec §1.2: single-shot override, no disk write.
|
|
if v, _ := c.Flags().GetString("context"); v != "" {
|
|
f.ContextOverride = v
|
|
}
|
|
},
|
|
}
|
|
// Match `weknora version` line format so both forms output the same.
|
|
cmd.SetVersionTemplate("weknora {{.Version}}\n")
|
|
addGlobalFlags(cmd)
|
|
cmd.SetHelpFunc(agentAwareHelpFunc(cmd.HelpFunc()))
|
|
// Wrap cobra's flag-parsing errors as FlagError so cmdutil.ExitCode maps
|
|
// them to exit 2 (gh-style). "unknown command" errors are detected by
|
|
// message prefix in Execute() since cobra emits them as plain errors.
|
|
cmd.SetFlagErrorFunc(func(c *cobra.Command, err error) error {
|
|
return cmdutil.NewFlagError(err)
|
|
})
|
|
|
|
cmd.AddCommand(newVersionCmd(f))
|
|
cmd.AddCommand(auth.NewCmdAuth(f))
|
|
cmd.AddCommand(search.NewCmdSearch(f))
|
|
cmd.AddCommand(whoami.NewCmd(f))
|
|
cmd.AddCommand(doctor.NewCmd(f))
|
|
cmd.AddCommand(kb.NewCmd(f))
|
|
cmd.AddCommand(contextcmd.NewCmd(f))
|
|
return cmd
|
|
}
|
|
|
|
// addGlobalFlags registers persistent flags available on every subcommand.
|
|
// Only flags whose behavior is actually wired are listed — a flag that
|
|
// accepts values but does nothing is a worse contract than no flag.
|
|
//
|
|
// --context lands here in v0.1 (spec §1.2); --no-version-check waits for
|
|
// v0.7's compat probe consumer.
|
|
func addGlobalFlags(cmd *cobra.Command) {
|
|
pf := cmd.PersistentFlags()
|
|
pf.Bool("agent", false, "Agent mode: envelope JSON output + no interactive prompts + no progress UI")
|
|
pf.Bool("no-interactive", false, "Refuse interactive prompts; missing input becomes a hard error")
|
|
pf.Bool("no-progress", false, "Suppress progress bars and spinners")
|
|
pf.BoolP("yes", "y", false, "Skip confirmation prompts on destructive operations")
|
|
pf.String("context", "", "Override the active context for this invocation (no disk write)")
|
|
}
|
|
|
|
// agentAwareHelpFunc wraps cobra's default help to append the AI agent guidance
|
|
// (Annotations[agent.AIAgentHelpKey]) only when agent mode is active.
|
|
// Stripe pkg/cmd/templates.go pattern.
|
|
func agentAwareHelpFunc(orig func(*cobra.Command, []string)) func(*cobra.Command, []string) {
|
|
return func(c *cobra.Command, args []string) {
|
|
orig(c, args)
|
|
if !agent.ShouldUseAgentMode(c) {
|
|
return
|
|
}
|
|
extra := agent.FormatAgentGuidance(c)
|
|
if extra == "" {
|
|
return
|
|
}
|
|
w := c.OutOrStdout()
|
|
fmt.Fprintln(w)
|
|
fmt.Fprintln(w, "AI Agent guidance:")
|
|
fmt.Fprintln(w, " "+extra)
|
|
}
|
|
}
|
|
|
|
// newVersionCmd is the only leaf command shipped in the foundation PR. It
|
|
// doubles as the smoke test that proves Factory + iostreams + cobra wiring works.
|
|
func newVersionCmd(f *cmdutil.Factory) *cobra.Command {
|
|
var jsonOut bool
|
|
cmd := &cobra.Command{
|
|
Use: "version",
|
|
Short: "Show CLI build metadata",
|
|
RunE: func(c *cobra.Command, args []string) error {
|
|
v, commit, date := build.Info()
|
|
if jsonOut {
|
|
return cmdutil.NewJSONExporter().Write(c.OutOrStdout(), format.Success(map[string]string{
|
|
"version": v,
|
|
"commit": commit,
|
|
"date": date,
|
|
}, nil))
|
|
}
|
|
fmt.Fprintf(c.OutOrStdout(), "weknora %s (commit %s, built %s)\n", v, commit, date)
|
|
return nil
|
|
},
|
|
}
|
|
cmd.Flags().BoolVar(&jsonOut, "json", false, "Output JSON envelope")
|
|
return cmd
|
|
}
|