Потоковые данные с Node.js
Я хочу знать, можно ли передавать данные с сервера клиенту с помощью Node.js. Я хочу отправить один запрос AJAX на Node.js, а затем оставить соединение открытым и непрерывно передавать данные клиенту. Клиент будет получать этот поток и постоянно обновлять страницу.
Обновление:
Как обновление этого ответа - я не могу заставить это работать. response.write
не отправляется перед вызовом close
. Я создал примерную программу, которую я использую для этого:
Node.js
var sys = require('sys'),
http = require('http');
http.createServer(function (req, res) {
res.writeHead(200, {'Content-Type': 'text/html'});
var currentTime = new Date();
setInterval(function(){
res.write(
currentTime.getHours()
+ ':' +
currentTime.getMinutes()
+ ':' +
currentTime.getSeconds()
);
},1000);
}).listen(8000);
HTML:
<html>
<head>
<title>Testnode</title>
</head>
<body>
<!-- This fields needs to be updated -->
Server time: <span id="time"> </span>
<!-- import jQuery from google -->
<script type="text/javascript" src="http://ajax.googleapis.com/ajax/libs/jquery/1.4.2/jquery.min.js"></script>
<!-- import jQuery -->
<script type="text/javascript">
$(document).ready(function(){
// I call here node.localhost nginx ports this to port 8000
$('#time').load('http://node.localhost');
});
</script>
</body>
</html>
Используя этот метод, я ничего не получаю, пока не позвоню close()
. Возможно ли это, или я должен пойти с использованием метода длительного опроса, вместо которого я снова вызываю функцию загрузки, когда вы входите?
Ответы
Ответ 1
Это возможно. Просто используйте response.write() несколько раз.
var body = ["hello world", "early morning", "richard stallman", "chunky bacon"];
// send headers
response.writeHead(200, {
"Content-Type": "text/plain"
});
// send data in chunks
for (piece in body) {
response.write(body[piece], "ascii");
}
// close connection
response.end();
Возможно, вам придется закрыть и снова открыть соединение каждые 30 секунд или около того.
EDIT: это код, который я действительно тестировал:
var sys = require('sys'),
http = require('http');
http.createServer(function (req, res) {
res.writeHead(200, {'Content-Type': 'text/html'});
var currentTime = new Date();
sys.puts('Starting sending time');
setInterval(function(){
res.write(
currentTime.getHours()
+ ':' +
currentTime.getMinutes()
+ ':' +
currentTime.getSeconds() + "\n"
);
setTimeout(function() {
res.end();
}, 10000);
},1000);
}).listen(8090, '192.168.175.128');
Я подключился к нему с помощью Telnet, и он действительно выдал ответный ответ. Но для его использования в AJAX браузер должен поддерживать XHR.readyState = 3 (частичный ответ). Насколько мне известно, не все браузеры поддерживают это. Поэтому вам лучше использовать длительный опрос (или Websockets для Chrome/Firefox).
EDIT2. Кроме того, если вы используете nginx в качестве обратного прокси-сервера для Node, он иногда хочет собрать все куски и отправить его пользователю сразу. Вам нужно настроить его.
Ответ 2
Посмотрите на Sockets.io. Он обеспечивает потоковое HTTP/HTTPS и использует различные транспорты для этого:
- WebSocket
- Поддержка WebSocket по Flash (поддержка политики безопасности XML)
- Опрос XHR
- Многостраничная потоковая передача XHR
- Forever Iframe
- Опрос JSONP (для перекрестного домена)
А! Он работает без проблем с Node.JS. Это также пакет NPM.
https://github.com/LearnBoost/Socket.IO
https://github.com/LearnBoost/Socket.IO-node
Ответ 3
Вы также можете прервать бесконечный цикл:
app.get('/sse/events', function(req, res) {
res.header('Content-Type', 'text/event-stream');
var interval_id = setInterval(function() {
res.write("some data");
}, 50);
req.socket.on('close', function() {
clearInterval(interval_id);
});
});
Это пример выражения. Я считаю, что без expressjs будет что-то вроде.