草庐IT

node.js - 当数据库再次可访问时,Mongoose 连接不会恢复

coder 2023-11-06 原文

问题是 Mongoose 在以下情况下失去连接​​:

  1. Node App 已启动,一切正常(可访问数据库等)
  2. 本地运行的 Mongo 服务器进程(版本 2.6.10)停止,然后在 15 秒后启动,一切似乎都很好(可访问数据库等)
  3. Mongo 进程已停止。
  4. 已发出对 Express 的请求,试图转到数据库(使用 Mongoose)——如预期的那样无法访问。
  5. Mongo 进程启动。
  6. 发出相同的请求,即使 Mongo 已启动,也无法访问数据库。

如果我重新启动 Node JS 应用程序,一切正常。

为什么对 Mongo 的失败查询(使用 Mongoose 版本 4.4.5)会阻止在数据库进程再次启动时恢复连接?

我们是否应该实现重试机制来尝试恢复连接,直到数据库变得可访问?

我试过提到的配置 here但它没有用。

如有任何帮助,我们将不胜感激。

下面是我们连接助手的示例代码:

'use strict';

const _ = require('lodash');
const mongoose = require('mongoose');
const config = require('../config');
const logger = require('../logger');

const _instance = Symbol('instance');
const _enforcer = Symbol('enforcer');
const _members = Symbol('members');

/**
 * Singleton implementation of ConnectionHelper module
 * This module is responsible to reusing common db connections in the application

 * @type {ConnectionsHelper}
 */
module.exports = class ConnectionsHelper {
  constructor(enforcer) {
    if (enforcer !== _enforcer) {
      throw new Error('invalid singleton instantiation');
    }
    initConnectionFromConfig.call(this);
  }

  /**
   * The single instance
   * @returns {*}
     */
  static get instance() {
    if (!this[_instance]) {
      this[_instance] = new ConnectionsHelper(_enforcer);
    }
    return this[_instance];
  }

  /**
   * method retrieves connection by its name
   * @param connectionKey
   * @returns {*}
   */
  getConnection(connectionKey) {
    return this[_members][connectionKey];
  }

  /**
   * method disconnects all underlying connections
   * @returns {void|MongooseThenable}
   */
  closeConnections() {
    return mongoose.disconnect();
  }
};

function initConnectionFromConfig() {
  this[_members] = {};
  const dbsConnections = config.get('dbsConnections');

  _.forEach(dbsConnections, (connection, connectionName) => {

    const protocol = connection.protocol;
    const repSetPath = connection.mongoPath.join(',');
    const options = connection.options;

    options.server = {auto_reconnect: true, socketOptions: {keepAlive: 1, connectTimeoutMS: 30000 }};
    options.replset = {socketOptions: {keepAlive: 1, connectTimeoutMS: 30000 }};

    this[_members][connectionName] = mongoose.createConnection(protocol + repSetPath, options);

    addConnectionEvents.call(this, connectionName);
  });
}

function initConnectionIfNeeded() {
  this[_members] = {};
  const dbsConnections = config.get('dbsConnections');

  _.forEach(dbsConnections, (connection, connectionName) => {
    const protocol = connection.protocol;
    const repSetPath = connection.mongoPath.join(',');
    const options = connection.options;
    //options.server = {auto_reconnect: true, socketOptions: {keepAlive: 1, connectTimeoutMS: 30000 }};
    //options.replset = {socketOptions: {keepAlive: 1, connectTimeoutMS: 30000 }};

    this[_members][connectionName] = mongoose.createConnection(protocol + repSetPath, options);
    addConnectionEvents.call(this, connectionName);
  });
}

function addConnectionEvents(connectionName) {
  const connection = this[_members][connectionName];

  connection.on('connected', () => {
    logger.debug(`Mongoose connection open`);
  });
  connection.on('error', (err) => {
    logger.debug(`Mongoose connection error: ${err}`);
  });
  connection.on('exception', (err) => {
    logger.debug(`Mongoose connection exception: ${err}`);
  });
  connection.on('disconnected', () => {
    logger.debug(`Mongoose connection disconnected`);
  });
  connection.on('reconnected', () => {
    logger.debug(`Mongoose connection reconnected`);
  });
  connection.on('open', () => {
    logger.debug(`Mongoose connection is open`);
  });
  // If the Node process ends, close the Mongoose connection
  process.on('SIGINT', () => {
    connection.close(() => {
      logger.debug(`Mongoose default connection disconnected through app termination`);
      process.exit(0);
    });
  });
}

最佳答案

我遇到了同样的问题。我所做的是在这里重启 Mongoose :

connection.on('disconnected', () => {
    logger.debug(`Mongoose connection disconnected`);
    mongoose.connect(path);
  });

而且我还使用 setTimeout() 以便它每 3 秒尝试一次。

mongoose.connection.on('disconnected', function () {  
  console.log('Mongoose default connection disconnected'); 
  console.log("Trying to reconnect");

  function connect(){
    mongoose.connect(dbPath, function(err){
      if(err) console.log("Error while trying to reconnect to MongoDB");
    });
  }
  setTimeout(connect, 3000);
});

