惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

GbyAI
GbyAI
博客园_首页
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
阮一峰的网络日志
阮一峰的网络日志
酷 壳 – CoolShell
酷 壳 – CoolShell
博客园 - 司徒正美
V
V2EX
Cloudbric
Cloudbric
Hugging Face - Blog
Hugging Face - Blog
腾讯CDC
量子位
博客园 - 三生石上(FineUI控件)
博客园 - 叶小钗
K
Kaspersky official blog
博客园 - 【当耐特】
T
Tenable Blog
L
Lohrmann on Cybersecurity
The Cloudflare Blog
S
Schneier on Security
A
Arctic Wolf
Latest news
Latest news
C
Cyber Attacks, Cyber Crime and Cyber Security
罗磊的独立博客
T
The Exploit Database - CXSecurity.com
Cisco Talos Blog
Cisco Talos Blog
小众软件
小众软件
P
Privacy & Cybersecurity Law Blog
WordPress大学
WordPress大学
Simon Willison's Weblog
Simon Willison's Weblog
雷峰网
雷峰网
NISL@THU
NISL@THU
人人都是产品经理
人人都是产品经理
月光博客
月光博客
J
Java Code Geeks
V
Visual Studio Blog
S
Security Affairs
博客园 - Franky
T
Tailwind CSS Blog
Apple Machine Learning Research
Apple Machine Learning Research
H
Heimdal Security Blog
有赞技术团队
有赞技术团队
V2EX - 技术
V2EX - 技术
AWS News Blog
AWS News Blog
G
GRAHAM CLULEY
T
Troy Hunt's Blog
SecWiki News
SecWiki News
Spread Privacy
Spread Privacy
宝玉的分享
宝玉的分享
www.infosecurity-magazine.com
www.infosecurity-magazine.com
博客园 - 聂微东

少数派

