mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-25 07:58:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			356 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			356 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import time
 | |
| import uuid
 | |
| from os import getenv
 | |
| from typing import cast
 | |
| 
 | |
| import pytest
 | |
| 
 | |
| from core.app.entities.app_invoke_entities import InvokeFrom
 | |
| from core.workflow.entities.node_entities import NodeRunResult
 | |
| from core.workflow.entities.variable_pool import VariablePool
 | |
| from core.workflow.enums import SystemVariableKey
 | |
| from core.workflow.graph_engine.entities.graph import Graph
 | |
| from core.workflow.graph_engine.entities.graph_init_params import GraphInitParams
 | |
| from core.workflow.graph_engine.entities.graph_runtime_state import GraphRuntimeState
 | |
| from core.workflow.nodes.code.code_node import CodeNode
 | |
| from core.workflow.nodes.code.entities import CodeNodeData
 | |
| from models.enums import UserFrom
 | |
| from models.workflow import WorkflowNodeExecutionStatus, WorkflowType
 | |
| from tests.integration_tests.workflow.nodes.__mock.code_executor import setup_code_executor_mock
 | |
| 
 | |
| CODE_MAX_STRING_LENGTH = int(getenv("CODE_MAX_STRING_LENGTH", "10000"))
 | |
| 
 | |
| 
 | |
| def init_code_node(code_config: dict):
 | |
|     graph_config = {
 | |
|         "edges": [
 | |
|             {
 | |
|                 "id": "start-source-code-target",
 | |
|                 "source": "start",
 | |
|                 "target": "code",
 | |
|             },
 | |
|         ],
 | |
|         "nodes": [{"data": {"type": "start"}, "id": "start"}, code_config],
 | |
|     }
 | |
| 
 | |
|     graph = Graph.init(graph_config=graph_config)
 | |
| 
 | |
|     init_params = GraphInitParams(
 | |
|         tenant_id="1",
 | |
|         app_id="1",
 | |
|         workflow_type=WorkflowType.WORKFLOW,
 | |
|         workflow_id="1",
 | |
|         graph_config=graph_config,
 | |
|         user_id="1",
 | |
|         user_from=UserFrom.ACCOUNT,
 | |
|         invoke_from=InvokeFrom.DEBUGGER,
 | |
|         call_depth=0,
 | |
|     )
 | |
| 
 | |
|     # construct variable pool
 | |
|     variable_pool = VariablePool(
 | |
|         system_variables={SystemVariableKey.FILES: [], SystemVariableKey.USER_ID: "aaa"},
 | |
|         user_inputs={},
 | |
|         environment_variables=[],
 | |
|         conversation_variables=[],
 | |
|     )
 | |
|     variable_pool.add(["code", "123", "args1"], 1)
 | |
|     variable_pool.add(["code", "123", "args2"], 2)
 | |
| 
 | |
|     node = CodeNode(
 | |
|         id=str(uuid.uuid4()),
 | |
|         graph_init_params=init_params,
 | |
|         graph=graph,
 | |
|         graph_runtime_state=GraphRuntimeState(variable_pool=variable_pool, start_at=time.perf_counter()),
 | |
|         config=code_config,
 | |
|     )
 | |
| 
 | |
|     return node
 | |
| 
 | |
| 
 | |
| @pytest.mark.parametrize("setup_code_executor_mock", [["none"]], indirect=True)
 | |
| def test_execute_code(setup_code_executor_mock):
 | |
|     code = """
 | |
|     def main(args1: int, args2: int) -> dict:
 | |
|         return {
 | |
|             "result": args1 + args2,
 | |
|         }
 | |
|     """
 | |
|     # trim first 4 spaces at the beginning of each line
 | |
|     code = "\n".join([line[4:] for line in code.split("\n")])
 | |
| 
 | |
|     code_config = {
 | |
|         "id": "code",
 | |
|         "data": {
 | |
|             "outputs": {
 | |
|                 "result": {
 | |
|                     "type": "number",
 | |
|                 },
 | |
|             },
 | |
|             "title": "123",
 | |
|             "variables": [
 | |
|                 {
 | |
|                     "variable": "args1",
 | |
|                     "value_selector": ["1", "123", "args1"],
 | |
|                 },
 | |
|                 {"variable": "args2", "value_selector": ["1", "123", "args2"]},
 | |
|             ],
 | |
|             "answer": "123",
 | |
|             "code_language": "python3",
 | |
|             "code": code,
 | |
|         },
 | |
|     }
 | |
| 
 | |
|     node = init_code_node(code_config)
 | |
|     node.graph_runtime_state.variable_pool.add(["1", "123", "args1"], 1)
 | |
