带有exress的Node.js处理中间件外部的数据库连接错误

时间:2018-06-22 09:42:40

标签: mysql node.js express error-handling

我对node还是很陌生,我正在尝试使用passwordjs创建一个简单的登录/注册系统。我有我的护照配置文件,其中将护照对象作为参数传递,如下所示。

我的护照配置文件:

var LocalStrategy   = require('passport-local').Strategy;

var User = require('./../models/user');

var mysql = require('./../database/mysql_setup');
var mysqlPool = mysql.pool;

// expose this function to our app using module.exports
module.exports = function(passport) {

    mysqlPool.getConnection(function(error, connection) {

        if (error) throw error;

        connection.query('USE vidyawxx_build2');

        // =========================================================================
        // passport session setup ==================================================
        // =========================================================================
        // required for persistent login sessions
        // passport needs ability to serialize and deserialize users out of session

        // used to serialize the user for the session
        passport.serializeUser(function(user, done) {
            done(null, user.username);
        });

        // used to deserialize the user
        passport.deserializeUser(function(username, done) {
            connection.query("SELECT * FROM `"+mysql.dbSpecs.prefix+"users` WHERE username = " + connection.escape(username), function(err,rows){
                done(err, rows[0]);
            });
        });


        // =========================================================================
        // LOCAL SIGNUP ============================================================
        // =========================================================================
        // we are using named strategies since we have one for login and one for signup
        // by default, if there was no name, it would just be called 'local'

        passport.use('local-signup', new LocalStrategy({
                // by default, local strategy uses username and password, we will override with email
                usernameField : 'username',
                passwordField : 'password',
                passReqToCallback : true // allows us to pass back the entire request to the callback
            },
            function(req, username, password, done) {

                // find a user whose username is the same as the forms username
                // we are checking to see if the user trying to login already exists
                connection.query("SELECT * FROM `"+mysql.dbSpecs.prefix+"users` WHERE `username` = "+connection.escape(username),function(err,rows){
                    if (err)
                        return done(err);
                    if (rows.length) {
                        return done(null, false, req.flash('error', 'This username is already in use.'));
                    } else {

                        // if there is no user with that username
                        // create the user
                        var newUserMysql = new User(username, password);

                        newUserMysql.generateHash(function(error, hash) {

                            if(error) {
                                return done(error);
                            }

                            var insertQuery = "INSERT INTO `"+mysql.dbSpecs.prefix+"users` ( username, password ) values (" + connection.escape(newUserMysql.username) +",'"+ hash +"')";

                            connection.query(insertQuery,function(err,rows){
                                if(err) {
                                    return done(error);
                                }
                                return done(null, rows);
                            });

                        });

                    }
                });
            }
        ));

        // =========================================================================
        // LOCAL LOGIN =============================================================
        // =========================================================================
        // we are using named strategies since we have one for login and one for signup
        // by default, if there was no name, it would just be called 'local'

        passport.use('local-login', new LocalStrategy({
                // by default, local strategy uses username and password
                usernameField : 'username',
                passwordField : 'password',
                passReqToCallback : true // allows us to pass back the entire request to the callback
            },
            function(req, username, password, done) { // callback with email and password from our form

                connection.query("SELECT * FROM `"+mysql.dbSpecs.prefix+"users` WHERE `username` = " + connection.escape(username), function(err,rows){

                    if (err) {
                        return done(err);
                    }
                    if (rows.length === 0) {
                        return done(null, false, req.flash('error', 'Oops! Wrong username or password.')); // req.flash is the way to set flashdata using connect-flash
                    }

                    // if the user is found but the password is wrong
                    var newUser = new User(username, password);
                    newUser.compareHash(function(error, result) {
                        if(result) {
                            return done(null, rows[0]);
                        } else {
                            return done(null, false, req.flash('error', 'Oops! Wrong username or password.')); // create the loginMessage and save it to session as flashdata
                        }
                    });

                });



            }
        ));

        connection.release();

    });
};

