Files
ragflow/internal/agent/component/memory_save.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

103 lines
3.1 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.
//
// Memory persistence scaffold. The Python Message component,
// when configured with memory_ids and memory_save=true, calls
// `queue_save_to_memory_task` from
// `api.db.joint_services.memory_message_service`. The Go runtime
// has a partial port (see internal/service/memory_message_service.go);
// this file defines the contract surface the Message component
// uses and a stub saver that returns ErrMemoryServiceMissing
// when no real implementation is wired.
//
// The interface (MemorySaver) is intentionally small: it takes a
// MemorySaveRequest and returns nil. A follow-up phase wires the
// real implementation, which will translate the request into the
// same DB row shape Python produces (api4conversation table per
// plan §2.11.6).
package component
import (
"context"
"errors"
"sync"
)
// ErrMemoryServiceMissing is the deferred-state sentinel for
// memory persistence. The Message component wraps persistence
// calls in errors.Is checks so callers can detect the gap.
var ErrMemoryServiceMissing = errors.New(
"component: memory persistence not yet wired in Go — " +
"defer to Python Canvas or implement MemorySaver",
)
// MemorySaveRequest is the wire shape. It mirrors the Python
// `message_dict` built in message.py:_save_to_memory:
//
// {
// "user_id": ...,
// "agent_id": ...,
// "session_id": ...,
// "user_input": ...,
// "agent_response": ...,
// }
type MemorySaveRequest struct {
MemoryIDs []string // the canvas-declared memory_ids
UserID string
AgentID string
SessionID string
UserInput string
AgentResponse string
}
// MemorySaver is the abstract interface for memory persistence.
// The default implementation returns ErrMemoryServiceMissing.
type MemorySaver interface {
Save(ctx context.Context, req MemorySaveRequest) error
}
var (
memSaverMu sync.RWMutex
memSaverImpl MemorySaver = stubMemorySaver{}
)
// SetMemorySaver installs a custom saver. Passing nil reverts to
// the default stub. Production code calls this at boot once
// internal/service/memory_message_service lands.
func SetMemorySaver(s MemorySaver) {
memSaverMu.Lock()
defer memSaverMu.Unlock()
if s == nil {
memSaverImpl = stubMemorySaver{}
return
}
memSaverImpl = s
}
// GetMemorySaver returns the registered saver.
func GetMemorySaver() MemorySaver {
memSaverMu.RLock()
defer memSaverMu.RUnlock()
return memSaverImpl
}
type stubMemorySaver struct{}
func (stubMemorySaver) Save(_ context.Context, _ MemorySaveRequest) error {
return ErrMemoryServiceMissing
}