|     node.graph_runtime_state.variable_pool.add(["1", "123", "args2"], 2)
 | |
| 
 | |
|     # execute node
 | |
|     result = node._run()
 | |
|     assert isinstance(result, NodeRunResult)
 | |
|     assert result.status == WorkflowNodeExecutionStatus.SUCCEEDED
 | |
|     assert result.outputs is not None
 | |
|     assert result.outputs["result"] == 3
 | |
|     assert result.error is None
 | |
| 
 | |
| 
 | |
| @pytest.mark.parametrize("setup_code_executor_mock", [["none"]], indirect=True)
 | |
| def test_execute_code_output_validator(setup_code_executor_mock):
 | |
|     code = """
 | |
|     def main(args1: int, args2: int) -> dict:
 | |
|         return {
 | |
|             "result": args1 + args2,
 | |
|         }
 | |
|     """
 | |
|     # trim first 4 spaces at the beginning of each line
 | |
|     code = "\n".join([line[4:] for line in code.split("\n")])
 | |
| 
 | |
|     code_config = {
 | |
|         "id": "code",
 | |
|         "data": {
 | |
|             "outputs": {
 | |
|                 "result": {
 | |
|                     "type": "string",
 | |
|                 },
 | |
|             },
 | |
|             "title": "123",
 | |
|             "variables": [
 | |
|                 {
 | |
|                     "variable": "args1",
 | |
|                     "value_selector": ["1", "123", "args1"],
 | |
|                 },
 | |
|                 {"variable": "args2", "value_selector": ["1", "123", "args2"]},
 | |
|             ],
 | |
|             "answer": "123",
 | |
|             "code_language": "python3",
 | |
|             "code": code,
 | |
|         },
 | |
|     }
 | |
| 
 | |
|     node = init_code_node(code_config)
 | |
|     node.graph_runtime_state.variable_pool.add(["1", "123", "args1"], 1)
 | |
|     node.graph_runtime_state.variable_pool.add(["1", "123", "args2"], 2)
 | |
| 
 | |
|     # execute node
 | |
|     result = node._run()
 | |
|     assert isinstance(result, NodeRunResult)
 | |
|     assert result.status == WorkflowNodeExecutionStatus.FAILED
 | |
|     assert result.error == "Output variable `result` must be a string"
 | |
| 
 | |
| 
 | |
| def test_execute_code_output_validator_depth():
 | |
|     code = """
 | |
|     def main(args1: int, args2: int) -> dict:
 | |
|         return {
 | |
|             "result": {
 | |
|                 "result": args1 + args2,
 | |
|             }
 | |
|         }
 | |
|     """
 | |
|     # trim first 4 spaces at the beginning of each line
 | |
|     code = "\n".join([line[4:] for line in code.split("\n")])
 | |
| 
 | |
|     code_config = {
 | |
|         "id": "code",
 | |
|         "data": {
 | |
|             "outputs": {
 | |
|                 "string_validator": {
 | |
|                     "type": "string",
 | |
|                 },
 | |
|                 "number_validator": {
 | |
|                     "type": "number",
 | |
|                 },
 | |
|                 "number_array_validator": {
 | |
|                     "type": "array[number]",
 | |
|                 },
 | |
|                 "string_array_validator": {
 | |
|                     "type": "array[string]",
 | |
|                 },
 | |
|                 "object_validator": {
 | |
|                     "type": "object",
 | |
|                     "children": {
 | |
|                         "result": {
 | |
|                             "type": "number",
 | |
|                         },
 | |
|                         "depth": {
 | |
|                             "type": "object",
 | |
|                             "children": {
 | |
|                                 "depth": {
 | |
|                                     "type": "object",
 | |
|                                     "children": {
 | |
|                                         "depth": {
 | |
|                                             "type": "number",
 | |
|                                         }
 | |
|                                     },
 | |
|                                 }
 | |
|                             },
 | |
|                         },
 | |
|                     },
 | |
|                 },
 | |
|             },
 | |
|             "title": "123",
 | |
|             "variables": [
 | |
|                 {
 | |
|                     "variable": "args1",
 | |
|                     "value_selector": ["1", "123", "args1"],
 | |
|                 },
 | |
|                 {"variable": "args2", "value_selector": ["1", "123", "args2"]},
 | |
|             ],
 | |
|             "answer": "123",
 | |
|             "code_language": "python3",
 | |
|             "code": code,
 | |
|         },
 | |
|     }
 | |
| 
 | |
|     node = init_code_node(code_config)
 | |
| 
 | |
|     # construct result
 | |
