代码之家  ›  专栏  ›  技术社区  ›  Mika Saari

节点。js和express:如何等待udp响应

  •  4
  • Mika Saari  · 技术社区  · 9 年前

    我正在研究一个比特节点。js编程,并面临一个问题。

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

    如果我应该等待radius响应(非常快),我应该如何在节点中进行等待。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 回复  |  直到 9 年前
        1
  •  5
  •   ralh    9 年前

    您需要在 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 回调。

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

        2
  •  0
  •   Ian Sebastian    6 年前

    如果你不能使用 res handleAuthentication 函数,否则,出于某些代码设计目的,您不想在那里响应客户端,可以使用新的JSasync/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,返回语句将在后台转换为相应的resoly/reject,然后您可以在函数之外做任何想做的事情。