mirror of
https://github.com/Ed1s0nZ/CyberStrikeAI.git
synced 2026-05-15 12:58:01 +02:00
124 lines
4.0 KiB
Go
124 lines
4.0 KiB
Go
package multiagent
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/cloudwego/eino/adk"
|
|
"github.com/cloudwego/eino/compose"
|
|
"github.com/cloudwego/eino/schema"
|
|
)
|
|
|
|
type hitlInterceptorKey struct{}
|
|
|
|
type HITLToolInterceptor func(ctx context.Context, toolName, arguments string) (string, error)
|
|
|
|
type humanRejectError struct {
|
|
reason string
|
|
}
|
|
|
|
func (e *humanRejectError) Error() string {
|
|
if strings.TrimSpace(e.reason) == "" {
|
|
return "rejected by user"
|
|
}
|
|
return "rejected by user: " + strings.TrimSpace(e.reason)
|
|
}
|
|
|
|
func NewHumanRejectError(reason string) error {
|
|
return &humanRejectError{reason: strings.TrimSpace(reason)}
|
|
}
|
|
|
|
func IsHumanRejectError(err error) bool {
|
|
var target *humanRejectError
|
|
return errors.As(err, &target)
|
|
}
|
|
|
|
func WithHITLToolInterceptor(ctx context.Context, fn HITLToolInterceptor) context.Context {
|
|
if fn == nil {
|
|
return ctx
|
|
}
|
|
return context.WithValue(ctx, hitlInterceptorKey{}, fn)
|
|
}
|
|
|
|
// hitlToolCallMiddleware 同时注册 Invokable 与 Streamable。
|
|
// Eino filesystem 的 execute 为流式工具(StreamableTool),仅挂 Invokable 时人机协同不会拦截,会直接执行。
|
|
func hitlToolCallMiddleware() compose.ToolMiddleware {
|
|
return compose.ToolMiddleware{
|
|
Invokable: hitlInvokableToolCallMiddleware(),
|
|
Streamable: hitlStreamableToolCallMiddleware(),
|
|
}
|
|
}
|
|
|
|
func hitlClearReturnDirectlyIfTransfer(ctx context.Context, toolName string) {
|
|
if !strings.EqualFold(strings.TrimSpace(toolName), adk.TransferToAgentToolName) {
|
|
return
|
|
}
|
|
_ = compose.ProcessState[*adk.State](ctx, func(_ context.Context, st *adk.State) error {
|
|
if st == nil {
|
|
return nil
|
|
}
|
|
st.ReturnDirectlyToolCallID = ""
|
|
st.HasReturnDirectly = false
|
|
st.ReturnDirectlyEvent = nil
|
|
return nil
|
|
})
|
|
}
|
|
|
|
func hitlInvokableToolCallMiddleware() compose.InvokableToolMiddleware {
|
|
return func(next compose.InvokableToolEndpoint) compose.InvokableToolEndpoint {
|
|
return func(ctx context.Context, input *compose.ToolInput) (*compose.ToolOutput, error) {
|
|
if input != nil {
|
|
if fn, ok := ctx.Value(hitlInterceptorKey{}).(HITLToolInterceptor); ok && fn != nil {
|
|
edited, err := fn(ctx, input.Name, input.Arguments)
|
|
if err != nil {
|
|
if IsHumanRejectError(err) {
|
|
// Human rejection should be a soft tool result so the model can continue iterating.
|
|
msg := fmt.Sprintf("[HITL Reject] Tool '%s' was rejected by human reviewer. Reason: %s\nPlease adjust parameters/plan and continue without this call.",
|
|
input.Name, strings.TrimSpace(err.Error()))
|
|
// transfer_to_agent 在 Eino 中标记为 returnDirectly:工具成功后 ReAct 子图会直接 END,
|
|
// 并依赖真实工具内的 SendToolGenAction 触发移交。HITL 拒绝时不会执行真实工具,
|
|
// 若仍走 returnDirectly 分支,监督者会在无 Transfer 动作的情况下结束,模型不再迭代。
|
|
hitlClearReturnDirectlyIfTransfer(ctx, input.Name)
|
|
return &compose.ToolOutput{Result: msg}, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
if edited != "" {
|
|
input.Arguments = edited
|
|
}
|
|
}
|
|
}
|
|
return next(ctx, input)
|
|
}
|
|
}
|
|
}
|
|
|
|
func hitlStreamableToolCallMiddleware() compose.StreamableToolMiddleware {
|
|
return func(next compose.StreamableToolEndpoint) compose.StreamableToolEndpoint {
|
|
return func(ctx context.Context, input *compose.ToolInput) (*compose.StreamToolOutput, error) {
|
|
if input != nil {
|
|
if fn, ok := ctx.Value(hitlInterceptorKey{}).(HITLToolInterceptor); ok && fn != nil {
|
|
edited, err := fn(ctx, input.Name, input.Arguments)
|
|
if err != nil {
|
|
if IsHumanRejectError(err) {
|
|
msg := fmt.Sprintf("[HITL Reject] Tool '%s' was rejected by human reviewer. Reason: %s\nPlease adjust parameters/plan and continue without this call.",
|
|
input.Name, strings.TrimSpace(err.Error()))
|
|
hitlClearReturnDirectlyIfTransfer(ctx, input.Name)
|
|
return &compose.StreamToolOutput{
|
|
Result: schema.StreamReaderFromArray([]string{msg}),
|
|
}, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
if edited != "" {
|
|
input.Arguments = edited
|
|
}
|
|
}
|
|
}
|
|
return next(ctx, input)
|
|
}
|
|
}
|
|
}
|