无法读取云功能中的托管文件

时间:2018-01-23 18:33:35

标签: javascript node.js firebase google-cloud-functions firebase-hosting

我的Firebase功能无效。这是日志:

enter image description here

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被包裹在令牌中。最后,该文件将写入目录中的某个位置。最后一步似乎是错误所在。

这是我的目录结构的另一个视图:

enter image description here

感谢您的帮助!

2 个答案:

答案 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来加快速度。