mirror of
https://github.com/moparisthebest/kaiwa
synced 2024-11-22 17:22:22 -05:00
Add disco caps caching, typing notifications.
This commit is contained in:
parent
c54bb6f7fe
commit
69c4c33b8e
@ -4,12 +4,55 @@
|
|||||||
var crypto = XMPP.crypto;
|
var crypto = XMPP.crypto;
|
||||||
|
|
||||||
var _ = require('underscore');
|
var _ = require('underscore');
|
||||||
|
var async = require('async');
|
||||||
var log = require('andlog');
|
var log = require('andlog');
|
||||||
var Contact = require('../models/contact');
|
var Contact = require('../models/contact');
|
||||||
var Resource = require('../models/resource');
|
var Resource = require('../models/resource');
|
||||||
var Message = require('../models/message');
|
var Message = require('../models/message');
|
||||||
|
|
||||||
|
|
||||||
|
var discoCapsQueue = async.queue(function (pres, cb) {
|
||||||
|
var jid = pres.from;
|
||||||
|
var caps = pres.caps;
|
||||||
|
|
||||||
|
log.debug('Checking storage for ' + caps.ver);
|
||||||
|
|
||||||
|
var contact = me.getContact(jid);
|
||||||
|
var resource = null;
|
||||||
|
if (contact) {
|
||||||
|
resource = contact.resources.get(jid);
|
||||||
|
}
|
||||||
|
|
||||||
|
app.storage.disco.get(caps.ver, function (err, existing) {
|
||||||
|
log.debug(err, existing);
|
||||||
|
if (existing) {
|
||||||
|
log.debug('Already found info for ' + caps.ver);
|
||||||
|
if (resource) resource.discoInfo = existing;
|
||||||
|
return cb();
|
||||||
|
}
|
||||||
|
log.debug('getting info for ' + caps.ver + ' from ' + jid);
|
||||||
|
client.getDiscoInfo(jid, caps.node + '#' + caps.ver, function (err, result) {
|
||||||
|
log.debug(caps.ver, err, result);
|
||||||
|
if (err) {
|
||||||
|
log.debug('Couldnt get info for ' + caps.ver);
|
||||||
|
return cb();
|
||||||
|
}
|
||||||
|
if (client.verifyVerString(result.discoInfo, caps.hash, caps.ver)) {
|
||||||
|
log.debug('Saving info for ' + caps.ver);
|
||||||
|
var data = result.discoInfo.toJSON();
|
||||||
|
app.storage.disco.add(caps.ver, data, function () {
|
||||||
|
if (resource) resource.discoInfo = existing;
|
||||||
|
cb();
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
log.debug('Couldnt verify info for ' + caps.ver + ' from ' + jid);
|
||||||
|
cb();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
module.exports = function (client, app) {
|
module.exports = function (client, app) {
|
||||||
|
|
||||||
client.on('*', function (name, data) {
|
client.on('*', function (name, data) {
|
||||||
@ -39,7 +82,7 @@ module.exports = function (client, app) {
|
|||||||
});
|
});
|
||||||
|
|
||||||
client.on('auth:failed', function () {
|
client.on('auth:failed', function () {
|
||||||
console.log('auth failed');
|
log.debug('auth failed');
|
||||||
window.location = '/login';
|
window.location = '/login';
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -54,17 +97,18 @@ module.exports = function (client, app) {
|
|||||||
app.storage.rosterver.set(me.barejid, resp.roster.ver);
|
app.storage.rosterver.set(me.barejid, resp.roster.ver);
|
||||||
|
|
||||||
_.each(resp.roster.items, function (item) {
|
_.each(resp.roster.items, function (item) {
|
||||||
console.log(item);
|
|
||||||
me.setContact(item, true);
|
me.setContact(item, true);
|
||||||
});
|
});
|
||||||
|
|
||||||
client.updateCaps();
|
var caps = client.updateCaps();
|
||||||
|
app.storage.disco.add(caps.ver, caps.discoInfo, function () {
|
||||||
client.sendPresence({
|
client.sendPresence({
|
||||||
caps: client.disco.caps
|
caps: client.disco.caps
|
||||||
});
|
});
|
||||||
client.enableCarbons();
|
client.enableCarbons();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
||||||
client.on('roster:update', function (iq) {
|
client.on('roster:update', function (iq) {
|
||||||
iq = iq.toJSON();
|
iq = iq.toJSON();
|
||||||
@ -214,4 +258,11 @@ module.exports = function (client, app) {
|
|||||||
|
|
||||||
client.emit('message', msg);
|
client.emit('message', msg);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
client.on('disco:caps', function (pres) {
|
||||||
|
if (pres.from !== client.jid && pres.caps.hash) {
|
||||||
|
log.debug('Caps from ' + pres.from + ' ver: ' + pres.caps.ver);
|
||||||
|
discoCapsQueue.push(pres);
|
||||||
|
}
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
@ -36,6 +36,7 @@ var WildEmitter = require('wildemitter');
|
|||||||
var _ = require('../vendor/lodash');
|
var _ = require('../vendor/lodash');
|
||||||
var async = require('async');
|
var async = require('async');
|
||||||
var uuid = require('node-uuid');
|
var uuid = require('node-uuid');
|
||||||
|
var paddle = require('paddle');
|
||||||
var SASL = require('./stanza/sasl');
|
var SASL = require('./stanza/sasl');
|
||||||
var Message = require('./stanza/message');
|
var Message = require('./stanza/message');
|
||||||
var Presence = require('./stanza/presence');
|
var Presence = require('./stanza/presence');
|
||||||
@ -402,6 +403,10 @@ Client.prototype.connect = function (opts) {
|
|||||||
|
|
||||||
_.extend(self.config, opts || {});
|
_.extend(self.config, opts || {});
|
||||||
|
|
||||||
|
// Default iq timeout of 15 seconds
|
||||||
|
self.timeoutMonitor = new paddle.Paddle(self.config.timeout || 15);
|
||||||
|
self.timeoutMonitor.start();
|
||||||
|
|
||||||
if (self.config.wsURL) {
|
if (self.config.wsURL) {
|
||||||
return self.conn.connect(self.config);
|
return self.conn.connect(self.config);
|
||||||
}
|
}
|
||||||
@ -417,6 +422,7 @@ Client.prototype.connect = function (opts) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
Client.prototype.disconnect = function () {
|
Client.prototype.disconnect = function () {
|
||||||
|
this.timeoutMonitor.stop();
|
||||||
if (this.sessionStarted) {
|
if (this.sessionStarted) {
|
||||||
this.emit('session:end');
|
this.emit('session:end');
|
||||||
this.releaseGroup('session');
|
this.releaseGroup('session');
|
||||||
@ -461,12 +467,25 @@ Client.prototype.sendIq = function (data, cb) {
|
|||||||
if (!data.id) {
|
if (!data.id) {
|
||||||
data.id = this.nextId();
|
data.id = this.nextId();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var called = false;
|
||||||
|
var rescb = function (err, result) {
|
||||||
|
if (!called) {
|
||||||
|
called = true;
|
||||||
|
cb(err, result);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
if (data.type === 'get' || data.type === 'set') {
|
if (data.type === 'get' || data.type === 'set') {
|
||||||
|
var timeoutCheck = this.timeoutMonitor.insure(function () {
|
||||||
|
rescb({type: 'error', error: {condition: 'timeout'}}, null);
|
||||||
|
});
|
||||||
this.once('id:' + data.id, 'session', function (resp) {
|
this.once('id:' + data.id, 'session', function (resp) {
|
||||||
|
timeoutCheck.check_in();
|
||||||
if (resp._extensions.error) {
|
if (resp._extensions.error) {
|
||||||
cb(resp, null);
|
rescb(resp, null);
|
||||||
} else {
|
} else {
|
||||||
cb(null, resp);
|
rescb(null, resp);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -530,7 +549,7 @@ Client.prototype.denySubscription = function (jid) {
|
|||||||
|
|
||||||
module.exports = Client;
|
module.exports = Client;
|
||||||
|
|
||||||
},{"../vendor/lodash":90,"./stanza/bind":23,"./stanza/error":30,"./stanza/iq":33,"./stanza/message":35,"./stanza/presence":37,"./stanza/roster":41,"./stanza/sasl":43,"./stanza/session":44,"./stanza/sm":45,"./stanza/stream":46,"./stanza/streamError":47,"./stanza/streamFeatures":48,"./websocket":52,"async":53,"hostmeta":67,"node-uuid":76,"sasl-anonymous":78,"sasl-digest-md5":80,"sasl-external":82,"sasl-plain":84,"sasl-scram-sha-1":86,"saslmechanisms":88,"wildemitter":89}],3:[function(require,module,exports){
|
},{"../vendor/lodash":91,"./stanza/bind":23,"./stanza/error":30,"./stanza/iq":33,"./stanza/message":35,"./stanza/presence":37,"./stanza/roster":41,"./stanza/sasl":43,"./stanza/session":44,"./stanza/sm":45,"./stanza/stream":46,"./stanza/streamError":47,"./stanza/streamFeatures":48,"./websocket":52,"async":53,"hostmeta":67,"node-uuid":76,"paddle":77,"sasl-anonymous":79,"sasl-digest-md5":81,"sasl-external":83,"sasl-plain":85,"sasl-scram-sha-1":87,"saslmechanisms":89,"wildemitter":90}],3:[function(require,module,exports){
|
||||||
require('../stanza/attention');
|
require('../stanza/attention');
|
||||||
|
|
||||||
|
|
||||||
@ -685,7 +704,7 @@ function verifyVerString(info, hash, check) {
|
|||||||
if (hash === 'sha-1') {
|
if (hash === 'sha-1') {
|
||||||
hash = 'sha1';
|
hash = 'sha1';
|
||||||
}
|
}
|
||||||
var computed = this._generatedVerString(info, hash);
|
var computed = generateVerString(info, hash);
|
||||||
return computed && computed == check;
|
return computed && computed == check;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -862,17 +881,52 @@ module.exports = function (client) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
client.updateCaps = function () {
|
client.updateCaps = function () {
|
||||||
this.disco.caps = {
|
var node = this.config.capsNode || 'https://stanza.io';
|
||||||
node: this.config.capsNode || 'https://stanza.io',
|
var data = JSON.parse(JSON.stringify({
|
||||||
hash: 'sha-1',
|
|
||||||
ver: generateVerString({
|
|
||||||
identities: this.disco.identities[''],
|
identities: this.disco.identities[''],
|
||||||
features: this.disco.features[''],
|
features: this.disco.features[''],
|
||||||
extensions: this.disco.extensions['']
|
extensions: this.disco.extensions['']
|
||||||
}, 'sha-1')
|
}));
|
||||||
|
|
||||||
|
var ver = generateVerString(data, 'sha-1');
|
||||||
|
|
||||||
|
this.disco.caps = {
|
||||||
|
node: node,
|
||||||
|
hash: 'sha-1',
|
||||||
|
ver: ver
|
||||||
|
};
|
||||||
|
|
||||||
|
node = node + '#' + ver;
|
||||||
|
this.disco.features[node] = data.features;
|
||||||
|
this.disco.identities[node] = data.identities;
|
||||||
|
this.disco.extensions[node] = data.extensions;
|
||||||
|
|
||||||
|
return client.getCurrentCaps();
|
||||||
|
};
|
||||||
|
|
||||||
|
client.getCurrentCaps = function () {
|
||||||
|
var caps = client.disco.caps;
|
||||||
|
if (!caps.ver) {
|
||||||
|
return {ver: null, discoInfo: null};
|
||||||
|
}
|
||||||
|
|
||||||
|
var node = caps.node + '#' + caps.ver;
|
||||||
|
return {
|
||||||
|
ver: caps.ver,
|
||||||
|
discoInfo: {
|
||||||
|
identities: client.disco.identities[node],
|
||||||
|
features: client.disco.features[node],
|
||||||
|
extensions: client.disco.extensions[node]
|
||||||
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
client.on('presence', function (pres) {
|
||||||
|
if (pres._extensions.caps) {
|
||||||
|
client.emit('disco:caps', pres);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
client.on('iq:get:discoInfo', function (iq) {
|
client.on('iq:get:discoInfo', function (iq) {
|
||||||
var node = iq.discoInfo.node;
|
var node = iq.discoInfo.node;
|
||||||
var reportedNode = iq.discoInfo.node;
|
var reportedNode = iq.discoInfo.node;
|
||||||
@ -900,9 +954,12 @@ module.exports = function (client) {
|
|||||||
}
|
}
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
client.verifyVerString = verifyVerString;
|
||||||
|
client.generateVerString = generateVerString;
|
||||||
};
|
};
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"../stanza/caps":24,"../stanza/disco":29,"crypto":60}],10:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"../stanza/caps":24,"../stanza/disco":29,"crypto":60}],10:[function(require,module,exports){
|
||||||
var stanzas = require('../stanza/forwarded');
|
var stanzas = require('../stanza/forwarded');
|
||||||
|
|
||||||
|
|
||||||
@ -1843,7 +1900,7 @@ Avatar.prototype = {
|
|||||||
|
|
||||||
module.exports = Avatar;
|
module.exports = Avatar;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"./pubsub":38,"jxt":74}],23:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"./pubsub":38,"jxt":74}],23:[function(require,module,exports){
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
var Iq = require('./iq');
|
var Iq = require('./iq');
|
||||||
var StreamFeatures = require('./streamFeatures');
|
var StreamFeatures = require('./streamFeatures');
|
||||||
@ -2311,7 +2368,7 @@ stanza.extend(Message, DataForm);
|
|||||||
exports.DataForm = DataForm;
|
exports.DataForm = DataForm;
|
||||||
exports.Field = Field;
|
exports.Field = Field;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"./message":35,"jxt":74}],28:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"./message":35,"jxt":74}],28:[function(require,module,exports){
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
var Message = require('./message');
|
var Message = require('./message');
|
||||||
var Presence = require('./presence');
|
var Presence = require('./presence');
|
||||||
@ -2516,7 +2573,7 @@ stanza.extend(DiscoItems, RSM);
|
|||||||
exports.DiscoInfo = DiscoInfo;
|
exports.DiscoInfo = DiscoInfo;
|
||||||
exports.DiscoItems = DiscoItems;
|
exports.DiscoItems = DiscoItems;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"./dataforms":27,"./iq":33,"./rsm":42,"jxt":74}],30:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"./dataforms":27,"./iq":33,"./rsm":42,"jxt":74}],30:[function(require,module,exports){
|
||||||
var _ = require('../../vendor/lodash');
|
var _ = require('../../vendor/lodash');
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
var Message = require('./message');
|
var Message = require('./message');
|
||||||
@ -2631,7 +2688,7 @@ stanza.extend(Iq, Error);
|
|||||||
|
|
||||||
module.exports = Error;
|
module.exports = Error;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"./iq":33,"./message":35,"./presence":37,"jxt":74}],31:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"./iq":33,"./message":35,"./presence":37,"jxt":74}],31:[function(require,module,exports){
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
var Message = require('./message');
|
var Message = require('./message');
|
||||||
var Presence = require('./presence');
|
var Presence = require('./presence');
|
||||||
@ -2993,7 +3050,7 @@ stanza.topLevel(Message);
|
|||||||
|
|
||||||
module.exports = Message;
|
module.exports = Message;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"jxt":74}],36:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"jxt":74}],36:[function(require,module,exports){
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
var Message = require('./message');
|
var Message = require('./message');
|
||||||
var Presence = require('./presence');
|
var Presence = require('./presence');
|
||||||
@ -3156,7 +3213,7 @@ stanza.topLevel(Presence);
|
|||||||
|
|
||||||
module.exports = Presence;
|
module.exports = Presence;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"jxt":74}],38:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"jxt":74}],38:[function(require,module,exports){
|
||||||
var _ = require('../../vendor/lodash');
|
var _ = require('../../vendor/lodash');
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
var Iq = require('./iq');
|
var Iq = require('./iq');
|
||||||
@ -3594,7 +3651,7 @@ exports.Pubsub = Pubsub;
|
|||||||
exports.Item = Item;
|
exports.Item = Item;
|
||||||
exports.EventItem = EventItem;
|
exports.EventItem = EventItem;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"./dataforms":27,"./iq":33,"./message":35,"./rsm":42,"jxt":74}],39:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"./dataforms":27,"./iq":33,"./message":35,"./rsm":42,"jxt":74}],39:[function(require,module,exports){
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
var Message = require('./message');
|
var Message = require('./message');
|
||||||
|
|
||||||
@ -3773,7 +3830,7 @@ stanza.extend(Iq, Roster);
|
|||||||
|
|
||||||
module.exports = Roster;
|
module.exports = Roster;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"./iq":33,"jxt":74}],42:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"./iq":33,"jxt":74}],42:[function(require,module,exports){
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
|
|
||||||
|
|
||||||
@ -4091,7 +4148,7 @@ exports.Success = Success;
|
|||||||
exports.Failure = Failure;
|
exports.Failure = Failure;
|
||||||
exports.Abort = Abort;
|
exports.Abort = Abort;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"./streamFeatures":48,"jxt":74}],44:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"./streamFeatures":48,"jxt":74}],44:[function(require,module,exports){
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
var Iq = require('./iq');
|
var Iq = require('./iq');
|
||||||
var StreamFeatures = require('./streamFeatures');
|
var StreamFeatures = require('./streamFeatures');
|
||||||
@ -4452,7 +4509,7 @@ stanza.topLevel(StreamError);
|
|||||||
|
|
||||||
module.exports = StreamError;
|
module.exports = StreamError;
|
||||||
|
|
||||||
},{"../../vendor/lodash":90,"jxt":74}],48:[function(require,module,exports){
|
},{"../../vendor/lodash":91,"jxt":74}],48:[function(require,module,exports){
|
||||||
var stanza = require('jxt');
|
var stanza = require('jxt');
|
||||||
|
|
||||||
|
|
||||||
@ -4845,7 +4902,7 @@ WSConnection.prototype.send = function (data) {
|
|||||||
|
|
||||||
module.exports = WSConnection;
|
module.exports = WSConnection;
|
||||||
|
|
||||||
},{"../vendor/lodash":90,"./sm":20,"./stanza/iq":33,"./stanza/message":35,"./stanza/presence":37,"./stanza/stream":46,"async":53,"node-uuid":76,"wildemitter":89}],53:[function(require,module,exports){
|
},{"../vendor/lodash":91,"./sm":20,"./stanza/iq":33,"./stanza/message":35,"./stanza/presence":37,"./stanza/stream":46,"async":53,"node-uuid":76,"wildemitter":90}],53:[function(require,module,exports){
|
||||||
var process=require("__browserify_process");/*global setImmediate: false, setTimeout: false, console: false */
|
var process=require("__browserify_process");/*global setImmediate: false, setTimeout: false, console: false */
|
||||||
(function () {
|
(function () {
|
||||||
|
|
||||||
@ -13875,6 +13932,178 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
}());
|
}());
|
||||||
|
|
||||||
},{"__browserify_Buffer":65,"crypto":60}],77:[function(require,module,exports){
|
},{"__browserify_Buffer":65,"crypto":60}],77:[function(require,module,exports){
|
||||||
|
/**
|
||||||
|
* Written by Nathan Fritz. Copyright © 2011 by &yet, LLC. Released under the
|
||||||
|
* terms of the MIT License:
|
||||||
|
*
|
||||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
* of this software and associated documentation files (the "Software"), to deal
|
||||||
|
* in the Software without restriction, including without limitation the rights
|
||||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
* copies of the Software, and to permit persons to whom the Software is
|
||||||
|
* furnished to do so, subject to the following conditions:
|
||||||
|
*
|
||||||
|
* The above copyright notice and this permission notice shall be included in
|
||||||
|
* all copies or substantial portions of the Software.
|
||||||
|
*
|
||||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
|
* THE SOFTWARE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
var EventEmitter = require("events").EventEmitter;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* You're up a creek; here's your Paddle. In Javascript, we rely on callback
|
||||||
|
* execution, often times without knowing for sure that it will happen. With
|
||||||
|
* Paddle, you can know. Paddle is a simple way of noting that your code should
|
||||||
|
* reach one of several code-execution points within a timelimit. If the time-
|
||||||
|
* limit is exceeded, an error callback is executed.
|
||||||
|
*
|
||||||
|
* @param freq: number of seconds between timeout checks
|
||||||
|
*/
|
||||||
|
function Paddle(freq) {
|
||||||
|
EventEmitter.call(this);
|
||||||
|
this.registry = new Object();
|
||||||
|
this.insureids = 0;
|
||||||
|
if(freq === undefined) {
|
||||||
|
this.freq = 5;
|
||||||
|
} else {
|
||||||
|
this.freq = freq;
|
||||||
|
}
|
||||||
|
this.run = false;
|
||||||
|
this.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
//extend Paddle with EventEmitter
|
||||||
|
Paddle.super_ = EventEmitter;
|
||||||
|
Paddle.prototype = Object.create(EventEmitter.prototype, {
|
||||||
|
constructor: {
|
||||||
|
value: Paddle,
|
||||||
|
enumerable: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Register a new check. If the check times out, error_callback will be called
|
||||||
|
* with the optionally specified args. You may specify an id, but one will be
|
||||||
|
* created otherwise.
|
||||||
|
*
|
||||||
|
* @param error_callback: function called when timeout is reached without check-in
|
||||||
|
* @param timeout: seconds to wait for check-in
|
||||||
|
* @param args: Array of arguments to call error_callback with
|
||||||
|
* @param id: optional -- insure will generate one for you
|
||||||
|
*
|
||||||
|
* @return Object: returns the insurance obj you just created
|
||||||
|
* {paddle, error_callback, args, id, timeout, done, check_in}
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
function insure(error_callback, timeout, args, id) {
|
||||||
|
if(id === undefined) {
|
||||||
|
++this.insureids;
|
||||||
|
this.insureids %= 65000;
|
||||||
|
id = this.insureids;
|
||||||
|
}
|
||||||
|
expiretime = Date.now() + timeout * 1000;
|
||||||
|
var that = this;
|
||||||
|
var insurance = {
|
||||||
|
paddle: that,
|
||||||
|
error_callback: error_callback,
|
||||||
|
args: args,
|
||||||
|
id: id,
|
||||||
|
timeout: expiretime,
|
||||||
|
done: false,
|
||||||
|
check_in: function() {
|
||||||
|
return this.paddle.check_in(this.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//this.registry[id] = [expiretime, error_callback, args];
|
||||||
|
this.registry[id] = insurance;
|
||||||
|
return insurance;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Check in with an id or paddle to confirm that your end-execution point occurred. This
|
||||||
|
* will cancel the timeout error, and delete the entry for this id.
|
||||||
|
*
|
||||||
|
* @param id or insurance: id from insure or insure obj
|
||||||
|
*/
|
||||||
|
function check_in(id) {
|
||||||
|
if(id.id !== undefined) {
|
||||||
|
//perhaps this is an insure object
|
||||||
|
id = id.id;
|
||||||
|
}
|
||||||
|
if(id in this.registry) {
|
||||||
|
this.emit('check_in', this.registry[id]);
|
||||||
|
this.registry[id].done = true;
|
||||||
|
delete this.registry[id];
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Executed internally to occasionally make sure all insurance ids are within
|
||||||
|
* their timeouts.
|
||||||
|
*/
|
||||||
|
function checkEnsures() {
|
||||||
|
var now = Date.now();
|
||||||
|
for(var id in this.registry) {
|
||||||
|
if(now > this.registry[id].timeout) {
|
||||||
|
this.registry[id].error_callback.apply(this, this.registry[id].args);
|
||||||
|
this.emit('timeout', this.registry[id]);
|
||||||
|
delete this.registry[id];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(this.run) {
|
||||||
|
setTimeout(function() { this.checkEnsures() }.bind(this), this.freq * 1000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Start checking paddle timeouts. Optionally reset frequency.
|
||||||
|
*
|
||||||
|
* @return bool: true if running, false if it was already running.
|
||||||
|
*/
|
||||||
|
function start(freq) {
|
||||||
|
if(freq !== undefined) {
|
||||||
|
this.freq = freq;
|
||||||
|
}
|
||||||
|
if(!this.run) {
|
||||||
|
this.run = true;
|
||||||
|
setTimeout(function() { this.checkEnsures() }.bind(this), this.freq * 1000);
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Stop checking Paddle timeouts.
|
||||||
|
* @return bool: true if stopped, false if it was already stopped.
|
||||||
|
*/
|
||||||
|
function stop() {
|
||||||
|
if(this.run) {
|
||||||
|
this.run = false;
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Paddle.prototype.insure = insure;
|
||||||
|
Paddle.prototype.check_in = check_in;
|
||||||
|
Paddle.prototype.checkEnsures = checkEnsures;
|
||||||
|
Paddle.prototype.start = start;
|
||||||
|
Paddle.prototype.stop = stop;
|
||||||
|
|
||||||
|
exports.Paddle = Paddle;
|
||||||
|
|
||||||
|
},{"events":55}],78:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -13930,7 +14159,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{}],78:[function(require,module,exports){
|
},{}],79:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -13950,7 +14179,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{"./lib/mechanism":77}],79:[function(require,module,exports){
|
},{"./lib/mechanism":78}],80:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14140,7 +14369,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{"crypto":60}],80:[function(require,module,exports){
|
},{"crypto":60}],81:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14160,7 +14389,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{"./lib/mechanism":79}],81:[function(require,module,exports){
|
},{"./lib/mechanism":80}],82:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14216,7 +14445,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{}],82:[function(require,module,exports){
|
},{}],83:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14236,7 +14465,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{"./lib/mechanism":81}],83:[function(require,module,exports){
|
},{"./lib/mechanism":82}],84:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14303,7 +14532,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{}],84:[function(require,module,exports){
|
},{}],85:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14323,7 +14552,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{"./lib/mechanism":83}],85:[function(require,module,exports){
|
},{"./lib/mechanism":84}],86:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14581,7 +14810,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
exports = module.exports = Mechanism;
|
exports = module.exports = Mechanism;
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{"buffer":58,"crypto":60}],86:[function(require,module,exports){
|
},{"buffer":58,"crypto":60}],87:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14601,7 +14830,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{"./lib/mechanism":85}],87:[function(require,module,exports){
|
},{"./lib/mechanism":86}],88:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14674,7 +14903,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{}],88:[function(require,module,exports){
|
},{}],89:[function(require,module,exports){
|
||||||
(function(root, factory) {
|
(function(root, factory) {
|
||||||
if (typeof exports === 'object') {
|
if (typeof exports === 'object') {
|
||||||
// CommonJS
|
// CommonJS
|
||||||
@ -14694,7 +14923,7 @@ var Buffer=require("__browserify_Buffer").Buffer;// uuid.js
|
|||||||
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
},{"./lib/factory":87}],89:[function(require,module,exports){
|
},{"./lib/factory":88}],90:[function(require,module,exports){
|
||||||
/*
|
/*
|
||||||
WildEmitter.js is a slim little event emitter by @henrikjoreteg largely based
|
WildEmitter.js is a slim little event emitter by @henrikjoreteg largely based
|
||||||
on @visionmedia's Emitter from UI Kit.
|
on @visionmedia's Emitter from UI Kit.
|
||||||
@ -14831,7 +15060,7 @@ WildEmitter.prototype.getWildcardCallbacks = function (eventName) {
|
|||||||
return result;
|
return result;
|
||||||
};
|
};
|
||||||
|
|
||||||
},{}],90:[function(require,module,exports){
|
},{}],91:[function(require,module,exports){
|
||||||
var global=self;/**
|
var global=self;/**
|
||||||
* @license
|
* @license
|
||||||
* Lo-Dash 1.3.1 (Custom Build) lodash.com/license
|
* Lo-Dash 1.3.1 (Custom Build) lodash.com/license
|
||||||
|
@ -23,6 +23,7 @@ module.exports = HumanModel.define({
|
|||||||
seal: true,
|
seal: true,
|
||||||
type: 'contact',
|
type: 'contact',
|
||||||
props: {
|
props: {
|
||||||
|
inRoster: ['bool', true, false],
|
||||||
jid: ['string', true],
|
jid: ['string', true],
|
||||||
name: ['string', true, ''],
|
name: ['string', true, ''],
|
||||||
subscription: ['string', true, 'none'],
|
subscription: ['string', true, 'none'],
|
||||||
@ -190,6 +191,8 @@ module.exports = HumanModel.define({
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
save: function () {
|
save: function () {
|
||||||
|
if (!this.inRoster) return;
|
||||||
|
|
||||||
var data = {
|
var data = {
|
||||||
jid: this.jid,
|
jid: this.jid,
|
||||||
name: this.name,
|
name: this.name,
|
||||||
|
@ -52,6 +52,7 @@ module.exports = BaseCollection.extend({
|
|||||||
|
|
||||||
contacts.forEach(function (contact) {
|
contacts.forEach(function (contact) {
|
||||||
contact = new Contact(contact);
|
contact = new Contact(contact);
|
||||||
|
contact.inRoster = true;
|
||||||
contact.save();
|
contact.save();
|
||||||
self.add(contact);
|
self.add(contact);
|
||||||
});
|
});
|
||||||
|
@ -11,6 +11,7 @@ module.exports = HumanModel.define({
|
|||||||
status: ['string', true, ''],
|
status: ['string', true, ''],
|
||||||
show: ['string', true, ''],
|
show: ['string', true, ''],
|
||||||
priority: ['number', true, 0],
|
priority: ['number', true, 0],
|
||||||
idleSince: 'date'
|
idleSince: 'date',
|
||||||
|
discoInfo: 'object'
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -17,8 +17,8 @@ module.exports = BasePage.extend({
|
|||||||
this.render();
|
this.render();
|
||||||
},
|
},
|
||||||
events: {
|
events: {
|
||||||
'submit #chatInput': 'killForm',
|
'keydown #chatBuffer': 'handleKeyDown',
|
||||||
'keydown #chatBuffer': 'handleKeyDown'
|
'keyup #chatBuffer': 'handleKeyUp'
|
||||||
},
|
},
|
||||||
srcBindings: {
|
srcBindings: {
|
||||||
avatar: 'header .avatar'
|
avatar: 'header .avatar'
|
||||||
@ -29,16 +29,14 @@ module.exports = BasePage.extend({
|
|||||||
},
|
},
|
||||||
render: function () {
|
render: function () {
|
||||||
this.renderAndBind();
|
this.renderAndBind();
|
||||||
|
this.typingTimer = null;
|
||||||
this.$chatBuffer = this.$('#chatBuffer');
|
this.$chatBuffer = this.$('#chatBuffer');
|
||||||
this.renderCollection(this.model.messages, Message, this.$('#conversation'));
|
this.renderCollection(this.model.messages, Message, this.$('#conversation'));
|
||||||
this.registerBindings();
|
this.registerBindings();
|
||||||
return this;
|
return this;
|
||||||
},
|
},
|
||||||
killForm: function (e) {
|
|
||||||
e.preventDefault();
|
|
||||||
return false;
|
|
||||||
},
|
|
||||||
handleKeyDown: function (e) {
|
handleKeyDown: function (e) {
|
||||||
|
clearTimeout(this.typingTimer);
|
||||||
if (e.which === 13 && !e.shiftKey) {
|
if (e.which === 13 && !e.shiftKey) {
|
||||||
this.sendChat();
|
this.sendChat();
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@ -54,6 +52,34 @@ module.exports = BasePage.extend({
|
|||||||
this.$chatBuffer.removeClass('editing');
|
this.$chatBuffer.removeClass('editing');
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
return false;
|
return false;
|
||||||
|
} else {
|
||||||
|
if (!this.typing) {
|
||||||
|
this.typing = true;
|
||||||
|
client.sendMessage({
|
||||||
|
to: this.model.lockedResource || this.model.jid,
|
||||||
|
chatState: 'composing'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
handleKeyUp: function (e) {
|
||||||
|
this.typingTimer = setTimeout(this.pausedTyping.bind(this), 5000);
|
||||||
|
if (this.typing && this.$chatBuffer.val().length === 0) {
|
||||||
|
this.typing = false;
|
||||||
|
client.sendMessage({
|
||||||
|
to: this.model.lockedResource || this.model.jid,
|
||||||
|
chatState: 'active'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
pausedTyping: function () {
|
||||||
|
console.log('paused?', this.typing);
|
||||||
|
if (this.typing) {
|
||||||
|
this.typing = false;
|
||||||
|
client.sendMessage({
|
||||||
|
to: this.model.lockedResource || this.model.jid,
|
||||||
|
chatState: 'paused'
|
||||||
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
sendChat: function () {
|
sendChat: function () {
|
||||||
@ -79,11 +105,13 @@ module.exports = BasePage.extend({
|
|||||||
this.model.lastSentMessage.correct(message);
|
this.model.lastSentMessage.correct(message);
|
||||||
} else {
|
} else {
|
||||||
var msgModel = new MessageModel(message);
|
var msgModel = new MessageModel(message);
|
||||||
|
msgModel.cid = id;
|
||||||
this.model.messages.add(msgModel);
|
this.model.messages.add(msgModel);
|
||||||
this.model.lastSentMessage = msgModel;
|
this.model.lastSentMessage = msgModel;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.editMode = false;
|
this.editMode = false;
|
||||||
|
this.typing = false;
|
||||||
this.$chatBuffer.removeClass('editing');
|
this.$chatBuffer.removeClass('editing');
|
||||||
this.$chatBuffer.val('');
|
this.$chatBuffer.val('');
|
||||||
}
|
}
|
||||||
|
@ -23,7 +23,11 @@ DiscoStorage.prototype = {
|
|||||||
ver: ver,
|
ver: ver,
|
||||||
disco: disco
|
disco: disco
|
||||||
};
|
};
|
||||||
|
var request = this.transaction('readwrite').put(data);
|
||||||
|
request.onsuccess = function () {
|
||||||
|
cb(false, data);
|
||||||
|
};
|
||||||
|
request.onerror = cb;
|
||||||
},
|
},
|
||||||
get: function (ver, cb) {
|
get: function (ver, cb) {
|
||||||
cb = cb || function () {};
|
cb = cb || function () {};
|
||||||
|
@ -18,7 +18,7 @@
|
|||||||
"node-uuid": "1.4.1",
|
"node-uuid": "1.4.1",
|
||||||
"semi-static": "0.0.4",
|
"semi-static": "0.0.4",
|
||||||
"sound-effect-manager": "0.0.5",
|
"sound-effect-manager": "0.0.5",
|
||||||
"human-model": "1.0.0",
|
"human-model": "1.1.0",
|
||||||
"human-view": "1.1.2",
|
"human-view": "1.1.2",
|
||||||
"templatizer": "0.1.2",
|
"templatizer": "0.1.2",
|
||||||
"underscore": "1.5.1"
|
"underscore": "1.5.1"
|
||||||
|
@ -133,6 +133,7 @@ html, body {
|
|||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
position: relative;
|
position: relative;
|
||||||
margin-top: 50px;
|
margin-top: 50px;
|
||||||
|
padding-top: 50px;
|
||||||
bottom: 50px;
|
bottom: 50px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user