模板方法模式实战:重构Agent工具审批,告别重复代码

设计模式-模板方法实战

什么是模板方法

在软件设计模式中,模板方法设计模式是一种行为型设计模式,其核心思想是在抽象类中定义一个算法的骨架(即模板),并将某些具体的实现步骤延迟到子类去完成

实战概览

在自己编写的agent框架中实现工具调用需要审批

为什么做这篇

在学习设计模式时,应当结合实战。不久前刚学到的模板方法设计模式,今天在搓自己的agent框架实现功能------工具调用需要审批时刚好用得上

代码实战

未使用模板方法的缺陷

要实现的功能:工具调用需要审批

tool_registry.py中,调用的逻辑伪代码如下

python 复制代码
# 触发审批功能
approved = self.approval_tool.approve(tool, parameters, tool_call_id)
# 根据审批结果,调用对应的功能
if not approved:
	return '未通过审批!'
#开始调用工具
tool.run(parameters, tool_call_id)
# ...省略

根据要实现的功能,不难得出在调用Tool前需要触发审批的逻辑,由于审批的实现有多种,比如可以实现以下审批:

  1. 交互式命令行审批
  2. Web界面审批
  3. AI辅助审批

鉴于有多种实现,我们需要先创建一个抽象类,让子类来实现对应的审批逻辑,上述代码块approval_tool是对应的实现类对象

python 复制代码
class ApprovalTool(ABC):
    """审批工具基类"""
    
    @abstractmethod
    def approve(self, tool: Tool, parameters: Dict[str, Any], tool_call_id: str) -> bool:
        """具体的审批逻辑,由子类实现
        
        Args:
            tool (Tool): 被调用的工具实例
            parameters (Dict[str, Any]): 工具参数
            tool_call_id (str): 工具调用ID
        """
        pass

然后让子类去实现该抽象方法,例子如下(交互式命令行审批):

python 复制代码
class DefaultApprovalTool(ApprovalTool):
    """默认审批工具"""
    def approve(self, tool: Tool, parameters: Dict[str, Any], tool_call_id: str) -> bool:
        max_attempts = self.max_attempts
        while max_attempts > 0:
            is_approved = input(f"{prompt}\n是否批准该工具调用?(y/n): ")
            if is_approved.lower() == "y":
                return True
            elif is_approved.lower() == "n":
                return False
            else:
                print(f"无效输入,请输入 'y' 或 'n'\n 剩余尝试次数: {max_attempts - 1}")
                max_attempts -= 1
        print("超过最大尝试次数,默认拒绝该工具调用")
        return False

至此,一个审批工具实现类便以完成,伪代码使用示例如下(approval_tool为对应的实现类对象):

python 复制代码
# 触发审批功能
approved = self.approval_tool.approve(tool, parameters, tool_call_id)
# 根据审批结果,调用对应的功能
if not approved:
	return '未通过审批!'
#开始调用工具
tool.run(parameters, tool_call_id)
# ...省略

缺陷展现

在完成该功能后,假设想为工具审批 添加自动批准指定工具、需要审批的指定工具 功能,那么这些逻辑都需要在实现类中编写,这些功能在代码编写上都是相似的,每个实现类都要为实现这些功能编写重复性的代码,这并不利于日后的维护和保持代码简洁优雅

模板方法设计模式可以有效解决上述缺陷

使用模板方法

再次回顾:在软件设计模式中,模板方法设计模式是一种行为型设计模式,其核心思想是在抽象类中定义一个算法的骨架(即模板),并将某些具体的实现步骤延迟到子类去完成

为了解决上述缺陷,我们需要使用模板方法设计模式,现在我们修改ApprovalTool类,将抽象方法改为do_approve()

python 复制代码
class ApprovalTool(ABC):
    """审批工具基类"""
    
    def __init__(self,
                 auto_approve_tools: Optional[List[Type[Tool]]] = None,
                 require_approval_tools: Optional[List[Type[Tool]]] = None,
                 auto_approve_if_no_rules: bool = False
                 ):
        """初始化审批工具
        假设一个tool在auto_approve_tools和require_approval_tools中都存在,则以require_approval_tools为准,即需要审批。
        假设一个tool既不在auto_approve_tools中,也不在require_approval_tools中,则根据auto_approve_if_no_rules的值来决定是否需要审批。

        
        Args:
            auto_approve_tools (Optional[List[Type[Tool]]], optional): 自动批准的工具列表. Defaults to None.
            require_approval_tools (Optional[List[Type[Tool]]], optional): 需要审批的工具列表. Defaults to None.
            auto_approve_if_no_rules (bool, optional): 当工具不在任何列表中时是否自动批准. Defaults to False.
        """
        self.auto_approve_tools = auto_approve_tools or []
        self.require_approval_tools = require_approval_tools or []
        self.auto_approve_if_no_rules = auto_approve_if_no_rules
        
    def approve(self, tool: Tool, parameters: Dict[str, Any], tool_call_id: str) -> bool:
        """
        根据工具类型和审批规则决定是否批准工具调用
        Args:
            tool (Tool): 被调用的工具实例
            parameters (Dict[str, Any]): 工具参数
            tool_call_id (str): 工具调用ID
        Returns:
            bool: 是否批准工具调用
        """
        if type(tool) in self.require_approval_tools:
            return self.do_approve(tool, parameters, tool_call_id)
        if type(tool) in self.auto_approve_tools:
            return True
        if self.auto_approve_if_no_rules == False:
            return self.do_approve(tool, parameters, tool_call_id)
        else:
            return True

    @abstractmethod
    def do_approve(self, tool: Tool, parameters: Dict[str, Any], tool_call_id: str) -> bool:
        """具体的审批逻辑,由子类实现
        
        Args:
            tool (Tool): 被调用的工具实例
            parameters (Dict[str, Any]): 工具参数
            tool_call_id (str): 工具调用ID
        """
        pass

