【AgentScope Java新手村系列】(6)Hook与Middleware

简介: Hook与Middleware — 五类 Middleware 回调点(onModelCall/onActing等)替代 1.x Hook,实现日志埋点与限流。

第六章 Hook 与 Middleware:五类插桩点,替代 1.x Hook,覆盖模型调用与系统提示

"我们想给 agent 加上日志埋点、限流、token 计费、敏感词过滤……这些需求有一个共同点:在 agent 干活的特定时刻插一段自己的代码。1.x 用 Hook 实现,2.0 用更通用的 Middleware。"

6.0 Middleware 是什么?

Middleware = 一段可以在 agent 执行流程的特定节点被自动调用的代码。

把它类比成 Web 框架的过滤器(Filter / Interceptor):

HTTP 请求                                                                 Agent 执行
─────────                                                                 ─────────
客户端 → [Filter] → [Filter] → Controller         用户提问 → [MW] → [MW] → LLM 推理 → 工具调用
                                                                  ↑ 你可以在这 5 个节点插代码

每次 agent.call() 的执行经过 5 个阶段,每个阶段前后都可以插入 Middleware:

agent.call(msg, rt)
  │
  ├─ ① onAgent         ← 整轮调用的起点(记日志、计时、限流)
  │
  ├─ ② onSystemPrompt  ← 系统提示词拼好之后、发给 LLM 之前(动态注入时间/角色)
  │
  ├─ ③ onReasoning     ← LLM 做推理、吐文字(审计、敏感词检测)
  │
  ├─ ④ onActing        ← LLM 决定调工具了(HITL 审批、工具调用审计)
  │
  └─ ⑤ onModelCall     ← 真正打 HTTP 给 LLM API 之前/之后(token 计费、缓存、熔断)

下面这个例子——每次 agent 被调用时打一行日志——是最直观的"为什么用 Middleware":

class LoggingMiddleware extends MiddlewareBase {
   
    @Override
    public Mono<HookEvent> onAgent(MiddlewareContext ctx, HookEvent event) {
   
        System.out.println(">>> agent 被调用了, session=" + ctx.runtime().getSessionId());
        return Mono.just(event);
    }
}

挂到 agent 上之后,每次 agent.call() 都会自动打印这行日志——你不需要在每个调用点手写 System.out.println

再看一个更实用的:每次调 LLM 之前打印消耗了多少 token,还能做计费

@Override
public Mono<ModelCallResponse> onModelCall(MiddlewareContext ctx, ModelCallResponse resp) {
   
    long tokens = resp.getUsage().totalTokens();
    long cost = tokens * 2 / 1000;  // 假设 2 元/千 token
    System.out.printf("本轮消耗 %d token,费用约 %d 分%n", tokens, cost);
    return Mono.just(resp);
}

挂上之后,所有 agent 调用的 token 消耗和费用自动打印出来——不写 Middleware 的话,你需要在每一处 agent.call() 后面手动算。

Middleware 的核心价值:把"每次都要做的事"抽出来,写一次,挂一次,自动生效。

6.1 与 1.x Hook 的关系

1.x 旧写法(仅供对照,不要再写新代码)

import io.agentscope.core.hook.Hook;
import io.agentscope.core.hook.HookEvent;

class LoggingHook implements Hook {
   
    @Override
    public void onReasoning(HookEvent event) {
   
        System.out.println("[reasoning] " + event.getMessage().getTextContent());
    }
}

ReActAgent agent = ReActAgent.builder()
        ...
        .hook(new LoggingHook())
        .build();

2.0 新写法

import io.agentscope.core.hook.HookEvent;
import io.agentscope.core.middleware.MiddlewareBase;
import io.agentscope.core.middleware.MiddlewareContext;
import io.agentscope.core.middleware.ModelCallRequest;
import io.agentscope.core.middleware.ModelCallResponse;
import reactor.core.publisher.Mono;

class LoggingMiddleware extends MiddlewareBase {
   

    @Override
    public Mono<HookEvent> onReasoning(MiddlewareContext ctx, HookEvent event) {
   
        System.out.println("[reasoning] " + event.getMessage().getTextContent());
        return Mono.just(event);
    }
}

HarnessAgent agent = HarnessAgent.builder()
        ...
        .middleware(new LoggingMiddleware())
        .build();

可以看到:

  • Hookvoid 同步方法;新 Middleware 全部返回 Mono<T>,方便链式组合。
  • 旧版只能接 ReActAgent;新版既可以装在 HarnessAgent,也可以装在 ReActAgent

