API Gmail для отправки писем в Node.js
Отказ от ответственности:
- Я следовал собственному руководству Google по Node.js и успешно подключился и использую функциональность
gmail.users.labels.list()
. - Я проверил здесь вопросы/ответы, например, этот (который не использует API Node.js, о котором я спрашиваю), или этот (похожий на этот), который, по-видимому, является той же проблемой, что и у меня, но решение не дает Работа.
Моя проблема:
При использовании Google Node.js API я получаю сообщение об ошибке при попытке отправить электронное письмо. Ошибка:
{
"code": 403,
"errors": [{
"domain": "global",
"reason": "insufficientPermissions",
"message": "Insufficient Permission"
}]
}
Моя настройка:
fs.readFile(secretlocation, function processClientSecrets(err, content) {
if (err) {
console.log('Error loading client secret file: ' + err);
return;
}
authorize(JSON.parse(content), sendMessage);
});
function sendMessage(auth) {
var raw = makeBody('[email protected]', '[email protected]', 'subject', 'message test');
gmail.users.messages.send({
auth: auth,
userId: 'me',
message: {
raw: raw
}
}, function(err, response) {
res.send(err || response)
});
}
Функция processClientSecrets
из руководства Google, о котором я упоминал выше. Он читает мой файл .json
котором есть мои access_token
и refresh_token
. Функция makeBody
- это создание закодированного сообщения тела.
В конфиге variabels у меня также:
var SCOPES = [
'https://mail.google.com/',
'https://www.googleapis.com/auth/gmail.modify',
'https://www.googleapis.com/auth/gmail.compose',
'https://www.googleapis.com/auth/gmail.send'
];
Почему это должно работать:
- процесс авторизации работает для
gmail.users.labels.list()
. - тело сообщения, которое я тестирую, работает, если я тестирую его на тестовой странице Google.
Мой вопрос:
Мои настройки неверны? Были ли изменения в API? Что мне не хватает?
Ответы
Ответ 1
Итак, я нашел проблему (проблемы).
Проблема № 1
Следуя руководству Node.js quickstart, в этом уроке есть
var SCOPES = ['https://www.googleapis.com/auth/gmail.readonly'];
И когда я получил .json
, который выглядит так:
{
"access_token": "xxx_a_long_secret_string_i_hided_xxx",
"token_type": "Bearer",
"refresh_token": "xxx_a_token_i_hided_xxx",
"expiry_date": 1451721044161
}
те токены, которые производятся с учетом только области auth/gmail.readonly
в коде учебника.
Итак, я удалил первый .json
, добавил области видимости из моего окончательного массива областей (я опубликовал в вопросе) и снова запустил настройку учебника, получив новый токен.
Проблема № 2
В объекте, переданном API, я отправлял:
{
auth: auth,
userId: 'me',
message: {
raw: raw
}
}
но это неправильно, message
следует называть resource
.
Окончательная настройка:
Это то, что я добавил в код учебника:
function makeBody(to, from, subject, message) {
var str = ["Content-Type: text/plain; charset=\"UTF-8\"\n",
"MIME-Version: 1.0\n",
"Content-Transfer-Encoding: 7bit\n",
"to: ", to, "\n",
"from: ", from, "\n",
"subject: ", subject, "\n\n",
message
].join('');
var encodedMail = new Buffer(str).toString("base64").replace(/\+/g, '-').replace(/\//g, '_');
return encodedMail;
}
function sendMessage(auth) {
var raw = makeBody('[email protected]', '[email protected]', 'test subject', 'test message');
gmail.users.messages.send({
auth: auth,
userId: 'me',
resource: {
raw: raw
}
}, function(err, response) {
res.send(err || response)
});
}
И назовите все с помощью:
fs.readFile(secretlocation, function processClientSecrets(err, content) {
if (err) {
console.log('Error loading client secret file: ' + err);
return;
}
// Authorize a client with the loaded credentials, then call the
// Gmail API.
authorize(JSON.parse(content), sendMessage);
});
Ответ 2
Так что для тех, кто смотрит на это, пытается получить тестовое электронное письмо от своего API, но не может получить эту работу, вот что вам нужно сделать:
Шаг 1: заменить
var SCOPES = ['https://www.googleapis.com/auth/gmail.readonly'];
с этим:
var SCOPES = [
'https://mail.google.com/',
'https://www.googleapis.com/auth/gmail.modify',
'https://www.googleapis.com/auth/gmail.compose',
'https://www.googleapis.com/auth/gmail.send'
];
Шаг 2: В конце примера кода googles добавьте это:
function makeBody(to, from, subject, message) {
var str = ["Content-Type: text/plain; charset=\"UTF-8\"\n",
"MIME-Version: 1.0\n",
"Content-Transfer-Encoding: 7bit\n",
"to: ", to, "\n",
"from: ", from, "\n",
"subject: ", subject, "\n\n",
message
].join('');
var encodedMail = new Buffer(str).toString("base64").replace(/\+/g, '-').replace(/\//g, '_');
return encodedMail;
}
function sendMessage(auth) {
var raw = makeBody('[email protected]', 'whereyou'[email protected]', 'This is your subject', 'I got this working finally!!!');
const gmail = google.gmail({version: 'v1', auth});
gmail.users.messages.send({
auth: auth,
userId: 'me',
resource: {
raw: raw
}
}, function(err, response) {
return(err || response)
});
}
fs.readFile('credentials.json', function processClientSecrets(err, content) {
if (err) {
console.log('Error loading client secret file: ' + err);
return;
}
// Authorize a client with the loaded credentials, then call the
// Gmail API.
authorize(JSON.parse(content), sendMessage);
});
Шаг 3 (необязательно)
Удалить эту строку:
authorize(JSON.parse(content), listLabels);
И эти:
/**
* Lists the labels in the user account.
*
* @param {google.auth.OAuth2} auth An authorized OAuth2 client.
*/
function listLabels(auth) {
const gmail = google.gmail({version: 'v1', auth});
gmail.users.labels.list({
userId: 'me',
}, (err, res) => {
if (err) return console.log('The API returned an error: ' + err);
const labels = res.data.labels;
if (labels.length) {
console.log('Labels:');
labels.forEach((label) => {
console.log('- ${label.name}');
});
} else {
console.log('No labels found.');
}
});
}
(Таким образом, вы не получаете случайные метки в вашей консоли)