eventsource
Version:
W3C compliant EventSource client for Node.js
205 lines (178 loc) • 6.48 kB
JavaScript
var http = require('http')
, https =require('https')
, util = require('util')
, events = require('events')
, eventstream = require('./eventstream');
function isPlainObject(obj) {
return Object.getPrototypeOf(obj) === Object.prototype;
}
/**
* Creates a new EventSource object
*
* @param {String} url the URL to which to connect
* @param {Object} headers headers to use
* @api public
**/
function EventSource(url, eventSourceInitDict) {
var readyState = EventSource.CONNECTING;
Object.defineProperty(this, 'readyState', {
get: function() {
return readyState;
}
});
Object.defineProperty(this, 'url', {
get: function() {
return url;
}
});
var self = this;
self.reconnectInterval = 1000;
var connectPending = false;
function onConnectionClosed() {
if (connectPending || readyState === EventSource.CLOSED) return;
connectPending = true;
readyState = EventSource.CONNECTING;
_emit('error');
// The url may have been changed by a temporary
// redirect. If that's the case, revert it now.
if (reconnectUrl) {
url = reconnectUrl;
reconnectUrl = null;
}
setTimeout(connect, self.reconnectInterval);
}
var req;
var lastEventId = '';
var reconnectUrl = null;
function connect() {
connectPending = false;
var options = require('url').parse(url);
var isSecure = options.protocol == 'https:';
options.headers = { 'Cache-Control': 'no-cache', 'Accept': 'text/event-stream' };
if (lastEventId) options.headers['last-event-id'] = lastEventId;
if (eventSourceInitDict && eventSourceInitDict.headers && isPlainObject(eventSourceInitDict.headers)) {
for (var i in eventSourceInitDict.headers) {
var header = eventSourceInitDict.headers[i];
options.headers[i] = header;
}
}
req = (isSecure ? https : http).request(options, function(res) {
// Handle HTTP redirects
if (res.statusCode == 301 || res.statusCode == 307) {
if (!res.headers.location) {
_emit('error', 'Server sent redirect response without Location header.');
return;
}
if (res.statusCode == 307) reconnectUrl = url;
url = res.headers.location;
process.nextTick(connect);
return;
}
if (res.statusCode == 204) return self.close();
if (res.statusCode == 403) {
_emit('error', 'Access denied');
return self.close();
}
readyState = EventSource.OPEN;
res.on('close', onConnectionClosed);
res.on('end', onConnectionClosed);
_emit('open');
var buf = '';
res.on('data', function (chunk) {
buf = buf + chunk;
var messages = buf.match(/(.|\r\n|\n|\r)*(\n\n|\r\r|\r\n\r\n)/);
if (!messages) return;
messages = messages[0];
buf = buf.slice(messages.length);
try {
messages = eventstream.parse(messages);
if (!messages) return;
messages.forEach(function(message) {
var data = message.data.replace(/\n$/, '');
if (data == '' || (message.event != null && message.event == '')) {
return;
}
if (message.id) lastEventId = message.id;
_emit(message.event || 'message', new MessageEvent(data));
});
} catch(e) {
_emit('error', e);
}
});
});
req.on('error', onConnectionClosed);
req.setNoDelay(true);
req.end();
}
connect();
function _emit() {
if(self.listeners(arguments[0]).length > 0) {
self.emit.apply(self, arguments);
}
}
this.close = function() {
if (readyState == EventSource.CLOSED) return;
readyState = EventSource.CLOSED;
req.abort();
}
}
module.exports = EventSource;
util.inherits(EventSource, events.EventEmitter);
EventSource.prototype.constructor = EventSource; // make stacktraces readable
['open', 'error', 'message'].forEach(function(method) {
Object.defineProperty(EventSource.prototype, 'on' + method, {
/**
* Returns the current listener
*
* @return {Mixed} the set function or undefined
* @api private
*/
get: function get() {
var listener = this.listeners(method)[0];
return listener ? (listener._listener ? listener._listener : listener) : undefined;
},
/**
* Start listening for events
*
* @param {Function} listener the listener
* @return {Mixed} the set function or undefined
* @api private
*/
set: function set(listener) {
this.removeAllListeners(method);
this.addEventListener(method, listener);
}
});
});
/**
* Ready states
*/
Object.defineProperty(EventSource, 'CONNECTING', { enumerable: true, value: 0});
Object.defineProperty(EventSource, 'OPEN', { enumerable: true, value: 1});
Object.defineProperty(EventSource, 'CLOSED', { enumerable: true, value: 2});
/**
* Emulates the W3C Browser based WebSocket interface using addEventListener.
*
* @param {String} method Listen for an event
* @param {Function} event callback
* @see https://developer.mozilla.org/en/DOM/element.addEventListener
* @see http://dev.w3.org/html5/websockets/#the-websocket-interface
* @api public
*/
EventSource.prototype.addEventListener = function addEventListener(method, listener) {
if (typeof listener === 'function') {
// store a reference so we can return the original function again
listener._listener = listener;
this.on(method, listener);
}
};
/**
* W3C MessageEvent
*
* @see http://www.w3.org/TR/html5/comms.html
* @api private
*/
function MessageEvent(dataArg) {
// Currently only the data attribute is implemented. More can be added later if needed.
Object.defineProperty(this, 'data', { writable: false, value: dataArg });
}