NodeJS - 响应流

时间:2017-08-13 15:29:11

标签: node.js sockets sails.js rxjs

我使用Sails.js在NodeJS上构建了一个简单的API端点。

当有人访问我的API端点时,服务器开始等待数据,每当出现新数据时,他都会使用套接字进行广播。每个客户端都应根据用户输入接收自己的数据流。

var Cap = require('cap').Cap;

collect: function (req, res) {

var iface = req.param("ip");

var c = new Cap(),
device = Cap.findDevice(ip);

    c.on('data', function(myData) {
        sails.sockets.blast('message', {"host": myData});
    });
});

响应没有完成(我从不发送res.json() - 实际发生的是浏览器继续加载 - 但上述功能有效)。

2个问题:

  • 我尝试从我的客户端订阅和取消订阅此API端点(使用RxJS)。当我订阅时,我开始通过套接字接收数据 - 但我无法取消订阅API端点(浏览器希望请求完成)。

  • 每个客户端都应根据请求IP参数订阅自己的套接字室(请参阅更新的代码)。目前,它向所有人发布消息。

如何使用Sails.js创建类似流/服务的API端点,根据他的输入向每个用户发送新数据?

我的目标是能够从每个客户端订阅/取消订阅此API端点。

1 个答案:

答案 0 :(得分:1)

修订答案

我们假设您的API端点在config/routes.js中定义如下:

...
'get     /collect': 'SomeController.collectSubscribe',
'delete  /collect': 'SomeController.collectUnsubscribe',

由于每个Cap实例都绑定到一个设备,因此每个订阅需要一个实例。我们不是使用sails join / leave方法,而是跟踪请求套接字ID中的Cap个实例和broadcast个实例。这是有效的,因为Sails套接字默认订阅了自己的ID。

api/controllers/SomeController.js

// In order for the `Cap` instances to persist after `collectSubscribe` finishes, we store them all in an Object, associated with which socket the were created for.
var caps = {/* req.socket.id: <instance of Cap>, */};

module.exports = {

...

  collectSubscribe: function(req, res) {
    if (!res.isSocket) return res.badRequest("I need a websocket! Help!");
    if (!!caps[req.socket.id]) return res.badRequest("Dude, you are already subscribed.");

    caps[req.socket.id] = new Cap();
    var c = caps[req.socket.id]; // remember that `c` is a reference to our new `Cap`, not a copy.
    var device = c.findDevice(req.param('ip'));

    c.open(device, ...);
    c.on('data', function(myData) {
      sails.sockets.broadcast(req.socket.id, 'message', {host: myData});
    });

    return res.ok();
  },

  collectUnsubscribe: function(req, res) {
    if (!res.isSocket) return res.badRequest("I need a websocket! Help!");
    if (!caps[req.socket.id]) return res.badRequest("I can't unsubscribe you unless you actually subscribe first.");

    caps[req.socket.id].removeAllListeners('data');
    delete caps[req.socket.id]; 

    return res.ok();
  }
}

基本上,它是这样的:当浏览器请求触发collectSubscribe时,新的Cap实例会侦听提供的IP。当浏览器触发collectUnsubscribe时,服务器会检索该Cap实例,告诉它停止侦听,然后删除它。

制作注意事项:请注意Cap的列表不是PERSISTENT(因为它存储在内存而不是数据库中)!因此,如果您的服务器已关闭并重新启动(由于雷电风暴等),列表将被清除,但考虑到所有websocket连接都将被丢弃,我认为没有必要担心这一点。

旧答案,保留参考

您可以使用sails.sockets.join(req, room)sails.sockets.leave(req, room)来管理套接字会议室。基本上你有一个名为“收集”的房间,只有在该房间加入的套接字才会收到sails.sockets.broadcast(room, eventName, data)

有关如何使用sails.sockets here的更多信息。

api/controllers/SomeController.js

collectSubscribe: function(req, res) {
  if (!res.isSocket) return res.badRequest();

  sails.sockets.join(req, 'collect');
  return res.ok();

},

collectUnsubscribe: function(req, res) {
  if (!res.isSocket) return res.badRequest();

  sails.sockets.leave(req, 'collect');
  return res.ok();
}

最后,我们需要告诉服务器向我们的'collect'房间广播消息。 请注意,这只需要一次,因此您可以在config/目录下的文件中执行此操作。

对于此示例,我将其放在config/sockets.js

module.exports = {
  // ...
};


c.on('data', function(myData) {
  var eventName = 'message';
  var data = {host: myData};
  sails.sockets.broadcast('collect', eventName, data);
});

我假设c可以在这里访问;如果没有,您可以将其定义为sails.c = ...,以使其可全局访问。