如何在Firebase的云函数中访问多个实时数据库实例

时间:2017-12-18 02:25:01

标签: javascript firebase firebase-realtime-database google-cloud-functions firebase-admin

我在Firebase项目中使用多个数据库。主(默认)数据库的云功能很好,但是,我不能使它们适用于辅助数据库。例如,我想在具有管理员权限的节点上发出读取请求:

//this works
admin.database().ref(nodePath).once('value')...

这在主数据库中有效,但是,如果我想在另一个数据库上执行该命令,它不起作用:

//this doesn't work
admin.database(secondaryDatabaseUrl).ref(nodePath).once('value')...

虽然部署了这些功能,但在尝试执行云功能时,我在控制台上收到错误。

这里是带有https触发器的云功能代码:

const functions = require('firebase-functions');
const admin = require('firebase-admin');
admin.initializeApp(functions.config().firebase);
const secureCompare = require('secure-compare');

exports.testFunction= functions.https.onRequest((req, res) => {
  const key = req.query.key;
  // Exit if the keys don't match
  if (!secureCompare(key, functions.config().cron.key)) {
    console.error('keys do not match');
    res.status(403).send('error1');
    return;
  }
  //test read request
  //the line below crashes the function
  return admin.database('https://secondary_db_url.firebaseio.com').ref(`/testNode`).once('value').then(dataSnapshot=> {
    console.log('value', dataSnapshot.val());
    return;
  }).catch(er => {
    console.error('error', er);
    res.status(403).send('error2');
  });
});

以下是Firebase控制台中的错误日志:

TypeError: ns.ensureApp(...).database is not a function
    at FirebaseNamespace.fn (/user_code/node_modules/firebase-admin/lib/firebase-namespace.js:251:42)
    at exports.testFunction.functions.https.onRequest (/user_code/index.js:16:16)
    at cloudFunction (/user_code/node_modules/firebase-functions/lib/providers/https.js:26:41)
    at /var/tmp/worker/worker.js:671:7
    at /var/tmp/worker/worker.js:655:9
    at _combinedTickCallback (internal/process/next_tick.js:73:7)
    at process._tickDomainCallback (internal/process/next_tick.js:128:9)

如果我没有指定辅助数据库URL,该函数将在我的主数据库上生成读取请求,该请求非常有效:

//this works
return admin.database().ref(`/testNode`).once('value').then(dataSnapshot=> {
...

我使用的是最新的SDK版本:"firebase-admin": "^5.5.1""firebase-functions": "^0.7.3"

那么,如何使用管理员权限在云功能中获取辅助数据库的实例?

5 个答案:

答案 0 :(得分:10)

以下是使用Admin SDK按URL访问数据库的方法:

let app = admin.app();
let ref = app.database('https://secondary_db_url.firebaseio.com').ref();

以下是Admin SDK集成测试的示例:https://github.com/firebase/firebase-admin-node/blob/master/test/integration/database.js#L52

答案 1 :(得分:8)

所以看起来您正在尝试使用javascript Web客户端API访问多个数据库。将数据库的URL传递给这样的API并不适用于Admin SDK:

admin.database('https://secondary_db_url.firebaseio.com').ref(`/testNode`)

相反,您必须initialize a second app,为其命名,并将该应用程序传递给Admin SDK API。这是一个完整的示例,它将相同的数据写入同一项目中的两个不同的数据库实例:

const functions = require('firebase-functions')
const admin = require('firebase-admin')
admin.initializeApp(functions.config().firebase)

const otherConfig = Object.assign({}, functions.config().firebase)
otherConfig.databaseURL = 'https://your-other-db.firebaseio.com/'
const otherApp = admin.initializeApp(otherConfig, 'otherAppName')

exports.foo = functions.https.onRequest((req, res) => {
    const data = { foo: 'bar' }
    const p1 = admin.database().ref('data').set(data)
    const p2 = admin.database(otherApp).ref('data').set(data)
    Promise.all([p1, p2]).then(() => {
        res.send("OK")
    })
    .catch(error => {
        res.status(500).send(error)
    })
})

答案 2 :(得分:4)

现在使用云功能> 1.1,这是在此问题上挽救了我生命的文档链接。

https://firebase.google.com/docs/database/usage/sharding#connect_your_app_to_multiple_database_instances

因此,它看起来像在我的云函数index.js的顶部:

class TodayViewController: NSViewController, NSTableViewDataSource, NSTableViewDelegate, NCWidgetProviding  {

    @IBOutlet weak var tableView: NSTableView!

    override var nibName: NSNib.Name? {
        return NSNib.Name("TodayViewController")
    }

    func widgetPerformUpdate(completionHandler: (@escaping (NCUpdateResult) -> Void)) {
        // Update your data and prepare for a snapshot. Call completion handler when you are done
        // with NoData if nothing has changed or NewData if there is new data since the last
        // time we called you
        completionHandler(.noData)
    }

    override func viewDidLoad() {
        super.viewDidLoad()

        tableView.delegate = self
        tableView.dataSource = self

        tableView.headerView = nil

        tableView.target = self
        tableView.action = #selector(self.onItemClicked)

        tableView.isEnabled = true
        fetchData() // -> this method contains tableView.reloadData()
    }

    func tableView(_ tableView: NSTableView, viewFor tableColumn: NSTableColumn?, row: Int) -> NSView? { 
    // ...
    }

    func numberOfRows(in tableView: NSTableView) -> Int {
    // ...
    }
}

然后,在我的克隆函数函数代码中,我可以执行以下操作:

const functions = require('firebase-functions');
const admin = require('firebase-admin');
admin.initializeApp();

const dev = admin.initializeApp({
  databaseURL: "https://appdev.firebaseio.com"
}, 'dev');

const v2 = admin.initializeApp({
  databaseURL: "https://appv2.firebaseio.com"
}, 'v2');

答案 3 :(得分:2)

在 Firebase Functions v3.14.0 上更新此内容。这些答案都不适合我,所以我实施了 this solution

instance 注册一个函数来触发来自特定 Firebase 实时数据库实例的事件

functions.database.instance('my-app-db-2').ref('/foo/bar')

使用数据库实例的名称就可以了,不需要url。 functions.database.ref 不使用实例监视事件的默认实例。

答案 4 :(得分:0)

因此,如果两个答案都不起作用。

我发生的事情是这个方法没有任何错误,但第二个数据库实例没有得到更新。

我更新了npm和firebase CLI。

另外@Dough Stevenson你Passing the URL of the database to the API like this **does** work with the Admin SDK

这是一个来自Firebase的好博客 Firebase Blog : Easier scaling with multi-database support!