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

MCP项目笔记三(server)

MCP Server服务端设计请求分发、回调注册、通知发送与 transport 连接Server实现了一个典型 MCP 服务端骨架的核心结构。它并不直接承载具体业务而是提供了一组通用能力包括请求接收与 JSON 解析方法分发回调覆盖异步通知发送transport 抽象与连接管理同步 / 异步运行模式从职责划分上看这个Server更接近一个通用框架通信层由transport负责协议处理与分发由Server负责具体业务逻辑则通过默认命令函数或回调覆盖注入。一、Server 的整体角色该Server的主要职责可以概括为以下几部分持有并管理transport_通过functionMap建立 method 与 handler 的映射关系通过HandleRequest()统一处理客户端请求通过OverrideCallback()提供运行时覆盖默认行为的能力通过notification_queue_与WriterLoop()实现异步通知发送通过同步或异步连接函数驱动整个生命周期从结构上看可以将它抽象为Server 协议层 分发层 生命周期管理 transport 底层通信通道 callback / plugin 业务逻辑实现在这种设计中Server不直接依赖底层通信细节也不将业务逻辑硬编码在主流程中而是将协议处理、传输层和业务扩展点区分开来。二、请求分发请求分发是整个Server的核心处理机制主要由两部分构成functionMap方法注册表HandleRequest()统一分发入口客户端请求进入系统后的基本路径如下客户端发送 JSON ↓ transport_-Read() / ReadAsync() ↓ json::parse() ↓ HandleRequest() ↓ 根据 method 查找 functionMap ↓ 调用对应 XxxCmd() ↓ 返回 json response ↓ transport_-Write()这条链路说明HandleRequest()是服务端请求处理的中心入口。1. functionMap 的作用在构造阶段Server会注册一组 method 与对应处理函数的映射例如initializepingresources/listresources/readtools/listtools/callprompts/listnotifications/...其本质是method 字符串 → std::functionjson(const json)即通过 method 字符串定位具体 handler。例如initialize对应InitializeCmdping对应PingCmdtools/call对应ToolsCallCmd这里使用 lambda 的原因在于成员函数需要绑定this而functionMap需要保存统一签名的可调用对象因此 lambda 充当了包装层。2. HandleRequest 的处理过程HandleRequest()的职责包括可选地输出请求日志校验请求中是否包含method在functionMap中查找对应 handler调用 handler 并返回其结果在找不到 method 时构造标准错误响应其核心流程可表示为JSON Request ↓ HandleRequest() ↓ 是否包含 method? ↓ ↓ 否 是 ↓ ↓ InvalidRequest functionMap.find(method) ↓ 是否找到 handler? ↓ ↓ 否 是 ↓ ↓ MethodNotFound handler(request) ↓ 返回 response这种分发方式避免了大量if-else或switch逻辑使 method 扩展与默认处理的管理集中在统一路由表中。3. request 与 notification 的区别在读循环中处理结果通常会经过如下判断if(response!nullptr){transport_-Write(response.dump());}这意味着handler 返回非空 JSON表示该消息是 request需要返回 responsehandler 返回nullptr表示该消息是 notification不返回任何响应因此请求与通知的区分并不依赖额外的独立流程而是依赖 handler 的返回值语义。三、回调注册回调注册机制用于在运行时覆盖默认 method 的处理逻辑。它的核心接口是boolServer::OverrideCallback(conststd::stringmethod,std::functionjson(constjson)function)其本质是修改functionMap中指定 method 的 handler。1. OverrideCallback 的工作方式其核心行为可以概括为functionMap[method]std::move(function);但现有实现只允许覆盖已存在的 method即先检查该 method 是否已经注册。因此它的语义不是“动态添加任意新方法”而是“替换框架已有方法的默认实现”。2. 在整体流程中的位置覆盖完成后请求链路变为客户端请求 ↓ HandleRequest() ↓ 查 functionMap ↓ 调用 handler其中 handler 既可能是构造函数中预注册的默认命令函数也可能是通过OverrideCallback()替换后的函数。也就是说回调注册改变的是“查表后的落点”。3. 框架与业务的分工这种设计体现了清晰的分工框架负责transport 生命周期管理JSON 解析请求分发线程管理默认协议方法实现业务层负责具体的工具调用逻辑资源读取逻辑prompt 获取逻辑对默认方法进行覆盖例如tools/call默认实现只是一个占位逻辑提示需要在插件中重写这表明该方法本身就是为业务层覆盖预留的。4. 设计边界该实现中的回调注册有几个明确边界默认仅允许覆盖已有 method对functionMap的修改未显式加锁运行期动态覆盖不具备线程安全保障通知类 method 若被覆盖仍应遵循 notification 语义返回nullptr因此该机制更适合在服务启动前完成注册而不是在高并发运行过程中频繁调整。四、通知发送通知发送采用的是异步生产者-消费者模型而不是调用方直接执行transport_-Write()。其基本链路如下业务代码 / 插件 ↓ SendNotification(pluginName, notification) ↓ notification_queue_ ↓ queue_cv_.notify_one() ↓ WriterLoop() ↓ transport_-Write(notification)这意味着通知发送被拆分为两个阶段业务线程提交通知写线程统一串行发送通知1. SendNotification 的职责SendNotification()的职责主要包括检查服务是否处于停止状态将通知写入notification_queue_通过条件变量唤醒写线程它本身并不直接执行底层写操作而只是充当通知生产者。2. WriterLoop 的职责WriterLoop()是发送通知的消费者线程。其典型逻辑为等待队列中出现待发送通知从队列中取出一条通知在适当的锁保护下调用transport_-Write()当前实现中WriterLoop只消费通知队列因此它在语义上是“通知专用写线程”。3. 为什么通知采用队列而不是直接写采用通知队列有三个直接原因第一避免并发写 transport多个线程若同时调用transport_-Write()容易导致输出交叉或线程安全问题。第二保证通知顺序队列天然提供 FIFO 顺序能够保证通知按入队顺序发送。第三避免业务线程阻塞如果底层 transport 写操作较慢直接写会拖慢调用方线程改为入队后由写线程统一消费可以将发送阻塞与业务线程解耦。4. response 与 notification 的写路径差异该实现中存在两类输出路径请求响应由读循环在处理 request 后直接写回transport_-Write(response.dump());服务端主动通知由SendNotification()提交到队列再由WriterLoop()发送transport_-Write(notification_to_send);因此当前实现并没有将所有输出统一到单写线程而是将 notification 单独放入异步写路径。5. output_mutex_ 的作用由于 response 与 notification 可能来自不同线程底层写操作可能并发发生因此output_mutex_同时保护队列访问transport 写操作这样可以避免响应输出与通知输出在底层通道中发生交叉。虽然这种设计的锁粒度较大但实现相对简单。五、连接 transporttransport是Server与外部世界之间的抽象通信层。Server并不直接依赖具体通信方式而是通过ITransport接口与底层 IO 解耦。1. transport 的抽象角色在类中transport_的类型为std::shared_ptrITransporttransport_;这意味着Server只依赖如下抽象能力Start()Read()/ReadAsync()Write()Stop()至于底层是标准输入输出、SSE 还是其他传输方式都由具体ITransport实现决定。这体现的是协议层与传输层分离的设计。2. Connect同步连接模式同步模式入口为boolConnect(conststd::shared_ptrITransporttransport);其主要流程为校验 transport 非空保存到transport_重置停止状态启动writer_thread_调用transport_-Start()进入循环调用transport-Read()对读到的 JSON 执行解析、分发与响应写回在结束时执行停止流程在该模式下Connect()调用后当前线程会被阻塞在读循环中。3. ConnectAsync异步连接模式异步模式入口为boolConnectAsync(conststd::shared_ptrITransporttransport);它与同步模式的区别在于同样保存 transport、初始化状态并启动写线程额外启动reader_thread_在读线程中调用transport_-ReadAsync()ConnectAsync()自身会快速返回因此异步模式中主线程负责启动读线程负责接收请求写线程负责发送通知4. Stoptransport 生命周期结束点Stop()的典型逻辑包括设置isStopping_ true调用transport_-Stop()释放transport_停止写线程并唤醒条件变量join 写线程停止读线程并 join 读线程因此transport 生命周期是由Server统一驱动的连接时注入并启动停止时关闭并释放从这个角度看Connect()/ConnectAsync()的含义更接近“将当前 Server 绑定到一个通信通道并启动运行”而不是传统意义上的主动建立网络连接。六、完整调用链将上述四部分合并后可以得到整个Server从启动到处理消息的完整调用链。1. 客户端 request → 服务端 responseServer::Connect / ConnectAsync ↓ 绑定 transport_初始化状态 ↓ 启动 WriterLoop ↓ 同步模式调用 Read() 异步模式读线程中调用 ReadAsync() ↓ 得到 json_string ↓ json::parse() ↓ HandleRequest() ↓ 根据 method 查找 functionMap ↓ 调用对应 XxxCmd() 或覆盖后的 callback ↓ 返回 response ↓ if (response ! nullptr) ↓ transport_-Write(response.dump())这条链路对应标准 request-response 处理过程。2. 客户端 notification → 服务端处理但不回包客户端发送 notification ↓ Read / ReadAsync ↓ json::parse() ↓ HandleRequest() ↓ 调用 NotificationXXXCmd() ↓ 返回 nullptr ↓ 不执行 transport_-Write()在这条链路中notification 与 request 的处理入口相同但由于返回值为空因此不会产生回包。3. 服务端主动 notification → 客户端接收通知业务代码 / 插件 ↓ SendNotification() ↓ notification_queue_ ↓ queue_cv_.notify_one() ↓ WriterLoop() ↓ transport_-Write(notification)这条链路独立于请求分发流程属于服务端主动推送机制。七、总结这份Server实现的整体结构可以概括为通过Connect()/ConnectAsync()绑定并启动 transport通过Read()/ReadAsync()接收客户端消息通过json::parse()与HandleRequest()完成请求分发通过functionMap建立 method 到 handler 的映射通过OverrideCallback()提供运行时覆盖默认逻辑的能力通过SendNotification()、通知队列和WriterLoop()实现异步主动通知通过Stop()统一收束 transport 与线程生命周期从设计上看这是一套将传输层、协议处理层与业务扩展层分离的服务端骨架实现。

