支持私有化部署
AI知识库

53AI知识库

学习大模型的前沿技术与行业应用场景


什么?动动嘴就能让大模型帮你整理电脑桌面?!

发布日期:2025-04-30 08:32:00 浏览次数: 1526 作者:AI进化纪
推荐语

利用AI技术简化桌面管理,释放双手从繁琐的文件整理中。
核心内容:
1. MCP协议与Golang结合实现桌面智能清理系统
2. MCP Server与Client的架构设计与通信流程
3. 代码示例:实现桌面路径获取与文件管理工具定义

杨芳贤
53A创始人/腾讯云(TVP)最具价值专家
图片
大家好,我是苦咖啡。
这样的桌面,你是否似曾相识,或者每天都能看
到? 哈哈哈


很多人的桌面常被快捷方式、临时文件、测试日志、
调试截图等等占据,手动清理费时费力。

本文将基于MCP协议,用Golang构建一个智能
清理系统
MCP Server提供文件管理工具
MCP Client通过自然语言指令触发自动化操作

生活在AI时代,我们要学会使用AI工具。

一、MCP架构设计

1.1 技术方案

我们通过Golang来实现MCP Client和Server,MCP是Model Context Protocol,用于连接大语言模型和外部工具,它采用客户端-服务器架构。
MCP Server提供工具,Client调用这些工具。所以我们需要设计一个MCP Server,提供清理桌面的工具,比如扫描临时文件、删除文件、整理文件等功能,然后Client部分需要调用这些工具,然后结合LLM(qwen2.5:3b)进行智能清理决策。
传输层我们使用stdio(或HTTP SSE),适合本地进程间通信,本次演示我们直接编译为二进制后运行(或通过Docker运行也可以)。
双进程架构:
  • MCP Server驻留后台,提供scan_temp_files(扫描)、delete_files(删除)、organize_desktop(整理)三个工具。

  • MCP Client:解析自然语言指令(如"搜索桌面以.log结尾的文件"),调用Server工具链。

1.2 通信流程

二、MCP Server实现

2.1 工具定义

