问题描述
我已经被这个问题困扰了好几天了,几乎没有进展,如果可以,请帮忙!
我有一个 Node.js (v12) AWS Lambda,它需要从我的 Firebase 实时数据库中提取数据并将每条记录处理到 Redis 缓存中(如果它不存在)。该函数启动但从未完成,而是从 AWS 收到 Task timed out after 180.10 seconds
。
我尝试过的事情:
- 使用
exports.handler = async function(event)
与exports.handler = function(event,context,callback)
; - 对于上面的同步尝试,我尝试使用
context.callbackWaitsForEmptyEventLoop = false
与 not; - 使用 promise 与级联函数与将一堆
.then()
拼接在一起; - 将
firebase-admin
与https
模块与 Firebase REST API 结合使用; - 稍后使用
settimeout
触发回调对比不; - 将
GOOGLE_APPLICATION_CREDENTIALS
环境变量设置为我的服务帐户凭据对比直接在代码中引用文件; - 我什至已将 Lambda 本身的内存和超时时间提高到最大程度,并将我想从 Firebase 提取的数据减少到仅 1 条记录。
我根据上述尝试得到的回复:
- AWS(最常见的):
Task timed out after 180.10 seconds
; - AWS(
.then
拼接方法):Function completed successfully
(但实际上没有处理数据); - 节点 HTTPS(REST API 方法):
ETIMEDOUT
或ECONNREFUSED
;
下面是我的工作,但仍然没有运气。我已经删除了缓存代码,因为我知道它可以正常工作。您看到的 settimeout
是我联系这里之前最后的选择。
const admin = require("firebase-admin");
admin.initializeApp({
credential: admin.credential.applicationDefault(),databaseURL: "https://{projectName}.firebaseio.com"
});
var result = [];
exports.handler = (event,callback) => {
context.callbackWaitsForEmptyEventLoop = false;
try {
admin.database().ref("data").orderByChild("timestamp").limitToLast(1).once("value",snapshot => {
if (snapshot.exists()) {
console.log('snapshot exists...');
let posts = snapshot.val();
result = Object.keys(posts);
}
setTimeout(() => {
admin.database().goOffline();
admin.app().delete();
callback(null,`Success! ${JSON.stringify(result)}`); // <-- NEVER RETURNS
},2000);
},error => {
setTimeout(() => {
admin.database().goOffline();
admin.app().delete();
callback(error); // <-- NEVER RETURNS
},2000);
});
} catch (error) {
setTimeout(() => {
admin.database().goOffline();
admin.app().delete();
callback(error); // <-- NEVER RETURNS
},2000);
}
};
解决方法
您似乎没有在函数的根级别存储或使用 setTimeout。您应该存储它,以便回调函数可以继续运行,因为它只在它在作用域内时才存在。这样做还需要您绑定对象,以便在您决定将其推入数组以进行多个回调时拥有自引用
var result = [];
var timeOut;
//...
timeOut = setTimeout(() => {
admin.database().goOffline();
admin.app().delete();
callback(error);
}.bind(this),2000);
来源:MSDN Function.prototype.bind()
如果 Binding 不是解决方案,而您想要一个阻塞方法,您可能对延迟函数感兴趣,它的行为与 setTimeout 相同,但适用于 promise
function sleep(ms) {
return new Promise(resolve => setTimeout(resolve,ms));
}
// Async function solution
await sleep(2000);
console.log('Two seconds later,showing sleep in a loop...');
// non-Async solution
sleep(2000)
.then(()=> {
admin.database().goOffline();
admin.app().delete();
callback(error);
})
.catch(e => console.log(e));