可以看到抽象方法改成了do_approve()自动批准指定工具、需要审批的指定工具的功能 已在approve()实现,使得每个实现类都不用重复性地编写自动批准指定工具、需要审批的指定工具相关代码。我们再来看下子类(交互式命令行审批)代码:

python 复制代码
class DefaultApprovalTool(ApprovalTool):
    """默认审批工具"""

    def __init__(self,
                 auto_approve_tools: Optional[List[Type[Tool]]] = None,
                 require_approval_tools: Optional[List[Type[Tool]]] = None,
                 auto_approve_if_no_rules: bool = False,                  
                 max_attempts: int = 10):
        """初始化审批工具
        假设一个tool在auto_approve_tools和require_approval_tools中都存在,则以require_approval_tools为准,即需要审批。
        假设一个tool既不在auto_approve_tools中,也不在require_approval_tools中,则根据auto_approve_if_no_rules的值来决定是否需要审批。

        
        Args:
            auto_approve_tools (Optional[List[Type[Tool]]], optional): 自动批准的工具列表. Defaults to None.
            require_approval_tools (Optional[List[Type[Tool]]], optional): 需要审批的工具列表. Defaults to None.
            auto_approve_if_no_rules (bool, optional): 当工具不在任何列表中时是否自动批准. Defaults to False.
            max_attempts (int, optional): 最大尝试次数. Defaults to 10.
        """
        super().__init__(auto_approve_tools, require_approval_tools, auto_approve_if_no_rules)
        self.max_attempts = max_attempts

    def do_approve(self, tool: Tool, parameters: Dict[str, Any], tool_call_id: str) -> bool:
        if isinstance(tool, TerminalTool):
            current_dir = tool.current_dir
            prompt = f"{current_dir}> {parameters.get('command', '')}"
        else:
            prompt = f"工具名称: {tool.name}\n工具描述: {tool.description}\n工具参数: {parameters}"
        max_attempts = self.max_attempts
        while max_attempts > 0:
            is_approved = input(f"{prompt}\n是否批准该工具调用?(y/n): ")
            if is_approved.lower() == "y":
                return True
            elif is_approved.lower() == "n":
                return False
            else:
                print(f"无效输入,请输入 'y' 或 'n'\n 剩余尝试次数: {max_attempts - 1}")
                max_attempts -= 1
        print("超过最大尝试次数,默认拒绝该工具调用")
        return False

可以看到子类只关心实现do_approve(),而自动批准指定工具、需要审批的指定工具 不需要子类关心,这种重复的代码已在ApprovalTool类中的approve()方法编写。我们再来看下实际调用

python 复制代码
class ToolRegistry:
    """
    工具注册表,用于管理和调用工具
    """
    def __init__(self, approval_tool: Optional[ApprovalTool] = None):
        self._tools: Dict[str, Tool] = {}
        self.approval_tool = approval_tool

    def execute_tool(self, tool_call: Union[Dict[str, Any], ChatCompletionMessageFunctionToolCall]) -> Message:
        """
        执行工具

        Args:
            tool_call (Union[Dict[str, Any], ChatCompletionMessageFunctionToolCall]): 工具调用信息,llm返回的工具调用格式,包含工具名称、参数等信息

        Returns:
            Message: 工具执行结果封装的Message对象
        """
        
        # ...省略代码
        tool = self._tools[tool_name]
        if self.approval_tool:
            approved = self.approval_tool.approve(tool, parameters, tool_call_id)
            if not approved:
                return Message(role="tool", content=f"❌ 工具调用未通过用户的审批,已被用户拒绝: {tool_name}", tool_call_id=tool_call_id)

        return tool.run(parameters, tool_call_id)

参考文章

模板模式 | 菜鸟教程

相关推荐
PersonalViolet1 小时前
模板方法模式实战:重构Agent工具审批,告别重复代码
设计模式·agent
HjhIron1 小时前
Python列表与LLM接口实战:从切片到DeepSeek调用
python
搬砖的小码农_Sky1 小时前
macOS Sequoia上如何安装Python开发环境?
开发语言·python·macos
Rauser Mack1 小时前
编程纯小白,五分钟用AI做了个小游戏(附Prompt)
人工智能·python·html·prompt·ai编程
py小王子1 小时前
期刊复现| Python 实现带边缘密度与残差检验的回归拟合图
python·期刊复现
deepin_sir1 小时前
14 - 面向对象编程
开发语言·python
知识分享小能手1 小时前
Flask入门学习教程,从入门到精通,Flask智能租房——列表页 知识点详解(7)
python·学习·flask
极客小云1 小时前
【从 while 循环到可视化智能体:深入拆解 Agent Loop、Codex 风格工具调用、OpenClaw 与 Hermes 背后的技术细节】
数据库·python·大模型·agent·codex·openclaw·hermes
Artech1 小时前
[MAF的Agent管道详解-03]连接LLM的IChatClient对象
ai·llm·openai·agent·maf·agent管道·ichatclient