提问者:小点点

node.js和express:如何等待udp响应


我在这里学习了一点Node.js编程,遇到了一个问题。

当express收到POST请求时,需要通过dgram模块在UDP上进行radius认证。Radius身份验证的响应稍后出现,但POST请求回调已经退出,req和res变量丢失,使用相同TCP连接的响应无法完成。

如果我应该等待radius响应(非常快),我应该如何在node.js&;快车?

下面的代码执行基本的POST处理和radius身份验证。

非常感谢你提供的任何提示。


    var http = require('http');
    var express = require('express');
    var bodyParser = require('body-parser');
    var radius = require('radius');
    var dgram = require('dgram');

    var radius_secret = 'really_secret';
    var radius_server = '127.0.0.1';
    var radius_port = '1812';

    function handleAuthentication(req, res) {
        console.log("BODY:",req.body);
        var client = dgram.createSocket("udp4");
        var account = req.body.account; 
        var password = req.body.password;
        var packet = {
            code: 'Access-Request',
            secret: radius_secret,
            identifier: 0,
            attributes: [
                ['NAS-IP-Address', radius_server],
                ['User-Name', account + "@exampledomain.something"],
                ['User-Password', password]
            ]
        };

        client.on('error', function(e) {
                throw e;
            });

        client.on('message', function(msg, rinfo) {
            var radius_response = radius.decode({packet: msg, secret: radius_secret});
            console.log(radius_response);
        });

        var encoded = radius.encode(packet);
        client.send(encoded, 0, encoded.length, radius_port, radius_server);
    }

    var app = express();
    app.use(bodyParser.json());
    app.use(bodyParser.urlencoded({ extended: false}));

    app.post('/authenticate', function(req, res) {
        // Authenticate against radius server
        handleAuthentication(req, res);

        // The radius response is not received yet, and we will exit the post request
        // here and loose the req and res. 
    });

    var server = http.createServer(app);
    server.listen(80);


共2个答案

匿名用户

您需要在client回调中进行响应。因此,例如,要在消息之后响应:

client.on('message', function(msg, rinfo) {
     var radius_response = radius.decode({packet: msg, secret: radius_secret});
     console.log(radius_response);
     res.send('somethingCalculatedWithUDPResponse');
 });

您应该在error回调中类似地处理它。

resreq在函数结束时不会“死亡”,它们仍然可以被作为回调传递的闭包引用。

匿名用户

如果您不能在handleauthentication函数中使用res,或者出于某些代码设计目的不想在那里响应客户端,则可以使用新的JS async/await函数。

app.post('/authenticate', async function(req, res) {
    // Authenticate against radius server
    try{
        let udpResponse = await handleAuthentication(req, res);
    } catch (error){
        // handle error. use 'res' to notify about error
    }

    // if ok
    res.send("your response")

    //or better
    res.send(udpResponse)
});

在职能范围内:

client.on('message', function(msg, rinfo) {
        var radius_response = radius.decode({packet: msg, secret: radius_secret});
        return radius_response;
    });

Async/Await将把函数包装成promise,并且return语句将在后台相应地转换为resolve/reject,然后您可以在函数之外做您想做的任何事情。