mirror of
https://github.com/infiniflow/ragflow.git
synced 2026-06-29 23:41:12 +08:00
Ports the agent canvas subsystem from Python to Go.
## What's included
### Canvas Engine (Phase 0/1)
- State engine, scheduler, variable resolver, Redis checkpoint store,
cancel protocol
- **209 tests** across canvas / component / io packages
### 22 Components (P0–P4)
| Tier | Components |
|---|---|
| P0 T1+T2+T3 | LLM, Agent, ExitLoop, Switch, Categorize, Begin,
Message, Invoke |
| P1 T3 | VariableAggregator, VariableAssigner, StringTransform,
ListOperations, DataOperations |
| P2 T3 | Iteration, IterationItem, Loop, LoopItem |
| P3 T3 | UserFillUp, Fillup |
| P4 T5 | Browser, ExcelProcessor, DocsGenerator |
### DSL v2 Schema (Phase 2.5)
- Typed v2 in-memory model with v1-to-v2 auto-detect converter
- v1 legacy field stripping per plan §2.11.7
### HTTP Endpoints & Bug Fixes (Plans PR1–PR3)
- **DELETE SQL bug fix**: gorm v2 `Where("id = ?", id).Delete(...)`
pattern
- **CreateAgent validation**: title/DSL required, duplicate check, 103
envelope
- **13 new endpoints**: templates, prompts, tags, sessions CRUD,
chat/completions (SSE + non-stream stubs), rerun, test_db_connection,
logs, webhook/logs
- **756 Go unit tests** (745 → 756, +18)
- **17 → 0 Python integration test failures** (test_agents.py +
test_session_management/)
### Tools
21 eino tools: HTTPHelper, search tools, financial/data tools, mandatory
stubs
### Infrastructure
OTel observability, NATS message queue, DeepDoc gRPC client, SSRF
guards, IDOR mitigation
147 lines
4.2 KiB
Go
147 lines
4.2 KiB
Go
// Package component — Categorize unit tests (Phase 2 P0, plan §2.11.3 row 6).
|
|
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 [] (Phase 5 placeholder)", 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())
|
|
}
|
|
}
|