138 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			138 lines
		
	
	
		
			4.0 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 nodes
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 
 | |
| 	"github.com/cloudwego/eino/compose"
 | |
| 
 | |
| 	"github.com/coze-dev/coze-studio/backend/domain/workflow/entity/vo"
 | |
| 	"github.com/coze-dev/coze-studio/backend/domain/workflow/internal/schema"
 | |
| )
 | |
| 
 | |
| var KeyIsFinished = "\x1FKey is finished\x1F"
 | |
| 
 | |
| type DynamicStreamContainer interface {
 | |
| 	SaveDynamicChoice(nodeKey vo.NodeKey, groupToChoice map[string]int)
 | |
| 	GetDynamicChoice(nodeKey vo.NodeKey) map[string]int
 | |
| 	GetDynamicStreamType(nodeKey vo.NodeKey, group string) (schema.FieldStreamType, error)
 | |
| 	GetAllDynamicStreamTypes(nodeKey vo.NodeKey) (map[string]schema.FieldStreamType, error)
 | |
| }
 | |
| 
 | |
| // ResolveStreamSources resolves incoming field sources for a node, deciding their stream type.
 | |
| func ResolveStreamSources(ctx context.Context, sources map[string]*schema.SourceInfo) (map[string]*schema.SourceInfo, error) {
 | |
| 	resolved := make(map[string]*schema.SourceInfo, len(sources))
 | |
| 
 | |
| 	nodeKey2Skipped := make(map[vo.NodeKey]bool)
 | |
| 
 | |
| 	var resolver func(path string, sInfo *schema.SourceInfo) (*schema.SourceInfo, error)
 | |
| 	resolver = func(path string, sInfo *schema.SourceInfo) (*schema.SourceInfo, error) {
 | |
| 		resolvedNode := &schema.SourceInfo{
 | |
| 			IsIntermediate: sInfo.IsIntermediate,
 | |
| 			FieldType:      sInfo.FieldType,
 | |
| 			FromNodeKey:    sInfo.FromNodeKey,
 | |
| 			FromPath:       sInfo.FromPath,
 | |
| 			TypeInfo:       sInfo.TypeInfo,
 | |
| 		}
 | |
| 
 | |
| 		if len(sInfo.SubSources) > 0 {
 | |
| 			resolvedNode.SubSources = make(map[string]*schema.SourceInfo, len(sInfo.SubSources))
 | |
| 
 | |
| 			for k, subInfo := range sInfo.SubSources {
 | |
| 				resolvedSub, err := resolver(k, subInfo)
 | |
| 				if err != nil {
 | |
| 					return nil, err
 | |
| 				}
 | |
| 
 | |
| 				resolvedNode.SubSources[k] = resolvedSub
 | |
| 			}
 | |
| 
 | |
| 			return resolvedNode, nil
 | |
| 		}
 | |
| 
 | |
| 		if sInfo.FromNodeKey == "" { // static values and variables, always non-streaming and available
 | |
| 			return resolvedNode, nil
 | |
| 		}
 | |
| 
 | |
| 		var skipped, ok bool
 | |
| 		if skipped, ok = nodeKey2Skipped[sInfo.FromNodeKey]; !ok {
 | |
| 			_ = compose.ProcessState(ctx, func(ctx context.Context, state NodeExecuteStatusAware) error {
 | |
| 				skipped = !state.NodeExecuted(sInfo.FromNodeKey)
 | |
| 				return nil
 | |
| 			})
 | |
| 			nodeKey2Skipped[sInfo.FromNodeKey] = skipped
 | |
| 		}
 | |
| 
 | |
| 		if skipped {
 | |
| 			resolvedNode.FieldType = schema.FieldSkipped
 | |
| 			return resolvedNode, nil
 | |
| 		}
 | |
| 
 | |
| 		if sInfo.FieldType == schema.FieldMaybeStream {
 | |
| 			if len(sInfo.SubSources) > 0 {
 | |
| 				panic("a maybe stream field should not have sub sources")
 | |
| 			}
 | |
| 
 | |
| 			var streamType schema.FieldStreamType
 | |
| 			err := compose.ProcessState(ctx, func(ctx context.Context, state DynamicStreamContainer) error {
 | |
| 				var e error
 | |
| 				streamType, e = state.GetDynamicStreamType(sInfo.FromNodeKey, sInfo.FromPath[0])
 | |
| 				return e
 | |
| 			})
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 
 | |
| 			return &schema.SourceInfo{
 | |
| 				IsIntermediate: sInfo.IsIntermediate,
 | |
| 				FieldType:      streamType,
 | |
| 				FromNodeKey:    sInfo.FromNodeKey,
 | |
| 				FromPath:       sInfo.FromPath,
 | |
| 				SubSources:     sInfo.SubSources,
 | |
| 				TypeInfo:       sInfo.TypeInfo,
 | |
| 			}, nil
 | |
| 		}
 | |
| 
 | |
| 		return resolvedNode, nil
 | |
| 	}
 | |
| 
 | |
| 	for k, sInfo := range sources {
 | |
| 		resolvedInfo, err := resolver(k, sInfo)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		resolved[k] = resolvedInfo
 | |
| 	}
 | |
| 
 | |
| 	return resolved, nil
 | |
| }
 | |
| 
 | |
| type NodeExecuteStatusAware interface {
 | |
| 	NodeExecuted(key vo.NodeKey) bool
 | |
| }
 | |
| 
 | |
| func IsStreamingField(s *schema.NodeSchema, path compose.FieldPath,
 | |
| 	sc *schema.WorkflowSchema) (schema.FieldStreamType, error) {
 | |
| 	sg, ok := s.Configs.(StreamGenerator)
 | |
| 	if !ok {
 | |
| 		return schema.FieldNotStream, nil
 | |
| 	}
 | |
| 
 | |
| 	return sg.FieldStreamType(path, s, sc)
 | |
| }
 |