spec/utils/javascripts/faye.js in faye-authentication-0.4.0 vs spec/utils/javascripts/faye.js in faye-authentication-1.6.0
- old
+ new
@@ -1,10 +1,10 @@
(function() {
'use strict';
var Faye = {
- VERSION: '1.0.1',
+ VERSION: '1.0.3',
BAYEUX_VERSION: '1.0',
ID_LENGTH: 160,
JSONP_CALLBACK: 'jsonpcallback',
CONNECTION_TYPES: ['long-polling', 'cross-origin-long-polling', 'callback-polling', 'websocket', 'eventsource', 'in-process'],
@@ -24,11 +24,14 @@
return dest;
},
random: function(bitlength) {
bitlength = bitlength || this.ID_LENGTH;
- return csprng(bitlength, 36);
+ var maxLength = Math.ceil(bitlength * Math.log(2) / Math.log(36));
+ var string = csprng(bitlength, 36);
+ while (string.length < maxLength) string = '0' + string;
+ return string;
},
clientIdFromMessages: function(messages) {
var connect = this.filter([].concat(messages), function(message) {
return message.channel === '/meta/connect';
@@ -742,11 +745,11 @@
},
setDeferredStatus: function(status, value) {
if (this._timer) Faye.ENV.clearTimeout(this._timer);
- var promise = this.then();
+ this.then();
if (status === 'succeeded')
this._fulfill(value);
else if (status === 'failed')
this._reject(value);
@@ -799,11 +802,11 @@
removeTimeout: function(name) {
this._timeouts = this._timeouts || {};
var timeout = this._timeouts[name];
if (!timeout) return;
- clearTimeout(timeout);
+ Faye.ENV.clearTimeout(timeout);
delete this._timeouts[name];
},
removeAllTimeouts: function() {
this._timeouts = this._timeouts || {};
@@ -821,17 +824,17 @@
},
writeLog: function(messageArgs, level) {
if (!Faye.logger) return;
- var messageArgs = Array.prototype.slice.apply(messageArgs),
- banner = '[Faye',
- klass = this.className,
+ var args = Array.prototype.slice.apply(messageArgs),
+ banner = '[Faye',
+ klass = this.className,
- message = messageArgs.shift().replace(/\?/g, function() {
+ message = args.shift().replace(/\?/g, function() {
try {
- return Faye.toJSON(messageArgs.shift());
+ return Faye.toJSON(args.shift());
} catch (e) {
return '[Object]';
}
});
@@ -850,15 +853,15 @@
}
};
(function() {
for (var key in Faye.Logging.LOG_LEVELS)
- (function(level, value) {
+ (function(level) {
Faye.Logging[level] = function() {
this.writeLog(arguments, level);
};
- })(key, Faye.Logging.LOG_LEVELS[key]);
+ })(key);
})();
Faye.Grammar = {
CHANNEL_NAME: /^\/(((([a-z]|[A-Z])|[0-9])|(\-|\_|\!|\~|\(|\)|\$|\@)))+(\/(((([a-z]|[A-Z])|[0-9])|(\-|\_|\!|\~|\(|\)|\$|\@)))+)*$/,
CHANNEL_PATTERN: /^(\/(((([a-z]|[A-Z])|[0-9])|(\-|\_|\!|\~|\(|\)|\$|\@)))+)*\/\*{1,2}$/,
@@ -997,16 +1000,15 @@
hasSubscription: function(name) {
return this._channels.hasOwnProperty(name);
},
subscribe: function(names, callback, context) {
- if (!callback) return;
var name;
for (var i = 0, n = names.length; i < n; i++) {
name = names[i];
var channel = this._channels[name] = this._channels[name] || new Faye.Channel(name);
- channel.bind('message', callback, context);
+ if (callback) channel.bind('message', callback, context);
}
},
unsubscribe: function(name, callback, context) {
var channel = this._channels[name];
@@ -1030,21 +1032,10 @@
}
}
})
});
-Faye.Envelope = Faye.Class({
- initialize: function(message, timeout) {
- this.id = message.id;
- this.message = message;
-
- if (timeout !== undefined) this.timeout(timeout / 1000, false);
- }
-});
-
-Faye.extend(Faye.Envelope.prototype, Faye.Deferrable);
-
Faye.Publication = Faye.Class(Faye.Deferrable);
Faye.Subscription = Faye.Class({
initialize: function(client, channels, callback, context) {
this._client = client;
@@ -1066,69 +1057,59 @@
});
Faye.extend(Faye.Subscription.prototype, Faye.Deferrable);
Faye.Client = Faye.Class({
- UNCONNECTED: 1,
- CONNECTING: 2,
- CONNECTED: 3,
- DISCONNECTED: 4,
+ UNCONNECTED: 1,
+ CONNECTING: 2,
+ CONNECTED: 3,
+ DISCONNECTED: 4,
- HANDSHAKE: 'handshake',
- RETRY: 'retry',
- NONE: 'none',
+ HANDSHAKE: 'handshake',
+ RETRY: 'retry',
+ NONE: 'none',
- CONNECTION_TIMEOUT: 60,
- DEFAULT_RETRY: 5,
- MAX_REQUEST_SIZE: 2048,
+ CONNECTION_TIMEOUT: 60,
- DEFAULT_ENDPOINT: '/bayeux',
- INTERVAL: 0,
+ DEFAULT_ENDPOINT: '/bayeux',
+ INTERVAL: 0,
initialize: function(endpoint, options) {
this.info('New client created for ?', endpoint);
+ options = options || {};
- this._options = options || {};
- this.endpoint = Faye.URI.parse(endpoint || this.DEFAULT_ENDPOINT);
- this.endpoints = this._options.endpoints || {};
- this.transports = {};
- this.cookies = Faye.CookieJar && new Faye.CookieJar();
- this.headers = {};
- this.ca = this._options.ca;
- this._disabled = [];
- this._retry = this._options.retry || this.DEFAULT_RETRY;
+ this._endpoint = endpoint || this.DEFAULT_ENDPOINT;
+ this._channels = new Faye.Channel.Set();
+ this._dispatcher = new Faye.Dispatcher(this, this._endpoint, options);
- for (var key in this.endpoints)
- this.endpoints[key] = Faye.URI.parse(this.endpoints[key]);
-
- this.maxRequestSize = this.MAX_REQUEST_SIZE;
-
- this._state = this.UNCONNECTED;
- this._channels = new Faye.Channel.Set();
this._messageId = 0;
+ this._state = this.UNCONNECTED;
this._responseCallbacks = {};
this._advice = {
reconnect: this.RETRY,
- interval: 1000 * (this._options.interval || this.INTERVAL),
- timeout: 1000 * (this._options.timeout || this.CONNECTION_TIMEOUT)
+ interval: 1000 * (options.interval || this.INTERVAL),
+ timeout: 1000 * (options.timeout || this.CONNECTION_TIMEOUT)
};
+ this._dispatcher.timeout = this._advice.timeout / 1000;
+ this._dispatcher.bind('message', this._receiveMessage, this);
+
if (Faye.Event && Faye.ENV.onbeforeunload !== undefined)
Faye.Event.on(Faye.ENV, 'beforeunload', function() {
- if (Faye.indexOf(this._disabled, 'autodisconnect') < 0)
+ if (Faye.indexOf(this._dispatcher._disabled, 'autodisconnect') < 0)
this.disconnect();
}, this);
},
disable: function(feature) {
- this._disabled.push(feature);
+ return this._dispatcher.disable(feature);
},
setHeader: function(name, value) {
- this.headers[name] = value;
+ return this._dispatcher.setHeader(name, value);
},
// Request
// MUST include: * channel
// * version
@@ -1153,34 +1134,34 @@
if (this._state !== this.UNCONNECTED) return;
this._state = this.CONNECTING;
var self = this;
- this.info('Initiating handshake with ?', Faye.URI.stringify(this.endpoint));
- this._selectTransport(Faye.MANDATORY_CONNECTION_TYPES);
+ this.info('Initiating handshake with ?', Faye.URI.stringify(this._endpoint));
+ this._dispatcher.selectTransport(Faye.MANDATORY_CONNECTION_TYPES);
- this._send({
+ this._sendMessage({
channel: Faye.Channel.HANDSHAKE,
version: Faye.BAYEUX_VERSION,
- supportedConnectionTypes: [this._transport.connectionType]
+ supportedConnectionTypes: [this._dispatcher.connectionType]
- }, function(response) {
+ }, {}, function(response) {
if (response.successful) {
- this._state = this.CONNECTED;
- this._clientId = response.clientId;
+ this._state = this.CONNECTED;
+ this._dispatcher.clientId = response.clientId;
- this._selectTransport(response.supportedConnectionTypes);
+ this._dispatcher.selectTransport(response.supportedConnectionTypes);
- this.info('Handshake successful: ?', this._clientId);
+ this.info('Handshake successful: ?', this._dispatcher.clientId);
this.subscribe(this._channels.getKeys(), true);
if (callback) Faye.Promise.defer(function() { callback.call(context) });
} else {
this.info('Handshake unsuccessful');
- Faye.ENV.setTimeout(function() { self.handshake(callback, context) }, this._advice.interval);
+ Faye.ENV.setTimeout(function() { self.handshake(callback, context) }, this._dispatcher.retry * 1000);
this._state = this.UNCONNECTED;
}
}, this);
},
@@ -1201,25 +1182,25 @@
return this.handshake(function() { this.connect(callback, context) }, this);
this.callback(callback, context);
if (this._state !== this.CONNECTED) return;
- this.info('Calling deferred actions for ?', this._clientId);
+ this.info('Calling deferred actions for ?', this._dispatcher.clientId);
this.setDeferredStatus('succeeded');
this.setDeferredStatus('unknown');
if (this._connectRequest) return;
this._connectRequest = true;
- this.info('Initiating connection for ?', this._clientId);
+ this.info('Initiating connection for ?', this._dispatcher.clientId);
- this._send({
+ this._sendMessage({
channel: Faye.Channel.CONNECT,
- clientId: this._clientId,
- connectionType: this._transport.connectionType
+ clientId: this._dispatcher.clientId,
+ connectionType: this._dispatcher.connectionType
- }, this._cycleConnection, this);
+ }, {}, this._cycleConnection, this);
},
// Request Response
// MUST include: * channel MUST include: * channel
// * clientId * successful
@@ -1229,23 +1210,21 @@
// * id
disconnect: function() {
if (this._state !== this.CONNECTED) return;
this._state = this.DISCONNECTED;
- this.info('Disconnecting ?', this._clientId);
+ this.info('Disconnecting ?', this._dispatcher.clientId);
- this._send({
+ this._sendMessage({
channel: Faye.Channel.DISCONNECT,
- clientId: this._clientId
+ clientId: this._dispatcher.clientId
- }, function(response) {
- if (!response.successful) return;
- this._transport.close();
- delete this._transport;
+ }, {}, function(response) {
+ if (response.successful) this._dispatcher.close();
}, this);
- this.info('Clearing channel listeners for ?', this._clientId);
+ this.info('Clearing channel listeners for ?', this._dispatcher.clientId);
this._channels = new Faye.Channel.Set();
},
// Request Response
// MUST include: * channel MUST include: * channel
@@ -1272,26 +1251,26 @@
subscription.setDeferredStatus('succeeded');
return subscription;
}
this.connect(function() {
- this.info('Client ? attempting to subscribe to ?', this._clientId, channel);
+ this.info('Client ? attempting to subscribe to ?', this._dispatcher.clientId, channel);
if (!force) this._channels.subscribe([channel], callback, context);
- this._send({
+ this._sendMessage({
channel: Faye.Channel.SUBSCRIBE,
- clientId: this._clientId,
+ clientId: this._dispatcher.clientId,
subscription: channel
- }, function(response) {
+ }, {}, function(response) {
if (!response.successful) {
subscription.setDeferredStatus('failed', Faye.Error.parse(response.error));
return this._channels.unsubscribe(channel, callback, context);
}
var channels = [].concat(response.subscription);
- this.info('Subscription acknowledged for ? to ?', this._clientId, channels);
+ this.info('Subscription acknowledged for ? to ?', this._dispatcher.clientId, channels);
subscription.setDeferredStatus('succeeded');
}, this);
}, this);
return subscription;
@@ -1315,284 +1294,389 @@
var dead = this._channels.unsubscribe(channel, callback, context);
if (!dead) return;
this.connect(function() {
- this.info('Client ? attempting to unsubscribe from ?', this._clientId, channel);
+ this.info('Client ? attempting to unsubscribe from ?', this._dispatcher.clientId, channel);
- this._send({
+ this._sendMessage({
channel: Faye.Channel.UNSUBSCRIBE,
- clientId: this._clientId,
+ clientId: this._dispatcher.clientId,
subscription: channel
- }, function(response) {
+ }, {}, function(response) {
if (!response.successful) return;
var channels = [].concat(response.subscription);
- this.info('Unsubscription acknowledged for ? from ?', this._clientId, channels);
+ this.info('Unsubscription acknowledged for ? from ?', this._dispatcher.clientId, channels);
}, this);
}, this);
},
// Request Response
// MUST include: * channel MUST include: * channel
// * data * successful
// MAY include: * clientId MAY include: * id
// * id * error
// * ext * ext
- publish: function(channel, data) {
+ publish: function(channel, data, options) {
var publication = new Faye.Publication();
this.connect(function() {
- this.info('Client ? queueing published message to ?: ?', this._clientId, channel, data);
+ this.info('Client ? queueing published message to ?: ?', this._dispatcher.clientId, channel, data);
- this._send({
+ this._sendMessage({
channel: channel,
data: data,
- clientId: this._clientId
+ clientId: this._dispatcher.clientId
- }, function(response) {
+ }, options, function(response) {
if (response.successful)
publication.setDeferredStatus('succeeded');
else
publication.setDeferredStatus('failed', Faye.Error.parse(response.error));
}, this);
}, this);
return publication;
},
- receiveMessage: function(message) {
- var id = message.id, timeout, callback;
+ _sendMessage: function(message, options, callback, context) {
+ message.id = this._generateMessageId();
+ var timeout = this._advice.timeout
+ ? 1.2 * this._advice.timeout / 1000
+ : 1.2 * this._dispatcher.retry;
+
+ this.pipeThroughExtensions('outgoing', message, null, function(message) {
+ if (!message) return;
+ if (callback) this._responseCallbacks[message.id] = [callback, context];
+ this._dispatcher.sendMessage(message, timeout, options || {});
+ }, this);
+ },
+
+ _generateMessageId: function() {
+ this._messageId += 1;
+ if (this._messageId >= Math.pow(2,32)) this._messageId = 0;
+ return this._messageId.toString(36);
+ },
+
+ _receiveMessage: function(message) {
+ var id = message.id, callback;
+
if (message.successful !== undefined) {
callback = this._responseCallbacks[id];
delete this._responseCallbacks[id];
}
this.pipeThroughExtensions('incoming', message, null, function(message) {
if (!message) return;
-
if (message.advice) this._handleAdvice(message.advice);
this._deliverMessage(message);
-
if (callback) callback[0].call(callback[1], message);
}, this);
-
- if (this._transportUp === true) return;
- this._transportUp = true;
- this.trigger('transport:up');
},
- messageError: function(messages, immediate) {
- var retry = this._retry,
- self = this,
- id, message, timeout;
+ _handleAdvice: function(advice) {
+ Faye.extend(this._advice, advice);
+ this._dispatcher.timeout = this._advice.timeout / 1000;
- for (var i = 0, n = messages.length; i < n; i++) {
- message = messages[i];
- id = message.id;
+ if (this._advice.reconnect === this.HANDSHAKE && this._state !== this.DISCONNECTED) {
+ this._state = this.UNCONNECTED;
+ this._dispatcher.clientId = null;
+ this._cycleConnection();
+ }
+ },
- if (immediate)
- this._transportSend(message);
- else
- Faye.ENV.setTimeout(function() { self._transportSend(message) }, retry * 1000);
+ _deliverMessage: function(message) {
+ if (!message.channel || message.data === undefined) return;
+ this.info('Client ? calling listeners for ? with ?', this._dispatcher.clientId, message.channel, message.data);
+ this._channels.distributeMessage(message);
+ },
+
+ _cycleConnection: function() {
+ if (this._connectRequest) {
+ this._connectRequest = null;
+ this.info('Closed connection for ?', this._dispatcher.clientId);
}
+ var self = this;
+ Faye.ENV.setTimeout(function() { self.connect() }, this._advice.interval);
+ }
+});
- if (immediate || this._transportUp === false) return;
- this._transportUp = false;
- this.trigger('transport:down');
+Faye.extend(Faye.Client.prototype, Faye.Deferrable);
+Faye.extend(Faye.Client.prototype, Faye.Publisher);
+Faye.extend(Faye.Client.prototype, Faye.Logging);
+Faye.extend(Faye.Client.prototype, Faye.Extensible);
+
+Faye.Dispatcher = Faye.Class({
+ MAX_REQUEST_SIZE: 2048,
+ DEFAULT_RETRY: 5,
+
+ UP: 1,
+ DOWN: 2,
+
+ initialize: function(client, endpoint, options) {
+ this._client = client;
+ this.endpoint = Faye.URI.parse(endpoint);
+ this._alternates = options.endpoints || {};
+
+ this.ca = options.ca;
+ this.cookies = Faye.Cookies && new Faye.Cookies.CookieJar();
+ this._disabled = [];
+ this._envelopes = {};
+ this.headers = {};
+ this.retry = options.retry || this.DEFAULT_RETRY;
+ this._state = 0;
+ this.transports = {};
+
+ for (var type in this._alternates)
+ this._alternates[type] = Faye.URI.parse(this._alternates[type]);
+
+ this.maxRequestSize = this.MAX_REQUEST_SIZE;
},
- _selectTransport: function(transportTypes) {
+ endpointFor: function(connectionType) {
+ return this._alternates[connectionType] || this.endpoint;
+ },
+
+ disable: function(feature) {
+ this._disabled.push(feature);
+ },
+
+ setHeader: function(name, value) {
+ this.headers[name] = value;
+ },
+
+ close: function() {
+ var transport = this._transport;
+ delete this._transport;
+ if (transport) transport.close();
+ },
+
+ selectTransport: function(transportTypes) {
Faye.Transport.get(this, transportTypes, this._disabled, function(transport) {
this.debug('Selected ? transport for ?', transport.connectionType, Faye.URI.stringify(transport.endpoint));
if (transport === this._transport) return;
if (this._transport) this._transport.close();
this._transport = transport;
+ this.connectionType = transport.connectionType;
}, this);
},
- _send: function(message, callback, context) {
+ sendMessage: function(message, timeout, options) {
if (!this._transport) return;
- message.id = message.id || this._generateMessageId();
+ options = options || {};
- this.pipeThroughExtensions('outgoing', message, null, function(message) {
- if (!message) return;
- if (callback) this._responseCallbacks[message.id] = [callback, context];
- this._transportSend(message);
- }, this);
- },
+ var self = this,
+ id = message.id,
+ attempts = options.attempts,
+ deadline = options.deadline && new Date().getTime() + (options.deadline * 1000),
- _transportSend: function(message) {
- if (!this._transport) return;
+ envelope = this._envelopes[id] = this._envelopes[id] ||
+ {message: message, timeout: timeout, attempts: attempts, deadline: deadline};
- var timeout = 1.2 * (this._advice.timeout || this._retry * 1000),
- envelope = new Faye.Envelope(message, timeout);
+ if (envelope.request || envelope.timer) return;
- envelope.errback(function(immediate) {
- this.messageError([message], immediate);
- }, this);
+ if (this._attemptsExhausted(envelope) || this._deadlinePassed(envelope)) {
+ delete this._envelopes[id];
+ return;
+ }
- this._transport.send(envelope);
+ envelope.timer = Faye.ENV.setTimeout(function() {
+ self.handleError(message);
+ }, timeout * 1000);
+
+ envelope.request = this._transport.sendMessage(message);
},
- _generateMessageId: function() {
- this._messageId += 1;
- if (this._messageId >= Math.pow(2,32)) this._messageId = 0;
- return this._messageId.toString(36);
+ handleResponse: function(reply) {
+ var envelope = this._envelopes[reply.id];
+
+ if (reply.successful !== undefined && envelope) {
+ delete this._envelopes[reply.id];
+ Faye.ENV.clearTimeout(envelope.timer);
+ }
+
+ this.trigger('message', reply);
+
+ if (this._state === this.UP) return;
+ this._state = this.UP;
+ this._client.trigger('transport:up');
},
- _handleAdvice: function(advice) {
- Faye.extend(this._advice, advice);
+ handleError: function(message, immediate) {
+ var envelope = this._envelopes[message.id],
+ request = envelope && envelope.request,
+ self = this;
- if (this._advice.reconnect === this.HANDSHAKE && this._state !== this.DISCONNECTED) {
- this._state = this.UNCONNECTED;
- this._clientId = null;
- this._cycleConnection();
+ if (!request) return;
+
+ request.then(function(req) {
+ if (req && req.abort) req.abort();
+ });
+
+ Faye.ENV.clearTimeout(envelope.timer);
+ envelope.request = envelope.timer = null;
+
+ if (immediate) {
+ this.sendMessage(envelope.message, envelope.timeout);
+ } else {
+ envelope.timer = Faye.ENV.setTimeout(function() {
+ envelope.timer = null;
+ self.sendMessage(envelope.message, envelope.timeout);
+ }, this.retry * 1000);
}
+
+ if (this._state === this.DOWN) return;
+ this._state = this.DOWN;
+ this._client.trigger('transport:down');
},
- _deliverMessage: function(message) {
- if (!message.channel || message.data === undefined) return;
- this.info('Client ? calling listeners for ? with ?', this._clientId, message.channel, message.data);
- this._channels.distributeMessage(message);
+ _attemptsExhausted: function(envelope) {
+ if (envelope.attempts === undefined) return false;
+ envelope.attempts -= 1;
+ if (envelope.attempts >= 0) return false;
+ return true;
},
- _cycleConnection: function() {
- if (this._connectRequest) {
- this._connectRequest = null;
- this.info('Closed connection for ?', this._clientId);
- }
- var self = this;
- Faye.ENV.setTimeout(function() { self.connect() }, this._advice.interval);
+ _deadlinePassed: function(envelope) {
+ var deadline = envelope.deadline;
+ if (deadline === undefined) return false;
+ if (new Date().getTime() <= deadline) return false;
+ return true;
}
});
-Faye.extend(Faye.Client.prototype, Faye.Deferrable);
-Faye.extend(Faye.Client.prototype, Faye.Publisher);
-Faye.extend(Faye.Client.prototype, Faye.Logging);
-Faye.extend(Faye.Client.prototype, Faye.Extensible);
+Faye.extend(Faye.Dispatcher.prototype, Faye.Publisher);
+Faye.extend(Faye.Dispatcher.prototype, Faye.Logging);
Faye.Transport = Faye.extend(Faye.Class({
MAX_DELAY: 0,
batching: true,
- initialize: function(client, endpoint) {
- this._client = client;
- this.endpoint = endpoint;
- this._outbox = [];
+ initialize: function(dispatcher, endpoint) {
+ this._dispatcher = dispatcher;
+ this.endpoint = endpoint;
+ this._outbox = [];
},
close: function() {},
- encode: function(envelopes) {
+ encode: function(messages) {
return '';
},
- send: function(envelope) {
- var message = envelope.message;
-
+ sendMessage: function(message) {
this.debug('Client ? sending message to ?: ?',
- this._client._clientId, Faye.URI.stringify(this.endpoint), message);
+ this._dispatcher.clientId, Faye.URI.stringify(this.endpoint), message);
- if (!this.batching) return this.request([envelope]);
+ if (!this.batching) return Faye.Promise.fulfilled(this.request([message]));
- this._outbox.push(envelope);
+ this._outbox.push(message);
+ this._flushLargeBatch();
+ this._promise = this._promise || new Faye.Promise();
- if (message.channel === Faye.Channel.HANDSHAKE)
- return this.addTimeout('publish', 0.01, this.flush, this);
+ if (message.channel === Faye.Channel.HANDSHAKE) {
+ this.addTimeout('publish', 0.01, this._flush, this);
+ return this._promise;
+ }
if (message.channel === Faye.Channel.CONNECT)
this._connectMessage = message;
- this.flushLargeBatch();
- this.addTimeout('publish', this.MAX_DELAY, this.flush, this);
+ this.addTimeout('publish', this.MAX_DELAY, this._flush, this);
+ return this._promise;
},
- flush: function() {
+ _flush: function() {
this.removeTimeout('publish');
if (this._outbox.length > 1 && this._connectMessage)
this._connectMessage.advice = {timeout: 0};
- this.request(this._outbox);
+ Faye.Promise.fulfill(this._promise, this.request(this._outbox));
+ delete this._promise;
this._connectMessage = null;
this._outbox = [];
},
- flushLargeBatch: function() {
+ _flushLargeBatch: function() {
var string = this.encode(this._outbox);
- if (string.length < this._client.maxRequestSize) return;
+ if (string.length < this._dispatcher.maxRequestSize) return;
var last = this._outbox.pop();
- this.flush();
+ this._flush();
if (last) this._outbox.push(last);
},
- receive: function(envelopes, responses) {
- var n = envelopes.length;
- while (n--) envelopes[n].setDeferredStatus('succeeded');
+ _receive: function(replies) {
+ replies = [].concat(replies);
- responses = [].concat(responses);
+ this.debug('Client ? received from ? via ?: ?',
+ this._dispatcher.clientId, Faye.URI.stringify(this.endpoint), this.connectionType, replies);
- this.debug('Client ? received from ?: ?',
- this._client._clientId, Faye.URI.stringify(this.endpoint), responses);
-
- for (var i = 0, n = responses.length; i < n; i++)
- this._client.receiveMessage(responses[i]);
+ for (var i = 0, n = replies.length; i < n; i++)
+ this._dispatcher.handleResponse(replies[i]);
},
- handleError: function(envelopes, immediate) {
- var n = envelopes.length;
- while (n--) envelopes[n].setDeferredStatus('failed', immediate);
+ _handleError: function(messages, immediate) {
+ messages = [].concat(messages);
+
+ this.debug('Client ? failed to send to ? via ?: ?',
+ this._dispatcher.clientId, Faye.URI.stringify(this.endpoint), this.connectionType, messages);
+
+ for (var i = 0, n = messages.length; i < n; i++)
+ this._dispatcher.handleError(messages[i]);
},
_getCookies: function() {
- var cookies = this._client.cookies;
+ var cookies = this._dispatcher.cookies,
+ url = Faye.URI.stringify(this.endpoint);
+
if (!cookies) return '';
- return cookies.getCookies({
- domain: this.endpoint.hostname,
- path: this.endpoint.path,
- secure: this.endpoint.protocol === 'https:'
- }).toValueString();
+ return Faye.map(cookies.getCookiesSync(url), function(cookie) {
+ return cookie.cookieString();
+ }).join('; ');
},
_storeCookies: function(setCookie) {
- if (!setCookie || !this._client.cookies) return;
+ var cookies = this._dispatcher.cookies,
+ url = Faye.URI.stringify(this.endpoint),
+ cookie;
+
+ if (!setCookie || !cookies) return;
setCookie = [].concat(setCookie);
- var cookie;
for (var i = 0, n = setCookie.length; i < n; i++) {
- cookie = this._client.cookies.setCookie(setCookie[i]);
- cookie = cookie[0] || cookie;
- cookie.domain = cookie.domain || this.endpoint.hostname;
+ cookie = Faye.Cookies.Cookie.parse(setCookie[i]);
+ cookies.setCookieSync(cookie, url);
}
}
}), {
- get: function(client, allowed, disabled, callback, context) {
- var endpoint = client.endpoint;
+ get: function(dispatcher, allowed, disabled, callback, context) {
+ var endpoint = dispatcher.endpoint;
Faye.asyncEach(this._transports, function(pair, resume) {
var connType = pair[0], klass = pair[1],
- connEndpoint = client.endpoints[connType] || endpoint;
+ connEndpoint = dispatcher.endpointFor(connType);
if (Faye.indexOf(disabled, connType) >= 0)
return resume();
if (Faye.indexOf(allowed, connType) < 0) {
- klass.isUsable(client, connEndpoint, function() {});
+ klass.isUsable(dispatcher, connEndpoint, function() {});
return resume();
}
- klass.isUsable(client, connEndpoint, function(isUsable) {
+ klass.isUsable(dispatcher, connEndpoint, function(isUsable) {
if (!isUsable) return resume();
- var transport = klass.hasOwnProperty('create') ? klass.create(client, connEndpoint) : new klass(client, connEndpoint);
+ var transport = klass.hasOwnProperty('create') ? klass.create(dispatcher, connEndpoint) : new klass(dispatcher, connEndpoint);
callback.call(context, transport);
});
}, function() {
throw new Error('Could not find a usable connection type for ' + Faye.URI.stringify(endpoint));
});
@@ -2152,20 +2236,27 @@
this.callback(function() { callback.call(context, true) });
this.errback(function() { callback.call(context, false) });
this.connect();
},
- request: function(envelopes) {
+ request: function(messages) {
this._pending = this._pending || new Faye.Set();
- for (var i = 0, n = envelopes.length; i < n; i++) this._pending.add(envelopes[i]);
+ for (var i = 0, n = messages.length; i < n; i++) this._pending.add(messages[i]);
+ var promise = new Faye.Promise();
+
this.callback(function(socket) {
if (!socket) return;
- var messages = Faye.map(envelopes, function(e) { return e.message });
socket.send(Faye.toJSON(messages));
+ Faye.Promise.fulfill(promise, socket);
}, this);
+
this.connect();
+
+ return {
+ abort: function() { promise.then(function(ws) { ws.close() }) }
+ };
},
connect: function() {
if (Faye.Transport.WebSocket._unloaded) return;
@@ -2202,43 +2293,41 @@
var pending = self._pending ? self._pending.toArray() : [];
delete self._pending;
if (wasConnected) {
- self.handleError(pending, true);
+ self._handleError(pending, true);
} else if (self._everConnected) {
- self.handleError(pending);
+ self._handleError(pending);
} else {
self.setDeferredStatus('failed');
}
};
socket.onmessage = function(event) {
- var messages = JSON.parse(event.data),
- envelopes = [],
- envelope;
+ var replies = JSON.parse(event.data);
+ if (!replies) return;
- if (!messages) return;
- messages = [].concat(messages);
+ replies = [].concat(replies);
- for (var i = 0, n = messages.length; i < n; i++) {
- if (messages[i].successful === undefined) continue;
- envelope = self._pending.remove(messages[i]);
- if (envelope) envelopes.push(envelope);
+ for (var i = 0, n = replies.length; i < n; i++) {
+ if (replies[i].successful === undefined) continue;
+ self._pending.remove(replies[i]);
}
- self.receive(envelopes, messages);
+ self._receive(replies);
};
},
close: function() {
if (!this._socket) return;
this._socket.close();
},
_createSocket: function() {
var url = Faye.Transport.WebSocket.getSocketUrl(this.endpoint),
- options = {headers: Faye.copyObject(this._client.headers), ca: this._client.ca};
+ headers = Faye.copyObject(this._dispatcher.headers),
+ options = {headers: headers, ca: this._dispatcher.ca};
options.headers['Cookie'] = this._getCookies();
if (Faye.WebSocket) return new Faye.WebSocket.Client(url, [], options);
if (Faye.ENV.MozWebSocket) return new MozWebSocket(url);
@@ -2246,53 +2335,53 @@
},
_ping: function() {
if (!this._socket) return;
this._socket.send('[]');
- this.addTimeout('ping', this._client._advice.timeout/2000, this._ping, this);
+ this.addTimeout('ping', this._dispatcher.timeout / 2, this._ping, this);
}
}), {
PROTOCOLS: {
'http:': 'ws:',
'https:': 'wss:'
},
- create: function(client, endpoint) {
- var sockets = client.transports.websocket = client.transports.websocket || {};
- sockets[endpoint.href] = sockets[endpoint.href] || new this(client, endpoint);
+ create: function(dispatcher, endpoint) {
+ var sockets = dispatcher.transports.websocket = dispatcher.transports.websocket || {};
+ sockets[endpoint.href] = sockets[endpoint.href] || new this(dispatcher, endpoint);
return sockets[endpoint.href];
},
getSocketUrl: function(endpoint) {
endpoint = Faye.copyObject(endpoint);
endpoint.protocol = this.PROTOCOLS[endpoint.protocol];
return Faye.URI.stringify(endpoint);
},
- isUsable: function(client, endpoint, callback, context) {
- this.create(client, endpoint).isUsable(callback, context);
+ isUsable: function(dispatcher, endpoint, callback, context) {
+ this.create(dispatcher, endpoint).isUsable(callback, context);
}
});
Faye.extend(Faye.Transport.WebSocket.prototype, Faye.Deferrable);
Faye.Transport.register('websocket', Faye.Transport.WebSocket);
-if (Faye.Event)
+if (Faye.Event && Faye.ENV.onbeforeunload !== undefined)
Faye.Event.on(Faye.ENV, 'beforeunload', function() {
Faye.Transport.WebSocket._unloaded = true;
});
Faye.Transport.EventSource = Faye.extend(Faye.Class(Faye.Transport, {
- initialize: function(client, endpoint) {
- Faye.Transport.prototype.initialize.call(this, client, endpoint);
+ initialize: function(dispatcher, endpoint) {
+ Faye.Transport.prototype.initialize.call(this, dispatcher, endpoint);
if (!Faye.ENV.EventSource) return this.setDeferredStatus('failed');
- this._xhr = new Faye.Transport.XHR(client, endpoint);
+ this._xhr = new Faye.Transport.XHR(dispatcher, endpoint);
endpoint = Faye.copyObject(endpoint);
- endpoint.pathname += '/' + client._clientId;
+ endpoint.pathname += '/' + dispatcher.clientId;
var socket = new EventSource(Faye.URI.stringify(endpoint)),
self = this;
socket.onopen = function() {
@@ -2300,19 +2389,19 @@
self.setDeferredStatus('succeeded');
};
socket.onerror = function() {
if (self._everConnected) {
- self._client.messageError([]);
+ self._handleError([]);
} else {
self.setDeferredStatus('failed');
socket.close();
}
};
socket.onmessage = function(event) {
- self.receive([], JSON.parse(event.data));
+ self._receive(JSON.parse(event.data));
};
this._socket = socket;
},
@@ -2326,114 +2415,113 @@
isUsable: function(callback, context) {
this.callback(function() { callback.call(context, true) });
this.errback(function() { callback.call(context, false) });
},
- encode: function(envelopes) {
- return this._xhr.encode(envelopes);
+ encode: function(messages) {
+ return this._xhr.encode(messages);
},
- request: function(envelopes) {
- this._xhr.request(envelopes);
+ request: function(messages) {
+ return this._xhr.request(messages);
}
}), {
- isUsable: function(client, endpoint, callback, context) {
- var id = client._clientId;
+ isUsable: function(dispatcher, endpoint, callback, context) {
+ var id = dispatcher.clientId;
if (!id) return callback.call(context, false);
- Faye.Transport.XHR.isUsable(client, endpoint, function(usable) {
+ Faye.Transport.XHR.isUsable(dispatcher, endpoint, function(usable) {
if (!usable) return callback.call(context, false);
- this.create(client, endpoint).isUsable(callback, context);
+ this.create(dispatcher, endpoint).isUsable(callback, context);
}, this);
},
- create: function(client, endpoint) {
- var sockets = client.transports.eventsource = client.transports.eventsource || {},
- id = client._clientId;
+ create: function(dispatcher, endpoint) {
+ var sockets = dispatcher.transports.eventsource = dispatcher.transports.eventsource || {},
+ id = dispatcher.clientId;
endpoint = Faye.copyObject(endpoint);
endpoint.pathname += '/' + (id || '');
var url = Faye.URI.stringify(endpoint);
- sockets[url] = sockets[url] || new this(client, endpoint);
+ sockets[url] = sockets[url] || new this(dispatcher, endpoint);
return sockets[url];
}
});
Faye.extend(Faye.Transport.EventSource.prototype, Faye.Deferrable);
Faye.Transport.register('eventsource', Faye.Transport.EventSource);
Faye.Transport.XHR = Faye.extend(Faye.Class(Faye.Transport, {
- encode: function(envelopes) {
- var messages = Faye.map(envelopes, function(e) { return e.message });
+ encode: function(messages) {
return Faye.toJSON(messages);
},
- request: function(envelopes) {
- var path = this.endpoint.path,
+ request: function(messages) {
+ var href = this.endpoint.href,
xhr = Faye.ENV.ActiveXObject ? new ActiveXObject('Microsoft.XMLHTTP') : new XMLHttpRequest(),
self = this;
- xhr.open('POST', path, true);
+ xhr.open('POST', href, true);
xhr.setRequestHeader('Content-Type', 'application/json');
xhr.setRequestHeader('Pragma', 'no-cache');
xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
- var headers = this._client.headers;
+ var headers = this._dispatcher.headers;
for (var key in headers) {
if (!headers.hasOwnProperty(key)) continue;
xhr.setRequestHeader(key, headers[key]);
}
var abort = function() { xhr.abort() };
- Faye.Event.on(Faye.ENV, 'beforeunload', abort);
+ if (Faye.ENV.onbeforeunload !== undefined) Faye.Event.on(Faye.ENV, 'beforeunload', abort);
xhr.onreadystatechange = function() {
if (!xhr || xhr.readyState !== 4) return;
- var parsedMessage = null,
- status = xhr.status,
- text = xhr.responseText,
- successful = (status >= 200 && status < 300) || status === 304 || status === 1223;
+ var replies = null,
+ status = xhr.status,
+ text = xhr.responseText,
+ successful = (status >= 200 && status < 300) || status === 304 || status === 1223;
- Faye.Event.detach(Faye.ENV, 'beforeunload', abort);
+ if (Faye.ENV.onbeforeunload !== undefined) Faye.Event.detach(Faye.ENV, 'beforeunload', abort);
xhr.onreadystatechange = function() {};
xhr = null;
- if (!successful) return self.handleError(envelopes);
+ if (!successful) return self._handleError(messages);
try {
- parsedMessage = JSON.parse(text);
+ replies = JSON.parse(text);
} catch (e) {}
- if (parsedMessage)
- self.receive(envelopes, parsedMessage);
+ if (replies)
+ self._receive(replies);
else
- self.handleError(envelopes);
+ self._handleError(messages);
};
- xhr.send(this.encode(envelopes));
+ xhr.send(this.encode(messages));
+ return xhr;
}
}), {
- isUsable: function(client, endpoint, callback, context) {
+ isUsable: function(dispatcher, endpoint, callback, context) {
callback.call(context, Faye.URI.isSameOrigin(endpoint));
}
});
Faye.Transport.register('long-polling', Faye.Transport.XHR);
Faye.Transport.CORS = Faye.extend(Faye.Class(Faye.Transport, {
- encode: function(envelopes) {
- var messages = Faye.map(envelopes, function(e) { return e.message });
+ encode: function(messages) {
return 'message=' + encodeURIComponent(Faye.toJSON(messages));
},
- request: function(envelopes) {
+ request: function(messages) {
var xhrClass = Faye.ENV.XDomainRequest ? XDomainRequest : XMLHttpRequest,
xhr = new xhrClass(),
- headers = this._client.headers,
+ headers = this._dispatcher.headers,
self = this,
key;
xhr.open('POST', Faye.URI.stringify(this.endpoint), true);
@@ -2450,33 +2538,34 @@
xhr.onload = xhr.onerror = xhr.ontimeout = xhr.onprogress = null;
xhr = null;
};
xhr.onload = function() {
- var parsedMessage = null;
+ var replies = null;
try {
- parsedMessage = JSON.parse(xhr.responseText);
+ replies = JSON.parse(xhr.responseText);
} catch (e) {}
cleanUp();
- if (parsedMessage)
- self.receive(envelopes, parsedMessage);
+ if (replies)
+ self._receive(replies);
else
- self.handleError(envelopes);
+ self._handleError(messages);
};
xhr.onerror = xhr.ontimeout = function() {
cleanUp();
- self.handleError(envelopes);
+ self._handleError(messages);
};
xhr.onprogress = function() {};
- xhr.send(this.encode(envelopes));
+ xhr.send(this.encode(messages));
+ return xhr;
}
}), {
- isUsable: function(client, endpoint, callback, context) {
+ isUsable: function(dispatcher, endpoint, callback, context) {
if (Faye.URI.isSameOrigin(endpoint))
return callback.call(context, false);
if (Faye.ENV.XDomainRequest)
return callback.call(context, endpoint.protocol === Faye.ENV.location.protocol);
@@ -2490,52 +2579,61 @@
});
Faye.Transport.register('cross-origin-long-polling', Faye.Transport.CORS);
Faye.Transport.JSONP = Faye.extend(Faye.Class(Faye.Transport, {
- encode: function(envelopes) {
- var messages = Faye.map(envelopes, function(e) { return e.message });
+ encode: function(messages) {
var url = Faye.copyObject(this.endpoint);
url.query.message = Faye.toJSON(messages);
url.query.jsonp = '__jsonp' + Faye.Transport.JSONP._cbCount + '__';
return Faye.URI.stringify(url);
},
- request: function(envelopes) {
- var messages = Faye.map(envelopes, function(e) { return e.message }),
- head = document.getElementsByTagName('head')[0],
+ request: function(messages) {
+ var head = document.getElementsByTagName('head')[0],
script = document.createElement('script'),
callbackName = Faye.Transport.JSONP.getCallbackName(),
endpoint = Faye.copyObject(this.endpoint),
self = this;
endpoint.query.message = Faye.toJSON(messages);
endpoint.query.jsonp = callbackName;
- Faye.ENV[callbackName] = function(data) {
+ var cleanup = function() {
if (!Faye.ENV[callbackName]) return false;
Faye.ENV[callbackName] = undefined;
try { delete Faye.ENV[callbackName] } catch (e) {}
script.parentNode.removeChild(script);
- self.receive(envelopes, data);
};
+ Faye.ENV[callbackName] = function(replies) {
+ cleanup();
+ self._receive(replies);
+ };
+
script.type = 'text/javascript';
script.src = Faye.URI.stringify(endpoint);
head.appendChild(script);
+
+ script.onerror = function() {
+ cleanup();
+ self._handleError(messages);
+ };
+
+ return {abort: cleanup};
}
}), {
_cbCount: 0,
getCallbackName: function() {
this._cbCount += 1;
return '__jsonp' + this._cbCount + '__';
},
- isUsable: function(client, endpoint, callback, context) {
+ isUsable: function(dispatcher, endpoint, callback, context) {
callback.call(context, true);
}
});
Faye.Transport.register('callback-polling', Faye.Transport.JSONP);
-})();
+})();
\ No newline at end of file