派早报:Google 发布 Fitbit Air 等 - 少数派 「新人报到」確認需求,再開始 - 少数派 从 SOLO 独立开发者社区,我看到了越来越多开发者开始做自己的产品 - 少数派 我怎么管理那些"不常做,但总会忘"的生活事项 - 少数派 人形机器人量产元年,数据才是具身智能的“生死线” - 少数派 BuhoLaunchpad 高度还原 Mac 启动台:开发历程与思考 - 少数派 五年陪伴依然不舍,DIY 换壳后让罗技 MX Master 3 继续服役 - 少数派 新玩意 240|少数派的编辑们最近买了啥? - 少数派 一日一技|为什么你应该关闭 iOS 的键盘声音 - 少数派 我做了个插件和 Skills,一键提取任何网站的设计规范 Design.md - 少数派 住在三四线城市的你,该开始录播客了 - 少数派 甘南秘境,大白高国 - 少数派 AI的审美:谁让把我变成川内倫子 - 少数派 返工怎能不烦恼,打工人片单总有一部是你的「嘴替」 - 少数派 为了让「上厕所」更健康,我做了一个小工具 - 少数派 AI + Skill,能够让生成的文章去除 AI 味吗? - 少数派 新玩意|韶音OpenDots ONE 耳夹式耳机 - 少数派 《美满》| 在每一个春天的晚上相爱(362) - 少数派 新玩意|优篮子 PS01 MagSnap 磁吸支架 - 少数派 自我整合手记 | 我开始早睡了:用稳定规则,为自由托底 - 少数派 用龙虾(OpenClaw)两个多月,我最深的12个体会 - 少数派 听歌时间到,12 张你可能错过的 2025 华语乐坛好专辑 - 少数派 承诺能追吗 - 少数派 macOS 26启动台没了? 我做了个不一样的App启动器 - Keboard - 少数派 《四海为家的人》| INTJ对话INTJ(361) - 少数派 你发过的那些黑历史,是时候一次清干净了 - 少数派 新玩意:安安静静玩,越玩越专注:计客密码机 - 少数派 iPad 用户首次体验 Android 平板:vivo Pad6 Pro - 少数派 数据逻辑强 - 少数派 极北行+ | 一路向北,探访日本至北之地 | 001 - 少数派 万字剖析:千问App深度体验报告(2026) - 少数派 在2026年,如何真正防止别人抄袭你的作品 - 少数派 怎么用 50 块搭个 AI 语音助手?我踩了 3 天坑 - 少数派 YeeroAI:让 AI 对话真正成为知识管理的一部分 - 少数派 爬泰山 - 少数派 「旅图显影」 App 更新:这次,我们补上了一点「手感」 - 少数派 假期出门太折磨?我的 23 条经验帮你规划惬意旅行 - 少数派 工作流会变吗 - 少数派 Claude Opus 4.6 怎么用最省钱?我测了 5 种方案 - 少数派 GPT Image 2 让图文并茂不再稀罕 - 少数派 用户侧出发——什么是AI,我要不要学习? - 少数派 找片、转存、整理、播放一条龙!让你的付费网盘值回票价 - 少数派 欢迎试用!日课一问2.0插件 - 少数派 自己做的MDeditor,原本想购买 Typora 试了两次支付不成功,干脆自己做一个 - 少数派 vibe coding了一个 3MB 的小工具,让 ~/Downloads 彻底告别混乱 - 少数派 因为受不了 Mac 的风扇策略,我做了一个风扇控制工具 - 少数派 别只怪模型 - 少数派 Warp 终端的 AI 功能怎么用?我测了一周的体验 - 少数派 AI 写代码老是出 bug?这 5 个配置我后悔没早知道 - 少数派 「新玩意」苹果出相机可能就这样:Sigma BF + 45mm F2.8 DG Contemporary - 少数派 一个面向2030年的AI操作系统是什么样子的:浅谈cola这款有灵魂的Agent - 少数派 别只看写代码 - 少数派 每天解决10个问题,还是一口气攻坚解决400个? - 少数派 AI 交易机器人怎么搭?我用 Claude 跑了一周实盘 - 少数派 Maptoposter Online:把你爱的城市画成艺术海报 - 少数派 Function Calling 怎么用?我测了 3 个模型发现差距真大 - 少数派 Legend Talk:我做了个 AI 圆桌,让 160 位思想家围着你的问题转 - 少数派 如何找到自己的蓝方?在小县城寻找压力测试 - 少数派 语音输入与软件接口|2026年聊AI时,我们都聊些什么(上) - 少数派 混动已经卖爆,纯电又来补刀——钛7闪充版简直“不讲武德” - 少数派 本月玩什么|朋友收藏、识质存在、沙罗周期 - 少数派 为什么要每天坚持输出? - 少数派 Claude API 挂了好几个小时,你的项目有备用方案吗? - 少数派 Function Calling 没你想的复杂——我用它做了个有点用的工具 - 少数派 登录系统立即播放视频或者图片音乐的软件 - 少数派 我为什么创建 FlipHTML5 下载工具 - 少数派 残局没电?多品牌外设电量统一管理软件EasyBluetooth已支持RTSS游戏内显示以及AIDA64 - 少数派 前往通义路的路 - 少数派 太好看了,媲美Sun的个人导航页,NAS部署星云门户 - 少数派 乌黑嘴唇“一键检测”上线了 - 少数派 派早报:Claude AI 接入多个创意软件生态、FILCO 生产方接手品牌等 - 少数派 【更新】BearCLI、Claude 连接器与 MCP 服务器 - 少数派 记了上千条流水,还是看不懂财务?我做了一个让 AI 读懂账本的工作台 - 少数派 MINI R56 升级原厂 Sport 模式 - 少数派 新玩意 | 一棵柠檬树(仿真版) - 少数派 Momenta的“物理AI”野望,需迈过“含摩量”这道关 - 少数派 网页直接投屏控制手机!NAS一键部署PandaScrcpy,流畅丝滑可远程。 - 少数派 众测|邀你一同探索随身 AI 硬件入口 YoooClaw C·ONE - 少数派 2050大会:分享时间是真诚 参会记 - 少数派 iPad 赋能电影创作:国内首部宣纸手绘长片《燃比娃》的幕后故事 - 少数派 AI的审美:我用 8 个大模型给 100 张旅行照片打分 - 少数派 普通人如何破圈?去参加一个本地协会 - 少数派 把极空间的图标全换了,主题DIY全攻略打造你的专属NAS桌面 - 少数派 电子便签墙,帮你实现便签自由 - 少数派 我如何用三个 CLI 工具取代文档创建需求 - 少数派 原来真的有人可以玩一辈子 - 少数派 社区速递 139 | 派友热议三月买了啥、复古单反尼康 Df 体验 - 少数派 06 作品的赏析与评价 - 少数派 TDS REVIEW|索尼 WF-1000XM6 降噪真无线耳机体验 - 少数派 35.98万起售的第二代腾势D9,我看重的不是堆料,而是不凑合 - 少数派 鼠须管 Squirrel 皮肤配置指北 - 少数派 从watch ultra2换到redmi watch6 - 少数派 派早报:阿里巴巴发布视频生成模型 HappyHorse 1.0 等 - 少数派 别迷信1M - 少数派 家人们天塌了!网盘“大封杀”,多个渠道多条路,NAS部署PanHub - 少数派 AI与人勾心斗角!NAS一键部署AI狼人杀,假日休闲必备。 - 少数派 电商必备!Comfyui工作流批量生图插件,一次生成12张!支持Nano banana pro模型 - 少数派 Comfyui工作流配置Gpt-image-2模型教程,0.03/张 - 少数派 OpenClaw第三方APi怎么配置?可使用Gpt-image-2模型 - 少数派 会员社区话题精选 Ep. 103 - 少数派
给Mac邮箱装上“大脑”:自动抓取、自动总结、自动汇报 - 少数派
2025-11-20 · via 少数派