相关文章:

MCP项目笔记三(server)

MCP Server服务端设计:请求分发、回调注册、通知发送与 transport 连接 Server 实现了一个典型 MCP 服务端骨架的核心结构。它并不直接承载具体业务,而是提供了一组通用能力,包括: 请求接收与 JSON 解析方法分发回调覆盖异步通知发…...

uni.requestMerchantTransfer-安卓APP中商家转账用户确认模式下,拉起页面请求用户确认收款

刚使用这个插件的时候很迷茫,不知道从何下手,问社区和专门的插件交流群几乎石层大海,索性在自己两天的摸索中终于成功了,下面是我成功的经验进行分享;目前只使用了安卓版 ①下载插件: ②下载后&#xff0c…...

智慧大屏+数字人:企业数据交互的下一个十年

【导语】 还在用鼠标点点点查数据?现在,说一句"展示本月销售数据",数字人就能秒级调取并语音播报。这不是科幻,而是正在落地的智慧大屏新形态。一、什么是"智慧大屏数字人"?简单说,就是…...

运维那些事儿(9):运维知识库,串联全流程的运维效率神器

前八期我们从IT资产管理、监控工具、U 位管理到自动化方案,搭建起了完整的精细化运维体系。但后台不少小伙伴反馈:“故障排查时翻记录半天找不到方案”“新人上手慢,老员工经验没法复用”“流程太多记混操作步骤”—— 这些痛点的核心&#x…...