而且我也使用 ForeverJS。

关于node.js - 当数据库再次可访问时,Mongoose 连接不会恢复,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35739873/

有关node.js - 当数据库再次可访问时,Mongoose 连接不会恢复的更多相关文章

  1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  2. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  3. ruby-on-rails - 在混合/模块中覆盖模型的属性访问器 - 2

    我有一个包含模块的模型。我想在模块中覆盖模型的访问器方法。例如:classBlah这显然行不通。有什么想法可以实现吗? 最佳答案 您的代码看起来是正确的。我们正在毫无困难地使用这个确切的模式。如果我没记错的话,Rails使用#method_missing作为属性setter,因此您的模块将优先,阻止ActiveRecord的setter。如果您正在使用ActiveSupport::Concern(参见thisblogpost),那么您的实例方法需要进入一个特殊的模块:classBlah

  4. ruby - Highline 询问方法不会使用同一行 - 2

    设置:狂欢ruby1.9.2高线(1.6.13)描述:我已经相当习惯在其他一些项目中使用highline,但已经有几个月没有使用它了。现在,在Ruby1.9.2上全新安装时,它似乎不允许在同一行回答提示。所以以前我会看到类似的东西:require"highline/import"ask"Whatisyourfavoritecolor?"并得到:Whatisyourfavoritecolor?|现在我看到类似的东西:Whatisyourfavoritecolor?|竖线(|)符号是我的终端光标。知道为什么会发生这种变化吗? 最佳答案

  5. ruby - 续集在添加关联时访问many_to_many连接表 - 2

    我正在使用Sequel构建一个愿望list系统。我有一个wishlists和itemstable和一个items_wishlists连接表(该名称是续集选择的名称)。items_wishlists表还有一个用于facebookid的额外列(因此我可以存储opengraph操作),这是一个NOTNULL列。我还有Wishlist和Item具有续集many_to_many关联的模型已建立。Wishlist类也有:selectmany_to_many关联的选项设置为select:[:items.*,:items_wishlists__facebook_action_id].有没有一种方法可以

  6. ruby-on-rails - 项目升级后 Pow 不会更改 ruby​​ 版本 - 2

    我在我的Rails项目中使用Pow和powifygem。现在我尝试升级我的ruby​​版本(从1.9.3到2.0.0,我使用RVM)当我切换ruby​​版本、安装所有gem依赖项时,我通过运行railss并访问localhost:3000确保该应用程序正常运行以前,我通过使用pow访问http://my_app.dev来浏览我的应用程序。升级后,由于错误Bundler::RubyVersionMismatch:YourRubyversionis1.9.3,butyourGemfilespecified2.0.0,此url不起作用我尝试过的:重新创建pow应用程序重启pow服务器更新战俘

  7. ruby - Ruby 有 `Pair` 数据类型吗? - 2

    有时我需要处理键/值数据。我不喜欢使用数组,因为它们在大小上没有限制(很容易不小心添加超过2个项目,而且您最终需要稍后验证大小)。此外,0和1的索引变成了魔数(MagicNumber),并且在传达含义方面做得很差(“当我说0时,我的意思是head...”)。散列也不合适,因为可能会不小心添加额外的条目。我写了下面的类来解决这个问题:classPairattr_accessor:head,:taildefinitialize(h,t)@head,@tail=h,tendend它工作得很好并且解决了问题,但我很想知道:Ruby标准库是否已经带有这样一个类? 最佳

  8. ruby - 我如何添加二进制数据来遏制 POST - 2

    我正在尝试使用Curbgem执行以下POST以解析云curl-XPOST\-H"X-Parse-Application-Id:PARSE_APP_ID"\-H"X-Parse-REST-API-Key:PARSE_API_KEY"\-H"Content-Type:image/jpeg"\--data-binary'@myPicture.jpg'\https://api.parse.com/1/files/pic.jpg用这个:curl=Curl::Easy.new("https://api.parse.com/1/files/lion.jpg")curl.multipart_form_

  9. 世界前沿3D开发引擎HOOPS全面讲解——集3D数据读取、3D图形渲染、3D数据发布于一体的全新3D应用开发工具 - 2

    无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD

  10. FOHEART H1数据手套驱动Optitrack光学动捕双手运动(Unity3D) - 2

    本教程将在Unity3D中混合Optitrack与数据手套的数据流,在人体运动的基础上,添加双手手指部分的运动。双手手背的角度仍由Optitrack提供,数据手套提供双手手指的角度。 01  客户端软件分别安装MotiveBody与MotionVenus并校准人体与数据手套。MotiveBodyMotionVenus数据手套使用、校准流程参照:https://gitee.com/foheart_1/foheart-h1-data-summary.git02  数据转发打开MotiveBody软件的Streaming,开始向Unity3D广播数据;MotionVenus中设置->选项选择Unit

随机推荐