2020-07-14 22:41:26 +02:00
|
|
|
/*global converse */
|
|
|
|
|
2020-04-22 12:10:39 +02:00
|
|
|
const mock = {};
|
|
|
|
window.mock = mock;
|
|
|
|
let _converse, initConverse;
|
|
|
|
|
2020-04-22 13:11:48 +02:00
|
|
|
const converseLoaded = new Promise(resolve => window.addEventListener('converse-loaded', resolve));
|
|
|
|
|
2020-06-01 16:05:00 +02:00
|
|
|
jasmine.DEFAULT_TIMEOUT_INTERVAL = 7000;
|
2020-05-15 14:33:31 +02:00
|
|
|
|
2020-04-22 12:10:39 +02:00
|
|
|
mock.initConverse = function (promise_names=[], settings=null, func) {
|
|
|
|
if (typeof promise_names === "function") {
|
|
|
|
func = promise_names;
|
|
|
|
promise_names = []
|
|
|
|
settings = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return async done => {
|
|
|
|
if (_converse && _converse.api.connection.connected()) {
|
|
|
|
await _converse.api.user.logout();
|
|
|
|
}
|
|
|
|
const el = document.querySelector('#conversejs');
|
|
|
|
if (el) {
|
|
|
|
el.parentElement.removeChild(el);
|
|
|
|
}
|
|
|
|
document.title = "Converse Tests";
|
|
|
|
|
2020-04-22 13:11:48 +02:00
|
|
|
await converseLoaded;
|
2020-04-22 12:10:39 +02:00
|
|
|
await initConverse(settings);
|
|
|
|
await Promise.all((promise_names || []).map(_converse.api.waitUntil));
|
|
|
|
try {
|
|
|
|
await func(done, _converse);
|
|
|
|
} catch(e) {
|
|
|
|
console.error(e);
|
|
|
|
fail(e);
|
|
|
|
await done();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
window.addEventListener('converse-loaded', () => {
|
2020-09-25 21:20:29 +02:00
|
|
|
const { u, sizzle, Strophe, dayjs, $iq, $msg, $pres } = converse.env;
|
2020-04-22 13:11:48 +02:00
|
|
|
|
|
|
|
mock.waitUntilDiscoConfirmed = async function (_converse, entity_jid, identities, features=[], items=[], type='info') {
|
|
|
|
const iq = await u.waitUntil(() => {
|
2020-09-25 21:20:29 +02:00
|
|
|
return _converse.connection.IQ_stanzas.filter(
|
|
|
|
iq => sizzle(`iq[to="${entity_jid}"] query[xmlns="http://jabber.org/protocol/disco#${type}"]`, iq).length
|
2020-04-22 13:11:48 +02:00
|
|
|
).pop();
|
|
|
|
}, 300);
|
|
|
|
const stanza = $iq({
|
|
|
|
'type': 'result',
|
|
|
|
'from': entity_jid,
|
|
|
|
'to': 'romeo@montague.lit/orchard',
|
|
|
|
'id': iq.getAttribute('id'),
|
|
|
|
}).c('query', {'xmlns': 'http://jabber.org/protocol/disco#'+type});
|
|
|
|
|
2020-09-25 21:20:29 +02:00
|
|
|
identities?.forEach(identity => stanza.c('identity', {'category': identity.category, 'type': identity.type}).up());
|
|
|
|
features?.forEach(feature => stanza.c('feature', {'var': feature}).up());
|
|
|
|
items?.forEach(item => stanza.c('item', {'jid': item}).up());
|
2020-04-22 13:11:48 +02:00
|
|
|
_converse.connection._dataRecv(mock.createRequest(stanza));
|
|
|
|
}
|
|
|
|
|
|
|
|
mock.createRequest = function (iq) {
|
|
|
|
iq = typeof iq.tree == "function" ? iq.tree() : iq;
|
|
|
|
var req = new Strophe.Request(iq, function() {});
|
|
|
|
req.getResponse = function () {
|
|
|
|
var env = new Strophe.Builder('env', {type: 'mock'}).tree();
|
|
|
|
env.appendChild(iq);
|
|
|
|
return env;
|
|
|
|
};
|
|
|
|
return req;
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.closeAllChatBoxes = function (_converse) {
|
|
|
|
return Promise.all(_converse.chatboxviews.map(view => view.close()));
|
|
|
|
};
|
|
|
|
|
2020-10-20 06:45:35 +02:00
|
|
|
mock.toggleControlBox = function () {
|
|
|
|
const toggle = document.querySelector(".toggle-controlbox");
|
2020-04-22 13:11:48 +02:00
|
|
|
if (!u.isVisible(document.querySelector("#controlbox"))) {
|
|
|
|
if (!u.isVisible(toggle)) {
|
|
|
|
u.removeClass('hidden', toggle);
|
|
|
|
}
|
|
|
|
toggle.click();
|
|
|
|
}
|
2020-10-20 06:45:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
mock.openControlBox = async function (_converse) {
|
|
|
|
const model = await _converse.api.controlbox.open();
|
|
|
|
await u.waitUntil(() => model.get('connected'));
|
|
|
|
mock.toggleControlBox();
|
2020-04-22 13:11:48 +02:00
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.closeControlBox = function () {
|
|
|
|
const controlbox = document.querySelector("#controlbox");
|
|
|
|
if (u.isVisible(controlbox)) {
|
|
|
|
const button = controlbox.querySelector(".close-chatbox-button");
|
2020-09-25 21:20:29 +02:00
|
|
|
(button !== null) && button.click();
|
2020-04-22 13:11:48 +02:00
|
|
|
}
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.waitUntilBookmarksReturned = async function (_converse, bookmarks=[]) {
|
|
|
|
await mock.waitUntilDiscoConfirmed(
|
|
|
|
_converse, _converse.bare_jid,
|
|
|
|
[{'category': 'pubsub', 'type': 'pep'}],
|
|
|
|
['http://jabber.org/protocol/pubsub#publish-options']
|
|
|
|
);
|
|
|
|
const IQ_stanzas = _converse.connection.IQ_stanzas;
|
|
|
|
const sent_stanza = await u.waitUntil(
|
|
|
|
() => IQ_stanzas.filter(s => sizzle('items[node="storage:bookmarks"]', s).length).pop()
|
|
|
|
);
|
|
|
|
const stanza = $iq({
|
|
|
|
'to': _converse.connection.jid,
|
|
|
|
'type':'result',
|
|
|
|
'id':sent_stanza.getAttribute('id')
|
|
|
|
}).c('pubsub', {'xmlns': Strophe.NS.PUBSUB})
|
|
|
|
.c('items', {'node': 'storage:bookmarks'})
|
|
|
|
.c('item', {'id': 'current'})
|
|
|
|
.c('storage', {'xmlns': 'storage:bookmarks'});
|
|
|
|
bookmarks.forEach(bookmark => {
|
|
|
|
stanza.c('conference', {
|
|
|
|
'name': bookmark.name,
|
|
|
|
'autojoin': bookmark.autojoin,
|
|
|
|
'jid': bookmark.jid
|
|
|
|
}).c('nick').t(bookmark.nick).up().up()
|
|
|
|
});
|
|
|
|
_converse.connection._dataRecv(mock.createRequest(stanza));
|
|
|
|
await _converse.api.waitUntil('bookmarksInitialized');
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.openChatBoxes = function (converse, amount) {
|
|
|
|
for (let i=0; i<amount; i++) {
|
|
|
|
const jid = mock.cur_names[i].replace(/ /g,'.').toLowerCase() + '@montague.lit';
|
2020-04-26 15:46:45 +02:00
|
|
|
converse.roster.get(jid).openChat();
|
2020-04-22 13:11:48 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.openChatBoxFor = async function (_converse, jid) {
|
|
|
|
await _converse.api.waitUntil('rosterContactsFetched');
|
2020-04-26 15:46:45 +02:00
|
|
|
_converse.roster.get(jid).openChat();
|
2020-04-22 13:11:48 +02:00
|
|
|
return u.waitUntil(() => _converse.chatboxviews.get(jid), 1000);
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.openChatRoomViaModal = async function (_converse, jid, nick='') {
|
|
|
|
// Opens a new chatroom
|
|
|
|
const model = await _converse.api.controlbox.open('controlbox');
|
|
|
|
await u.waitUntil(() => model.get('connected'));
|
|
|
|
await mock.openControlBox(_converse);
|
|
|
|
const view = await _converse.chatboxviews.get('controlbox');
|
|
|
|
const roomspanel = view.roomspanel;
|
|
|
|
roomspanel.el.querySelector('.show-add-muc-modal').click();
|
|
|
|
mock.closeControlBox(_converse);
|
|
|
|
const modal = roomspanel.add_room_modal;
|
|
|
|
await u.waitUntil(() => u.isVisible(modal.el), 1500)
|
|
|
|
modal.el.querySelector('input[name="chatroom"]').value = jid;
|
|
|
|
if (nick) {
|
|
|
|
modal.el.querySelector('input[name="nickname"]').value = nick;
|
|
|
|
}
|
|
|
|
modal.el.querySelector('form input[type="submit"]').click();
|
|
|
|
await u.waitUntil(() => _converse.chatboxviews.get(jid), 1000);
|
|
|
|
return _converse.chatboxviews.get(jid);
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.openChatRoom = function (_converse, room, server) {
|
|
|
|
return _converse.api.rooms.open(`${room}@${server}`);
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.getRoomFeatures = async function (_converse, muc_jid, features=[]) {
|
|
|
|
const room = Strophe.getNodeFromJid(muc_jid);
|
|
|
|
muc_jid = muc_jid.toLowerCase();
|
|
|
|
const stanzas = _converse.connection.IQ_stanzas;
|
|
|
|
const stanza = await u.waitUntil(() => stanzas.filter(
|
|
|
|
iq => iq.querySelector(
|
|
|
|
`iq[to="${muc_jid}"] query[xmlns="http://jabber.org/protocol/disco#info"]`
|
|
|
|
)).pop()
|
|
|
|
);
|
|
|
|
const features_stanza = $iq({
|
|
|
|
'from': muc_jid,
|
|
|
|
'id': stanza.getAttribute('id'),
|
|
|
|
'to': 'romeo@montague.lit/desktop',
|
|
|
|
'type': 'result'
|
|
|
|
}).c('query', { 'xmlns': 'http://jabber.org/protocol/disco#info'})
|
|
|
|
.c('identity', {
|
|
|
|
'category': 'conference',
|
|
|
|
'name': room[0].toUpperCase() + room.slice(1),
|
|
|
|
'type': 'text'
|
|
|
|
}).up();
|
|
|
|
|
|
|
|
features = features.length ? features : mock.default_muc_features;
|
|
|
|
features.forEach(f => features_stanza.c('feature', {'var': f}).up());
|
|
|
|
features_stanza.c('x', { 'xmlns':'jabber:x:data', 'type':'result'})
|
|
|
|
.c('field', {'var':'FORM_TYPE', 'type':'hidden'})
|
|
|
|
.c('value').t('http://jabber.org/protocol/muc#roominfo').up().up()
|
|
|
|
.c('field', {'type':'text-single', 'var':'muc#roominfo_description', 'label':'Description'})
|
|
|
|
.c('value').t('This is the description').up().up()
|
|
|
|
.c('field', {'type':'text-single', 'var':'muc#roominfo_occupants', 'label':'Number of occupants'})
|
|
|
|
.c('value').t(0);
|
|
|
|
_converse.connection._dataRecv(mock.createRequest(features_stanza));
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
mock.waitForReservedNick = async function (_converse, muc_jid, nick) {
|
|
|
|
const stanzas = _converse.connection.IQ_stanzas;
|
|
|
|
const selector = `iq[to="${muc_jid.toLowerCase()}"] query[node="x-roomuser-item"]`;
|
|
|
|
const iq = await u.waitUntil(() => stanzas.filter(s => sizzle(selector, s).length).pop());
|
|
|
|
|
|
|
|
// We remove the stanza, otherwise we might get stale stanzas returned in our filter above.
|
|
|
|
stanzas.splice(stanzas.indexOf(iq), 1)
|
|
|
|
|
|
|
|
// The XMPP server returns the reserved nick for this user.
|
|
|
|
const IQ_id = iq.getAttribute('id');
|
|
|
|
const stanza = $iq({
|
|
|
|
'type': 'result',
|
|
|
|
'id': IQ_id,
|
|
|
|
'from': muc_jid,
|
|
|
|
'to': _converse.connection.jid
|
|
|
|
}).c('query', {'xmlns': 'http://jabber.org/protocol/disco#info', 'node': 'x-roomuser-item'});
|
|
|
|
if (nick) {
|
|
|
|
stanza.c('identity', {'category': 'conference', 'name': nick, 'type': 'text'});
|
|
|
|
}
|
|
|
|
_converse.connection._dataRecv(mock.createRequest(stanza));
|
|
|
|
if (nick) {
|
|
|
|
return u.waitUntil(() => nick);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
mock.returnMemberLists = async function (_converse, muc_jid, members=[], affiliations=['member', 'owner', 'admin']) {
|
|
|
|
if (affiliations.length === 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const stanzas = _converse.connection.IQ_stanzas;
|
|
|
|
|
|
|
|
if (affiliations.includes('member')) {
|
2020-09-25 21:20:29 +02:00
|
|
|
const member_IQ = await u.waitUntil(() =>
|
|
|
|
stanzas.filter(s => sizzle(`iq[to="${muc_jid}"] query[xmlns="${Strophe.NS.MUC_ADMIN}"] item[affiliation="member"]`, s).length
|
2020-04-22 13:11:48 +02:00
|
|
|
).pop());
|
|
|
|
const member_list_stanza = $iq({
|
|
|
|
'from': 'coven@chat.shakespeare.lit',
|
|
|
|
'id': member_IQ.getAttribute('id'),
|
|
|
|
'to': 'romeo@montague.lit/orchard',
|
|
|
|
'type': 'result'
|
|
|
|
}).c('query', {'xmlns': Strophe.NS.MUC_ADMIN});
|
|
|
|
members.filter(m => m.affiliation === 'member').forEach(m => {
|
|
|
|
member_list_stanza.c('item', {
|
|
|
|
'affiliation': m.affiliation,
|
|
|
|
'jid': m.jid,
|
|
|
|
'nick': m.nick
|
|
|
|
});
|
|
|
|
});
|
|
|
|
_converse.connection._dataRecv(mock.createRequest(member_list_stanza));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (affiliations.includes('admin')) {
|
2020-09-25 21:20:29 +02:00
|
|
|
const admin_IQ = await u.waitUntil(() => stanzas.filter(
|
2020-04-22 13:11:48 +02:00
|
|
|
s => sizzle(`iq[to="${muc_jid}"] query[xmlns="${Strophe.NS.MUC_ADMIN}"] item[affiliation="admin"]`, s).length
|
|
|
|
).pop());
|
|
|
|
const admin_list_stanza = $iq({
|
|
|
|
'from': 'coven@chat.shakespeare.lit',
|
|
|
|
'id': admin_IQ.getAttribute('id'),
|
|
|
|
'to': 'romeo@montague.lit/orchard',
|
|
|
|
'type': 'result'
|
|
|
|
}).c('query', {'xmlns': Strophe.NS.MUC_ADMIN});
|
|
|
|
members.filter(m => m.affiliation === 'admin').forEach(m => {
|
|
|
|
admin_list_stanza.c('item', {
|
|
|
|
'affiliation': m.affiliation,
|
|
|
|
'jid': m.jid,
|
|
|
|
'nick': m.nick
|
|
|
|
});
|
|
|
|
});
|
|
|
|
_converse.connection._dataRecv(mock.createRequest(admin_list_stanza));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (affiliations.includes('owner')) {
|
2020-09-25 21:20:29 +02:00
|
|
|
const owner_IQ = await u.waitUntil(() => stanzas.filter(
|
2020-04-22 13:11:48 +02:00
|
|
|
s => sizzle(`iq[to="${muc_jid}"] query[xmlns="${Strophe.NS.MUC_ADMIN}"] item[affiliation="owner"]`, s).length
|
|
|
|
).pop());
|
|
|
|
const owner_list_stanza = $iq({
|
|
|
|
'from': 'coven@chat.shakespeare.lit',
|
|
|
|
'id': owner_IQ.getAttribute('id'),
|
|
|
|
'to': 'romeo@montague.lit/orchard',
|
|
|
|
'type': 'result'
|
|
|
|
}).c('query', {'xmlns': Strophe.NS.MUC_ADMIN});
|
|
|
|
members.filter(m => m.affiliation === 'owner').forEach(m => {
|
|
|
|
owner_list_stanza.c('item', {
|
|
|
|
'affiliation': m.affiliation,
|
|
|
|
'jid': m.jid,
|
|
|
|
'nick': m.nick
|
|
|
|
});
|
|
|
|
});
|
|
|
|
_converse.connection._dataRecv(mock.createRequest(owner_list_stanza));
|
|
|
|
}
|
|
|
|
return new Promise(resolve => _converse.api.listen.on('membersFetched', resolve));
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.receiveOwnMUCPresence = async function (_converse, muc_jid, nick) {
|
|
|
|
const sent_stanzas = _converse.connection.sent_stanzas;
|
|
|
|
await u.waitUntil(() => sent_stanzas.filter(iq => sizzle('presence history', iq).length).pop());
|
|
|
|
const presence = $pres({
|
|
|
|
to: _converse.connection.jid,
|
|
|
|
from: `${muc_jid}/${nick}`,
|
|
|
|
id: u.getUniqueId()
|
|
|
|
}).c('x').attrs({xmlns:'http://jabber.org/protocol/muc#user'})
|
|
|
|
.c('item').attrs({
|
|
|
|
affiliation: 'owner',
|
|
|
|
jid: _converse.bare_jid,
|
|
|
|
role: 'moderator'
|
|
|
|
}).up()
|
|
|
|
.c('status').attrs({code:'110'});
|
|
|
|
_converse.connection._dataRecv(mock.createRequest(presence));
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
mock.openAndEnterChatRoom = async function (_converse, muc_jid, nick, features=[], members=[]) {
|
|
|
|
muc_jid = muc_jid.toLowerCase();
|
|
|
|
const room_creation_promise = _converse.api.rooms.open(muc_jid);
|
|
|
|
await mock.getRoomFeatures(_converse, muc_jid, features);
|
|
|
|
await mock.waitForReservedNick(_converse, muc_jid, nick);
|
|
|
|
// The user has just entered the room (because join was called)
|
|
|
|
// and receives their own presence from the server.
|
|
|
|
// See example 24: https://xmpp.org/extensions/xep-0045.html#enter-pres
|
|
|
|
await mock.receiveOwnMUCPresence(_converse, muc_jid, nick);
|
|
|
|
|
|
|
|
await room_creation_promise;
|
|
|
|
const view = _converse.chatboxviews.get(muc_jid);
|
|
|
|
await u.waitUntil(() => (view.model.session.get('connection_status') === converse.ROOMSTATUS.ENTERED));
|
|
|
|
|
|
|
|
const affs = _converse.muc_fetch_members;
|
|
|
|
const all_affiliations = Array.isArray(affs) ? affs : (affs ? ['member', 'admin', 'owner'] : []);
|
|
|
|
await mock.returnMemberLists(_converse, muc_jid, members, all_affiliations);
|
|
|
|
await view.model.messages.fetched;
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.createContact = async function (_converse, name, ask, requesting, subscription) {
|
|
|
|
const jid = name.replace(/ /g,'.').toLowerCase() + '@montague.lit';
|
|
|
|
if (_converse.roster.get(jid)) {
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
const contact = await new Promise((success, error) => {
|
|
|
|
_converse.roster.create({
|
|
|
|
'ask': ask,
|
|
|
|
'fullname': name,
|
|
|
|
'jid': jid,
|
|
|
|
'requesting': requesting,
|
|
|
|
'subscription': subscription
|
|
|
|
}, {success, error});
|
|
|
|
});
|
|
|
|
return contact;
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.createContacts = async function (_converse, type, length) {
|
|
|
|
/* Create current (as opposed to requesting or pending) contacts
|
|
|
|
* for the user's roster.
|
|
|
|
*
|
|
|
|
* These contacts are not grouped. See below.
|
|
|
|
*/
|
|
|
|
await _converse.api.waitUntil('rosterContactsFetched');
|
|
|
|
let names, subscription, requesting, ask;
|
|
|
|
if (type === 'requesting') {
|
|
|
|
names = mock.req_names;
|
|
|
|
subscription = 'none';
|
|
|
|
requesting = true;
|
|
|
|
ask = null;
|
|
|
|
} else if (type === 'pending') {
|
|
|
|
names = mock.pend_names;
|
|
|
|
subscription = 'none';
|
|
|
|
requesting = false;
|
|
|
|
ask = 'subscribe';
|
|
|
|
} else if (type === 'current') {
|
|
|
|
names = mock.cur_names;
|
|
|
|
subscription = 'both';
|
|
|
|
requesting = false;
|
|
|
|
ask = null;
|
|
|
|
} else if (type === 'all') {
|
|
|
|
await this.createContacts(_converse, 'current');
|
|
|
|
await this.createContacts(_converse, 'requesting')
|
|
|
|
await this.createContacts(_converse, 'pending');
|
|
|
|
return this;
|
|
|
|
} else {
|
|
|
|
throw Error("Need to specify the type of contact to create");
|
|
|
|
}
|
|
|
|
const promises = names.slice(0, length).map(n => this.createContact(_converse, n, ask, requesting, subscription));
|
|
|
|
await Promise.all(promises);
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.waitForRoster = async function (_converse, type='current', length=-1, include_nick=true, grouped=true) {
|
|
|
|
const s = `iq[type="get"] query[xmlns="${Strophe.NS.ROSTER}"]`;
|
|
|
|
const iq = await u.waitUntil(() => _converse.connection.IQ_stanzas.filter(iq => sizzle(s, iq).length).pop());
|
|
|
|
|
|
|
|
const result = $iq({
|
|
|
|
'to': _converse.connection.jid,
|
|
|
|
'type': 'result',
|
|
|
|
'id': iq.getAttribute('id')
|
|
|
|
}).c('query', {
|
|
|
|
'xmlns': 'jabber:iq:roster'
|
|
|
|
});
|
|
|
|
if (type === 'pending' || type === 'all') {
|
|
|
|
const pend_names = (length > -1) ? mock.pend_names.slice(0, length) : mock.pend_names;
|
|
|
|
pend_names.map(name =>
|
|
|
|
result.c('item', {
|
|
|
|
jid: name.replace(/ /g,'.').toLowerCase() + '@montague.lit',
|
|
|
|
name: include_nick ? name : undefined,
|
|
|
|
subscription: 'none',
|
|
|
|
ask: 'subscribe'
|
|
|
|
}).up()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (type === 'current' || type === 'all') {
|
|
|
|
const cur_names = Object.keys(mock.current_contacts_map);
|
|
|
|
const names = (length > -1) ? cur_names.slice(0, length) : cur_names;
|
|
|
|
names.forEach(name => {
|
|
|
|
result.c('item', {
|
|
|
|
jid: name.replace(/ /g,'.').toLowerCase() + '@montague.lit',
|
|
|
|
name: include_nick ? name : undefined,
|
|
|
|
subscription: 'both',
|
|
|
|
ask: null
|
|
|
|
});
|
|
|
|
if (grouped) {
|
|
|
|
mock.current_contacts_map[name].forEach(g => result.c('group').t(g).up());
|
|
|
|
}
|
|
|
|
result.up();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
_converse.connection._dataRecv(mock.createRequest(result));
|
|
|
|
await _converse.api.waitUntil('rosterContactsFetched');
|
|
|
|
};
|
|
|
|
|
|
|
|
mock.createChatMessage = function (_converse, sender_jid, message) {
|
|
|
|
return $msg({
|
|
|
|
from: sender_jid,
|
|
|
|
to: _converse.connection.jid,
|
|
|
|
type: 'chat',
|
|
|
|
id: (new Date()).getTime()
|
|
|
|
})
|
|
|
|
.c('body').t(message).up()
|
2020-06-01 17:30:20 +02:00
|
|
|
.c('markable', {'xmlns': Strophe.NS.MARKERS}).up()
|
2020-04-22 13:11:48 +02:00
|
|
|
.c('active', {'xmlns': Strophe.NS.CHATSTATES}).tree();
|
|
|
|
}
|
|
|
|
|
|
|
|
mock.sendMessage = function (view, message) {
|
2020-05-15 14:33:31 +02:00
|
|
|
const promise = new Promise(resolve => view.model.messages.once('rendered', resolve));
|
2020-04-22 13:11:48 +02:00
|
|
|
view.el.querySelector('.chat-textarea').value = message;
|
|
|
|
view.onKeyDown({
|
|
|
|
target: view.el.querySelector('textarea.chat-textarea'),
|
2020-09-25 21:20:29 +02:00
|
|
|
preventDefault: () => {},
|
2020-04-22 13:11:48 +02:00
|
|
|
keyCode: 13
|
|
|
|
});
|
|
|
|
return promise;
|
|
|
|
};
|
|
|
|
|
2018-02-07 13:26:39 +01:00
|
|
|
|
2018-05-13 14:12:53 +02:00
|
|
|
window.libsignal = {
|
2018-07-25 12:59:12 +02:00
|
|
|
'SignalProtocolAddress': function (name, device_id) {
|
|
|
|
this.name = name;
|
|
|
|
this.deviceId = device_id;
|
|
|
|
},
|
|
|
|
'SessionCipher': function (storage, remote_address) {
|
|
|
|
this.remoteAddress = remote_address;
|
|
|
|
this.storage = storage;
|
|
|
|
this.encrypt = () => Promise.resolve({
|
2018-07-28 16:36:23 +02:00
|
|
|
'type': 1,
|
|
|
|
'body': 'c1ph3R73X7',
|
2019-02-12 14:21:45 +01:00
|
|
|
'registrationId': '1337'
|
2018-07-25 12:59:12 +02:00
|
|
|
});
|
2018-08-27 14:25:34 +02:00
|
|
|
this.decryptPreKeyWhisperMessage = (key_and_tag) => {
|
2018-08-31 18:49:47 +02:00
|
|
|
return Promise.resolve(key_and_tag);
|
2018-08-27 14:25:34 +02:00
|
|
|
};
|
2018-08-04 19:41:06 +02:00
|
|
|
this.decryptWhisperMessage = (key_and_tag) => {
|
2018-08-31 18:49:47 +02:00
|
|
|
return Promise.resolve(key_and_tag);
|
2018-08-04 19:41:06 +02:00
|
|
|
}
|
2018-07-25 12:59:12 +02:00
|
|
|
},
|
2018-11-18 19:14:22 +01:00
|
|
|
'SessionBuilder': function (storage, remote_address) { // eslint-disable-line no-unused-vars
|
2018-07-25 12:59:12 +02:00
|
|
|
this.processPreKey = function () {
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
},
|
2018-05-13 14:12:53 +02:00
|
|
|
'KeyHelper': {
|
|
|
|
'generateIdentityKeyPair': function () {
|
|
|
|
return Promise.resolve({
|
2018-07-21 21:51:50 +02:00
|
|
|
'pubKey': new TextEncoder('utf-8').encode('1234'),
|
|
|
|
'privKey': new TextEncoder('utf-8').encode('4321')
|
2018-05-13 14:12:53 +02:00
|
|
|
});
|
|
|
|
},
|
|
|
|
'generateRegistrationId': function () {
|
2018-05-13 16:04:52 +02:00
|
|
|
return '123456789';
|
2018-05-13 14:12:53 +02:00
|
|
|
},
|
|
|
|
'generatePreKey': function (keyid) {
|
|
|
|
return Promise.resolve({
|
|
|
|
'keyId': keyid,
|
|
|
|
'keyPair': {
|
2018-07-21 21:51:50 +02:00
|
|
|
'pubKey': new TextEncoder('utf-8').encode('1234'),
|
|
|
|
'privKey': new TextEncoder('utf-8').encode('4321')
|
2018-05-13 14:12:53 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
'generateSignedPreKey': function (identity_keypair, keyid) {
|
|
|
|
return Promise.resolve({
|
2018-07-21 21:51:50 +02:00
|
|
|
'signature': new TextEncoder('utf-8').encode('11112222333344445555'),
|
2018-05-13 14:12:53 +02:00
|
|
|
'keyId': keyid,
|
|
|
|
'keyPair': {
|
2018-07-21 21:51:50 +02:00
|
|
|
'pubKey': new TextEncoder('utf-8').encode('1234'),
|
|
|
|
'privKey': new TextEncoder('utf-8').encode('4321')
|
2018-05-13 14:12:53 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-11-20 18:26:41 +01:00
|
|
|
mock.default_muc_features = [
|
|
|
|
'http://jabber.org/protocol/muc',
|
|
|
|
'jabber:iq:register',
|
|
|
|
Strophe.NS.SID,
|
|
|
|
Strophe.NS.MAM,
|
|
|
|
'muc_passwordprotected',
|
|
|
|
'muc_hidden',
|
|
|
|
'muc_temporary',
|
|
|
|
'muc_open',
|
|
|
|
'muc_unmoderated',
|
|
|
|
'muc_anonymous'
|
|
|
|
];
|
|
|
|
|
2018-02-07 13:26:39 +01:00
|
|
|
mock.view_mode = 'overlayed';
|
|
|
|
|
2013-11-02 09:56:20 +01:00
|
|
|
// Names from http://www.fakenamegenerator.com/
|
|
|
|
mock.req_names = [
|
2019-09-04 19:11:26 +02:00
|
|
|
'Escalus, prince of Verona', 'The Nurse', 'Paris'
|
2013-11-02 09:56:20 +01:00
|
|
|
];
|
|
|
|
mock.pend_names = [
|
2019-10-24 14:29:15 +02:00
|
|
|
'Lord Capulet', 'Guard', 'Servant'
|
2013-11-02 09:56:20 +01:00
|
|
|
];
|
2019-10-24 14:29:15 +02:00
|
|
|
mock.current_contacts_map = {
|
|
|
|
'Mercutio': ['Colleagues', 'friends & acquaintences'],
|
|
|
|
'Juliet Capulet': ['friends & acquaintences'],
|
|
|
|
'Lady Montague': ['Colleagues', 'Family'],
|
|
|
|
'Lord Montague': ['Family'],
|
|
|
|
'Friar Laurence': ['friends & acquaintences'],
|
|
|
|
'Tybalt': ['friends & acquaintences'],
|
|
|
|
'Lady Capulet': ['ænemies'],
|
|
|
|
'Benviolo': ['friends & acquaintences'],
|
|
|
|
'Balthasar': ['Colleagues'],
|
|
|
|
'Peter': ['Colleagues'],
|
|
|
|
'Abram': ['Colleagues'],
|
|
|
|
'Sampson': ['Colleagues'],
|
|
|
|
'Gregory': ['friends & acquaintences'],
|
|
|
|
'Potpan': [],
|
|
|
|
'Friar John': []
|
|
|
|
};
|
|
|
|
|
|
|
|
const map = mock.current_contacts_map;
|
|
|
|
const groups_map = {};
|
|
|
|
Object.keys(map).forEach(k => {
|
|
|
|
const groups = map[k].length ? map[k] : ["Ungrouped"];
|
|
|
|
Object.values(groups).forEach(g => {
|
|
|
|
groups_map[g] = groups_map[g] ? [...groups_map[g], k] : [k]
|
|
|
|
});
|
|
|
|
});
|
|
|
|
mock.groups_map = groups_map;
|
|
|
|
|
|
|
|
mock.cur_names = Object.keys(mock.current_contacts_map);
|
2013-11-02 09:56:20 +01:00
|
|
|
mock.num_contacts = mock.req_names.length + mock.pend_names.length + mock.cur_names.length;
|
|
|
|
|
2014-08-18 20:37:38 +02:00
|
|
|
mock.groups = {
|
|
|
|
'colleagues': 3,
|
|
|
|
'friends & acquaintences': 3,
|
|
|
|
'Family': 4,
|
|
|
|
'ænemies': 3,
|
|
|
|
'Ungrouped': 2
|
|
|
|
};
|
|
|
|
|
2013-11-02 09:56:20 +01:00
|
|
|
mock.chatroom_names = [
|
2019-07-03 14:54:50 +02:00
|
|
|
'Dyon van de Wege',
|
|
|
|
'Thomas Kalb',
|
|
|
|
'Dirk Theissen',
|
|
|
|
'Felix Hofmann',
|
|
|
|
'Ka Lek',
|
|
|
|
'Anne Ebersbacher'
|
2013-11-02 09:56:20 +01:00
|
|
|
];
|
2015-03-01 11:58:07 +01:00
|
|
|
// TODO: need to also test other roles and affiliations
|
|
|
|
mock.chatroom_roles = {
|
|
|
|
'Anne Ebersbacher': { affiliation: "owner", role: "moderator" },
|
|
|
|
'Dirk Theissen': { affiliation: "admin", role: "moderator" },
|
2015-10-31 17:30:06 +01:00
|
|
|
'Dyon van de Wege': { affiliation: "member", role: "occupant" },
|
|
|
|
'Felix Hofmann': { affiliation: "member", role: "occupant" },
|
|
|
|
'Ka Lek': { affiliation: "member", role: "occupant" },
|
|
|
|
'Thomas Kalb': { affiliation: "member", role: "occupant" }
|
2015-03-01 11:58:07 +01:00
|
|
|
};
|
2013-11-02 09:56:20 +01:00
|
|
|
|
2013-11-03 10:36:31 +01:00
|
|
|
mock.event = {
|
|
|
|
'preventDefault': function () {}
|
|
|
|
};
|
|
|
|
|
2019-12-17 13:39:32 +01:00
|
|
|
function clearIndexedDB () {
|
|
|
|
const promise = u.getResolveablePromise();
|
2020-01-06 10:37:09 +01:00
|
|
|
const db_request = window.indexedDB.open("converse-test-persistent");
|
|
|
|
db_request.onsuccess = function () {
|
|
|
|
const db = db_request.result;
|
2019-12-17 13:39:32 +01:00
|
|
|
const bare_jid = "romeo@montague.lit";
|
2020-01-21 12:45:34 +01:00
|
|
|
let store;
|
2020-01-06 10:37:09 +01:00
|
|
|
try {
|
2020-01-21 12:45:34 +01:00
|
|
|
store= db.transaction([bare_jid], "readwrite").objectStore(bare_jid);
|
2020-01-06 10:37:09 +01:00
|
|
|
} catch (e) {
|
|
|
|
return promise.resolve();
|
|
|
|
}
|
|
|
|
const request = store.clear();
|
|
|
|
request.onsuccess = promise.resolve();
|
|
|
|
request.onerror = promise.resolve();
|
2019-12-17 13:39:32 +01:00
|
|
|
};
|
2020-01-28 17:45:34 +01:00
|
|
|
db_request.onerror = function (ev) {
|
|
|
|
return promise.reject(ev.target.error);
|
|
|
|
}
|
2019-12-17 13:39:32 +01:00
|
|
|
return promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
function clearStores () {
|
|
|
|
[localStorage, sessionStorage].forEach(
|
|
|
|
s => Object.keys(s).forEach(k => k.match(/^converse-test-/) && s.removeItem(k))
|
|
|
|
);
|
2020-03-31 17:22:05 +02:00
|
|
|
const cache_key = `converse.room-bookmarksromeo@montague.lit`;
|
|
|
|
window.sessionStorage.removeItem(cache_key+'fetched');
|
2019-12-17 13:39:32 +01:00
|
|
|
}
|
|
|
|
|
2020-04-22 12:10:39 +02:00
|
|
|
initConverse = async (settings) => {
|
2019-12-17 13:39:32 +01:00
|
|
|
clearStores();
|
|
|
|
await clearIndexedDB();
|
2017-05-08 20:44:35 +02:00
|
|
|
|
2020-02-18 23:04:30 +01:00
|
|
|
_converse = await converse.initialize(Object.assign({
|
2018-11-18 19:14:22 +01:00
|
|
|
'animate': false,
|
2017-05-08 20:44:35 +02:00
|
|
|
'auto_subscribe': false,
|
2019-06-03 07:58:51 +02:00
|
|
|
'bosh_service_url': 'montague.lit/http-bind',
|
2020-04-10 21:13:37 +02:00
|
|
|
'discover_connection_methods': false,
|
2019-12-03 13:54:28 +01:00
|
|
|
'enable_smacks': false,
|
2018-11-18 19:14:22 +01:00
|
|
|
'i18n': 'en',
|
2019-12-17 13:39:32 +01:00
|
|
|
// 'persistent_store': 'IndexedDB',
|
2019-12-03 13:54:28 +01:00
|
|
|
'loglevel': 'warn',
|
2017-05-08 20:44:35 +02:00
|
|
|
'no_trimming': true,
|
2018-11-18 19:14:22 +01:00
|
|
|
'play_sounds': false,
|
|
|
|
'use_emojione': false,
|
2019-10-24 14:29:15 +02:00
|
|
|
'view_mode': mock.view_mode
|
2017-04-05 11:01:31 +02:00
|
|
|
}, settings || {}));
|
2018-02-07 17:30:44 +01:00
|
|
|
|
2018-05-08 19:24:50 +02:00
|
|
|
_converse.ChatBoxViews.prototype.trimChat = function () {};
|
2018-05-25 16:25:02 +02:00
|
|
|
|
2018-05-08 14:46:00 +02:00
|
|
|
_converse.api.vcard.get = function (model, force) {
|
2019-10-24 14:29:15 +02:00
|
|
|
let jid;
|
2020-09-25 21:20:29 +02:00
|
|
|
if (typeof model === 'string' || model instanceof String) {
|
2019-10-24 14:29:15 +02:00
|
|
|
jid = model;
|
|
|
|
} else if (!model.get('vcard_updated') || force) {
|
|
|
|
jid = model.get('jid') || model.get('muc_jid');
|
|
|
|
}
|
|
|
|
let fullname;
|
|
|
|
if (!jid || jid == 'romeo@montague.lit') {
|
|
|
|
jid = 'romeo@montague.lit';
|
|
|
|
fullname = 'Romeo Montague' ;
|
|
|
|
} else {
|
|
|
|
const name = jid.split('@')[0].replace(/\./g, ' ').split(' ');
|
|
|
|
const last = name.length-1;
|
|
|
|
name[0] = name[0].charAt(0).toUpperCase()+name[0].slice(1);
|
|
|
|
name[last] = name[last].charAt(0).toUpperCase()+name[last].slice(1);
|
|
|
|
fullname = name.join(' ');
|
|
|
|
}
|
|
|
|
const vcard = $iq().c('vCard').c('FN').t(fullname).nodeTree;
|
|
|
|
return {
|
|
|
|
'vcard': vcard,
|
2020-09-25 21:20:29 +02:00
|
|
|
'fullname': vcard.querySelector('FN')?.textContent,
|
|
|
|
'image': vcard.querySelector('PHOTO BINVAL')?.textContent,
|
|
|
|
'image_type': vcard.querySelector('PHOTO TYPE')?.textContent,
|
|
|
|
'url': vcard.querySelector('URL')?.textContent,
|
2019-10-24 14:29:15 +02:00
|
|
|
'vcard_updated': dayjs().format(),
|
|
|
|
'vcard_error': undefined
|
|
|
|
};
|
2018-05-08 14:46:00 +02:00
|
|
|
};
|
2020-09-25 21:20:29 +02:00
|
|
|
if (settings?.auto_login !== false) {
|
2019-06-25 22:53:39 +02:00
|
|
|
_converse.api.user.login('romeo@montague.lit/orchard', 'secret');
|
2018-05-29 12:00:23 +02:00
|
|
|
await _converse.api.waitUntil('afterResourceBinding');
|
2018-05-08 19:24:50 +02:00
|
|
|
}
|
2017-08-15 21:23:30 +02:00
|
|
|
window.converse_disable_effects = true;
|
2017-07-15 15:15:37 +02:00
|
|
|
return _converse;
|
2017-04-05 11:01:31 +02:00
|
|
|
}
|
2020-04-22 12:10:39 +02:00
|
|
|
});
|
2020-04-22 13:11:48 +02:00
|
|
|
|
|
|
|
converse.load();
|