Matrix 首页推荐 

Matrix 是少数派的写作社区,我们主张分享真实的产品体验,有实用价值的经验与思考。我们会不定期挑选 Matrix 最优质的文章,展示来自用户的最真实的体验和观点。 
文章代表作者个人观点,少数派仅对标题和排版略作修改。


虽然市面上的邮件客户端或插件已具备单封邮件的总结与翻译功能,但往往「缺乏对当天所有邮件进行全局摘要的能力」。因此我借助系统自带邮件应用 (Mail app) 和扣子智能体,实现了以下的工作流:

「通过 AppleScript 获取 Mail app 的邮件内容,调用扣子智能体总结,并将总结内容发送到自己的邮箱里,每天自动执行。」

具体的实现流程可以分为三步:

  1. 获取邮件内容
  2. 调用扣子智能体总结内容
  3. 配置自动操作自动执行

最终实现的效果:每天你会准时收到一封包含所有未读邮件内容总结的邮件,如下图所示。

image.png|500
临时使用了一些旧的邮件用于测试,因此邮件内容比较老

获取邮件内容

为避免影响文章整体的阅读体验,我将获取邮件内容的完整 AppleScript 放在文末,这里主要说明几个核心逻辑:

  1. 我在脚本中指定了邮箱账户 iCloud 和邮箱文件夹 newsletter ,因为newsletter 文件夹会归集我所有的邮件订阅;
  2. 脚本只会获取未读邮件的内容,并会在成功获取内容后将邮件设置为已读;
  3. 获取到的邮件内容会以 txt 文件形式保存到指定文件夹中,使用邮件主题作为文件名。既能简化数据存储,又能实现与后续 AI 处理步骤的解耦,方便独立调试;
  4. 获取邮件的 message-ID 并拼接成 message://%3c'messageID'%3e 格式的 URI。这种 URI 可以在点击时激活 Mail app,并以弹窗形式打开对应的邮件,方便我在阅读摘要后快速打开原文。

经过这一步,我们已经获取了指定邮箱文件夹中未读邮件的内容,并以 txt 文件的形式保存在文件夹中。

调用扣子智能体总结邮件内容

