Lambda(Node.js 4.3)でこれを実行しようとすると「モジュール 'firebase'が見つかりません」と表示される
var Firebase = require('firebase');
Node_modules/firebaseを含む圧縮されたパッケージをアップロードしようとすると、同じことが起こります
誰かが動作する「ラムダからfirebaseへの書き込み」実装を持っていますか?
Firebase npmパッケージ(バージョン3.3.0)をAWS Lambda(Nodejs 4.3)で安全に使用するには、次を実行してください。
'use strict';
var firebase = require("firebase");
exports.handler = (event, context, callback) => {
context.callbackWaitsForEmptyEventLoop = false; //<---Important
var config = {
apiKey: "<<apikey>>",
authDomain: "<<app_id>>.firebaseapp.com",
databaseURL: "https://<<app_id>>.firebaseio.com",
storageBucket: "<<app_id>>.appspot.com",
};
if(firebase.apps.length == 0) { // <---Important!!! In lambda, it will cause double initialization.
firebase.initializeApp(config);
}
...
<Your Logic here...>
...
};
Firebase REST apiを使用して問題を解決しました
var https = require('https');
exports.handler = function(event, context, callback) {
var body = JSON.stringify({
foo: "bar"
})
var https = require('https');
var options = {
Host: 'project-XXXXX.firebaseio.com',
port: 443,
path: '/.json',
method: 'POST'
};
var req = https.request(options, function(res) {
console.log(res.statusCode);
res.on('data', function(d) {
process.stdout.write(d);
});
});
req.end(body);
req.on('error', function(e) {
console.error(e);
});
callback(null, "some success message");
}
これは遅いですが、誰かが探している場合:
プロジェクトフォルダーのcontentsの代わりにプロジェクトフォルダーを圧縮すると、これが発生する可能性があります。 zipされたフォルダーは、展開されたときに、ラムダファイルを含むフォルダーを含むべきではありませんが、ルートレベルにindex.jsファイルとnode_modulesフォルダーを持っている必要があります。
ラムダ関数の実例は次のとおりです(最新の光沢のあるfirebaseのものを使用*ため息*):
var firebase = require('firebase');
// Your service account details
var credentials = {
"type": "service_account",
"project_id": "project-123451234512345123",
"private_key_id": "my1private2key3id",
"private_key": "-----BEGIN PRIVATE KEY-----InsertKeyHere-----END PRIVATE KEY-----\n",
"client_email": "[email protected]",
"client_id": "1111222223333344444",
"auth_uri": "https://accounts.google.com/o/oauth2/auth",
"token_uri": "https://accounts.google.com/o/oauth2/token",
"auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs",
"client_x509_cert_url": "https://www.googleapis.com/robot/v1/metadata/x509/projectname%40project-123451234512345123.iam.gserviceaccount.com"
};
firebase.initializeApp({
serviceAccount: credentials,
databaseURL: "https://project-123451234512345123.firebaseio.com"
});
exports.handler = function (event, context, callback) {
// I use some data passed in from AWS API Gateway:
if (!event.firebaseUid) {
callback('Missing param for id');
}
firebase.database().ref().child('users').child(firebaseUid).child('at').set(newTokens.access_token).then(function (data) {
console.log('Firebase data: ', data);
firebase.database().goOffline();
callback(null, 'Firebase data: ', data);
}).catch(function (error) {
callback('Database set error ' + error);
});
};
さて、注意点です。 firebaseコールバックが発生した後でも、ラムダ関数がタイムアウトする原因になりました。 set関数は、正しいデータが返されてもラムダ関数を開いたままにするリスナーを作成するようです。
更新:firebase.database()。goOffline()を呼び出すと、私が経験していたLambda関数のタイムアウトの問題が修正されます。
セキュリティが検証されていない、または適切でないことに関する通常の注意、およびこれを使用してスペースと時間を停止する可能性が適用されます。
2017-03-22編集:googleが発表しました firebaseクラウド機能 、これはこれを行うためのより良い方法です。クラウド関数はラムダのように機能し、firebaseイベントからトリガーできます。
REST api (したがって、require
何もする必要はありません)を使用した私のソリューションです:
var https = require('https');
var firebaseHost = "yourapp.firebaseio.com";
function fbGet(key){
return new Promise((resolve, reject) => {
var options = {
hostname: firebaseHost,
port: 443,
path: key + ".json",
method: 'GET'
};
var req = https.request(options, function (res) {
res.setEncoding('utf8');
var body = '';
res.on('data', function(chunk) {
body += chunk;
});
res.on('end', function() {
resolve(JSON.parse(body))
});
});
req.end();
req.on('error', reject);
});
}
function fbPut(key, value){
return new Promise((resolve, reject) => {
var options = {
hostname: firebaseHost,
port: 443,
path: key + ".json",
method: 'PUT'
};
var req = https.request(options, function (res) {
console.log("request made")
res.setEncoding('utf8');
var body = '';
res.on('data', function(chunk) {
body += chunk;
});
res.on('end', function() {
resolve(body)
});
});
req.end(JSON.stringify(value));
req.on('error', reject);
});
}
次のように使用できます。
fbPut("/foo/bar", "lol").then(res => {
console.log("wrote data")
})
その後:
fbGet("/foo/bar").then(data => {
console.log(data); // prints "lol"
}).catch(e => {
console.log("error saving to firebase: ");
console.log(e);
})
ノードベースの開発セットアップを使用している場合の別の方法は、node-lambda
パッケージの here から。基本的には、ラムダをセットアップ、テスト、デプロイするためのラッパーを提供します。 node-lambda deploy
は、インストールしたモジュールをパッケージ化します(例:npm i --save firebase
)Lambda自体で利用できることを確認します。外部モジュールの管理に非常に役立つことがわかりました。
私にとって、firebase-adminはこのトリックを行うべきです。 https://firebase.google.com/docs/admin/setup
ただし、context.callbackWaitsForEmptyEventLoopを提案してくれたJosiah Choiに感謝します。そのため、ラムダは毎回Firebaseを初期化する必要はありません。私の最初の実行は本当に遅かった。
var firebase = require('firebase-admin');
module.exports.Test = (event, context, callback) => {
context.callbackWaitsForEmptyEventLoop = false; //<---Important
if(firebase.apps.length == 0) { // <---Important!!! In lambda, it will cause double initialization.
firebase.initializeApp({
credential: firebase.credential.cert("serviceAccount.json"),
databaseURL: <YOUR FIREBASE URL>
});
}
firebase.database().ref('conversation').once('value').then(function(snapshot) {
console.log (snapshot.val()) ;
var bodyReturn = {
input: snapshot.val()
} ;
callback(null,bodyReturn);
context.succeed() ;
});
};
いくつかのことを試した後、これは私のために働くようです(v 3.10.8):
for(var i=0;i<5;i++)
{
var firebase = require('firebase');
var config = {
apiKey: "",
authDomain: "",
databaseURL: "",
storageBucket: "",
messagingSenderId: ""
};
if(firebase.apps)
if(firebase.apps.length==0)
firebase.initializeApp(config)
firebase.database().ref().child("test").once('value').
then(function(snapshot) {
console.log(snapshot.val());
});
}