-
-
Notifications
You must be signed in to change notification settings - Fork 70
/
Copy pathgmail-tester.js
230 lines (217 loc) · 7.51 KB
/
gmail-tester.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
const gmail = require("./gmail");
const tokenStore = require("./token-store")
function _get_header(name, headers) {
const found = headers.find(h => h.name === name);
return found && found.value;
}
function _init_query(options) {
const { to, from, subject, before, after } = options;
let query = "";
if (to) {
query += `to:"${to}" `;
}
if (from) {
query += `from:"${from}" `;
}
if (subject) {
query += `subject:(${subject}) `;
}
if (after) {
const after_epoch = Math.round(new Date(after).getTime() / 1000);
query += `after:${after_epoch} `;
}
if (before) {
const before_epoch = Math.round(new Date(before).getTime() / 1000);
query += `before:${before_epoch} `;
}
query = query.trim();
return query;
}
async function _get_recent_email(credentials, token, options = {}) {
const emails = [];
const query = _init_query(options);
// Load client secrets from a local file.
const oAuth2Client = await gmail.authorize(credentials, token);
const gmail_emails = await gmail.get_recent_email(
oAuth2Client,
query,
options.label
);
for (const gmail_email of gmail_emails) {
const email = {
from: _get_header("From", gmail_email.payload.headers),
subject: _get_header("Subject", gmail_email.payload.headers),
receiver: _get_header("Delivered-To", gmail_email.payload.headers),
date: new Date(+gmail_email["internalDate"])
};
if (options.include_body) {
let email_body = {
html: "",
text: ""
};
const { body } = gmail_email.payload;
if (body.size) {
switch (gmail_email.payload.mimeType) {
case "text/html":
email_body.html = Buffer.from(body.data, "base64").toString("utf8");
break;
case "text/plain":
default:
email_body.text = Buffer.from(body.data, "base64").toString("utf8");
break;
}
} else {
let parts = [...gmail_email.payload.parts];
while (parts.length) {
let part = parts.shift();
if (part.parts) {
parts = parts.concat(part.parts);
}
if (part.mimeType === "text/plain") {
email_body.text = Buffer.from(part.body.data, "base64").toString(
"utf8"
);
} else if (part.mimeType === "text/html") {
email_body.html = Buffer.from(part.body.data, "base64").toString(
"utf8"
);
}
}
}
email.body = email_body;
}
if (options.include_attachments) {
email.attachments = await gmail.get_email_attachments(oAuth2Client, gmail_email);
}
emails.push(email);
}
return emails;
}
async function __check_inbox(credentials, token, options = {}) {
const { subject, from, to, wait_time_sec, max_wait_time_sec } = options;
try {
console.log(
`[gmail] Checking for message from '${from}', to: ${to}, contains '${subject}' in subject...`
);
let found_emails = null;
let done_waiting_time = 0;
do {
const emails = await _get_recent_email(
credentials,
token,
options
);
if (emails.length > 0) {
console.log(`[gmail] Found!`);
found_emails = emails;
break;
}
console.log(
`[gmail] Message not found. Waiting ${wait_time_sec} seconds...`
);
done_waiting_time += wait_time_sec;
if (done_waiting_time >= max_wait_time_sec) {
console.log("[gmail] Maximum waiting time exceeded!");
break;
}
await new Promise(resolve => setTimeout(resolve, wait_time_sec * 1000));
} while (!found_emails);
return found_emails;
} catch (err) {
console.log("[gmail] Error:", err);
throw err;
}
}
/**
* Poll inbox.
*
* @param {string | Object} credentials - Path to credentials json file or credentials Object.
* @param {string | Object} token - Path to token json file or token Object.
* @param {CheckInboxOptions} [options]
* @param {boolean} [options.include_body] - Set to `true` to fetch decoded email bodies.
* @param {string} [options.from] - Filter on the email address of the receiver.
* @param {string} [options.to] - Filter on the email address of the sender.
* @param {string} [options.subject] - Filter on the subject of the email.
* @param {Date} [options.before] - Date. Filter messages received _after_ the specified date.
* @param {Date} [options.after] - Date. Filter messages received _before_ the specified date.
* @param {number} [options.wait_time_sec] - Interval between inbox checks (in seconds). Default: 30 seconds.
* @param {number} [options.max_wait_time_sec] - Maximum wait time (in seconds). When reached and the email was not found, the script exits. Default: 60 seconds.
* @param {string} [options.label] - String. The default label is 'INBOX', but can be changed to 'SPAM', 'TRASH' or a custom label. For a full list of built-in labels, see https://developers.google.com/gmail/api/guides/labels?hl=en
*/
async function check_inbox(
credentials,
token,
options = {
subject: undefined,
from: undefined,
to: undefined,
wait_time_sec: 30,
max_wait_time_sec: 30,
include_body: false,
label: "INBOX"
}
) {
if (typeof options !== "object") {
console.error(
"[gmail-tester] This functionality is obsolete! Please pass all params of check_inbox() in options object."
);
process.exit(1);
}
return __check_inbox(credentials, token, options);
}
/**
* Get an array of messages
*
* @param {string | Object} credentials - Path to credentials json file or credentials Object.
* @param {string | Object} token - Path to token json file or token Object.
* @param {GetMessagesOptions} options
* @param {boolean} options.include_body - Return message body string.
* @param {string} options.from - Filter on the email address of the receiver.
* @param {string} options.to - Filter on the email address of the sender.
* @param {string} options.subject - Filter on the subject of the email.
* @param {Object} options.before - Date. Filter messages received _after_ the specified date.
* @param {Object} options.after - Date. Filter messages received _before_ the specified date.
*/
async function get_messages(credentials, token, options) {
try {
return await _get_recent_email(credentials, token, options);
} catch (err) {
console.log("[gmail] Error:", err);
}
}
/**
* Refreshes Access Token
*
* @param {string | Object} credentials - Path to credentials json file or credentials Object.
* @param {string | Object} token - Path to token json file or token Object.
*/
async function refresh_access_token(credentials, token) {
const oAuth2Client = await gmail.authorize(credentials, token);
const refresh_token_result = await oAuth2Client.refreshToken(
oAuth2Client.credentials.refresh_token
);
if (refresh_token_result && refresh_token_result.tokens) {
const new_token = tokenStore.get(token);
if (refresh_token_result.tokens.access_token) {
new_token.access_token = refresh_token_result.tokens.access_token;
}
if (refresh_token_result.tokens.refresh_token) {
new_token.refresh_token = refresh_token_result.tokens.refresh_token;
}
if (refresh_token_result.tokens.expiry_date) {
new_token.expiry_date = refresh_token_result.tokens.expiry_date;
}
tokenStore.store(new_token, token);
} else {
throw new Error(
`Refresh access token failed! Respose: ${JSON.stringify(
refresh_token_result
)}`
);
}
}
module.exports = {
check_inbox,
get_messages,
refresh_access_token
};