Nodejs- Установить тело в очередь облачных задач Google
У меня есть сервер nodejs в движке приложений Google, где я хочу делегировать долго выполняющиеся задачи в очереди задач с помощью очереди задач Google Cloud.
Очередь задач не доставляет тело запроса, но достигает конечной точки:
Вот как я добавляю задачу в очередь задач:
// Imports the Google Cloud Tasks library.
const cloudTasks = require('@google-cloud/tasks');
// Instantiates a client.
const client = new cloudTasks.CloudTasksClient();
const project = 'projectname';
const queue = 'queuename';
const location = 'us-central1';
const parent = client.queuePath(project, location, queue);
// Send create task request.
exports.sender = async function (options) {
// Construct the fully qualified queue name.
let myMap = new Map();
myMap.set("Content-Type", "application/json");
const task = {
appEngineHttpRequest: {
httpMethod: 'POST',
relativeUri: '/log_payload',
headers: myMap,
body: options/* Buffer.from(JSON.stringify(options)).toString('base64')*/
},
};
/* if (options.payload !== undefined) {
task.appEngineHttpRequest.body = Buffer.from(options.payload).toString(
'base64'
);
}*/
if (options.inSeconds !== undefined) {
task.scheduleTime = {
seconds: options.inSeconds + Date.now() / 1000,
};
}
const request = {
parent: parent,
task: task,
};
client.createTask(request)
.then(response => {
const task = response[0].name;
//console.log(`Created task ${task}`);
return {'Response': String(response)}
})
.catch(err => {
//console.error(`Error in createTask: ${err.message || err}`);
return `Error in createTask: ${err.message || err}`;
});
};
И это конечная точка получения:
app.post('/log_payload', async (req, res) => {
let mailOptions = {
subject: 'Message Delivered',
from: 'sender@example.com',
to: "receiver@example.com",
text: String(JSON.stringify(JSON.stringify(req.body)))
};
return await mailer.sendEmail(mailOptions).then(value => {
return res.send(`Received task payload: ${value}`).end()
}).catch(reason => {
res.send(`Worker error: ${reason.message}`).end()
});
});
Когда электронное письмо получено, оба тела являются пустыми объектами Json. Что я делаю неправильно?
2 ответа
Просто чтобы дополнить ответ Джоан Грау:
Если вы добавите этот фильтр, вы можете быть уверены, что результат будет правильно проанализирован:
var bodyParser = require('body-parser');
var rawBodySaver = function (req, res, buf, encoding) {
if (buf && buf.length) {
req.rawBody = buf.toString(encoding || 'utf8');
}
}
app.use(bodyParser.json({ verify: rawBodySaver }));
app.use(bodyParser.urlencoded({ verify: rawBodySaver, extended: true }));
app.use(bodyParser.raw({ verify: rawBodySaver, type: function () { return true } }));
(идея взята из этого поста)
В частности, хорошей практикой является применение парсера только там, где он вам нужен. И, кроме того, рассмотрите возможность отправки содержимого тела в виде строки base64 (это необходимо для работы облачных задач). Переписать свой код со всеми этими:
const task = {
appEngineHttpRequest: {
httpMethod: 'POST',
relativeUri: '/log_payload',
headers: myMap,
body: Buffer.from(JSON.stringify(options)).toString('base64')
},
};
и конечная точка:
var bodyParser = require('body-parser');
var rawBodySaver = function (req, res, buf, encoding) {
if (buf && buf.length) {
req.rawBody = buf.toString(encoding || 'utf8');
}
}
app.use('/log_payload', bodyParser.json({ verify: rawBodySaver }));
app.use('/log_payload', bodyParser.urlencoded({ verify: rawBodySaver, extended: true }));
app.use('/log_payload', bodyParser.raw({ verify: rawBodySaver, type: function () { return true } }));
app.post('/log_payload', async (req, res) => {
let mailOptions = {
subject: 'Message Delivered',
from: 'sender@example.com',
to: "receiver@example.com",
text: req.body //no need to parse here!
};
return await mailer.sendEmail(mailOptions).then(value => {
return res.send(`Received task payload: ${value}`).end()
}).catch(reason => {
res.send(`Worker error: ${reason.message}`).end()
});
});
Руководствуясь этим примером, я считаю, что лучший способ разобрать тело из запроса - использовать пакет body-parser.
Вы можете реализовать это, изменив файл получающей конечной точки, включив следующее:
const bodyParser = require('body-parser');
app.use(bodyParser.raw());
app.use(bodyParser.json());
app.use(bodyParser.text());
app.post('/log_payload', async (req, res) => {
let mailOptions = {
subject: 'Message Delivered',
from: 'sender@example.com',
to: "receiver@example.com",
text: req.body
};
return await mailer.sendEmail(mailOptions).then(value => {
return res.send(`Received task payload: ${value}`).end()
}).catch(reason => {
res.send(`Worker error: ${reason.message}`).end()
});
});