coze-studio/backend/domain/workflow/internal/nodes/receiver/input_receiver.go

203 lines
5.5 KiB
Go

/*
* Copyright 2025 coze-dev Authors
*
* 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 receiver
import (
"context"
"errors"
"fmt"
"github.com/bytedance/sonic"
"github.com/cloudwego/eino/compose"
"github.com/coze-dev/coze-studio/backend/domain/workflow"
"github.com/coze-dev/coze-studio/backend/domain/workflow/entity"
"github.com/coze-dev/coze-studio/backend/domain/workflow/entity/vo"
"github.com/coze-dev/coze-studio/backend/domain/workflow/internal/canvas/convert"
"github.com/coze-dev/coze-studio/backend/domain/workflow/internal/execute"
"github.com/coze-dev/coze-studio/backend/domain/workflow/internal/nodes"
"github.com/coze-dev/coze-studio/backend/domain/workflow/internal/schema"
"github.com/coze-dev/coze-studio/backend/pkg/ctxcache"
"github.com/coze-dev/coze-studio/backend/pkg/errorx"
"github.com/coze-dev/coze-studio/backend/pkg/logs"
sonic2 "github.com/coze-dev/coze-studio/backend/pkg/sonic"
"github.com/coze-dev/coze-studio/backend/types/errno"
)
type Config struct {
OutputSchema string
}
func (c *Config) Adapt(_ context.Context, n *vo.Node, _ ...nodes.AdaptOption) (*schema.NodeSchema, error) {
c.OutputSchema = n.Data.Inputs.OutputSchema
ns := &schema.NodeSchema{
Key: vo.NodeKey(n.ID),
Type: entity.NodeTypeInputReceiver,
Name: n.Data.Meta.Title,
Configs: c,
}
if err := convert.SetOutputTypesForNodeSchema(n, ns); err != nil {
return nil, err
}
return ns, nil
}
func (c *Config) Build(_ context.Context, ns *schema.NodeSchema, _ ...schema.BuildOption) (any, error) {
nodeMeta := entity.NodeMetaByNodeType(entity.NodeTypeInputReceiver)
if nodeMeta == nil {
return nil, errors.New("node meta not found for input receiver")
}
interruptData := map[string]string{
"content_type": "form_schema",
"content": c.OutputSchema,
}
interruptDataStr, err := sonic.ConfigStd.MarshalToString(interruptData) // keep the order of the keys
if err != nil {
return nil, err
}
return &InputReceiver{
outputTypes: ns.OutputTypes, // so the node can refer to its output types during execution
nodeMeta: *nodeMeta,
nodeKey: ns.Key,
interruptData: interruptDataStr,
}, nil
}
func (c *Config) RequireCheckpoint() bool {
return true
}
type InputReceiver struct {
outputTypes map[string]*vo.TypeInfo
interruptData string
nodeKey vo.NodeKey
nodeMeta entity.NodeTypeMeta
}
const (
ReceivedDataKey = "$received_data"
receiverWarningKey = "receiver_warning_%d_%s"
)
func (i *InputReceiver) Invoke(ctx context.Context, in map[string]any) (map[string]any, error) {
var input string
if in != nil {
receivedData, ok := in[ReceivedDataKey]
if ok {
input = receivedData.(string)
}
}
if len(input) == 0 {
err := compose.ProcessState(ctx, func(ctx context.Context, ieStore nodes.InterruptEventStore) error {
_, found, e := ieStore.GetInterruptEvent(i.nodeKey) // TODO: try not use InterruptEventStore or state in general
if e != nil {
return e
}
if !found { // only generate a new event if it doesn't exist
eventID, err := workflow.GetRepository().GenID(ctx)
if err != nil {
return err
}
return ieStore.SetInterruptEvent(i.nodeKey, &entity.InterruptEvent{
ID: eventID,
NodeKey: i.nodeKey,
NodeType: entity.NodeTypeInputReceiver,
NodeTitle: i.nodeMeta.Name,
NodeIcon: i.nodeMeta.IconURL,
InterruptData: i.interruptData,
EventType: entity.InterruptEventInput,
})
}
return nil
})
if err != nil {
return nil, err
}
return nil, compose.InterruptAndRerun
}
out, err := jsonParseRelaxed(ctx, input, i.outputTypes)
if err != nil {
return nil, err
}
return out, nil
}
func jsonParseRelaxed(ctx context.Context, data string, schema_ map[string]*vo.TypeInfo) (map[string]any, error) {
var result map[string]any
err := sonic2.UnmarshalString(data, &result)
if err != nil {
return nil, err
}
r, ws, err := nodes.ConvertInputs(ctx, result, schema_, nodes.SkipUnknownFields())
if err != nil {
return nil, err
}
if ws != nil && len(*ws) > 0 {
logs.CtxWarnf(ctx, "convert inputs warnings: %v", *ws)
var (
executeID int64
nodeKey vo.NodeKey
)
if c := execute.GetExeCtx(ctx); c != nil {
executeID = c.RootExecuteID
nodeKey = c.NodeKey
}
warningKey := fmt.Sprintf(receiverWarningKey, executeID, nodeKey)
ctxcache.Store(ctx, warningKey, *ws)
}
return r, nil
}
func (i *InputReceiver) ToCallbackOutput(ctx context.Context, output map[string]any) (
*nodes.StructuredCallbackOutput, error) {
var (
executeID int64
nodeKey vo.NodeKey
)
if c := execute.GetExeCtx(ctx); c != nil {
executeID = c.RootExecuteID
nodeKey = c.NodeKey
}
warningKey := fmt.Sprintf(receiverWarningKey, executeID, nodeKey)
var wfe vo.WorkflowError
if warnings, ok := ctxcache.Get[nodes.ConversionWarnings](ctx, warningKey); ok {
wfe = vo.WrapWarn(errno.ErrNodeOutputParseFail, warnings, errorx.KV("warnings", warnings.Error()))
}
return &nodes.StructuredCallbackOutput{
Output: output,
RawOutput: output,
Error: wfe,
}, nil
}