我使用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端点。
答案 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 = ...
,以使其可全局访问。