铺布机在服装厂不同裁剪方式中的配套应用与工艺衔接

服装厂的裁剪方式多种多样,从传统的手工电剪到现代化的自动裁剪机,每种裁剪方式对铺布工序的要求各不相同。铺布机与裁剪方式的配套衔接,直接影响裁剪单元的整体效率和裁片质量。本文从裁剪方式角度,探讨铺布机在不同裁剪场景中的…...

直接上结论:全领域适配!最强AI论文工具 —— 千笔·专业论文写作工具

你是否曾在论文写作中感到力不从心?选题无头绪、文献难查找、框架混乱、查重率高、格式反复出错……这些困扰让无数学生在学术路上步履维艰。而如今,一款真正能解决这些问题的AI论文工具——千笔AI,正悄然改变着学术写作的方式。它不仅具备全…...

IOP-JPCS出版|EI会议2026年计算力学与智能系统国际学术会议(CMSS 2026)

高录用 | EI稳定检索 | 福耀科技大学主办 组团投稿越多,优惠则越多(2篇即可成团,联系会务组) 早投稿早审核早录用!*会务组张老师微信/电话:14748150307 • 福耀科技大学 福耀科技大学坚持社会主义办学方向…...

不必远行,武汉家门口的徒步路线,美到不想停

武汉的山水格局,为城市徒步提供了丰富的舞台。无论是中心城区的湖光山色,还是郊野的苍翠古寨,多条徒步路线各具风貌,吸引着市民用脚步丈量身边的风景。其中,东湖磨山景区以其独特的立体景观,成为一条融合了…...

