我在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"
那么,如何使用管理员权限在云功能中获取辅助数据库的实例?
答案 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,这是在此问题上挽救了我生命的文档链接。
因此,它看起来像在我的云函数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!