3

我有连接到 db 并管理连接的 database.js 文件我导出连接并在我的应用程序中重用它

    var mysql = require('mysql');

    pool = mysql.createPool({
        host: cfg.mysql.host,
        user: cfg.mysql.user,
        password: cfg.mysql.pass,
        database: cfg.mysql.db,
        port: cfg.mysql.port
    });


    function handleDisconnect() {
    pool.getConnection(function(err, cnt) {
    module.exports.connection = cnt;
    });

    pool.on('error', function (err) {
      console.log(err);
    });
    };

    handleDisconnect();
    process.on('uncaughtException', function (err) {
      console.error(err.code);
    if(err.code === "PROTOCOL_CONNECTION_LOST")
    handleDisconnect();
      console.log("Node NOT Exiting...");
    });

应用程序.js

    var db = require('./database');
     app.get('/test', function(req, res) {
        db.connection.query('SELECT * from table', function(err, result) {
           console.log(result);
          });
        }

这工作正常。我的问题是mysql服务器断开连接时。我通过调用 handleDisconnect() 函数来获取新连接来处理此错误。但是,当我导航到浏览器 /test 时,我的 app.js 中的连接未定义

TypeError: Cannot call method 'query' of undefined

想法?

4

3 回答 3

2

您可以在数据库服务器上的每个查询之后简单地将连接放回池中,这是一个示例

数据库.js

var mysql = require('mysql');
var pool  = mysql.createPool({
  host     : 'localhost',
  user     : 'root',
  password : 'root',
  database : 'test',
  port: 8889
});

exports.getConnection = function(callback) {
    pool.getConnection(function(err, connection) {
        callback(err, connection);
    });
};

应用程序.js

var db = require('./db');
var http = require("http");
var server = http.createServer(function(request, response) {
    db.getConnection(function(err, connection){
        if(!err){
            connection.query('SELECT * from users' , function(err, rows) {
                var users = '';
                for (var i = 0; i < rows.length; i++) {
                    users += rows[i].name + "<br />";
                }
                connection.release(); // always put connection back in pool after last query
                response.writeHead(200, {"Content-Type": "text/html"});
                response.write("<!DOCTYPE html>");
                response.write("<html>");
                response.write("<head>");
                response.write("<title>Hello World Page</title>");
                response.write("</head>");
                response.write("<body>");
                response.write(users);
                response.write("</body>");
                response.write("</html>");
                response.end();
           });
        }else{
            response.writeHead(200, {"Content-Type": "text/html"});
            response.write("<!DOCTYPE html>");
            response.write("<html>");
            response.write("<head>");
            response.write("<title>Hello World Page</title>");
            response.write("</head>");
            response.write("<body>");
            response.write(err.toString());
            response.write("</body>");
            response.write("</html>");
            response.end();
        }
    });
});
server.listen(9999);
console.log("Server is listening localhost:9999");
于 2013-10-26T18:41:46.847 回答
1

更新:请参阅下面的第一条评论,以更正我对问题的[错误]理解。我留下这个回复,而不是删除它,因为它涉及多个问题。

您不能继续重新分配给 module.exports。它不起作用,因为导出仅在 require() 时间发生一次(实际上,我怀疑您上面的代码完全起作用只是因为 pool.getConnection 同步执行,即在回调中分配给 module.exports 通常不安全) . 这是发生了什么:

在 app.js 你做:var db = require('./database');

它存储的不是对 module.exports(database.js)的引用,而是对分配给 module.exports 的对象(即连接对象)的引用。当该对象由于断开连接而被销毁时,app.js 中的变量“db”什么都没有(请注意,node-mysql 文档很清楚您不能重用连接对象)。您真正想要做的是向 app.js 返回一个新的连接对象,但这是不可能的,因为 app.js 调用只需要一次。

我了解到您想隐藏 app.js 中的连接重新建立。一种方法是将 node-mysql 连接对象包装在您自己的 database.js 对象中,并将后者返回给 app.js。这个被包装的对象可以在调用它的query()方法时从池中获取一个连接,然后调用获取到的连接的query()方法。很快。

于 2013-10-27T14:46:52.343 回答
0

首先签入会pool.getConnection引发handleDisconnect错误。由于您没有在重新连接之前建立超时以防连接丢失,因此立即调用pool.getConnection可能会因为同样的问题而失败。

此外,在连接丢失和handleDisconnect重新连接之间,任何传入的 HTTP 请求都无法执行数据库查询,因此应db.connection在尝试运行查询之前检查是否确实有效。

那,或者您也需要处理 Express 路由中的错误,例如使用错误处理程序路由(应该放置在您的路由/中间件设置的最后某个位置):

// middleware setup
...
// routes setup
...
// finally:
app.use(function(err, req, res, next) {
  res.send(500, err.message); // or whatever you want to send back in case of errors.
});
于 2013-10-26T12:55:14.210 回答