我的问题在于以下事实:如果我的mysql服务器由于任何原因而关闭,则该错误会引发在我的第一行中。我希望能够将用户重定向到一个简单的页面,该页面向他显示诸如“数据库有问题,请稍后再试”。问题是,当我抛出错误时,我的应用程序会立即关闭,并向任何访问者提供ERR_CONNECTION_REFUSED响应。(我目前正在本地进行此操作。

这是我的app.js文件:

var createError = require('http-errors');
var express = require('express');
var path = require('path');
var cookieParser = require('cookie-parser');
var logger = require('morgan');

var passport = require('passport');
var passportConfig = require('./config/passport');

var session = require("express-session");
var flash = require("connect-flash");

var indexRouter = require('./routes/index');
var usersRouter = require('./routes/users');

var app = express();
app.use(express.static(path.join(__dirname, 'public')));

// view engine setup
app.set('views', path.join(__dirname, 'views'));
app.set('view engine', 'ejs');

app.use(logger('dev'));
app.use(express.json());
app.use(express.urlencoded({ extended: false }));
app.use(cookieParser());

passportConfig(passport);
app.use(session({
    secret: "aPa1fgOed(&fjkKLN34%#$lpv@@",
    resave: true,
    saveUninitialized: true,
    cookie: { maxAge: 1000*60*15 } //15 minutes in milliseconds
}));

app.use(passport.initialize());
app.use(passport.session());
app.use(flash());

//create local vaariables for all our templates to use
app.use(function(req, res, next) {
    res.locals.errors = req.flash("error");
    res.locals.infos = req.flash("info");
    res.locals.successes = req.flash("success");
    next();
});

app.use('/', indexRouter);
app.use('/users', usersRouter);

// catch 404 and forward to error handler
app.use(function(req, res, next) {
  next(createError(404));
});

// error handler
app.use(function(err, req, res, next) {
  // set locals, only providing error in development
  res.locals.message = err.message;
  res.locals.error = req.app.get('env') === 'development' ? err : {};

  // render the error page
  res.status(err.status || 500);
  res.render('error');
});

module.exports = app;

话虽如此,由于错误是在我的password-config文件中引发的,它不遵循具有req,res和next params的中间件约定,因此我如何才能将用户正确地重定向到上述页面?

请确定,我再说一遍,这仅涉及mysql连接错误。我知道我可以使用done()通过我的password-config方法返回其他错误,但是数据库连接错误发生在具有完成参数的函数之外。

预先感谢

1 个答案:

答案 0 :(得分:0)

经过仔细研究后,我想到的唯一解决方法是将查询合并到通行证配置方法中,以便我可以通过done()函数将任何数据库连接错误传回;

这是我修改过的护照配置文件

// load all the things we need
var LocalStrategy   = require('passport-local').Strategy;

var User = require('./../models/user');

var mysql = require('./../database/mysql_setup');
var mysqlPool = mysql.pool;

// expose this function to our app using module.exports
module.exports = function(passport) {

    //connection.query('USE vidyawxx_build2');

    // =========================================================================
    // passport session setup ==================================================
    // =========================================================================
    // required for persistent login sessions
    // passport needs ability to serialize and deserialize users out of session

    // used to serialize the user for the session
    passport.serializeUser(function(user, done) {
        done(null, user.username);
    });

    // used to deserialize the user
    passport.deserializeUser(function(username, done) {
        mysqlPool.getConnection(function(dbError, connection) {

            if(dbError) {
                return done(dbError);
            }

            connection.query("SELECT * FROM `"+mysql.dbSpecs.prefix+"users` WHERE username = " + connection.escape(username), function(err,rows){
                if(err) {
                    done(err);
                    connection.release();
                    return;
                }
                connection.release();
                done(err, rows[0]);
            });
        });
    });


    // =========================================================================
    // LOCAL SIGNUP ============================================================
    // =========================================================================
    // we are using named strategies since we have one for login and one for signup
    // by default, if there was no name, it would just be called 'local'

    passport.use('local-signup', new LocalStrategy({
            // by default, local strategy uses username and password, we will override with email
            usernameField : 'username',
            passwordField : 'password',
            passReqToCallback : true // allows us to pass back the entire request to the callback
        },
        function(req, username, password, done) {

            // find a user whose username is the same as the forms username
            // we are checking to see if the user trying to login already exists
            mysqlPool.getConnection(function(dbError, connection) {

                if(dbError) {
                    return done(dbError);
                }

                connection.query("SELECT * FROM `"+mysql.dbSpecs.prefix+"users` WHERE `username` = "+connection.escape(username),function(err,rows){

                    if (err) {
                        connection.release();
                        return done(err);
                    }

                    if (rows.length) {
                        connection.release();
                        return done(null, false, req.flash('error', 'This username is already in use.'));
                    } else {

                        // if there is no user with that username
                        // create the user
                        var newUserMysql = new User(username, password);

                        newUserMysql.generateHash(function(error, hash) {

                            if(error) {
                                connection.release();
                                return done(error);
                            }

                            var insertQuery = "INSERT INTO `"+mysql.dbSpecs.prefix+"users` ( username, password ) values (" + connection.escape(newUserMysql.username) +",'"+ hash +"')";

                            mysqlPool.query(insertQuery,function(err,rows){
                                if(err) {
                                    connection.release();
                                    return done(error);
                                }
                                connection.release();
                                return done(null, rows);
                            });

                        });

                    }
                    connection.release();
                });

            });
        }
    ));

    // =========================================================================
    // LOCAL LOGIN =============================================================
    // =========================================================================
    // we are using named strategies since we have one for login and one for signup
    // by default, if there was no name, it would just be called 'local'

    passport.use('local-login', new LocalStrategy({
            // by default, local strategy uses username and password
            usernameField : 'username',
            passwordField : 'password',
            passReqToCallback : true // allows us to pass back the entire request to the callback
        },
        function(req, username, password, done) { // callback with email and password from our form

            mysqlPool.getConnection(function(dbError, connection) {
                if(dbError) {
                    return done(dbError);
                }
                connection.query("SELECT * FROM `"+mysql.dbSpecs.prefix+"users` WHERE `username` = " + connection.escape(username), function(err,rows){

                    if (err) {
                        connection.release();
                        return done(err);
                    }
                    if (rows.length === 0) {
                        connection.release();
                        return done(null, false, req.flash('error', 'Oops! Wrong username or password.')); // req.flash is the way to set flashdata using connect-flash
                    }

                    // if the user is found but the password is wrong
                    var newUser = new User(username, password);
                    newUser.compareHash(function(error, result) {
                        if(result) {
                            connection.release();
                            return done(null, rows[0]);
                        } else {
                            connection.release();
                            return done(null, false, req.flash('error', 'Oops! Wrong username or password.')); // create the loginMessage and save it to session as flashdata
                        }
                    });

                });
            });
        }
    ));


};