0

0

MCP SDK 快速接入 DeepSeek 并添加工具!万万没想到MCP这么简单好用!

看不見的法師

看不見的法師

发布时间:2025-05-11 11:14:16

|

260人浏览过

|

来源于php中文网

原创

前言

重新整理了上篇文章,主要修正了错误的地方,加上了正确的截图和代码!感谢大家的积极指正!

这篇文章记录一下我用 MCPTypeScriptSDK 实现一个自包含的 AI 聊天应用的过程:内部包含 MCP 服务器提供上下文,客户端拿上下文再去调 LLM 接口拿回答!

正文MCP 是什么?

简单说,MCP 是一个给 AI 应用提供上下文的标准协议。你可以把它理解成一个服务标准,它规定了“资源”和“工具”的接口规范,然后通过客户端连接这些接口,就可以组合出丰富的上下文数据。比如说资源可以是“当前时间”、“用户历史记录”,工具可以是“数据库搜索”、“调用外部 API”。

它采用的是客户端-服务器架构,Server 暴露上下文能力,Client 拉取这些上下文,再拿去调语言模型生成回答,而 Transport 负责 ServerClient 的通信部分!

MCP SDK 快速接入 DeepSeek 并添加工具!万万没想到MCP这么简单好用!MCP 架构

MCP 架构

(AI 帮我画的图)

其中图片中的 Transport 层还分为:

StdioServerTransport:用于 CLI 工具对接 stdin/stdoutSSEServerTransport:用于HTTP通信StdioClientTransport:客户端以子进程方式拉起服务端,这个不常用

另外,Server 层分为:

Server 基本类:原始的类,适合自己定制功能!McpServer基于Server 封装好了可以快速使用的方法!安装依赖

用的是官方的 TypeScriptSDK

仓库:https://github.com/modelcontextprotocol/typescript-sdk

官网:https://modelcontextprotocol.io

代码语言:javascript代码运行次数:0运行复制
<code class="javascript">npm install @modelcontextprotocol/sdk axios</code>

DeepSeek 没有官方 SDK,用的是 HTTP API,所以需要 axios

记得把 API Key 放到 .env 或直接配置成环境变量,我用的 DEEPSEEK_API_KEY

实现一个 McpServer

我们先实现一个本地 McpServer,实现两个东西:

当前时间(资源)本地“知识库”搜索(工具)

代码如下:

字狐AI
字狐AI

由GPT-4 驱动的AI全能助手,支持回答复杂问题、撰写邮件、阅读文章、智能搜索

下载
代码语言:javascript代码运行次数:0运行复制
<code class="javascript">// src/server.jsimport {  McpServer,  ResourceTemplate,} from"@modelcontextprotocol/sdk/server/mcp.js";import { StdioServerTransport } from"@modelcontextprotocol/sdk/server/stdio.js";import { z } from"zod";const facts = ["公理1: 生存是文明的第一需要.","公理2: 文明不断增长和扩张,但宇宙中的物质总量保持不变.",].map((f) => f.toLowerCase());try {const server = new McpServer({    name: "mcp-cli-server",    version: "1.0.0",  });// 使用 Zod 定义工具的输入模式  server.tool(    "search_local_database",   {      query: z.string(),    },    async ({ query }) => {      console.log("Tool called with query:", query);      const queryTerms = query.toLowerCase().split(/\s+/);      const results = facts.filter((fact) =>        queryTerms.some((term) => fact.includes(term))      );      return {        content: [          {            type: "text",            text: results.length === 0 ? "未找到相关公理" : results.join("\n"),          },        ],      };    }  );// 定义资源  server.resource(    "current_time",    new ResourceTemplate("time://current", { list: undefined }),    async (uri) => ({      contents: [{ uri: uri.href, text: newDate().toLocaleString() }],    })  );await server.connect(new StdioServerTransport());console.log("Server is running...");} catch (err) {console.error("Server connection failed:", err);}</code>

这样一来,我们的服务端就能通过 MCP 协议对外暴露两个上下文能力了。

配置 MCP Client

