2
0

omemo.js 79 KB

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