我的Firebase功能无效。这是日志:
11:01:03.932 AM warning getWatsonToken Error: ENOENT: no such file or directory, open '../public/javascript/services/watsonTokenValue.js'
at Error (native)
11:01:03.831 AM warning getWatsonToken Uncaught exception
11:01:03.233 AM info getWatsonToken 5YGY3R%2FBP0zelDOaob9PnxMWDj...
11:01:00.139 AM outlined_flag getWatsonToken Function execution took 804 ms, finished with status: 'ok'
11:00:59.834 AM info getWatsonToken Executing function!
11:00:59.335 AM outlined_flag getWatsonToken Function execution started
我不知道“Uncaught exception”是指什么。
“ENOENT:没有这样的文件或目录”错误可能是节点路径错误。这是我的目录结构:
── functions
├── index.js // this is my function
── public
├── javascript
│ ├── services
│ │ ├── watsonTokenValue.js // this is the file I want to write to
这是显然导致错误的一行:
fs.writeFile('../public/javascript/services/watsonTokenValue.js', tokenService, (err) => {
该行使用Node编写文件。路径有问题吗?
这是完整的功能:
// Node modules
const functions = require('firebase-functions');
const admin = require('firebase-admin');
const request = require('request'); // node module to send HTTP requests
const fs = require('fs');
admin.initializeApp(functions.config().firebase);
exports.getWatsonToken = functions.database.ref('userLoginEvent').onUpdate(event => { // authentication trigger when user logs in
console.log("Executing function!");
var username = '56ae6a1e7854',
password = 'swordfish',
url = 'https://' + username + ':' + password + '@stream.watsonplatform.net/authorization/api/v1/token?url=https://stream.watsonplatform.net/speech-to-text/api';
request({url: url}, function (error, response, body) {
console.log(body);
var tokenService = "app.value('watsonToken','" + body + "');";
fs.writeFile('../public/javascript/services/watsonTokenValue.js', tokenService, (err) => {
if (err) throw err;
console.log('The file has been saved!');
}); // close fs.writeFile
}); // close request
return 0; // prevents an error message "Function returned undefined, expected Promise or value"
}); // close getWatsonToken
前四行加载节点模块。该函数称为“getWatsonToken”。它在用户登录时触发,或者,具体地说,当登录过程中的一行代码将用户的Auth ID写入Firebase实时数据库中的某个位置时触发,然后onUpdate触发该功能。接下来,为HTTP请求定义参数,然后将HTTP请求发送到IBM Watson。 IBM为其语音到文本服务返回一个令牌,作为HTTP响应的主体。 (一个console.log显示了令牌,它正在工作。)然后一些Angular JavaScript被包裹在令牌中。最后,该文件将写入目录中的某个位置。最后一步似乎是错误所在。
这是我的目录结构的另一个视图:
感谢您的帮助!
答案 0 :(得分:3)
即使在同一个项目中,您也无法使用云功能来读取和写入Firebase托管服务的文件。
使用Firebase CLI进行部署时,会单独部署每个产品。因此,您public
文件夹中Firebase托管的所有静态内容都会发送到一个位置,并且您的所有云功能代码都会发送到其他位置。在那之后,他们没有任何共同点。
部署后,可以读取functions
文件夹下的文件。但显然,这些与部署后的托管无关。
我强烈建议不要尝试在云函数中编写文件,特别是对于您的用例。通过将持久性数据写入实时数据库,Firestore甚至云存储,可以更好地完成您要做的工作。云功能中的磁盘文件不是永久性的,您可能有多个服务器实例运行的功能彼此无关。
答案 1 :(得分:0)
谢谢,道格!这是我的Firebase功能:
// Node modules
const functions = require('firebase-functions');
const admin = require('firebase-admin');
const request = require('request'); // node module to send HTTP requests
const fs = require('fs');
admin.initializeApp(functions.config().firebase);
exports.getWatsonToken = functions.database.ref('userLoginEvent').onUpdate(event => { // authentication trigger when user logs in
var username = '56ae6a1e7854',
password = 'swordfish',
url = 'https://' + username + ':' + password + '@stream.watsonplatform.net/authorization/api/v1/token?url=https://stream.watsonplatform.net/speech-to-text/api';
request({url: url}, function (error, response, body) {
admin.database().ref('IBMWatsonToken').set({'token': body})
.then(function() {
console.log("Token saved!");
}); // close promise
}); // close request
return 0; // prevents an error message "Function returned undefined, expected Promise or value"
}); // close getWatsonToken
这会将令牌写入我的Firebase实时数据库。我的控制器代码是:
firebase.database().ref('IBMWatsonToken').on('value', function(snapshot) {
snapshot.forEach(function(childSnapshot) {
$scope.watsonToken = childSnapshot.val();
});
});
此代码引用Firebase实时数据库中的位置,然后on
侦听值的更改,然后forEach
读取值并将值放在$scope
上以用于控制器函数使用。
我会通过删除函数中的promise和console.log
来加快速度。