我们通过代码来说明:
package main
import (  "context"  "fmt"  "github.com/mark3labs/mcp-go/mcp"  "github.com/mark3labs/mcp-go/server"  "os"  "path/filepath"  "strings"  "time")
// 获取桌面路径(跨平台兼容)func getDesktopPath() string { home, _ := os.UserHomeDir()// 创建工作目录如果不存在 path := filepath.Join(home, "Desktop")        if _, err := os.Stat(path); os.IsNotExist(err) { os.MkdirAll(path, 0755) }        return path}
func main() { s := server.NewMCPServer("DesktopCleaner""1.0", server.WithLogging(), server.WithRecovery(), )// 工具1:扫描临时文件 scanTool := mcp.NewTool("scan_temp_files", mcp.WithDescription("扫描桌面指定后缀的临时文件"), mcp.WithString("suffix", mcp.Required(), mcp.Description("文件后缀如.log/.tmp"), mcp.Pattern(`^\.[a-zA-Z0-9]+$`)), mcp.WithNumber("days", mcp.Description("查找最近N天内的文件")), ) s.AddTool(scanTool, scanHandler)// 工具2:批量删除 delTool := mcp.NewTool("delete_files", mcp.WithDescription("删除指定路径的文件"), mcp.WithArray("paths", mcp.Required(), mcp.Description("文件路径数组")), ) s.AddTool(delTool, deleteHandler)// 工具3:桌面整理 orgTool := mcp.NewTool("organize_desktop", mcp.WithDescription("按类型整理文件"), mcp.WithString("strategy", mcp.Enum("type""date"), // 按类型/日期分类 mcp.Description("整理策略")), ) s.AddTool(orgTool, organizeHandler)       if err := server.ServeStdio(s); err != nil { fmt.Printf("Server error: %v\n", err) }}

2.1 核心处理器实现

// 扫描处理器func scanHandler(ctx context.Context, req mcp.CallToolRequest) (*mcp.CallToolResult, error) {	suffix := req.Params.Arguments["suffix"].(string)       // 获取days参数,如果不存在则默认为0,不做日期过滤       var days int       if daysVal, ok := req.Params.Arguments["days"]; ok {		days, _ = daysVal.(int)	}	desktopPath := getDesktopPath()	fmt.Printf("扫描目录: %s,后缀: %s,天数: %d\n", desktopPath, suffix, days)
       var files []string err := filepath.Walk(desktopPath, func(path string, info os.FileInfo, err error) error {          // 处理错误情况,防止空指针          if err != nil { fmt.Printf("访问路径出错: %s, 错误: %v\n", path, err)               return nil // 继续扫描其他文件   }          if info == nil { fmt.Printf("文件信息为空: %s\n", path)               return nil   }
         // 检查文件后缀名,忽略大小写         if info.IsDir() {            return nil         }
         ext := filepath.Ext(path)         if strings.ToLower(ext) != strings.ToLower(suffix) {            return nil   }
         // 如果指定了天数,检查文件修改时间         if days > 0 {            if time.Since(info.ModTime()) > time.Duration(days)*24*time.Hour {                return nil // 文件太老,跳过     }         }
         // 文件符合条件,添加到结果中     files = append(files, path)  fmt.Printf("找到文件: %s\n", path)          return nil })
        if err != nil {    fmt.Printf("扫描出错: %v\n", err)            return mcp.NewToolResultErrorFromErr("扫描失败", err), nil }
result := "找到" + fmt.Sprintf("%d"len(files)) + "个文件:\n"        for _, f := range files { result += f + "\n" }
        return mcp.NewToolResultText(result), nil}// 删除处理器func deleteHandler(ctx context.Context, req mcp.CallToolRequest) (*mcp.CallToolResult, error) { paths := req.Params.Arguments["paths"].([]interface{}) results := make(map[string]string)       for _, p := range paths { path := p.(string)            if !isUnderDesktop(path) { // 安全校验              return mcp.NewToolResultError("非法路径: " + path), nil      }      err := os.Remove(path)            if err != nil { results[path] = "删除失败: " + err.Error()      } else { results[path] = "删除成功"      } }
result := "删除结果:\n"        for path, status := range results { result += path + ": " + status + "\n" }
        return mcp.NewToolResultText(result), nil}// 整理桌面处理器func organizeHandler(ctx context.Context, req mcp.CallToolRequest) (*mcp.CallToolResult, error) { strategy, _ := req.Params.Arguments["strategy"].(string)        // 实际的整理逻辑可以在这里实现
        return mcp.NewToolResultText("桌面文件已按" + strategy + "整理完成"), nil}// 路径合法性校验func isUnderDesktop(path string) bool { rel, err := filepath.Rel(getDesktopPath(), path)        return err == nil && !strings.HasPrefix(rel, "..")}

三、MCP Client实现

3.1 客户端初始化

