当前位置: 首页 > article >正文

OpenClaw 大结局——接入个人

背景StreamJsonRpc 是微软官方维护的用于 .NET 和 TypeScript 的 JSON-RPC 通信库以其强大的类型安全、自动代理生成和成熟的异常处理机制著称。在 HagiCode 项目中为了通过 ACP (Agent Communication Protocol) 与外部 AI 工具如 iflow CLI、OpenCode CLI进行通信并消除早期自定义 JSON-RPC 实现带来的维护成本和潜在 Bug项目决定集成 StreamJsonRpc。然而在集成过程中遇到了流式 JSON-RPC 特有的挑战特别是在处理代理目标绑定和泛型参数识别时。为了解决这些痛点我们做了一个大胆的决定整个构建系统推倒重来。这个决定带来的变化可能比你想象的还要大——稍后我会具体说。关于 HagiCode先介绍一下本文的主角项目如果你在开发中遇到过这些烦恼多项目、多技术栈构建脚本维护成本高CI/CD 流水线配置繁琐每次改都要查文档跨平台兼容性问题层出不穷想让 AI 帮忙写代码但现有工具不够智能那么我们正在做的 HagiCode 可能你会感兴趣。HagiCode 是什么一款 AI 驱动的代码智能助手支持多语言、跨平台的代码生成与优化内置游戏化机制让编码不再枯燥为什么在这里提它本文分享的 StreamJsonRpc 集成方案正是我们在开发 HagiCode 过程中实践总结出来的。如果你觉得这套工程化方案有价值说明我们的技术品味还不错——那么 HagiCode 本身也值得关注一下。想了当前项目处于 ACP 协议集成的关键阶段面临着以下几个技术痛点和架构挑战1. 自定义实现的局限原有的 JSON-RPC 实现位于?src/HagiCode.ClaudeHelper/AcpImp/包含?JsonRpcEndpoint?和?ClientSideConnection?等组件。维护这套自定义代码成本高且缺乏成熟库的高级功能如进度报告、取消支持。2. StreamJsonRpc 集成障碍在尝试将现有的?CallbackProxyTarget?模式迁移到 StreamJsonRpc 时发现?_rpc.AddLocalRpcTarget(target)?方法无法识别通过代理模式创建的目标。具体表现为StreamJsonRpc 无法自动将泛型类型?T?的属性拆分为 RPC 方法参数导致服务器端无法正确处理客户端发起的方法调用。3. 架构分层混乱现有的?ClientSideConnection?混合了传输层WebSocket/Stdio、协议层JSON-RPC和业务层ACP Agent 接口导致职责不清且存在?AcpAgentCallbackRpcAdapter?方法绑定缺失的问题。4. 日志缺失WebSocket 传输层缺少对原始 JSON 内容的日志输出导致在调试 RPC 通信问题时难以定位是序列化问题还是网络问题。解决针对上述问题我们采用了以下系统化的解决方案从架构重构、库集成和调试增强三个维度进行优化1. 全面迁移至 StreamJsonRpc移除旧代码删除?JsonRpcEndpoint.cs、AgentSideConnection.cs?及相关的自定义序列化转换器JsonRpcMessageJsonConverter?等。集成官方库引入?StreamJsonRpc?NuGet 包利用其?JsonRpc?类处理核心通信逻辑。抽象传输层定义?IAcpTransport?接口统一处理?WebSocket?和?Stdio?两种传输模式确保协议层与传输层解耦。// IAcpTransport 接口定义public interface IAcpTransport{Task SendAsync(string message, CancellationToken cancellationToken default);Task ReceiveAsync(CancellationToken cancellationToken default);Task CloseAsync(CancellationToken cancellationToken default);}// WebSocket 传输实现public class WebSocketTransport : IAcpTransport{private readonly WebSocket _webSocket;public WebSocketTransport(WebSocket webSocket){_webSocket webSocket;}// 实现发送和接收方法// ...}// Stdio 传输实现public class StdioTransport : IAcpTransport{private readonly StreamReader _reader;private readonly StreamWriter _writer;public StdioTransport(StreamReader reader, StreamWriter writer){_reader reader;_writer writer;}// 实现发送和接收方法// ...}2. 修复代理目标识别问题分析?CallbackProxyTarget检查现有的动态代理生成逻辑确定 StreamJsonRpc 无法识别的根本原因通常是因为代理对象没有公开实际的方法签名或者使用了 StreamJsonRpc 不支持的参数类型。重构参数传递将泛型属性拆分为明确的 RPC 方法参数。不再依赖动态属性而是定义具体的 Request/Response DTO数据传输对象确保 StreamJsonRpc 能通过反射正确识别方法签名。// 原有的泛型属性方式public class CallbackProxyTarget{public Func Callback { get; set; }}// 重构后的具体方法方式public class ReadTextFileRequest{public string FilePath { get; set; }}public class ReadTextFileResponse{public string Content { get; set; }}public interface IAcpAgentCallback{Task ReadTextFileAsync(ReadTextFileRequest request);// 其他方法...}使用?Attach?替代?AddLocalRpcTarget在某些复杂场景下手动代理?JsonRpc?对象并处理?RpcConnection?可能比直接添加目标更灵活。3. 实现方法绑定与日志增强实现?AcpAgentCallbackRpcAdapter确保该组件显式实现 StreamJsonRpc 的代理接口将 ACP 协议定义的方法如?ReadTextFileAsync映射到 StreamJsonRpc 的回调处理器上。集成日志记录在 WebSocket 或 Stdio 的消息处理管道中拦截并记录 JSON-RPC 请求和响应的原始文本。利用?ILogger?在解析前和序列化后输出原始 payload以便排查格式错误。// 日志增强的传输包装器public class LoggingAcpTransport : IAcpTransport{private readonly IAcpTransport _innerTransport;private readonly ILogger _logger;public LoggingAcpTransport(IAcpTransport innerTransport, ILogger logger){_innerTransport innerTransport;_logger logger;}public async Task SendAsync(string message, CancellationToken cancellationToken default){_logger.LogTrace(Sending message: {Message}, message);await _innerTransport.SendAsync(message, cancellationToken);}public async Task ReceiveAsync(CancellationToken cancellationToken default){var message await _innerTransport.ReceiveAsync(cancellationToken);_logger.LogTrace(Received message: {Message}, message);return message;}public async Task CloseAsync(CancellationToken cancellationToken default){_logger.LogDebug(Closing connection);await _innerTransport.CloseAsync(cancellationToken);}}4. 架构分层重构传输层 (AcpRpcClient)封装 StreamJsonRpc 连接负责?InvokeAsync?和连接生命周期管理。public class AcpRpcClient : IDisposable{private readonly JsonRpc _rpc;private readonly IAcpTransport _transport;public AcpRpcClient(IAcpTransport transport){_transport transport;_rpc new JsonRpc(new StreamRpcTransport(transport));_rpc.StartListening();}public async Task InvokeAsync(string methodName, object parameters){return await _rpc.InvokeAsync(methodName, parameters);}public void Dispose(){_rpc.Dispose();_transport.Dispose();}// StreamRpcTransport 是对 IAcpTransport 的 StreamJsonRpc 适配器private class StreamRpcTransport : IDuplexPipe{// 实现 IDuplexPipe 接口// ...}}协议层 (IAcpAgentClient?/?IAcpAgentCallback)定义清晰的 client-to-agent 和 agent-to-client 接口移除?Func?这种循环依赖的工厂模式改用依赖注入或直接注册回调。实践基于 StreamJsonRpc 的最佳实践和项目经验以下是实施过程中的关键建议1. 强类型 DTO 优于动态对象StreamJsonRpc 的核心优势在于强类型。不要使用?dynamic?或?JObject?传递参数。应为每个 RPC 方法定义明确的 C# POCO 类作为参数。这不仅解决了代理目标识别问题还能在编译时发现类型错误。示例将?CallbackProxyTarget?中的泛型属性替换为?ReadTextFileRequest?和?WriteTextFileRequest?等具体类。2. 显式声明 Method Name使用?[JsonRpcMethod]?特性显式指定 RPC 方法名称不要依赖默认的方法名映射。这可以防止因命名风格差异如 PascalCase vs camelCase导致的调用失败。public interface IAcpAgentCallback{[JsonRpcMethod(readTextFile)]Task ReadTextFileAsync(ReadTextFileRequest request);[JsonRpcMethod(writeTextFile)]Task WriteTextFileAsync(WriteTextFileRequest request);}3. 利用连接状态回调StreamJsonRpc 提供了?JsonRpc.ConnectionLost?事件。务必监听此事件以处理进程意外退出或网络断开的情况这比单纯依赖 Orleans 的 Grain 失效检测更及时。_rpc.ConnectionLost (sender, e) {_logger.LogError(RPC connection lost: {Reason}, e.ToString());// 处理重连逻辑或通知用户};4. 日志分层记录Trace 级别记录完整的 JSON Request/Response 原文。Debug 级别记录方法调用栈和参数摘要。注意确保日志中不包含敏感的 Authorization Token 或大文件内容的 Base64 编码。5. 处理流式传输的特殊性StreamJsonRpc 原生支持?IAsyncEnumerable。在实现 ACP 的流式 Prompt 响应时应直接使用?IAsyncEnumerable?而不是自定义的分页逻辑。这能极大简化流式处理的代码量。public interface IAcpAgentCallback{[JsonRpcMethod(streamText)]IAsyncEnumerable StreamTextAsync(StreamTextRequest request);}6. 适配器模式 (Adapter Pattern)保持?ACPSession?和?ClientSideConnection?的分离。ACPSession?应专注于 Orleans 的状态管理和业务逻辑如消息入队通过组合而非继承的方式使用 StreamJsonRpc 连接对象。总结通过全面集成 StreamJsonRpcHagiCode 项目成功解决了原自定义实现的维护成本高、功能局限性和架构分层混乱等问题。关键改进包括采用强类型 DTO 替代动态属性提高了代码的可维护性和可靠性实现了传输层抽象和协议层分离提升了架构的清晰性增强了日志记录功能便于排查通信问题航该兑撂

