node.js - 在 firebase 中动态导出函数

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

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

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 示例,但我使用的是 typescript 。我尝试了多种变体,虽然进行了一些构建,但最后我总是坚持我的函数没有被导出,并且部署警告我我将删除现有函数。

这是众多尝试之一:

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 block ,从不进入 catch),但导出的变量没有被替换。

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

所以问题是 - 如何避免导入/导出每个特定功能不需要的任何内容?

最佳答案

您可以坚持使用原始的 index.js,稍作改动。我已经尝试了几次并提出了解决方案。我包括一个示例目录结构和两个 typescript 文件(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/索引.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”中

关于node.js - 在 firebase 中动态导出函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52959193/

相关文章:

javascript - .get() 函数获取 firebase 文档 "is not a function"

java - 如何从内部具有监听器的 Jersey REST api 方法返回响应

node.js - 当我在函数中使用请求 promise 并返回一个值时,它显示未定义

Node.js 解释 REPL 中的错误?

typescript 如何提取嵌套类型

reactjs - React FormEvent<HTMLFormElement> 表单输入 Prop 类型

node.js - 使用异步请求

Node.js 服务器自动启动

javascript - 未找到命名导出 'Types'。请求的模块 'mongoose' 是一个 CommonJS 模块,它可能不支持所有 module.exports 作为命名导出

android - 所有 gms/firebase 库必须使用完全相同的版本