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

推荐订阅源

Project Zero
Project Zero
D
Darknet – Hacking Tools, Hacker News & Cyber Security
Scott Helme
Scott Helme
Know Your Adversary
Know Your Adversary
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
WordPress大学
WordPress大学
AWS News Blog
AWS News Blog
小众软件
小众软件
www.infosecurity-magazine.com
www.infosecurity-magazine.com
Jina AI
Jina AI
AI
AI
美团技术团队
人人都是产品经理
人人都是产品经理
S
Secure Thoughts
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
V
Visual Studio Blog
宝玉的分享
宝玉的分享
Security Latest
Security Latest
P
Privacy & Cybersecurity Law Blog
C
Cisco Blogs
大猫的无限游戏
大猫的无限游戏
Google Online Security Blog
Google Online Security Blog
L
LINUX DO - 最新话题
罗磊的独立博客
Recent Announcements
Recent Announcements
H
Hacker News: Front Page
博客园 - 【当耐特】
K
Kaspersky official blog
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
SecWiki News
SecWiki News
Schneier on Security
Schneier on Security
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Apple Machine Learning Research
Apple Machine Learning Research
F
Full Disclosure
Google DeepMind News
Google DeepMind News
V
V2EX
博客园 - 聂微东
量子位
云风的 BLOG
云风的 BLOG
C
Check Point Blog
J
Java Code Geeks
Exploit-DB.com RSS Feed
Exploit-DB.com RSS Feed
W
WeLiveSecurity
Engineering at Meta
Engineering at Meta
V2EX - 技术
V2EX - 技术
Vercel News
Vercel News
L
LINUX DO - 热门话题
T
The Exploit Database - CXSecurity.com
L
Lohrmann on Cybersecurity
The GitHub Blog
The GitHub Blog

少数派

派早报: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);
  });
}

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

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