Compare commits
27 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| e6dea40c8c | |||
| 7953bc39ff | |||
| bf39b7b44b | |||
| 85ba40ac32 | |||
| e6edb23681 | |||
| d40de878b2 | |||
| ed3907d5fd | |||
| 84b6f50f1a | |||
| 8d19ee823e | |||
| 3bee0366c5 | |||
| 9f2782da14 | |||
| c752c76712 | |||
| ad8f08f19e | |||
| d343c7d21b | |||
| 0b089ad921 | |||
| e65e837968 | |||
| cda22e3bef | |||
| 1ae37d6efe | |||
| 82c42a6ce5 | |||
| aad1519f3f | |||
| 11a9b75185 | |||
| d93c853414 | |||
| 2dff35d3e4 | |||
| 6588928445 | |||
| 151fddd206 | |||
| 0b1f16a7ee | |||
| f15b331a34 |
@@ -6,3 +6,6 @@ node_js:
|
||||
- "0.10"
|
||||
- "0.11"
|
||||
|
||||
before_install:
|
||||
- '[ "${TRAVIS_NODE_VERSION}" = "0.6" ] && npm conf set strict-ssl false || true'
|
||||
|
||||
|
||||
@@ -1,3 +1,23 @@
|
||||
### 0.3.4 / 2014-05-08
|
||||
|
||||
* Don't hold memory-leaking references to I/O buffers after they have been parsed
|
||||
|
||||
### 0.3.3 / 2014-04-24
|
||||
|
||||
* Correct the draft-76 status line reason phrase
|
||||
|
||||
### 0.3.2 / 2013-12-29
|
||||
|
||||
* Expand `maxLength` to cover sequences of continuation frames and `draft-{75,76}`
|
||||
* Decrease default maximum frame buffer size to 64MB
|
||||
* Stop parsing when the protocol enters a failure mode, to save CPU cycles
|
||||
|
||||
### 0.3.1 / 2013-12-03
|
||||
|
||||
* Add a `maxLength` option to limit allowed frame size
|
||||
* Don't pre-allocate a message buffer until the whole frame has arrived
|
||||
* Fix compatibility with Node v0.11 `HTTPParser`
|
||||
|
||||
### 0.3.0 / 2013-09-09
|
||||
|
||||
* Support client URLs with Basic Auth credentials
|
||||
|
||||
@@ -173,6 +173,8 @@ with masking enabled on outgoing frames.
|
||||
The `options` argument is optional, and is an object. It may contain the
|
||||
following fields:
|
||||
|
||||
* `maxLength` - the maximum allowed size of incoming message frames, in bytes.
|
||||
The default value is `2^26 - 1`, or 1 byte short of 64 MiB.
|
||||
* `protocols` - an array of strings representing acceptable subprotocols for
|
||||
use over the socket. The driver will negotiate one of these to use via the
|
||||
`Sec-WebSocket-Protocol` header if supported by the other peer.
|
||||
@@ -283,7 +285,7 @@ after `emit('open')` has fired.
|
||||
|
||||
(The MIT License)
|
||||
|
||||
Copyright (c) 2010-2013 James Coglan
|
||||
Copyright (c) 2010-2014 James Coglan
|
||||
|
||||
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
|
||||
|
||||
@@ -1,8 +1,10 @@
|
||||
var net = require('net'),
|
||||
websocket = require('../lib/websocket/driver');
|
||||
websocket = require('../lib/websocket/driver'),
|
||||
deflate = require('../lib/websocket/extensions/per_message_deflate');
|
||||
|
||||
var server = net.createServer(function(connection) {
|
||||
var driver = websocket.server();
|
||||
driver.addExtension('permessage-deflate', deflate.create());
|
||||
|
||||
driver.on('connect', function() {
|
||||
if (websocket.isWebSocket(driver)) driver.start();
|
||||
|
||||
@@ -6,12 +6,14 @@ var Emitter = require('events').EventEmitter,
|
||||
var Base = function(request, url, options) {
|
||||
Emitter.call(this);
|
||||
|
||||
this._request = request;
|
||||
this._options = options || {};
|
||||
this.__headers = new Headers();
|
||||
this.__queue = [];
|
||||
this.readyState = 0;
|
||||
this.url = url;
|
||||
this._request = request;
|
||||
this._options = options || {};
|
||||
this._extensions = {};
|
||||
this._maxLength = this._options.maxLength || this.MAX_LENGTH;
|
||||
this.__headers = new Headers();
|
||||
this.__queue = [];
|
||||
this.readyState = 0;
|
||||
this.url = url;
|
||||
|
||||
this.io = new streams.IO(this);
|
||||
this.messages = new streams.Messages(this);
|
||||
@@ -20,6 +22,10 @@ var Base = function(request, url, options) {
|
||||
util.inherits(Base, Emitter);
|
||||
|
||||
var instance = {
|
||||
// This is 64MB, small enough for an average VPS to handle without
|
||||
// crashing from process out of memory
|
||||
MAX_LENGTH: 0x3ffffff,
|
||||
|
||||
STATES: ['connecting', 'open', 'closing', 'closed'],
|
||||
|
||||
_bindEventListeners: function() {
|
||||
@@ -46,6 +52,10 @@ var instance = {
|
||||
});
|
||||
},
|
||||
|
||||
addExtension: function(name, extension) {
|
||||
this._extensions[name] = extension;
|
||||
},
|
||||
|
||||
getState: function() {
|
||||
return this.STATES[this.readyState] || null;
|
||||
},
|
||||
@@ -103,6 +113,9 @@ var instance = {
|
||||
for (var key in instance)
|
||||
Base.prototype[key] = instance[key];
|
||||
|
||||
Base.parseHeader = function(header) {
|
||||
return Headers.parseHeader(header);
|
||||
};
|
||||
|
||||
Base.ConnectEvent = function() {};
|
||||
|
||||
|
||||
@@ -17,7 +17,7 @@ util.inherits(Client, Hybi);
|
||||
|
||||
Client.generateKey = function() {
|
||||
var buffer = new Buffer(16), i = buffer.length;
|
||||
while (i--) buffer[i] = Math.floor(Math.random() * 256);
|
||||
while (i--) buffer[i] = ~~(Math.random() * 256);
|
||||
return buffer.toString('base64');
|
||||
};
|
||||
|
||||
|
||||
@@ -9,7 +9,16 @@ var Draft75 = function(request, url, options) {
|
||||
util.inherits(Draft75, Base);
|
||||
|
||||
var instance = {
|
||||
close: function() {
|
||||
if (this.readyState === 3) return false;
|
||||
this.readyState = 3;
|
||||
this.emit('close', new Base.CloseEvent(null, null));
|
||||
return true;
|
||||
},
|
||||
|
||||
parse: function(buffer) {
|
||||
if (this.readyState > 1) return;
|
||||
|
||||
var data, message, value;
|
||||
for (var i = 0, n = buffer.length; i < n; i++) {
|
||||
data = buffer[i];
|
||||
@@ -29,8 +38,7 @@ var instance = {
|
||||
this._length = value + 128 * this._length;
|
||||
|
||||
if (this._closing && this._length === 0) {
|
||||
this.readyState = 3;
|
||||
this.emit('close', new Base.CloseEvent(null, null));
|
||||
return this.close();
|
||||
}
|
||||
else if ((0x80 & data) !== 0x80) {
|
||||
if (this._length === 0) {
|
||||
@@ -56,6 +64,7 @@ var instance = {
|
||||
this._stage = 0;
|
||||
} else {
|
||||
this._buffer.push(data);
|
||||
if (this._buffer.length > this._maxLength) return this.close();
|
||||
}
|
||||
}
|
||||
break;
|
||||
|
||||
@@ -48,7 +48,7 @@ var instance = {
|
||||
},
|
||||
|
||||
_handshakeResponse: function() {
|
||||
return new Buffer('HTTP/1.1 101 Web Socket Protocol Handshake\r\n' +
|
||||
return new Buffer('HTTP/1.1 101 WebSocket Protocol Handshake\r\n' +
|
||||
'Upgrade: WebSocket\r\n' +
|
||||
'Connection: Upgrade\r\n' +
|
||||
'Sec-WebSocket-Origin: ' + this._request.headers.origin + '\r\n' +
|
||||
|
||||
@@ -3,6 +3,22 @@ var Headers = function() {
|
||||
this._lines = [];
|
||||
};
|
||||
|
||||
Headers.parseHeader = function(header) {
|
||||
if (!header) return [];
|
||||
|
||||
return header.split(/\s*,\s*/).map(function(value) {
|
||||
var parts = value.split(/\s*;\s*/),
|
||||
name = parts.shift(),
|
||||
params = {};
|
||||
|
||||
parts.forEach(function(part) {
|
||||
var pair = part.split(/\s*=\s*/);
|
||||
params[pair[0]] = pair[1] || true;
|
||||
});
|
||||
return {name: name, params: params};
|
||||
});
|
||||
};
|
||||
|
||||
Headers.prototype.ALLOWED_DUPLICATES = ['set-cookie', 'set-cookie2', 'warning', 'www-authenticate']
|
||||
|
||||
Headers.prototype.set = function(name, value) {
|
||||
|
||||
@@ -19,26 +19,57 @@ var HttpParser = function(type) {
|
||||
};
|
||||
|
||||
this._parser.onHeaderValue = function(b, start, length) {
|
||||
self.headers[current] = b.toString('utf8', start, start + length);
|
||||
self.set(current, b.toString('utf8', start, start + length));
|
||||
};
|
||||
|
||||
this._parser.onHeadersComplete = function(info) {
|
||||
self.method = info.method;
|
||||
this._parser.onHeadersComplete = this._parser[HTTPParser.kOnHeadersComplete] = function(info) {
|
||||
self.method = (typeof info.method === 'number') ? HttpParser.METHODS[info.method] : info.method;
|
||||
self.statusCode = info.statusCode;
|
||||
self.url = info.url;
|
||||
|
||||
var headers = info.headers;
|
||||
var headers = info.headers, name, values;
|
||||
if (!headers) return;
|
||||
|
||||
for (var i = 0, n = headers.length; i < n; i += 2)
|
||||
self.headers[headers[i].toLowerCase()] = headers[i+1];
|
||||
self.set(headers[i].toLowerCase(), headers[i+1]);
|
||||
};
|
||||
|
||||
this._parser.onMessageComplete = function() {
|
||||
this._parser.onMessageComplete = this._parser[HTTPParser.kOnMessageComplete] = function() {
|
||||
self._complete = true;
|
||||
};
|
||||
};
|
||||
|
||||
HttpParser.METHODS = {
|
||||
0: 'DELETE',
|
||||
1: 'GET',
|
||||
2: 'HEAD',
|
||||
3: 'POST',
|
||||
4: 'PUT',
|
||||
5: 'CONNECT',
|
||||
6: 'OPTIONS',
|
||||
7: 'TRACE',
|
||||
8: 'COPY',
|
||||
9: 'LOCK',
|
||||
10: 'MKCOL',
|
||||
11: 'MOVE',
|
||||
12: 'PROPFIND',
|
||||
13: 'PROPPATCH',
|
||||
14: 'SEARCH',
|
||||
15: 'UNLOCK',
|
||||
16: 'REPORT',
|
||||
17: 'MKACTIVITY',
|
||||
18: 'CHECKOUT',
|
||||
19: 'MERGE',
|
||||
24: 'PATCH'
|
||||
};
|
||||
|
||||
HttpParser.prototype.set = function(name, value) {
|
||||
var values = [];
|
||||
if (this.headers.hasOwnProperty(name)) values.push(this.headers[name]);
|
||||
values.push(value);
|
||||
this.headers[name] = values.join(', ');
|
||||
};
|
||||
|
||||
HttpParser.prototype.isComplete = function() {
|
||||
return this._complete;
|
||||
};
|
||||
|
||||
@@ -1,14 +1,15 @@
|
||||
var crypto = require('crypto'),
|
||||
util = require('util'),
|
||||
Base = require('./base'),
|
||||
Reader = require('./hybi/stream_reader');
|
||||
var crypto = require('crypto'),
|
||||
util = require('util'),
|
||||
Base = require('./base'),
|
||||
Concat = require('./hybi/concat'),
|
||||
Mask = require('./hybi/mask'),
|
||||
Reader = require('./hybi/stream_reader');
|
||||
|
||||
var Hybi = function(request, url, options) {
|
||||
Base.apply(this, arguments);
|
||||
this._reset();
|
||||
|
||||
this._reader = new Reader();
|
||||
this._stage = 0;
|
||||
this._reader = new Reader({context: this});
|
||||
this._masking = this._options.masking;
|
||||
this._protocols = this._options.protocols || [];
|
||||
|
||||
@@ -22,19 +23,11 @@ var Hybi = function(request, url, options) {
|
||||
var version = this._request.headers['sec-websocket-version'];
|
||||
this.version = 'hybi-' + version;
|
||||
}
|
||||
|
||||
this._reader.read(1, this._parseOpcode);
|
||||
};
|
||||
util.inherits(Hybi, Base);
|
||||
|
||||
Hybi.mask = function(payload, mask, offset) {
|
||||
if (mask.length === 0) return payload;
|
||||
offset = offset || 0;
|
||||
|
||||
for (var i = 0, n = payload.length - offset; i < n; i++) {
|
||||
payload[offset + i] = payload[offset + i] ^ mask[i % 4];
|
||||
}
|
||||
return payload;
|
||||
};
|
||||
|
||||
Hybi.generateAccept = function(key) {
|
||||
var sha1 = crypto.createHash('sha1');
|
||||
sha1.update(key + Hybi.GUID);
|
||||
@@ -66,7 +59,6 @@ var instance = {
|
||||
FRAGMENTED_OPCODES: [0, 1, 2],
|
||||
OPENING_OPCODES: [1, 2],
|
||||
|
||||
MAX_LENGTH: Math.pow(2, 53) - 1,
|
||||
TWO_POWERS: [0, 1, 2, 3, 4, 5, 6, 7].map(function(n) { return Math.pow(2, 8 * n) }),
|
||||
|
||||
ERRORS: {
|
||||
@@ -89,43 +81,7 @@ var instance = {
|
||||
UTF8_MATCH: /^([\x00-\x7F]|[\xC2-\xDF][\x80-\xBF]|\xE0[\xA0-\xBF][\x80-\xBF]|[\xE1-\xEC\xEE\xEF][\x80-\xBF]{2}|\xED[\x80-\x9F][\x80-\xBF]|\xF0[\x90-\xBF][\x80-\xBF]{2}|[\xF1-\xF3][\x80-\xBF]{3}|\xF4[\x80-\x8F][\x80-\xBF]{2})*$/,
|
||||
|
||||
parse: function(data) {
|
||||
this._reader.put(data);
|
||||
var buffer = true;
|
||||
while (buffer) {
|
||||
switch (this._stage) {
|
||||
case 0:
|
||||
buffer = this._reader.read(1);
|
||||
if (buffer) this._parseOpcode(buffer[0]);
|
||||
break;
|
||||
|
||||
case 1:
|
||||
buffer = this._reader.read(1);
|
||||
if (buffer) this._parseLength(buffer[0]);
|
||||
break;
|
||||
|
||||
case 2:
|
||||
buffer = this._reader.read(this._lengthSize);
|
||||
if (buffer) this._parseExtendedLength(buffer);
|
||||
break;
|
||||
|
||||
case 3:
|
||||
buffer = this._reader.read(4);
|
||||
if (buffer) {
|
||||
this._mask = buffer;
|
||||
this._stage = 4;
|
||||
}
|
||||
break;
|
||||
|
||||
case 4:
|
||||
buffer = this._reader.read(this._length);
|
||||
if (buffer) {
|
||||
this._payload = buffer;
|
||||
this._emitFrame();
|
||||
this._stage = 0;
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
this._reader.write(data);
|
||||
},
|
||||
|
||||
frame: function(data, type, code) {
|
||||
@@ -152,31 +108,31 @@ var instance = {
|
||||
frame[1] = masked | length;
|
||||
} else if (length <= 65535) {
|
||||
frame[1] = masked | 126;
|
||||
frame[2] = Math.floor(length / 256);
|
||||
frame[2] = ~~(length / 256);
|
||||
frame[3] = length & BYTE;
|
||||
} else {
|
||||
frame[1] = masked | 127;
|
||||
frame[2] = Math.floor(length / Math.pow(2,56)) & BYTE;
|
||||
frame[3] = Math.floor(length / Math.pow(2,48)) & BYTE;
|
||||
frame[4] = Math.floor(length / Math.pow(2,40)) & BYTE;
|
||||
frame[5] = Math.floor(length / Math.pow(2,32)) & BYTE;
|
||||
frame[6] = Math.floor(length / Math.pow(2,24)) & BYTE;
|
||||
frame[7] = Math.floor(length / Math.pow(2,16)) & BYTE;
|
||||
frame[8] = Math.floor(length / Math.pow(2,8)) & BYTE;
|
||||
frame[2] = ~~(length / Math.pow(2, 56)) & BYTE;
|
||||
frame[3] = ~~(length / Math.pow(2, 48)) & BYTE;
|
||||
frame[4] = ~~(length / Math.pow(2, 40)) & BYTE;
|
||||
frame[5] = ~~(length / Math.pow(2, 32)) & BYTE;
|
||||
frame[6] = ~~(length / Math.pow(2, 24)) & BYTE;
|
||||
frame[7] = ~~(length / Math.pow(2, 16)) & BYTE;
|
||||
frame[8] = ~~(length / Math.pow(2, 8)) & BYTE;
|
||||
frame[9] = length & BYTE;
|
||||
}
|
||||
|
||||
if (code) {
|
||||
frame[offset] = Math.floor(code / 256) & BYTE;
|
||||
frame[offset+1] = code & BYTE;
|
||||
frame[offset] = ~~(code / 256) & BYTE;
|
||||
frame[offset + 1] = code & BYTE;
|
||||
}
|
||||
buffer.copy(frame, offset + insert);
|
||||
|
||||
if (this._masking) {
|
||||
mask = [Math.floor(Math.random() * 256), Math.floor(Math.random() * 256),
|
||||
Math.floor(Math.random() * 256), Math.floor(Math.random() * 256)];
|
||||
mask = [~~(Math.random() * 256), ~~(Math.random() * 256),
|
||||
~~(Math.random() * 256), ~~(Math.random() * 256)];
|
||||
new Buffer(mask).copy(frame, header);
|
||||
Hybi.mask(frame, mask, offset);
|
||||
Mask.mask(frame, mask, offset);
|
||||
}
|
||||
|
||||
this._write(frame);
|
||||
@@ -219,6 +175,7 @@ var instance = {
|
||||
if (!secKey) return '';
|
||||
|
||||
var accept = Hybi.generateAccept(secKey),
|
||||
offers = Base.parseHeader(this._request.headers['sec-websocket-extensions']),
|
||||
protos = this._request.headers['sec-websocket-protocol'],
|
||||
supported = this._protocols,
|
||||
proto,
|
||||
@@ -239,12 +196,27 @@ var instance = {
|
||||
}
|
||||
}
|
||||
|
||||
// TODO: implement a selection procedure for extensions
|
||||
// see http://tools.ietf.org/html/draft-ietf-hybi-permessage-compression-17#section-5
|
||||
var extensions = offers.map(function(offer) {
|
||||
var ext = this._extensions[offer.name];
|
||||
return ext && ext.createSession(offer.params);
|
||||
}, this).filter(function(k) {
|
||||
return k;
|
||||
});
|
||||
|
||||
if (extensions.length > 0) {
|
||||
extensions = extensions.map(function(e) { return e.responseHeader() });
|
||||
headers.push('Sec-WebSocket-Extensions: ' + extensions.join(', '));
|
||||
}
|
||||
|
||||
return new Buffer(headers.concat(this.__headers.toString(), '').join('\r\n'), 'utf8');
|
||||
},
|
||||
|
||||
_shutdown: function(code, reason) {
|
||||
this.frame(reason, 'close', code);
|
||||
this.readyState = 3;
|
||||
this._reader.end();
|
||||
this.emit('close', new Base.CloseEvent(code, reason));
|
||||
},
|
||||
|
||||
@@ -253,7 +225,9 @@ var instance = {
|
||||
this._shutdown(this.ERRORS[type], message);
|
||||
},
|
||||
|
||||
_parseOpcode: function(data) {
|
||||
_parseOpcode: function(buffer) {
|
||||
var data = buffer[0];
|
||||
|
||||
var rsvs = [this.RSV1, this.RSV2, this.RSV3].map(function(rsv) {
|
||||
return (data & rsv) === rsv;
|
||||
});
|
||||
@@ -264,10 +238,9 @@ var instance = {
|
||||
', reserved2 = ' + (rsvs[1] ? 1 : 0) +
|
||||
', reserved3 = ' + (rsvs[2] ? 1 : 0));
|
||||
|
||||
this._final = (data & this.FIN) === this.FIN;
|
||||
this._opcode = (data & this.OPCODE);
|
||||
this._mask = [];
|
||||
this._payload = [];
|
||||
this._final = (data & this.FIN) === this.FIN;
|
||||
this._opcode = (data & this.OPCODE);
|
||||
this._mask = null;
|
||||
|
||||
if (this.OPCODE_CODES.indexOf(this._opcode) < 0)
|
||||
return this._fail('protocol_error', 'Unrecognized frame opcode: ' + this._opcode);
|
||||
@@ -278,10 +251,12 @@ var instance = {
|
||||
if (this._mode && this.OPENING_OPCODES.indexOf(this._opcode) >= 0)
|
||||
return this._fail('protocol_error', 'Received new data frame but previous continuous frame is unfinished');
|
||||
|
||||
this._stage = 1;
|
||||
this._reader.read(1, this._parseLength);
|
||||
},
|
||||
|
||||
_parseLength: function(data) {
|
||||
_parseLength: function(buffer) {
|
||||
var data = buffer[0];
|
||||
|
||||
this._masked = (data & this.MASK) === this.MASK;
|
||||
if (this._requireMasking && !this._masked)
|
||||
return this._fail('unacceptable', 'Received unmasked frame but masking is required');
|
||||
@@ -289,11 +264,11 @@ var instance = {
|
||||
this._length = (data & this.LENGTH);
|
||||
|
||||
if (this._length >= 0 && this._length <= 125) {
|
||||
this._stage = this._masked ? 3 : 4;
|
||||
if (!this._checkFrameLength()) return;
|
||||
this._readMask();
|
||||
} else {
|
||||
this._lengthBuffer = [];
|
||||
this._lengthSize = (this._length === 126 ? 2 : 8);
|
||||
this._stage = 2;
|
||||
var lengthSize = (this._length === 126 ? 2 : 8);
|
||||
this._reader.read(lengthSize, this._parseExtendedLength);
|
||||
}
|
||||
},
|
||||
|
||||
@@ -303,21 +278,47 @@ var instance = {
|
||||
if (this.FRAGMENTED_OPCODES.indexOf(this._opcode) < 0 && this._length > 125)
|
||||
return this._fail('protocol_error', 'Received control frame having too long payload: ' + this._length);
|
||||
|
||||
if (this._length > this.MAX_LENGTH)
|
||||
return this._fail('too_large', 'WebSocket frame length too large');
|
||||
if (!this._checkFrameLength()) return;
|
||||
|
||||
this._stage = this._masked ? 3 : 4;
|
||||
this._readMask();
|
||||
},
|
||||
|
||||
_emitFrame: function() {
|
||||
var payload = Hybi.mask(this._payload, this._mask),
|
||||
opcode = this._opcode;
|
||||
_checkFrameLength: function() {
|
||||
if (this.__blength + this._length > this._maxLength) {
|
||||
this._fail('too_large', 'WebSocket frame length too large');
|
||||
return false;
|
||||
} else {
|
||||
return true;
|
||||
}
|
||||
},
|
||||
|
||||
_readMask: function() {
|
||||
if (this._masked)
|
||||
this._reader.read(4, function(buffer) {
|
||||
this._mask = new Mask(buffer);
|
||||
this._readPayload();
|
||||
});
|
||||
else
|
||||
this._readPayload();
|
||||
},
|
||||
|
||||
_readPayload: function() {
|
||||
var stream = this._reader.fork(this._length);
|
||||
if (this._mask) {
|
||||
stream.pipe(this._mask);
|
||||
stream = this._mask;
|
||||
}
|
||||
stream.pipe(new Concat(this._emitFrame, this));
|
||||
},
|
||||
|
||||
_emitFrame: function(payload) {
|
||||
var opcode = this._opcode;
|
||||
|
||||
if (opcode === this.OPCODES.continuation) {
|
||||
if (!this._mode) return this._fail('protocol_error', 'Received unexpected continuation frame');
|
||||
this._buffer(payload);
|
||||
if (this._final) {
|
||||
var message = new Buffer(this.__buffer);
|
||||
var message = Concat.concatBuffers(this.__buffer, this.__blength);
|
||||
if (this._mode === 'text') message = this._encode(message);
|
||||
this._reset();
|
||||
if (message === null)
|
||||
@@ -371,16 +372,19 @@ var instance = {
|
||||
delete callbacks[message];
|
||||
if (callback) callback()
|
||||
}
|
||||
|
||||
this._reader.read(1, this._parseOpcode);
|
||||
},
|
||||
|
||||
_buffer: function(fragment) {
|
||||
for (var i = 0, n = fragment.length; i < n; i++)
|
||||
this.__buffer.push(fragment[i]);
|
||||
this.__buffer.push(fragment);
|
||||
this.__blength += fragment.length;
|
||||
},
|
||||
|
||||
_reset: function() {
|
||||
this._mode = null;
|
||||
this.__buffer = [];
|
||||
this._mode = null;
|
||||
this.__buffer = [];
|
||||
this.__blength = 0;
|
||||
},
|
||||
|
||||
_encode: function(buffer) {
|
||||
|
||||
@@ -0,0 +1,45 @@
|
||||
var Stream = require('stream').Stream,
|
||||
util = require('util');
|
||||
|
||||
var Concat = function(callback, context) {
|
||||
this._callback = callback;
|
||||
this._context = context;
|
||||
this._chunks = [];
|
||||
this._size = 0;
|
||||
this.writable = true;
|
||||
};
|
||||
util.inherits(Concat, Stream);
|
||||
|
||||
Concat.prototype.write = function(buffer) {
|
||||
if (!this.writable) return false;
|
||||
this._chunks.push(buffer);
|
||||
this._size += buffer.length;
|
||||
return true;
|
||||
};
|
||||
|
||||
Concat.prototype.end = function(buffer) {
|
||||
if (buffer) this.write(buffer);
|
||||
this.writable = false;
|
||||
this._callback.call(this._context, Concat.concatBuffers(this._chunks, this._size));
|
||||
};
|
||||
|
||||
Concat.concatBuffers = function(chunks, size) {
|
||||
if (size === undefined) {
|
||||
size = 0;
|
||||
var c = chunks.length;
|
||||
while (c--) size += chunks[c].length;
|
||||
}
|
||||
|
||||
var concat = new Buffer(size),
|
||||
offset = 0;
|
||||
|
||||
for (var i = 0, n = chunks.length; i < n; i++) {
|
||||
chunks[i].copy(concat, offset);
|
||||
offset += chunks[i].length;
|
||||
}
|
||||
return concat;
|
||||
};
|
||||
|
||||
Concat.prototype.destroy = function() {};
|
||||
|
||||
module.exports = Concat;
|
||||
@@ -0,0 +1,43 @@
|
||||
var Stream = require('stream').Stream,
|
||||
util = require('util');
|
||||
|
||||
var Mask = function(bytes) {
|
||||
this.readable = this.writable = true;
|
||||
|
||||
this._bytes = bytes;
|
||||
this._index = 0;
|
||||
};
|
||||
util.inherits(Mask, Stream);
|
||||
|
||||
Mask.mask = function(payload, mask, offset, index) {
|
||||
offset = offset || 0;
|
||||
index = index || 0;
|
||||
|
||||
for (var i = 0, n = payload.length - offset; i < n; i++)
|
||||
payload[offset + i] ^= mask[(index + i) % 4];
|
||||
};
|
||||
|
||||
Mask.prototype.write = function(chunk) {
|
||||
Mask.mask(chunk, this._bytes, 0, this._index);
|
||||
this._index = (this._index + chunk.length) % 4;
|
||||
this.emit('data', chunk);
|
||||
return !this._paused;
|
||||
};
|
||||
|
||||
Mask.prototype.end = function(chunk) {
|
||||
if (chunk) this.write(chunk);
|
||||
this.readable = this.writable = false;
|
||||
this.emit('end');
|
||||
};
|
||||
|
||||
Mask.prototype.pause = function() {
|
||||
this._paused = true;
|
||||
};
|
||||
|
||||
Mask.prototype.resume = function() {
|
||||
this._paused = false;
|
||||
this.emit('drain');
|
||||
};
|
||||
|
||||
module.exports = Mask;
|
||||
|
||||
@@ -1,41 +1,136 @@
|
||||
var StreamReader = function() {
|
||||
this._queue = [];
|
||||
this._cursor = 0;
|
||||
};
|
||||
var Stream = require('stream').Stream,
|
||||
Concat = require('./concat'),
|
||||
util = require('util');
|
||||
|
||||
StreamReader.prototype.read = function(bytes) {
|
||||
return this._readBuffer(bytes);
|
||||
};
|
||||
var defer = (typeof setImmediate === 'function')
|
||||
? setImmediate
|
||||
: process.nextTick;
|
||||
|
||||
var StreamReader = function(options, parent) {
|
||||
this.readable = !!parent;
|
||||
this.writable = !parent;
|
||||
this._streams = [];
|
||||
this._context = options.context;
|
||||
this._parent = parent;
|
||||
this._queue = [];
|
||||
this._queueSize = 0;
|
||||
this._cursor = 0;
|
||||
};
|
||||
util.inherits(StreamReader, Stream);
|
||||
|
||||
StreamReader.prototype.write = function(buffer) {
|
||||
if (!this.writable) return false;
|
||||
if (!buffer || buffer.length === 0) return !this._paused;
|
||||
|
||||
StreamReader.prototype.put = function(buffer) {
|
||||
if (!buffer || buffer.length === 0) return;
|
||||
if (!buffer.copy) buffer = new Buffer(buffer);
|
||||
|
||||
this._queue.push(buffer);
|
||||
this._queueSize += buffer.length;
|
||||
this._flush();
|
||||
|
||||
return !this._paused;
|
||||
};
|
||||
|
||||
StreamReader.prototype._readBuffer = function(length) {
|
||||
var buffer = new Buffer(length),
|
||||
queue = this._queue,
|
||||
StreamReader.prototype.end = function(buffer) {
|
||||
if (buffer) this.write(buffer);
|
||||
this.writable = false;
|
||||
|
||||
for (var i = 0, n = this._streams.length; i < n; i++) {
|
||||
this._streams[i].emit('end');
|
||||
this._streams[i].readable = false;
|
||||
}
|
||||
|
||||
this._context = this._streams = this._queue = [];
|
||||
};
|
||||
|
||||
StreamReader.prototype.pause = function() {
|
||||
this._paused = true;
|
||||
if (this._parent) this._parent.pause();
|
||||
};
|
||||
|
||||
StreamReader.prototype.resume = function() {
|
||||
this._paused = false;
|
||||
this.emit('drain');
|
||||
if (this._parent) this._parent.resume();
|
||||
};
|
||||
|
||||
StreamReader.prototype.fork = function(length) {
|
||||
if (!this.writable) return null;
|
||||
|
||||
var stream = new StreamReader({context: this._context}, this),
|
||||
self = this;
|
||||
|
||||
stream._remaining = length;
|
||||
this._streams.push(stream);
|
||||
defer(function() { self._flush() });
|
||||
|
||||
return stream;
|
||||
};
|
||||
|
||||
StreamReader.prototype.read = function(length, callback) {
|
||||
if (!this.writable) return;
|
||||
|
||||
if (this._queueSize >= length)
|
||||
return callback.call(this._context, this._readBytes(length));
|
||||
|
||||
this.fork(length).pipe(new Concat(callback, this._context));
|
||||
};
|
||||
|
||||
StreamReader.prototype._flush = function() {
|
||||
var streams = this._streams, stream, size, buffer;
|
||||
|
||||
while (streams.length > 0) {
|
||||
stream = streams[0];
|
||||
size = Math.min(stream._remaining, this._queueSize);
|
||||
buffer = this._readBytes(size);
|
||||
|
||||
if (size > 0) stream.emit('data', buffer);
|
||||
stream._remaining -= size;
|
||||
|
||||
if (stream._remaining > 0) break;
|
||||
|
||||
stream.readable = false;
|
||||
stream.emit('end');
|
||||
streams.shift();
|
||||
}
|
||||
};
|
||||
|
||||
StreamReader.prototype._readBytes = function(length) {
|
||||
var queue = this._queue,
|
||||
remain = length,
|
||||
n = queue.length,
|
||||
first = queue[0],
|
||||
i = 0,
|
||||
chunk, offset, size;
|
||||
buffer, chunk, size;
|
||||
|
||||
if (remain === 0) return buffer;
|
||||
if (length === 0) return new Buffer(0);
|
||||
|
||||
if (remain <= first.length - this._cursor) {
|
||||
buffer = first.slice(this._cursor, this._cursor + remain);
|
||||
|
||||
this._queueSize -= remain;
|
||||
this._cursor = (this._cursor + remain) % first.length;
|
||||
if (this._cursor === 0) this._queue.shift();
|
||||
|
||||
return buffer;
|
||||
}
|
||||
|
||||
buffer = new Buffer(length);
|
||||
|
||||
while (remain > 0 && i < n) {
|
||||
chunk = queue[i];
|
||||
offset = (i === 0) ? this._cursor : 0;
|
||||
size = Math.min(remain, chunk.length - offset);
|
||||
chunk.copy(buffer, length - remain, offset, offset + size);
|
||||
remain -= size;
|
||||
size = Math.min(remain, chunk.length - this._cursor);
|
||||
|
||||
chunk.copy(buffer, length - remain, this._cursor, this._cursor + size);
|
||||
|
||||
remain -= size;
|
||||
this._queueSize -= size;
|
||||
this._cursor = (this._cursor + size) % chunk.length;
|
||||
|
||||
i += 1;
|
||||
}
|
||||
|
||||
if (remain > 0) return null;
|
||||
|
||||
queue.splice(0, i-1);
|
||||
this._cursor = (i === 1 ? this._cursor : 0) + size;
|
||||
queue.splice(0, this._cursor === 0 ? i : i - 1);
|
||||
|
||||
return buffer;
|
||||
};
|
||||
|
||||
@@ -34,6 +34,7 @@ var instance = {
|
||||
this._delegate = Server.http(this, this._options);
|
||||
this._delegate.messages = this.messages;
|
||||
this._delegate.io = this.io;
|
||||
this._delegate._extensions = this._extensions;
|
||||
|
||||
this._delegate.on('open', function() { self._open() });
|
||||
this.EVENTS.forEach(function(event) {
|
||||
|
||||
@@ -0,0 +1,22 @@
|
||||
var PerMessageDeflate = function() {
|
||||
this.name = 'permessage-deflate';
|
||||
};
|
||||
|
||||
PerMessageDeflate.create = function() {
|
||||
return new this();
|
||||
};
|
||||
|
||||
PerMessageDeflate.prototype.createSession = function(params) {
|
||||
return new Session(this, params);
|
||||
};
|
||||
|
||||
var Session = function(extension, params) {
|
||||
this._ext = extension;
|
||||
this._params = params;
|
||||
};
|
||||
|
||||
Session.prototype.responseHeader = function() {
|
||||
return this._ext.name;
|
||||
};
|
||||
|
||||
module.exports = PerMessageDeflate;
|
||||
+1
-1
@@ -5,7 +5,7 @@
|
||||
, "keywords" : ["websocket"]
|
||||
, "license" : "MIT"
|
||||
|
||||
, "version" : "0.3.0"
|
||||
, "version" : "0.3.4"
|
||||
, "engines" : {"node": ">=0.4.0"}
|
||||
, "main" : "./lib/websocket/driver"
|
||||
, "devDependencies" : {"jstest": ""}
|
||||
|
||||
@@ -131,7 +131,10 @@ test.describe("Client", function() { with(this) {
|
||||
before(function() { this.driver().start() })
|
||||
|
||||
describe("with a valid response", function() { with(this) {
|
||||
before(function() { this.driver().parse(new Buffer(this.response())) })
|
||||
before(function(resume) { with(this) {
|
||||
driver().parse(new Buffer(response()))
|
||||
setTimeout(resume, 10)
|
||||
}})
|
||||
|
||||
it("changes the state to open", function() { with(this) {
|
||||
assertEqual( true, open )
|
||||
@@ -149,11 +152,12 @@ test.describe("Client", function() { with(this) {
|
||||
}})
|
||||
|
||||
describe("with a valid response followed by a frame", function() { with(this) {
|
||||
before(function() { with(this) {
|
||||
before(function(resume) { with(this) {
|
||||
var resp = new Buffer(response().length + 4)
|
||||
new Buffer(response()).copy(resp)
|
||||
new Buffer([0x81, 0x02, 72, 105]).copy(resp, resp.length - 4)
|
||||
driver().parse(resp)
|
||||
setTimeout(resume, 10)
|
||||
}})
|
||||
|
||||
it("changes the state to open", function() { with(this) {
|
||||
@@ -168,10 +172,11 @@ test.describe("Client", function() { with(this) {
|
||||
}})
|
||||
|
||||
describe("with a bad status line", function() { with(this) {
|
||||
before(function() {
|
||||
var resp = this.response().replace(/101/g, "4")
|
||||
this.driver().parse(new Buffer(resp))
|
||||
})
|
||||
before(function(resume) { with(this) {
|
||||
var resp = response().replace(/101/g, "4")
|
||||
driver().parse(new Buffer(resp))
|
||||
setTimeout(resume, 10)
|
||||
}})
|
||||
|
||||
it("changes the state to closed", function() { with(this) {
|
||||
assertEqual( false, open )
|
||||
@@ -182,10 +187,11 @@ test.describe("Client", function() { with(this) {
|
||||
}})
|
||||
|
||||
describe("with a bad Upgrade header", function() { with(this) {
|
||||
before(function() {
|
||||
var resp = this.response().replace(/websocket/g, "wrong")
|
||||
this.driver().parse(new Buffer(resp))
|
||||
})
|
||||
before(function(resume) { with(this) {
|
||||
var resp = response().replace(/websocket/g, "wrong")
|
||||
driver().parse(new Buffer(resp))
|
||||
setTimeout(resume, 10)
|
||||
}})
|
||||
|
||||
it("changes the state to closed", function() { with(this) {
|
||||
assertEqual( false, open )
|
||||
@@ -196,10 +202,11 @@ test.describe("Client", function() { with(this) {
|
||||
}})
|
||||
|
||||
describe("with a bad Accept header", function() { with(this) {
|
||||
before(function() {
|
||||
var resp = this.response().replace(/QV3/g, "wrong")
|
||||
this.driver().parse(new Buffer(resp))
|
||||
})
|
||||
before(function(resume) { with(this) {
|
||||
var resp = response().replace(/QV3/g, "wrong")
|
||||
driver().parse(new Buffer(resp))
|
||||
setTimeout(resume, 10)
|
||||
}})
|
||||
|
||||
it("changes the state to closed", function() { with(this) {
|
||||
assertEqual( false, open )
|
||||
@@ -212,10 +219,11 @@ test.describe("Client", function() { with(this) {
|
||||
describe("with valid subprotocols", function() { with(this) {
|
||||
define("protocols", function() { return ["foo", "xmpp"] })
|
||||
|
||||
before(function() {
|
||||
var resp = this.response().replace(/\r\n\r\n/, "\r\nSec-WebSocket-Protocol: xmpp\r\n\r\n")
|
||||
this.driver().parse(new Buffer(resp))
|
||||
})
|
||||
before(function(resume) { with(this) {
|
||||
var resp = response().replace(/\r\n\r\n/, "\r\nSec-WebSocket-Protocol: xmpp\r\n\r\n")
|
||||
driver().parse(new Buffer(resp))
|
||||
setTimeout(resume, 10)
|
||||
}})
|
||||
|
||||
it("changs the state to open", function() { with(this) {
|
||||
assertEqual( true, open )
|
||||
@@ -231,10 +239,11 @@ test.describe("Client", function() { with(this) {
|
||||
describe("with invalid subprotocols", function() { with(this) {
|
||||
define("protocols", function() { return ["foo", "xmpp"] })
|
||||
|
||||
before(function() {
|
||||
var resp = this.response().replace(/\r\n\r\n/, "\r\nSec-WebSocket-Protocol: irc\r\n\r\n")
|
||||
this.driver().parse(new Buffer(resp))
|
||||
})
|
||||
before(function(resume) { with(this) {
|
||||
var resp = response().replace(/\r\n\r\n/, "\r\nSec-WebSocket-Protocol: irc\r\n\r\n")
|
||||
driver().parse(new Buffer(resp))
|
||||
setTimeout(resume, 10)
|
||||
}})
|
||||
|
||||
it("changs the state to closed", function() { with(this) {
|
||||
assertEqual( false, open )
|
||||
|
||||
@@ -53,7 +53,7 @@ test.describe("Draft76", function() { with(this) {
|
||||
describe("start", function() { with(this) {
|
||||
it("writes the handshake response to the socket", function() { with(this) {
|
||||
expect(driver().io, "emit").given("data", buffer(
|
||||
"HTTP/1.1 101 Web Socket Protocol Handshake\r\n" +
|
||||
"HTTP/1.1 101 WebSocket Protocol Handshake\r\n" +
|
||||
"Upgrade: WebSocket\r\n" +
|
||||
"Connection: Upgrade\r\n" +
|
||||
"Sec-WebSocket-Origin: http://www.example.com\r\n" +
|
||||
@@ -95,7 +95,7 @@ test.describe("Draft76", function() { with(this) {
|
||||
|
||||
it("queues the frames until the handshake has been sent", function() { with(this) {
|
||||
expect(driver().io, "emit").given("data", buffer(
|
||||
"HTTP/1.1 101 Web Socket Protocol Handshake\r\n" +
|
||||
"HTTP/1.1 101 WebSocket Protocol Handshake\r\n" +
|
||||
"Upgrade: WebSocket\r\n" +
|
||||
"Connection: Upgrade\r\n" +
|
||||
"Sec-WebSocket-Origin: http://www.example.com\r\n" +
|
||||
@@ -116,7 +116,7 @@ test.describe("Draft76", function() { with(this) {
|
||||
|
||||
it("writes the handshake response with no body", function() { with(this) {
|
||||
expect(driver().io, "emit").given("data", buffer(
|
||||
"HTTP/1.1 101 Web Socket Protocol Handshake\r\n" +
|
||||
"HTTP/1.1 101 WebSocket Protocol Handshake\r\n" +
|
||||
"Upgrade: WebSocket\r\n" +
|
||||
"Connection: Upgrade\r\n" +
|
||||
"Sec-WebSocket-Origin: http://www.example.com\r\n" +
|
||||
|
||||
@@ -199,42 +199,54 @@ test.describe("Hybi", function() { with(this) {
|
||||
return output
|
||||
})
|
||||
|
||||
it("parses unmasked text frames", function() { with(this) {
|
||||
it("parses unmasked text frames", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x05, 0x48, 0x65, 0x6c, 0x6c, 0x6f])
|
||||
assertEqual( "Hello", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "Hello", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses multiple frames from the same packet", function() { with(this) {
|
||||
it("parses multiple frames from the same packet", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x05, 0x48, 0x65, 0x6c, 0x6c, 0x6f, 0x81, 0x05, 0x48, 0x65, 0x6c, 0x6c, 0x6f])
|
||||
assertEqual( "HelloHello", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "HelloHello", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses empty text frames", function() { with(this) {
|
||||
it("parses empty text frames", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x00])
|
||||
assertEqual( "", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses fragmented text frames", function() { with(this) {
|
||||
it("parses fragmented text frames", function(resume) { with(this) {
|
||||
driver().parse([0x01, 0x03, 0x48, 0x65, 0x6c])
|
||||
driver().parse([0x80, 0x02, 0x6c, 0x6f])
|
||||
assertEqual( "Hello", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "Hello", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses masked text frames", function() { with(this) {
|
||||
it("parses masked text frames", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x85])
|
||||
driver().parse(mask())
|
||||
driver().parse(maskMessage([0x48, 0x65, 0x6c, 0x6c, 0x6f]))
|
||||
assertEqual( "Hello", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "Hello", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses masked empty text frames", function() { with(this) {
|
||||
it("parses masked empty text frames", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x80])
|
||||
driver().parse(mask())
|
||||
driver().parse(maskMessage([]))
|
||||
assertEqual( "", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses masked fragmented text frames", function() { with(this) {
|
||||
it("parses masked fragmented text frames", function(resume) { with(this) {
|
||||
driver().parse([0x01, 0x81])
|
||||
driver().parse(mask())
|
||||
driver().parse(maskMessage([0x48]))
|
||||
@@ -243,49 +255,67 @@ test.describe("Hybi", function() { with(this) {
|
||||
driver().parse(mask())
|
||||
driver().parse(maskMessage([0x65, 0x6c, 0x6c, 0x6f]))
|
||||
|
||||
assertEqual( "Hello", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "Hello", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("closes the socket if the frame has an unrecognized opcode", function() { with(this) {
|
||||
it("closes the socket if the frame has an unrecognized opcode", function(resume) { with(this) {
|
||||
driver().parse([0x83, 0x00])
|
||||
assertEqual( [0x88, 0x1e, 0x03, 0xea], collector().bytes.slice(0,4) )
|
||||
assertEqual( "Unrecognized frame opcode: 3", error.message )
|
||||
assertEqual( [1002, "Unrecognized frame opcode: 3"], close )
|
||||
assertEqual( "closed", driver().getState() )
|
||||
setTimeout(function() {
|
||||
resume(function() {
|
||||
assertEqual( [0x88, 0x1e, 0x03, 0xea], collector().bytes.slice(0,4) )
|
||||
assertEqual( "Unrecognized frame opcode: 3", error.message )
|
||||
assertEqual( [1002, "Unrecognized frame opcode: 3"], close )
|
||||
assertEqual( "closed", driver().getState() )
|
||||
})
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("closes the socket if a close frame is received", function() { with(this) {
|
||||
it("closes the socket if a close frame is received", function(resume) { with(this) {
|
||||
driver().parse([0x88, 0x07, 0x03, 0xe8, 0x48, 0x65, 0x6c, 0x6c, 0x6f])
|
||||
assertEqual( [0x88, 0x07, 0x03, 0xe8, 0x48, 0x65, 0x6c, 0x6c, 0x6f], collector().bytes )
|
||||
assertEqual( [1000, "Hello"], close )
|
||||
assertEqual( "closed", driver().getState() )
|
||||
setTimeout(function() {
|
||||
resume(function() {
|
||||
assertEqual( [0x88, 0x07, 0x03, 0xe8, 0x48, 0x65, 0x6c, 0x6c, 0x6f], collector().bytes )
|
||||
assertEqual( [1000, "Hello"], close )
|
||||
assertEqual( "closed", driver().getState() )
|
||||
})
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses unmasked multibyte text frames", function() { with(this) {
|
||||
it("parses unmasked multibyte text frames", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x0b, 0x41, 0x70, 0x70, 0x6c, 0x65, 0x20, 0x3d, 0x20, 0xef, 0xa3, 0xbf])
|
||||
assertEqual( "Apple = ", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "Apple = ", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses frames received in several packets", function() { with(this) {
|
||||
it("parses frames received in several packets", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x0b, 0x41, 0x70, 0x70, 0x6c])
|
||||
driver().parse([0x65, 0x20, 0x3d, 0x20, 0xef, 0xa3, 0xbf])
|
||||
assertEqual( "Apple = ", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "Apple = ", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses fragmented multibyte text frames", function() { with(this) {
|
||||
it("parses fragmented multibyte text frames", function(resume) { with(this) {
|
||||
driver().parse([0x01, 0x0a, 0x41, 0x70, 0x70, 0x6c, 0x65, 0x20, 0x3d, 0x20, 0xef, 0xa3])
|
||||
driver().parse([0x80, 0x01, 0xbf])
|
||||
assertEqual( "Apple = ", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "Apple = ", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parse masked multibyte text frames", function() { with(this) {
|
||||
it("parse masked multibyte text frames", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x8b])
|
||||
driver().parse(mask())
|
||||
driver().parse(maskMessage([0x41, 0x70, 0x70, 0x6c, 0x65, 0x20, 0x3d, 0x20, 0xef, 0xa3, 0xbf]))
|
||||
assertEqual( "Apple = ", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "Apple = ", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses masked fragmented multibyte text frames", function() { with(this) {
|
||||
it("parses masked fragmented multibyte text frames", function(resume) { with(this) {
|
||||
driver().parse([0x01, 0x8a])
|
||||
driver().parse(mask())
|
||||
driver().parse(maskMessage([0x41, 0x70, 0x70, 0x6c, 0x65, 0x20, 0x3d, 0x20, 0xef, 0xa3]))
|
||||
@@ -294,27 +324,35 @@ test.describe("Hybi", function() { with(this) {
|
||||
driver().parse(mask())
|
||||
driver().parse(maskMessage([0xbf]))
|
||||
|
||||
assertEqual( "Apple = ", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "Apple = ", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses unmasked medium-length text frames", function() { with(this) {
|
||||
it("parses unmasked medium-length text frames", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x7e, 0x00, 0xc8])
|
||||
var i = 40, result = ""
|
||||
while (i--) {
|
||||
driver().parse([0x48, 0x65, 0x6c, 0x6c, 0x6f])
|
||||
result += "Hello"
|
||||
}
|
||||
assertEqual( result, message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( result, message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("returns an error for too-large frames", function() { with(this) {
|
||||
driver().parse([0x81, 0x7f, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00])
|
||||
assertEqual( "WebSocket frame length too large", error.message )
|
||||
assertEqual( [1009, "WebSocket frame length too large"], close )
|
||||
assertEqual( "closed", driver().getState() )
|
||||
it("returns an error for too-large frames", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x7f, 0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00])
|
||||
setTimeout(function() {
|
||||
resume(function() {
|
||||
assertEqual( "WebSocket frame length too large", error.message )
|
||||
assertEqual( [1009, "WebSocket frame length too large"], close )
|
||||
assertEqual( "closed", driver().getState() )
|
||||
})
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("parses masked medium-length text frames", function() { with(this) {
|
||||
it("parses masked medium-length text frames", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0xfe, 0x00, 0xc8])
|
||||
driver().parse(mask())
|
||||
var i = 40, result = "", packet = []
|
||||
@@ -323,12 +361,18 @@ test.describe("Hybi", function() { with(this) {
|
||||
result += "Hello"
|
||||
}
|
||||
driver().parse(maskMessage(packet))
|
||||
assertEqual( result, message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( result, message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("replies to pings with a pong", function() { with(this) {
|
||||
it("replies to pings with a pong", function(resume) { with(this) {
|
||||
driver().parse([0x89, 0x04, 0x4f, 0x48, 0x41, 0x49])
|
||||
assertEqual( [0x8a, 0x04, 0x4f, 0x48, 0x41, 0x49], collector().bytes )
|
||||
setTimeout(function() {
|
||||
resume(function() {
|
||||
assertEqual( [0x8a, 0x04, 0x4f, 0x48, 0x41, 0x49], collector().bytes )
|
||||
})
|
||||
}, 10)
|
||||
}})
|
||||
}})
|
||||
|
||||
@@ -379,18 +423,22 @@ test.describe("Hybi", function() { with(this) {
|
||||
assertEqual( true, driver().ping() )
|
||||
}})
|
||||
|
||||
it("runs the given callback on mathing pong", function() { with(this) {
|
||||
it("runs the given callback on mathing pong", function(resume) { with(this) {
|
||||
var reply = null
|
||||
driver().ping("Hi", function() { reply = true })
|
||||
driver().parse([0x8a, 0x02, 72, 105])
|
||||
assert( reply )
|
||||
setTimeout(function() {
|
||||
resume(function() { assert( reply ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("does not run the callback on non-matching pong", function() { with(this) {
|
||||
it("does not run the callback on non-matching pong", function(resume) { with(this) {
|
||||
var reply = null
|
||||
driver().ping("Hi", function() { reply = true })
|
||||
driver().parse([0x8a, 0x03, 119, 97, 116])
|
||||
assert( !reply )
|
||||
setTimeout(function() {
|
||||
resume(function() { assert( !reply ) })
|
||||
}, 10)
|
||||
}})
|
||||
}})
|
||||
|
||||
@@ -427,15 +475,21 @@ test.describe("Hybi", function() { with(this) {
|
||||
this.driver().start()
|
||||
})
|
||||
|
||||
it("does not emit a message", function() { with(this) {
|
||||
it("does not emit a message", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x05, 0x48, 0x65, 0x6c, 0x6c, 0x6f])
|
||||
assertEqual( "", message )
|
||||
setTimeout(function() {
|
||||
resume(function() { assertEqual( "", message ) })
|
||||
}, 10)
|
||||
}})
|
||||
|
||||
it("returns an error", function() { with(this) {
|
||||
it("returns an error", function(resume) { with(this) {
|
||||
driver().parse([0x81, 0x05, 0x48, 0x65, 0x6c, 0x6c, 0x6f])
|
||||
assertEqual( "Received unmasked frame but masking is required", error.message )
|
||||
assertEqual( [1003, "Received unmasked frame but masking is required"], close )
|
||||
setTimeout(function() {
|
||||
resume(function() {
|
||||
assertEqual( "Received unmasked frame but masking is required", error.message )
|
||||
assertEqual( [1003, "Received unmasked frame but masking is required"], close )
|
||||
})
|
||||
}, 10)
|
||||
}})
|
||||
}})
|
||||
|
||||
@@ -479,8 +533,9 @@ test.describe("Hybi", function() { with(this) {
|
||||
}})
|
||||
|
||||
describe("receiving a close frame", function() { with(this) {
|
||||
before(function() {
|
||||
before(function(resume) {
|
||||
this.driver().parse([0x88, 0x04, 0x03, 0xe9, 0x4f, 0x4b])
|
||||
setTimeout(resume, 10)
|
||||
})
|
||||
|
||||
it("triggers the onclose event", function() { with(this) {
|
||||
@@ -494,10 +549,11 @@ test.describe("Hybi", function() { with(this) {
|
||||
}})
|
||||
|
||||
describe("in the closed state", function() { with(this) {
|
||||
before(function() {
|
||||
before(function(resume) {
|
||||
this.driver().start()
|
||||
this.driver().close()
|
||||
this.driver().parse([0x88, 0x02, 0x03, 0xea])
|
||||
setTimeout(resume, 10)
|
||||
})
|
||||
|
||||
describe("frame", function() { with(this) {
|
||||
|
||||
Reference in New Issue
Block a user