javascript - 使用 AppScript 回复 Gmail 中更改收件人的电子邮件最终会出现在新线程中
问题描述
我的邮箱中有一封电子邮件,我希望 AppScript 程序仅以我和一个特殊的 google 组作为收件人来回复它。这样做的目的是程序与我进行通信,因为程序在处理完消息后会回复消息,并在回复正文中提供有关处理的必要细节。原始消息中可能还有除我之外的其他收件人,我不希望程序将回复发送给他们。
所以我需要回复一组更改的收件人。当我在 Gmail GUI 中执行此操作时,它工作得很好,我点击回复,更改收件人,发送消息,回复最终出现在原始线程中。但是,当我在脚本中执行此操作时,回复总是以新线程结束。最初我认为 Gmail 会根据电子邮件的主题做出决定,但似乎还有更多内容(也许它最近发生了变化,因为我认为它曾经以这种方式工作)。
我尝试了多种略有不同的方法,其中之一是:
var messageBody = "foo";
var newRecipients = "me@gmail.com, my-group@gmail.com";
var messageToReplyTo = ...;
var advancedParams = {from : "my-alias@gmail.com"};
var replyDraft = messageToReplyTo.createDraftReply(messageBody);
var replySubject = replyDraft.getMessage().getSubject();
var replyBody = replyDraft.getMessage().getBody();
replyDraft.update(newRecipients, replySubject, replyBody, advancedParams);
replyDraft.send();
解决方案
为了实现这一目标,您需要做一些有趣的事情,但您可以轻松完成。您绝对应该查看草稿指南。
为了成为线程的一部分,消息或草稿必须满足以下条件:
- 请求
threadId
必须在您的请求中指定Message
或Draft.Message
您提供。- References和In-Reply-To标头必须按照 RFC 2822 标准进行设置。
- 主题标头必须匹配。
首先,您需要获取要更新的草稿的参考。这可能是最简单的使用GmailApp
:
const thread = /** get the thread somehow */;
const newBody = /** your plaintext here */;
const reply = thread.createDraftReply(newBody);
Gmail 和草稿的主要问题是 aDraft
是对服务器资源的不可变消息。如果你改变其中任何一个,你就会改变所有。因此,要更改诸如收件人地址之类的标头值,您需要完全重建邮件。这就是为什么使用GmailApp
更新草稿的方法无法维护现有线程信息的原因 - 您不能将其指定为构建新消息的高级选项之一。因此,您必须为此任务使用 Gmail REST API:
const rawMsg = Gmail.Users.Drafts.get("me", reply.getId(), {format: "raw"}).message;
要更新草稿,您需要提供以 base64 编码的 RFC 2822 格式的消息。如果您愿意将丰富格式的消息部分转换为这样的有效字符串,请务必使用非原始格式,因为您可以直接访问message.payload
.
要处理原始消息,请知道 Apps 脚本在上述调用中将描述的 base64 编码字符串转换为字节数组。然后,飞跃就是将该字节数组视为字符串字节,特别是charCode
s:
const msg_string = rawMsg.raw.reduce(function (acc, b) { return acc + String.fromCharCode(b); }, "");
console.log({message: "Converted byte[] to str", bytes: rawMsg.raw, str: msg_string});
将消息作为字符串后,您可以使用正则表达式来更新所需的标头:
const pattern = /^To: .+$/m;
var new_msg_string = msg_string.replace(pattern, "To: <....>");
// new_msg_string += ....
由于到 a 的 Gmail API 端点update
需要Draft
一个 base64 网络安全编码字符串,因此您可以计算:
const encoded_msg = Utilities.base64EncodeWebSafe(new_msg_string);
剩下的唯一一点就是执行调用(和/或send
更新的草稿)。
const resource = {
id: <draft id>, // e.g. reply.getId()
message: {
threadId: <thread id>, // e.g. thread.getId()
raw: encoded_msg
}
}
const resp = Gmail.Users.Drafts.update(resource, "me", reply.getId());
const sent_msg = Gmail.Users.Drafts.send({id: resp.id}, "me");
console.log({message: "Sent the draft", msg: sent_msg});
我并没有声称Byte
从属性返回的数组的处理Message.raw
是 100% 正确的,只是它看起来是正确的并且在我发送的测试消息中没有导致任何错误。可能还有一种更简单的方法,因为 Apps 脚本服务有一个Drafts.update
接受Blob
输入的端点,我还没有研究过如何使用它。
推荐阅读
- java - JVM:通过三个步骤签署 PDF 文件
- python - Python 找不到模块,即使在路径更新和为子目录创建空白 __init__.py 文件之后
- ios - iOS 按钮与自定义键盘上的边缘手势冲突
- sql - 如何用 SQL 确定小数类型列的小数位数
- linux - 在期望脚本中发送不可打印的字符
- javascript - 如何在 React Native 中从 ImagePicker 中检索 blob/文件
- python - Kivy:使用 kivy.storage.jsonstore 在另一个 JSON 对象中添加 JSON 对象
- python - 创建一个列表,包括行名、列名和数据框中的值
- jquery - Django Infinite Scroll 没有运行 Ajax 的东西
- html - 通过 CSS 渲染图像的 HTML 表格无法正常工作