package mainimport (  "context"  "encoding/json"  "fmt"  "github.com/mark3labs/mcp-go/client"  "github.com/mark3labs/mcp-go/mcp"  "strings"  "time")
func main() {// 连接本地Server      mcpClient, err := client.NewStdioMCPClient("./desktop_cleaner"nil)      if err != nil {        panic(err) }      defer mcpClient.Close()// Create context with timeout      ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)      defer cancel()// 初始化握手 fmt.Println("Initializing client...") initRequest := mcp.InitializeRequest{} initRequest.Params.ProtocolVersion = mcp.LATEST_PROTOCOL_VERSION initRequest.Params.ClientInfo = mcp.Implementation{ Name:    "CleanBot", Version: "1.0", }      if _, err := mcpClient.Initialize(ctx, initRequest); err != nil {          panic(err) }// 执行工具链       if err := cleanTempFiles(ctx, mcpClient); err != nil { fmt.Printf("清理失败: %v\n", err) }}

3.2 工具调用

func cleanTempFiles(ctx context.Context, client *client.Client) error {    // 步骤1:扫描.log文件    scanReq := mcp.CallToolRequest{        Request: mcp.Request{            Method: "tools/call",        },    }    scanReq.Params.Name = "scan_temp_files"    scanReq.Params.Arguments = map[string]interface{}{        "suffix"".log",        "days":   7,    }    scanRes, err := client.CallTool(ctx, scanReq)    if err != nil {        return err    }    printToolResult(scanRes)    fmt.Println()    // 解析扫描结果    files := parseFiles(scanRes)    // 步骤2:删除文件    delReq := mcp.CallToolRequest{        Request: mcp.Request{            Method: "tools/call",        },    }    delReq.Params.Name = "delete_files"    delReq.Params.Arguments = map[string]interface{}{        "paths": files,    }    if _, err := client.CallTool(ctx, delReq); err != nil {        return err    }    fmt.Printf("成功清理%d个文件\n"len(files))    return nil}// Helper function to print tool resultsfunc printToolResult(result *mcp.CallToolResult) {    for _, content := range result.Content {        if textContent, ok := content.(mcp.TextContent); ok {            fmt.Println(textContent.Text)        } else {            jsonBytes, _ := json.MarshalIndent(content, """  ")            fmt.Println(string(jsonBytes))        }    }}// 解析calltoolresultfunc parseFiles(result *mcp.CallToolResult) (files []string) {    for _, content := range result.Content {        if textContent, ok := content.(mcp.TextContent); ok {            for _, line := range strings.Split(textContent.Text, "\n") {                if strings.HasPrefix(line, "/") {                    files = append(files, line)                }            }        } else {            jsonBytes, _ := json.MarshalIndent(content, """  ")            // 解析成[]string            var result []string            if err := json.Unmarshal(jsonBytes, &result); err != nil {                fmt.Println("解析失败:", err)            }            files = append(files, result...)        }    }    return}

四、场景演示

结合LLM实现智能决策:

首先我们需要先安装mcphost工具和Ollama

# 安装mcphost工具go install github.com/mark3labs/mcphost@latest
# 启动AI协调模式  mcphost --config mcp-systemfile/clean_config.json --model ollama:qwen2.5:3b

Ollama安装qwen2.5:3b大模型,大小1.9G

最终效果

Thinking...

Assistant:

写在最后


文中我们通过“桌面清理小工具”完整介绍了Golang实现的MCP服务,由于只是在本地演示,所以使用了stdio方式,也可以提交到公有云AI托管平台(如阿里云),实现SSE方式传输。

不知大家有无发现,我们标题是“动动嘴”,也没看到动动嘴呀,哈哈,这个需要接入TTS和ASR服务,这两个也是大模型里非常重要的功能,如果大家感兴趣,我们可以单开一篇来分解。


希望本文能让大家对当下非常火热的MCP有一个整体的认识。


如果你什么问题,欢迎在评论区讨论,也可以加我微信tianxingjianlrk与我深入探讨。


参考链接:

[1] https://github.com/mark3labs/mcp-go/tree/main

[2] https://cloud.tencent.com/developer/article/2512798

[3] https://mp.weixin.qq.com/s?__biz=MzA4ODg0NDkzOA==&mid=2247510628&idx=1&sn=1ce134bf3a10a214e5590346a9f81d78&scene=21#wechat_redirect

图片
欢迎添加个人微信交流:tianxingjianlrk

53AI,企业落地大模型首选服务商

产品:场景落地咨询+大模型应用平台+行业解决方案

承诺:免费场景POC验证,效果验证后签署服务协议。零风险落地应用大模型,已交付160+中大型企业

联系我们

售前咨询
186 6662 7370
预约演示
185 8882 0121

微信扫码

添加专属顾问

回到顶部

加载中...

扫码咨询