相关文章:

OpenClaw 大结局——接入个人

背景 StreamJsonRpc 是微软官方维护的用于 .NET 和 TypeScript 的 JSON-RPC 通信库,以其强大的类型安全、自动代理生成和成熟的异常处理机制著称。在 HagiCode 项目中,为了通过 ACP (Agent Communication Protocol) 与外部 AI 工具(如 iflow …...

番茄小说下载器完全指南:从零开始打造个人离线图书馆

番茄小说下载器完全指南:从零开始打造个人离线图书馆 【免费下载链接】Tomato-Novel-Downloader 番茄小说下载器不精简版 项目地址: https://gitcode.com/gh_mirrors/to/Tomato-Novel-Downloader 你是否曾经想在通勤路上、旅行途中或网络不佳时阅读番茄小说&…...

DownKyi哔哩下载姬:B站视频下载的终极解决方案,轻松构建个人离线资源库

DownKyi哔哩下载姬:B站视频下载的终极解决方案,轻松构建个人离线资源库 【免费下载链接】downkyi 哔哩下载姬downkyi,哔哩哔哩网站视频下载工具,支持批量下载,支持8K、HDR、杜比视界,提供工具箱&#xff08…...

明月家书小程序|监狱寄信、看守所线上写信全攻略(2026 最新)

明月家书是宜昌高新技术企业,专注监所寄信写信服务长达 10 年,在同类写信软件中口碑靠前。对于监狱、看守所服刑人员的家属而言,一封家书是跨越高墙的牵挂,是维系亲情最直接的桥梁。但传统寄信流程繁琐、合规要求严格,…...

八大网盘直链解析:高效下载解决方案全面解析

八大网盘直链解析:高效下载解决方案全面解析 【免费下载链接】Online-disk-direct-link-download-assistant 一个基于 JavaScript 的网盘文件下载地址获取工具。基于【网盘直链下载助手】修改 ,支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼云盘 / 迅…...

抖音无水印视频下载技术实现:基于链接解析与双架构方案

抖音无水印视频下载技术实现:基于链接解析与双架构方案 【免费下载链接】douyin_downloader 抖音短视频无水印下载 win编译版本下载:https://www.lanzous.com/i9za5od 项目地址: https://gitcode.com/gh_mirrors/dou/douyin_downloader 在处理社交…...

3分钟上手TMSpeech:打造Windows本地实时语音转文字神器

3分钟上手TMSpeech:打造Windows本地实时语音转文字神器 【免费下载链接】TMSpeech 腾讯会议摸鱼工具 项目地址: https://gitcode.com/gh_mirrors/tm/TMSpeech 想要在会议中轻松走神却不错过关键信息?担心云端语音识别泄露隐私?TMSpeec…...

别再乱调散热片间距了!手把手教你用热边界层和烟囱效应搞定自然对流散热器设计

散热器设计实战:用热边界层与烟囱效应破解翅片间距迷思 当你的LED驱动电源在满载运行时突然降频,或是工控设备在高温环境下频繁死机,问题往往指向同一个罪魁祸首——散热失效。我曾亲眼见证一个价值百万的自动化产线因为散热器设计不当&#…...

ROS进阶(三)——Xacro优化与Arbotix运动控制实战(Rviz可视化)

1. Xacro宏语言:让机器人建模效率翻倍 第一次接触机器人建模时,我像大多数新手一样直接从URDF开始编写。当模型复杂度超过四个轮子时,代码就开始变得难以维护——每次修改轮子尺寸都要手动调整八个地方,稍不留神就会漏改某个参数。…...

科研必备:Mathpix+Mathtype一键转换PDF/图片公式为可编辑格式

1. 科研公式处理的痛点与解决方案 每次阅读文献时看到需要引用的公式,手动输入总让人头疼。特别是遇到复杂的分式、积分符号或矩阵时,光是调整格式就可能花费半小时。我曾为了输入一个三重积分公式,反复调试了40分钟仍不满意。 传统方法主要有…...

ComfyUI翻译节点实战:一键将中文提示词精准转化为英文

1. ComfyUI翻译节点:中文用户的效率救星 每次用ComfyUI生成图片时,最头疼的就是写英文提示词。作为一个母语中文的用户,我经常要反复切换浏览器查词典、用翻译软件,好不容易凑出来的英文提示词还经常词不达意。直到发现了ComfyUI的…...

从一笔转账看懂银行账务:客户、账户、科目与总账的完整数据流转(附实操SQL)

从一笔转账透视银行账务系统的技术架构与数据流转 当你在手机银行点击"确认转账"按钮时,系统背后发生了什么?这个看似简单的操作,实际上触发了一场精密的数据交响乐。作为金融科技从业者,理解资金在银行系统中的完整流转…...

SR-IOV与NVMe SSD的QoS性能隔离实践

1. 为什么需要SR-IOV与NVMe SSD的性能隔离? 在云计算和大数据场景中,存储性能的稳定性和隔离性一直是工程师们头疼的问题。想象一下,你租用了一台云服务器,明明配置很高,但磁盘性能却时好时坏——这可能就是多虚拟机共…...

在Ascend NPU上构建并运行onnxruntime的实战指南

1. 为什么要在Ascend NPU上运行onnxruntime? 最近几年国产AI加速硬件发展迅猛,Ascend NPU凭借出色的算力和能效比,在推理场景中表现亮眼。但很多开发者手上积累了大量ONNX格式的模型,直接迁移到新硬件平台总会遇到各种兼容性问题。…...

保姆级教程:在Ubuntu 22.04上从零编译ArmSoM Sige7的RK3588 Linux固件(含环境配置与常见错误排查)

RK3588 Linux固件编译实战:从环境搭建到烧录全流程指南 1. 开发环境准备与SDK获取 在Ubuntu 22.04系统上编译RK3588 Linux固件前,需要确保主机满足以下硬件要求: 磁盘空间:至少40GB可用空间(多系统编译需更大&#…...

从ECU复位到产线下线:深度拆解ControlDTCSetting(0x85)在汽车电子生命周期中的4种角色

ECU生命周期中的ControlDTCSetting(0x85)服务:从研发到售后的四维实践指南 当ECU完成最后一次产线测试即将装车时,产线工程师老张习惯性地在EOL终端上输入了一组UDS指令。其中那条ControlDTCSetting(0x85)服务的执行结果让他确认了这个控制单元已经准备好…...

SITS2026多模态工具链极简入门,1小时搭建可商用图文生成流水线(含私有化部署Checklist与License激活密钥生成逻辑)

第一章:SITS2026发布:多模态大模型工具链 2026奇点智能技术大会(https://ml-summit.org) SITS2026是面向工业级多模态AI研发场景推出的开源工具链,聚焦视觉-语言-时序信号联合建模能力,支持从数据预处理、多模态对齐训练、轻量化…...

如何用WeChatMsg永久珍藏微信聊天记录:你的数字记忆终极守护指南

如何用WeChatMsg永久珍藏微信聊天记录:你的数字记忆终极守护指南 【免费下载链接】WeChatMsg 提取微信聊天记录,将其导出成HTML、Word、CSV文档永久保存,对聊天记录进行分析生成年度聊天报告 项目地址: https://gitcode.com/GitHub_Trendin…...

如何高效使用BUAAthesis:北航毕业论文LaTeX模板的完整指南

如何高效使用BUAAthesis:北航毕业论文LaTeX模板的完整指南 【免费下载链接】BUAAthesis 北航毕设论文LaTeX模板 项目地址: https://gitcode.com/gh_mirrors/bu/BUAAthesis 对于每一位北航学子来说,毕业论文不仅是学术研究的结晶,更是一…...

神经网络架构图可视化宝典:轻松绘制专业深度学习图表

神经网络架构图可视化宝典:轻松绘制专业深度学习图表 【免费下载链接】Neural-Network-Architecture-Diagrams Diagrams for visualizing neural network architecture 项目地址: https://gitcode.com/gh_mirrors/ne/Neural-Network-Architecture-Diagrams 你…...

避开这些坑!百度智能云AppBuilder API调用中的5个常见错误及解决方案

百度智能云AppBuilder API实战避坑指南:从鉴权到调用的深度解析 第一次接触百度智能云AppBuilder API时,我像大多数开发者一样,以为这不过是又一个标准的RESTful接口。直到凌晨三点被报警短信惊醒——某个未做限流的API密钥在短短两小时内耗尽…...

深入解析BioBERT:高效生物医学文本挖掘的实战应用完全指南

深入解析BioBERT:高效生物医学文本挖掘的实战应用完全指南 【免费下载链接】biobert Bioinformatics2020: BioBERT: a pre-trained biomedical language representation model for biomedical text mining 项目地址: https://gitcode.com/gh_mirrors/bi/biobert …...

比对智能体,偏置群体:多智能体系统中的偏置放大测量

摘要尽管多智能体系统正越来越多地部署于复杂工作流中,但其涌现特性——尤其是偏见的累积机制——仍鲜为人知。由于现实世界中的多智能体系统过于复杂而难以全面分析,评估其伦理鲁棒性首先需要剥离其基础运行机制。本文开展了一项基线实证研究&#xff0…...

深入解析Camera矩阵:从Intrinsic到Extrinsic的完整指南

1. 相机矩阵基础:从成像原理到坐标转换 当你用手机拍照时,有没有想过镜头背后的数学魔法?相机矩阵就是这场视觉盛宴的幕后导演。简单来说,它就像一套精确的数学公式,告诉计算机如何把三维世界"压扁"成手机里…...

2026届毕业生推荐的十大降AI率神器实测分析

Ai论文网站排名(开题报告、文献综述、降aigc率、降重综合对比) TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 将AIGC率予以降低的关键核心之处在于,把文本里那些能够被机器识别出来的规律性特…...

避坑指南:在Win11上搞定PSCAD 5.0与MATLAB R2022a联合仿真(附VS2015+Intel Fortran配置)

Win11下PSCAD与MATLAB联合仿真避坑全攻略 电力系统仿真研究者们常常需要在PSCAD和MATLAB之间搭建联合仿真环境,但这个过程就像在雷区行走——稍有不慎就会触发各种兼容性问题。本文将带你避开所有已知的"地雷",从版本选择到配置修复&#xff…...

用PPO算法训练AI团队:从零实现多智能体协作捕猎(附完整代码)

用PPO算法训练AI团队:从零实现多智能体协作捕猎 想象一下,你正在设计一个虚拟生态系统,需要让一群AI控制的捕食者学会协同围猎敏捷的猎物。这不是简单的单智能体任务,而是需要多个智能体在局部观测条件下做出分布式决策的复杂场景…...

SystemVerilog三大专用always块:如何避免RTL设计中的常见陷阱

1. SystemVerilog专用always块的前世今生 在Verilog时代,我们只有一个万能的always块来处理所有类型的逻辑。这就像给你一把瑞士军刀,虽然什么都能干,但切菜不如菜刀顺手,拧螺丝不如螺丝刀专业。SystemVerilog带来的always_ff、al…...

新概念英语第一册137_A pleasant dream

Lesson 137: A pleasant dream 美好的梦 Watch the story and answer the question What would Julie like to do, if she had the money? She would like to travel and see the world.Key words and expressions football 足球(美:橄榄球)pool 赌注win …...

别再踩坑了!ZYNQ7000双核FreeRTOS通信与LwIP API调用的五个关键细节

ZYNQ7000双核FreeRTOS通信与LwIP API调用的五个关键细节 调试ZYNQ7000双核系统就像在走钢丝——稍有不慎就会掉进隐蔽的陷阱。去年我们团队在开发工业网关时,曾因一个Cache未刷新的问题浪费了两周时间。本文将分享那些手册上不会写的实战经验,特别是当Fr…...