6.2 五个插桩点速查

重写 MiddlewareBase 的以下方法即可。每个点对应 agent 执行流程中的一个时刻:

插桩点 触发时机 典型用途
onAgent agent.call() 开始和结束 全链路日志、计时、限流
onSystemPrompt 系统提示词拼好后,发给 LLM 前 动态注入时间、角色、计划摘要
onReasoning LLM 推理过程中(每段文字输出时) 内容审计、敏感词检测
onActing LLM 决定调工具时 HITL 审批、工具调用审计
onModelCall 真正向 LLM API 发 HTTP 请求的前后 token 计费、缓存、熔断、提示词脱敏

下面逐一看每个点的代码写法:

onAgent —— 整轮 call 的入口和出口

@Override
public Mono<HookEvent> onAgent(MiddlewareContext ctx, HookEvent event) {
   
    System.out.println(">>> call 开始, session=" + ctx.runtime().getSessionId());
    return Mono.just(event);
}

用途:日志开头、整轮计时、traceId 注入、整体限流。

onReasoning —— 推理阶段

@Override
public Mono<HookEvent> onReasoning(MiddlewareContext ctx, HookEvent event) {
   
    System.out.println("[reason] " + event.getMessage().getTextContent());
    return Mono.just(event);
}

用途:思维链审计、敏感词检测、reasoning 阶段限流。

onActing —— 行动阶段(工具调用之前)

@Override
public Mono<HookEvent> onActing(MiddlewareContext ctx, HookEvent event) {
   
    System.out.println("[act] tools=" + event.getToolCalls().size());
    return Mono.just(event);
}

用途:判断 LLM 想调什么工具、决定是否要先把这次调用转人工。

onModelCall

@Override
public Mono<ModelCallRequest> onModelCall(MiddlewareContext ctx, ModelCallRequest req) {
   
    return Mono.fromSupplier(() -> {
   
        System.out.println("[model] in=" + req.getMessages().size() + " msgs");
        return req;
    });
}

@Override
public Mono<ModelCallResponse> onModelCall(MiddlewareContext ctx, ModelCallResponse resp) {
   
    return Mono.fromSupplier(() -> {
   
        System.out.println("[model] out tokens=" + resp.getUsage().totalTokens());
        return resp;
    });
}

