76 lines
1.6 KiB
JavaScript
76 lines
1.6 KiB
JavaScript
|
|
||
|
/**
|
||
|
* Module dependencies.
|
||
|
*/
|
||
|
|
||
|
var Polling = require('./polling');
|
||
|
var qs = require('querystring');
|
||
|
var rDoubleSlashes = /\\\\n/g;
|
||
|
var rSlashes = /(\\)?\\n/g;
|
||
|
var util = require('util');
|
||
|
|
||
|
/**
|
||
|
* Module exports.
|
||
|
*/
|
||
|
|
||
|
module.exports = JSONP;
|
||
|
|
||
|
/**
|
||
|
* JSON-P polling transport.
|
||
|
*
|
||
|
* @api public
|
||
|
*/
|
||
|
|
||
|
function JSONP (req) {
|
||
|
Polling.call(this, req);
|
||
|
|
||
|
this.head = '___eio[' + (req._query.j || '').replace(/[^0-9]/g, '') + '](';
|
||
|
this.foot = ');';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Inherits from Polling.
|
||
|
*/
|
||
|
|
||
|
util.inherits(JSONP, Polling);
|
||
|
|
||
|
/**
|
||
|
* Handles incoming data.
|
||
|
* Due to a bug in \n handling by browsers, we expect a escaped string.
|
||
|
*
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
JSONP.prototype.onData = function (data) {
|
||
|
// we leverage the qs module so that we get built-in DoS protection
|
||
|
// and the fast alternative to decodeURIComponent
|
||
|
data = qs.parse(data).d;
|
||
|
if ('string' === typeof data) {
|
||
|
// client will send already escaped newlines as \\\\n and newlines as \\n
|
||
|
// \\n must be replaced with \n and \\\\n with \\n
|
||
|
data = data.replace(rSlashes, function (match, slashes) {
|
||
|
return slashes ? match : '\n';
|
||
|
});
|
||
|
Polling.prototype.onData.call(this, data.replace(rDoubleSlashes, '\\n'));
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Performs the write.
|
||
|
*
|
||
|
* @api private
|
||
|
*/
|
||
|
|
||
|
JSONP.prototype.doWrite = function (data, options, callback) {
|
||
|
// we must output valid javascript, not valid json
|
||
|
// see: http://timelessrepo.com/json-isnt-a-javascript-subset
|
||
|
var js = JSON.stringify(data)
|
||
|
.replace(/\u2028/g, '\\u2028')
|
||
|
.replace(/\u2029/g, '\\u2029');
|
||
|
|
||
|
// prepare response
|
||
|
data = this.head + js + this.foot;
|
||
|
|
||
|
Polling.prototype.doWrite.call(this, data, options, callback);
|
||
|
};
|