/*global mock */ const Strophe = converse.env.Strophe; const $iq = converse.env.$iq; const { _, sizzle} = converse.env; const u = converse.env.utils; describe("The Registration Panel", function () { it("is not available unless allow_registration=true", mock.initConverse( ['chatBoxesInitialized'], { auto_login: false, allow_registration: false }, async function (done, _converse) { await u.waitUntil(() => _converse.chatboxviews.get('controlbox')); const cbview = _converse.api.controlbox.get(); expect(cbview.el.querySelectorAll('a.register-account').length).toBe(0); done(); })); it("can be opened by clicking on the registration tab", mock.initConverse( ['chatBoxesInitialized'], { auto_login: false, allow_registration: true }, async function (done, _converse) { const toggle = document.querySelector(".toggle-controlbox"); if (!u.isVisible(document.querySelector("#controlbox"))) { if (!u.isVisible(toggle)) { u.removeClass('hidden', toggle); } toggle.click(); } await u.waitUntil(() => _.get(_converse.chatboxviews.get('controlbox'), 'registerpanel'), 300); const cbview = _converse.chatboxviews.get('controlbox'); const panels = cbview.el.querySelector('.controlbox-panes'); const login = panels.firstElementChild; const registration = panels.childNodes[1]; const register_link = cbview.el.querySelector('a.register-account'); expect(register_link.textContent).toBe("Create an account"); register_link.click(); await u.waitUntil(() => u.isVisible(registration)); expect(u.isVisible(login)).toBe(false); done(); })); it("allows the user to choose an XMPP provider's domain", mock.initConverse( ['chatBoxesInitialized'], { auto_login: false, discover_connection_methods: false, allow_registration: true }, async function (done, _converse) { spyOn(Strophe.Connection.prototype, 'connect'); await u.waitUntil(() => _.get(_converse.chatboxviews.get('controlbox'), 'registerpanel')); const toggle = document.querySelector(".toggle-controlbox"); toggle.click(); const cbview = _converse.api.controlbox.get(); await u.waitUntil(() => u.isVisible(cbview.el)); const registerview = cbview.registerpanel; spyOn(registerview, 'onProviderChosen').and.callThrough(); registerview.delegateEvents(); // We need to rebind all events otherwise our spy won't be called // Open the register panel cbview.el.querySelector('.toggle-register-login').click(); // Check the form layout const form = cbview.el.querySelector('#converse-register'); expect(form.querySelectorAll('input').length).toEqual(2); expect(form.querySelectorAll('input')[0].getAttribute('name')).toEqual('domain'); expect(sizzle('input:last', form).pop().getAttribute('type')).toEqual('submit'); // Check that the input[type=domain] input is required const submit_button = form.querySelector('input[type=submit]'); submit_button.click(); expect(registerview.onProviderChosen).not.toHaveBeenCalled(); // Check that the form is accepted if input[type=domain] has a value form.querySelector('input[name=domain]').value = 'conversejs.org'; submit_button.click(); expect(registerview.onProviderChosen).toHaveBeenCalled(); await u.waitUntil(() => _converse.connection.connect.calls.count()); done(); })); it("will render a registration form as received from the XMPP provider", mock.initConverse( ['chatBoxesInitialized'], { auto_login: false, discover_connection_methods: false, allow_registration: true }, async function (done, _converse) { spyOn(Strophe.Connection.prototype, 'connect'); await u.waitUntil(() => _.get(_converse.chatboxviews.get('controlbox'), 'registerpanel')); const cbview = _converse.api.controlbox.get(); cbview.el.querySelector('.toggle-register-login').click(); const registerview = _converse.chatboxviews.get('controlbox').registerpanel; spyOn(registerview, 'onProviderChosen').and.callThrough(); spyOn(registerview, 'getRegistrationFields').and.callThrough(); spyOn(registerview, 'onRegistrationFields').and.callThrough(); spyOn(registerview, 'renderRegistrationForm').and.callThrough(); registerview.delegateEvents(); // We need to rebind all events otherwise our spy won't be called expect(registerview._registering).toBeFalsy(); expect(_converse.api.connection.connected()).toBeFalsy(); registerview.el.querySelector('input[name=domain]').value = 'conversejs.org'; registerview.el.querySelector('input[type=submit]').click(); expect(registerview.onProviderChosen).toHaveBeenCalled(); expect(registerview._registering).toBeTruthy(); await u.waitUntil(() => _converse.connection.connect.calls.count()); let stanza = new Strophe.Builder("stream:features", { 'xmlns:stream': "http://etherx.jabber.org/streams", 'xmlns': "jabber:client" }) .c('register', {xmlns: "http://jabber.org/features/iq-register"}).up() .c('mechanisms', {xmlns: "urn:ietf:params:xml:ns:xmpp-sasl"}); _converse.connection._connect_cb(mock.createRequest(stanza)); expect(registerview.getRegistrationFields).toHaveBeenCalled(); stanza = $iq({ 'type': 'result', 'id': 'reg1' }).c('query', {'xmlns': 'jabber:iq:register'}) .c('instructions') .t('Please choose a username, password and provide your email address').up() .c('username').up() .c('password').up() .c('email'); _converse.connection._dataRecv(mock.createRequest(stanza)); expect(registerview.onRegistrationFields).toHaveBeenCalled(); expect(registerview.renderRegistrationForm).toHaveBeenCalled(); expect(registerview.el.querySelectorAll('input').length).toBe(5); expect(registerview.el.querySelectorAll('input[type=submit]').length).toBe(1); expect(registerview.el.querySelectorAll('input[type=button]').length).toBe(1); done(); })); it("will set form_type to legacy and submit it as legacy", mock.initConverse( ['chatBoxesInitialized'], { auto_login: false, discover_connection_methods: false, allow_registration: true }, async function (done, _converse) { await u.waitUntil(() => _.get(_converse.chatboxviews.get('controlbox'), 'registerpanel')); const toggle = document.querySelector(".toggle-controlbox"); if (!u.isVisible(document.querySelector("#controlbox"))) { if (!u.isVisible(toggle)) { u.removeClass('hidden', toggle); } toggle.click(); } const cbview = _converse.api.controlbox.get(); cbview.el.querySelector('.toggle-register-login').click(); const registerview = cbview.registerpanel; spyOn(registerview, 'onProviderChosen').and.callThrough(); spyOn(registerview, 'getRegistrationFields').and.callThrough(); spyOn(registerview, 'onRegistrationFields').and.callThrough(); spyOn(registerview, 'renderRegistrationForm').and.callThrough(); registerview.delegateEvents(); // We need to rebind all events otherwise our spy won't be called registerview.el.querySelector('input[name=domain]').value = 'conversejs.org'; registerview.el.querySelector('input[type=submit]').click(); let stanza = new Strophe.Builder("stream:features", { 'xmlns:stream': "http://etherx.jabber.org/streams", 'xmlns': "jabber:client" }) .c('register', {xmlns: "http://jabber.org/features/iq-register"}).up() .c('mechanisms', {xmlns: "urn:ietf:params:xml:ns:xmpp-sasl"}); _converse.connection._connect_cb(mock.createRequest(stanza)); stanza = $iq({ 'type': 'result', 'id': 'reg1' }).c('query', {'xmlns': 'jabber:iq:register'}) .c('instructions') .t('Please choose a username, password and provide your email address').up() .c('username').up() .c('password').up() .c('email'); _converse.connection._dataRecv(mock.createRequest(stanza)); expect(registerview.form_type).toBe('legacy'); registerview.el.querySelector('input[name=username]').value = 'testusername'; registerview.el.querySelector('input[name=password]').value = 'testpassword'; registerview.el.querySelector('input[name=email]').value = 'test@email.local'; spyOn(_converse.connection, 'send'); registerview.el.querySelector('input[type=submit]').click(); expect(_converse.connection.send).toHaveBeenCalled(); stanza = _converse.connection.send.calls.argsFor(0)[0].tree(); expect(stanza.querySelector('query').childNodes.length).toBe(3); expect(stanza.querySelector('query').firstElementChild.tagName).toBe('username'); delete _converse.connection; done(); })); it("will set form_type to xform and submit it as xform", mock.initConverse( ['chatBoxesInitialized'], { auto_login: false, discover_connection_methods: false, allow_registration: true }, async function (done, _converse) { await u.waitUntil(() => _.get(_converse.chatboxviews.get('controlbox'), 'registerpanel')); const toggle = document.querySelector(".toggle-controlbox"); if (!u.isVisible(document.querySelector("#controlbox"))) { if (!u.isVisible(toggle)) { u.removeClass('hidden', toggle); } toggle.click(); } const cbview = _converse.api.controlbox.get(); cbview.el.querySelector('.toggle-register-login').click(); const registerview = _converse.chatboxviews.get('controlbox').registerpanel; spyOn(registerview, 'onProviderChosen').and.callThrough(); spyOn(registerview, 'getRegistrationFields').and.callThrough(); spyOn(registerview, 'onRegistrationFields').and.callThrough(); spyOn(registerview, 'renderRegistrationForm').and.callThrough(); registerview.delegateEvents(); // We need to rebind all events otherwise our spy won't be called registerview.el.querySelector('input[name=domain]').value = 'conversejs.org'; registerview.el.querySelector('input[type=submit]').click(); let stanza = new Strophe.Builder("stream:features", { 'xmlns:stream': "http://etherx.jabber.org/streams", 'xmlns': "jabber:client" }) .c('register', {xmlns: "http://jabber.org/features/iq-register"}).up() .c('mechanisms', {xmlns: "urn:ietf:params:xml:ns:xmpp-sasl"}); _converse.connection._connect_cb(mock.createRequest(stanza)); stanza = $iq({ 'type': 'result', 'id': 'reg1' }).c('query', {'xmlns': 'jabber:iq:register'}) .c('instructions') .t('Using xform data').up() .c('x', { 'xmlns': 'jabber:x:data', 'type': 'form' }) .c('instructions').t('xform instructions').up() .c('field', {'type': 'text-single', 'var': 'username'}).c('required').up().up() .c('field', {'type': 'text-private', 'var': 'password'}).c('required').up().up() .c('field', {'type': 'text-single', 'var': 'email'}).c('required').up().up(); _converse.connection._dataRecv(mock.createRequest(stanza)); expect(registerview.form_type).toBe('xform'); registerview.el.querySelector('input[name=username]').value = 'testusername'; registerview.el.querySelector('input[name=password]').value = 'testpassword'; registerview.el.querySelector('input[name=email]').value = 'test@email.local'; spyOn(_converse.connection, 'send'); registerview.el.querySelector('input[type=submit]').click(); expect(_converse.connection.send).toHaveBeenCalled(); stanza = _converse.connection.send.calls.argsFor(0)[0].tree(); expect(Strophe.serialize(stanza).toLocaleString().trim().replace(/(\n|\s{2,})/g, '')).toEqual( ''+ ''+ ''+ ''+ 'testusername'+ ''+ ''+ 'testpassword'+ ''+ ''+ 'test@email.local'+ ''+ ''+ ''+ '' ); delete _converse.connection; done(); })); it("renders the account registration form", mock.initConverse( ['chatBoxesInitialized'], { auto_login: false, view_mode: 'fullscreen', discover_connection_methods: false, allow_registration: true }, async function (done, _converse) { await u.waitUntil(() => _.get(_converse.chatboxviews.get('controlbox'), 'registerpanel')); const toggle = document.querySelector(".toggle-controlbox"); if (!u.isVisible(document.querySelector("#controlbox"))) { if (!u.isVisible(toggle)) { u.removeClass('hidden', toggle); } toggle.click(); } const cbview = _converse.chatboxviews.get('controlbox'); cbview.el.querySelector('.toggle-register-login').click(); const registerview = _converse.chatboxviews.get('controlbox').registerpanel; spyOn(registerview, 'onProviderChosen').and.callThrough(); spyOn(registerview, 'getRegistrationFields').and.callThrough(); spyOn(registerview, 'onRegistrationFields').and.callThrough(); spyOn(registerview, 'renderRegistrationForm').and.callThrough(); registerview.delegateEvents(); // We need to rebind all events otherwise our spy won't be called registerview.el.querySelector('input[name=domain]').value = 'conversejs.org'; registerview.el.querySelector('input[type=submit]').click(); let stanza = new Strophe.Builder("stream:features", { 'xmlns:stream': "http://etherx.jabber.org/streams", 'xmlns': "jabber:client" }) .c('register', {xmlns: "http://jabber.org/features/iq-register"}).up() .c('mechanisms', {xmlns: "urn:ietf:params:xml:ns:xmpp-sasl"}); _converse.connection._connect_cb(mock.createRequest(stanza)); stanza = u.toStanza(` Choose a username and password to register with this server urn:xmpp:captcha conversations.im 15376320046808160053 ad1e0d50-5adb-4397-a997-5feab56fe418:sendIQ cid:sha1+2df8c1b366f1e90ce60354f97d1fe75237290b8a@bob.xmpp.org iVBORw0KGgoAAAANSUhEUgAAALQAAAA8BAMAAAA9AI20AAAAMFBMVEX///8AAADf39+fn59fX19/f3+/v78fHx8/Pz9PT08bGxsvLy9jY2NTU1MXFxcnJyc84bkWAAAACXBIWXMAAA7EAAAOxAGVKw4bAAAERUlEQVRYhe1WTXMaRxDdDxY4JWpYvDinpVyxdATLin0MiRLlCHEi+7hYUcVHTSI7urhK6yr5//gn5N/4Z7inX89+CQkTcFUO6gOwS8/r7tdvesbzvoT5ROR5JJ9bB97xAK22XWAY1WznlnUr7QaAzSOsWufXQ6wH/FmO60b4D936LJr8TWRwW4SNgOsodZr8m4vZUoRt2xZ3xHXgna1FCE5+f5aWwPU//bXgg8eHjyqPp4aXJeOlwLUIt0O39zOvPWW3WfHmCCkli816FxlK0rnFGKZ484dN+eIXsw1R+G+JfjwgOpMnm+r5SxA63gS2Q8MchO1RLN8jSn4W4F5OPed2evhTthKLG3bsfjLL874XGBpWHLrU0953i/ev7JsfViHbhsWSQTunJDOppeAe0hVGokJUHBOphmjrbBlgabviJKXbIP0B//gKSBHZh2rvJnQp3wsapMFz+VsTPNhPr0Hn9N57YOjywaxFSU6S79fUF39KBDgnt6yjZOeSffk+4IXDZovbQl9E96m34EzQKMepQcbzijAGiBmDsO+LaqzqG3m3kEf+DQ2mY+vdk5c2n2Iaj5QGi6n59FHDmcuP4t8MGlRaF39P6ENyIaB2EXdpjLnQq9IgdVxfax3ilBc10u4gowX9K6BaKiZNmCC7CF/WpkJvWxN00OjuoqGYLqAnpILLE68Ymrt9M0S9hcznUJ8RykdlLalUfFaDjvA8pT2kxmsl5fuMaM6mSWUpUhDoudSucdhiZFDwphEHwsMwhEpH0jsm+/UBK2wCzFIiitalN7YjWkyIBgTNPgpDXX4rjk4UH+yPPgfK4HNZQCP/KZ0fGnrnKl8+pXl3X7FwZuwNUdwDGO+BjPUn6XaKtbkm+MJ6vtaXSnIz6wBT/m+VvZNIhz7ayabQLSeRQDmYkjt0KlmHDa555v9DzFxx+CCvCG4K3dbx6mTYtfPs1Dgdh0i3W+cl4lnnhblMKKBBA23X1Ezc3E5ZoPS5KHjPiU1rKTviYe1fTsa6e3UwXGWI4ykB8uiGqkmA6Cbf3K4JTH3LOBlbX+yPWll57LKVeH8CTEvyVPV2TXL8kPnPqtA51CaFYxOH2rJoZunSnvsSj48WiaDccl6KEgiMSarITsa+rWWBnqFloYlT1qWW2GKw9nPSbEvoVHFst967XgNQjxdA66Q6VFEUh488xfaSo7cHB52XYzA4eRlVteeT8ostWfuPea0oF6MwzlwgZE9gQI+uUV0gzK+WlpUrNI8juhhX/OyNwZnRrsDfxOqS1aDR+gC6NUPvJpvQeVZ9eiNr9aDUuddY3bLnA4tH4r/49UboznH1ia8PV/uP3WUB3dxtzj1uxfDZgbEbZx17Itwrf0Jyc8N4en+5dhivtKeYjGJ8yXgUzKvSU/uWJZmsuAYtseDku+K3zMHi4lC1h0suPmtZaEp2tm3hEV2lXwb6zu7szv6f9glF5rPGT5xR7AAAAABJRU5ErkJggg== You need a client that supports x:data and CAPTCHA to register `); _converse.connection._dataRecv(mock.createRequest(stanza)); expect(registerview.form_type).toBe('xform'); expect(registerview.el.querySelectorAll('#converse-register input[required="required"]').length).toBe(3); // Hide the controlbox so that we can see whether the test // passed or failed u.addClass('hidden', _converse.chatboxviews.get('controlbox').el); delete _converse.connection; done(); })); it("renders errors", mock.initConverse( ['chatBoxesInitialized'], { auto_login: false, view_mode: 'fullscreen', discover_connection_methods: false, allow_registration: true }, async function (done, _converse) { await u.waitUntil(() => _converse.chatboxviews.get('controlbox')?.registerpanel); const toggle = document.querySelector(".toggle-controlbox"); if (!u.isVisible(document.querySelector("#controlbox"))) { if (!u.isVisible(toggle)) { u.removeClass('hidden', toggle); } toggle.click(); } const cbview = _converse.chatboxviews.get('controlbox'); cbview.el.querySelector('.toggle-register-login').click(); const view = _converse.chatboxviews.get('controlbox').registerpanel; view.delegateEvents(); // We need to rebind all events otherwise our spy won't be called view.el.querySelector('input[name=domain]').value = 'conversejs.org'; view.el.querySelector('input[type=submit]').click(); let stanza = new Strophe.Builder("stream:features", { 'xmlns:stream': "http://etherx.jabber.org/streams", 'xmlns': "jabber:client" }) .c('register', {xmlns: "http://jabber.org/features/iq-register"}).up() .c('mechanisms', {xmlns: "urn:ietf:params:xml:ns:xmpp-sasl"}); _converse.connection._connect_cb(mock.createRequest(stanza)); stanza = u.toStanza(` Choose a username and password to register with this server urn:xmpp:captcha conversejs.org 15376320046808160053 ad1e0d50-5adb-4397-a997-5feab56fe418:sendIQ cid:sha1+2df8c1b366f1e90ce60354f97d1fe75237290b8a@bob.xmpp.org iVBORw0KGgoAAAANSUhEUgAAALQAAAA8BAMAAAA9AI20AAAAMFBMVEX///8AAADf39+fn59fX19/f3+/v78fHx8/Pz9PT08bGxsvLy9jY2NTU1MXFxcnJyc84bkWAAAACXBIWXMAAA7EAAAOxAGVKw4bAAAERUlEQVRYhe1WTXMaRxDdDxY4JWpYvDinpVyxdATLin0MiRLlCHEi+7hYUcVHTSI7urhK6yr5//gn5N/4Z7inX89+CQkTcFUO6gOwS8/r7tdvesbzvoT5ROR5JJ9bB97xAK22XWAY1WznlnUr7QaAzSOsWufXQ6wH/FmO60b4D936LJr8TWRwW4SNgOsodZr8m4vZUoRt2xZ3xHXgna1FCE5+f5aWwPU//bXgg8eHjyqPp4aXJeOlwLUIt0O39zOvPWW3WfHmCCkli816FxlK0rnFGKZ484dN+eIXsw1R+G+JfjwgOpMnm+r5SxA63gS2Q8MchO1RLN8jSn4W4F5OPed2evhTthKLG3bsfjLL874XGBpWHLrU0953i/ev7JsfViHbhsWSQTunJDOppeAe0hVGokJUHBOphmjrbBlgabviJKXbIP0B//gKSBHZh2rvJnQp3wsapMFz+VsTPNhPr0Hn9N57YOjywaxFSU6S79fUF39KBDgnt6yjZOeSffk+4IXDZovbQl9E96m34EzQKMepQcbzijAGiBmDsO+LaqzqG3m3kEf+DQ2mY+vdk5c2n2Iaj5QGi6n59FHDmcuP4t8MGlRaF39P6ENyIaB2EXdpjLnQq9IgdVxfax3ilBc10u4gowX9K6BaKiZNmCC7CF/WpkJvWxN00OjuoqGYLqAnpILLE68Ymrt9M0S9hcznUJ8RykdlLalUfFaDjvA8pT2kxmsl5fuMaM6mSWUpUhDoudSucdhiZFDwphEHwsMwhEpH0jsm+/UBK2wCzFIiitalN7YjWkyIBgTNPgpDXX4rjk4UH+yPPgfK4HNZQCP/KZ0fGnrnKl8+pXl3X7FwZuwNUdwDGO+BjPUn6XaKtbkm+MJ6vtaXSnIz6wBT/m+VvZNIhz7ayabQLSeRQDmYkjt0KlmHDa555v9DzFxx+CCvCG4K3dbx6mTYtfPs1Dgdh0i3W+cl4lnnhblMKKBBA23X1Ezc3E5ZoPS5KHjPiU1rKTviYe1fTsa6e3UwXGWI4ykB8uiGqkmA6Cbf3K4JTH3LOBlbX+yPWll57LKVeH8CTEvyVPV2TXL8kPnPqtA51CaFYxOH2rJoZunSnvsSj48WiaDccl6KEgiMSarITsa+rWWBnqFloYlT1qWW2GKw9nPSbEvoVHFst967XgNQjxdA66Q6VFEUh488xfaSo7cHB52XYzA4eRlVteeT8ostWfuPea0oF6MwzlwgZE9gQI+uUV0gzK+WlpUrNI8juhhX/OyNwZnRrsDfxOqS1aDR+gC6NUPvJpvQeVZ9eiNr9aDUuddY3bLnA4tH4r/49UboznH1ia8PV/uP3WUB3dxtzj1uxfDZgbEbZx17Itwrf0Jyc8N4en+5dhivtKeYjGJ8yXgUzKvSU/uWJZmsuAYtseDku+K3zMHi4lC1h0suPmtZaEp2tm3hEV2lXwb6zu7szv6f9glF5rPGT5xR7AAAAABJRU5ErkJggg== You need a client that supports x:data and CAPTCHA to register `); _converse.connection._dataRecv(mock.createRequest(stanza)); spyOn(view, 'submitRegistrationForm').and.callThrough(); const username_input = view.el.querySelector('[name="username"]'); username_input.value = 'romeo'; const password_input = view.el.querySelector('[name="password"]'); password_input.value = 'secret'; const ocr_input = view.el.querySelector('[name="ocr"]'); ocr_input.value = '8m9D88'; view.el.querySelector('[type="submit"]').click(); expect(view.submitRegistrationForm).toHaveBeenCalled(); const response_IQ = u.toStanza(` Too many CAPTCHA requests `); _converse.connection._dataRecv(mock.createRequest(response_IQ)); expect(view.el.querySelector('.error')?.textContent.trim()).toBe('Too many CAPTCHA requests'); delete _converse.connection; done(); })); });