mirror of
https://github.com/Tencent/WeKnora.git
synced 2026-06-04 13:30:32 +08:00
Post-review polish on the v0.7 wire / surface contract. Bundles five
follow-ups that landed after the main BREAKING feat commit:
1. Complete context→profile cascade (internal API + YAML schema)
The prior commit renamed only the user-visible surface (commands /
flags / env / project link / envelope field). The internal Go API
and on-disk config schema were still half-renamed — an L-25
self-consistency violation flagged by post-merge review. Closed here:
Internal Go API:
- config.Context → config.Profile
- config.Config.CurrentContext → CurrentProfile
- config.Config.Contexts → Profiles
- LoginOptions.Context → LoginOptions.Profile
- clearContextSecrets() → clearProfileSecrets()
- saveContextRef() → saveProfileRef()
- secrets.Store: param name `context` → `profile` (interface +
FileStore + KeyringStore + MemStore)
- cmdutil.LoadSecret(store, context, key) → LoadSecret(store, profile, key)
- cmdutil.RefreshAndPersist's ctxName → profileName
- Local var `ctx := &config.Profile{...}` → `prof := &config.Profile{...}`
in auth/login.go to eliminate the visual collision with Go stdlib
context.Context that motivated the whole rename in the first place.
On-disk config.yaml schema:
- current_context: → current_profile:
- contexts: → profiles:
- Pre-1.0 break, no compat alias. Users on v0.6 dogfooded configs
must delete ~/.config/weknora/config.yaml or hand-rename the two
keys (CHANGELOG migration note added).
Tests / fixtures / golden files:
- factory_test.go YAML fixture + assertion updated.
- acceptance/e2e/e2e_test.go writeContextYAML → writeProfileYAML,
fixture YAML keys updated.
- acceptance/testdata/wire/doctor.error_network.json golden updated
("active context" → "active profile" in hint string).
User-visible prose sweep:
- cmd/mcp/serve.go --help Long: "active context (or --context)" →
"active profile (or --profile)" — most-visible miss.
- cmd/{kb/list, search/kb, session/list, api/api} Short/Long help.
- cmd/auth/login.go stdout: `(context=%s)` → `(profile=%s)`.
- cmd/auth/logout.go error: `"no current context"` → `"no current profile"`.
- cmd/doctor/doctor.go hint string (also the wire golden above).
- cmd/auth/refresh.go error: `"refresh token missing for context"` →
`"refresh token missing for profile"`.
- README.md: `## Multi-context` H2 → `## Multi-profile`; code-block
comment `# current context` → `# current profile`.
Code-comment / docstring sweep across cli/cmd/auth/ and
cli/internal/cmdutil/. Comments referencing Go stdlib context.Context,
the RAG / LLM "context window" concept, and historical CHANGELOG
entries for v0.4 / v0.5 were left alone.
CHANGELOG v0.7 BREAKING entry gains the on-disk-schema bullet under
the existing "context → profile" item.
2. Profile name validation (shell-injection guard)
`envelope.error.retry_command` is a single shell-string field. An
AI agent that exec()s it via `sh -c <retry_command>` was injectable
through a maliciously-named profile:
weknora auth logout --name 'x; rm -rf ~'
# would produce: retry_command = "weknora auth logout --name x; rm -rf ~ -y"
`cmd/profile/add.go` already enforced an alphanumeric + `-_.`
allowlist via `validateName`. The `auth login` and `auth logout`
paths bypassed it.
- Moved validation from `cmd/profile/add.go` to
`cli/internal/cmdutil/profilename.go` as exported
`ValidateProfileName` (cmdutil is the import-cycle-safe home;
internal/config can't depend on cmdutil).
- `auth login` runs the validator before any persist call.
- `auth logout` runs the validator on `opts.Name` before
constructing `retry_command`.
- Unit tests (`profilename_test.go`) cover the allowlist, empty
rejection, path-traversal, shell metacharacters (`;`, `&`, `|`,
`$()`, backticks, quotes, whitespace, glob, redirects), and the
user-facing hint text. The shell-metachar test exists as a
regression guard.
Wire shape (`retry_command` string → `retry_command_argv []string`)
remains a v0.8 additive change per ROADMAP — this fix removes the
practical exploit path without touching the wire contract.
3. AI-agent terminology disambiguation
"agent" has three referents in this codebase: (a) WeKnora's
server-side Custom Agent resource, (b) the removed `agent invoke`
verb, (c) external LLM/automation consumers. Per project memory
feedback_no_meta_disambiguation_in_docs, the fix is full-term
naming, not "X has N meanings" prose. Surgical changes at section
headers + ambiguous prose:
- AGENTS.md: "Agent decision shortcuts" → "AI agent decision
shortcuts"; "agent-callable surface" → "AI-agent-callable
surface".
- README.md: "Designed to be agent-first" → "AI-agent-first";
"Other agent ergonomics" → "Other AI-agent ergonomics"; "in
agent contexts" → "in AI-agent contexts"; "for CI / agents" →
"for CI / AI agents".
Anaphoric "agents" inside paragraphs that already established
"AI agents" was left alone — full substitution everywhere would
have been prose noise without clarity gain.
4. Wire-contract review follow-ups
Real findings from a second-pass review of the v0.7 envelope /
streaming / surface design. Per project memory
feedback_check_in_domain_anchor_first, candidate findings were
first verified against the in-domain peer CLI explicitly cited as
the envelope anchor; two earlier-flagged issues turned out to be
in-pattern and were withdrawn.
Surviving fixes:
- AGENTS.md success-envelope example rewritten. The prior example
showed `has_more: false` / `_notice: {}` as if they were always
present, but both fields are `omitempty` and never serialize
when zero / nil. Replaced with three realistic shapes (list /
single resource / mutation with no payload) and added a note
that optional fields are omitted when empty.
- cmd/chat/chat.go Args: MinimumNArgs(1) → ExactArgs(1).
v0.6 silently joined `weknora chat hello world` into
`"hello world"`. v0.7 now rejects multi-arg with exit 2,
matching `weknora session ask`. BREAKING; CHANGELOG entry
added under v0.7 BREAKING.
- internal/output/envelope.go extracts NewEnvelope(data, meta,
profile) constructor. The jq-filter path in
cmdutil.FormatOptions.Emit was manually rebuilding the
envelope literal alongside the canonical WriteEnvelope path —
drift risk when fields are added. Single construction point now.
- internal/cmdutil/factory.go adds AddKBFlag(cmd) helper.
Five files (chat, doc/list, doc/upload, doc/create, doc/fetch)
had verbatim-identical `cmd.Flags().String("kb", ...)`
declarations. Centralised so flag name + help text stay
in sync with Factory.ResolveKB. Docstring reordering + gofmt
fixup landed in the same edit to keep ResolveKB's own godoc
attached to its function.
5. OSS-readiness comment / doc sweep
Pre-publication scrub of code, comments, and shipped Markdown to
remove references that only make sense in the development repo:
- AGENTS.md "Deliberate deviations + mainstream alignments"
section: removed peer-project name-drops from the comparison
table; rewrote as five flagged design decisions with rationale
but no specific competitor named. The four rows that previously
contrasted against a named peer CLI now state WeKnora's choice
+ rationale directly. Section header renamed to "Design
decisions worth flagging" since it is no longer a
deviation/alignment matrix.
- CHANGELOG v0.7 BREAKING rationales: three references to a
named peer CLI removed; the context→profile rationale now
cites only mainstream multi-credential CLIs by category (AWS /
Stripe / OpenAI / Anthropic), and the `api -d/--data` removal
rationale cites only `gh api` / `curl`. `chat` BREAKING entry
rationale similarly simplified.
- 35 cross-references to design-spec section numbers (§4.1 /
§4.5 / §5.3 etc.) removed from Go doc comments and test
comments across 13 files. The referenced spec lives outside
the shipped tree; readers of the public repo cannot resolve
them. Each reference replaced with a self-contained semantic
description (e.g. "the batch envelope" / "AGENTS.md section
on the success path").
- Mixed-language strings translated to English:
- Four Go comments: internal/cmdutil/exit.go:213,215,
internal/cmdutil/errors.go:156,
internal/output/batch_test.go:90,
internal/output/envelope_test.go:27.
- One CHANGELOG section title:
`v0.7 — Agent-first wire contract + 命令面集中清理` →
`... + command-surface cleanup`.
- CJK test fixtures (internal/text/truncate_test.go CJK
truncation cases, cmd/session/list_test.go Chinese session
title, acceptance/e2e/e2e_test.go Chinese RAG corpus)
retained — they are intentional test inputs, not stray prose.
- Makefile help comment: `golangci-lint added in PR-9` →
`golangci-lint planned`. Internal PR numbering should not
surface in shipped Makefile prose.
Build green, 28/28 packages, +5 new ValidateProfileName tests.
go vet / gofmt / go mod verify / go mod tidy all clean.
Rationale for the cascade: pre-1.0 is the cheapest moment to close
L-25 self-consistency (L-26). The half-finished internal rename
would have perpetuated the very `context` vs `context.Context`
ambiguity that motivated v0.7's user-visible rename in the first
place.
418 lines
13 KiB
Go
418 lines
13 KiB
Go
package cmdutil
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/spf13/cobra"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/Tencent/WeKnora/cli/internal/config"
|
|
"github.com/Tencent/WeKnora/cli/internal/projectlink"
|
|
"github.com/Tencent/WeKnora/cli/internal/prompt"
|
|
"github.com/Tencent/WeKnora/cli/internal/secrets"
|
|
sdk "github.com/Tencent/WeKnora/client"
|
|
)
|
|
|
|
// TestFactory_Lazy ensures none of the closures execute work at construction
|
|
// time - `--help` / `completion` must not trigger HTTP / keyring access.
|
|
func TestFactory_Lazy(t *testing.T) {
|
|
var configCalls, clientCalls, prompterCalls int
|
|
f := &Factory{
|
|
Config: func() (*config.Config, error) {
|
|
configCalls++
|
|
return &config.Config{}, nil
|
|
},
|
|
Client: func() (*sdk.Client, error) {
|
|
clientCalls++
|
|
return nil, nil
|
|
},
|
|
Prompter: func() prompt.Prompter {
|
|
prompterCalls++
|
|
return prompt.AgentPrompter{}
|
|
},
|
|
}
|
|
// Asserting on closure presence - none should have run yet.
|
|
assert.Equal(t, 0, configCalls)
|
|
assert.Equal(t, 0, clientCalls)
|
|
assert.Equal(t, 0, prompterCalls)
|
|
// Smoke: each closure runs exactly once when called.
|
|
_, err := f.Config()
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 1, configCalls)
|
|
_, _ = f.Client()
|
|
assert.Equal(t, 1, clientCalls)
|
|
_ = f.Prompter()
|
|
assert.Equal(t, 1, prompterCalls)
|
|
}
|
|
|
|
// TestNew_FoundationDefaults verifies the production New() returns a usable
|
|
// Factory and that Client surfaces auth.unauthenticated when no current
|
|
// profile is configured (the precondition for `weknora auth login`).
|
|
func TestNew_FoundationDefaults(t *testing.T) {
|
|
t.Setenv("XDG_CONFIG_HOME", t.TempDir()) // empty config → no current profile
|
|
f := New()
|
|
require.NotNil(t, f)
|
|
require.NotNil(t, f.Config)
|
|
require.NotNil(t, f.Client)
|
|
require.NotNil(t, f.Prompter)
|
|
require.NotNil(t, f.Secrets)
|
|
|
|
_, err := f.Client()
|
|
require.Error(t, err)
|
|
var typed *Error
|
|
require.True(t, errors.As(err, &typed), "expected *cmdutil.Error")
|
|
assert.Equal(t, CodeAuthUnauthenticated, typed.Code)
|
|
}
|
|
|
|
// TestFactory_ProfileOverride verifies the global --profile flag mechanism:
|
|
// f.ProfileOverride replaces config.CurrentProfile for this invocation only,
|
|
// without writing to disk.
|
|
func TestFactory_ProfileOverride(t *testing.T) {
|
|
dir := t.TempDir()
|
|
t.Setenv("XDG_CONFIG_HOME", dir)
|
|
|
|
// Seed config with two profiles; CurrentProfile = "default"
|
|
cfgPath := dir + "/weknora/config.yaml"
|
|
require.NoError(t, os.MkdirAll(dir+"/weknora", 0o700))
|
|
require.NoError(t, os.WriteFile(cfgPath, []byte(`
|
|
current_profile: default
|
|
profiles:
|
|
default:
|
|
host: https://default.example
|
|
other:
|
|
host: https://other.example
|
|
`), 0o600))
|
|
|
|
f := New()
|
|
|
|
t.Run("no override: returns CurrentProfile from disk", func(t *testing.T) {
|
|
f.ProfileOverride = ""
|
|
cfg, err := f.Config()
|
|
require.NoError(t, err)
|
|
assert.Equal(t, "default", cfg.CurrentProfile)
|
|
})
|
|
|
|
t.Run("override applied: ProfileOverride wins over disk", func(t *testing.T) {
|
|
f.ProfileOverride = "other"
|
|
cfg, err := f.Config()
|
|
require.NoError(t, err)
|
|
assert.Equal(t, "other", cfg.CurrentProfile)
|
|
})
|
|
|
|
t.Run("override does not persist to disk", func(t *testing.T) {
|
|
// Reload from disk: should still be "default" (the original).
|
|
raw, err := os.ReadFile(cfgPath)
|
|
require.NoError(t, err)
|
|
assert.Contains(t, string(raw), "current_profile: default")
|
|
})
|
|
}
|
|
|
|
// TestTypedPredicates exercises the namespace and code matchers.
|
|
func TestTypedPredicates(t *testing.T) {
|
|
t.Run("IsAuthError matches auth.* prefix", func(t *testing.T) {
|
|
err := NewError(CodeAuthUnauthenticated, "no creds")
|
|
assert.True(t, IsAuthError(err))
|
|
assert.False(t, IsNotFound(err))
|
|
})
|
|
t.Run("IsNotFound matches resource.not_found exactly", func(t *testing.T) {
|
|
err := NewError(CodeResourceNotFound, "kb missing")
|
|
assert.True(t, IsNotFound(err))
|
|
assert.False(t, IsTransient(err))
|
|
})
|
|
t.Run("IsTransient matches network.* and server.timeout / rate_limited", func(t *testing.T) {
|
|
assert.True(t, IsTransient(NewError(CodeNetworkError, "")))
|
|
assert.True(t, IsTransient(NewError(CodeServerTimeout, "")))
|
|
assert.True(t, IsTransient(NewError(CodeServerRateLimited, "")))
|
|
assert.False(t, IsTransient(NewError(CodeServerError, "")))
|
|
})
|
|
t.Run("predicates walk the wrap chain", func(t *testing.T) {
|
|
inner := NewError(CodeAuthTokenExpired, "expired")
|
|
wrapped := Wrapf(CodeServerError, inner, "while calling foo")
|
|
// IsAuthExpired matches the wrapped *Error first; outer Wrapf has
|
|
// CodeServerError so the predicate returns false on the outer match.
|
|
// This documents current behavior: predicates report the first *Error
|
|
// in the chain, not deep walks.
|
|
assert.False(t, IsAuthExpired(wrapped))
|
|
// Direct match works.
|
|
assert.True(t, IsAuthExpired(inner))
|
|
})
|
|
t.Run("non-typed errors are never matched", func(t *testing.T) {
|
|
assert.False(t, IsAuthError(errors.New("plain error")))
|
|
assert.False(t, IsNotFound(nil))
|
|
})
|
|
}
|
|
|
|
// TestError_Format checks the Error/Unwrap surface.
|
|
func TestError_Format(t *testing.T) {
|
|
cause := errors.New("dial tcp: refused")
|
|
e := Wrapf(CodeNetworkError, cause, "connect to %s", "host")
|
|
assert.Contains(t, e.Error(), "network.error")
|
|
assert.Contains(t, e.Error(), "connect to host")
|
|
assert.Contains(t, e.Error(), "dial tcp: refused")
|
|
assert.Same(t, cause, errors.Unwrap(e))
|
|
}
|
|
|
|
func memSecretsFn(s *secrets.MemStore) func() (secrets.Store, error) {
|
|
return func() (secrets.Store, error) { return s, nil }
|
|
}
|
|
|
|
func TestBuildClient_NoCurrentProfile(t *testing.T) {
|
|
f := &Factory{
|
|
Config: func() (*config.Config, error) { return &config.Config{}, nil },
|
|
Secrets: memSecretsFn(secrets.NewMemStore()),
|
|
}
|
|
_, err := buildClient(f)
|
|
require.Error(t, err)
|
|
var typed *Error
|
|
require.ErrorAs(t, err, &typed)
|
|
assert.Equal(t, CodeAuthUnauthenticated, typed.Code)
|
|
}
|
|
|
|
func TestBuildClient_UnknownContext(t *testing.T) {
|
|
f := &Factory{
|
|
Config: func() (*config.Config, error) {
|
|
return &config.Config{CurrentProfile: "ghost"}, nil
|
|
},
|
|
Secrets: memSecretsFn(secrets.NewMemStore()),
|
|
}
|
|
_, err := buildClient(f)
|
|
require.Error(t, err)
|
|
var typed *Error
|
|
require.ErrorAs(t, err, &typed)
|
|
assert.Equal(t, CodeLocalConfigCorrupt, typed.Code)
|
|
}
|
|
|
|
func TestBuildClient_MissingHost(t *testing.T) {
|
|
f := &Factory{
|
|
Config: func() (*config.Config, error) {
|
|
return &config.Config{
|
|
CurrentProfile: "p",
|
|
Profiles: map[string]config.Profile{"p": {Host: ""}},
|
|
}, nil
|
|
},
|
|
Secrets: memSecretsFn(secrets.NewMemStore()),
|
|
}
|
|
_, err := buildClient(f)
|
|
require.Error(t, err)
|
|
var typed *Error
|
|
require.ErrorAs(t, err, &typed)
|
|
assert.Equal(t, CodeLocalConfigCorrupt, typed.Code)
|
|
}
|
|
|
|
func TestBuildClient_HappyPath(t *testing.T) {
|
|
store := secrets.NewMemStore()
|
|
require.NoError(t, store.Set("p", "access", "jwt"))
|
|
require.NoError(t, store.Set("p", "api_key", "sk-x"))
|
|
f := &Factory{
|
|
Config: func() (*config.Config, error) {
|
|
return &config.Config{
|
|
CurrentProfile: "p",
|
|
Profiles: map[string]config.Profile{
|
|
"p": {
|
|
Host: "https://kb.example.com",
|
|
TenantID: 7,
|
|
TokenRef: "mem://p/access",
|
|
APIKeyRef: "mem://p/api_key",
|
|
},
|
|
},
|
|
}, nil
|
|
},
|
|
Secrets: memSecretsFn(store),
|
|
}
|
|
cli, err := buildClient(f)
|
|
require.NoError(t, err)
|
|
require.NotNil(t, cli)
|
|
}
|
|
|
|
func TestBuildClient_SkipsUnreferencedSecrets(t *testing.T) {
|
|
// If the profile doesn't list APIKeyRef, buildClient must not call
|
|
// Get(api_key) - a perf invariant: avoid keychain trips for unused creds.
|
|
store := &countingSecrets{MemStore: secrets.NewMemStore()}
|
|
require.NoError(t, store.Set("p", "access", "jwt"))
|
|
f := &Factory{
|
|
Config: func() (*config.Config, error) {
|
|
return &config.Config{
|
|
CurrentProfile: "p",
|
|
Profiles: map[string]config.Profile{
|
|
"p": {Host: "https://x", TokenRef: "mem://p/access"},
|
|
},
|
|
}, nil
|
|
},
|
|
Secrets: func() (secrets.Store, error) { return store, nil },
|
|
}
|
|
_, err := buildClient(f)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 1, store.gets, "must fetch only access; api_key was not referenced")
|
|
}
|
|
|
|
// countingSecrets wraps MemStore to count Get invocations.
|
|
type countingSecrets struct {
|
|
*secrets.MemStore
|
|
gets int
|
|
}
|
|
|
|
func (c *countingSecrets) Get(ctx, key string) (string, error) {
|
|
c.gets++
|
|
return c.MemStore.Get(ctx, key)
|
|
}
|
|
|
|
// makeResolveKBCmd builds a minimal cobra.Command carrying the single --kb
|
|
// local flag so cmd.Flags().GetString lookups in ResolveKB exercise the flag
|
|
// path. The single flag accepts either a kb_<id> or a name.
|
|
func makeResolveKBCmd(t *testing.T, kb string) *cobra.Command {
|
|
t.Helper()
|
|
c := &cobra.Command{Use: "x"}
|
|
c.Flags().String("kb", "", "")
|
|
if kb != "" {
|
|
require.NoError(t, c.Flags().Set("kb", kb))
|
|
}
|
|
c.SetContext(context.Background())
|
|
return c
|
|
}
|
|
|
|
// resolveKBChdir switches cwd to dir for the duration of t (auto-restored).
|
|
// ResolveKB walks up from os.Getwd() so tests must isolate cwd.
|
|
func resolveKBChdir(t *testing.T, dir string) {
|
|
t.Helper()
|
|
prev, err := os.Getwd()
|
|
require.NoError(t, err)
|
|
require.NoError(t, os.Chdir(dir))
|
|
t.Cleanup(func() { _ = os.Chdir(prev) })
|
|
}
|
|
|
|
// fakeKBServer returns an httptest server that answers GET /api/v1/knowledge-bases
|
|
// with kbs (KnowledgeBaseListResponse), so a real *sdk.Client can talk to it.
|
|
func fakeKBServer(t *testing.T, kbs []sdk.KnowledgeBase) *httptest.Server {
|
|
t.Helper()
|
|
mux := http.NewServeMux()
|
|
mux.HandleFunc("/api/v1/knowledge-bases", func(w http.ResponseWriter, r *http.Request) {
|
|
w.Header().Set("Content-Type", "application/json")
|
|
_ = json.NewEncoder(w).Encode(sdk.KnowledgeBaseListResponse{Success: true, Data: kbs})
|
|
})
|
|
srv := httptest.NewServer(mux)
|
|
t.Cleanup(srv.Close)
|
|
return srv
|
|
}
|
|
|
|
// TestFactory_ActiveProfile_EnvVarFallback verifies WEKNORA_PROFILE is honoured
|
|
// when no override or config is present.
|
|
func TestFactory_ActiveProfile_EnvVarFallback(t *testing.T) {
|
|
t.Setenv("WEKNORA_PROFILE", "staging")
|
|
f := &Factory{} // no override, no config
|
|
if got := f.ActiveProfile(); got != "staging" {
|
|
t.Errorf("expected env fallback to staging; got %q", got)
|
|
}
|
|
}
|
|
|
|
// TestFactory_ActiveProfile_OverrideWinsEnv verifies ProfileOverride takes
|
|
// priority over the WEKNORA_PROFILE env var.
|
|
func TestFactory_ActiveProfile_OverrideWinsEnv(t *testing.T) {
|
|
t.Setenv("WEKNORA_PROFILE", "staging")
|
|
f := &Factory{ProfileOverride: "prod"}
|
|
if got := f.ActiveProfile(); got != "prod" {
|
|
t.Errorf("override should win over env; got %q", got)
|
|
}
|
|
}
|
|
|
|
// TestResolveKB_Chain exercises the 4-level fallback chain. Each sub-test
|
|
// isolates cwd / env / closure from the others.
|
|
func TestResolveKB_Chain(t *testing.T) {
|
|
t.Run("flag_kb_id_wins", func(t *testing.T) {
|
|
// UUID form on --kb → pass-through; no SDK call, no env, no disk.
|
|
t.Setenv("WEKNORA_KB_ID", "kb_env_should_lose")
|
|
dir := t.TempDir()
|
|
resolveKBChdir(t, dir)
|
|
// Drop a project link too - must be ignored.
|
|
require.NoError(t, projectlink.Save(filepath.Join(dir, ".weknora", "project.yaml"), &projectlink.Project{KBID: "kb_disk_should_lose"}))
|
|
|
|
clientCalls := 0
|
|
f := &Factory{
|
|
Client: func() (*sdk.Client, error) {
|
|
clientCalls++
|
|
return nil, errors.New("must not be called")
|
|
},
|
|
}
|
|
got, err := f.ResolveKB(makeResolveKBCmd(t, "aaaaaaaa-aaaa-4aaa-8aaa-aaaaaaaaaaaa"))
|
|
require.NoError(t, err)
|
|
assert.Equal(t, "aaaaaaaa-aaaa-4aaa-8aaa-aaaaaaaaaaaa", got)
|
|
assert.Equal(t, 0, clientCalls)
|
|
})
|
|
|
|
t.Run("flag_kb_name_resolves", func(t *testing.T) {
|
|
t.Setenv("WEKNORA_KB_ID", "")
|
|
srv := fakeKBServer(t, []sdk.KnowledgeBase{
|
|
{ID: "kb_a", Name: "foo"},
|
|
{ID: "kb_b", Name: "bar"},
|
|
})
|
|
f := &Factory{
|
|
Client: func() (*sdk.Client, error) { return sdk.NewClient(srv.URL), nil },
|
|
}
|
|
got, err := f.ResolveKB(makeResolveKBCmd(t, "foo"))
|
|
require.NoError(t, err)
|
|
assert.Equal(t, "kb_a", got)
|
|
})
|
|
|
|
t.Run("flag_kb_name_not_found", func(t *testing.T) {
|
|
t.Setenv("WEKNORA_KB_ID", "")
|
|
srv := fakeKBServer(t, []sdk.KnowledgeBase{{ID: "kb_a", Name: "foo"}})
|
|
f := &Factory{
|
|
Client: func() (*sdk.Client, error) { return sdk.NewClient(srv.URL), nil },
|
|
}
|
|
_, err := f.ResolveKB(makeResolveKBCmd(t, "missing"))
|
|
require.Error(t, err)
|
|
var typed *Error
|
|
require.ErrorAs(t, err, &typed)
|
|
assert.Equal(t, CodeKBNotFound, typed.Code)
|
|
})
|
|
|
|
t.Run("env_var", func(t *testing.T) {
|
|
// No flag, env wins over disk.
|
|
t.Setenv("WEKNORA_KB_ID", "kb_env")
|
|
dir := t.TempDir()
|
|
resolveKBChdir(t, dir)
|
|
require.NoError(t, projectlink.Save(filepath.Join(dir, ".weknora", "project.yaml"), &projectlink.Project{KBID: "kb_disk_should_lose"}))
|
|
|
|
f := &Factory{}
|
|
got, err := f.ResolveKB(makeResolveKBCmd(t, ""))
|
|
require.NoError(t, err)
|
|
assert.Equal(t, "kb_env", got)
|
|
})
|
|
|
|
t.Run("project_link_walk_up", func(t *testing.T) {
|
|
t.Setenv("WEKNORA_KB_ID", "")
|
|
root := t.TempDir()
|
|
require.NoError(t, projectlink.Save(filepath.Join(root, ".weknora", "project.yaml"), &projectlink.Project{KBID: "kb_proj"}))
|
|
// Run from a deep child to exercise walk-up.
|
|
deep := filepath.Join(root, "a", "b", "c")
|
|
require.NoError(t, os.MkdirAll(deep, 0o755))
|
|
resolveKBChdir(t, deep)
|
|
|
|
f := &Factory{}
|
|
got, err := f.ResolveKB(makeResolveKBCmd(t, ""))
|
|
require.NoError(t, err)
|
|
assert.Equal(t, "kb_proj", got)
|
|
})
|
|
|
|
t.Run("none", func(t *testing.T) {
|
|
// No flag, no env, no project link → CodeKBIDRequired.
|
|
t.Setenv("WEKNORA_KB_ID", "")
|
|
dir := t.TempDir()
|
|
resolveKBChdir(t, dir)
|
|
|
|
f := &Factory{}
|
|
_, err := f.ResolveKB(makeResolveKBCmd(t, ""))
|
|
require.Error(t, err)
|
|
var typed *Error
|
|
require.ErrorAs(t, err, &typed)
|
|
assert.Equal(t, CodeKBIDRequired, typed.Code)
|
|
})
|
|
}
|