- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我的邮箱中有一封电子邮件,我希望 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();
最佳答案
为了实现这一目标,您需要做几件有趣的事情,但您可以轻松完成。您一定要查看 Drafts 的指南.
In order to be part of a thread, a message or draft must meet the following criteria:
- The requested
threadId
must be specified on theMessage
orDraft.Message
you supply with your request.- The References and In-Reply-To headers must be set in compliance with the RFC 2822 standard.
- The Subject headers must match.
首先,您需要获得对要更新的草稿的引用。这可能是使用 GmailApp
最简单的方法:
const thread = /** get the thread somehow */;
const newBody = /** your plaintext here */;
const reply = thread.createDraftReply(newBody);
Gmail 和 Drafts 的主要问题是 Draft
是发送到服务器资源的不可变消息。如果你改变了其中的任何一个,你就改变了它的全部。因此,要更改收件人地址等 header 值,您需要完全重建邮件。这就是为什么使用 GmailApp
methods to update a draft无法维护现有线程信息 - 您不能将其指定为构建新消息的高级选项之一。因此,您必须使用 Gmail REST API 来完成此任务:
const rawMsg = Gmail.Users.Drafts.get("me", reply.getId(), {format: "raw"}).message;
要更新草稿,您需要提供以 base64 编码的 RFC 2822 格式消息。如果您愿意将丰富格式的消息部分转换为这样一个有效的字符串,请务必使用非原始格式,因为您可以直接访问 message.payload
中的 header 。 .
要处理原始消息,请知道 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});
一旦您将消息作为字符串,您就可以使用正则表达式来更新所需的 header :
const pattern = /^To: .+$/m;
var new_msg_string = msg_string.replace(pattern, "To: <....>");
// new_msg_string += ....
由于 Gmail API 端点为 update
a 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});
我并不声称对从 Message.raw
属性返回的 Byte
数组的处理是 100% 正确的,只是它看起来是正确的,但实际上并不正确导致我发送的测试消息出现任何错误。可能还有一种更简单的方法,因为 Apps 脚本服务有一个 Drafts.update
端点,它接受 Blob
输入,我还没有研究如何使用它。
关于javascript - 使用 AppScript 回复 Gmail 中更改收件人的电子邮件最终进入新线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51786588/
如果我 mov, eax 12345 和之后的 mov var, eax (假设 var 是一个 32 位的 int 等..等等)并输出 var 稍后它会正确输出。 与 ax 相同。 mov ax,
我有这个代码: for($nrt=0; $nrt"; if($sidesIndexes[$nrt]==$nrt) { echo "am I in??? ".$sidesInde
我正在阅读The Go Programming Language的8.5章,并陷入一些代码。下面的代码列表。 func main() { naturals := make(chan int)
我写了一个 MySQL 查询用于将数据导出到文本文件。 查询运行成功,但结果与我的预期不符。 我想在列之间没有间距的结果。 select sample_export_record1_2013.
在普通的 Excel 窗口中,我可以打开 VBE 并通过触摸键序列插入一个新模块:ALT+F11、ALTim 全部不使用鼠标。有没有办法打开 VBE 并导航到 本工作手册 不使用鼠标的代码区域? 最佳
我正在使用 axios 发出 http 请求。在 .then() 内部,我正在使用另一个 axios 调用。最后,我有第三个 then(),它应该在第二个 then 之后运行,但实际上并没有这样做。
我需要在 cocos2d 项目中播放视频..我的问题是:如何将 MPMoviePlayerController 放入我的 View 中,如下所示:? UIView *theView = [[CCDir
我正在学习 Angular。以下代码有效: .controller('abc', function ($scope, $http) { $http.get("/Handlers/Authenticat
目前我正在使用 WPF 学习 C#。我的主要方法是尽我所能使用 MVVM 模式,但现在我有点困惑。 在我所有 View 的应用程序中,我有一个 View 模型: private DruckviewVi
关于将 G 邮件提取到 Google 电子表格,我该如何添加 IF 以按主题驳回特定电子邮件?例如:电子邮件回复(主题中带有“RE:”)。我不希望这些电子邮件出现在我的电子表格中。 我尝试过使用 LO
我正在尝试使用 Spotify API 并进入数组。 const App = () => { const [isLoading, setIsLoading] = useState(true);
我有一个 html 模板,并且有一个条件为 --> 的代码 --> window.jQuery || document.write(""+"");
我正在开发一个 Android 应用程序,该应用程序会暴力破解从 int 创建的 MD5 和。 暴力破解部分工作正常。 (我可以sysout最终值并且它是正确的。) 我在将输出值发送到警报对话框时遇到
我正在创建一个界面,用户可以通过该界面生成多系列折线图,并控制绘制哪些数据集。 要绘制哪些数据集由复选框控制。页面加载时,默认数据集以图表形式呈现,并且 $('input:checkbox.data-
我尝试将有向无环图绘制为力布局。 但是我注意到,尽管为每个组元素灌输了进入/退出条件,弹出的节点/链接并没有从 DOM 中删除它们自己。 相反,弹出的节点/链接在力布局中卡住;这意味着没有调用进入/退
这里是新手。我不知道它是怎么发生的,但我正在处理一个程序,当我去调试并进入时,黄色箭头走到了我代码的最后并跳过了整个 block 。有快速解决方法吗? 最佳答案 按 F11,或单击工具栏上的“Step
我的 Action 栏 sherlock 中有一个列表。我想在用户点击该列表时得到。我不想知道用户何时点击某个项目,我已经知道 (onNavigationItemSelected)。 在我的 onCr
** 你好 **我如何从 ci 中的 mysql 日期获取 eurodate 来工作......无法弄清楚 - 请帮忙 想要获取日期 YY-mm- dd -> dd-mm-yy提前致谢 最佳答案 $t
我有以下脚本: #!/bin/bash ls -1 | while read d do [[ -f "$d" ]] && continue echo $d cd $d done
TL;DR - 跳转到最后一段 背景 我正在执行一些数据驱动测试,并将日志文件用作测试输出之一。它的工作原理是这样的- 读取文件夹中的第一个文件 处理第一行并转换为测试 运行测试 执行验证 1 ...
我是一名优秀的程序员,十分优秀!