我使用 JavaScript 脚本来读取上一步保存的文件内容,并调用扣子 API 总结邮件内容。当然,你也可以使用 DeepSeek 或是任意一家厂商的 API。使用扣子 API 是看中了它的异步查询功能。完整的脚本同样放在文末。

脚本执行逻辑

  1. 读取指定目录下的所有文件:脚本会遍历指定文件夹目录下的所有文件;
  2. 筛选当天创建的 .txt 文件:只处理创建日期为当天且扩展名为 .txt 的文件;
  3. 读取文件内容并调用发起对话接口:对每个有效文件,读取内容后调用 扣子 API 的 chat 接口,提交内容进行处理;
  4. 轮询等待 AI 处理完成:通过轮询 chat/retrieve 接口,等待 AI 处理状态变为 completed
  5. 查询 AI 结果并整理内容:查询 chat/message/list 接口,获取 AI 的回答内容,并将结果格式化为 HTML,累计到 allContents 变量;
  6. 所有文件处理完毕后发送邮件:当所有有效文件都处理完毕后,使用 nodemailer 通过邮箱发送一封邮件,内容为当天所有文件的 AI 总结;
  7. 邮件内容包括未读邮件数量和所有 AI 总结:邮件主题为当天日期,正文包括未读邮件数量和所有文件的 AI 总结内容。

扣子智能体设置及调用

扣子空间创建智能体,并设置好智能体的 prompt。在这个页面的地址栏中,你可以获取到 Bot_ID ,作为参数调用 API。

image.png|500

扣子 API 调用分为三步:发起对话 - 查询状态 - 查询结果。

「第一步发起对话」,相当于向扣子提交一个处理申请。

这一步你需要提供 API KeyBot_ID 以及自定义的 User_ID ,扣子会返回 Chat_idConversation_ID 作为唯一标识。

image.png|500

「第二步查询状态」,这一步相当于询问扣子我们的申请处理好了没有,当获取到 completed 状态后,再去获取处理结果,否则可能会获取到不完整的内容。

在这一步需要提供上一步返回的 Chat_IDConversation_ID 来查询结果。

image.png|500

「第三步查询结果」,这一步就是获取智能体最终的处理结果。同样需要提供第一步返回的 Chat_IDConversation_ID 来查询结果。

image.png|500

配置自动操作自动执行

有了获取邮件内容的 AppleScript 脚本和调用扣子智能体的 JavaScript 脚本,我们还需要创建一个自动操作来定期执行它们。

打开「自动操作」app 创建一个「日历提醒」类型的自动操作:

添加「运行 AppleScript」操作,将 AppleScript 脚本内容粘贴进去,记得要在外围包裹执行语句,类似:

on run{input, parameters}
    AppleScript 脚本
end run

添加「运行 Shell 脚本」操作,导航到脚本所在文件夹,并执行 JavaScript 脚本:

cd 脚本路径
/opt/homebrew/bin/node 脚本名称

添加「显示通知」来提示执行完成。

自动操作配置页面如下图:

image.png|500

创建完成后,在日历 app 中新建日程,将日程设置每天重复,并在「提醒」配置项中选择「自定义 - 打开文件 - 其他 - 选择你创建的自动操作」,就能自动执行啦!

image.png|500

至此,我们已经完成了全部的设置流程,不过在使用时请注意隐私安全,建议仅对 Newsletter 或公开资讯类邮件使用此流程,避免上传包含个人敏感信息的私人邮件。

获取邮件内容的 AppleScript

