How i get a complete list with ALL eMails - google-apps-script

i will get a simply list with all eMail which a have in my inbox - with settings "Conversation view off". My script currently looks like this:
function getAllMail(){
var threads = GmailApp.search('in:inbox');
for (var i = 0 ; i < threads.length; i++) {
var id = threads[i].getId();
var message = GmailApp.getMessageById(id);
Logger.log(i);
}
}
So i get a list with the view: "Conversation view on".
Where is my thinkingfault?

As per my understanding, you need something that will list all the threads or conversation in your inbox. Try to manage threads.
The Gmail API uses Thread resources to group email replies
with their original message into a single conversation or thread. This
allows you to retrieve all messages in a conversation, in order,
making it easier to have context for a message or to refine search
results.
To retrieve threads, here is what you need.
Threads provide a simple way of retrieving messages in a conversation
in order. By listing a set of threads you can choose to group messages
by conversation and provide additional context. You can retrieve a
list of threads using the threads.list method, or retrieve a specific
thread with threads.get. You can also filter threads using the same
query parameters as for the Message resource. If any message in a
thread matches the query, that thread is returned in the result.
You can study the sample codes provided in the quickstart guides.

Related

Is deleting Gmail thread specific enough to delete only messages from a specific email?

I made the following algorithm in Google Script which deletes all mails from a specific adress from all folders:
function deleteForever() {
var labelName
labelName="some_mail_address"
var threads = GmailApp.search("in: all from:" + labelName);
for (var i = 0; i < threads.length; i++) {
Gmail.Users.Messages.remove("me", threads[i].getId());
}
}
Is this safe to use for my gmail account? (With a trigger such that it runs every minute)
In particular, I am wondering whether a thread can contain multiple mails, and whether this can result in deleting mails which are not from the unwanted sender.
Also, I have seen "in: anywhere" on some webpages. Is this different from "in: all"?
Short answer: No.
It's not "safe" because:
the thread id is the same id for the first thread message but the first message might be from other sender.
the message will be deleted permanently
running a script every minute might consume the daily quota so you might first getting error messages and lately the trigger might be disabled due to having so many errors.
in:anywhere is not the same as in:all. The first will return threads labeled as spam and trash but the later doesn't.
Instead using the thread id, iterate over the thread messages to check the sender, if it matches then "delete" the message
Instead of using "remove" use "trash"
If you use "trash" instead of "remove" don't use "in:anywhere" otherwise the same message will be processed over an over until it's deleted permanently.
Instead of using a trigger scheduled to run every minute, run it every ten minutes or use a lower frequency if you plan to use other time driven triggers.
Related
Same thread id does not return the same message
Unthread or permanently delete individual gmail messages, without turning off threading

how to label and check only one message ( Gmail email) in Google Script?

I want my Google script to parse all emails within a label. But just once. The emails come from my bank. They store transaction details. Because I use thread conversation view I worry that the script will either - process one message more than once or miss a message in a case I mark the thread with a label let's say "done" once the email message was processed.
The transaction details are coming with the same subject, so all of them will be part of one thread.
I do not want to store message ID or something like outside Google Script.
Could you think of any solution?
I was thinking that I can add email ID to the thread but the I would end up with too many labels.
I might be able to use Gmail API as suggested here
GmailApp - Add label to specific message, not the thread and
Google Apps Script - Changing label of individual email in Gmail
but I do not know how to check if a particular message got specific label.
You can query for messages containing a certain label with Gmail.Users.Messages.list specifying the id of a label
Sample:
var starredMessages = Gmail.Users.Messages.list("me", {"labelIds":"Label_981438383934575828"}).messages;
var starredIds = [];
starredMessages.forEach(function(message){starredIds.push(message.id)});
Once you have this, you can retrieve all inbox messages, push their ids into an array and filter them to remove the already labelled messages from the array:
var AllIds = [];
var allMessages = GmailApp.getInboxThreads().forEach(function(thread){thread.getMessages().forEach(function(message){AllIds.push(message.getId())})});
var filteredIds= AllIds.filter(function (id) { return starredIds.indexOf(id) == -1;});
As a result, you obtain an array with message Ids that do not have the specified label (yet).
Now, do with those messages what you desire, e.g. in a loop and subsequently add the label to them with e.g.
filteredIds.forEach(function(id){Gmail.Users.Messages.modify({'addLabelIds': [LabelId]}, 'me', id)});
UPDATE
Another way to retrieve messages without the label is with the query parameter q.
Thereby, the query should be specified as -label:MY LABEL :
var NotStarredMessages = Gmail.Users.Messages.list("me", {"q":"-label:MY LABEL"}).messages;
HOWEVER, KEEP IN MIND THAT IF ONE MESSAGE OF A THREAD HAS THE LABEL, THIS SECOND METHOD WILL NOT EXCLUDE ALL MESSAGES OF THE CORRESPONDING THREAD FROM THE LIST.
So in your case the second method is not recommended.

GmailApp.getMessagesForThreads() returns too many messages

