195 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			195 lines
		
	
	
		
			6.4 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 compose
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"strconv"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/coze-dev/coze-studio/backend/domain/workflow/entity/vo"
 | 
						|
	"github.com/coze-dev/coze-studio/backend/domain/workflow/internal/nodes"
 | 
						|
	"github.com/coze-dev/coze-studio/backend/domain/workflow/internal/nodes/selector"
 | 
						|
)
 | 
						|
 | 
						|
type selectorCallbackField struct {
 | 
						|
	Key   string      `json:"key"`
 | 
						|
	Type  vo.DataType `json:"type"`
 | 
						|
	Value any         `json:"value"`
 | 
						|
}
 | 
						|
 | 
						|
type selectorCondition struct {
 | 
						|
	Left     selectorCallbackField  `json:"left"`
 | 
						|
	Operator vo.OperatorType        `json:"operator"`
 | 
						|
	Right    *selectorCallbackField `json:"right,omitempty"`
 | 
						|
}
 | 
						|
 | 
						|
type selectorBranch struct {
 | 
						|
	Conditions []*selectorCondition `json:"conditions"`
 | 
						|
	Logic      vo.LogicType         `json:"logic"`
 | 
						|
	Name       string               `json:"name"`
 | 
						|
}
 | 
						|
 | 
						|
