【问题标题】:Dynamically exporting functions in firebase在 Firebase 中动态导出函数
【发布时间】:2019-03-28 06:58:12
【问题描述】:

我遇到了云功能中冷启动时间的典型(根据许多帖子)问题。一个似乎承诺的解决方案建议只导入/导出实际正在执行的函数,如下所示:

https://github.com/firebase/functions-samples/issues/170#issuecomment-323375462

if (!process.env.FUNCTION_NAME || process.env.FUNCTION_NAME === 'sendFollowerNotification') {
  exports.sendFollowerNotification = require('./sendFollowerNotification');
}

这是一个 Javascript 示例,但我使用的是打字稿。我尝试了许多变体,虽然有一些构建,但最后我总是卡在我的函数没有被导出并部署警告我要删除现有函数。

这是众多尝试之一:

if (!process.env.FUNCTION_NAME || process.env.FUNCTION_NAME === 'generateInviteURL') {
  import ('./invite_functions').then ((mod) => { console.log ("mod follows" ); console.log (mod);  exports.generateInviteURL = functions.https.onRequest( mod.generateInviteURL ); } )
  .catch ((err) => {console.log ("Trying to import/export generateInviteURL ", err);}) ;
}

如上所述,在部署时发生的情况是我收到有关该功能被删除的警告。

我能够通过以下方式“避免”该消息:

console.log ("Function name: ", process.env.FUNCTION_NAME);

function dummy_generateInviteURL (req, res) { ; }
exports.generateInviteURL = functions.https.onRequest( dummy_generateInviteURL );

if (!process.env.FUNCTION_NAME || process.env.FUNCTION_NAME === 'generateInviteURL') {
console.log ("Doing the good import");
import ('./invite_functions').then ((mod) => { console.log ("mod follows" ); console.log (mod);  exports.generateInviteURL = functions.https.onRequest( mod.generateInviteURL ); } )
  .catch ((err) => {console.log ("Trying to import/export generateInviteURL ", err);}) ;
}

console.log ("Exported");
console.log (exports.generateInviteURL);

当然,空函数总是会被导出,但如果是被调用的函数,就会被真实的函数替换。

在这种情况下,日志如下所示:

generateInviteURL Function name:  generateInviteURL generateInviteURL  
generateInviteURL Exported generateInviteURL
{ [Function: cloudFunction] __trigger: { httpsTrigger: {} } }

所以第一部分看起来很有希望(定义了环境变量),然后导入做了一些事情(进入 then 块,从不捕获),但导出的变量没有被替换。

我不确定这是 TypeScript 问题、firebase 问题还是开发人员问题 - 可能我只是遗漏了一些明显的东西。

那么问题 - 我如何避免导入/导出每个特定功能不需要的任何东西?

【问题讨论】:

  • 所有 JavaScript 都是有效的 TypeScript。您是否尝试将原始代码保持原样以查看它是否有效?还是你需要更多的东西?
  • @DougStevenson 它的一个变体(使用 require 而不是 import)确实有效(因为它构建和部署,但似乎对冷启动时间没有显着影响)。老实说,我开始导入的东西是因为 typescript headers 上的 require 会导致错误/警告。
  • 我通常会忽略 VS Code 中的这些警告。 import 与 require 不完全相同。此时通常无法避免一些冷启动时间(尤其是 Firestore)。

标签: node.js typescript firebase google-cloud-functions firebase-cli


【解决方案1】:

您可以坚持使用原始 index.js,只需稍作改动。我已经尝试了几次并提出了解决方案。我包括一个示例目录结构和两个打字稿文件(index.ts 和另一个用于您的自定义函数)。有了这个,您将永远不必更改 index.ts 来修改或添加功能。

目录结构

+ functions
|
 -- + src
|   |
|    -- index.ts
|   |
|    -- + get
|       |
|        -- status.f.ts
 -- package.json (auto generated)
|
 -- package-lock.json (auto generated)
|
 -- tsconfig.json (auto generated)
|
 -- tslint.json (auto generated)
|
 -- + lib (auto generated)
|
 -- + node_modules (auto generated)

src/index.ts

import * as glob from "glob";
import * as camelCase from "camelcase";
const files = glob.sync('./**/*.f.js', { cwd: __dirname, ignore: './node_modules/**'});
for(let f=0,fl=files.length; f<fl; f++){
  const file = files[f];
  const functionName = camelCase(file.slice(0, -5).split('/').join('_')); // Strip off '.f.js'
  if (!process.env.FUNCTION_NAME || process.env.FUNCTION_NAME === functionName) {
    exports[functionName] = require(file);
  }
}

src/get/status.f.ts

import * as functions from 'firebase-functions';

exports = module.exports = functions.https.onRequest((req, res) => {
    res.status(200).json({'status':'OK'})
})

创建上述文件后,安装 npm 包 'glob' 和 'camelcase', 然后尝试部署 firebase 功能

Firebase 将部署一个名为“getStatus”的函数

注意函数名是驼峰版的文件夹名和函数所在的文件名,所以你只能每个.f.ts文件导出一个函数

编辑

我已经更新了目录结构。请注意,index.ts 和所有后续文件和文件夹都位于父文件夹“src”中

【讨论】:

  • 这不起作用没有功能上传您的示例
  • 您是否按照我提到的方式构建了目录?代码的工作非常简单。首先将 TypeScript 文件转换为 JavaScript (.js) 文件,然后 index.js(从 index.ts 转换而来)查找所有以 .f.js 结尾的文件并上传 .f.js 文件导出的函数.
  • 还要注意我已经更新了文件夹结构。除此之外,如果您遇到任何错误,请告诉我们。
  • 如果有人在 process.env.FUNCTION_NAME 中未定义,请改用 procecess.env.K_SERVICE
【解决方案2】:

我一直有类似的问题,我写了一个非常酷的解决方案,对我有用。

我决定将它作为一个开源包发布 - 我以前从未这样做过,所以如果有人可以查看、贡献或提供反馈,那就太好了。

包名为better-firebase-functions - https://www.npmjs.com/package/better-firebase-functions

您所要做的就是导入并运行它。两条线。几乎所有其他事情都是自动化的。

它将获取您目录中所有函数文件的默认导出并将它们部署为正确命名的函数。 这是一个例子:

import { exportFunctions } from 'better-firebase-functions';

exportFunctions({
   __filename,                         // standard node var (leave as is).
   exports,                            // standard node var (leave as is).
   functionDirectoryPath: './myFuncs', // define root functions folder
                                       // relative to this file.
   searchGlob: '**/*.js'               // file search glob pattern.
});

您唯一需要做的另一件事是将您的function.http.onRequest()... 导出为每个包含您要部署的云功能的文件的默认导出。所以:

export default functions.http.onRequest()...
/// OR
const func1 = functions.http.onRequest()...
export default func1;

就是这样! 更新:已编辑答案以反映更新版本的软件包。

【讨论】:

  • 这可以在打字稿中完成吗?尝试此操作时,我不断收到错误“文件 lib/index.js 无法加载”。谢谢。
  • 是的,这是一个 TS 和 JS 兼容的库。如果您在 github 问题跟踪器中留下更详细的问题,我将很乐意帮助您使其正常工作。尝试同时使用最新版本和 v2.0.1,看看是否可以为您修复它。
猜你喜欢
  • 2020-12-28
  • 2019-03-08
  • 2022-06-18
  • 2019-02-22
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-09-17
  • 2019-10-17
相关资源
最近更新 更多