[Pkg-javascript-commits] [node-eventsource] 01/02: Imported Upstream version 0.1.6

Thorsten Alteholz alteholz at moszumanska.debian.org
Mon Feb 15 20:54:10 UTC 2016


This is an automated email from the git hooks/post-receive script.

alteholz pushed a commit to branch master
in repository node-eventsource.

commit c23567825afda82f58b93f84d84a4965af6fc9a3
Author: Thorsten Alteholz <debian at alteholz.de>
Date:   Mon Feb 15 21:53:54 2016 +0100

    Imported Upstream version 0.1.6
---
 .gitignore               |   3 +
 .travis.yml              |   6 +
 CONTRIBUTING.md          |  11 +
 History.md               |  78 +++++
 LICENSE                  |  22 ++
 README.md                |  71 ++++
 example.js               |   9 +
 lib/eventsource.js       | 312 +++++++++++++++++
 package.json             |  43 +++
 test/certificate.pem     |  13 +
 test/eventsource_test.js | 875 +++++++++++++++++++++++++++++++++++++++++++++++
 test/key.pem             |  15 +
 12 files changed, 1458 insertions(+)

diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..1b7e22c
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,3 @@
+/node_modules/
+npm-debug.log
+.DS_Store
diff --git a/.travis.yml b/.travis.yml
new file mode 100644
index 0000000..3a3d23b
--- /dev/null
+++ b/.travis.yml
@@ -0,0 +1,6 @@
+language: node_js
+
+node_js:
+  - 0.8.28
+  - 0.10.36
+  - 0.12.0
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
new file mode 100644
index 0000000..01e11e8
--- /dev/null
+++ b/CONTRIBUTING.md
@@ -0,0 +1,11 @@
+# Contributing to EventSource
+
+If you add or fix something, add tests.
+
+## Release process
+
+Update `History.md`, Then:
+
+    npm outdated --depth 0 # See if you can upgrade something
+    npm version [major|minor|patch]
+    npm publish
diff --git a/History.md b/History.md
new file mode 100644
index 0000000..1a80588
--- /dev/null
+++ b/History.md
@@ -0,0 +1,78 @@
+# [0.1.6](https://github.com/aslakhellesoy/eventsource-node/compare/v0.1.5...v0.1.6)
+
+* Ignore headers without a value. ([#41](https://github.com/aslakhellesoy/eventsource-node/issues/41), [#43](https://github.com/aslakhellesoy/eventsource-node/pull/43) Adriano Raiano)
+
+# [0.1.5](https://github.com/aslakhellesoy/eventsource-node/compare/v0.1.4...v0.1.5)
+
+* Refactor tests to support Node.js 0.12.0 and Io.js 1.1.0. (Aslak Hellesøy)
+
+# [0.1.4](https://github.com/aslakhellesoy/eventsource-node/compare/v0.1.3...master)
+
+* Bugfix: Added missing origin property. ([#39](https://github.com/aslakhellesoy/eventsource-node/pull/39), [#38](https://github.com/aslakhellesoy/eventsource-node/issues/38) Arnout Kazemier)
+* Expose `status` property on `error` events. ([#40](https://github.com/aslakhellesoy/eventsource-node/pull/40) Adriano Raiano)
+
+# [0.1.3](https://github.com/aslakhellesoy/eventsource-node/compare/v0.1.2...v0.1.3)
+
+* Bugfix: Made message properties enumerable. ([#37](https://github.com/aslakhellesoy/eventsource-node/pull/37) Golo Roden)
+
+# [0.1.2](https://github.com/aslakhellesoy/eventsource-node/compare/v0.1.1...v0.1.2)
+
+* Bugfix: Blank lines not read. ([#35](https://github.com/aslakhellesoy/eventsource-node/issues/35), [#36](https://github.com/aslakhellesoy/eventsource-node/pull/36) Lesterpig)
+
+# [0.1.1](https://github.com/aslakhellesoy/eventsource-node/compare/v0.1.0...v0.1.1)
+
+* Bugfix: Fix message type. ([#33](https://github.com/aslakhellesoy/eventsource-node/pull/33) Romain Gauthier)
+
+# [0.1.0](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.10...v0.1.0)
+
+* Bugfix: High CPU usage by replacing Jison with port of WebKit's parser. ([#25](https://github.com/aslakhellesoy/eventsource-node/issues/25), [#32](https://github.com/aslakhellesoy/eventsource-node/pull/32), [#18](https://github.com/aslakhellesoy/eventsource-node/issues/18) qqueue)
+* Reformatted all code to 2 spaces.
+
+# [0.0.10](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.9...v0.0.10)
+
+* Provide `Event` argument on `open` and `error` event ([#30](https://github.com/aslakhellesoy/eventsource-node/issues/30), [#31](https://github.com/aslakhellesoy/eventsource-node/pull/31) Donghwan Kim)
+* Expose `lastEventId` on messages. ([#28](https://github.com/aslakhellesoy/eventsource-node/pull/28) mbieser)
+
+# [0.0.9](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.8...v0.0.9)
+
+* Bugfix: old "last-event-id" used on reconnect ([#27](https://github.com/aslakhellesoy/eventsource-node/pull/27) Aslak Hellesøy)
+
+# [0.0.8](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.7...v0.0.8)
+
+* Bugfix: EventSource still reconnected when closed ([#24](https://github.com/aslakhellesoy/eventsource-node/pull/24) FrozenCow)
+* Allow unauthorized HTTPS connections by setting `rejectUnauthorized` to false. (Aslak Hellesøy)
+
+# [0.0.7](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.6...v0.0.7)
+
+* Explicitly raise an error when server returns http 403 and don't continue ([#20](https://github.com/aslakhellesoy/eventsource-node/pull/20) Scott Moak)
+* Added ability to send custom http headers to server ([#21](https://github.com/aslakhellesoy/eventsource-node/pull/21), [#9](https://github.com/aslakhellesoy/eventsource-node/issues/9) Scott Moak)
+* Fix Unicode support to cope with Javascript Unicode size limitations ([#23](https://github.com/aslakhellesoy/eventsource-node/pull/23), [#22](https://github.com/aslakhellesoy/eventsource-node/issues/22) Devon Adkisson)
+* Graceful handling of parse errors ([#19](https://github.com/aslakhellesoy/eventsource-node/issues/19) Aslak Hellesøy)
+* Switched from testing with Nodeunit to Mocha (Aslak Hellesøy)
+
+# [0.0.6](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.5...v0.0.6)
+
+* Add Accept: text/event-stream header ([#17](https://github.com/aslakhellesoy/eventsource-node/pull/17) William Wicks)
+
+# [0.0.5](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.4...v0.0.5)
+
+* Add no-cache and https support ([#10](https://github.com/aslakhellesoy/eventsource-node/pull/10) Einar Otto Stangvik)
+* Ensure that Last-Event-ID is sent to the server for reconnects, as defined in the spec ([#8](https://github.com/aslakhellesoy/eventsource-node/pull/8) Einar Otto Stangvik)
+* Verify that CR and CRLF are accepted alongside LF ([#7](https://github.com/aslakhellesoy/eventsource-node/pull/7) Einar Otto Stangvik)
+* Emit 'open' event ([#4](https://github.com/aslakhellesoy/eventsource-node/issues/4) Einar Otto Stangvik)
+
+# [0.0.4](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.3...v0.0.4)
+
+* Automatic reconnect every second if the server is down. Reconnect interval can be set with `reconnectInterval` (not in W3C spec). (Aslak Hellesøy)
+
+# [0.0.3](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.2...v0.0.3)
+
+* Jison based eventstream parser ([#2](https://github.com/aslakhellesoy/eventsource-node/pull/2) Einar Otto Stangvik)
+
+# [0.0.2](https://github.com/aslakhellesoy/eventsource-node/compare/v0.0.1...v0.0.2)
+
+* Use native EventListener (Aslak Hellesøy)
+
+# 0.0.1
+
+* First release
diff --git a/LICENSE b/LICENSE
new file mode 100644
index 0000000..86329b9
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,22 @@
+The MIT License
+
+Copyright (c) 2012, 2013, 2014 Aslak Hellesøy
+
+Permission is hereby granted, free of charge, to any person obtaining
+a copy of this software and associated documentation files (the
+"Software"), to deal in the Software without restriction, including
+without limitation the rights to use, copy, modify, merge, publish,
+distribute, sublicense, and/or sell copies of the Software, and to
+permit persons to whom the Software is furnished to do so, subject to
+the following conditions:
+
+The above copyright notice and this permission notice shall be
+included in all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
diff --git a/README.md b/README.md
new file mode 100644
index 0000000..8a29769
--- /dev/null
+++ b/README.md
@@ -0,0 +1,71 @@
+# EventSource [![Build Status](https://secure.travis-ci.org/aslakhellesoy/eventsource-node.png)](http://travis-ci.org/aslakhellesoy/eventsource-node) [![Dependencies](https://david-dm.org/aslakhellesoy/eventsource-node.png)](https://david-dm.org/aslakhellesoy/eventsource-node) [![Bitdeli Badge](https://d2weczhvl823v0.cloudfront.net/aslakhellesoy/eventsource-node/trend.png)](https://bitdeli.com/free "Bitdeli Badge")
+
+
+[![NPM](https://nodei.co/npm/eventsource.png?stars&downloads)](https://nodei.co/npm/eventsource/)
+[![NPM](https://nodei.co/npm-dl/eventsource.png)](https://nodei.co/npm/eventsource/)
+
+This library implements the [EventSource](http://dev.w3.org/html5/eventsource/) client for Node.js. The API aims to be W3C compatible.
+
+## Install
+
+    npm install eventsource
+
+## Usage
+
+```javascript
+var EventSource = require('eventsource');
+
+var es = new EventSource('http://demo-eventsource.rhcloud.com/');
+es.onmessage = function(e) {
+  console.log(e.data);
+};
+es.onerror = function() {
+  console.log('ERROR!');
+};
+```
+
+See the [spec](http://dev.w3.org/html5/eventsource/) for API docs.
+
+## Example
+
+See https://github.com/einaros/sse-example
+
+## Extensions to the W3C API
+
+### Setting HTTP request headers
+
+You can define custom HTTP headers for the initial HTTP request. This can be useful for e.g. sending cookies
+or to specify an initial `Last-Event-ID` value.
+
+HTTP headers are defined by assigning a `headers` attribute to the optional `eventSourceInitDict` argument:
+
+```javascript
+var eventSourceInitDict = {headers: {'Cookie': 'test=test'}};
+var es = new EventSource(url, eventSourceInitDict);
+```
+
+### Allow unauthorized HTTPS requests
+
+By default, https requests that cannot be authorized will cause connection to fail and an exception
+to be emitted. You can override this behaviour:
+
+```javascript
+var eventSourceInitDict = {rejectUnauthorized: false};
+var es = new EventSource(url, eventSourceInitDict);
+```
+
+Note that for Node.js < v0.10.x this option has no effect - unauthorized HTTPS requests are *always* allowed.
+
+### HTTP status code on error events
+
+Unauthorized and redirect error status codes (for example 401, 403, 301, 307) are available in the `status` property in the error event.
+
+```javascript
+es.onerror = function (err) {
+  if (err) {
+    if (err.status === 401 || err.status === 403) {
+      console.log('not authorized');
+    }
+  }
+};
+```
diff --git a/example.js b/example.js
new file mode 100644
index 0000000..f6d3237
--- /dev/null
+++ b/example.js
@@ -0,0 +1,9 @@
+var EventSource = require('./lib/eventsource');
+
+var es = new EventSource('http://demo-eventsource.rhcloud.com/');
+es.onmessage = function(e) {
+  console.log(e.data);
+};
+es.onerror = function() {
+  console.log('ERROR!');
+};
diff --git a/lib/eventsource.js b/lib/eventsource.js
new file mode 100644
index 0000000..e71455c
--- /dev/null
+++ b/lib/eventsource.js
@@ -0,0 +1,312 @@
+var original = require('original')
+  , parse = require('url').parse
+  , events = require('events')
+  , https = require('https')
+  , http = require('http')
+  , util = require('util');
+
+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} eventSourceInitDict extra init params. See README for details.
+ * @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', new Event('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(function () {
+      if (readyState !== EventSource.CONNECTING) {
+        return;
+      }
+      connect();
+    }, self.reconnectInterval);
+  }
+
+  var req;
+  var lastEventId = '';
+  if (eventSourceInitDict && eventSourceInitDict.headers && isPlainObject(eventSourceInitDict.headers) && eventSourceInitDict.headers['Last-Event-ID']) {
+    lastEventId = eventSourceInitDict.headers['Last-Event-ID'];
+    delete eventSourceInitDict.headers['Last-Event-ID'];
+  }
+
+  var discardTrailingNewline = false
+    , data = ''
+    , eventName = '';
+
+  var reconnectUrl = null;
+
+  function connect() {
+    connectPending = false;
+
+    var options = 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];
+        if (header) {
+          options.headers[i] = header;
+        }
+      }
+    }
+
+    options.rejectUnauthorized = !(eventSourceInitDict && eventSourceInitDict.rejectUnauthorized == false);
+
+    req = (isSecure ? https : http).request(options, function (res) {
+      // Handle HTTP redirects
+      if (res.statusCode == 301 || res.statusCode == 307) {
+        if (!res.headers.location) {
+          // Server sent redirect response without Location header.
+          _emit('error', new Event('error', {status: res.statusCode}));
+          return;
+        }
+        if (res.statusCode == 307) reconnectUrl = url;
+        url = res.headers.location;
+        process.nextTick(connect);
+        return;
+      }
+
+      if (res.statusCode !== 200) {
+        _emit('error', new Event('error', {status: res.statusCode}));
+        return self.close();
+      }
+
+      readyState = EventSource.OPEN;
+      res.on('close', onConnectionClosed);
+      res.on('end', onConnectionClosed);
+      _emit('open', new Event('open'));
+
+      // text/event-stream parser adapted from webkit's
+      // Source/WebCore/page/EventSource.cpp
+      var buf = '';
+      res.on('data', function (chunk) {
+        buf += chunk;
+
+        var pos = 0
+          , length = buf.length;
+        while (pos < length) {
+          if (discardTrailingNewline) {
+            if (buf[pos] === '\n') {
+              ++pos;
+            }
+            discardTrailingNewline = false;
+          }
+
+          var lineLength = -1
+            , fieldLength = -1
+            , c;
+
+          for (var i = pos; lineLength < 0 && i < length; ++i) {
+            c = buf[i];
+            if (c === ':') {
+              if (fieldLength < 0) {
+                fieldLength = i - pos;
+              }
+            } else if (c === '\r') {
+              discardTrailingNewline = true;
+              lineLength = i - pos;
+            } else if (c === '\n') {
+              lineLength = i - pos;
+            }
+          }
+
+          if (lineLength < 0) {
+            break;
+          }
+
+          parseEventStreamLine(buf, pos, fieldLength, lineLength);
+
+          pos += lineLength + 1;
+        }
+
+        if (pos === length) {
+          buf = '';
+        } else if (pos > 0) {
+          buf = buf.slice(pos);
+        }
+      });
+    });
+
+    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();
+  };
+
+  function parseEventStreamLine(buf, pos, fieldLength, lineLength) {
+    if (lineLength === 0) {
+      if (data.length > 0) {
+        var type = eventName || 'message';
+        _emit(type, new MessageEvent(type, {
+          data: data.slice(0, -1), // remove trailing newline
+          lastEventId: lastEventId,
+          origin: original(url)
+        }));
+        data = '';
+      }
+      eventName = void 0;
+    } else if (fieldLength > 0) {
+      var noValue = fieldLength < 0
+        , step = 0
+        , field = buf.slice(pos, pos + (noValue ? lineLength : fieldLength));
+
+      if (noValue) {
+        step = lineLength;
+      } else if (buf[pos + fieldLength + 1] !== ' ') {
+        step = fieldLength + 1;
+      } else {
+        step = fieldLength + 2;
+      }
+      pos += step;
+      var valueLength = lineLength - step
+        , value = buf.slice(pos, pos + valueLength);
+
+      if (field === 'data') {
+        data += value + '\n';
+      } else if (field === 'event') {
+        eventName = value;
+      } else if (field === 'id') {
+        lastEventId = value;
+      } else if (field === 'retry') {
+        var retry = parseInt(value, 10);
+        if (!Number.isNaN(retry)) {
+          self.reconnectInterval = retry;
+        }
+      }
+    }
+  }
+}
+
+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} listener 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 Event
+ *
+ * @see http://www.w3.org/TR/DOM-Level-3-Events/#interface-Event
+ * @api private
+ */
+function Event(type, optionalProperties) {
+  Object.defineProperty(this, 'type', { writable: false, value: type, enumerable: true });
+  if (optionalProperties) {
+    for (var f in optionalProperties) {
+      if (optionalProperties.hasOwnProperty(f)) {
+        Object.defineProperty(this, f, { writable: false, value: optionalProperties[f], enumerable: true });
+      }
+    }
+  }
+}
+
+/**
+ * W3C MessageEvent
+ *
+ * @see http://www.w3.org/TR/webmessaging/#event-definitions
+ * @api private
+ */
+function MessageEvent(type, eventInitDict) {
+  Object.defineProperty(this, 'type', { writable: false, value: type, enumerable: true });
+  for (var f in eventInitDict) {
+    if (eventInitDict.hasOwnProperty(f)) {
+      Object.defineProperty(this, f, { writable: false, value: eventInitDict[f], enumerable: true });
+    }
+  }
+}
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..5ac1c80
--- /dev/null
+++ b/package.json
@@ -0,0 +1,43 @@
+{
+  "name": "eventsource",
+  "version": "0.1.6",
+  "description": "W3C compliant EventSource client for Node.js",
+  "keywords": [
+    "eventsource",
+    "http",
+    "streaming",
+    "sse"
+  ],
+  "homepage": "http://github.com/aslakhellesoy/eventsource-node",
+  "author": "Aslak Hellesøy <aslak.hellesoy at gmail.com>",
+  "repository": {
+    "type": "git",
+    "url": "git://github.com/aslakhellesoy/eventsource-node.git"
+  },
+  "bugs": {
+    "url": "http://github.com/aslakhellesoy/eventsource-node/issues"
+  },
+  "directories": {
+    "lib": "./lib"
+  },
+  "main": "./lib/eventsource",
+  "licenses": [
+    {
+      "type": "MIT",
+      "url": "http://github.com/aslakhellesoy/eventsource-node/raw/master/LICENSE"
+    }
+  ],
+  "devDependencies": {
+    "mocha": ">=1.21.4"
+  },
+  "scripts": {
+    "test": "mocha --reporter spec",
+    "postpublish": "git push && git push --tags"
+  },
+  "engines": {
+    "node": ">=0.8.0"
+  },
+  "dependencies": {
+    "original": ">=0.0.5"
+  }
+}
diff --git a/test/certificate.pem b/test/certificate.pem
new file mode 100644
index 0000000..0efc2ef
--- /dev/null
+++ b/test/certificate.pem
@@ -0,0 +1,13 @@
+-----BEGIN CERTIFICATE-----
+MIICATCCAWoCCQDPufXH86n2QzANBgkqhkiG9w0BAQUFADBFMQswCQYDVQQGEwJu
+bzETMBEGA1UECAwKU29tZS1TdGF0ZTEhMB8GA1UECgwYSW50ZXJuZXQgV2lkZ2l0
+cyBQdHkgTHRkMB4XDTEyMDEwMTE0NDQwMFoXDTIwMDMxOTE0NDQwMFowRTELMAkG
+A1UEBhMCbm8xEzARBgNVBAgMClNvbWUtU3RhdGUxITAfBgNVBAoMGEludGVybmV0
+IFdpZGdpdHMgUHR5IEx0ZDCBnzANBgkqhkiG9w0BAQEFAAOBjQAwgYkCgYEAtrQ7
++r//2iV/B6F+4boH0XqFn7alcV9lpjvAmwRXNKnxAoa0f97AjYPGNLKrjpkNXXhB
+JROIdbRbZnCNeC5fzX1a+JCo7KStzBXuGSZr27TtFmcV4H+9gIRIcNHtZmJLnxbJ
+sIhkGR8yVYdmJZe4eT5ldk1zoB1adgPF1hZhCBMCAwEAATANBgkqhkiG9w0BAQUF
+AAOBgQCeWBEHYJ4mCB5McwSSUox0T+/mJ4W48L/ZUE4LtRhHasU9hiW92xZkTa7E
+QLcoJKQiWfiLX2ysAro0NX4+V8iqLziMqvswnPzz5nezaOLE/9U/QvH3l8qqNkXu
+rNbsW1h/IO6FV8avWFYVFoutUwOaZ809k7iMh2F2JMgXQ5EymQ==
+-----END CERTIFICATE-----
diff --git a/test/eventsource_test.js b/test/eventsource_test.js
new file mode 100644
index 0000000..79f0709
--- /dev/null
+++ b/test/eventsource_test.js
@@ -0,0 +1,875 @@
+var EventSource = require('../lib/eventsource')
+  , http = require('http')
+  , https = require('https')
+  , fs = require('fs')
+  , assert = require('assert')
+  , u = require('url');
+
+var _port = 20000;
+var servers = 0;
+process.on('exit', function () {
+  if (servers != 0) {
+    console.error("************ Didn't kill all servers - there is still %d running.", servers);
+  }
+});
+
+function createServer(callback) {
+  var server = http.createServer();
+  configureServer(server, 'http', _port++, callback);
+}
+
+function createHttpsServer(callback) {
+  var options = {
+    key: fs.readFileSync(__dirname + '/key.pem'),
+    cert: fs.readFileSync(__dirname + '/certificate.pem')
+  };
+  var server = https.createServer(options);
+  configureServer(server, 'https', _port++, callback);
+}
+
+function configureServer(server, protocol, port, callback) {
+  var responses = [];
+
+  var oldClose = server.close;
+  server.close = function() {
+    responses.forEach(function (res) {
+      res.end();
+    });
+    servers--;
+    oldClose.apply(this, arguments);
+  };
+
+  server.on('request', function (req, res) {
+    responses.push(res);
+  });
+
+  server.url = protocol + '://localhost:' + port;
+
+  server.listen(port, function onOpen(err) {
+    servers++;
+    callback(err, server);
+  });
+}
+
+function writeEvents(chunks) {
+  return function (req, res) {
+    res.writeHead(200, {'Content-Type': 'text/event-stream'});
+    chunks.forEach(function (chunk) {
+      res.write(chunk);
+    });
+    res.write(':'); // send a dummy comment to ensure that the head is flushed
+  };
+}
+
+describe('Parser', function () {
+  it('parses multibyte characters', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["id: 1\ndata: €豆腐\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = function (m) {
+        assert.equal("€豆腐", m.data);
+        server.close(done);
+      };
+    });
+  });
+
+  it('parses empty lines with multibyte characters', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["\n\n\n\nid: 1\ndata: 我現在都看實況不玩遊戲\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = function (m) {
+        assert.equal("我現在都看實況不玩遊戲", m.data);
+        server.close(done);
+      };
+    });
+  });
+
+  it('parses one one-line message in one chunk', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["data: Hello\n\n"]));
+      var es = new EventSource(server.url);
+      es.onmessage = function (m) {
+        assert.equal("Hello", m.data);
+        server.close(done);
+      };
+    });
+  });
+
+  it('parses one one-line message in two chunks', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["data: Hel", "lo\n\n"]));
+      var es = new EventSource(server.url);
+      es.onmessage = function (m) {
+        assert.equal("Hello", m.data);
+        server.close(done);
+      };
+    });
+  });
+
+  it('parses two one-line messages in one chunk', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["data: Hello\n\n", "data: World\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = first;
+
+      function first(m) {
+        assert.equal("Hello", m.data);
+        es.onmessage = second;
+      }
+
+      function second(m) {
+        assert.equal("World", m.data);
+        server.close(done);
+      }
+    });
+  });
+
+  it('parses one two-line message in one chunk', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["data: Hello\ndata:World\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = function (m) {
+        assert.equal("Hello\nWorld", m.data);
+        server.close(done);
+      };
+    });
+  });
+
+  it('parses really chopped up unicode data', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      var chopped = "data: Aslak\n\ndata: Hellesøy\n\n".split("");
+      server.on('request', writeEvents(chopped));
+      var es = new EventSource(server.url);
+
+      es.onmessage = first;
+
+      function first(m) {
+        assert.equal("Aslak", m.data);
+        es.onmessage = second;
+      }
+
+      function second(m) {
+        assert.equal("Hellesøy", m.data);
+        server.close(done);
+      }
+    });
+  });
+
+  it('accepts CRLF as separator', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      var chopped = "data: Aslak\r\n\r\ndata: Hellesøy\r\n\r\n".split("");
+      server.on('request', writeEvents(chopped));
+      var es = new EventSource(server.url);
+
+      es.onmessage = first;
+
+      function first(m) {
+        assert.equal("Aslak", m.data);
+        es.onmessage = second;
+      }
+
+      function second(m) {
+        assert.equal("Hellesøy", m.data);
+        server.close(done);
+      }
+    });
+  });
+
+  it('accepts CR as separator', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      var chopped = "data: Aslak\r\rdata: Hellesøy\r\r".split("");
+      server.on('request', writeEvents(chopped));
+      var es = new EventSource(server.url);
+
+      es.onmessage = first;
+
+      function first(m) {
+        assert.equal("Aslak", m.data);
+        es.onmessage = second;
+      }
+
+      function second(m) {
+        assert.equal("Hellesøy", m.data);
+        server.close(done);
+      }
+    });
+  });
+
+  it('delivers message with explicit event', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["event: greeting\ndata: Hello\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.addEventListener('greeting', function (m) {
+        assert.equal("Hello", m.data);
+        server.close(done);
+      });
+    });
+  });
+
+  it('ignores comments', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["data: Hello\n\n:nothing to see here\n\ndata: World\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = first;
+
+      function first(m) {
+        assert.equal("Hello", m.data);
+        es.onmessage = second;
+      }
+
+      function second(m) {
+        assert.equal("World", m.data);
+        server.close(done);
+      }
+    });
+  });
+
+  it('ignores empty comments', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["data: Hello\n\n:\n\ndata: World\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = first;
+
+      function first(m) {
+        assert.equal("Hello", m.data);
+        es.onmessage = second;
+      }
+
+      function second(m) {
+        assert.equal("World", m.data);
+        server.close(done);
+      }
+    });
+  });
+
+  it('does not ignore multilines strings', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["data: line one\ndata:\ndata: line two\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = function (m) {
+        assert.equal('line one\n\nline two', m.data);
+        server.close(done);
+      };
+    });
+  });
+
+  it('does not ignore multilines strings even in data beginning', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["data:\ndata:line one\ndata: line two\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = function (m) {
+        assert.equal('\nline one\nline two', m.data);
+        server.close(done);
+      };
+    });
+  });
+
+  it('causes entire event to be ignored for empty event field', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', writeEvents(["event:\n\ndata: Hello\n\n"]));
+      var es = new EventSource(server.url);
+
+      var originalEmit = es.emit;
+      es.emit = function (event) {
+        assert.ok(event === 'message' || event === 'newListener');
+        return originalEmit.apply(this, arguments);
+      };
+      es.onmessage = function (m) {
+        assert.equal('Hello', m.data);
+        server.close(done);
+      };
+    });
+  });
+
+  it('parses relatively huge messages efficiently', function (done) {
+    this.timeout(1000);
+
+    createServer(function (err, server) {
+      if (err) return done(err);
+      var longMessage = "data: " + new Array(100000).join('a') + "\n\n";
+      server.on('request', writeEvents([longMessage]));
+
+      var es = new EventSource(server.url);
+
+      es.onmessage = function () {
+        server.close(done);
+      };
+    });
+  });
+});
+
+describe('HTTP Request', function () {
+  it('passes cache-control: no-cache to server', function (done) {
+    createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', function (req) {
+        assert.equal('no-cache', req.headers['cache-control']);
+        server.close(done);
+      });
+      new EventSource(server.url);
+    });
+  });
+
+  it('sets request headers', function (done) {
+    var server = createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', function (req) {
+        assert.equal(req.headers['user-agent'], 'test');
+        assert.equal(req.headers['cookie'], 'test=test');
+        assert.equal(req.headers['last-event-id'], '99');
+        server.close(done);
+      });
+
+      var headers = {
+        'User-Agent': 'test',
+        'Cookie': 'test=test',
+        'Last-Event-ID': '99'
+      };
+      new EventSource(server.url, {headers: headers});
+    });
+  });
+
+  it("does not set request headers that don't have a value", function (done) {
+    var server = createServer(function (err, server) {
+      if (err) return done(err);
+
+      server.on('request', function (req) {
+        assert.equal(req.headers['user-agent'], 'test');
+        assert.equal(req.headers['cookie'], 'test=test');
+        assert.equal(req.headers['last-event-id'], '99');
+        assert.equal(req.headers['X-Something'], undefined);
+        server.close(done);
+      });
+
+      var headers = {
+        'User-Agent': 'test',
+        'Cookie': 'test=test',
+        'Last-Event-ID': '99',
+        'X-Something': null
+      };
+
+      assert.doesNotThrow(
+        function() {
+          new EventSource(server.url, {headers: headers});
+        }
+      );
+    });
+  });
+
+  [301, 307].forEach(function (status) {
+    it('follows http ' + status + ' redirect', function (done) {
+      var redirectSuffix = '/foobar';
+      var clientRequestedRedirectUrl = false;
+      createServer(function (err, server) {
+        if(err) return done(err);
+
+        server.on('request', function (req, res) {
+          if (req.url === '/') {
+            res.writeHead(status, {
+              'Connection': 'Close',
+              'Location': server.url + redirectSuffix
+            });
+            res.end();
+          } else if (req.url === redirectSuffix) {
+            clientRequestedRedirectUrl = true;
+            res.writeHead(200, {'Content-Type': 'text/event-stream'});
+            res.end();
+          }
+        });
+
+        var es = new EventSource(server.url);
+        es.onopen = function () {
+          assert.ok(clientRequestedRedirectUrl);
+          assert.equal(server.url + redirectSuffix, es.url);
+          server.close(done);
+        };
+      });
+    });
+
+
+    it('causes error event when response is ' + status + ' with missing location', function (done) {
+      var redirectSuffix = '/foobar';
+      var clientRequestedRedirectUrl = false;
+      createServer(function (err, server) {
+        if(err) return done(err);
+
+        server.on('request', function (req, res) {
+          res.writeHead(status, {
+            'Connection': 'Close'
+          });
+          res.end();
+        });
+
+        var es = new EventSource(server.url);
+        es.onerror = function (err) {
+          assert.equal(err.status, status);
+          server.close(done);
+        };
+      });
+    });
+  });
+
+  [401, 403].forEach(function (status) {
+    it('causes error event when response status is ' + status, function (done) {
+      createServer(function (err, server) {
+        if(err) return done(err);
+
+        server.on('request', function (req, res) {
+          res.writeHead(status, {'Content-Type': 'text/html'});
+          res.end();
+        });
+
+        var es = new EventSource(server.url);
+        es.onerror = function (err) {
+          assert.equal(err.status, status);
+          server.close(done);
+        };
+      });
+    });
+  });
+});
+
+describe('HTTPS Support', function () {
+  it('uses https for https urls', function (done) {
+    createHttpsServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents(["data: hello\n\n"]));
+      var es = new EventSource(server.url, {rejectUnauthorized: false});
+
+      es.onmessage = function (m) {
+        assert.equal("hello", m.data);
+        server.close(done);
+      }
+    });
+  });
+});
+
+describe('Reconnection', function () {
+  it('is attempted when server is down', function (done) {
+    var es = new EventSource('http://localhost:' + _port);
+    es.reconnectInterval = 0;
+
+    es.onerror = function () {
+      es.onerror = null;
+      createServer(function (err, server) {
+        if(err) return done(err);
+
+        server.on('request', writeEvents(["data: hello\n\n"]));
+
+        es.onmessage = function (m) {
+          assert.equal("hello", m.data);
+          server.close(done);
+        }
+      });
+    };
+  });
+
+  it('is attempted when server goes down after connection', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents(["data: hello\n\n"]));
+      var es = new EventSource(server.url);
+      es.reconnectInterval = 0;
+
+      es.onmessage = function (m) {
+        assert.equal("hello", m.data);
+        server.close(function (err) {
+          if(err) return done(err);
+
+          var port = u.parse(es.url).port;
+          configureServer(http.createServer(), 'http', port, function (err, server2) {
+            if(err) return done(err);
+
+            server2.on('request', writeEvents(["data: world\n\n"]));
+            es.onmessage = function (m) {
+              assert.equal("world", m.data);
+              server2.close(done);
+            };
+          });
+        });
+      };
+    });
+  });
+
+  it('is stopped when server goes down and eventsource is being closed', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents(["data: hello\n\n"]));
+      var es = new EventSource(server.url);
+      es.reconnectInterval = 0;
+
+      es.onmessage = function (m) {
+        assert.equal("hello", m.data);
+        server.close(function (err) {
+          if(err) return done(err);
+          // The server has closed down. es.onerror should now get called,
+          // because es's remote connection was dropped.
+        });
+      };
+
+      es.onerror = function () {
+        // We received an error because the remote connection was closed.
+        // We close es, so we do not want es to reconnect.
+        es.close();
+
+        var port = u.parse(es.url).port;
+        configureServer(http.createServer(), 'http', port, function (err, server2) {
+          if(err) return done(err);
+          server2.on('request', writeEvents(["data: world\n\n"]));
+
+          es.onmessage = function (m) {
+            return done(new Error("Unexpected message: " + m.data));
+          };
+
+          setTimeout(function () {
+            // We have not received any message within 100ms, we can
+            // presume this works correctly.
+            server2.close(done);
+          }, 100);
+        });
+      };
+    });
+  });
+
+  it('is not attempted when server responds with HTTP 204', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', function (req, res) {
+        res.writeHead(204);
+        res.end();
+      });
+
+      var es = new EventSource(server.url);
+      es.reconnectInterval = 0;
+
+      es.onerror = function (e) {
+        assert.equal(e.status, 204);
+        server.close(function (err) {
+          if(err) return done(err);
+
+          var port = u.parse(es.url).port;
+          configureServer(http.createServer(), 'http', port, function (err, server2) {
+            if(err) return done(err);
+
+            // this will be verified by the readyState
+            // going from CONNECTING to CLOSED,
+            // along with the tests verifying that the
+            // state is CONNECTING when a server closes.
+            // it's next to impossible to write a fail-safe
+            // test for this, though.
+            var ival = setInterval(function () {
+              if (es.readyState == EventSource.CLOSED) {
+                clearInterval(ival);
+                server2.close(done);
+              }
+            }, 5);
+          });
+        });
+      };
+    });
+  });
+
+  it('sends Last-Event-ID http header when it has previously been passed in an event from the server', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents(['id: 10\ndata: Hello\n\n']));
+
+      var es = new EventSource(server.url);
+      es.reconnectInterval = 0;
+
+      es.onmessage = function () {
+        server.close(function (err) {
+          if(err) return done(err);
+
+          var port = u.parse(es.url).port;
+          configureServer(http.createServer(), 'http', port, function (err, server2) {
+            server2.on('request', function (req, res) {
+              assert.equal('10', req.headers['last-event-id']);
+              server2.close(done);
+            });
+          });
+        });
+      };
+    });
+  });
+
+  it('sends correct Last-Event-ID http header when an initial Last-Event-ID header was specified in the constructor', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', function (req, res) {
+        assert.equal('9', req.headers['last-event-id']);
+        server.close(done);
+      });
+
+      new EventSource(server.url, {headers: {'Last-Event-ID': '9'}});
+    });
+  });
+
+  it('does not send Last-Event-ID http header when it has not been previously sent by the server', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents(['data: Hello\n\n']));
+
+      var es = new EventSource(server.url);
+      es.reconnectInterval = 0;
+
+      es.onmessage = function () {
+        server.close(function (err) {
+          if(err) return done(err);
+
+          var port = u.parse(es.url).port;
+          configureServer(http.createServer(), 'http', port, function (err, server2) {
+            server2.on('request', function (req, res) {
+              assert.equal(undefined, req.headers['last-event-id']);
+              server2.close(done);
+            });
+          });
+        });
+      };
+    });
+  });
+});
+
+describe('readyState', function () {
+  it('has CONNECTING constant', function () {
+    assert.equal(0, EventSource.CONNECTING);
+  });
+
+  it('has OPEN constant', function () {
+    assert.equal(1, EventSource.OPEN);
+  });
+
+  it('has CLOSED constant', function () {
+    assert.equal(2, EventSource.CLOSED);
+  });
+
+  it('is CONNECTING before connection has been established', function (done) {
+    var es = new EventSource('http://localhost:' + _port);
+    assert.equal(EventSource.CONNECTING, es.readyState);
+    es.onerror = function () {
+      es.close();
+      done();
+    }
+  });
+
+  it('is CONNECTING when server has closed the connection', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents([]));
+      var es = new EventSource(server.url);
+      es.reconnectInterval = 0;
+
+      es.onopen = function (m) {
+        server.close(function (err) {
+          if(err) return done(err);
+
+          es.onerror = function () {
+            es.onerror = null;
+            assert.equal(EventSource.CONNECTING, es.readyState);
+            done();
+          };
+        });
+      };
+    });
+  });
+
+  it('is OPEN when connection has been established', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents([]));
+      var es = new EventSource(server.url);
+
+      es.onopen = function () {
+        assert.equal(EventSource.OPEN, es.readyState);
+        server.close(done);
+      }
+    });
+  });
+
+  it('is CLOSED after connection has been closed', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents([]));
+      var es = new EventSource(server.url);
+
+      es.onopen = function () {
+        es.close();
+        assert.equal(EventSource.CLOSED, es.readyState);
+        server.close(done);
+      }
+    });
+  });
+});
+
+describe('Properties', function () {
+  it('url exposes original request url', function () {
+    var url = 'http://localhost:' + _port;
+    var es = new EventSource(url);
+    assert.equal(url, es.url);
+  });
+});
+
+describe('Events', function () {
+  it('calls onopen when connection is established', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents([]));
+      var es = new EventSource(server.url);
+
+      es.onopen = function (event) {
+        assert.equal(event.type, 'open');
+        server.close(done);
+      }
+    });
+  });
+
+  it('supplies the correct origin', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents(["data: hello\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = function (event) {
+        assert.equal(event.origin, server.url);
+        server.close(done);
+      }
+    });
+  });
+
+  it('emits open event when connection is established', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents([]));
+      var es = new EventSource(server.url);
+
+      es.addEventListener('open', function (event) {
+        assert.equal(event.type, 'open');
+        server.close(done);
+      });
+    });
+  });
+
+  it('does not emit error when connection is closed by client', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents([]));
+      var es = new EventSource(server.url);
+
+      es.addEventListener('open', function () {
+        es.close();
+        process.nextTick(function () {
+          server.close(done);
+        });
+      });
+      es.addEventListener('error', function () {
+        done(new Error('error should not be emitted'));
+      });
+    });
+  });
+
+  it('populates message\'s lastEventId correctly when the last event has an associated id', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents(["id: 123\ndata: hello\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = function (m) {
+        assert.equal(m.lastEventId, "123");
+        server.close(done);
+      };
+    });
+  });
+
+  it('populates message\'s lastEventId correctly when the last event doesn\'t have an associated id', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents(["id: 123\ndata: Hello\n\n", "data: World\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = first;
+
+      function first() {
+        es.onmessage = second;
+      }
+
+      function second(m) {
+        assert.equal(m.data, "World");
+        assert.equal(m.lastEventId, "123");  //expect to get back the previous event id
+        server.close(done);
+      }
+    });
+  });
+
+  it('populates messages with enumerable properties so they can be inspected via console.log().', function (done) {
+    createServer(function (err, server) {
+      if(err) return done(err);
+
+      server.on('request', writeEvents(["data: World\n\n"]));
+      var es = new EventSource(server.url);
+
+      es.onmessage = function (m) {
+        var enumerableAttributes = Object.keys(m);
+        assert.notEqual(enumerableAttributes.indexOf("data"), -1);
+        assert.notEqual(enumerableAttributes.indexOf("type"), -1);
+        server.close(done);
+      };
+    });
+  });
+});
diff --git a/test/key.pem b/test/key.pem
new file mode 100644
index 0000000..176fe32
--- /dev/null
+++ b/test/key.pem
@@ -0,0 +1,15 @@
+-----BEGIN RSA PRIVATE KEY-----
+MIICXAIBAAKBgQC2tDv6v//aJX8HoX7hugfReoWftqVxX2WmO8CbBFc0qfEChrR/
+3sCNg8Y0squOmQ1deEElE4h1tFtmcI14Ll/NfVr4kKjspK3MFe4ZJmvbtO0WZxXg
+f72AhEhw0e1mYkufFsmwiGQZHzJVh2Yll7h5PmV2TXOgHVp2A8XWFmEIEwIDAQAB
+AoGAAlVY8sHi/aE+9xT77twWX3mGHV0SzdjfDnly40fx6S1Gc7bOtVdd9DC7pk6l
+3ENeJVR02IlgU8iC5lMHq4JEHPE272jtPrLlrpWLTGmHEqoVFv9AITPqUDLhB9Kk
+Hjl7h8NYBKbr2JHKICr3DIPKOT+RnXVb1PD4EORbJ3ooYmkCQQDfknUnVxPgxUGs
+ouABw1WJIOVgcCY/IFt4Ihf6VWTsxBgzTJKxn3HtgvE0oqTH7V480XoH0QxHhjLq
+DrgobWU9AkEA0TRJ8/ouXGnFEPAXjWr9GdPQRZ1Use2MrFjneH2+Sxc0CmYtwwqL
+Kr5kS6mqJrxprJeluSjBd+3/ElxURrEXjwJAUvmlN1OPEhXDmRHd92mKnlkyKEeX
+OkiFCiIFKih1S5Y/sRJTQ0781nyJjtJqO7UyC3pnQu1oFEePL+UEniRztQJAMfav
+AtnpYKDSM+1jcp7uu9BemYGtzKDTTAYfoiNF42EzSJiGrWJDQn4eLgPjY0T0aAf/
+yGz3Z9ErbhMm/Ysl+QJBAL4kBxRT8gM4ByJw4sdOvSeCCANFq8fhbgm8pGWlCPb5
+JGmX3/GHFM8x2tbWMGpyZP1DLtiNEFz7eCGktWK5rqE=
+-----END RSA PRIVATE KEY-----

-- 
Alioth's /usr/local/bin/git-commit-notice on /srv/git.debian.org/git/pkg-javascript/node-eventsource.git



More information about the Pkg-javascript-commits mailing list