func (s *NodeSchema) toSelectorCallbackInput(sc *WorkflowSchema) func(_ context.Context, in map[string]any) (map[string]any, error) {
 | 
						|
	return func(_ context.Context, in map[string]any) (map[string]any, error) {
 | 
						|
		config := mustGetKey[[]*selector.OneClauseSchema]("Clauses", s.Configs)
 | 
						|
		count := len(config)
 | 
						|
 | 
						|
		output := make([]*selectorBranch, count)
 | 
						|
 | 
						|
		for _, source := range s.InputSources {
 | 
						|
			targetPath := source.Path
 | 
						|
			if len(targetPath) == 2 {
 | 
						|
				indexStr := targetPath[0]
 | 
						|
				index, err := strconv.Atoi(indexStr)
 | 
						|
				if err != nil {
 | 
						|
					return nil, err
 | 
						|
				}
 | 
						|
 | 
						|
				branch := output[index]
 | 
						|
				if branch == nil {
 | 
						|
					output[index] = &selectorBranch{
 | 
						|
						Conditions: []*selectorCondition{
 | 
						|
							{
 | 
						|
								Operator: config[index].Single.ToCanvasOperatorType(),
 | 
						|
							},
 | 
						|
						},
 | 
						|
						Logic: selector.ClauseRelationAND.ToVOLogicType(),
 | 
						|
					}
 | 
						|
				}
 | 
						|
 | 
						|
				if targetPath[1] == selector.LeftKey {
 | 
						|
					leftV, ok := nodes.TakeMapValue(in, targetPath)
 | 
						|
					if !ok {
 | 
						|
						return nil, fmt.Errorf("failed to take left value of %s", targetPath)
 | 
						|
					}
 | 
						|
					if source.Source.Ref.VariableType != nil {
 | 
						|
						if *source.Source.Ref.VariableType == vo.ParentIntermediate {
 | 
						|
							parentNodeKey, ok := sc.Hierarchy[s.Key]
 | 
						|
							if !ok {
 | 
						|
								return nil, fmt.Errorf("failed to find parent node key of %s", s.Key)
 | 
						|
							}
 | 
						|
							parentNode := sc.GetNode(parentNodeKey)
 | 
						|
							output[index].Conditions[0].Left = selectorCallbackField{
 | 
						|
								Key:   parentNode.Name + "." + strings.Join(source.Source.Ref.FromPath, "."),
 | 
						|
								Type:  s.InputTypes[targetPath[0]].Properties[targetPath[1]].Type,
 | 
						|
								Value: leftV,
 | 
						|
							}
 | 
						|
						} else {
 | 
						|
							output[index].Conditions[0].Left = selectorCallbackField{
 | 
						|
								Key:   "",
 | 
						|
								Type:  s.InputTypes[targetPath[0]].Properties[targetPath[1]].Type,
 | 
						|
								Value: leftV,
 | 
						|
							}
 | 
						|
						}
 | 
						|
					} else {
 | 
						|
						output[index].Conditions[0].Left = selectorCallbackField{
 | 
						|
							Key:   sc.GetNode(source.Source.Ref.FromNodeKey).Name + "." + strings.Join(source.Source.Ref.FromPath, "."),
 | 
						|
							Type:  s.InputTypes[targetPath[0]].Properties[targetPath[1]].Type,
 | 
						|
							Value: leftV,
 | 
						|
						}
 | 
						|
					}
 | 
						|
				} else if targetPath[1] == selector.RightKey {
 | 
						|
					rightV, ok := nodes.TakeMapValue(in, targetPath)
 | 
						|
					if !ok {
 | 
						|
						return nil, fmt.Errorf("failed to take right value of %s", targetPath)
 | 
						|
					}
 | 
						|
					output[index].Conditions[0].Right = &selectorCallbackField{
 | 
						|
						Type:  s.InputTypes[targetPath[0]].Properties[targetPath[1]].Type,
 | 
						|
						Value: rightV,
 | 
						|
					}
 | 
						|
				}
 | 
						|
			} else if len(targetPath) == 3 {
 | 
						|
				indexStr := targetPath[0]
 | 
						|
				index, err := strconv.Atoi(indexStr)
 | 
						|
				if err != nil {
 | 
						|
					return nil, err
 | 
						|
				}
 | 
						|
 | 
						|
				multi := config[index].Multi
 | 
						|
 | 
						|
				branch := output[index]
 | 
						|
				if branch == nil {
 | 
						|
					output[index] = &selectorBranch{
 | 
						|
						Conditions: make([]*selectorCondition, len(multi.Clauses)),
 | 
						|
						Logic:      multi.Relation.ToVOLogicType(),
 | 
						|
					}
 | 
						|
				}
 | 
						|
 | 
						|
				clauseIndexStr := targetPath[1]
 | 
						|
				clauseIndex, err := strconv.Atoi(clauseIndexStr)
 | 
						|
				if err != nil {
 | 
						|
					return nil, err
 | 
						|
				}
 | 
						|
 | 
						|
				clause := multi.Clauses[clauseIndex]
 | 
						|
 | 
						|
				if output[index].Conditions[clauseIndex] == nil {
 | 
						|
					output[index].Conditions[clauseIndex] = &selectorCondition{
 | 
						|
						Operator: clause.ToCanvasOperatorType(),
 | 
						|
					}
 | 
						|
				}
 | 
						|
 | 
						|
				if targetPath[2] == selector.LeftKey {
 | 
						|
					leftV, ok := nodes.TakeMapValue(in, targetPath)
 | 
						|
					if !ok {
 | 
						|
						return nil, fmt.Errorf("failed to take left value of %s", targetPath)
 | 
						|
					}
 | 
						|
					if source.Source.Ref.VariableType != nil {
 | 
						|
						if *source.Source.Ref.VariableType == vo.ParentIntermediate {
 | 
						|
							parentNodeKey, ok := sc.Hierarchy[s.Key]
 | 
						|
							if !ok {
 | 
						|
								return nil, fmt.Errorf("failed to find parent node key of %s", s.Key)
 | 
						|
							}
 | 
						|
							parentNode := sc.GetNode(parentNodeKey)
 | 
						|
							output[index].Conditions[clauseIndex].Left = selectorCallbackField{
 | 
						|
								Key:   parentNode.Name + "." + strings.Join(source.Source.Ref.FromPath, "."),
 | 
						|
								Type:  s.InputTypes[targetPath[0]].Properties[targetPath[1]].Properties[targetPath[2]].Type,
 | 
						|
								Value: leftV,
 | 
						|
							}
 | 
						|
						} else {
 | 
						|
							output[index].Conditions[clauseIndex].Left = selectorCallbackField{
 | 
						|
								Key:   "",
 | 
						|
								Type:  s.InputTypes[targetPath[0]].Properties[targetPath[1]].Properties[targetPath[2]].Type,
 | 
						|
								Value: leftV,
 | 
						|
							}
 | 
						|
						}
 | 
						|
					} else {
 | 
						|
						output[index].Conditions[clauseIndex].Left = selectorCallbackField{
 | 
						|
							Key:   sc.GetNode(source.Source.Ref.FromNodeKey).Name + "." + strings.Join(source.Source.Ref.FromPath, "."),
 | 
						|
							Type:  s.InputTypes[targetPath[0]].Properties[targetPath[1]].Properties[targetPath[2]].Type,
 | 
						|
							Value: leftV,
 | 
						|
						}
 | 
						|
					}
 | 
						|
				} else if targetPath[2] == selector.RightKey {
 | 
						|
					rightV, ok := nodes.TakeMapValue(in, targetPath)
 | 
						|
					if !ok {
 | 
						|
						return nil, fmt.Errorf("failed to take right value of %s", targetPath)
 | 
						|
					}
 | 
						|
					output[index].Conditions[clauseIndex].Right = &selectorCallbackField{
 | 
						|
						Type:  s.InputTypes[targetPath[0]].Properties[targetPath[1]].Properties[targetPath[2]].Type,
 | 
						|
						Value: rightV,
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		return map[string]any{"branches": output}, nil
 | 
						|
	}
 | 
						|
}
 |