Files
ragflow/internal/agent/canvas/compile_test.go
Zhichang Yu 2362210caf refactor(log): unify Go logging to zap with rotation, strip per-package levels (#16261)
Refactor the Go agent port's logging so every log line — gin access,
agent canvas events, harness warnings, fatal boot errors — flows through
a single common.Logger (zap) backed by a rotated file, with structured
fields, level filtering, and configurable rotation.

---------

Co-authored-by: Claude <noreply@anthropic.com>
2026-06-23 16:21:46 +08:00

122 lines
3.8 KiB
Go

//
// Copyright 2026 The InfiniFlow Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package canvas
import (
"bytes"
"context"
"strings"
"testing"
"go.uber.org/zap"
"go.uber.org/zap/zapcore"
"ragflow/internal/common"
)
// TestCompile_LogsWhenLegacyNodesPresent exercises the
// decoder-bypass guard in Compile: a Canvas that carries
// LoopItem/IterationItem entries in `Components` (i.e. one that
// never went through dsl.NormalizeForCanvas) must produce a
// visible warning through common.Logger. The guard is
// intentionally a log, not a panic, so internal drivers / legacy
// fixtures can still drive Compile; the log makes the regression
// observable.
//
// The test swaps common.Logger for a buffer-backed encoder so
// we can assert on the structured log message. We don't fail on
// `Compile` itself failing — the legacy fixture graph is
// intentionally minimal and may not compile end-to-end without a
// Begin node; the assertion is strictly about the log surface.
func TestCompile_LogsWhenLegacyNodesPresent(t *testing.T) {
var buf bytes.Buffer
prev := common.Logger
common.Logger = zap.New(
zapcore.NewCore(
zapcore.NewConsoleEncoder(zap.NewProductionEncoderConfig()),
zapcore.AddSync(&buf),
zapcore.InfoLevel,
),
)
t.Cleanup(func() { common.Logger = prev })
c := &Canvas{
Components: map[string]CanvasComponent{
"Loop:abc": {
Obj: CanvasComponentObj{ComponentName: "Loop", Params: map[string]any{}},
},
"LoopItem:def": {
Obj: CanvasComponentObj{ComponentName: "LoopItem", Params: map[string]any{}},
Downstream: []string{"Body:1"},
},
"Body:1": {
Obj: CanvasComponentObj{ComponentName: "Message", Params: map[string]any{}},
},
},
}
// Compile may return an error from downstream BuildWorkflow —
// we ignore it; the assertion is on the log line.
_, _ = Compile(context.Background(), c)
got := buf.String()
if !strings.Contains(got, "LoopItem/IterationItem") {
t.Errorf("expected legacy-node log warning, got %q", got)
}
if !strings.Contains(got, "bypassed dsl.NormalizeForCanvas") {
t.Errorf("expected bypass warning, got %q", got)
}
}
// TestCompile_NoLogOnCleanCanvas is the negative case: a Canvas
// whose components carry only modern names must NOT trip the
// guard. This guards against an over-eager regex that fires on
// every Compile.
func TestCompile_NoLogOnCleanCanvas(t *testing.T) {
var buf bytes.Buffer
prev := common.Logger
common.Logger = zap.New(
zapcore.NewCore(
zapcore.NewConsoleEncoder(zap.NewProductionEncoderConfig()),
zapcore.AddSync(&buf),
zapcore.InfoLevel,
),
)
t.Cleanup(func() { common.Logger = prev })
c := &Canvas{
Components: map[string]CanvasComponent{
"begin": {
Obj: CanvasComponentObj{ComponentName: "Begin", Params: map[string]any{}},
},
"llm:0": {
Obj: CanvasComponentObj{ComponentName: "LLM", Params: map[string]any{}},
Downstream: []string{},
},
},
}
// We don't fail on Compile's own error (it may fail for many
// reasons unrelated to legacy names); the assertion is on the
// absence of the legacy log line.
_, _ = Compile(context.Background(), c)
got := buf.String()
if strings.Contains(got, "LoopItem/IterationItem") {
t.Errorf("unexpected legacy-node log on clean canvas: %q", got)
}
}