tell application "Mail"
	-- 获取 iCloud 账户中的 newsletter 邮箱
	set theAccount to account "iCloud"
	set theMailbox to mailbox "newsletter" of theAccount

	-- 获取 newsletter 邮箱中的所有未读邮件
	set unreadMessages to (every message of theMailbox whose read status is false)

	-- 遍历所有未读邮件
	repeat with eachMessage in unreadMessages
		-- 获取邮件的主题作为文件名的一部分
		set theSubject to subject of eachMessage
		-- 获取邮件的正文并转换为纯文本格式
		set theContent to content of eachMessage
		set plainTextContent to do shell script "echo " & quoted form of theContent & " | textutil -convert txt -stdin -stdout"

		-- 获取邮件的 message-ID 并生成 URI
		set messageID to message id of eachMessage
		set messageURL to "message://" & "%3c" & messageID & "%3e"
	
		-- 创建文件名,移除文件名中的不合法字符
		set cleanSubject to do shell script "echo " & quoted form of theSubject & " | tr -d '\"/:<>?\\|+[]{};=,'"
		set fileName to (cleanSubject & ".txt")
	
		-- 使用指定的 POSIX 路径并转换为 AppleScript 路径格式
		set savePath to POSIX file "保存 txt 文件的路径"
	
		-- 完整的文件路径
		set filePath to (savePath as string) & fileName
	
		-- 将邮件内容写入到指定路径的 txt 文件
		try
			set fileReference to open for access file filePath with write permission
			write plainTextContent to fileReference starting at eof
	
			-- 在文件末尾添加邮件 URI
			write return & "emailURL=" & messageURL to fileReference starting at eof
			close access fileReference
		on error errMsg
			close access file filePath
			display dialog "Error: " & errMsg
		end try
	
		-- 将邮件标记为已读
		set read status of eachMessage to true
	end repeat
end tell

return input

调用扣子智能体的脚本

const fs = require('fs');
const path = require('path');
const axios = require('axios');
const nodemailer = require('nodemailer');

const directoryPath = '存放文本文件的目录路径'; // 指定存放文本文件的目录路径
const today = new Date().toLocaleDateString('zh-CN'); // 获取系统本地日期

let allContents = ''; // 用于存储所有查询接口返回的内容
let fileCounter = 0; // 用于文件计数
let validFileCount = 0; // 有效文件计数
let emailSent = false; // 添加一个标志来跟踪邮件是否已发送

// 读取指定目录下的文件
fs.readdir(directoryPath, async (err, files) => {
  if (err) {
    console.error('无法读取目录:', err);
    return;
  }

  const filePromises = files.map(file => processFile(file));
  await Promise.all(filePromises);
  checkAllFilesProcessed();
});

async function processFile(file) {
  const filePath = path.join(directoryPath, file);

  try {
    const stats = await fs.promises.stat(filePath);
    const fileCreationDate = stats.birthtime.toLocaleDateString('zh-CN'); // 获取文件创建的本地日期
    if (fileCreationDate === today && path.extname(file) === '.txt') {
      validFileCount++; // 增加有效文件计数
      const content = await fs.promises.readFile(filePath, 'utf8');
      await callAIAPI(content, file);
    }
  } catch (err) {
    console.error('处理文件时出错:', err);
  }
}

// 使用 async/await 优化异步处理
async function callAIAPI(content, fileName) {
  const apiUrl = 'https://api.coze.cn/v3/chat';
  const headers = getHeaders();
  
  // 根据 content 的长度设置 bot_id
  const botId = content.length > 32000 ? 'bot_id' : 'bot_id';
  
  const data = {
    bot_id: botId, // 使用动态 bot_id
    user_id: '**',
    stream: false,
    auto_save_history: true,
    additional_messages: [
      {
        role: 'user',
        content: content,
        content_type: 'text'
      }
    ]
  };

  let attempts = 0;
  const maxAttempts = 5;

  while (attempts < maxAttempts) {
    try {
      const response = await axios.post(apiUrl, data, { headers, timeout: 5000 });
      console.log('API响应:', response.data);
      const { id, conversation_id } = response.data.data;
      await new Promise(resolve => setTimeout(resolve, 1000)); // 增加等待时间,1秒钟
      await pollConversationStatus(id, conversation_id, fileName);
      break; // 成功后退出循环
    } catch (error) {
      attempts++;
      console.error(`API调用错误 (尝试 ${attempts}/${maxAttempts}):`, error);
      if (attempts >= maxAttempts) {
        console.error('多次尝试后仍然失败');
      }
    }
  }
}

