如何使用云函数正确连接MongoDB?

Tom*_*652 5 mongodb node.js firebase google-cloud-platform google-cloud-functions

我只想为每个运行 Cloud Functions 的实例连接一次 Atlas 集群。

这是我的实例代码:

const MongoClient = require("mongodb").MongoClient;

const client = new MongoClient("myUrl", {
  useNewUrlParser: true,
  useUnifiedTopology: true,
});

exports.myHttpMethod = functions.region("europe-west1").runWith({
  memory: "128MB",
  timeoutSeconds: 20,
}).https.onCall((data, context) => {
  console.log("Data is: ", data);
  client.connect(() => {
    const testCollection = client.db("myDB").collection("test");
    testCollection.insertOne(data);
  });
});
Run Code Online (Sandbox Code Playgroud)

我想避免client.connect()在每个函数调用中看起来确实太多了。

我想做这样的事情:

const MongoClient = require("mongodb").MongoClient;

const client = await MongoClient.connect("myUrl", {
  useNewUrlParser: true,
  useUnifiedTopology: true,
});

const db = client.db("myDB");

exports.myHttpMethod = functions.region("europe-west1").runWith({
  memory: "128MB",
  timeoutSeconds: 20,
}).https.onCall((data, context) => {
  console.log("Data is: ", data);
  const testCollection = db.collection("test");
  testCollection.insertOne(data);
});
Run Code Online (Sandbox Code Playgroud)

但我不能await喜欢这个。
在我的 AWS Lambda 函数(在 python 中运行)中,我没有这个问题,并且每个实例只能连接一次,所以我猜有一个等效的函数,但我不太了解 JS / Node JS。

Dha*_*raj 9

您可以将数据库客户端存储为全局变量。从文档来看,

Cloud Functions 通常会回收先前调用的执行环境。如果在全局范围内声明变量,则可以在后续调用中重用其值,而无需重新计算。

尝试重构代码,如下所示:

import * as functions from "firebase-functions";

import { MongoClient } from "mongodb"; 

let client: MongoClient | null;

const getClient = async () => {
  if (!client) {
    const mClient = new MongoClient("[MONGODB_URI]", {});
    client = await mClient.connect();
    functions.logger.log("Connected to MongoDB");
  } else {
    functions.logger.log("Using existing MongoDB connection");
  }
  functions.logger.log("Returning client");
  return client;
};

export const helloWorld = functions.https.onRequest(
  async (request, response) => {
    const db = (await getClient()).db("[DATABASE]");
    const result = await db.collection("[COLLECTION]").findOne({});
    response.send("Hello from Firebase!");
  }
);
Run Code Online (Sandbox Code Playgroud)

这应该重用该实例的连接。

在此输入图像描述