fix: append only current field mapping when merging inputs for parent (#2022)
This commit is contained in:
@ -368,23 +368,7 @@ func (w *Workflow) getInnerWorkflow(ctx context.Context, cNode *schema.Composite
|
||||
continue
|
||||
}
|
||||
|
||||
if _, ok := carryOvers[fromNodeKey]; !ok {
|
||||
carryOvers[fromNodeKey] = make([]*compose.FieldMapping, 0)
|
||||
}
|
||||
|
||||
for _, fm := range fieldMappings {
|
||||
duplicate := false
|
||||
for _, existing := range carryOvers[fromNodeKey] {
|
||||
if fm.Equals(existing) {
|
||||
duplicate = true
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
if !duplicate {
|
||||
carryOvers[fromNodeKey] = append(carryOvers[fromNodeKey], fieldMappings...)
|
||||
}
|
||||
}
|
||||
addFieldMappingsWithDeduplication(carryOvers, fromNodeKey, fieldMappings)
|
||||
}
|
||||
}
|
||||
|
||||
@ -882,3 +866,29 @@ func (w *Workflow) resolveDependenciesAsParent(n vo.NodeKey, sourceWithPaths []*
|
||||
variableInfos: variableInfos,
|
||||
}, nil
|
||||
}
|
||||
|
||||
// addFieldMappingsWithDeduplication adds field mappings to carryOvers while avoiding duplicates
|
||||
func addFieldMappingsWithDeduplication(
|
||||
carryOvers map[vo.NodeKey][]*compose.FieldMapping,
|
||||
fromNodeKey vo.NodeKey,
|
||||
fieldMappings []*compose.FieldMapping,
|
||||
) {
|
||||
if _, ok := carryOvers[fromNodeKey]; !ok {
|
||||
carryOvers[fromNodeKey] = make([]*compose.FieldMapping, 0)
|
||||
}
|
||||
|
||||
for i := range fieldMappings {
|
||||
fm := fieldMappings[i]
|
||||
duplicate := false
|
||||
for _, existing := range carryOvers[fromNodeKey] {
|
||||
if fm.Equals(existing) {
|
||||
duplicate = true
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
if !duplicate {
|
||||
carryOvers[fromNodeKey] = append(carryOvers[fromNodeKey], fm)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@ -0,0 +1,174 @@
|
||||
/*
|
||||
* 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 (
|
||||
"testing"
|
||||
|
||||
"github.com/cloudwego/eino/compose"
|
||||
"github.com/stretchr/testify/assert"
|
||||
|
||||
"github.com/coze-dev/coze-studio/backend/domain/workflow/entity/vo"
|
||||
)
|
||||
|
||||
func TestAddFieldMappingsWithDeduplication(t *testing.T) {
|
||||
tests := []struct {
|
||||
name string
|
||||
initialCarryOvers map[vo.NodeKey][]*compose.FieldMapping
|
||||
fromNodeKey vo.NodeKey
|
||||
fieldMappings []*compose.FieldMapping
|
||||
expectedCount int
|
||||
description string
|
||||
}{
|
||||
{
|
||||
name: "empty_carry_overs",
|
||||
initialCarryOvers: make(map[vo.NodeKey][]*compose.FieldMapping),
|
||||
fromNodeKey: "node1",
|
||||
fieldMappings: []*compose.FieldMapping{
|
||||
compose.MapFieldPaths(compose.FieldPath{"input1"}, compose.FieldPath{"output1"}),
|
||||
compose.MapFieldPaths(compose.FieldPath{"input2"}, compose.FieldPath{"output2"}),
|
||||
},
|
||||
expectedCount: 2,
|
||||
description: "should add all mappings when carryOvers is empty",
|
||||
},
|
||||
{
|
||||
name: "no_duplicates",
|
||||
initialCarryOvers: map[vo.NodeKey][]*compose.FieldMapping{
|
||||
"node1": {
|
||||
compose.MapFieldPaths(compose.FieldPath{"input1"}, compose.FieldPath{"output1"}),
|
||||
},
|
||||
},
|
||||
fromNodeKey: "node1",
|
||||
fieldMappings: []*compose.FieldMapping{
|
||||
compose.MapFieldPaths(compose.FieldPath{"input2"}, compose.FieldPath{"output2"}),
|
||||
compose.MapFieldPaths(compose.FieldPath{"input3"}, compose.FieldPath{"output3"}),
|
||||
},
|
||||
expectedCount: 3,
|
||||
description: "should add new mappings when no duplicates exist",
|
||||
},
|
||||
{
|
||||
name: "with_duplicates",
|
||||
initialCarryOvers: map[vo.NodeKey][]*compose.FieldMapping{
|
||||
"node1": {
|
||||
compose.MapFieldPaths(compose.FieldPath{"input1"}, compose.FieldPath{"output1"}),
|
||||
compose.MapFieldPaths(compose.FieldPath{"input2"}, compose.FieldPath{"output2"}),
|
||||
},
|
||||
},
|
||||
fromNodeKey: "node1",
|
||||
fieldMappings: []*compose.FieldMapping{
|
||||
compose.MapFieldPaths(compose.FieldPath{"input1"}, compose.FieldPath{"output1"}), // duplicate
|
||||
compose.MapFieldPaths(compose.FieldPath{"input3"}, compose.FieldPath{"output3"}), // new
|
||||
compose.MapFieldPaths(compose.FieldPath{"input2"}, compose.FieldPath{"output2"}), // duplicate
|
||||
},
|
||||
expectedCount: 3,
|
||||
description: "should skip duplicates and only add new mappings",
|
||||
},
|
||||
{
|
||||
name: "all_duplicates",
|
||||
initialCarryOvers: map[vo.NodeKey][]*compose.FieldMapping{
|
||||
"node1": {
|
||||
compose.MapFieldPaths(compose.FieldPath{"input1"}, compose.FieldPath{"output1"}),
|
||||
compose.MapFieldPaths(compose.FieldPath{"input2"}, compose.FieldPath{"output2"}),
|
||||
},
|
||||
},
|
||||
fromNodeKey: "node1",
|
||||
fieldMappings: []*compose.FieldMapping{
|
||||
compose.MapFieldPaths(compose.FieldPath{"input1"}, compose.FieldPath{"output1"}),
|
||||
compose.MapFieldPaths(compose.FieldPath{"input2"}, compose.FieldPath{"output2"}),
|
||||
},
|
||||
expectedCount: 2,
|
||||
description: "should not add any mappings when all are duplicates",
|
||||
},
|
||||
{
|
||||
name: "new_node_key",
|
||||
initialCarryOvers: map[vo.NodeKey][]*compose.FieldMapping{
|
||||
"node1": {
|
||||
compose.MapFieldPaths(compose.FieldPath{"input1"}, compose.FieldPath{"output1"}),
|
||||
},
|
||||
},
|
||||
fromNodeKey: "node2",
|
||||
fieldMappings: []*compose.FieldMapping{
|
||||
compose.MapFieldPaths(compose.FieldPath{"input1"}, compose.FieldPath{"output1"}),
|
||||
compose.MapFieldPaths(compose.FieldPath{"input2"}, compose.FieldPath{"output2"}),
|
||||
},
|
||||
expectedCount: 2,
|
||||
description: "should add all mappings for new node key",
|
||||
},
|
||||
{
|
||||
name: "empty_field_mappings",
|
||||
initialCarryOvers: map[vo.NodeKey][]*compose.FieldMapping{
|
||||
"node1": {
|
||||
compose.MapFieldPaths(compose.FieldPath{"input1"}, compose.FieldPath{"output1"}),
|
||||
},
|
||||
},
|
||||
fromNodeKey: "node1",
|
||||
fieldMappings: []*compose.FieldMapping{},
|
||||
expectedCount: 1,
|
||||
description: "should not change carryOvers when fieldMappings is empty",
|
||||
},
|
||||
}
|
||||
|
||||
for _, tt := range tests {
|
||||
t.Run(tt.name, func(t *testing.T) {
|
||||
// Make a copy of initial carryOvers to avoid modifying the test data
|
||||
carryOvers := make(map[vo.NodeKey][]*compose.FieldMapping)
|
||||
for k, v := range tt.initialCarryOvers {
|
||||
carryOvers[k] = make([]*compose.FieldMapping, len(v))
|
||||
copy(carryOvers[k], v)
|
||||
}
|
||||
|
||||
// Call the function under test
|
||||
addFieldMappingsWithDeduplication(carryOvers, tt.fromNodeKey, tt.fieldMappings)
|
||||
|
||||
// Verify the result
|
||||
actualCount := len(carryOvers[tt.fromNodeKey])
|
||||
assert.Equal(t, tt.expectedCount, actualCount, tt.description)
|
||||
|
||||
// Verify no duplicates exist in the result
|
||||
mappings := carryOvers[tt.fromNodeKey]
|
||||
for i := 0; i < len(mappings); i++ {
|
||||
for j := i + 1; j < len(mappings); j++ {
|
||||
assert.False(t, mappings[i].Equals(mappings[j]),
|
||||
"found duplicate mappings at indices %d and %d", i, j)
|
||||
}
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
func TestAddFieldMappingsWithDeduplication_NilSafety(t *testing.T) {
|
||||
t.Run("nil_field_mappings", func(t *testing.T) {
|
||||
carryOvers := make(map[vo.NodeKey][]*compose.FieldMapping)
|
||||
fromNodeKey := vo.NodeKey("node1")
|
||||
|
||||
// Should not panic with nil fieldMappings
|
||||
assert.NotPanics(t, func() {
|
||||
addFieldMappingsWithDeduplication(carryOvers, fromNodeKey, nil)
|
||||
})
|
||||
|
||||
// Should initialize empty slice for the node
|
||||
assert.NotNil(t, carryOvers[fromNodeKey])
|
||||
assert.Equal(t, 0, len(carryOvers[fromNodeKey]))
|
||||
})
|
||||
|
||||
t.Run("nil_carry_overs", func(t *testing.T) {
|
||||
// Should panic with nil carryOvers - this is expected behavior
|
||||
assert.Panics(t, func() {
|
||||
addFieldMappingsWithDeduplication(nil, "node1", []*compose.FieldMapping{})
|
||||
})
|
||||
})
|
||||
}
|
||||
Reference in New Issue
Block a user