Files
ragflow/internal/agent/component/categorize_test.go
Zhichang Yu e45659868a feat(agent): ship the Go agent canvas port — eino interrupt/resume + Redis check-pointing (#16035)
Replaces the Python agent canvas runtime with a Go implementation that
runs inside `cmd/server_main`.

The canvas compiles into an eino Workflow that pauses on wait-for-user
via native Interrupt/Resume (no sentinel flag) and resumes from a
Redis-backed CheckPointStore.

All 21 Python agent components and ~35 tools are ported with functional
parity.

Sandbox providers now read their JSON config from the admin-panel
system_settings table with env fallback.

234 files / +35,413 / -6,111. All Go files are gofmt-clean (CI gate
added); drops the v2 DSL E2E step and the gap-analysis plan (both
redundant after the port ships).

## Type of change

- [x] Refactoring
- [x] New feature
- [x] Bug fix

🤖 Generated with [Claude Code](https://claude.com/claude-code)

---------

Co-authored-by: Claude <noreply@anthropic.com>
2026-06-17 13:24:03 +08:00

147 lines
4.2 KiB
Go

// Package component — Categorize unit tests.
package component
import (
"context"
"strings"
"testing"
)
func TestCategorize_ChosenCategory(t *testing.T) {
stub := &stubInvoker{resp: &ChatInvokeResponse{Content: "support", Model: "stub"}}
withStubInvoker(t, stub)
c := NewCategorizeComponent(CategorizeParam{
ModelID: "stub",
Categories: []string{"sales", "support", "billing"},
DefaultCategory: "support",
})
out, err := c.Invoke(context.Background(), map[string]any{})
if err != nil {
t.Fatalf("Invoke: %v", err)
}
if got, want := out["category"], "support"; got != want {
t.Errorf("category=%v, want %v", got, want)
}
scores, ok := out["scores"].(map[string]float64)
if !ok {
t.Fatalf("scores missing or wrong type: %T", out["scores"])
}
if scores["support"] != 1 {
t.Errorf("support score=%v, want 1", scores["support"])
}
if scores["sales"] != 0 || scores["billing"] != 0 {
t.Errorf("non-chosen categories should score 0; got %v", scores)
}
next, ok := out["_next"].([]string)
if !ok {
t.Fatalf("_next missing or wrong type: %T", out["_next"])
}
if len(next) != 0 {
t.Errorf("_next=%v, want [] (placeholder; MultiBranch wires the actual routing)", next)
}
}
func TestCategorize_FallbackToDefault(t *testing.T) {
stub := &stubInvoker{resp: &ChatInvokeResponse{Content: "totally not in the list", Model: "stub"}}
withStubInvoker(t, stub)
c := NewCategorizeComponent(CategorizeParam{
ModelID: "stub",
Categories: []string{"a", "b", "c"},
DefaultCategory: "b",
})
out, err := c.Invoke(context.Background(), map[string]any{})
if err != nil {
t.Fatalf("Invoke: %v", err)
}
if got, want := out["category"], "b"; got != want {
t.Errorf("category=%v, want %v (default fallback)", got, want)
}
}
func TestCategorize_DefaultDefaultsToFirstCategory(t *testing.T) {
stub := &stubInvoker{resp: &ChatInvokeResponse{Content: "garbage", Model: "stub"}}
withStubInvoker(t, stub)
c := NewCategorizeComponent(CategorizeParam{
ModelID: "stub",
Categories: []string{"alpha", "beta", "gamma"},
// no default_category
})
out, err := c.Invoke(context.Background(), map[string]any{})
if err != nil {
t.Fatalf("Invoke: %v", err)
}
if got, want := out["category"], "alpha"; got != want {
t.Errorf("category=%v, want %v (auto-default to first)", got, want)
}
}
func TestCategorize_CaseInsensitive(t *testing.T) {
stub := &stubInvoker{resp: &ChatInvokeResponse{Content: "SUPPORT", Model: "stub"}}
withStubInvoker(t, stub)
c := NewCategorizeComponent(CategorizeParam{
ModelID: "stub",
Categories: []string{"sales", "support", "billing"},
DefaultCategory: "sales",
})
out, err := c.Invoke(context.Background(), map[string]any{})
if err != nil {
t.Fatalf("Invoke: %v", err)
}
if got, want := out["category"], "support"; got != want {
t.Errorf("category=%v, want %v (case-insensitive match)", got, want)
}
}
func TestCategorize_PromptListsCategories(t *testing.T) {
// Verify the prompt passed to the invoker includes the categories
// so a model choosing between A and B has the context to do so.
stub := &stubInvoker{resp: &ChatInvokeResponse{Content: "x", Model: "stub"}}
withStubInvoker(t, stub)
c := NewCategorizeComponent(CategorizeParam{
ModelID: "stub",
Categories: []string{"x", "y", "z"},
DefaultCategory: "x",
Items: []string{"foo", "bar"},
})
_, err := c.Invoke(context.Background(), map[string]any{})
if err != nil {
t.Fatalf("Invoke: %v", err)
}
if stub.captured == nil {
t.Fatal("invoker not called")
}
var userContent string
for _, m := range stub.captured.Messages {
if m.Role == "user" {
userContent = m.Content
}
}
if userContent == "" {
t.Fatal("no user message in captured invoker request")
}
for _, want := range []string{"x", "y", "z", "foo", "bar"} {
if !strings.Contains(userContent, want) {
t.Errorf("prompt missing %q; got: %s", want, userContent)
}
}
}
func TestCategorize_Registered(t *testing.T) {
c, err := New("Categorize", map[string]any{
"model_id": "stub",
"categories": []any{"a", "b"},
"default_category": "a",
})
if err != nil {
t.Fatalf("New(Categorize): %v", err)
}
if c.Name() != "Categorize" {
t.Errorf("Name()=%q, want Categorize", c.Name())
}
}