mysql - Socket.io 和 Node.js 与 mysql 未按预期返回结果

标签 mysql node.js socket.io

编辑:我正在重新表述我的问题:Socket.io 不等待回调并且连接从未被接受。请参阅下面的编辑 2/尝试 1

这是为了根据数据库检查身份验证 token 。有人能发现这里出了什么问题吗?

var checkauth = function(auth) {
    var rs = 0;
    var sql = 'SELECT * FROM clients WHERE pword=\''+auth+'\''; 
    // Copied from debug session and got results: SELECT * FROM clients WHERE pword='d98e623c7a74a178703d17e1fd536b1488724acd41e71f178331c768c385bda2c82d2bcb60cbb4650be375ad4734c63fb694bd164c138f9abe0c51f37f9a7e33'
    var query = connection.query(sql);
    query
        .on('error', function(err) {
            console.log( err );
        })
        .on('result', function( row ) {
            rs = Number(row.client_id); // This never fires!!?!?!
        })
        .on('end',function(){

        });
    if (rs == 0) {
        sql = 'SELECT * FROM users WHERE pword=\''+auth+'\'';
        query = connection.query(sql);
        query
            .on('error', function(err) {
                console.log( err );
            })
            .on('result', function( row ) {
                rs = Number(row.client_id); // This never fires!!?!?!
            })
            .on('end',function(){
            });
    }
    return rs;
}

编辑1:我使用下面的循环来验证运行该函数的socket.io连接,好吧,所以我明白我必须等待数据库完成,我将回调放在哪里?

io.use(function(socket, next){
    console.log("Query: ", socket.handshake.query);
    // return the result of next() to accept the connection.
    socket.clientid = 0;
    socket.clientid = checkauth(socket.handshake.query.auth);
    console.log("CID:"+socket.clientid);
    if (socket.clientid != 0) {
        return next();
    }
    // call next() with an Error if you need to reject the connection.
    next(new Error('Authentication error'));
});

根据约翰内斯的建议编辑2/尝试1:

var checkauth = function(auth, cb) {
    var rs = 0;
    var sql = 'SELECT * FROM clients WHERE pword=?';
    var query = connection.query(sql, [auth]);
    query
        .on('error', function(err) {
            console.log(err);
        })
        .on('result', function(row) {
            rs = Number(row.client_id);
        })
        .on('end', function() {
            if (rs == 0) {
                sql = 'SELECT * FROM users WHERE pword=?';
                query = connection.query(sql, [auth]);
                query
                    .on('error', function(err) {
                        console.log(err);
                    })
                    .on('result', function(row) {
                        rs = Number(row.client_id);
                    })
                    .on('end', function() {
                        cb(rs);
                    });
            }
        });
}

io.use(function(socket, next){
    console.log("Query: ", socket.handshake.query);
    // return the result of next() to accept the connection.
    socket.clientid = 0;
    var auth = socket.handshake.query.auth;

    checkauth(auth, function(clientid){
        socket.clientid = clientid;
        if (clientid != 0) {
            return next();
        }
        console.log('CLIENID', clientid);
        next(new Error('Authentication error'));
    });

    // if (socket.clientid != 0) {
    //     return next();
    // }
    // // call next() with an Error if you need to reject the connection.
    // next(new Error('Authentication error'));
});

最佳答案

整个过程是异步的。你必须做两件事:

将所有内容移至事件回调中并 向您的 checkAuth 函数添加回调或 promise 。您还应该转义插入的数据。

编辑解决方案

var checkauth = function(auth, cb) {
    var rs = 0;
    var sql = 'SELECT * FROM clients WHERE pword=?';
    var query = connection.query(sql, [auth], (err, client) => {
        if(err || !client || client.length < 1) return cb(err || new Error('unkown client')); 
        sql = 'SELECT * FROM users WHERE pword=?';
        query = connection.query(sql, [auth], (err, user) => {
            if(err || !user || user.length < 1) return cb(err || new Error('unkown user'));
            cb(null, client[0].client_id);
        });
    });
}

注意:为什么您首先要执行 2 个查询并存储用户的每个客户端的密码?

原答案:

var checkauth = function(auth, cb) {
    var rs = 0;
    var sql = 'SELECT * FROM clients WHERE pword=?';
    var query = connection.query(sql, [auth]);
    query
        .on('error', function(err) {
            console.log(err);
        })
        .on('result', function(row) {
            rs = Number(row.client_id); // This never fires!!?!?!
        })
        .on('end', function() {
            if (rs == 0) {
                sql = 'SELECT * FROM users WHERE pword=?';
                query = connection.query(sql, [auth]);
                query
                    .on('error', function(err) {
                        console.log(err);
                    })
                    .on('result', function(row) {
                        rs = Number(row.client_id); // This never fires!!?!?!
                    })
                    .on('end', function() {
                        cb(rs);
                    });
            }
        });
}

checkauth(123, function(clientId){
    console.log('CLIENID', clientId);
});

关于mysql - Socket.io 和 Node.js 与 mysql 未按预期返回结果,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39371779/

相关文章:

node.js - WebSocket 握手期间出错 : 'Connection' header is missing

mysql - 显示相反值的 SQL 查询

mysql - inner join mysql的思考

Mysql重置id字段

node.js - cron 作业的 Spotify API 授权

node.js - 如何在 socket.io 中禁用客户端日志记录

mysql - 家谱关系数据库

node.js - mongoDB + NodeJS : can't keep a stream cursor open

node.js - 如何从大型 ZIP 存档中提取单个文件

node.js - 如何使用 WebSockets 进行请求-响应模式