Author: ceposta
Date: Mon Dec 3 13:06:35 2012
New Revision: 1416498
URL: http://svn.apache.org/viewvc?rev=1416498&view=rev
Log:
https://issues.apache.org/jira/browse/APLO-273 - Update the websocket example to use the latest
stomp.js which supports STOMP 1.1 (including heart-beating).
Jeff Mensil provided patch. Thank you!
Modified:
activemq/activemq-apollo/trunk/apollo-distro/src/main/release/examples/websocket/js/stomp.js
Modified: activemq/activemq-apollo/trunk/apollo-distro/src/main/release/examples/websocket/js/stomp.js
URL: http://svn.apache.org/viewvc/activemq/activemq-apollo/trunk/apollo-distro/src/main/release/examples/websocket/js/stomp.js?rev=1416498&r1=1416497&r2=1416498&view=diff
==============================================================================
--- activemq/activemq-apollo/trunk/apollo-distro/src/main/release/examples/websocket/js/stomp.js
(original)
+++ activemq/activemq-apollo/trunk/apollo-distro/src/main/release/examples/websocket/js/stomp.js
Mon Dec 3 13:06:35 2012
@@ -1,55 +1,45 @@
-// Generated by CoffeeScript 1.3.1
+// Generated by CoffeeScript 1.3.3
+(function() {
+ var Byte, Client, Frame, Stomp,
+ __hasProp = {}.hasOwnProperty;
-/*
-Copyright (C) 2010 Jeff Mesnil -- http://jmesnil.net/
-Copyright (C) 2012 FuseSource, Inc. -- http://fusesource.com
-*/
+ Byte = {
+ LF: '\x0A',
+ NULL: '\x00'
+ };
+ Frame = (function() {
-(function() {
- var Client, Stomp, WebSocketStompMock,
- __hasProp = {}.hasOwnProperty;
+ function Frame(command, headers, body) {
+ this.command = command;
+ this.headers = headers != null ? headers : {};
+ this.body = body != null ? body : '';
+ }
- Stomp = {
- frame: function(command, headers, body) {
- if (headers == null) {
- headers = [];
+ Frame.prototype.toString = function() {
+ var lines, name, value, _ref;
+ lines = [this.command];
+ _ref = this.headers;
+ for (name in _ref) {
+ if (!__hasProp.call(_ref, name)) continue;
+ value = _ref[name];
+ lines.push("" + name + ":" + value);
}
- if (body == null) {
- body = '';
+ if (this.body) {
+ lines.push("content-length:" + ('' + this.body).length);
}
- return {
- command: command,
- headers: headers,
- body: body,
- id: headers.id,
- receipt: headers.receipt,
- transaction: headers.transaction,
- destination: headers.destination,
- subscription: headers.subscription,
- error: null,
- toString: function() {
- var lines, name, value;
- lines = [command];
- for (name in headers) {
- if (!__hasProp.call(headers, name)) continue;
- value = headers[name];
- lines.push("" + name + ":" + value);
- }
- lines.push('\n' + body);
- return lines.join('\n');
- }
- };
- },
- unmarshal: function(data) {
- var body, chr, command, divider, headerLines, headers, i, idx, line, trim, _i, _j,
_ref, _ref1, _ref2;
- divider = data.search(/\n\n/);
- headerLines = data.substring(0, divider).split('\n');
+ lines.push(Byte.LF + this.body);
+ return lines.join(Byte.LF);
+ };
+
+ Frame._unmarshallSingle = function(data) {
+ var body, chr, command, divider, headerLines, headers, i, idx, len, line, start, trim,
_i, _j, _ref, _ref1;
+ divider = data.search(RegExp("" + Byte.LF + Byte.LF));
+ headerLines = data.substring(0, divider).split(Byte.LF);
command = headerLines.shift();
headers = {};
- body = '';
trim = function(str) {
- return str.replace(/^\s+/g, '').replace(/\s+$/g, '');
+ return str.replace(/^\s+|\s+$/g, '');
};
line = idx = null;
for (i = _i = 0, _ref = headerLines.length; 0 <= _ref ? _i < _ref : _i > _ref;
i = 0 <= _ref ? ++_i : --_i) {
@@ -57,134 +47,221 @@ Copyright (C) 2012 FuseSource, Inc. -- h
idx = line.indexOf(':');
headers[trim(line.substring(0, idx))] = trim(line.substring(idx + 1));
}
- chr = null;
- for (i = _j = _ref1 = divider + 2, _ref2 = data.length; _ref1 <= _ref2 ? _j <
_ref2 : _j > _ref2; i = _ref1 <= _ref2 ? ++_j : --_j) {
- chr = data.charAt(i);
- if (chr === '\x00') {
- break;
+ body = '';
+ start = divider + 2;
+ if (headers['content-length']) {
+ len = parseInt(headers['content-length']);
+ body = ('' + data).substring(start, start + len);
+ } else {
+ chr = null;
+ for (i = _j = start, _ref1 = data.length; start <= _ref1 ? _j < _ref1 : _j
> _ref1; i = start <= _ref1 ? ++_j : --_j) {
+ chr = data.charAt(i);
+ if (chr === Byte.NULL) {
+ break;
+ }
+ body += chr;
}
- body += chr;
}
- return Stomp.frame(command, headers, body);
- },
- unmarshal_multi: function(multi_datas) {
- var data, datas;
- datas = (function() {
+ return new Frame(command, headers, body);
+ };
+
+ Frame.unmarshall = function(datas) {
+ var data;
+ return (function() {
var _i, _len, _ref, _results;
- _ref = multi_datas.split(/\x00\n*/);
+ _ref = datas.split(RegExp("" + Byte.NULL + Byte.LF + "*"));
_results = [];
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
data = _ref[_i];
- if (data && data.length > 0) {
- _results.push(Stomp.unmarshal(data));
+ if ((data != null ? data.length : void 0) > 0) {
+ _results.push(Frame._unmarshallSingle(data));
}
}
return _results;
})();
- return datas;
- },
- marshal: function(command, headers, body) {
- return Stomp.frame(command, headers, body).toString() + '\x00';
- },
- client: function(url) {
- return new Client(url);
- }
- };
+ };
- Client = (function() {
+ Frame.marshall = function(command, headers, body) {
+ var frame;
+ frame = new Frame(command, headers, body);
+ return frame.toString() + Byte.NULL;
+ };
+
+ return Frame;
+
+ })();
- Client.name = 'Client';
+ Client = (function() {
- function Client(url) {
- this.url = url;
+ function Client(ws) {
+ this.ws = ws;
+ this.ws.binaryType = "arraybuffer";
this.counter = 0;
this.connected = false;
+ this.heartbeat = {
+ outgoing: 10000,
+ incoming: 10000
+ };
this.subscriptions = {};
}
Client.prototype._transmit = function(command, headers, body) {
var out;
- out = Stomp.marshal(command, headers, body);
+ out = Frame.marshall(command, headers, body);
if (typeof this.debug === "function") {
this.debug(">>> " + out);
}
return this.ws.send(out);
};
- Client.prototype.connect = function(login_, passcode_, connectCallback, errorCallback)
{
- var klass,
+ Client.prototype._setupHeartbeat = function(headers) {
+ var serverIncoming, serverOutgoing, ttl, v, _ref, _ref1,
_this = this;
+ if ((_ref = headers.version) !== Stomp.VERSIONS.V1_1 && _ref !== Stomp.VERSIONS.V1_2)
{
+ return;
+ }
+ _ref1 = (function() {
+ var _i, _len, _ref1, _results;
+ _ref1 = headers['heart-beat'].split(",");
+ _results = [];
+ for (_i = 0, _len = _ref1.length; _i < _len; _i++) {
+ v = _ref1[_i];
+ _results.push(parseInt(v));
+ }
+ return _results;
+ })(), serverOutgoing = _ref1[0], serverIncoming = _ref1[1];
+ if (!(this.heartbeat.outgoing === 0 || serverIncoming === 0)) {
+ ttl = Math.max(this.heartbeat.outgoing, serverIncoming);
+ if (typeof this.debug === "function") {
+ this.debug("send PING every " + ttl + "ms");
+ }
+ this.pinger = typeof window !== "undefined" && window !== null ? window.setInterval(function()
{
+ _this.ws.send(Byte.LF);
+ return typeof _this.debug === "function" ? _this.debug(">>> PING") : void
0;
+ }, ttl) : void 0;
+ }
+ if (!(this.heartbeat.incoming === 0 || serverOutgoing === 0)) {
+ ttl = Math.max(this.heartbeat.incoming, serverOutgoing);
+ if (typeof this.debug === "function") {
+ this.debug("check PONG every " + ttl + "ms");
+ }
+ return this.ponger = typeof window !== "undefined" && window !== null ? window.setInterval(function()
{
+ var delta;
+ delta = Date.now() - _this.serverActivity;
+ if (delta > ttl * 2) {
+ if (typeof _this.debug === "function") {
+ _this.debug("did not receive server activity for the last " + delta + "ms");
+ }
+ return _this._cleanUp();
+ }
+ }, ttl) : void 0;
+ }
+ };
+
+ Client.prototype.connect = function(login, passcode, connectCallback, errorCallback,
vhost) {
+ var _this = this;
+ this.connectCallback = connectCallback;
if (typeof this.debug === "function") {
this.debug("Opening Web Socket...");
}
- klass = WebSocketStompMock || WebSocket;
- this.ws = new klass(this.url);
- this.ws.binaryType = "arraybuffer";
this.ws.onmessage = function(evt) {
- var data, frame, i, onreceive, view, _i, _len, _ref, _results;
- data = (function() {
- var _i, _len;
- if (evt.data instanceof ArrayBuffer) {
- view = new Uint8Array(evt.data);
- if (typeof this.debug === "function") {
- this.debug('--- got data length: ' + view.length);
- }
- data = "";
- for (_i = 0, _len = view.length; _i < _len; _i++) {
- i = view[_i];
- data += String.fromCharCode(i);
- }
- return data;
- } else {
- return evt.data;
+ var arr, c, data, frame, onreceive, _i, _len, _ref, _results;
+ data = typeof ArrayBuffer !== 'undefined' && evt.data instanceof ArrayBuffer
? (arr = new Uint8Array(evt.data), typeof _this.debug === "function" ? _this.debug("--- got
data length: " + arr.length) : void 0, ((function() {
+ var _i, _len, _results;
+ _results = [];
+ for (_i = 0, _len = arr.length; _i < _len; _i++) {
+ c = arr[_i];
+ _results.push(String.fromCharCode(c));
+ }
+ return _results;
+ })()).join('')) : evt.data;
+ _this.serverActivity = Date.now();
+ if (data === Byte.LF) {
+ if (typeof _this.debug === "function") {
+ _this.debug("<<< PONG");
}
- }).call(_this);
+ return;
+ }
if (typeof _this.debug === "function") {
- _this.debug('<<< ' + data);
+ _this.debug("<<< " + data);
}
- _ref = Stomp.unmarshal_multi(data);
+ _ref = Frame.unmarshall(data);
_results = [];
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
frame = _ref[_i];
- if (frame.command === "CONNECTED" && connectCallback) {
- _this.connected = true;
- _results.push(connectCallback(frame));
- } else if (frame.command === "MESSAGE") {
- onreceive = _this.subscriptions[frame.headers.subscription];
- _results.push(typeof onreceive === "function" ? onreceive(frame) : void 0);
- } else if (frame.command === "ERROR") {
- _results.push(typeof errorCallback === "function" ? errorCallback(frame) : void
0);
- } else {
- _results.push(typeof _this.debug === "function" ? _this.debug("Unhandled frame:
" + frame) : void 0);
+ switch (frame.command) {
+ case "CONNECTED":
+ if (typeof _this.debug === "function") {
+ _this.debug("connected to server " + frame.headers.server);
+ }
+ _this.connected = true;
+ _this._setupHeartbeat(frame.headers);
+ _results.push(typeof _this.connectCallback === "function" ? _this.connectCallback(frame)
: void 0);
+ break;
+ case "MESSAGE":
+ onreceive = _this.subscriptions[frame.headers.subscription];
+ _results.push(typeof onreceive === "function" ? onreceive(frame) : void 0);
+ break;
+ case "RECEIPT":
+ _results.push(typeof _this.onreceipt === "function" ? _this.onreceipt(frame)
: void 0);
+ break;
+ case "ERROR":
+ _results.push(typeof errorCallback === "function" ? errorCallback(frame) :
void 0);
+ break;
+ default:
+ _results.push(typeof _this.debug === "function" ? _this.debug("Unhandled frame:
" + frame) : void 0);
}
}
return _results;
};
this.ws.onclose = function() {
var msg;
- msg = "Whoops! Lost connection to " + _this.url;
+ msg = "Whoops! Lost connection to " + _this.ws.url;
if (typeof _this.debug === "function") {
_this.debug(msg);
}
return typeof errorCallback === "function" ? errorCallback(msg) : void 0;
};
- this.ws.onopen = function() {
+ return this.ws.onopen = function() {
+ var headers;
if (typeof _this.debug === "function") {
_this.debug('Web Socket Opened...');
}
- return _this._transmit("CONNECT", {
- login: login_,
- passcode: passcode_
- });
+ headers = {
+ "accept-version": Stomp.VERSIONS.supportedVersions(),
+ "heart-beat": [_this.heartbeat.outgoing, _this.heartbeat.incoming].join(',')
+ };
+ if (vhost) {
+ headers.host = vhost;
+ }
+ if (login) {
+ headers.login = login;
+ }
+ if (passcode) {
+ headers.passcode = passcode;
+ }
+ return _this._transmit("CONNECT", headers);
};
- return this.connectCallback = connectCallback;
};
Client.prototype.disconnect = function(disconnectCallback) {
this._transmit("DISCONNECT");
+ this.ws.onclose = null;
+ this._cleanUp();
+ return typeof disconnectCallback === "function" ? disconnectCallback() : void 0;
+ };
+
+ Client.prototype._cleanUp = function() {
this.ws.close();
this.connected = false;
- return typeof disconnectCallback === "function" ? disconnectCallback() : void 0;
+ if (this.pinger) {
+ if (typeof window !== "undefined" && window !== null) {
+ window.clearInterval(this.pinger);
+ }
+ }
+ if (this.ponger) {
+ return typeof window !== "undefined" && window !== null ? window.clearInterval(this.ponger)
: void 0;
+ }
};
Client.prototype.send = function(destination, headers, body) {
@@ -199,68 +276,95 @@ Copyright (C) 2012 FuseSource, Inc. -- h
};
Client.prototype.subscribe = function(destination, callback, headers) {
- var id;
if (headers == null) {
headers = {};
}
- id = "sub-" + this.counter++;
+ if (!headers.id) {
+ headers.id = "sub-" + this.counter++;
+ }
headers.destination = destination;
- headers.id = id;
- this.subscriptions[id] = callback;
+ this.subscriptions[headers.id] = callback;
this._transmit("SUBSCRIBE", headers);
- return id;
+ return headers.id;
};
- Client.prototype.unsubscribe = function(id, headers) {
- if (headers == null) {
- headers = {};
- }
- headers.id = id;
+ Client.prototype.unsubscribe = function(id) {
delete this.subscriptions[id];
- return this._transmit("UNSUBSCRIBE", headers);
+ return this._transmit("UNSUBSCRIBE", {
+ id: id
+ });
};
- Client.prototype.begin = function(transaction, headers) {
- if (headers == null) {
- headers = {};
- }
- headers.transaction = transaction;
- return this._transmit("BEGIN", headers);
+ Client.prototype.begin = function(transaction) {
+ return this._transmit("BEGIN", {
+ transaction: transaction
+ });
};
- Client.prototype.commit = function(transaction, headers) {
- if (headers == null) {
- headers = {};
- }
- headers.transaction = transaction;
- return this._transmit("COMMIT", headers);
+ Client.prototype.commit = function(transaction) {
+ return this._transmit("COMMIT", {
+ transaction: transaction
+ });
+ };
+
+ Client.prototype.abort = function(transaction) {
+ return this._transmit("ABORT", {
+ transaction: transaction
+ });
};
- Client.prototype.abort = function(transaction, headers) {
+ Client.prototype.ack = function(messageID, subscription, headers) {
if (headers == null) {
headers = {};
}
- headers.transaction = transaction;
- return this._transmit("ABORT", headers);
+ headers["message-id"] = messageID;
+ headers.subscription = subscription;
+ return this._transmit("ACK", headers);
};
- Client.prototype.ack = function(message_id, headers) {
+ Client.prototype.nack = function(messageID, subscription, headers) {
if (headers == null) {
headers = {};
}
- headers["message-id"] = message_id;
- return this._transmit("ACK", headers);
+ headers["message-id"] = messageID;
+ headers.subscription = subscription;
+ return this._transmit("NACK", headers);
};
return Client;
})();
+ Stomp = {
+ libVersion: "2.0.0-next",
+ VERSIONS: {
+ V1_0: '1.0',
+ V1_1: '1.1',
+ V1_2: '1.2',
+ supportedVersions: function() {
+ return '1.1,1.0';
+ }
+ },
+ client: function(url, protocols) {
+ var klass, ws;
+ if (protocols == null) {
+ protocols = ['v10.stomp', 'v11.stomp'];
+ }
+ klass = Stomp.WebSocketClass || WebSocket;
+ ws = new klass(url, protocols);
+ return new Client(ws);
+ },
+ over: function(ws) {
+ return new Client(ws);
+ },
+ Frame: Frame
+ };
+
if (typeof window !== "undefined" && window !== null) {
window.Stomp = Stomp;
} else {
exports.Stomp = Stomp;
- WebSocketStompMock = require('./test/server.mock.js').StompServerMock;
+ Stomp.WebSocketClass = require('./test/server.mock.js').StompServerMock;
}
}).call(this);
|