I want to return a list of all email subjects received at a specified day, at my Gmail account. If there is more than one message with the same subject, I want to have a number of how many there are. This is what I did:
var emailThreads = GmailApp.search("subject:<partial_subject> after:2020/8/30 before:2020/8/31");
var messages = GmailApp.getMessagesForThreads(emailThreads);
This returns "an array of arrays of messages, where each item in the outer array corresponds to a thread and the inner array contains the messages in that thread".
I try to access data I need like this:
for (i = 0; i < messages.length; i++) {
messages[i][0].getSubject() // a subject
messages[i].length // number of messages with the same subject
}
However, this returns more messages than is displayed in Gmail UI.
Here's what I've noticed:
it only happens to threads with more than one message
it doesn't happen to all threads with more than one message
every time a run my script for the same day, the result is the same
when I run my script for different days, the result is different even though the same emails were received
Considering that this code is very simple and I'm only using methods delivered by google, I assume I must have misunderstood something or Gmail is doing something weird with messages in threads.
Any ideas what could this be and how to work around it?
Edit:
I've noticed that there is a "Don't include chats" checkbox in gmail filter, which adds "-in:chats" into search query, but it didn't help either.
I also tried adding "in:inbox" to the query, but with no change in the result.
Explanation:
The reason you are getting too many messages is because your current
filter criteria does not query only emails but also hangout
messages or other threads rather than emails.
Use this to get the subjects of all emails:
var emailThreads = GmailApp.search("is:inbox after:2020/8/30 before:2020/8/31");
Solution:
This is how the full code would look like:
function myFunction() {
var emailThreads = GmailApp.search("is:inbox after:2020/8/30 before:2020/8/31");
var messages = GmailApp.getMessagesForThreads(emailThreads);
var messageArr = [];
for (i = 0; i < messages.length; i++) {
Logger.log(messages[i][0].getSubject()) // a subject
Logger.log(messages[i].length) // number of messages with the same subject
}
messages.forEach(m=>{
messageArr.push({
subject: m[0].getSubject(),
num_messages:m.length
})});
}
I also created for you a json array messageArr which contains all the
subject names and the number of message for each subject.
From the question
However, this returns more messages than is displayed in Gmail UI.
There might be some reasons for that
Messages in Trash and Spam are not displayend on the search results on the Gmail UI (unless you include in:anywhere, textually)
The UI is not showing all messages because the results might be paged and you missed that
After the receive/import/deletion of a large number of messages there is a noticible delay on getting them displayed on the UI, so even if it looks that you reached the last search results page, wait some minutes and regresh the Gmail webpage.
getMessagesForThreads return messages that might not be shown on the Gmail UI search results because they were sent to the Spam or Trash.
To correctly compare the results from the Gmail UI with the results of this method, on the Gmail UI include in:anywhere as part of your search. The complete search statement will look something like this:
in:anywhere subject:<partial_subject> after:2020/8/30 before:2020/8/31
As mentionend on the first part of this answer, you might have to refresh the Gmail webpage several times to be sure that there aren't messages recently made available through the web UI.
Related
How to keep gmail threads from coming back
Resources
Search operators you can use with Gmail

Google Apps Script - GMail to Google Sheets - Get Oldest Unread Email in Thread's .csv Attachment

First off: JavaScript novice. Any help/direction you could provide would be greatly appreciated.
I have a new business case in which a GMail box is going to receive an e-mail 4 times a day with the same subject line, but with a new .csv attachment each time. I've created the label within GMail, and it is attaching successfully to each email. I need to be able to use Apps Script to only process the oldest unread e-mail. The part to parse the .csv is already tested/successful/completed. However, each time I run it, it's looping through all the files, and processing them again.
Here is the current code for getting information from the GMail Box
// GMail Details
var gmailLabel = ('_Projects/ProjectLabel');
var label = GmailApp.getUserLabelByName(gmailLabel);
var unreadCount = label.getUnreadCount();
var gmailThreads = GmailApp.getUserLabelByName(gmailLabel).getThreads();
Logger.log(label.getUnreadCount());
for (var i = 0; i < gmailThreads.length; i++) {
Logger.log(gmailThreads[i].getFirstMessageSubject());
}
}
The log does show that there are 2 unread threads correctly, and does display the subject line in the logger. However, within the gmailThreads array, I can only see the object numbers for the threads. How could I get object numbers for each of the e-mails belonging to those threads? This would be the e-mail that I would need to parse the .csv attachment first, then mark as read (or change the label of) within the loop.
Thanks again for looking!
To get the IDs of the first messages in a given Gmail Thread, you need to call getMessages() to retrieve all the messages, and then access the first message in the resulting array, with [0]:
var firstMessage = thread.getMessages()[0];
Then you can do a number of things with this GmailMessage, such as retrieve its ID (.getId()) or attachments (.getAttachments()).

Get unread email and mark as read in one call using EWS Managed API

I have a standard routine to get an unread email from a mail server, mark it as read and then process it. It looks similar to this:
var view = new ItemView(1);
SearchFilter sf = new SearchFilter.SearchFilterCollection(LogicalOperator.And, new SearchFilter.IsEqualTo(EmailMessageSchema.IsRead, false));
var findResults = ExchangeService.FindItems(WellKnownFolderName.Inbox, sf, view);
if (findResults.Items.Any())
{
EmailMessage emailMsg = findResults.Items.First() as EmailMessage;
emailMsg.IsRead = true;
emailMsg.Update(ConflictResolutionMode.AutoResolve);
ProcessMail(emailMsg);
}
This could be invoked from various processes, so I need to ensure atomicity of "find unread / mark as read" operation. In other words between Service.FindItems() and mail.Update() the same email could be read twice by different processes and result in duplicated processing.
Now, I don't want to get involved in discussion about storing processed email IDs or whether I should be using multiple processes accessing the same email server.
All I want to find out is if there is a EWS API method which does the two operations in one call? Ideally that would be something like this:
var mail = ExchangeService.GetFirstUnreadEmailAndMarkItAsRead()
No, there isn't. An EWS GetItems call doesn't modify the item.