onModelCall 是 1.x Hook 没有的位点,专门为"模型调用前后"留出来——非常适合做:

  • 提示词脱敏(脱敏后再发到模型)
  • 模型响应缓存(命中后直接返回短路 ModelCallResponse
  • token 计数 / 限流 / 计费埋点
  • 模型熔断(连续失败 N 次后直接抛错)

onSystemPrompt

@Override
public Mono<String> onSystemPrompt(MiddlewareContext ctx, String sysPrompt) {
   
    return Mono.just(sysPrompt + "\n\n[organization] 当前时间: 2026-06-07");
}

用途:动态注入时间、组织名、当前角色身份、计划模式下的 plan 摘要。

6.3 一个完整的"生产可观测"中间件

把"trace 注入 / token 计数 / 推理审计"三件事放在一个 Middleware 里:

import io.agentscope.core.RuntimeContext;
import io.agentscope.core.hook.HookEvent;
import io.agentscope.core.middleware.MiddlewareBase;
import io.agentscope.core.middleware.MiddlewareContext;
import io.agentscope.core.middleware.ModelCallRequest;
import io.agentscope.core.middleware.ModelCallResponse;
import reactor.core.publisher.Mono;

public class ObservabilityMiddleware extends MiddlewareBase {
   

    @Override
    public Mono<HookEvent> onAgent(MiddlewareContext ctx, HookEvent event) {
   
        RuntimeContext rt = ctx.runtime();
        System.out.printf("[agent] start session=%s user=%s%n",
                rt.getSessionId(), rt.getUserId());
        return Mono.just(event);
    }

    @Override
    public Mono<HookEvent> onReasoning(MiddlewareContext ctx, HookEvent event) {
   
        if (event.getMessage() != null) {
   
            System.out.println("[reason] " + event.getMessage().getTextContent());
        }
        return Mono.just(event);
    }

    @Override
    public Mono<ModelCallRequest> onModelCall(MiddlewareContext ctx, ModelCallRequest req) {
   
        long t0 = System.nanoTime();
        ctx.putAttachment("model_t0", t0);   // 把计时写到 ctx,让对应 onModelCall 回调能读到
        return Mono.just(req);
    }

    @Override
    public Mono<ModelCallResponse> onModelCall(MiddlewareContext ctx, ModelCallResponse resp) {
   
        long t0 = (long) ctx.getAttachmentOrDefault("model_t0", 0L);
        long elapsed = (System.nanoTime() - t0) / 1_000_000;
        System.out.printf("[model] %d in / %d out / %d ms%n",
                resp.getUsage().inputTokens(),
                resp.getUsage().outputTokens(),
                elapsed);
        return Mono.just(resp);
    }
}

挂载:

HarnessAgent agent = HarnessAgent.builder()
        .name("weather_bot")
        .sysPrompt("...")
        .model(model)
        .workspace(Path.of("./workspace"))
        .middleware(new ObservabilityMiddleware())
        .build();

6.4 与 Permission 系统的协作

Middleware 拦截的是任意事件,Permission 系统只拦截工具调用。两者职责不重叠:

  • Permission —— 通过规则 / mode 决定某个工具调用能不能跑(ALLOW / DENY / ASK),不能修改事件内容。
  • Middleware.onActing / Middleware.onModelCall —— 修改事件内容、记录指标、做告警。

实战上推荐:业务级"全局跨工具"的事情放 Middleware;具体"这个工具允不允许跑"放 Permission。详见第 14 章。

6.5 完整可运行示例

import io.agentscope.core.RuntimeContext;
import io.agentscope.core.message.UserMessage;
import io.agentscope.core.model.DashScopeChatModel;
import io.agentscope.harness.HarnessAgent;

import java.nio.file.Path;
import java.util.List;

public class Chapter06_Middleware {
   
    public static void main(String[] args) {
   
        HarnessAgent agent = HarnessAgent.builder()
                .name("weather_bot")
                .sysPrompt("你是一个中文天气助手,每次回答不超过 50 字。")
                .model(DashScopeChatModel.builder()
                        .apiKey(System.getenv("DASHSCOPE_API_KEY"))
                        .modelName("qwen-plus")
                        .build())
                .workspace(Path.of("./workspace"))
                .middleware(new ObservabilityMiddleware())
                .build();

        agent.call(
                List.of(new UserMessage("user", "杭州今天多少度?")),
                RuntimeContext.builder().sessionId("s-1").userId("u-1").build())
                .block();
    }
}

运行后控制台类似:

[agent] start session=s-1 user=u-1
[reason] 用户问天气
[model] 51 in / 84 out / 612 ms
[agent] end session=s-1

6.6 本章小结

  • 2.0 推荐用 Middleware 替代 1.x 的 Hook,抽象更通用、能接 Mono 响应式。
  • 五个插桩点覆盖 agent 全生命周期:onAgent / onReasoning / onActing / onModelCall / onSystemPrompt
  • onModelCall 是 1.x 没有的新位点,特别适合做提示词脱敏、响应缓存、token 计费、模型熔断。
  • MiddlewarePermission 互补:Middleware 改事件 / 做埋点,Permission 决定工具调用能不能跑。

下一章我们把同样的 Middleware 思路推到「子 Agent」,用更轻量的 SubagentDeclaration + agent_spawn 工具构建层级化系统。

目录
相关文章
|
4天前
|
人工智能 自然语言处理 文字识别
阿里云百炼Qwen3.7-Max简介:能力、优势、支持订阅计划参考
Qwen3.7-Max是阿里云百炼面向智能体时代推出的新一代旗舰模型,对标GPT-5.5、Claude Opus 4.7等闭源旗舰。该模型支持百万级token上下文窗口,具备顶级推理能力、多模态搜索与视觉理解增强、流式输出低延迟响应等核心优势,覆盖编程、办公、长周期自主执行等复杂场景。同时支持OpenAI接口兼容,便于系统快速迁移。用户可通过Token Plan团队或节省计划等订阅方式灵活调用,适合企业级高要求场景使用。
8273 37
阿里云百炼Qwen3.7-Max简介:能力、优势、支持订阅计划参考
|
4天前
|
人工智能 运维 JavaScript
阿里云Qoder CN(原通义灵码)全解析 产品形态、版本划分与技术适配说明
在AI辅助开发与智能办公工具持续普及的当下,阿里云旗下原通义灵码正式更名为Qoder CN,同时延伸出QoderWork CN、Qoder CN CLI、Qoder CN Mobile等多款配套产品,形成覆盖代码开发、日常办公、终端交互、移动端使用的完整工具矩阵。Qoder CN核心定位为AI智能编码助手,深度适配主流代码编辑器、集成开发环境以及终端场景;QoderWork CN则偏向桌面端综合办公辅助,二者面向不同使用场景,划分了多个版本档位,搭配差异化资源配额、功能权限与计费规则,同时兼容多款主流大模型。
566 4
|
4天前
|
JavaScript 定位技术 API
CodeGraph 爆火:编程 Agent 需要的不是更多上下文,而是一张提前画好的代码地图
CodeGraph 是一款爆火的本地代码智能工具,通过 tree-sitter 解析 AST 构建结构化知识图谱(存于 SQLite),为编程 Agent 提前生成“代码地图”。它显著降低 Agent 在中大型项目中的探索成本——实测工具调用减少71%、Token 降57%、速度提升46%,支持19+语言及主流框架路由识别,完全离线、无需 API Key。
539 3
CodeGraph 爆火:编程 Agent 需要的不是更多上下文,而是一张提前画好的代码地图
|
3天前
|
缓存 测试技术 API
Qwen 3.7 Plus 与 Max 实测:性价比与多模态能力差异解析(2026)
2026 年 6 月 1 日,阿里悄无声息地发布了 Qwen 3.7 Plus,距 Qwen 3.7 Max 上线刚好 11 天。同样的 1M 上下文,同样的 35 小时自治上限。但价格才是头条:Plus 是 0.40/M输入,Max是 2.50/M——便宜约 6 倍——并且还能看图、看视频。Vision Arena 上 Plus 已经排到 #16。所以这周真正值得讨论的问题不是”要不要为视觉能力买单”,而是”Max 凭什么用 6 倍价格换来 2 个百分点的 benchmark 领先”。
|
4天前
|
数据采集 人工智能 前端开发
让 Coding Agent 从黑盒到透明:阿里云 Agent 观测审计数据采集实践
AI Agent 规模化落地带来执行黑盒、行为难追溯、成本难度量三大难题。阿里云基于 OTel 标准,面向 Coding Agent、个人通用助理和框架型 Agent,推出 LoongSuite Pilot、插件及探针等无侵入采集方案,让 Agent 实现可看见、可分析、可审计、可治理。
690 148
|
4天前
|
人工智能 缓存 自然语言处理
阿里Qwen3.7-Max评测:Agent能力显著提升,耗时与调用成本大幅下降
阿里云百炼推出面向智能体的旗舰大模型Qwen3.7-Max,具备长周期自主执行能力,显著提升编程、办公自动化等复杂任务处理水平;支持MCP集成与多框架兼容,并以限时5折+100万Tokens免费试用大幅降低使用门槛,助力企业高效落地AI应用。在阿里云百炼平台快速体验:https://t.aliyun.com/U/fPVHqY
1926 10
|
4天前
|
存储 安全 Java
AgentScope Java 2.0:打造分布式、企业级智能体底座
AgentScope 2.0 面向分布式部署、稳定运行、权限安全等企业级需求全面升级,打造支持多租户隔离与长期稳定运行的企业级智能体底座。
|
4天前
|
人工智能 安全 定位技术
CodeGraph深度解析 让Claude Code工具调用直降七成的核心原理与实操教程
如今以Claude Code为代表的AI编程智能体已经成为开发者日常编码、项目重构、漏洞修复的必备工具。但在长期使用过程中,几乎所有开发者都会遇到同一个明显痛点:AI虽然具备强大的代码生成与分析能力,却常常陷入盲目探索的循环中。
1325 2
|
4天前
|
人工智能 运维 API
2026年阿里云百炼通义千问Qwen3.7-plus深度介绍 功能特性、使用优势及618大促订阅方案指南
大模型技术的普及,让AI能力逐步融入个人办公、内容创作、代码编写、企业运营、教育培训等各类场景。不同定位的模型对应不同使用需求,旗舰级模型性能强劲但使用成本偏高,轻量化模型价格低廉却难以胜任复杂任务,而介于两者之间的中端主力模型,凭借均衡的能力、亲民的定价、广泛的场景适配性,成为绝大多数个人用户、小型团队、中小企业的首选。
694 1
|
4天前
|
人工智能 弹性计算 运维
阿里云发布堡垒机智能运维Agent,运维交互进入自然语言新时代
支持自然语言运维,提升效率与安全双保障。
1183 1

热门文章

最新文章