google-apps-script - 忽略具有不同标签的同线程电子邮件
问题描述
我正在将特定新电子邮件中的日期和主题写入Google 表格的新行。
- 我使用过滤器将标签应用于新邮件。
- 该脚本处理那些标记的电子邮件
- 标签被移除
- 应用了新标签,以便下次不会处理这些电子邮件。
问题:当有myLabel电子邮件时,脚本会处理同一线程中的所有电子邮件(例如相同的主题和发件人),而不管它们的标签(甚至是收件箱和垃圾箱)。
问题:如何只处理新电子邮件,即带有标签myLabel的电子邮件- 即使这些邮件的线程延伸到myLabel文件夹之外?
我当前的脚本:
function fetchmaildata() {
var ss = SpreadsheetApp.getActiveSpreadsheet();
var sheet = ss.getSheetByName('mySheetName');
var label = GmailApp.getUserLabelByName('myLabel');
var threads = label.getThreads();
for (var i = 0; i < threads.length; i++)
{
var messages = threads[i].getMessages();
for (var j = 0; j < messages.length; j++)
{
var sub = messages[j].getSubject();
var dat = messages[j].getDate();
ss.appendRow([dat, sub])
}
threads[i].removeLabel(label);
threads[i].addLabel(newlabel);
}
}
我通过将for
循环更改为:
for (var j = messages.length-1; j > messages.length-2; j--)
这表示仅处理线程中的最新电子邮件,即使myLabel文件夹中有多个线程的电子邮件也是如此。奇怪的是,该脚本仍然会更改所有myLabel电子邮件的标签,但只有最新的一个线程被写入电子表格,所以它适用于我。
我不得不对代码进行另一次更改,因为上面的代码不会作为时间触发的计划任务运行。我以这种方式更改了代码,它现在按时间表运行!
//var ss = SpreadsheetApp.getActiveSpreadsheet();
var ss = SpreadsheetApp.openById("myGoogleSheetID");
解决方案
由于标签位于所述线程中的单个消息上,因此标签可以位于线程上。您的代码只是进入label -> all label threads -> all thread messages,而不是仅访问具有给定标签的线程中的消息。这不是你的错 - 这是 Gmail 服务的限制。您可以使用两种方法来纠正此行为:
(使用前启用“高级服务”)Gmail REST API
REST API 支持对消息的详细查询,包括每条消息的标签状态,Gmail.Users.Messages.list
以及labelIds
可选参数。例如:
// Get all messages (not threads) with this label:
function getMessageIdsWithLabel_(labelClass) {
const labelId = labelClass.getId();
const options = {
labelIds: [ labelId ],
// Only retrieve the id metadata from each message.
fields: "nextPageToken,messages/id"
};
const messages = [];
// Could be multiple pages of results.
do {
var search = Gmail.Users.Messages.list("me", options);
if (search.messages && search.messages.length)
Array.prototype.push.apply(messages, search.messages);
options.pageToken = search.nextPageToken;
} while (options.pageToken);
// Return an array of the messages' ids.
return messages.map(function (m) { return m.id; });
}
使用 REST API 后,您还可以使用其他方法,例如批量消息标签调整:
function removeLabelFromMessages_(messageIds, labelClass) {
const labelId = labelClass.getId();
const resource = {
ids: messageIds,
// addLabelIds: [ ... ],
removeLabelIds: [ labelId ]
};
// https://developers.google.com/gmail/api/v1/reference/users/messages/batchModify
Gmail.Users.Messages.batchModify(resource, "me");
}
结果:
function foo() {
const myLabel = /* get the Label somehow */;
const ids = getMessageIdsWithLabel_(myLabel);
ids.forEach(function (messageId) {
var msg = GmailApp.getMessageById(messageId);
/* do stuff with the message */
});
removeLabelFromMessages_(ids, myLabel);
}
推荐阅读:
跟踪处理
您还可以将每个消息 ID 存储在某处,并使用存储的 ID 检查您是否已经处理了给定的消息。消息 ID 是唯一的。
此示例使用原生 JavaScript 对象进行极快的查找(相对于简单地将 id 存储在数组中并需要使用Array#indexOf
)。为了在脚本执行之间维护已处理的 id,它使用活动工作簿或您选择的工作簿上的工作表:
var MSG_HIST_NAME = "___processedMessages";
function getProcessedMessages(wb) {
// Read from a sheet on the given spreadsheet.
if (!wb) wb = SpreadsheetApp.getActive();
const sheet = wb.getSheetByName(MSG_HIST_NAME)
if (!sheet) {
try { wb.insertSheet(MSG_HIST_NAME).hideSheet(); }
catch (e) { }
// By definition, no processed messages.
return {};
}
const vals = sheet.getSheetValues(1, 1, sheet.getLastRow(), 1);
return vals.reduce(function (acc, row) {
// acc is our "accumulator", and row is an array with a single message id.
acc[ row[0] ] = true;
return acc;
}, {});
}
function setProcessedMessages(msgObject, wb) {
if (!wb) wb = SpreadsheetApp.getActive();
if (!msgObject) return;
var sheet = wb.getSheetByName(MSG_HIST_NAME);
if (!sheet) {
sheet = wb.insertSheet(MSG_HIST_NAME);
if (!sheet)
throw new Error("Unable to make sheet for storing data");
try { sheet.hideSheet(); }
catch (e) { }
}
// Convert the object into a serializable 2D array. Assumes we only care
// about the keys of the object, and not the values.
const data = Object.keys(msgObject).map(function (msgId) { return [msgId]; });
if (data.length) {
sheet.getDataRange().clearContent();
SpreadsheetApp.flush();
sheet.getRange(1, 1, data.length, data[0].length).setValues(data);
}
}
用法类似于:
function foo() {
const myLabel = /* get label somehow */;
const processed = getProcessedMessages();
myLabel.getThreads().forEach(function (thread) {
thread.getMessages().forEach(function (msg) {
var msgId = msg.getId();
if (processed[msgId])
return; // nothing to do for this message.
processed[msgId] = true;
// do stuff with this message
});
// do more stuff with the thread
});
setProcessedMessages(processed);
// do other stuff
}
推荐阅读:
推荐阅读
- hibernate - Hibernate Metamodel 注释处理器无法识别 Kotlin 实体类
- linux - 无法将磁盘挂载到目录
- c# - 如何使用 SpeechRecognizer EventArgs 变量在 TextBox 中进行换行
- python - 如何检查 2 个 DataFrames 是否关闭?(不相等)
- laravel - 在 Laravel 7 的特使脚本中获取 Dotenv 值
- html - 根据设备宽度的 HR 标签对齐在 IE11 浏览器上不起作用
- java - 使用 .or 和 .link 一起查询不起作用
- excel - 我需要使用 VBA 拖动数组公式并粘贴值,但 excel 不断崩溃
- html - 在带有 CDATA 的网站上显示 XML 文件中的可点击链接
- php - Laravel 6 中此集合实例上不存在属性 [名称]