|     result = {
 | |
|         "number_validator": 1,
 | |
|         "string_validator": "1",
 | |
|         "number_array_validator": [1, 2, 3, 3.333],
 | |
|         "string_array_validator": ["1", "2", "3"],
 | |
|         "object_validator": {"result": 1, "depth": {"depth": {"depth": 1}}},
 | |
|     }
 | |
| 
 | |
|     node.node_data = cast(CodeNodeData, node.node_data)
 | |
| 
 | |
|     # validate
 | |
|     node._transform_result(result, node.node_data.outputs)
 | |
| 
 | |
|     # construct result
 | |
|     result = {
 | |
|         "number_validator": "1",
 | |
|         "string_validator": 1,
 | |
|         "number_array_validator": ["1", "2", "3", "3.333"],
 | |
|         "string_array_validator": [1, 2, 3],
 | |
|         "object_validator": {"result": "1", "depth": {"depth": {"depth": "1"}}},
 | |
|     }
 | |
| 
 | |
|     # validate
 | |
|     with pytest.raises(ValueError):
 | |
|         node._transform_result(result, node.node_data.outputs)
 | |
| 
 | |
|     # construct result
 | |
|     result = {
 | |
|         "number_validator": 1,
 | |
|         "string_validator": (CODE_MAX_STRING_LENGTH + 1) * "1",
 | |
|         "number_array_validator": [1, 2, 3, 3.333],
 | |
|         "string_array_validator": ["1", "2", "3"],
 | |
|         "object_validator": {"result": 1, "depth": {"depth": {"depth": 1}}},
 | |
|     }
 | |
| 
 | |
|     # validate
 | |
|     with pytest.raises(ValueError):
 | |
|         node._transform_result(result, node.node_data.outputs)
 | |
| 
 | |
|     # construct result
 | |
|     result = {
 | |
|         "number_validator": 1,
 | |
|         "string_validator": "1",
 | |
|         "number_array_validator": [1, 2, 3, 3.333] * 2000,
 | |
|         "string_array_validator": ["1", "2", "3"],
 | |
|         "object_validator": {"result": 1, "depth": {"depth": {"depth": 1}}},
 | |
|     }
 | |
| 
 | |
|     # validate
 | |
|     with pytest.raises(ValueError):
 | |
|         node._transform_result(result, node.node_data.outputs)
 | |
| 
 | |
| 
 | |
| def test_execute_code_output_object_list():
 | |
|     code = """
 | |
|     def main(args1: int, args2: int) -> dict:
 | |
|         return {
 | |
|             "result": {
 | |
|                 "result": args1 + args2,
 | |
|             }
 | |
|         }
 | |
|     """
 | |
|     # trim first 4 spaces at the beginning of each line
 | |
|     code = "\n".join([line[4:] for line in code.split("\n")])
 | |
| 
 | |
|     code_config = {
 | |
|         "id": "code",
 | |
|         "data": {
 | |
|             "outputs": {
 | |
|                 "object_list": {
 | |
|                     "type": "array[object]",
 | |
|                 },
 | |
|             },
 | |
|             "title": "123",
 | |
|             "variables": [
 | |
|                 {
 | |
|                     "variable": "args1",
 | |
|                     "value_selector": ["1", "123", "args1"],
 | |
|                 },
 | |
|                 {"variable": "args2", "value_selector": ["1", "123", "args2"]},
 | |
|             ],
 | |
|             "answer": "123",
 | |
|             "code_language": "python3",
 | |
|             "code": code,
 | |
|         },
 | |
|     }
 | |
| 
 | |
|     node = init_code_node(code_config)
 | |
| 
 | |
|     # construct result
 | |
|     result = {
 | |
|         "object_list": [
 | |
|             {
 | |
|                 "result": 1,
 | |
|             },
 | |
|             {
 | |
|                 "result": 2,
 | |
|             },
 | |
|             {
 | |
|                 "result": [1, 2, 3],
 | |
|             },
 | |
|         ]
 | |
|     }
 | |
| 
 | |
|     node.node_data = cast(CodeNodeData, node.node_data)
 | |
| 
 | |
|     # validate
 | |
|     node._transform_result(result, node.node_data.outputs)
 | |
| 
 | |
|     # construct result
 | |
|     result = {
 | |
|         "object_list": [
 | |
|             {
 | |
|                 "result": 1,
 | |
|             },
 | |
|             {
 | |
|                 "result": 2,
 | |
|             },
 | |
|             {
 | |
|                 "result": [1, 2, 3],
 | |
|             },
 | |
|             1,
 | |
|         ]
 | |
|     }
 | |
| 
 | |
|     # validate
 | |
|     with pytest.raises(ValueError):
 | |
|         node._transform_result(result, node.node_data.outputs)
 | 
