omemo.js 79 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527
  1. /*global mock, converse */
  2. const { $iq, $pres, $msg, _, omemo, Strophe } = converse.env;
  3. const u = converse.env.utils;
  4. async function deviceListFetched (_converse, jid) {
  5. const selector = `iq[to="${jid}"] items[node="eu.siacs.conversations.axolotl.devicelist"]`;
  6. const stanza = await u.waitUntil(
  7. () => Array.from(_converse.connection.IQ_stanzas).filter(iq => iq.querySelector(selector)).pop()
  8. );
  9. await u.waitUntil(() => _converse.devicelists.get(jid));
  10. return stanza;
  11. }
  12. function ownDeviceHasBeenPublished (_converse) {
  13. return _.filter(
  14. Array.from(_converse.connection.IQ_stanzas),
  15. iq => iq.querySelector('iq[from="'+_converse.bare_jid+'"] publish[node="eu.siacs.conversations.axolotl.devicelist"]')
  16. ).pop();
  17. }
  18. function bundleHasBeenPublished (_converse) {
  19. const selector = 'publish[node="eu.siacs.conversations.axolotl.bundles:123456789"]';
  20. return Array.from(_converse.connection.IQ_stanzas).filter(iq => iq.querySelector(selector)).pop();
  21. }
  22. function bundleFetched (_converse, jid, device_id) {
  23. return _.filter(
  24. Array.from(_converse.connection.IQ_stanzas),
  25. iq => iq.querySelector(`iq[to="${jid}"] items[node="eu.siacs.conversations.axolotl.bundles:${device_id}"]`)
  26. ).pop();
  27. }
  28. async function initializedOMEMO (_converse) {
  29. await mock.waitUntilDiscoConfirmed(
  30. _converse, _converse.bare_jid,
  31. [{'category': 'pubsub', 'type': 'pep'}],
  32. ['http://jabber.org/protocol/pubsub#publish-options']
  33. );
  34. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, _converse.bare_jid));
  35. let stanza = $iq({
  36. 'from': _converse.bare_jid,
  37. 'id': iq_stanza.getAttribute('id'),
  38. 'to': _converse.bare_jid,
  39. 'type': 'result',
  40. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  41. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  42. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  43. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  44. .c('device', {'id': '482886413b977930064a5888b92134fe'});
  45. _converse.connection._dataRecv(mock.createRequest(stanza));
  46. iq_stanza = await u.waitUntil(() => ownDeviceHasBeenPublished(_converse))
  47. stanza = $iq({
  48. 'from': _converse.bare_jid,
  49. 'id': iq_stanza.getAttribute('id'),
  50. 'to': _converse.bare_jid,
  51. 'type': 'result'});
  52. _converse.connection._dataRecv(mock.createRequest(stanza));
  53. iq_stanza = await u.waitUntil(() => bundleHasBeenPublished(_converse))
  54. stanza = $iq({
  55. 'from': _converse.bare_jid,
  56. 'id': iq_stanza.getAttribute('id'),
  57. 'to': _converse.bare_jid,
  58. 'type': 'result'});
  59. _converse.connection._dataRecv(mock.createRequest(stanza));
  60. await _converse.api.waitUntil('OMEMOInitialized');
  61. }
  62. describe("The OMEMO module", function() {
  63. it("adds methods for encrypting and decrypting messages via AES GCM",
  64. mock.initConverse(
  65. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  66. async function (done, _converse) {
  67. const message = 'This message will be encrypted'
  68. await mock.waitForRoster(_converse, 'current', 1);
  69. const payload = await omemo.encryptMessage(message);
  70. const result = await omemo.decryptMessage(payload);
  71. expect(result).toBe(message);
  72. done();
  73. }));
  74. it("enables encrypted messages to be sent and received",
  75. mock.initConverse(
  76. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  77. async function (done, _converse) {
  78. let sent_stanza;
  79. await mock.waitForRoster(_converse, 'current', 1);
  80. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  81. await u.waitUntil(() => initializedOMEMO(_converse));
  82. await mock.openChatBoxFor(_converse, contact_jid);
  83. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  84. let stanza = $iq({
  85. 'from': contact_jid,
  86. 'id': iq_stanza.getAttribute('id'),
  87. 'to': _converse.connection.jid,
  88. 'type': 'result',
  89. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  90. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  91. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  92. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  93. .c('device', {'id': '555'});
  94. _converse.connection._dataRecv(mock.createRequest(stanza));
  95. await u.waitUntil(() => _converse.omemo_store);
  96. const devicelist = _converse.devicelists.get({'jid': contact_jid});
  97. await u.waitUntil(() => devicelist.devices.length === 1);
  98. const view = _converse.chatboxviews.get(contact_jid);
  99. view.model.set('omemo_active', true);
  100. const textarea = view.el.querySelector('.chat-textarea');
  101. textarea.value = 'This message will be encrypted';
  102. view.onKeyDown({
  103. target: textarea,
  104. preventDefault: function preventDefault () {},
  105. keyCode: 13 // Enter
  106. });
  107. iq_stanza = await u.waitUntil(() => bundleFetched(_converse, contact_jid, '555'));
  108. stanza = $iq({
  109. 'from': contact_jid,
  110. 'id': iq_stanza.getAttribute('id'),
  111. 'to': _converse.bare_jid,
  112. 'type': 'result',
  113. }).c('pubsub', {
  114. 'xmlns': 'http://jabber.org/protocol/pubsub'
  115. }).c('items', {'node': "eu.siacs.conversations.axolotl.bundles:555"})
  116. .c('item')
  117. .c('bundle', {'xmlns': 'eu.siacs.conversations.axolotl'})
  118. .c('signedPreKeyPublic', {'signedPreKeyId': '4223'}).t(btoa('1111')).up()
  119. .c('signedPreKeySignature').t(btoa('2222')).up()
  120. .c('identityKey').t(btoa('3333')).up()
  121. .c('prekeys')
  122. .c('preKeyPublic', {'preKeyId': '1'}).t(btoa('1001')).up()
  123. .c('preKeyPublic', {'preKeyId': '2'}).t(btoa('1002')).up()
  124. .c('preKeyPublic', {'preKeyId': '3'}).t(btoa('1003'));
  125. _converse.connection._dataRecv(mock.createRequest(stanza));
  126. iq_stanza = await u.waitUntil(() => bundleFetched(_converse, _converse.bare_jid, '482886413b977930064a5888b92134fe'));
  127. stanza = $iq({
  128. 'from': _converse.bare_jid,
  129. 'id': iq_stanza.getAttribute('id'),
  130. 'to': _converse.bare_jid,
  131. 'type': 'result',
  132. }).c('pubsub', {
  133. 'xmlns': 'http://jabber.org/protocol/pubsub'
  134. }).c('items', {'node': "eu.siacs.conversations.axolotl.bundles:482886413b977930064a5888b92134fe"})
  135. .c('item')
  136. .c('bundle', {'xmlns': 'eu.siacs.conversations.axolotl'})
  137. .c('signedPreKeyPublic', {'signedPreKeyId': '4223'}).t(btoa('100000')).up()
  138. .c('signedPreKeySignature').t(btoa('200000')).up()
  139. .c('identityKey').t(btoa('300000')).up()
  140. .c('prekeys')
  141. .c('preKeyPublic', {'preKeyId': '1'}).t(btoa('1991')).up()
  142. .c('preKeyPublic', {'preKeyId': '2'}).t(btoa('1992')).up()
  143. .c('preKeyPublic', {'preKeyId': '3'}).t(btoa('1993'));
  144. spyOn(_converse.connection, 'send').and.callFake(stanza => { sent_stanza = stanza });
  145. _converse.connection._dataRecv(mock.createRequest(stanza));
  146. await u.waitUntil(() => sent_stanza);
  147. expect(sent_stanza.toLocaleString()).toBe(
  148. `<message from="romeo@montague.lit/orchard" id="${sent_stanza.nodeTree.getAttribute("id")}" `+
  149. `to="mercutio@montague.lit" `+
  150. `type="chat" xmlns="jabber:client">`+
  151. `<body>This is an OMEMO encrypted message which your client doesn’t seem to support. Find more information on https://conversations.im/omemo</body>`+
  152. `<request xmlns="urn:xmpp:receipts"/>`+
  153. `<encrypted xmlns="eu.siacs.conversations.axolotl">`+
  154. `<header sid="123456789">`+
  155. `<key rid="482886413b977930064a5888b92134fe">YzFwaDNSNzNYNw==</key>`+
  156. `<key rid="555">YzFwaDNSNzNYNw==</key>`+
  157. `<iv>${sent_stanza.nodeTree.querySelector("iv").textContent}</iv>`+
  158. `</header>`+
  159. `<payload>${sent_stanza.nodeTree.querySelector("payload").textContent}</payload>`+
  160. `</encrypted>`+
  161. `<store xmlns="urn:xmpp:hints"/>`+
  162. `</message>`);
  163. // Test reception of an encrypted message
  164. let obj = await omemo.encryptMessage('This is an encrypted message from the contact')
  165. // XXX: Normally the key will be encrypted via libsignal.
  166. // However, we're mocking libsignal in the tests, so we include it as plaintext in the message.
  167. stanza = $msg({
  168. 'from': contact_jid,
  169. 'to': _converse.connection.jid,
  170. 'type': 'chat',
  171. 'id': _converse.connection.getUniqueId()
  172. }).c('body').t('This is a fallback message').up()
  173. .c('encrypted', {'xmlns': Strophe.NS.OMEMO})
  174. .c('header', {'sid': '555'})
  175. .c('key', {'rid': _converse.omemo_store.get('device_id')}).t(u.arrayBufferToBase64(obj.key_and_tag)).up()
  176. .c('iv').t(obj.iv)
  177. .up().up()
  178. .c('payload').t(obj.payload);
  179. _converse.connection._dataRecv(mock.createRequest(stanza));
  180. await new Promise(resolve => view.model.messages.once('rendered', resolve));
  181. expect(view.model.messages.length).toBe(2);
  182. expect(view.el.querySelectorAll('.chat-msg__body')[1].textContent.trim())
  183. .toBe('This is an encrypted message from the contact');
  184. // #1193 Check for a received message without <body> tag
  185. obj = await omemo.encryptMessage('Another received encrypted message without fallback')
  186. stanza = $msg({
  187. 'from': contact_jid,
  188. 'to': _converse.connection.jid,
  189. 'type': 'chat',
  190. 'id': _converse.connection.getUniqueId()
  191. }).c('encrypted', {'xmlns': Strophe.NS.OMEMO})
  192. .c('header', {'sid': '555'})
  193. .c('key', {'rid': _converse.omemo_store.get('device_id')}).t(u.arrayBufferToBase64(obj.key_and_tag)).up()
  194. .c('iv').t(obj.iv)
  195. .up().up()
  196. .c('payload').t(obj.payload);
  197. _converse.connection._dataRecv(mock.createRequest(stanza));
  198. await new Promise(resolve => view.model.messages.once('rendered', resolve));
  199. await u.waitUntil(() => view.model.messages.length > 1);
  200. expect(view.model.messages.length).toBe(3);
  201. expect(view.el.querySelectorAll('.chat-msg__body')[2].textContent.trim())
  202. .toBe('Another received encrypted message without fallback');
  203. done();
  204. }));
  205. it("enables encrypted groupchat messages to be sent and received",
  206. mock.initConverse(
  207. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  208. async function (done, _converse) {
  209. // MEMO encryption works only in members only conferences
  210. // that are non-anonymous.
  211. const features = [
  212. 'http://jabber.org/protocol/muc',
  213. 'jabber:iq:register',
  214. 'muc_passwordprotected',
  215. 'muc_hidden',
  216. 'muc_temporary',
  217. 'muc_membersonly',
  218. 'muc_unmoderated',
  219. 'muc_nonanonymous'
  220. ];
  221. await mock.openAndEnterChatRoom(_converse, 'lounge@montague.lit', 'romeo', features);
  222. const view = _converse.chatboxviews.get('lounge@montague.lit');
  223. await u.waitUntil(() => initializedOMEMO(_converse));
  224. const toolbar = view.el.querySelector('.chat-toolbar');
  225. toolbar.querySelector('.toggle-omemo').click();
  226. expect(view.model.get('omemo_active')).toBe(true);
  227. // newguy enters the room
  228. const contact_jid = 'newguy@montague.lit';
  229. let stanza = $pres({
  230. 'to': 'romeo@montague.lit/orchard',
  231. 'from': 'lounge@montague.lit/newguy'
  232. })
  233. .c('x', {xmlns: Strophe.NS.MUC_USER})
  234. .c('item', {
  235. 'affiliation': 'none',
  236. 'jid': 'newguy@montague.lit/_converse.js-290929789',
  237. 'role': 'participant'
  238. }).tree();
  239. _converse.connection._dataRecv(mock.createRequest(stanza));
  240. // Wait for Converse to fetch newguy's device list
  241. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  242. expect(Strophe.serialize(iq_stanza)).toBe(
  243. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="${contact_jid}" type="get" xmlns="jabber:client">`+
  244. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  245. `<items node="eu.siacs.conversations.axolotl.devicelist"/>`+
  246. `</pubsub>`+
  247. `</iq>`);
  248. // The server returns his device list
  249. stanza = $iq({
  250. 'from': contact_jid,
  251. 'id': iq_stanza.getAttribute('id'),
  252. 'to': _converse.bare_jid,
  253. 'type': 'result',
  254. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  255. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  256. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  257. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  258. .c('device', {'id': '4e30f35051b7b8b42abe083742187228'}).up()
  259. _converse.connection._dataRecv(mock.createRequest(stanza));
  260. await u.waitUntil(() => _converse.omemo_store);
  261. expect(_converse.devicelists.length).toBe(2);
  262. await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  263. const devicelist = _converse.devicelists.get(contact_jid);
  264. expect(devicelist.devices.length).toBe(1);
  265. expect(devicelist.devices.at(0).get('id')).toBe('4e30f35051b7b8b42abe083742187228');
  266. expect(view.model.get('omemo_active')).toBe(true);
  267. const icon = toolbar.querySelector('.toggle-omemo converse-icon');
  268. expect(u.hasClass('fa-unlock', icon)).toBe(false);
  269. expect(u.hasClass('fa-lock', icon)).toBe(true);
  270. const textarea = view.el.querySelector('.chat-textarea');
  271. textarea.value = 'This message will be encrypted';
  272. view.onKeyDown({
  273. target: textarea,
  274. preventDefault: function preventDefault () {},
  275. keyCode: 13 // Enter
  276. });
  277. iq_stanza = await u.waitUntil(() => bundleFetched(_converse, contact_jid, '4e30f35051b7b8b42abe083742187228'), 1000);
  278. console.log("Bundle fetched 4e30f35051b7b8b42abe083742187228");
  279. stanza = $iq({
  280. 'from': contact_jid,
  281. 'id': iq_stanza.getAttribute('id'),
  282. 'to': _converse.bare_jid,
  283. 'type': 'result',
  284. }).c('pubsub', {
  285. 'xmlns': 'http://jabber.org/protocol/pubsub'
  286. }).c('items', {'node': "eu.siacs.conversations.axolotl.bundles:4e30f35051b7b8b42abe083742187228"})
  287. .c('item')
  288. .c('bundle', {'xmlns': 'eu.siacs.conversations.axolotl'})
  289. .c('signedPreKeyPublic', {'signedPreKeyId': '4223'}).t(btoa('1111')).up()
  290. .c('signedPreKeySignature').t(btoa('2222')).up()
  291. .c('identityKey').t(btoa('3333')).up()
  292. .c('prekeys')
  293. .c('preKeyPublic', {'preKeyId': '1'}).t(btoa('1001')).up()
  294. .c('preKeyPublic', {'preKeyId': '2'}).t(btoa('1002')).up()
  295. .c('preKeyPublic', {'preKeyId': '3'}).t(btoa('1003'));
  296. _converse.connection._dataRecv(mock.createRequest(stanza));
  297. iq_stanza = await u.waitUntil(() => bundleFetched(_converse, _converse.bare_jid, '482886413b977930064a5888b92134fe'), 1000);
  298. console.log("Bundle fetched 482886413b977930064a5888b92134fe");
  299. stanza = $iq({
  300. 'from': _converse.bare_jid,
  301. 'id': iq_stanza.getAttribute('id'),
  302. 'to': _converse.bare_jid,
  303. 'type': 'result',
  304. }).c('pubsub', {
  305. 'xmlns': 'http://jabber.org/protocol/pubsub'
  306. }).c('items', {'node': "eu.siacs.conversations.axolotl.bundles:482886413b977930064a5888b92134fe"})
  307. .c('item')
  308. .c('bundle', {'xmlns': 'eu.siacs.conversations.axolotl'})
  309. .c('signedPreKeyPublic', {'signedPreKeyId': '4223'}).t(btoa('100000')).up()
  310. .c('signedPreKeySignature').t(btoa('200000')).up()
  311. .c('identityKey').t(btoa('300000')).up()
  312. .c('prekeys')
  313. .c('preKeyPublic', {'preKeyId': '1'}).t(btoa('1991')).up()
  314. .c('preKeyPublic', {'preKeyId': '2'}).t(btoa('1992')).up()
  315. .c('preKeyPublic', {'preKeyId': '3'}).t(btoa('1993'));
  316. spyOn(_converse.connection, 'send');
  317. _converse.connection._dataRecv(mock.createRequest(stanza));
  318. await u.waitUntil(() => _converse.connection.send.calls.count(), 1000);
  319. const sent_stanza = _converse.connection.send.calls.all()[0].args[0];
  320. expect(Strophe.serialize(sent_stanza)).toBe(
  321. `<message from="romeo@montague.lit/orchard" `+
  322. `id="${sent_stanza.nodeTree.getAttribute("id")}" `+
  323. `to="lounge@montague.lit" `+
  324. `type="groupchat" `+
  325. `xmlns="jabber:client">`+
  326. `<body>This is an OMEMO encrypted message which your client doesn’t seem to support. Find more information on https://conversations.im/omemo</body>`+
  327. `<encrypted xmlns="eu.siacs.conversations.axolotl">`+
  328. `<header sid="123456789">`+
  329. `<key rid="482886413b977930064a5888b92134fe">YzFwaDNSNzNYNw==</key>`+
  330. `<key rid="4e30f35051b7b8b42abe083742187228">YzFwaDNSNzNYNw==</key>`+
  331. `<iv>${sent_stanza.nodeTree.querySelector("iv").textContent}</iv>`+
  332. `</header>`+
  333. `<payload>${sent_stanza.nodeTree.querySelector("payload").textContent}</payload>`+
  334. `</encrypted>`+
  335. `<store xmlns="urn:xmpp:hints"/>`+
  336. `</message>`);
  337. done();
  338. }));
  339. it("will create a new device based on a received carbon message",
  340. mock.initConverse(
  341. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  342. async function (done, _converse) {
  343. await mock.waitUntilDiscoConfirmed(_converse, _converse.bare_jid, [], [Strophe.NS.SID]);
  344. await mock.waitForRoster(_converse, 'current', 1);
  345. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  346. await u.waitUntil(() => initializedOMEMO(_converse));
  347. await mock.openChatBoxFor(_converse, contact_jid);
  348. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  349. const my_devicelist = _converse.devicelists.get({'jid': _converse.bare_jid});
  350. expect(my_devicelist.devices.length).toBe(2);
  351. const stanza = $iq({
  352. 'from': contact_jid,
  353. 'id': iq_stanza.getAttribute('id'),
  354. 'to': _converse.connection.jid,
  355. 'type': 'result',
  356. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  357. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  358. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  359. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  360. .c('device', {'id': '555'});
  361. _converse.connection._dataRecv(mock.createRequest(stanza));
  362. await u.waitUntil(() => _converse.omemo_store);
  363. const contact_devicelist = _converse.devicelists.get({'jid': contact_jid});
  364. await u.waitUntil(() => contact_devicelist.devices.length === 1);
  365. const view = _converse.chatboxviews.get(contact_jid);
  366. view.model.set('omemo_active', true);
  367. // Test reception of an encrypted carbon message
  368. const obj = await omemo.encryptMessage('This is an encrypted carbon message from another device of mine')
  369. const carbon = u.toStanza(`
  370. <message xmlns="jabber:client" to="romeo@montague.lit/orchard" from="romeo@montague.lit" type="chat">
  371. <sent xmlns="urn:xmpp:carbons:2">
  372. <forwarded xmlns="urn:xmpp:forward:0">
  373. <message xmlns="jabber:client"
  374. from="romeo@montague.lit/gajim.HE02SW1L"
  375. xml:lang="en"
  376. to="${contact_jid}/gajim.0LATM5V2"
  377. type="chat" id="87141781-61d6-4eb3-9a31-429935a61b76">
  378. <archived xmlns="urn:xmpp:mam:tmp" by="romeo@montague.lit" id="1554033877043470"/>
  379. <stanza-id xmlns="urn:xmpp:sid:0" by="romeo@montague.lit" id="1554033877043470"/>
  380. <request xmlns="urn:xmpp:receipts"/>
  381. <active xmlns="http://jabber.org/protocol/chatstates"/>
  382. <origin-id xmlns="urn:xmpp:sid:0" id="87141781-61d6-4eb3-9a31-429935a61b76"/>
  383. <encrypted xmlns="eu.siacs.conversations.axolotl">
  384. <header sid="988349631">
  385. <key rid="${_converse.omemo_store.get('device_id')}"
  386. prekey="true">${u.arrayBufferToBase64(obj.key_and_tag)}</key>
  387. <iv>${obj.iv}</iv>
  388. </header>
  389. <payload>${obj.payload}</payload>
  390. </encrypted>
  391. <encryption xmlns="urn:xmpp:eme:0" namespace="eu.siacs.conversations.axolotl" name="OMEMO"/>
  392. <store xmlns="urn:xmpp:hints"/>
  393. </message>
  394. </forwarded>
  395. </sent>
  396. </message>
  397. `);
  398. _converse.connection._dataRecv(mock.createRequest(carbon));
  399. await new Promise(resolve => view.model.messages.once('rendered', resolve));
  400. expect(view.model.messages.length).toBe(1);
  401. expect(view.el.querySelector('.chat-msg__text').textContent.trim())
  402. .toBe('This is an encrypted carbon message from another device of mine');
  403. expect(contact_devicelist.devices.length).toBe(1);
  404. // Check that the new device id has been added to my devices
  405. expect(my_devicelist.devices.length).toBe(3);
  406. expect(my_devicelist.devices.at(0).get('id')).toBe('482886413b977930064a5888b92134fe');
  407. expect(my_devicelist.devices.at(1).get('id')).toBe('123456789');
  408. expect(my_devicelist.devices.at(2).get('id')).toBe('988349631');
  409. expect(my_devicelist.devices.get('988349631').get('active')).toBe(true);
  410. const textarea = view.el.querySelector('.chat-textarea');
  411. textarea.value = 'This is an encrypted message from this device';
  412. view.onKeyDown({
  413. target: textarea,
  414. preventDefault: function preventDefault () {},
  415. keyCode: 13 // Enter
  416. });
  417. iq_stanza = await u.waitUntil(() => bundleFetched(_converse, _converse.bare_jid, '988349631'));
  418. expect(Strophe.serialize(iq_stanza)).toBe(
  419. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="${_converse.bare_jid}" type="get" xmlns="jabber:client">`+
  420. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  421. `<items node="eu.siacs.conversations.axolotl.bundles:988349631"/>`+
  422. `</pubsub>`+
  423. `</iq>`);
  424. done();
  425. }));
  426. it("gracefully handles auth errors when trying to send encrypted groupchat messages",
  427. mock.initConverse(
  428. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  429. async function (done, _converse) {
  430. // MEMO encryption works only in members only conferences
  431. // that are non-anonymous.
  432. const features = [
  433. 'http://jabber.org/protocol/muc',
  434. 'jabber:iq:register',
  435. 'muc_passwordprotected',
  436. 'muc_hidden',
  437. 'muc_temporary',
  438. 'muc_membersonly',
  439. 'muc_unmoderated',
  440. 'muc_nonanonymous'
  441. ];
  442. await mock.openAndEnterChatRoom(_converse, 'lounge@montague.lit', 'romeo', features);
  443. const view = _converse.chatboxviews.get('lounge@montague.lit');
  444. await u.waitUntil(() => initializedOMEMO(_converse));
  445. const contact_jid = 'newguy@montague.lit';
  446. let stanza = $pres({
  447. 'to': 'romeo@montague.lit/orchard',
  448. 'from': 'lounge@montague.lit/newguy'
  449. })
  450. .c('x', {xmlns: Strophe.NS.MUC_USER})
  451. .c('item', {
  452. 'affiliation': 'none',
  453. 'jid': 'newguy@montague.lit/_converse.js-290929789',
  454. 'role': 'participant'
  455. }).tree();
  456. _converse.connection._dataRecv(mock.createRequest(stanza));
  457. const toolbar = view.el.querySelector('.chat-toolbar');
  458. const toggle = toolbar.querySelector('.toggle-omemo');
  459. toggle.click();
  460. expect(view.model.get('omemo_active')).toBe(true);
  461. expect(view.model.get('omemo_supported')).toBe(true);
  462. const textarea = view.el.querySelector('.chat-textarea');
  463. textarea.value = 'This message will be encrypted';
  464. view.onKeyDown({
  465. target: textarea,
  466. preventDefault: function preventDefault () {},
  467. keyCode: 13 // Enter
  468. });
  469. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  470. expect(Strophe.serialize(iq_stanza)).toBe(
  471. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="${contact_jid}" type="get" xmlns="jabber:client">`+
  472. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  473. `<items node="eu.siacs.conversations.axolotl.devicelist"/>`+
  474. `</pubsub>`+
  475. `</iq>`);
  476. stanza = $iq({
  477. 'from': contact_jid,
  478. 'id': iq_stanza.getAttribute('id'),
  479. 'to': _converse.bare_jid,
  480. 'type': 'result',
  481. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  482. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  483. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  484. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  485. .c('device', {'id': '4e30f35051b7b8b42abe083742187228'}).up()
  486. _converse.connection._dataRecv(mock.createRequest(stanza));
  487. await u.waitUntil(() => _converse.omemo_store);
  488. expect(_converse.devicelists.length).toBe(2);
  489. const devicelist = _converse.devicelists.get(contact_jid);
  490. await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  491. expect(devicelist.devices.length).toBe(1);
  492. expect(devicelist.devices.at(0).get('id')).toBe('4e30f35051b7b8b42abe083742187228');
  493. iq_stanza = await u.waitUntil(() => bundleFetched(_converse, _converse.bare_jid, '482886413b977930064a5888b92134fe'));
  494. stanza = $iq({
  495. 'from': _converse.bare_jid,
  496. 'id': iq_stanza.getAttribute('id'),
  497. 'to': _converse.bare_jid,
  498. 'type': 'result',
  499. }).c('pubsub', {
  500. 'xmlns': 'http://jabber.org/protocol/pubsub'
  501. }).c('items', {'node': "eu.siacs.conversations.axolotl.bundles:482886413b977930064a5888b92134fe"})
  502. .c('item')
  503. .c('bundle', {'xmlns': 'eu.siacs.conversations.axolotl'})
  504. .c('signedPreKeyPublic', {'signedPreKeyId': '4223'}).t(btoa('100000')).up()
  505. .c('signedPreKeySignature').t(btoa('200000')).up()
  506. .c('identityKey').t(btoa('300000')).up()
  507. .c('prekeys')
  508. .c('preKeyPublic', {'preKeyId': '1'}).t(btoa('1991')).up()
  509. .c('preKeyPublic', {'preKeyId': '2'}).t(btoa('1992')).up()
  510. .c('preKeyPublic', {'preKeyId': '3'}).t(btoa('1993'));
  511. iq_stanza = await u.waitUntil(() => bundleFetched(_converse, contact_jid, '4e30f35051b7b8b42abe083742187228'));
  512. /* <iq xmlns="jabber:client" to="jc@opkode.com/converse.js-34183907" type="error" id="945c8ab3-b561-4d8a-92da-77c226bb1689:sendIQ" from="joris@konuro.net">
  513. * <pubsub xmlns="http://jabber.org/protocol/pubsub">
  514. * <items node="eu.siacs.conversations.axolotl.bundles:7580"/>
  515. * </pubsub>
  516. * <error code="401" type="auth">
  517. * <presence-subscription-required xmlns="http://jabber.org/protocol/pubsub#errors"/>
  518. * <not-authorized xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"/>
  519. * </error>
  520. * </iq>
  521. */
  522. stanza = $iq({
  523. 'from': contact_jid,
  524. 'id': iq_stanza.getAttribute('id'),
  525. 'to': _converse.bare_jid,
  526. 'type': 'result',
  527. }).c('pubsub', {'xmlns': 'http://jabber.org/protocol/pubsub'})
  528. .c('items', {'node': "eu.siacs.conversations.axolotl.bundles:4e30f35051b7b8b42abe083742187228"}).up().up()
  529. .c('error', {'code': '401', 'type': 'auth'})
  530. .c('presence-subscription-required', {'xmlns':"http://jabber.org/protocol/pubsub#errors" }).up()
  531. .c('not-authorized', {'xmlns': "urn:ietf:params:xml:ns:xmpp-stanzas"});
  532. _converse.connection._dataRecv(mock.createRequest(stanza));
  533. await u.waitUntil(() => document.querySelectorAll('.alert-danger').length, 2000);
  534. const header = document.querySelector('.alert-danger .modal-title');
  535. expect(header.textContent).toBe("Error");
  536. expect(u.ancestor(header, '.modal-content').querySelector('.modal-body p').textContent.trim())
  537. .toBe("Sorry, we're unable to send an encrypted message because newguy@montague.lit requires you "+
  538. "to be subscribed to their presence in order to see their OMEMO information");
  539. expect(view.model.get('omemo_supported')).toBe(false);
  540. expect(view.el.querySelector('.chat-textarea').value).toBe('This message will be encrypted');
  541. done();
  542. }));
  543. it("can receive a PreKeySignalMessage",
  544. mock.initConverse(
  545. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  546. async function (done, _converse) {
  547. _converse.NUM_PREKEYS = 5; // Restrict to 5, otherwise the resulting stanza is too large to easily test
  548. await mock.waitForRoster(_converse, 'current', 1);
  549. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  550. await u.waitUntil(() => initializedOMEMO(_converse));
  551. const obj = await omemo.encryptMessage('This is an encrypted message from the contact');
  552. // XXX: Normally the key will be encrypted via libsignal.
  553. // However, we're mocking libsignal in the tests, so we include
  554. // it as plaintext in the message.
  555. let stanza = $msg({
  556. 'from': contact_jid,
  557. 'to': _converse.connection.jid,
  558. 'type': 'chat',
  559. 'id': 'qwerty'
  560. }).c('body').t('This is a fallback message').up()
  561. .c('encrypted', {'xmlns': Strophe.NS.OMEMO})
  562. .c('header', {'sid': '555'})
  563. .c('key', {
  564. 'prekey': 'true',
  565. 'rid': _converse.omemo_store.get('device_id')
  566. }).t(u.arrayBufferToBase64(obj.key_and_tag)).up()
  567. .c('iv').t(obj.iv)
  568. .up().up()
  569. .c('payload').t(obj.payload);
  570. const generateMissingPreKeys = _converse.omemo_store.generateMissingPreKeys;
  571. spyOn(_converse.omemo_store, 'generateMissingPreKeys').and.callFake(() => {
  572. // Since it's difficult to override
  573. // decryptPreKeyWhisperMessage, where a prekey will be
  574. // removed from the store, we do it here, before the
  575. // missing prekeys are generated.
  576. _converse.omemo_store.removePreKey(1);
  577. return generateMissingPreKeys.apply(_converse.omemo_store, arguments);
  578. });
  579. _converse.connection._dataRecv(mock.createRequest(stanza));
  580. let iq_stanza = await deviceListFetched(_converse, contact_jid);
  581. stanza = $iq({
  582. 'from': contact_jid,
  583. 'id': iq_stanza.getAttribute('id'),
  584. 'to': _converse.connection.jid,
  585. 'type': 'result',
  586. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  587. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  588. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  589. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  590. .c('device', {'id': '555'});
  591. // XXX: the bundle gets published twice, we want to make sure
  592. // that we wait for the 2nd, so we clear all the already sent
  593. // stanzas.
  594. _converse.connection.IQ_stanzas = [];
  595. _converse.connection._dataRecv(mock.createRequest(stanza));
  596. await u.waitUntil(() => _converse.omemo_store);
  597. iq_stanza = await u.waitUntil(() => bundleHasBeenPublished(_converse), 1000);
  598. expect(Strophe.serialize(iq_stanza)).toBe(
  599. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" type="set" xmlns="jabber:client">`+
  600. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  601. `<publish node="eu.siacs.conversations.axolotl.bundles:123456789">`+
  602. `<item>`+
  603. `<bundle xmlns="eu.siacs.conversations.axolotl">`+
  604. `<signedPreKeyPublic signedPreKeyId="0">${btoa("1234")}</signedPreKeyPublic>`+
  605. `<signedPreKeySignature>${btoa("11112222333344445555")}</signedPreKeySignature>`+
  606. `<identityKey>${btoa("1234")}</identityKey>`+
  607. `<prekeys>`+
  608. `<preKeyPublic preKeyId="0">${btoa("1234")}</preKeyPublic>`+
  609. `<preKeyPublic preKeyId="1">${btoa("1234")}</preKeyPublic>`+
  610. `<preKeyPublic preKeyId="2">${btoa("1234")}</preKeyPublic>`+
  611. `<preKeyPublic preKeyId="3">${btoa("1234")}</preKeyPublic>`+
  612. `<preKeyPublic preKeyId="4">${btoa("1234")}</preKeyPublic>`+
  613. `</prekeys>`+
  614. `</bundle>`+
  615. `</item>`+
  616. `</publish>`+
  617. `<publish-options>`+
  618. `<x type="submit" xmlns="jabber:x:data">`+
  619. `<field type="hidden" var="FORM_TYPE">`+
  620. `<value>http://jabber.org/protocol/pubsub#publish-options</value>`+
  621. `</field>`+
  622. `<field var="pubsub#access_model">`+
  623. `<value>open</value>`+
  624. `</field>`+
  625. `</x>`+
  626. `</publish-options>`+
  627. `</pubsub>`+
  628. `</iq>`)
  629. const own_device = _converse.devicelists.get(_converse.bare_jid).devices.get(_converse.omemo_store.get('device_id'));
  630. expect(own_device.get('bundle').prekeys.length).toBe(5);
  631. expect(_converse.omemo_store.generateMissingPreKeys).toHaveBeenCalled();
  632. done();
  633. }));
  634. it("updates device lists based on PEP messages",
  635. mock.initConverse(
  636. ['rosterGroupsFetched'], {'allow_non_roster_messaging': true},
  637. async function (done, _converse) {
  638. await mock.waitUntilDiscoConfirmed(
  639. _converse, _converse.bare_jid,
  640. [{'category': 'pubsub', 'type': 'pep'}],
  641. ['http://jabber.org/protocol/pubsub#publish-options']
  642. );
  643. await mock.waitForRoster(_converse, 'current', 1);
  644. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  645. // Wait until own devices are fetched
  646. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, _converse.bare_jid));
  647. expect(Strophe.serialize(iq_stanza)).toBe(
  648. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="romeo@montague.lit" type="get" xmlns="jabber:client">`+
  649. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  650. `<items node="eu.siacs.conversations.axolotl.devicelist"/>`+
  651. `</pubsub>`+
  652. `</iq>`);
  653. let stanza = $iq({
  654. 'from': _converse.bare_jid,
  655. 'id': iq_stanza.getAttribute('id'),
  656. 'to': _converse.bare_jid,
  657. 'type': 'result',
  658. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  659. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  660. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  661. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  662. .c('device', {'id': '555'});
  663. _converse.connection._dataRecv(mock.createRequest(stanza));
  664. await u.waitUntil(() => _converse.omemo_store);
  665. expect(_converse.chatboxes.length).toBe(1);
  666. expect(_converse.devicelists.length).toBe(1);
  667. const devicelist = _converse.devicelists.get(_converse.bare_jid);
  668. expect(devicelist.devices.length).toBe(2);
  669. expect(devicelist.devices.at(0).get('id')).toBe('555');
  670. expect(devicelist.devices.at(1).get('id')).toBe('123456789');
  671. iq_stanza = await u.waitUntil(() => ownDeviceHasBeenPublished(_converse));
  672. stanza = $iq({
  673. 'from': _converse.bare_jid,
  674. 'id': iq_stanza.getAttribute('id'),
  675. 'to': _converse.bare_jid,
  676. 'type': 'result'});
  677. _converse.connection._dataRecv(mock.createRequest(stanza));
  678. iq_stanza = await u.waitUntil(() => bundleHasBeenPublished(_converse));
  679. stanza = $iq({
  680. 'from': _converse.bare_jid,
  681. 'id': iq_stanza.getAttribute('id'),
  682. 'to': _converse.bare_jid,
  683. 'type': 'result'});
  684. _converse.connection._dataRecv(mock.createRequest(stanza));
  685. await _converse.api.waitUntil('OMEMOInitialized');
  686. stanza = $msg({
  687. 'from': contact_jid,
  688. 'to': _converse.bare_jid,
  689. 'type': 'headline',
  690. 'id': 'update_01',
  691. }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'})
  692. .c('items', {'node': 'eu.siacs.conversations.axolotl.devicelist'})
  693. .c('item')
  694. .c('list', {'xmlns': 'eu.siacs.conversations.axolotl'})
  695. .c('device', {'id': '1234'})
  696. .c('device', {'id': '4223'})
  697. _converse.connection._dataRecv(mock.createRequest(stanza));
  698. expect(_converse.devicelists.length).toBe(2);
  699. let devices = _converse.devicelists.get(contact_jid).devices;
  700. expect(devices.length).toBe(2);
  701. expect(_.map(devices.models, 'attributes.id').sort().join()).toBe('1234,4223');
  702. stanza = $msg({
  703. 'from': contact_jid,
  704. 'to': _converse.bare_jid,
  705. 'type': 'headline',
  706. 'id': 'update_02',
  707. }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'})
  708. .c('items', {'node': 'eu.siacs.conversations.axolotl.devicelist'})
  709. .c('item')
  710. .c('list', {'xmlns': 'eu.siacs.conversations.axolotl'})
  711. .c('device', {'id': '4223'})
  712. .c('device', {'id': '4224'})
  713. _converse.connection._dataRecv(mock.createRequest(stanza));
  714. expect(_converse.devicelists.length).toBe(2);
  715. expect(devices.length).toBe(3);
  716. expect(_.map(devices.models, 'attributes.id').sort().join()).toBe('1234,4223,4224');
  717. expect(devices.get('1234').get('active')).toBe(false);
  718. expect(devices.get('4223').get('active')).toBe(true);
  719. expect(devices.get('4224').get('active')).toBe(true);
  720. // Check that own devicelist gets updated
  721. stanza = $msg({
  722. 'from': _converse.bare_jid,
  723. 'to': _converse.bare_jid,
  724. 'type': 'headline',
  725. 'id': 'update_03',
  726. }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'})
  727. .c('items', {'node': 'eu.siacs.conversations.axolotl.devicelist'})
  728. .c('item')
  729. .c('list', {'xmlns': 'eu.siacs.conversations.axolotl'})
  730. .c('device', {'id': '123456789'})
  731. .c('device', {'id': '555'})
  732. .c('device', {'id': '777'})
  733. _converse.connection._dataRecv(mock.createRequest(stanza));
  734. expect(_converse.devicelists.length).toBe(2);
  735. devices = _converse.devicelists.get(_converse.bare_jid).devices;
  736. expect(devices.length).toBe(3);
  737. expect(_.map(devices.models, 'attributes.id').sort().join()).toBe('123456789,555,777');
  738. expect(devices.get('123456789').get('active')).toBe(true);
  739. expect(devices.get('555').get('active')).toBe(true);
  740. expect(devices.get('777').get('active')).toBe(true);
  741. _converse.connection.IQ_stanzas = [];
  742. // Check that own device gets re-added
  743. stanza = $msg({
  744. 'from': _converse.bare_jid,
  745. 'to': _converse.bare_jid,
  746. 'type': 'headline',
  747. 'id': 'update_04',
  748. }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'})
  749. .c('items', {'node': 'eu.siacs.conversations.axolotl.devicelist'})
  750. .c('item')
  751. .c('list', {'xmlns': 'eu.siacs.conversations.axolotl'})
  752. .c('device', {'id': '444'})
  753. _converse.connection._dataRecv(mock.createRequest(stanza));
  754. iq_stanza = await u.waitUntil(() => ownDeviceHasBeenPublished(_converse));
  755. // Check that our own device is added again, but that removed
  756. // devices are not added.
  757. expect(Strophe.serialize(iq_stanza)).toBe(
  758. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute(`id`)}" type="set" xmlns="jabber:client">`+
  759. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  760. `<publish node="eu.siacs.conversations.axolotl.devicelist">`+
  761. `<item>`+
  762. `<list xmlns="eu.siacs.conversations.axolotl">`+
  763. `<device id="123456789"/>`+
  764. `<device id="444"/>`+
  765. `</list>`+
  766. `</item>`+
  767. `</publish>`+
  768. `<publish-options>`+
  769. `<x type="submit" xmlns="jabber:x:data">`+
  770. `<field type="hidden" var="FORM_TYPE">`+
  771. `<value>http://jabber.org/protocol/pubsub#publish-options</value>`+
  772. `</field>`+
  773. `<field var="pubsub#access_model">`+
  774. `<value>open</value>`+
  775. `</field>`+
  776. `</x>`+
  777. `</publish-options>`+
  778. `</pubsub>`+
  779. `</iq>`);
  780. expect(_converse.devicelists.length).toBe(2);
  781. devices = _converse.devicelists.get(_converse.bare_jid).devices;
  782. // The device id for this device (123456789) was also generated and added to the list,
  783. // which is why we have 2 devices now.
  784. expect(devices.length).toBe(4);
  785. expect(_.map(devices.models, 'attributes.id').sort().join()).toBe('123456789,444,555,777');
  786. expect(devices.get('123456789').get('active')).toBe(true);
  787. expect(devices.get('444').get('active')).toBe(true);
  788. expect(devices.get('555').get('active')).toBe(false);
  789. expect(devices.get('777').get('active')).toBe(false);
  790. done();
  791. }));
  792. it("updates device bundles based on PEP messages",
  793. mock.initConverse(
  794. ['rosterGroupsFetched'], {},
  795. async function (done, _converse) {
  796. await mock.waitUntilDiscoConfirmed(
  797. _converse, _converse.bare_jid,
  798. [{'category': 'pubsub', 'type': 'pep'}],
  799. ['http://jabber.org/protocol/pubsub#publish-options']
  800. );
  801. await mock.waitForRoster(_converse, 'current');
  802. const contact_jid = mock.cur_names[3].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  803. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, _converse.bare_jid));
  804. expect(Strophe.serialize(iq_stanza)).toBe(
  805. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="romeo@montague.lit" type="get" xmlns="jabber:client">`+
  806. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  807. `<items node="eu.siacs.conversations.axolotl.devicelist"/>`+
  808. `</pubsub>`+
  809. `</iq>`);
  810. let stanza = $iq({
  811. 'from': contact_jid,
  812. 'id': iq_stanza.getAttribute('id'),
  813. 'to': _converse.bare_jid,
  814. 'type': 'result',
  815. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  816. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  817. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  818. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  819. .c('device', {'id': '555'});
  820. _converse.connection._dataRecv(mock.createRequest(stanza));
  821. await await u.waitUntil(() => _converse.omemo_store);
  822. expect(_converse.devicelists.length).toBe(1);
  823. let devicelist = _converse.devicelists.get(_converse.bare_jid);
  824. expect(devicelist.devices.length).toBe(2);
  825. expect(devicelist.devices.at(0).get('id')).toBe('555');
  826. expect(devicelist.devices.at(1).get('id')).toBe('123456789');
  827. iq_stanza = await u.waitUntil(() => ownDeviceHasBeenPublished(_converse));
  828. stanza = $iq({
  829. 'from': _converse.bare_jid,
  830. 'id': iq_stanza.getAttribute('id'),
  831. 'to': _converse.bare_jid,
  832. 'type': 'result'});
  833. _converse.connection._dataRecv(mock.createRequest(stanza));
  834. iq_stanza = await u.waitUntil(() => bundleHasBeenPublished(_converse));
  835. stanza = $iq({
  836. 'from': _converse.bare_jid,
  837. 'id': iq_stanza.getAttribute('id'),
  838. 'to': _converse.bare_jid,
  839. 'type': 'result'});
  840. _converse.connection._dataRecv(mock.createRequest(stanza));
  841. await _converse.api.waitUntil('OMEMOInitialized');
  842. stanza = $msg({
  843. 'from': contact_jid,
  844. 'to': _converse.bare_jid,
  845. 'type': 'headline',
  846. 'id': 'update_01',
  847. }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'})
  848. .c('items', {'node': 'eu.siacs.conversations.axolotl.bundles:555'})
  849. .c('item')
  850. .c('bundle', {'xmlns': 'eu.siacs.conversations.axolotl'})
  851. .c('signedPreKeyPublic', {'signedPreKeyId': '4223'}).t('1111').up()
  852. .c('signedPreKeySignature').t('2222').up()
  853. .c('identityKey').t('3333').up()
  854. .c('prekeys')
  855. .c('preKeyPublic', {'preKeyId': '1001'}).up()
  856. .c('preKeyPublic', {'preKeyId': '1002'}).up()
  857. .c('preKeyPublic', {'preKeyId': '1003'});
  858. _converse.connection._dataRecv(mock.createRequest(stanza));
  859. expect(_converse.devicelists.length).toBe(2);
  860. devicelist = _converse.devicelists.get(contact_jid);
  861. expect(devicelist.devices.length).toBe(1);
  862. let device = devicelist.devices.at(0);
  863. expect(device.get('bundle').identity_key).toBe('3333');
  864. expect(device.get('bundle').signed_prekey.public_key).toBe('1111');
  865. expect(device.get('bundle').signed_prekey.id).toBe(4223);
  866. expect(device.get('bundle').signed_prekey.signature).toBe('2222');
  867. expect(device.get('bundle').prekeys.length).toBe(3);
  868. expect(device.get('bundle').prekeys[0].id).toBe(1001);
  869. expect(device.get('bundle').prekeys[1].id).toBe(1002);
  870. expect(device.get('bundle').prekeys[2].id).toBe(1003);
  871. stanza = $msg({
  872. 'from': contact_jid,
  873. 'to': _converse.bare_jid,
  874. 'type': 'headline',
  875. 'id': 'update_02',
  876. }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'})
  877. .c('items', {'node': 'eu.siacs.conversations.axolotl.bundles:555'})
  878. .c('item')
  879. .c('bundle', {'xmlns': 'eu.siacs.conversations.axolotl'})
  880. .c('signedPreKeyPublic', {'signedPreKeyId': '4223'}).t('5555').up()
  881. .c('signedPreKeySignature').t('6666').up()
  882. .c('identityKey').t('7777').up()
  883. .c('prekeys')
  884. .c('preKeyPublic', {'preKeyId': '2001'}).up()
  885. .c('preKeyPublic', {'preKeyId': '2002'}).up()
  886. .c('preKeyPublic', {'preKeyId': '2003'});
  887. _converse.connection._dataRecv(mock.createRequest(stanza));
  888. expect(_converse.devicelists.length).toBe(2);
  889. devicelist = _converse.devicelists.get(contact_jid);
  890. expect(devicelist.devices.length).toBe(1);
  891. device = devicelist.devices.at(0);
  892. expect(device.get('bundle').identity_key).toBe('7777');
  893. expect(device.get('bundle').signed_prekey.public_key).toBe('5555');
  894. expect(device.get('bundle').signed_prekey.id).toBe(4223);
  895. expect(device.get('bundle').signed_prekey.signature).toBe('6666');
  896. expect(device.get('bundle').prekeys.length).toBe(3);
  897. expect(device.get('bundle').prekeys[0].id).toBe(2001);
  898. expect(device.get('bundle').prekeys[1].id).toBe(2002);
  899. expect(device.get('bundle').prekeys[2].id).toBe(2003);
  900. stanza = $msg({
  901. 'from': _converse.bare_jid,
  902. 'to': _converse.bare_jid,
  903. 'type': 'headline',
  904. 'id': 'update_03',
  905. }).c('event', {'xmlns': 'http://jabber.org/protocol/pubsub#event'})
  906. .c('items', {'node': 'eu.siacs.conversations.axolotl.bundles:123456789'})
  907. .c('item')
  908. .c('bundle', {'xmlns': 'eu.siacs.conversations.axolotl'})
  909. .c('signedPreKeyPublic', {'signedPreKeyId': '9999'}).t('8888').up()
  910. .c('signedPreKeySignature').t('3333').up()
  911. .c('identityKey').t('1111').up()
  912. .c('prekeys')
  913. .c('preKeyPublic', {'preKeyId': '3001'}).up()
  914. .c('preKeyPublic', {'preKeyId': '3002'}).up()
  915. .c('preKeyPublic', {'preKeyId': '3003'});
  916. _converse.connection._dataRecv(mock.createRequest(stanza));
  917. expect(_converse.devicelists.length).toBe(2);
  918. devicelist = _converse.devicelists.get(_converse.bare_jid);
  919. expect(devicelist.devices.length).toBe(2);
  920. expect(devicelist.devices.at(0).get('id')).toBe('555');
  921. expect(devicelist.devices.at(1).get('id')).toBe('123456789');
  922. device = devicelist.devices.at(1);
  923. expect(device.get('bundle').identity_key).toBe('1111');
  924. expect(device.get('bundle').signed_prekey.public_key).toBe('8888');
  925. expect(device.get('bundle').signed_prekey.id).toBe(9999);
  926. expect(device.get('bundle').signed_prekey.signature).toBe('3333');
  927. expect(device.get('bundle').prekeys.length).toBe(3);
  928. expect(device.get('bundle').prekeys[0].id).toBe(3001);
  929. expect(device.get('bundle').prekeys[1].id).toBe(3002);
  930. expect(device.get('bundle').prekeys[2].id).toBe(3003);
  931. done();
  932. }));
  933. it("publishes a bundle with which an encrypted session can be created",
  934. mock.initConverse(
  935. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  936. async function (done, _converse) {
  937. await mock.waitUntilDiscoConfirmed(
  938. _converse, _converse.bare_jid,
  939. [{'category': 'pubsub', 'type': 'pep'}],
  940. ['http://jabber.org/protocol/pubsub#publish-options']
  941. );
  942. _converse.NUM_PREKEYS = 2; // Restrict to 2, otherwise the resulting stanza is too large to easily test
  943. await mock.waitForRoster(_converse, 'current', 1);
  944. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  945. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, _converse.bare_jid));
  946. let stanza = $iq({
  947. 'from': contact_jid,
  948. 'id': iq_stanza.getAttribute('id'),
  949. 'to': _converse.bare_jid,
  950. 'type': 'result',
  951. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  952. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  953. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  954. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  955. .c('device', {'id': '482886413b977930064a5888b92134fe'});
  956. _converse.connection._dataRecv(mock.createRequest(stanza));
  957. expect(_converse.devicelists.length).toBe(1);
  958. await mock.openChatBoxFor(_converse, contact_jid);
  959. iq_stanza = await ownDeviceHasBeenPublished(_converse);
  960. stanza = $iq({
  961. 'from': _converse.bare_jid,
  962. 'id': iq_stanza.getAttribute('id'),
  963. 'to': _converse.bare_jid,
  964. 'type': 'result'});
  965. _converse.connection._dataRecv(mock.createRequest(stanza));
  966. iq_stanza = await u.waitUntil(() => bundleHasBeenPublished(_converse));
  967. expect(Strophe.serialize(iq_stanza)).toBe(
  968. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" type="set" xmlns="jabber:client">`+
  969. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  970. `<publish node="eu.siacs.conversations.axolotl.bundles:123456789">`+
  971. `<item>`+
  972. `<bundle xmlns="eu.siacs.conversations.axolotl">`+
  973. `<signedPreKeyPublic signedPreKeyId="0">${btoa("1234")}</signedPreKeyPublic>`+
  974. `<signedPreKeySignature>${btoa("11112222333344445555")}</signedPreKeySignature>`+
  975. `<identityKey>${btoa("1234")}</identityKey>`+
  976. `<prekeys>`+
  977. `<preKeyPublic preKeyId="0">${btoa("1234")}</preKeyPublic>`+
  978. `<preKeyPublic preKeyId="1">${btoa("1234")}</preKeyPublic>`+
  979. `</prekeys>`+
  980. `</bundle>`+
  981. `</item>`+
  982. `</publish>`+
  983. `<publish-options>`+
  984. `<x type="submit" xmlns="jabber:x:data">`+
  985. `<field type="hidden" var="FORM_TYPE">`+
  986. `<value>http://jabber.org/protocol/pubsub#publish-options</value>`+
  987. `</field>`+
  988. `<field var="pubsub#access_model">`+
  989. `<value>open</value>`+
  990. `</field>`+
  991. `</x>`+
  992. `</publish-options>`+
  993. `</pubsub>`+
  994. `</iq>`)
  995. stanza = $iq({
  996. 'from': _converse.bare_jid,
  997. 'id': iq_stanza.getAttribute('id'),
  998. 'to': _converse.bare_jid,
  999. 'type': 'result'});
  1000. _converse.connection._dataRecv(mock.createRequest(stanza));
  1001. await _converse.api.waitUntil('OMEMOInitialized');
  1002. done();
  1003. }));
  1004. it("adds a toolbar button for starting an encrypted chat session",
  1005. mock.initConverse(
  1006. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  1007. async function (done, _converse) {
  1008. await mock.waitUntilDiscoConfirmed(
  1009. _converse, _converse.bare_jid,
  1010. [{'category': 'pubsub', 'type': 'pep'}],
  1011. ['http://jabber.org/protocol/pubsub#publish-options']
  1012. );
  1013. await mock.waitForRoster(_converse, 'current', 1);
  1014. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  1015. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, _converse.bare_jid));
  1016. expect(Strophe.serialize(iq_stanza)).toBe(
  1017. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="romeo@montague.lit" type="get" xmlns="jabber:client">`+
  1018. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  1019. `<items node="eu.siacs.conversations.axolotl.devicelist"/>`+
  1020. `</pubsub>`+
  1021. `</iq>`);
  1022. let stanza = $iq({
  1023. 'from': _converse.bare_jid,
  1024. 'id': iq_stanza.getAttribute('id'),
  1025. 'to': _converse.bare_jid,
  1026. 'type': 'result',
  1027. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  1028. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  1029. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  1030. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  1031. .c('device', {'id': '482886413b977930064a5888b92134fe'});
  1032. _converse.connection._dataRecv(mock.createRequest(stanza));
  1033. await u.waitUntil(() => _converse.omemo_store);
  1034. expect(_converse.devicelists.length).toBe(1);
  1035. let devicelist = _converse.devicelists.get(_converse.bare_jid);
  1036. expect(devicelist.devices.length).toBe(2);
  1037. expect(devicelist.devices.at(0).get('id')).toBe('482886413b977930064a5888b92134fe');
  1038. expect(devicelist.devices.at(1).get('id')).toBe('123456789');
  1039. // Check that own device was published
  1040. iq_stanza = await u.waitUntil(() => ownDeviceHasBeenPublished(_converse));
  1041. expect(Strophe.serialize(iq_stanza)).toBe(
  1042. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute(`id`)}" type="set" xmlns="jabber:client">`+
  1043. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  1044. `<publish node="eu.siacs.conversations.axolotl.devicelist">`+
  1045. `<item>`+
  1046. `<list xmlns="eu.siacs.conversations.axolotl">`+
  1047. `<device id="482886413b977930064a5888b92134fe"/>`+
  1048. `<device id="123456789"/>`+
  1049. `</list>`+
  1050. `</item>`+
  1051. `</publish>`+
  1052. `<publish-options>`+
  1053. `<x type="submit" xmlns="jabber:x:data">`+
  1054. `<field type="hidden" var="FORM_TYPE">`+
  1055. `<value>http://jabber.org/protocol/pubsub#publish-options</value>`+
  1056. `</field>`+
  1057. `<field var="pubsub#access_model">`+
  1058. `<value>open</value>`+
  1059. `</field>`+
  1060. `</x>`+
  1061. `</publish-options>`+
  1062. `</pubsub>`+
  1063. `</iq>`);
  1064. stanza = $iq({
  1065. 'from': _converse.bare_jid,
  1066. 'id': iq_stanza.getAttribute('id'),
  1067. 'to': _converse.bare_jid,
  1068. 'type': 'result'});
  1069. _converse.connection._dataRecv(mock.createRequest(stanza));
  1070. const iq_el = await u.waitUntil(() => bundleHasBeenPublished(_converse));
  1071. expect(iq_el.getAttributeNames().sort().join()).toBe(["from", "type", "xmlns", "id"].sort().join());
  1072. expect(iq_el.querySelector('prekeys').childNodes.length).toBe(100);
  1073. const signed_prekeys = iq_el.querySelectorAll('signedPreKeyPublic');
  1074. expect(signed_prekeys.length).toBe(1);
  1075. const signed_prekey = signed_prekeys[0];
  1076. expect(signed_prekey.getAttribute('signedPreKeyId')).toBe('0')
  1077. expect(iq_el.querySelectorAll('signedPreKeySignature').length).toBe(1);
  1078. expect(iq_el.querySelectorAll('identityKey').length).toBe(1);
  1079. stanza = $iq({
  1080. 'from': _converse.bare_jid,
  1081. 'id': iq_el.getAttribute('id'),
  1082. 'to': _converse.bare_jid,
  1083. 'type': 'result'});
  1084. _converse.connection._dataRecv(mock.createRequest(stanza));
  1085. await _converse.api.waitUntil('OMEMOInitialized', 1000);
  1086. await mock.openChatBoxFor(_converse, contact_jid);
  1087. iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  1088. expect(Strophe.serialize(iq_stanza)).toBe(
  1089. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="${contact_jid}" type="get" xmlns="jabber:client">`+
  1090. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  1091. `<items node="eu.siacs.conversations.axolotl.devicelist"/>`+
  1092. `</pubsub>`+
  1093. `</iq>`);
  1094. stanza = $iq({
  1095. 'from': contact_jid,
  1096. 'id': iq_stanza.getAttribute('id'),
  1097. 'to': _converse.bare_jid,
  1098. 'type': 'result',
  1099. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  1100. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  1101. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  1102. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  1103. .c('device', {'id': '368866411b877c30064a5f62b917cffe'}).up()
  1104. .c('device', {'id': '3300659945416e274474e469a1f0154c'}).up()
  1105. .c('device', {'id': '4e30f35051b7b8b42abe083742187228'}).up()
  1106. .c('device', {'id': 'ae890ac52d0df67ed7cfdf51b644e901'});
  1107. _converse.connection._dataRecv(mock.createRequest(stanza));
  1108. devicelist = _converse.devicelists.get(contact_jid);
  1109. await u.waitUntil(() => devicelist.devices.length);
  1110. expect(_converse.devicelists.length).toBe(2);
  1111. devicelist = _converse.devicelists.get(contact_jid);
  1112. expect(devicelist.devices.length).toBe(4);
  1113. expect(devicelist.devices.at(0).get('id')).toBe('368866411b877c30064a5f62b917cffe');
  1114. expect(devicelist.devices.at(1).get('id')).toBe('3300659945416e274474e469a1f0154c');
  1115. expect(devicelist.devices.at(2).get('id')).toBe('4e30f35051b7b8b42abe083742187228');
  1116. expect(devicelist.devices.at(3).get('id')).toBe('ae890ac52d0df67ed7cfdf51b644e901');
  1117. await u.waitUntil(() => _converse.chatboxviews.get(contact_jid).el.querySelector('.chat-toolbar'));
  1118. const view = _converse.chatboxviews.get(contact_jid);
  1119. const toolbar = view.el.querySelector('.chat-toolbar');
  1120. expect(view.model.get('omemo_active')).toBe(undefined);
  1121. const toggle = toolbar.querySelector('.toggle-omemo');
  1122. expect(toggle === null).toBe(false);
  1123. expect(u.hasClass('fa-unlock', toggle.querySelector('converse-icon'))).toBe(true);
  1124. expect(u.hasClass('fa-lock', toggle.querySelector('.converse-icon'))).toBe(false);
  1125. view.delegateEvents(); // We need to rebind all events otherwise our spy won't be called
  1126. toolbar.querySelector('.toggle-omemo').click();
  1127. expect(view.model.get('omemo_active')).toBe(true);
  1128. await u.waitUntil(() => u.hasClass('fa-lock', toolbar.querySelector('.toggle-omemo converse-icon')));
  1129. let icon = toolbar.querySelector('.toggle-omemo converse-icon');
  1130. expect(u.hasClass('fa-unlock', icon)).toBe(false);
  1131. expect(u.hasClass('fa-lock', icon)).toBe(true);
  1132. const textarea = view.el.querySelector('.chat-textarea');
  1133. textarea.value = 'This message will be sent encrypted';
  1134. view.onKeyDown({
  1135. target: textarea,
  1136. preventDefault: function preventDefault () {},
  1137. keyCode: 13
  1138. });
  1139. view.model.save({'omemo_supported': false});
  1140. await u.waitUntil(() => toolbar.querySelector('.toggle-omemo').disabled);
  1141. icon = toolbar.querySelector('.toggle-omemo converse-icon');
  1142. expect(u.hasClass('fa-lock', icon)).toBe(false);
  1143. expect(u.hasClass('fa-unlock', icon)).toBe(true);
  1144. view.model.save({'omemo_supported': true});
  1145. await u.waitUntil(() => !toolbar.querySelector('.toggle-omemo').disabled);
  1146. icon = toolbar.querySelector('.toggle-omemo converse-icon');
  1147. expect(u.hasClass('fa-lock', icon)).toBe(false);
  1148. expect(u.hasClass('fa-unlock', icon)).toBe(true);
  1149. done();
  1150. }));
  1151. it("adds a toolbar button for starting an encrypted groupchat session",
  1152. mock.initConverse(
  1153. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  1154. async function (done, _converse) {
  1155. await mock.waitUntilDiscoConfirmed(
  1156. _converse, _converse.bare_jid,
  1157. [{'category': 'pubsub', 'type': 'pep'}],
  1158. ['http://jabber.org/protocol/pubsub#publish-options']
  1159. );
  1160. // MEMO encryption works only in members-only conferences that are non-anonymous.
  1161. const features = [
  1162. 'http://jabber.org/protocol/muc',
  1163. 'jabber:iq:register',
  1164. 'muc_passwordprotected',
  1165. 'muc_hidden',
  1166. 'muc_temporary',
  1167. 'muc_membersonly',
  1168. 'muc_unmoderated',
  1169. 'muc_nonanonymous'
  1170. ];
  1171. await mock.openAndEnterChatRoom(_converse, 'lounge@montague.lit', 'romeo', features);
  1172. const view = _converse.chatboxviews.get('lounge@montague.lit');
  1173. await u.waitUntil(() => initializedOMEMO(_converse));
  1174. const toolbar = view.el.querySelector('.chat-toolbar');
  1175. let toggle = toolbar.querySelector('.toggle-omemo');
  1176. expect(view.model.get('omemo_active')).toBe(undefined);
  1177. expect(view.model.get('omemo_supported')).toBe(true);
  1178. await u.waitUntil(() => !toggle.disabled);
  1179. let icon = toolbar.querySelector('.toggle-omemo converse-icon');
  1180. expect(u.hasClass('fa-unlock', icon)).toBe(true);
  1181. expect(u.hasClass('fa-lock', icon)).toBe(false);
  1182. toggle.click();
  1183. toggle = toolbar.querySelector('.toggle-omemo');
  1184. expect(!!toggle.disabled).toBe(false);
  1185. expect(view.model.get('omemo_active')).toBe(true);
  1186. expect(view.model.get('omemo_supported')).toBe(true);
  1187. await u.waitUntil(() => !u.hasClass('fa-unlock', toolbar.querySelector('.toggle-omemo converse-icon')));
  1188. expect(u.hasClass('fa-lock', toolbar.querySelector('.toggle-omemo converse-icon'))).toBe(true);
  1189. let contact_jid = 'newguy@montague.lit';
  1190. let stanza = $pres({
  1191. to: 'romeo@montague.lit/orchard',
  1192. from: 'lounge@montague.lit/newguy'
  1193. })
  1194. .c('x', {xmlns: Strophe.NS.MUC_USER})
  1195. .c('item', {
  1196. 'affiliation': 'none',
  1197. 'jid': 'newguy@montague.lit/_converse.js-290929789',
  1198. 'role': 'participant'
  1199. }).tree();
  1200. _converse.connection._dataRecv(mock.createRequest(stanza));
  1201. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  1202. expect(Strophe.serialize(iq_stanza)).toBe(
  1203. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="${contact_jid}" type="get" xmlns="jabber:client">`+
  1204. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  1205. `<items node="eu.siacs.conversations.axolotl.devicelist"/>`+
  1206. `</pubsub>`+
  1207. `</iq>`);
  1208. stanza = $iq({
  1209. 'from': contact_jid,
  1210. 'id': iq_stanza.getAttribute('id'),
  1211. 'to': _converse.bare_jid,
  1212. 'type': 'result',
  1213. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  1214. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  1215. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  1216. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  1217. .c('device', {'id': '4e30f35051b7b8b42abe083742187228'}).up()
  1218. .c('device', {'id': 'ae890ac52d0df67ed7cfdf51b644e901'});
  1219. _converse.connection._dataRecv(mock.createRequest(stanza));
  1220. await u.waitUntil(() => _converse.omemo_store);
  1221. expect(_converse.devicelists.length).toBe(2);
  1222. await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  1223. const devicelist = _converse.devicelists.get(contact_jid);
  1224. expect(devicelist.devices.length).toBe(2);
  1225. expect(devicelist.devices.at(0).get('id')).toBe('4e30f35051b7b8b42abe083742187228');
  1226. expect(devicelist.devices.at(1).get('id')).toBe('ae890ac52d0df67ed7cfdf51b644e901');
  1227. expect(view.model.get('omemo_active')).toBe(true);
  1228. toggle = toolbar.querySelector('.toggle-omemo');
  1229. expect(toggle === null).toBe(false);
  1230. expect(!!toggle.disabled).toBe(false);
  1231. expect(view.model.get('omemo_supported')).toBe(true);
  1232. await u.waitUntil(() => !u.hasClass('fa-unlock', toolbar.querySelector('.toggle-omemo converse-icon')));
  1233. expect(u.hasClass('fa-lock', toolbar.querySelector('.toggle-omemo converse-icon'))).toBe(true);
  1234. // Test that the button gets disabled when the room becomes
  1235. // anonymous or semi-anonymous
  1236. view.model.features.save({'nonanonymous': false, 'semianonymous': true});
  1237. await u.waitUntil(() => !view.model.get('omemo_supported'));
  1238. await u.waitUntil(() => view.el.querySelector('.toggle-omemo').disabled);
  1239. view.model.features.save({'nonanonymous': true, 'semianonymous': false});
  1240. await u.waitUntil(() => view.model.get('omemo_supported'));
  1241. await u.waitUntil(() => view.el.querySelector('.toggle-omemo') !== null);
  1242. expect(u.hasClass('fa-unlock', toolbar.querySelector('.toggle-omemo converse-icon'))).toBe(true);
  1243. expect(u.hasClass('fa-lock', toolbar.querySelector('.toggle-omemo converse-icon'))).toBe(false);
  1244. expect(!!view.el.querySelector('.toggle-omemo').disabled).toBe(false);
  1245. // Test that the button gets disabled when the room becomes open
  1246. view.model.features.save({'membersonly': false, 'open': true});
  1247. await u.waitUntil(() => !view.model.get('omemo_supported'));
  1248. await u.waitUntil(() => view.el.querySelector('.toggle-omemo').disabled);
  1249. view.model.features.save({'membersonly': true, 'open': false});
  1250. await u.waitUntil(() => view.model.get('omemo_supported'));
  1251. await u.waitUntil(() => !view.el.querySelector('.toggle-omemo').disabled);
  1252. expect(u.hasClass('fa-unlock', view.el.querySelector('.toggle-omemo converse-icon'))).toBe(true);
  1253. expect(u.hasClass('fa-lock', view.el.querySelector('.toggle-omemo converse-icon'))).toBe(false);
  1254. expect(view.model.get('omemo_supported')).toBe(true);
  1255. expect(view.model.get('omemo_active')).toBe(false);
  1256. view.el.querySelector('.toggle-omemo').click();
  1257. expect(view.model.get('omemo_active')).toBe(true);
  1258. // Someone enters the room who doesn't have OMEMO support, while we
  1259. // have OMEMO activated...
  1260. contact_jid = 'oldguy@montague.lit';
  1261. stanza = $pres({
  1262. to: 'romeo@montague.lit/orchard',
  1263. from: 'lounge@montague.lit/oldguy'
  1264. })
  1265. .c('x', {xmlns: Strophe.NS.MUC_USER})
  1266. .c('item', {
  1267. 'affiliation': 'none',
  1268. 'jid': `${contact_jid}/_converse.js-290929788`,
  1269. 'role': 'participant'
  1270. }).tree();
  1271. _converse.connection._dataRecv(mock.createRequest(stanza));
  1272. iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  1273. expect(Strophe.serialize(iq_stanza)).toBe(
  1274. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="${contact_jid}" type="get" xmlns="jabber:client">`+
  1275. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  1276. `<items node="eu.siacs.conversations.axolotl.devicelist"/>`+
  1277. `</pubsub>`+
  1278. `</iq>`);
  1279. stanza = $iq({
  1280. 'from': contact_jid,
  1281. 'id': iq_stanza.getAttribute('id'),
  1282. 'to': _converse.bare_jid,
  1283. 'type': 'error'
  1284. }).c('error', {'type': 'cancel'})
  1285. .c('item-not-found', {'xmlns': "urn:ietf:params:xml:ns:xmpp-stanzas"});
  1286. _converse.connection._dataRecv(mock.createRequest(stanza));
  1287. await u.waitUntil(() => !view.model.get('omemo_supported'));
  1288. await u.waitUntil(() => view.el.querySelector('.chat-error .chat-info__message')?.textContent.trim() ===
  1289. "oldguy doesn't appear to have a client that supports OMEMO. "+
  1290. "Encrypted chat will no longer be possible in this grouchat."
  1291. );
  1292. await u.waitUntil(() => toolbar.querySelector('.toggle-omemo').disabled);
  1293. icon = view.el.querySelector('.toggle-omemo converse-icon');
  1294. expect(u.hasClass('fa-unlock', icon)).toBe(true);
  1295. expect(u.hasClass('fa-lock', icon)).toBe(false);
  1296. expect(toolbar.querySelector('.toggle-omemo').title).toBe('This groupchat needs to be members-only and non-anonymous in order to support OMEMO encrypted messages');
  1297. done();
  1298. }));
  1299. it("shows OMEMO device fingerprints in the user details modal",
  1300. mock.initConverse(
  1301. ['rosterGroupsFetched', 'chatBoxesFetched'], {},
  1302. async function (done, _converse) {
  1303. await mock.waitUntilDiscoConfirmed(
  1304. _converse, _converse.bare_jid,
  1305. [{'category': 'pubsub', 'type': 'pep'}],
  1306. ['http://jabber.org/protocol/pubsub#publish-options']
  1307. );
  1308. await mock.waitForRoster(_converse, 'current', 1);
  1309. const contact_jid = mock.cur_names[0].replace(/ /g,'.').toLowerCase() + '@montague.lit';
  1310. await mock.openChatBoxFor(_converse, contact_jid)
  1311. // We simply emit, to avoid doing all the setup work
  1312. _converse.api.trigger('OMEMOInitialized');
  1313. const view = _converse.chatboxviews.get(contact_jid);
  1314. const show_modal_button = view.el.querySelector('.show-user-details-modal');
  1315. show_modal_button.click();
  1316. const modal = view.user_details_modal;
  1317. await u.waitUntil(() => u.isVisible(modal.el), 1000);
  1318. let iq_stanza = await u.waitUntil(() => deviceListFetched(_converse, contact_jid));
  1319. expect(Strophe.serialize(iq_stanza)).toBe(
  1320. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="mercutio@montague.lit" type="get" xmlns="jabber:client">`+
  1321. `<pubsub xmlns="http://jabber.org/protocol/pubsub"><items node="eu.siacs.conversations.axolotl.devicelist"/></pubsub>`+
  1322. `</iq>`);
  1323. let stanza = $iq({
  1324. 'from': contact_jid,
  1325. 'id': iq_stanza.getAttribute('id'),
  1326. 'to': _converse.bare_jid,
  1327. 'type': 'result',
  1328. }).c('pubsub', {'xmlns': "http://jabber.org/protocol/pubsub"})
  1329. .c('items', {'node': "eu.siacs.conversations.axolotl.devicelist"})
  1330. .c('item', {'xmlns': "http://jabber.org/protocol/pubsub"}) // TODO: must have an id attribute
  1331. .c('list', {'xmlns': "eu.siacs.conversations.axolotl"})
  1332. .c('device', {'id': '555'});
  1333. _converse.connection._dataRecv(mock.createRequest(stanza));
  1334. await u.waitUntil(() => u.isVisible(modal.el), 1000);
  1335. iq_stanza = await u.waitUntil(() => bundleFetched(_converse, contact_jid, '555'));
  1336. expect(Strophe.serialize(iq_stanza)).toBe(
  1337. `<iq from="romeo@montague.lit" id="${iq_stanza.getAttribute("id")}" to="mercutio@montague.lit" type="get" xmlns="jabber:client">`+
  1338. `<pubsub xmlns="http://jabber.org/protocol/pubsub">`+
  1339. `<items node="eu.siacs.conversations.axolotl.bundles:555"/>`+
  1340. `</pubsub>`+
  1341. `</iq>`);
  1342. stanza = $iq({
  1343. 'from': contact_jid,
  1344. 'id': iq_stanza.getAttribute('id'),
  1345. 'to': _converse.bare_jid,
  1346. 'type': 'result',
  1347. }).c('pubsub', {
  1348. 'xmlns': 'http://jabber.org/protocol/pubsub'
  1349. }).c('items', {'node': "eu.siacs.conversations.axolotl.bundles:555"})
  1350. .c('item')
  1351. .c('bundle', {'xmlns': 'eu.siacs.conversations.axolotl'})
  1352. .c('signedPreKeyPublic', {'signedPreKeyId': '4223'}).t(btoa('1111')).up()
  1353. .c('signedPreKeySignature').t(btoa('2222')).up()
  1354. .c('identityKey').t('BQmHEOHjsYm3w5M8VqxAtqJmLCi7CaxxsdZz6G0YpuMI').up()
  1355. .c('prekeys')
  1356. .c('preKeyPublic', {'preKeyId': '1'}).t(btoa('1001')).up()
  1357. .c('preKeyPublic', {'preKeyId': '2'}).t(btoa('1002')).up()
  1358. .c('preKeyPublic', {'preKeyId': '3'}).t(btoa('1003'));
  1359. _converse.connection._dataRecv(mock.createRequest(stanza));
  1360. await u.waitUntil(() => modal.el.querySelectorAll('.fingerprints .fingerprint').length);
  1361. expect(modal.el.querySelectorAll('.fingerprints .fingerprint').length).toBe(1);
  1362. const el = modal.el.querySelector('.fingerprints .fingerprint');
  1363. expect(el.textContent.trim()).toBe(
  1364. u.formatFingerprint(u.arrayBufferToHex(u.base64ToArrayBuffer('BQmHEOHjsYm3w5M8VqxAtqJmLCi7CaxxsdZz6G0YpuMI')))
  1365. );
  1366. expect(modal.el.querySelectorAll('input[type="radio"]').length).toBe(2);
  1367. const devicelist = _converse.devicelists.get(contact_jid);
  1368. expect(devicelist.devices.get('555').get('trusted')).toBe(0);
  1369. let trusted_radio = modal.el.querySelector('input[type="radio"][name="555"][value="1"]');
  1370. expect(trusted_radio.checked).toBe(true);
  1371. let untrusted_radio = modal.el.querySelector('input[type="radio"][name="555"][value="-1"]');
  1372. expect(untrusted_radio.checked).toBe(false);
  1373. // Test that the device can be set to untrusted
  1374. untrusted_radio.click();
  1375. trusted_radio = document.querySelector('input[type="radio"][name="555"][value="1"]');
  1376. expect(trusted_radio.hasAttribute('checked')).toBe(false);
  1377. expect(devicelist.devices.get('555').get('trusted')).toBe(-1);
  1378. untrusted_radio = document.querySelector('input[type="radio"][name="555"][value="-1"]');
  1379. expect(untrusted_radio.hasAttribute('checked')).toBe(true);
  1380. trusted_radio.click();
  1381. expect(devicelist.devices.get('555').get('trusted')).toBe(1);
  1382. done();
  1383. }));
  1384. });