【问题标题】:Why can't I catch error thrown from node-postgres?为什么我无法捕获 node-postgres 引发的错误?
【发布时间】:2021-03-21 04:09:40
【问题描述】:

我在捕获 Node-Postgres NPM 包引发的错误时遇到问题。

这个问题表面上看起来很简单,但我已经尝试了所有我能想到的。

我的代码如下:

import { Pool } from 'pg' // Import postgres connection pool
const pgPool = new Pool()

async function queryDatabase() {
  try {

    // Force TypeError by passing undefined
    let queryResult = await pgPool.query( undefined )

    if ( queryResult.rows.length > 0 ) {
      return queryResult.rows[0]
    }

    return false

  } catch( err ) {

    // Never Reached
    return new Error( 'Test error' )

  }
}
queryDatabase()

而且报错如下:

TypeError:客户端传递了一个空或未定义的查询
在 Client.query (~/.../node_modules/pg/lib/client.js:479:11)

错误本身是不言自明的。我在这里强制错误,以便在undefined 被错误传递的情况下尝试处理它。我意识到我可以简单地执行检查以确保输入永远不会为空或未定义,但这不是我主要关心的问题。

我担心的是,如果我无法捕获从这个包中抛出的这个错误,我会遇到多少其他无法预料的情况,我根本无法捕获和处理抛出的错误。

我尝试了许多不同的方法 - 上面显示的 Async/Await Try/Catch 方法 - 我尝试了 pgPool.query().then().catch() - 两者的组合。我什至尝试对 Pool 实例本身运行 catch。不管我做什么,不使用Node的process.on('unhandledRejection', ...)我都无法处理异常,这当然是个坏主意。

我已经为此绞尽脑汁好几个小时了。有什么方法可以捕捉和处理这样的错误,所以它不会每次都让我的服务器崩溃?提前致谢!

【问题讨论】:

  • 感谢您的快速响应,但这不起作用。 pg 包不会通过其错误处理程序路由此特定错误。他们只是扔它。
  • 如果传递undefined 导致上面的代码没有捕捉到异常,那就是一个错误。请确保您拥有最新版本的 pg,然后在 github 上报告问题。

标签: javascript node.js npm error-handling try-catch


【解决方案1】:

我能够重现这一点,它似乎是 pg-library 中的一个实际错误。

根据来源,如果您在池实例上调用.query,则此实例将attempt to connect and get a client。在此连接回调中,实际查询被分派到客户端模块,这将引发上述类型错误if the query is nil

这个错误是同步抛出的(即错误传递给callback参数,例如callback(new TypeError("...")),因为在池的client.query调用周围没有try/catch connect-callback,错误不会被你的try/catch捕获。

一个潜在的解决方法是将 client.query 调用包装在 try catch 中:

 client.once('error', onError)
      this.log('dispatching query')
      try {
        client.query(text, values, (err, res) => {
          this.log('query dispatched')
          client.removeListener('error', onError)
          if (clientReleased) {
            return
          }
          clientReleased = true
          client.release(err)
          if (err) {
            return cb(err)
          } else {
            return cb(undefined, res)
          }
        })
      }catch(err) {
        return cb(err)
      }

所以现在,恐怕你应该在 github 上创建一个问题并等待错误修复或 fork 存储库并使用上述解决方法。

【讨论】:

  • 感谢您的回复。不幸的是,这是包裹的问题。我想我的困惑源于不知道抛出的错误不会通过调用堆栈冒泡。似乎效率很低。
猜你喜欢
  • 1970-01-01
  • 2019-05-27
  • 1970-01-01
  • 2013-09-09
  • 1970-01-01
  • 2018-08-31
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多