【无标题】嵌入式控制器原理及应用——STM32微控制器开发工具

摘要本文围绕《嵌入式控制器原理及应用》教材第三章核心内容,聚焦STM32微控制器开发工具体系,从开发环境搭建、编译调试工具、硬件调试辅助工具三个维度展开详解,结合实操案例梳理工具选型逻辑与使用技巧,旨在帮助读者快速构建STM…...

云原生安全实战:如何在 K8s 集群中为 Ingress 开启 HTTPS 绿锁?

🔒 云原生安全实战:如何在 K8s 集群中为 Ingress 开启 HTTPS 绿锁? 在生产环境(Prod)的架构设计中,安全永远是“第一优先级”。对于涉及敏感数据的微服务平台,全站 HTTPS 加密是不可逾越的合规底…...

机票改签怎么收费?写给第一次坐飞机的你,改签操作与费用全攻略

作品声明:个人观点、仅供参考。对于没怎么坐过飞机的人来说,订完机票后最担心的不是航班延误,而是“万一那天有事去不了怎么办”。机票改签怎么收费?这个问题的答案往往藏在密密麻麻的订票条款里,看得人一头雾水。今天…...

vTaskDelay 与 vTaskDelayUntil 对比

vTaskDelay:相对延时,从调用时开始延时,周期不精准(受任务执行时间影响)vTaskDelayUntil:绝对延时,让任务固定周期精准运行,不受任务执行时间影响一、底层核心原理1. vTaskDelay( xT…...

16位元灰阶X光片增强的实践

采用多步骤产生较好的增强效果,如多尺度细节增强、gamma调整、增强锐度等等算法最终达到这样的效果,欢迎对这样技术感兴趣的人士交流讨论。联系邮箱:yinchine2019gmail.com...

AI建站工具零基础极速上手教程:10分钟从注册到发布

AI建站工具零基础极速上手教程:10分钟从注册到发布 “听说AI能建站,但我连域名是什么都搞不清楚,真的能自己搞定吗?” 这是很多小白的真实心声。面对“做网站”这件事,最大的障碍往往不是工具本身,而是那种…...

预算有限怎么选?2026 三角洲行动游戏笔记本,华硕天选6Pro 酷睿版解析

当前市场主流游戏笔记本在《三角洲行动》等3A大作高帧率需求下的表现差异显著。尤其对于预算卡在7K-8.5K区间的玩家,如何在高性能笔记本与实用体验间取得平衡,成为关键命题。本文将聚焦华硕天选6 Pro酷睿版、机械革命耀世16Ultra、联想拯救者Y7000P等热门…...

HarmonyOS6 foregroundBlurStyle 通用属性使用指南

文章目录1. 核心特性2. foregroundBlurStyle 完整参数说明2.1 基础语法2.2 BlurStyle(模糊强度)2.3 ForegroundBlurStyleOptions(可选配置)3. 完整示例代码解析3.1 代码结构说明4. 运行前置条件4.1 环境配置4.2 运行方式总结1. 核…...

google谷歌即将GMS服务设备收取激活费用?

是的,据渠道信息: 已收到邮件通知。近期关注到,谷歌正在调整 GMS 授权规则,面向设备厂商按单台收取激活费用的方案已进入落地阶段,直接影响全球安卓设备出海。本次收费并非面向普通用户,而是针对搭载 GMS 的手机、平板…...

2026大模型API中转网关选型指南:架构解析与主流方案对比

随着AI大模型在业务系统中的深入应用,如何高效整合各家主流API、降低迁移成本,成为2026年众多AI开发者和企业架构师关注的核心问题。面对层出不穷的大模型(如GPT-5.4、Gemini 3.1 pro等),采用中转网关实现业务解耦与弹…...

2026最新 SwitchHosts下载安装使用教程 及原理与本机Hosts关系(附最新安装包)

SwitchHosts下载安装使用;先说配置再讲原理SwitchHosts 下载安装使用全教程(Windows/Mac/Linux 通用)一、安装包下载二、安装步骤(根据自己系统安装,还是比较简单的)1. Windows 系统(安装版解压…...

使用org.redisson.api制作分布式锁

redis可以使用分布式锁的场景,redisson封装了该部分内容,下面看看怎么使用import lombok.extern.slf4j.Slf4j; import org.redisson.api.RLock; import org.redisson.api.RedissonClient; import org.springframework.beans.factory.annotation.Autowire…...

免费大模型 API 调用量被限制?试试这个企业级多账号轮询与高可用代理方案!

标签:大模型开发 | API 网关 | FastAPI | 负载均衡 | 架构设计 | 降本增效📖 引言:天下苦 “429 Too Many Requests” 久矣! 随着各类国产大模型(如 Kimi、DeepSeek、智谱等)以及国际巨头(OpenA…...

Dify(迪菲)LLM 应用开发平台 — 架构与组件总图

摘要:Dify(迪菲)是开源低代码生产级 LLM 应用开发平台,提供可视化工作流、RAG 知识库、Agent 智能体、模型网关及 LLMOps 能力。本图展示其四层架构(基础层 → 核心层 → 开发层 → 接入层)、核心组件及模块依赖关系。 Dify(中文名:迪菲,源自 Define + Modify / Do I…...

SEO_大幅提升自然搜索流量的七个SEO核心方法

SEO:大幅提升自然搜索流量的七个核心方法在当今数字化时代,拥有一个在搜索引擎上排名靠前的网站,不仅能带来大量的自然搜索流量,还能提升品牌知名度和销售机会。如何通过SEO(搜索引擎优化)大幅提升自然搜索流量呢&…...

水趣救生衣品牌推荐

水趣(waterfun)是上海水趣户外用品有限公司旗下专注于水域救援、水上运动户外及水乐园休闲装备的自主品牌。品牌致力于通过材料创新、场景化设计与严格的安全标准,为专业救援人员、户外运动爱好者及大众游客提供可靠的水上安全防护装备。产品…...

基于多维权重模型的女性周期护理产品评测:从合规性到功效匹配度的量化分析

很多女性受经期小腹坠凉、疼痛困扰,传统红糖水功效单一,多款补剂搭配麻烦又费钱,本次评测结合合规安全需求,对主流产品做了实测对比。本次评测权重分配为:功效匹配度40%、合规安全性30%、性价比20%、全场景适配性10%&a…...

JBL Live 780NC 耳机:开启 iPhone 用户 Auracast 新体验

JBL Live 780NC 耳机:Auracast 技术新突破 JBL 最新发布的 Live 780NC 耳机,最大亮点在于配备了 Auracast 技术。用户能够直接通过 JBL 耳机应用程序访问广播内容,这一功能的实现,让原本因苹果 iPhone 不支持 Auracast 而无缘该功…...

LLM 节点调参-AI不再胡扯

AI “胡说八道”?LLM 节点调参 Flow 搭好了,跑通了,上线了。然后用户反馈来了:“这 AI 怎么有时候像个诗人,有时候又像个杠精?”或者账单来了:“这 Token 怎么烧得比印钞还快?” 其实…...

day55 代码随想录算法训练营 图论专题9

1 今日打卡 dijkstra堆优化版 47. 参加科学大会(第六期模拟笔试) bellman_ford算法 94. 城市间货物运输 I 2 dijkstra堆优化版 2.1 思路 数据结构准备:用邻接表存储图的边信息,用优先队列(小顶堆)快速…...

欧盟小额包裹监管趋严低客单模式如何调整才能不亏

新规下的生存之道:跨境小包模式转型指南 近年来,随着全球电子商务的蓬勃发展,跨境小额包裹贸易成为许多中小卖家的主要业务模式。然而,欧盟海关监管政策的持续收紧,正在对这一传统模式构成严峻挑战。增值税&#xff0…...

锂电池测试设备采集到本地数据库的解决方案

锂电池凭借快速充放电、长循环寿命、无记忆效应等众多优势,在数码产品及电动汽车得到大规模应用,带动了电池工业的发展,同时对产品品质、生产稳定性的要求也越来越严格。因此,作为电池质检的重要设备,实现OCV测试设备的…...