# flake8: noqa: E501 import copy import io from contextlib import redirect_stdout from typing import Any, Optional, Type from asyncer import asyncify from lagent.actions.base_action import AsyncActionMixin, BaseAction, tool_api from lagent.actions.parser import BaseParser, JsonParser from lagent.schema import ActionReturn, ActionStatusCode class GenericRuntime: GLOBAL_DICT = {} LOCAL_DICT = None HEADERS = [] def __init__(self): self._global_vars = copy.copy(self.GLOBAL_DICT) self._local_vars = copy.copy(self.LOCAL_DICT) if self.LOCAL_DICT else None for c in self.HEADERS: self.exec_code(c) def exec_code(self, code_piece: str) -> None: exec(code_piece, self._global_vars) def eval_code(self, expr: str) -> Any: return eval(expr, self._global_vars) class PythonInterpreter(BaseAction): """A Python executor that can execute Python scripts. Args: answer_symbol (str, Optional): the answer symbol from LLM. Defaults to ``None``. answer_expr (str, Optional): the answer function name of the Python script. Defaults to ``'solution()'``. answer_from_stdout (boolean, Optional): whether the execution results is from stdout. Defaults to ``False``. timeout (int, Optional): Upper bound of waiting time for Python script execution. Defaults to ``20``. description (dict, Optional): The description of the action. Defaults to ``None``. parser (Type[BaseParser]): The parser class to process the action's inputs and outputs. Defaults to :class:`JsonParser`. """ def __init__( self, answer_symbol: Optional[str] = None, answer_expr: Optional[str] = 'solution()', answer_from_stdout: bool = False, timeout: int = 20, description: Optional[dict] = None, parser: Type[BaseParser] = JsonParser, ) -> None: super().__init__(description, parser) self.answer_symbol = answer_symbol self.answer_expr = answer_expr self.answer_from_stdout = answer_from_stdout self.timeout = timeout @tool_api def run(self, command: str) -> ActionReturn: """用来执行Python代码。代码必须是一个函数,函数名必须得是 'solution',代码对应你的思考过程。代码实例格式如下: ```python # import 依赖包 import xxx def solution(): # 初始化一些变量 variable_names_with_real_meaning = xxx # 步骤一 mid_variable = func(variable_names_with_real_meaning) # 步骤 x mid_variable = func(mid_variable) # 最后结果 final_answer = func(mid_variable) return final_answer ``` Args: command (:class:`str`): Python code snippet """ from func_timeout import FunctionTimedOut, func_set_timeout self.runtime = GenericRuntime() try: tool_return = func_set_timeout(self.timeout)(self._call)(command) except FunctionTimedOut as e: tool_return = ActionReturn(type=self.name) tool_return.errmsg = repr(e) tool_return.state = ActionStatusCode.API_ERROR return tool_return def _call(self, command: str) -> ActionReturn: tool_return = ActionReturn(type=self.name) try: if '```python' in command: command = command.split('```python')[1].split('```')[0] elif '```' in command: command = command.split('```')[1].split('```')[0] tool_return.args = dict(text='```python\n' + command + '\n```') command = command.split('\n') if self.answer_from_stdout: program_io = io.StringIO() with redirect_stdout(program_io): self.runtime.exec_code('\n'.join(command)) program_io.seek(0) res = program_io.readlines()[-1] elif self.answer_symbol: self.runtime.exec_code('\n'.join(command)) res = self.runtime._global_vars[self.answer_symbol] elif self.answer_expr: self.runtime.exec_code('\n'.join(command)) res = self.runtime.eval_code(self.answer_expr) else: self.runtime.exec_code('\n'.join(command[:-1])) res = self.runtime.eval_code(command[-1]) except Exception as e: tool_return.errmsg = repr(e) tool_return.type = self.name tool_return.state = ActionStatusCode.API_ERROR return tool_return try: tool_return.result = [dict(type='text', content=str(res))] tool_return.state = ActionStatusCode.SUCCESS except Exception as e: tool_return.errmsg = repr(e) tool_return.type = self.name tool_return.state = ActionStatusCode.API_ERROR return tool_return class AsyncPythonInterpreter(AsyncActionMixin, PythonInterpreter): """A Python executor that can execute Python scripts. Args: answer_symbol (str, Optional): the answer symbol from LLM. Defaults to ``None``. answer_expr (str, Optional): the answer function name of the Python script. Defaults to ``'solution()'``. answer_from_stdout (boolean, Optional): whether the execution results is from stdout. Defaults to ``False``. timeout (int, Optional): Upper bound of waiting time for Python script execution. Defaults to ``20``. description (dict, Optional): The description of the action. Defaults to ``None``. parser (Type[BaseParser]): The parser class to process the action's inputs and outputs. Defaults to :class:`JsonParser`. """ @tool_api @asyncify def run(self, command: str) -> ActionReturn: """用来执行Python代码。代码必须是一个函数,函数名必须得是 'solution',代码对应你的思考过程。代码实例格式如下: ```python # import 依赖包 import xxx def solution(): # 初始化一些变量 variable_names_with_real_meaning = xxx # 步骤一 mid_variable = func(variable_names_with_real_meaning) # 步骤 x mid_variable = func(mid_variable) # 最后结果 final_answer = func(mid_variable) return final_answer ``` Args: command (:class:`str`): Python code snippet """ return super().run(command)