fix: append only current field mapping when merging inputs for parent (#2022)

main
shentongmartin 1 month ago committed by GitHub
parent 3317fa3016
commit 9e8aadc8fc
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
  1. 44
      backend/domain/workflow/internal/compose/workflow.go
  2. 174
      backend/domain/workflow/internal/compose/workflow_field_mapping_test.go

@ -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{})
})
})
}
Loading…
Cancel
Save