// 提取 headers 设置函数
function getHeaders() {
  return {
    'Authorization': 'Bearer api_key',
    'Content-Type': 'application/json'
  };
}

// 轮询对话详情接口的函数
async function pollConversationStatus(chat_id, conversation_id, fileName) {
  const retrieveUrl = `https://api.coze.cn/v3/chat/retrieve?chat_id=${chat_id}&conversation_id=${conversation_id}`;
  const headers = getHeaders();

  let pollCount = 0;
  const maxPollCount = 120;

  return new Promise((resolve, reject) => {
    const intervalId = setInterval(async () => {
      pollCount++;
      if (pollCount > maxPollCount) {
        clearInterval(intervalId);
        reject(new Error(`轮询超时,chat_id: ${chat_id}, conversation_id: ${conversation_id}`));
        return;
      }

      try {
        const response = await axios.get(retrieveUrl, { headers });
        // 取消打印轮询接口的日志
        // console.log('对话详情API响应:', response.data);
        if (response.data.data.status === 'completed') {
          clearInterval(intervalId);
          await queryAIAPI(chat_id, conversation_id, fileName);
          resolve();
        }
      } catch (error) {
        console.error('对话详情API调用错误:', error);
        clearInterval(intervalId);
        reject(error);
      }
    }, 1000); // 每秒调用一次
  });
}

// 调用查询接口的函数
async function queryAIAPI(chat_id, conversation_id, fileName) {
  const queryUrl = `https://api.coze.cn/v3/chat/message/list?chat_id=${chat_id}&conversation_id=${conversation_id}`;
  const headers = getHeaders();

  try {
    const response = await axios.get(queryUrl, { headers });
    console.log('查询API响应:', response.data);
    const contents = response.data.data
      .filter(item => item.type === 'answer') // 过滤出 type 为 'answer' 的元素
      .map(item => item.content)
      .join('\n');

    // 读取文件的最后一行以获取 MessageID
    const filePath = path.join(directoryPath, fileName);
    const fileContent = await fs.promises.readFile(filePath, 'utf8');
    const lastLine = fileContent.trim().split('\n').pop();
    const messageId = lastLine.split('=')[1]; // 提取 MessageID

    fileCounter++; // 增加文件计数
    console.log(`文件计数: ${fileCounter}`); // 打印 fileCounter 的日志
    const fileNameWithoutExt = fileName.replace(/\.[^/.]+$/, ""); // 删除文件扩展名
    allContents += `<h3><a href="${messageId}">${fileCounter}. ${fileNameWithoutExt}</a></h3><br>${contents.replace(/\n/g, '<br>')}<br><br>`; // 将换行符替换为 <br>
  } catch (error) {
    console.error('查询API调用错误:', error);
  }
}

// 检查是否所有文件都已处理完毕
function checkAllFilesProcessed() {
  if (fileCounter === validFileCount && !emailSent) {
    emailSent = true; // 设置标志,表示邮件已发送
    sendEmail(allContents); // 调用发送邮件函数
  }
}

// 发送邮件的函数
function sendEmail(contents) {
  const transporter = nodemailer.createTransport({
    host: 'smtp.qq.com',
    port: 465,
    secure: true, // 使用 SSL
    auth: {
      user: '邮箱地址', // 你的邮箱地址
      pass: '邮箱授权码' // 你的邮箱授权码
    }
  });

  const mailOptions = {
    from: '邮箱地址', // 发送者邮箱地址
    to: '邮箱地址',      // 收件人邮箱地址
    subject: `${today} 当日邮件总结`, // 邮件主题
    html: `未读邮件数量: ${validFileCount}<br><br>${contents}` // 将内容类型改为 HTML
  };

  transporter.sendMail(mailOptions, (error, info) => {
    if (error) {
      return console.error('无法发送邮件:', error);
    }
    console.log('邮件已发送:', info.response);
  });
}

> 关注 少数派小红书,感受精彩数字生活 🍃

> 实用、好用的 正版软件,少数派为你呈现 🚀