MCP 的客户端用来连接服务器并获取资源或调用工具:

代码语言:javascript代码运行次数:0运行复制
<code class="javascript">// src/client.js;import { Client } from"@modelcontextprotocol/sdk/client/index.js";import { StdioClientTransport } from"@modelcontextprotocol/sdk/client/stdio.js";exportasyncfunction createClient() {const client = new Client({    name: "Demo",    version: "1.0.0",  });const transport = new StdioClientTransport({    command: "node",    args: ["src/server.js"],  });try {    await client.connect(transport);    console.log("Client connected successfully");  } catch (err) {    console.error("Client connection failed:", err);    throw err;  }// 可选:添加客户端方法调用后的调试return client;}</code>

连上之后,我们就可以开始调用服务端的资源和工具了。

获取上下文

我们设定一个简单的逻辑:每次用户提问,客户端都会获取当前时间;如果问题里包含 公理,那就调用搜索工具查一下本地知识库:

代码语言:javascript代码运行次数:0运行复制
<code class="javascript">async function getContext(client, question) {let currentTime = "";let additionalContext = "";try {    const resources = await client.readResource(      { uri: "time://current" },      { timeout: 15000 }    ); // 增加超时时间    console.log("Resources response:", resources);    currentTime = resources.contents[0]?.text ||      newDate().toLocaleString(); // 注意:resources 直接包含 contents  } catch (err) {    console.error("Resource read error:", err);    currentTime = newDate().toLocaleString();  }if (question.toLowerCase().includes("公理")) {    console.log("Searching for axioms...", question);    try {      const result = await client.getPrompt({        name: "search_local_database",        arguments: { query: question },      });      console.log("Tool result:", result);      additionalContext = result?.[0]?.text || "No results found.";    } catch (err) {      console.error("Tool call error:", err);      additionalContext = "Error searching database.";    }  }return { currentTime, additionalContext };}</code>
集成 DeepSeek,开始问答

DeepSeek 使用的是标准 OpenAI 接口风格,HTTP POST 请求即可。这里我们用 axios 调用:

代码语言:javascript代码运行次数:0运行复制
<code class="javascript">import axios from"axios";asyncfunction askLLM(prompt) {try {    console.log("Calling LLM with prompt:", prompt);    const res = await axios.post(      "https://api.deepseek.com/chat/completions",      {        model: "deepseek-chat",        messages: [{ role: "user", content: prompt }],        max_tokens: 2048,        stream: false,        temperature: 0.7,      },      {        headers: {          Authorization: `Bearer ${process.env.DEEPSEEK_API_KEY}`,          "Content-Type": "application/json",        },        timeout: 1000000,      }    );    console.log("LLM response:", res.data);    return res.data.choices[0].message.content;  } catch (err) {    console.error("LLM error:", err);    return"Error calling LLM.";  }}</code>

完整的代码,包含用命令行做一个简单的交互界面:

代码语言:javascript代码运行次数:0运行复制
<code class="javascript">// src/index.jsimport readline from"readline";import axios from"axios";import { createClient } from"./client.js";import { DEEPSEEK_API_KEY } from"./config.js";asyncfunction askLLM(prompt) {try {    console.log("Calling LLM with prompt:", prompt);    const res = await axios.post(      "https://api.deepseek.com/chat/completions",      {        model: "deepseek-chat",        messages: [{ role: "user", content: prompt }],        max_tokens: 2048,        stream: false,        temperature: 0.7,      },      {        headers: {          Authorization: `Bearer ${DEEPSEEK_API_KEY}`,          "Content-Type": "application/json",        },        timeout: 1000000,      }    );    return res.data.choices[0].message.content;  } catch (err) {    console.error("LLM error:", err);    return"Error calling LLM.";  }}asyncfunction getContext(client, question) {let currentTime = "";let additionalContext = "";try {    const resources = await client.readResource(      { uri: "time://current" },      { timeout: 15000 }    ); // 增加超时时间    currentTime = resources.contents[0]?.text || newDate().toLocaleString(); // 注意:resources 直接包含 contents  } catch (err) {    console.error("Resource read error:", err);    currentTime = newDate().toLocaleString();  }if (question.toLowerCase().includes("公理")) {    try {      // const result = await client.getPrompt({      //   name: "search_local_database",      //   arguments: { query: question },      // });            const toolResult = await client.callTool({        name: "search_local_database",        arguments: { query: question },      });      console.log("Tool result:", toolResult);      additionalContext = toolResult?.content?.[0]?.text || "No results found.";    } catch (err) {      console.error("Tool call error:", err);      additionalContext = "Error searching database.";    }  }return { currentTime, additionalContext };}const rl = readline.createInterface({  input: process.stdin,  output: process.stdout,});const client = await createClient();while (true) {const question = awaitnewPromise((resolve) =>    rl.question("You: ", resolve)  );if (question.toLowerCase() === "exit") {    console.log("Exiting...");    rl.close();    process.exit(0);  }// 使用上下文const context = await getContext(client, question);// 不使用上下文// const context = {};const prompt = `Time: ${context.currentTime}\nContext: ${context.additionalContext}\nQ: ${question}\nA:`;console.log("Prompt:", prompt);const answer = await askLLM(prompt);console.log('Assistant:', answer);}</code>

接着在终端运行:

代码语言:javascript代码运行次数:0运行复制
<code class="javascript"># 启动服务器node src/server.js</code>
代码语言:javascript代码运行次数:0运行复制
<code class="javascript"># 启动客户端node src/index.js</code>

运行结果:可以看到识别到关键字之后,答案更加集中在特定领域!

MCP SDK 快速接入 DeepSeek 并添加工具!万万没想到MCP这么简单好用!未命中关键词

未命中关键词

MCP SDK 快速接入 DeepSeek 并添加工具!万万没想到MCP这么简单好用!命中了关键词

命中了关键词

一些注意点

这个项目虽然小,但也踩了些坑,顺便分享几点:

MCP SDK 的 server 和 client 都是异步启动的,别忘了加上 await connect()。工具的入参和 schema 必须严格匹配,否则会抛错。

下面是我的目录结构,做个参考吧!

代码语言:javascript代码运行次数:0运行复制
<code class="javascript">mcp-mini/├── package.json├── src/│   ├── client.js│   ├── server.js│   └── index.js</code>
最后

总的来说,MCP TypeScriptSDK 用起来还是挺顺的,适合做一些轻量、模块化、支持上下文的 AI 应用。这种服务 + 客户端 + LLM 的组合模式挺适合本地测试,也方便后续扩展别的服务。

今天的分享就到这了,如果文章中有啥错误,欢迎指正!

相关文章

AI工具
AI工具

AI 智能聊天, 问答助手, AI 智能搜索, 免费无限量使用 DeepSeek R1 模型,支持联网搜索。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
TypeScript工程化开发与Vite构建优化实践
TypeScript工程化开发与Vite构建优化实践

本专题面向前端开发者,深入讲解 TypeScript 类型系统与大型项目结构设计方法,并结合 Vite 构建工具优化前端工程化流程。内容包括模块化设计、类型声明管理、代码分割、热更新原理以及构建性能调优。通过完整项目示例,帮助开发者提升代码可维护性与开发效率。

20

2026.02.13

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1507

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

403

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2254

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

37

2026.01.19

github中文官网入口 github中文版官网网页进入
github中文官网入口 github中文版官网网页进入

github中文官网入口https://docs.github.com/zh/get-started,GitHub 是一种基于云的平台,可在其中存储、共享并与他人一起编写代码。 通过将代码存储在GitHub 上的“存储库”中,你可以: “展示或共享”你的工作。 持续“跟踪和管理”对代码的更改。

2454

2026.01.21

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

373

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2093

2023.08.14

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

462

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Git 教程
Git 教程

共21课时 | 3.7万人学习

Git版本控制工具
Git版本控制工具

共8课时 | 1.5万人学习

Git中文开发手册
Git中文开发手册

共0课时 | 94人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号