diff options
Diffstat (limited to 'comm/chat/protocols/xmpp')
26 files changed, 8305 insertions, 0 deletions
diff --git a/comm/chat/protocols/xmpp/.eslintrc.js b/comm/chat/protocols/xmpp/.eslintrc.js new file mode 100644 index 0000000000..66953c7e25 --- /dev/null +++ b/comm/chat/protocols/xmpp/.eslintrc.js @@ -0,0 +1,12 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +"use strict"; + +module.exports = { + rules: { + // The following rules will not be enabled currently. + complexity: "off", + }, +}; diff --git a/comm/chat/protocols/xmpp/components.conf b/comm/chat/protocols/xmpp/components.conf new file mode 100644 index 0000000000..83943f1a34 --- /dev/null +++ b/comm/chat/protocols/xmpp/components.conf @@ -0,0 +1,15 @@ +# -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*- +# vim: set filetype=python: +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. + +Classes = [ + { + 'cid': '{dde786d1-6f59-43d0-9bc8-b505a757fb30}', + 'contract_ids': ['@mozilla.org/chat/xmpp;1'], + 'esModule': 'resource:///modules/xmpp.sys.mjs', + 'constructor': 'XMPPProtocol', + 'categories': {'im-protocol-plugin': 'prpl-jabber'}, + }, +] diff --git a/comm/chat/protocols/xmpp/icons/prpl-jabber-32.png b/comm/chat/protocols/xmpp/icons/prpl-jabber-32.png Binary files differnew file mode 100644 index 0000000000..98897f75fb --- /dev/null +++ b/comm/chat/protocols/xmpp/icons/prpl-jabber-32.png diff --git a/comm/chat/protocols/xmpp/icons/prpl-jabber-48.png b/comm/chat/protocols/xmpp/icons/prpl-jabber-48.png Binary files differnew file mode 100644 index 0000000000..805820c565 --- /dev/null +++ b/comm/chat/protocols/xmpp/icons/prpl-jabber-48.png diff --git a/comm/chat/protocols/xmpp/icons/prpl-jabber.png b/comm/chat/protocols/xmpp/icons/prpl-jabber.png Binary files differnew file mode 100644 index 0000000000..bb04c6e6df --- /dev/null +++ b/comm/chat/protocols/xmpp/icons/prpl-jabber.png diff --git a/comm/chat/protocols/xmpp/jar.mn b/comm/chat/protocols/xmpp/jar.mn new file mode 100644 index 0000000000..1f7ac54abc --- /dev/null +++ b/comm/chat/protocols/xmpp/jar.mn @@ -0,0 +1,5 @@ +chat.jar: +% skin prpl-jabber classic/1.0 %skin/classic/prpl/xmpp/ + skin/classic/prpl/xmpp/icon32.png (icons/prpl-jabber-32.png) + skin/classic/prpl/xmpp/icon48.png (icons/prpl-jabber-48.png) + skin/classic/prpl/xmpp/icon.png (icons/prpl-jabber.png) diff --git a/comm/chat/protocols/xmpp/lib/README.md b/comm/chat/protocols/xmpp/lib/README.md new file mode 100644 index 0000000000..c813b5d070 --- /dev/null +++ b/comm/chat/protocols/xmpp/lib/README.md @@ -0,0 +1,6 @@ +This directory contains sax-js from https://github.com/isaacs/sax-js. Current version is v1.2.4. + +## Updating sax-js + +1. Download a release version from https://github.com/isaacs/sax-js. +2. Copy `lib/sax.js` from the git repo to `sax/sax.js` in this directory. diff --git a/comm/chat/protocols/xmpp/lib/moz.build b/comm/chat/protocols/xmpp/lib/moz.build new file mode 100644 index 0000000000..b4f1787145 --- /dev/null +++ b/comm/chat/protocols/xmpp/lib/moz.build @@ -0,0 +1,8 @@ +# vim: set filetype=python: +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. + +EXTRA_JS_MODULES.sax += [ + "sax/sax.js", +] diff --git a/comm/chat/protocols/xmpp/lib/sax/LICENSE b/comm/chat/protocols/xmpp/lib/sax/LICENSE new file mode 100644 index 0000000000..ccffa082c9 --- /dev/null +++ b/comm/chat/protocols/xmpp/lib/sax/LICENSE @@ -0,0 +1,41 @@ +The ISC License + +Copyright (c) Isaac Z. Schlueter and Contributors + +Permission to use, copy, modify, and/or distribute this software for any +purpose with or without fee is hereby granted, provided that the above +copyright notice and this permission notice appear in all copies. + +THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR +IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + +==== + +`String.fromCodePoint` by Mathias Bynens used according to terms of MIT +License, as follows: + + Copyright Mathias Bynens <https://mathiasbynens.be/> + + Permission is hereby granted, free of charge, to any person obtaining + a copy of this software and associated documentation files (the + "Software"), to deal in the Software without restriction, including + without limitation the rights to use, copy, modify, merge, publish, + distribute, sublicense, and/or sell copies of the Software, and to + permit persons to whom the Software is furnished to do so, subject to + the following conditions: + + The above copyright notice and this permission notice shall be + included in all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, + EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF + MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND + NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE + LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION + OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION + WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/comm/chat/protocols/xmpp/lib/sax/sax.js b/comm/chat/protocols/xmpp/lib/sax/sax.js new file mode 100644 index 0000000000..564d8d4235 --- /dev/null +++ b/comm/chat/protocols/xmpp/lib/sax/sax.js @@ -0,0 +1,1648 @@ +/* This program is made available under an ISC-style license. */ +(function(sax) { + // wrapper for non-node envs + sax.parser = function(strict, opt) { + return new SAXParser(strict, opt); + }; + sax.SAXParser = SAXParser; + sax.SAXStream = SAXStream; + sax.createStream = createStream; + + // When we pass the MAX_BUFFER_LENGTH position, start checking for buffer overruns. + // When we check, schedule the next check for MAX_BUFFER_LENGTH - (max(buffer lengths)), + // since that's the earliest that a buffer overrun could occur. This way, checks are + // as rare as required, but as often as necessary to ensure never crossing this bound. + // Furthermore, buffers are only tested at most once per write(), so passing a very + // large string into write() might have undesirable effects, but this is manageable by + // the caller, so it is assumed to be safe. Thus, a call to write() may, in the extreme + // edge case, result in creating at most one complete copy of the string passed in. + // Set to Infinity to have unlimited buffers. + sax.MAX_BUFFER_LENGTH = 64 * 1024; + + var buffers = [ + "comment", + "sgmlDecl", + "textNode", + "tagName", + "doctype", + "procInstName", + "procInstBody", + "entity", + "attribName", + "attribValue", + "cdata", + "script", + ]; + + sax.EVENTS = [ + "text", + "processinginstruction", + "sgmldeclaration", + "doctype", + "comment", + "opentagstart", + "attribute", + "opentag", + "closetag", + "opencdata", + "cdata", + "closecdata", + "error", + "end", + "ready", + "script", + "opennamespace", + "closenamespace", + ]; + + function SAXParser(strict, opt) { + if (!(this instanceof SAXParser)) { + return new SAXParser(strict, opt); + } + + var parser = this; + clearBuffers(parser); + parser.q = parser.c = ""; + parser.bufferCheckPosition = sax.MAX_BUFFER_LENGTH; + parser.opt = opt || {}; + parser.opt.lowercase = parser.opt.lowercase || parser.opt.lowercasetags; + parser.looseCase = parser.opt.lowercase ? "toLowerCase" : "toUpperCase"; + parser.tags = []; + parser.closed = parser.closedRoot = parser.sawRoot = false; + parser.tag = parser.error = null; + parser.strict = !!strict; + parser.noscript = !!(strict || parser.opt.noscript); + parser.state = S.BEGIN; + parser.strictEntities = parser.opt.strictEntities; + parser.ENTITIES = parser.strictEntities + ? Object.create(sax.XML_ENTITIES) + : Object.create(sax.ENTITIES); + parser.attribList = []; + + // namespaces form a prototype chain. + // it always points at the current tag, + // which protos to its parent tag. + if (parser.opt.xmlns) { + parser.ns = Object.create(rootNS); + } + + // mostly just for error reporting + parser.trackPosition = parser.opt.position !== false; + if (parser.trackPosition) { + parser.position = parser.line = parser.column = 0; + } + emit(parser, "onready"); + } + + if (!Object.create) { + Object.create = function(o) { + function F() {} + F.prototype = o; + var newf = new F(); + return newf; + }; + } + + if (!Object.keys) { + Object.keys = function(o) { + var a = []; + for (var i in o) { + if (o.hasOwnProperty(i)) { + a.push(i); + } + } + return a; + }; + } + + function checkBufferLength(parser) { + var maxAllowed = Math.max(sax.MAX_BUFFER_LENGTH, 10); + var maxActual = 0; + for (var i = 0, l = buffers.length; i < l; i++) { + var len = parser[buffers[i]].length; + if (len > maxAllowed) { + // Text/cdata nodes can get big, and since they're buffered, + // we can get here under normal conditions. + // Avoid issues by emitting the text node now, + // so at least it won't get any bigger. + switch (buffers[i]) { + case "textNode": + closeText(parser); + break; + + case "cdata": + emitNode(parser, "oncdata", parser.cdata); + parser.cdata = ""; + break; + + case "script": + emitNode(parser, "onscript", parser.script); + parser.script = ""; + break; + + default: + error(parser, "Max buffer length exceeded: " + buffers[i]); + } + } + maxActual = Math.max(maxActual, len); + } + // schedule the next check for the earliest possible buffer overrun. + var m = sax.MAX_BUFFER_LENGTH - maxActual; + parser.bufferCheckPosition = m + parser.position; + } + + function clearBuffers(parser) { + for (var i = 0, l = buffers.length; i < l; i++) { + parser[buffers[i]] = ""; + } + } + + function flushBuffers(parser) { + closeText(parser); + if (parser.cdata !== "") { + emitNode(parser, "oncdata", parser.cdata); + parser.cdata = ""; + } + if (parser.script !== "") { + emitNode(parser, "onscript", parser.script); + parser.script = ""; + } + } + + SAXParser.prototype = { + end() { + end(this); + }, + write, + resume() { + this.error = null; + return this; + }, + close() { + return this.write(null); + }, + flush() { + flushBuffers(this); + }, + }; + + var Stream; + try { + Stream = require("stream").Stream; + } catch (ex) { + Stream = function() {}; + } + + var streamWraps = sax.EVENTS.filter(function(ev) { + return ev !== "error" && ev !== "end"; + }); + + function createStream(strict, opt) { + return new SAXStream(strict, opt); + } + + function SAXStream(strict, opt) { + if (!(this instanceof SAXStream)) { + return new SAXStream(strict, opt); + } + + Stream.apply(this); + + this._parser = new SAXParser(strict, opt); + this.writable = true; + this.readable = true; + + var me = this; + + this._parser.onend = function() { + me.emit("end"); + }; + + this._parser.onerror = function(er) { + me.emit("error", er); + + // if didn't throw, then means error was handled. + // go ahead and clear error, so we can write again. + me._parser.error = null; + }; + + this._decoder = null; + + streamWraps.forEach(function(ev) { + Object.defineProperty(me, "on" + ev, { + get() { + return me._parser["on" + ev]; + }, + set(h) { + if (!h) { + me.removeAllListeners(ev); + me._parser["on" + ev] = h; + return h; + } + me.on(ev, h); + }, + enumerable: true, + configurable: false, + }); + }); + } + + SAXStream.prototype = Object.create(Stream.prototype, { + constructor: { + value: SAXStream, + }, + }); + + SAXStream.prototype.write = function(data) { + if ( + typeof Buffer === "function" && + typeof Buffer.isBuffer === "function" && + Buffer.isBuffer(data) + ) { + if (!this._decoder) { + var SD = require("string_decoder").StringDecoder; + this._decoder = new SD("utf8"); + } + data = this._decoder.write(data); + } + + this._parser.write(data.toString()); + this.emit("data", data); + return true; + }; + + SAXStream.prototype.end = function(chunk) { + if (chunk && chunk.length) { + this.write(chunk); + } + this._parser.end(); + return true; + }; + + SAXStream.prototype.on = function(ev, handler) { + var me = this; + if (!me._parser["on" + ev] && streamWraps.indexOf(ev) !== -1) { + me._parser["on" + ev] = function() { + var args = + arguments.length === 1 + ? [arguments[0]] + : Array.apply(null, arguments); + args.splice(0, 0, ev); + me.emit.apply(me, args); + }; + } + + return Stream.prototype.on.call(me, ev, handler); + }; + + // this really needs to be replaced with character classes. + // XML allows all manner of ridiculous numbers and digits. + var CDATA = "[CDATA["; + var DOCTYPE = "DOCTYPE"; + var XML_NAMESPACE = "http://www.w3.org/XML/1998/namespace"; + var XMLNS_NAMESPACE = "http://www.w3.org/2000/xmlns/"; + var rootNS = { xml: XML_NAMESPACE, xmlns: XMLNS_NAMESPACE }; + + // http://www.w3.org/TR/REC-xml/#NT-NameStartChar + // This implementation works on strings, a single character at a time + // as such, it cannot ever support astral-plane characters (10000-EFFFF) + // without a significant breaking change to either this parser, or the + // JavaScript language. Implementation of an emoji-capable xml parser + // is left as an exercise for the reader. + var nameStart = /[:_A-Za-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD]/; + + var nameBody = /[:_A-Za-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD\u00B7\u0300-\u036F\u203F-\u2040.\d-]/; + + var entityStart = /[#:_A-Za-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD]/; + var entityBody = /[#:_A-Za-z\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u02FF\u0370-\u037D\u037F-\u1FFF\u200C-\u200D\u2070-\u218F\u2C00-\u2FEF\u3001-\uD7FF\uF900-\uFDCF\uFDF0-\uFFFD\u00B7\u0300-\u036F\u203F-\u2040.\d-]/; + + function isWhitespace(c) { + return c === " " || c === "\n" || c === "\r" || c === "\t"; + } + + function isQuote(c) { + return c === '"' || c === "'"; + } + + function isAttribEnd(c) { + return c === ">" || isWhitespace(c); + } + + function isMatch(regex, c) { + return regex.test(c); + } + + function notMatch(regex, c) { + return !isMatch(regex, c); + } + + var S = 0; + sax.STATE = { + BEGIN: S++, // leading byte order mark or whitespace + BEGIN_WHITESPACE: S++, // leading whitespace + TEXT: S++, // general stuff + TEXT_ENTITY: S++, // & and such. + OPEN_WAKA: S++, // < + SGML_DECL: S++, // <!BLARG + SGML_DECL_QUOTED: S++, // <!BLARG foo "bar + DOCTYPE: S++, // <!DOCTYPE + DOCTYPE_QUOTED: S++, // <!DOCTYPE "//blah + DOCTYPE_DTD: S++, // <!DOCTYPE "//blah" [ ... + DOCTYPE_DTD_QUOTED: S++, // <!DOCTYPE "//blah" [ "foo + COMMENT_STARTING: S++, // <!- + COMMENT: S++, // <!-- + COMMENT_ENDING: S++, // <!-- blah - + COMMENT_ENDED: S++, // <!-- blah -- + CDATA: S++, // <![CDATA[ something + CDATA_ENDING: S++, // ] + CDATA_ENDING_2: S++, // ]] + PROC_INST: S++, // <?hi + PROC_INST_BODY: S++, // <?hi there + PROC_INST_ENDING: S++, // <?hi "there" ? + OPEN_TAG: S++, // <strong + OPEN_TAG_SLASH: S++, // <strong / + ATTRIB: S++, // <a + ATTRIB_NAME: S++, // <a foo + ATTRIB_NAME_SAW_WHITE: S++, // <a foo _ + ATTRIB_VALUE: S++, // <a foo= + ATTRIB_VALUE_QUOTED: S++, // <a foo="bar + ATTRIB_VALUE_CLOSED: S++, // <a foo="bar" + ATTRIB_VALUE_UNQUOTED: S++, // <a foo=bar + ATTRIB_VALUE_ENTITY_Q: S++, // <foo bar=""" + ATTRIB_VALUE_ENTITY_U: S++, // <foo bar=" + CLOSE_TAG: S++, // </a + CLOSE_TAG_SAW_WHITE: S++, // </a > + SCRIPT: S++, // <script> ... + SCRIPT_ENDING: S++, // <script> ... < + }; + + sax.XML_ENTITIES = { + amp: "&", + gt: ">", + lt: "<", + quot: '"', + apos: "'", + }; + + sax.ENTITIES = { + amp: "&", + gt: ">", + lt: "<", + quot: '"', + apos: "'", + AElig: 198, + Aacute: 193, + Acirc: 194, + Agrave: 192, + Aring: 197, + Atilde: 195, + Auml: 196, + Ccedil: 199, + ETH: 208, + Eacute: 201, + Ecirc: 202, + Egrave: 200, + Euml: 203, + Iacute: 205, + Icirc: 206, + Igrave: 204, + Iuml: 207, + Ntilde: 209, + Oacute: 211, + Ocirc: 212, + Ograve: 210, + Oslash: 216, + Otilde: 213, + Ouml: 214, + THORN: 222, + Uacute: 218, + Ucirc: 219, + Ugrave: 217, + Uuml: 220, + Yacute: 221, + aacute: 225, + acirc: 226, + aelig: 230, + agrave: 224, + aring: 229, + atilde: 227, + auml: 228, + ccedil: 231, + eacute: 233, + ecirc: 234, + egrave: 232, + eth: 240, + euml: 235, + iacute: 237, + icirc: 238, + igrave: 236, + iuml: 239, + ntilde: 241, + oacute: 243, + ocirc: 244, + ograve: 242, + oslash: 248, + otilde: 245, + ouml: 246, + szlig: 223, + thorn: 254, + uacute: 250, + ucirc: 251, + ugrave: 249, + uuml: 252, + yacute: 253, + yuml: 255, + copy: 169, + reg: 174, + nbsp: 160, + iexcl: 161, + cent: 162, + pound: 163, + curren: 164, + yen: 165, + brvbar: 166, + sect: 167, + uml: 168, + ordf: 170, + laquo: 171, + not: 172, + shy: 173, + macr: 175, + deg: 176, + plusmn: 177, + sup1: 185, + sup2: 178, + sup3: 179, + acute: 180, + micro: 181, + para: 182, + middot: 183, + cedil: 184, + ordm: 186, + raquo: 187, + frac14: 188, + frac12: 189, + frac34: 190, + iquest: 191, + times: 215, + divide: 247, + OElig: 338, + oelig: 339, + Scaron: 352, + scaron: 353, + Yuml: 376, + fnof: 402, + circ: 710, + tilde: 732, + Alpha: 913, + Beta: 914, + Gamma: 915, + Delta: 916, + Epsilon: 917, + Zeta: 918, + Eta: 919, + Theta: 920, + Iota: 921, + Kappa: 922, + Lambda: 923, + Mu: 924, + Nu: 925, + Xi: 926, + Omicron: 927, + Pi: 928, + Rho: 929, + Sigma: 931, + Tau: 932, + Upsilon: 933, + Phi: 934, + Chi: 935, + Psi: 936, + Omega: 937, + alpha: 945, + beta: 946, + gamma: 947, + delta: 948, + epsilon: 949, + zeta: 950, + eta: 951, + theta: 952, + iota: 953, + kappa: 954, + lambda: 955, + mu: 956, + nu: 957, + xi: 958, + omicron: 959, + pi: 960, + rho: 961, + sigmaf: 962, + sigma: 963, + tau: 964, + upsilon: 965, + phi: 966, + chi: 967, + psi: 968, + omega: 969, + thetasym: 977, + upsih: 978, + piv: 982, + ensp: 8194, + emsp: 8195, + thinsp: 8201, + zwnj: 8204, + zwj: 8205, + lrm: 8206, + rlm: 8207, + ndash: 8211, + mdash: 8212, + lsquo: 8216, + rsquo: 8217, + sbquo: 8218, + ldquo: 8220, + rdquo: 8221, + bdquo: 8222, + dagger: 8224, + Dagger: 8225, + bull: 8226, + hellip: 8230, + permil: 8240, + prime: 8242, + Prime: 8243, + lsaquo: 8249, + rsaquo: 8250, + oline: 8254, + frasl: 8260, + euro: 8364, + image: 8465, + weierp: 8472, + real: 8476, + trade: 8482, + alefsym: 8501, + larr: 8592, + uarr: 8593, + rarr: 8594, + darr: 8595, + harr: 8596, + crarr: 8629, + lArr: 8656, + uArr: 8657, + rArr: 8658, + dArr: 8659, + hArr: 8660, + forall: 8704, + part: 8706, + exist: 8707, + empty: 8709, + nabla: 8711, + isin: 8712, + notin: 8713, + ni: 8715, + prod: 8719, + sum: 8721, + minus: 8722, + lowast: 8727, + radic: 8730, + prop: 8733, + infin: 8734, + ang: 8736, + and: 8743, + or: 8744, + cap: 8745, + cup: 8746, + int: 8747, + there4: 8756, + sim: 8764, + cong: 8773, + asymp: 8776, + ne: 8800, + equiv: 8801, + le: 8804, + ge: 8805, + sub: 8834, + sup: 8835, + nsub: 8836, + sube: 8838, + supe: 8839, + oplus: 8853, + otimes: 8855, + perp: 8869, + sdot: 8901, + lceil: 8968, + rceil: 8969, + lfloor: 8970, + rfloor: 8971, + lang: 9001, + rang: 9002, + loz: 9674, + spades: 9824, + clubs: 9827, + hearts: 9829, + diams: 9830, + }; + + Object.keys(sax.ENTITIES).forEach(function(key) { + var e = sax.ENTITIES[key]; + var s = typeof e === "number" ? String.fromCharCode(e) : e; + sax.ENTITIES[key] = s; + }); + + for (var s in sax.STATE) { + sax.STATE[sax.STATE[s]] = s; + } + + // shorthand + S = sax.STATE; + + function emit(parser, event, data) { + parser[event] && parser[event](data); + } + + function emitNode(parser, nodeType, data) { + if (parser.textNode) { + closeText(parser); + } + emit(parser, nodeType, data); + } + + function closeText(parser) { + parser.textNode = textopts(parser.opt, parser.textNode); + if (parser.textNode) { + emit(parser, "ontext", parser.textNode); + } + parser.textNode = ""; + } + + function textopts(opt, text) { + if (opt.trim) { + text = text.trim(); + } + if (opt.normalize) { + text = text.replace(/\s+/g, " "); + } + return text; + } + + function error(parser, er) { + closeText(parser); + if (parser.trackPosition) { + er += + "\nLine: " + + parser.line + + "\nColumn: " + + parser.column + + "\nChar: " + + parser.c; + } + er = new Error(er); + parser.error = er; + emit(parser, "onerror", er); + return parser; + } + + function end(parser) { + if (parser.sawRoot && !parser.closedRoot) { + strictFail(parser, "Unclosed root tag"); + } + if ( + parser.state !== S.BEGIN && + parser.state !== S.BEGIN_WHITESPACE && + parser.state !== S.TEXT + ) { + error(parser, "Unexpected end"); + } + closeText(parser); + parser.c = ""; + parser.closed = true; + emit(parser, "onend"); + SAXParser.call(parser, parser.strict, parser.opt); + return parser; + } + + function strictFail(parser, message) { + if (typeof parser !== "object" || !(parser instanceof SAXParser)) { + throw new Error("bad call to strictFail"); + } + if (parser.strict) { + error(parser, message); + } + } + + function newTag(parser) { + if (!parser.strict) { + parser.tagName = parser.tagName[parser.looseCase](); + } + var parent = parser.tags[parser.tags.length - 1] || parser; + var tag = (parser.tag = { name: parser.tagName, attributes: {} }); + + // will be overridden if tag contails an xmlns="foo" or xmlns:foo="bar" + if (parser.opt.xmlns) { + tag.ns = parent.ns; + } + parser.attribList.length = 0; + emitNode(parser, "onopentagstart", tag); + } + + function qname(name, attribute) { + var i = name.indexOf(":"); + var qualName = i < 0 ? ["", name] : name.split(":"); + var prefix = qualName[0]; + var local = qualName[1]; + + // <x "xmlns"="http://foo"> + if (attribute && name === "xmlns") { + prefix = "xmlns"; + local = ""; + } + + return { prefix, local }; + } + + function attrib(parser) { + if (!parser.strict) { + parser.attribName = parser.attribName[parser.looseCase](); + } + + if ( + parser.attribList.indexOf(parser.attribName) !== -1 || + parser.tag.attributes.hasOwnProperty(parser.attribName) + ) { + parser.attribName = parser.attribValue = ""; + return; + } + + if (parser.opt.xmlns) { + var qn = qname(parser.attribName, true); + var prefix = qn.prefix; + var local = qn.local; + + if (prefix === "xmlns") { + // namespace binding attribute. push the binding into scope + if (local === "xml" && parser.attribValue !== XML_NAMESPACE) { + strictFail( + parser, + "xml: prefix must be bound to " + + XML_NAMESPACE + + "\n" + + "Actual: " + + parser.attribValue + ); + } else if ( + local === "xmlns" && + parser.attribValue !== XMLNS_NAMESPACE + ) { + strictFail( + parser, + "xmlns: prefix must be bound to " + + XMLNS_NAMESPACE + + "\n" + + "Actual: " + + parser.attribValue + ); + } else { + var tag = parser.tag; + var parent = parser.tags[parser.tags.length - 1] || parser; + if (tag.ns === parent.ns) { + tag.ns = Object.create(parent.ns); + } + tag.ns[local] = parser.attribValue; + } + } + + // defer onattribute events until all attributes have been seen + // so any new bindings can take effect. preserve attribute order + // so deferred events can be emitted in document order + parser.attribList.push([parser.attribName, parser.attribValue]); + } else { + // in non-xmlns mode, we can emit the event right away + parser.tag.attributes[parser.attribName] = parser.attribValue; + emitNode(parser, "onattribute", { + name: parser.attribName, + value: parser.attribValue, + }); + } + + parser.attribName = parser.attribValue = ""; + } + + function openTag(parser, selfClosing) { + if (parser.opt.xmlns) { + // emit namespace binding events + var tag = parser.tag; + + // add namespace info to tag + var qn = qname(parser.tagName); + tag.prefix = qn.prefix; + tag.local = qn.local; + tag.uri = tag.ns[qn.prefix] || ""; + + if (tag.prefix && !tag.uri) { + strictFail( + parser, + "Unbound namespace prefix: " + JSON.stringify(parser.tagName) + ); + tag.uri = qn.prefix; + } + + var parent = parser.tags[parser.tags.length - 1] || parser; + if (tag.ns && parent.ns !== tag.ns) { + Object.keys(tag.ns).forEach(function(p) { + emitNode(parser, "onopennamespace", { + prefix: p, + uri: tag.ns[p], + }); + }); + } + + // handle deferred onattribute events + // Note: do not apply default ns to attributes: + // http://www.w3.org/TR/REC-xml-names/#defaulting + for (var i = 0, l = parser.attribList.length; i < l; i++) { + var nv = parser.attribList[i]; + var name = nv[0]; + var value = nv[1]; + var qualName = qname(name, true); + var prefix = qualName.prefix; + var local = qualName.local; + var uri = prefix === "" ? "" : tag.ns[prefix] || ""; + var a = { + name, + value, + prefix, + local, + uri, + }; + + // if there's any attributes with an undefined namespace, + // then fail on them now. + if (prefix && prefix !== "xmlns" && !uri) { + strictFail( + parser, + "Unbound namespace prefix: " + JSON.stringify(prefix) + ); + a.uri = prefix; + } + parser.tag.attributes[name] = a; + emitNode(parser, "onattribute", a); + } + parser.attribList.length = 0; + } + + parser.tag.isSelfClosing = !!selfClosing; + + // process the tag + parser.sawRoot = true; + parser.tags.push(parser.tag); + emitNode(parser, "onopentag", parser.tag); + if (!selfClosing) { + // special case for <script> in non-strict mode. + if (!parser.noscript && parser.tagName.toLowerCase() === "script") { + parser.state = S.SCRIPT; + } else { + parser.state = S.TEXT; + } + parser.tag = null; + parser.tagName = ""; + } + parser.attribName = parser.attribValue = ""; + parser.attribList.length = 0; + } + + function closeTag(parser) { + if (!parser.tagName) { + strictFail(parser, "Weird empty close tag."); + parser.textNode += "</>"; + parser.state = S.TEXT; + return; + } + + if (parser.script) { + if (parser.tagName !== "script") { + parser.script += "</" + parser.tagName + ">"; + parser.tagName = ""; + parser.state = S.SCRIPT; + return; + } + emitNode(parser, "onscript", parser.script); + parser.script = ""; + } + + // first make sure that the closing tag actually exists. + // <a><b></c></b></a> will close everything, otherwise. + var t = parser.tags.length; + var tagName = parser.tagName; + if (!parser.strict) { + tagName = tagName[parser.looseCase](); + } + var closeTo = tagName; + while (t--) { + var close = parser.tags[t]; + if (close.name !== closeTo) { + // fail the first time in strict mode + strictFail(parser, "Unexpected close tag"); + } else { + break; + } + } + + // didn't find it. we already failed for strict, so just abort. + if (t < 0) { + strictFail(parser, "Unmatched closing tag: " + parser.tagName); + parser.textNode += "</" + parser.tagName + ">"; + parser.state = S.TEXT; + return; + } + parser.tagName = tagName; + var s = parser.tags.length; + while (s-- > t) { + var tag = (parser.tag = parser.tags.pop()); + parser.tagName = parser.tag.name; + emitNode(parser, "onclosetag", parser.tagName); + + var x = {}; + for (var i in tag.ns) { + x[i] = tag.ns[i]; + } + + var parent = parser.tags[parser.tags.length - 1] || parser; + if (parser.opt.xmlns && tag.ns !== parent.ns) { + // remove namespace bindings introduced by tag + Object.keys(tag.ns).forEach(function(p) { + var n = tag.ns[p]; + emitNode(parser, "onclosenamespace", { prefix: p, uri: n }); + }); + } + } + if (t === 0) { + parser.closedRoot = true; + } + parser.tagName = parser.attribValue = parser.attribName = ""; + parser.attribList.length = 0; + parser.state = S.TEXT; + } + + function parseEntity(parser) { + var entity = parser.entity; + var entityLC = entity.toLowerCase(); + var num; + var numStr = ""; + + if (parser.ENTITIES[entity]) { + return parser.ENTITIES[entity]; + } + if (parser.ENTITIES[entityLC]) { + return parser.ENTITIES[entityLC]; + } + entity = entityLC; + if (entity.charAt(0) === "#") { + if (entity.charAt(1) === "x") { + entity = entity.slice(2); + num = parseInt(entity, 16); + numStr = num.toString(16); + } else { + entity = entity.slice(1); + num = parseInt(entity, 10); + numStr = num.toString(10); + } + } + entity = entity.replace(/^0+/, ""); + if (isNaN(num) || numStr.toLowerCase() !== entity) { + strictFail(parser, "Invalid character entity"); + return "&" + parser.entity + ";"; + } + + return String.fromCodePoint(num); + } + + function beginWhiteSpace(parser, c) { + if (c === "<") { + parser.state = S.OPEN_WAKA; + parser.startTagPosition = parser.position; + } else if (!isWhitespace(c)) { + // have to process this as a text node. + // weird, but happens. + strictFail(parser, "Non-whitespace before first tag."); + parser.textNode = c; + parser.state = S.TEXT; + } + } + + function charAt(chunk, i) { + var result = ""; + if (i < chunk.length) { + result = chunk.charAt(i); + } + return result; + } + + function write(chunk) { + var parser = this; + if (this.error) { + throw this.error; + } + if (parser.closed) { + return error( + parser, + "Cannot write after close. Assign an onready handler." + ); + } + if (chunk === null) { + return end(parser); + } + if (typeof chunk === "object") { + chunk = chunk.toString(); + } + var i = 0; + var c = ""; + while (true) { + c = charAt(chunk, i++); + parser.c = c; + + if (!c) { + break; + } + + if (parser.trackPosition) { + parser.position++; + if (c === "\n") { + parser.line++; + parser.column = 0; + } else { + parser.column++; + } + } + + switch (parser.state) { + case S.BEGIN: + parser.state = S.BEGIN_WHITESPACE; + if (c === "\uFEFF") { + continue; + } + beginWhiteSpace(parser, c); + continue; + + case S.BEGIN_WHITESPACE: + beginWhiteSpace(parser, c); + continue; + + case S.TEXT: + if (parser.sawRoot && !parser.closedRoot) { + var starti = i - 1; + while (c && c !== "<" && c !== "&") { + c = charAt(chunk, i++); + if (c && parser.trackPosition) { + parser.position++; + if (c === "\n") { + parser.line++; + parser.column = 0; + } else { + parser.column++; + } + } + } + parser.textNode += chunk.substring(starti, i - 1); + } + if ( + c === "<" && + !(parser.sawRoot && parser.closedRoot && !parser.strict) + ) { + parser.state = S.OPEN_WAKA; + parser.startTagPosition = parser.position; + } else { + if (!isWhitespace(c) && (!parser.sawRoot || parser.closedRoot)) { + strictFail(parser, "Text data outside of root node."); + } + if (c === "&") { + parser.state = S.TEXT_ENTITY; + } else { + parser.textNode += c; + } + } + continue; + + case S.SCRIPT: + // only non-strict + if (c === "<") { + parser.state = S.SCRIPT_ENDING; + } else { + parser.script += c; + } + continue; + + case S.SCRIPT_ENDING: + if (c === "/") { + parser.state = S.CLOSE_TAG; + } else { + parser.script += "<" + c; + parser.state = S.SCRIPT; + } + continue; + + case S.OPEN_WAKA: + // either a /, ?, !, or text is coming next. + if (c === "!") { + parser.state = S.SGML_DECL; + parser.sgmlDecl = ""; + } else if (isWhitespace(c)) { + // wait for it... + } else if (isMatch(nameStart, c)) { + parser.state = S.OPEN_TAG; + parser.tagName = c; + } else if (c === "/") { + parser.state = S.CLOSE_TAG; + parser.tagName = ""; + } else if (c === "?") { + parser.state = S.PROC_INST; + parser.procInstName = parser.procInstBody = ""; + } else { + strictFail(parser, "Unencoded <"); + // if there was some whitespace, then add that in. + if (parser.startTagPosition + 1 < parser.position) { + var pad = parser.position - parser.startTagPosition; + c = new Array(pad).join(" ") + c; + } + parser.textNode += "<" + c; + parser.state = S.TEXT; + } + continue; + + case S.SGML_DECL: + if ((parser.sgmlDecl + c).toUpperCase() === CDATA) { + emitNode(parser, "onopencdata"); + parser.state = S.CDATA; + parser.sgmlDecl = ""; + parser.cdata = ""; + } else if (parser.sgmlDecl + c === "--") { + parser.state = S.COMMENT; + parser.comment = ""; + parser.sgmlDecl = ""; + } else if ((parser.sgmlDecl + c).toUpperCase() === DOCTYPE) { + parser.state = S.DOCTYPE; + if (parser.doctype || parser.sawRoot) { + strictFail(parser, "Inappropriately located doctype declaration"); + } + parser.doctype = ""; + parser.sgmlDecl = ""; + } else if (c === ">") { + emitNode(parser, "onsgmldeclaration", parser.sgmlDecl); + parser.sgmlDecl = ""; + parser.state = S.TEXT; + } else if (isQuote(c)) { + parser.state = S.SGML_DECL_QUOTED; + parser.sgmlDecl += c; + } else { + parser.sgmlDecl += c; + } + continue; + + case S.SGML_DECL_QUOTED: + if (c === parser.q) { + parser.state = S.SGML_DECL; + parser.q = ""; + } + parser.sgmlDecl += c; + continue; + + case S.DOCTYPE: + if (c === ">") { + parser.state = S.TEXT; + emitNode(parser, "ondoctype", parser.doctype); + parser.doctype = true; // just remember that we saw it. + } else { + parser.doctype += c; + if (c === "[") { + parser.state = S.DOCTYPE_DTD; + } else if (isQuote(c)) { + parser.state = S.DOCTYPE_QUOTED; + parser.q = c; + } + } + continue; + + case S.DOCTYPE_QUOTED: + parser.doctype += c; + if (c === parser.q) { + parser.q = ""; + parser.state = S.DOCTYPE; + } + continue; + + case S.DOCTYPE_DTD: + parser.doctype += c; + if (c === "]") { + parser.state = S.DOCTYPE; + } else if (isQuote(c)) { + parser.state = S.DOCTYPE_DTD_QUOTED; + parser.q = c; + } + continue; + + case S.DOCTYPE_DTD_QUOTED: + parser.doctype += c; + if (c === parser.q) { + parser.state = S.DOCTYPE_DTD; + parser.q = ""; + } + continue; + + case S.COMMENT: + if (c === "-") { + parser.state = S.COMMENT_ENDING; + } else { + parser.comment += c; + } + continue; + + case S.COMMENT_ENDING: + if (c === "-") { + parser.state = S.COMMENT_ENDED; + parser.comment = textopts(parser.opt, parser.comment); + if (parser.comment) { + emitNode(parser, "oncomment", parser.comment); + } + parser.comment = ""; + } else { + parser.comment += "-" + c; + parser.state = S.COMMENT; + } + continue; + + case S.COMMENT_ENDED: + if (c !== ">") { + strictFail(parser, "Malformed comment"); + // allow <!-- blah -- bloo --> in non-strict mode, + // which is a comment of " blah -- bloo " + parser.comment += "--" + c; + parser.state = S.COMMENT; + } else { + parser.state = S.TEXT; + } + continue; + + case S.CDATA: + if (c === "]") { + parser.state = S.CDATA_ENDING; + } else { + parser.cdata += c; + } + continue; + + case S.CDATA_ENDING: + if (c === "]") { + parser.state = S.CDATA_ENDING_2; + } else { + parser.cdata += "]" + c; + parser.state = S.CDATA; + } + continue; + + case S.CDATA_ENDING_2: + if (c === ">") { + if (parser.cdata) { + emitNode(parser, "oncdata", parser.cdata); + } + emitNode(parser, "onclosecdata"); + parser.cdata = ""; + parser.state = S.TEXT; + } else if (c === "]") { + parser.cdata += "]"; + } else { + parser.cdata += "]]" + c; + parser.state = S.CDATA; + } + continue; + + case S.PROC_INST: + if (c === "?") { + parser.state = S.PROC_INST_ENDING; + } else if (isWhitespace(c)) { + parser.state = S.PROC_INST_BODY; + } else { + parser.procInstName += c; + } + continue; + + case S.PROC_INST_BODY: + if (!parser.procInstBody && isWhitespace(c)) { + continue; + } else if (c === "?") { + parser.state = S.PROC_INST_ENDING; + } else { + parser.procInstBody += c; + } + continue; + + case S.PROC_INST_ENDING: + if (c === ">") { + emitNode(parser, "onprocessinginstruction", { + name: parser.procInstName, + body: parser.procInstBody, + }); + parser.procInstName = parser.procInstBody = ""; + parser.state = S.TEXT; + } else { + parser.procInstBody += "?" + c; + parser.state = S.PROC_INST_BODY; + } + continue; + + case S.OPEN_TAG: + if (isMatch(nameBody, c)) { + parser.tagName += c; + } else { + newTag(parser); + if (c === ">") { + openTag(parser); + } else if (c === "/") { + parser.state = S.OPEN_TAG_SLASH; + } else { + if (!isWhitespace(c)) { + strictFail(parser, "Invalid character in tag name"); + } + parser.state = S.ATTRIB; + } + } + continue; + + case S.OPEN_TAG_SLASH: + if (c === ">") { + openTag(parser, true); + closeTag(parser); + } else { + strictFail( + parser, + "Forward-slash in opening tag not followed by >" + ); + parser.state = S.ATTRIB; + } + continue; + + case S.ATTRIB: + // haven't read the attribute name yet. + if (isWhitespace(c)) { + continue; + } else if (c === ">") { + openTag(parser); + } else if (c === "/") { + parser.state = S.OPEN_TAG_SLASH; + } else if (isMatch(nameStart, c)) { + parser.attribName = c; + parser.attribValue = ""; + parser.state = S.ATTRIB_NAME; + } else { + strictFail(parser, "Invalid attribute name"); + } + continue; + + case S.ATTRIB_NAME: + if (c === "=") { + parser.state = S.ATTRIB_VALUE; + } else if (c === ">") { + strictFail(parser, "Attribute without value"); + parser.attribValue = parser.attribName; + attrib(parser); + openTag(parser); + } else if (isWhitespace(c)) { + parser.state = S.ATTRIB_NAME_SAW_WHITE; + } else if (isMatch(nameBody, c)) { + parser.attribName += c; + } else { + strictFail(parser, "Invalid attribute name"); + } + continue; + + case S.ATTRIB_NAME_SAW_WHITE: + if (c === "=") { + parser.state = S.ATTRIB_VALUE; + } else if (isWhitespace(c)) { + continue; + } else { + strictFail(parser, "Attribute without value"); + parser.tag.attributes[parser.attribName] = ""; + parser.attribValue = ""; + emitNode(parser, "onattribute", { + name: parser.attribName, + value: "", + }); + parser.attribName = ""; + if (c === ">") { + openTag(parser); + } else if (isMatch(nameStart, c)) { + parser.attribName = c; + parser.state = S.ATTRIB_NAME; + } else { + strictFail(parser, "Invalid attribute name"); + parser.state = S.ATTRIB; + } + } + continue; + + case S.ATTRIB_VALUE: + if (isWhitespace(c)) { + continue; + } else if (isQuote(c)) { + parser.q = c; + parser.state = S.ATTRIB_VALUE_QUOTED; + } else { + strictFail(parser, "Unquoted attribute value"); + parser.state = S.ATTRIB_VALUE_UNQUOTED; + parser.attribValue = c; + } + continue; + + case S.ATTRIB_VALUE_QUOTED: + if (c !== parser.q) { + if (c === "&") { + parser.state = S.ATTRIB_VALUE_ENTITY_Q; + } else { + parser.attribValue += c; + } + continue; + } + attrib(parser); + parser.q = ""; + parser.state = S.ATTRIB_VALUE_CLOSED; + continue; + + case S.ATTRIB_VALUE_CLOSED: + if (isWhitespace(c)) { + parser.state = S.ATTRIB; + } else if (c === ">") { + openTag(parser); + } else if (c === "/") { + parser.state = S.OPEN_TAG_SLASH; + } else if (isMatch(nameStart, c)) { + strictFail(parser, "No whitespace between attributes"); + parser.attribName = c; + parser.attribValue = ""; + parser.state = S.ATTRIB_NAME; + } else { + strictFail(parser, "Invalid attribute name"); + } + continue; + + case S.ATTRIB_VALUE_UNQUOTED: + if (!isAttribEnd(c)) { + if (c === "&") { + parser.state = S.ATTRIB_VALUE_ENTITY_U; + } else { + parser.attribValue += c; + } + continue; + } + attrib(parser); + if (c === ">") { + openTag(parser); + } else { + parser.state = S.ATTRIB; + } + continue; + + case S.CLOSE_TAG: + if (!parser.tagName) { + if (isWhitespace(c)) { + continue; + } else if (notMatch(nameStart, c)) { + if (parser.script) { + parser.script += "</" + c; + parser.state = S.SCRIPT; + } else { + strictFail(parser, "Invalid tagname in closing tag."); + } + } else { + parser.tagName = c; + } + } else if (c === ">") { + closeTag(parser); + } else if (isMatch(nameBody, c)) { + parser.tagName += c; + } else if (parser.script) { + parser.script += "</" + parser.tagName; + parser.tagName = ""; + parser.state = S.SCRIPT; + } else { + if (!isWhitespace(c)) { + strictFail(parser, "Invalid tagname in closing tag"); + } + parser.state = S.CLOSE_TAG_SAW_WHITE; + } + continue; + + case S.CLOSE_TAG_SAW_WHITE: + if (isWhitespace(c)) { + continue; + } + if (c === ">") { + closeTag(parser); + } else { + strictFail(parser, "Invalid characters in closing tag"); + } + continue; + + case S.TEXT_ENTITY: + case S.ATTRIB_VALUE_ENTITY_Q: + case S.ATTRIB_VALUE_ENTITY_U: + var returnState; + var buffer; + switch (parser.state) { + case S.TEXT_ENTITY: + returnState = S.TEXT; + buffer = "textNode"; + break; + + case S.ATTRIB_VALUE_ENTITY_Q: + returnState = S.ATTRIB_VALUE_QUOTED; + buffer = "attribValue"; + break; + + case S.ATTRIB_VALUE_ENTITY_U: + returnState = S.ATTRIB_VALUE_UNQUOTED; + buffer = "attribValue"; + break; + } + + if (c === ";") { + parser[buffer] += parseEntity(parser); + parser.entity = ""; + parser.state = returnState; + } else if ( + isMatch(parser.entity.length ? entityBody : entityStart, c) + ) { + parser.entity += c; + } else { + strictFail(parser, "Invalid character in entity name"); + parser[buffer] += "&" + parser.entity + c; + parser.entity = ""; + parser.state = returnState; + } + + continue; + + default: + throw new Error(parser, "Unknown state: " + parser.state); + } + } // while + + if (parser.position >= parser.bufferCheckPosition) { + checkBufferLength(parser); + } + return parser; + } + + /*! http://mths.be/fromcodepoint v0.1.0 by @mathias */ + /* istanbul ignore next */ + if (!String.fromCodePoint) { + (function() { + var stringFromCharCode = String.fromCharCode; + var floor = Math.floor; + var fromCodePoint = function() { + var MAX_SIZE = 0x4000; + var codeUnits = []; + var highSurrogate; + var lowSurrogate; + var index = -1; + var length = arguments.length; + if (!length) { + return ""; + } + var result = ""; + while (++index < length) { + var codePoint = Number(arguments[index]); + if ( + !isFinite(codePoint) || // `NaN`, `+Infinity`, or `-Infinity` + codePoint < 0 || // not a valid Unicode code point + codePoint > 0x10ffff || // not a valid Unicode code point + floor(codePoint) !== codePoint // not an integer + ) { + throw RangeError("Invalid code point: " + codePoint); + } + if (codePoint <= 0xffff) { + // BMP code point + codeUnits.push(codePoint); + } else { + // Astral code point; split in surrogate halves + // http://mathiasbynens.be/notes/javascript-encoding#surrogate-formulae + codePoint -= 0x10000; + highSurrogate = (codePoint >> 10) + 0xd800; + lowSurrogate = (codePoint % 0x400) + 0xdc00; + codeUnits.push(highSurrogate, lowSurrogate); + } + if (index + 1 === length || codeUnits.length > MAX_SIZE) { + result += stringFromCharCode.apply(null, codeUnits); + codeUnits.length = 0; + } + } + return result; + }; + /* istanbul ignore next */ + if (Object.defineProperty) { + Object.defineProperty(String, "fromCodePoint", { + value: fromCodePoint, + configurable: true, + writable: true, + }); + } else { + String.fromCodePoint = fromCodePoint; + } + })(); + } +})(typeof exports === "undefined" ? (this.sax = {}) : exports); diff --git a/comm/chat/protocols/xmpp/moz.build b/comm/chat/protocols/xmpp/moz.build new file mode 100644 index 0000000000..56140aeef6 --- /dev/null +++ b/comm/chat/protocols/xmpp/moz.build @@ -0,0 +1,26 @@ +# vim: set filetype=python: +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. + +XPCSHELL_TESTS_MANIFESTS += ["test/xpcshell.ini"] + +DIRS += [ + "lib", +] + +EXTRA_JS_MODULES += [ + "sax.sys.mjs", + "xmpp-authmechs.sys.mjs", + "xmpp-base.sys.mjs", + "xmpp-commands.sys.mjs", + "xmpp-session.sys.mjs", + "xmpp-xml.sys.mjs", + "xmpp.sys.mjs", +] + +JAR_MANIFESTS += ["jar.mn"] + +XPCOM_MANIFESTS += [ + "components.conf", +] diff --git a/comm/chat/protocols/xmpp/sax.sys.mjs b/comm/chat/protocols/xmpp/sax.sys.mjs new file mode 100644 index 0000000000..3098c6df79 --- /dev/null +++ b/comm/chat/protocols/xmpp/sax.sys.mjs @@ -0,0 +1,7 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +let scope = {}; +Services.scriptloader.loadSubScript("resource:///modules/sax/sax.js", scope); +export var SAX = scope.sax; diff --git a/comm/chat/protocols/xmpp/test/test_authmechs.js b/comm/chat/protocols/xmpp/test/test_authmechs.js new file mode 100644 index 0000000000..f935026dbc --- /dev/null +++ b/comm/chat/protocols/xmpp/test/test_authmechs.js @@ -0,0 +1,160 @@ +/* Any copyright is dedicated to the Public Domain. + * http://creativecommons.org/publicdomain/zero/1.0/ */ + +var { XMPPAuthMechanisms } = ChromeUtils.importESModule( + "resource:///modules/xmpp-authmechs.sys.mjs" +); +var { Stanza } = ChromeUtils.importESModule( + "resource:///modules/xmpp-xml.sys.mjs" +); + +/* + * Test PLAIN using the examples given in section 6 of RFC 6120. + */ +add_task(async function testPlain() { + const username = "juliet"; + const password = "r0m30myr0m30"; + + let mech = XMPPAuthMechanisms.PLAIN(username, password, undefined); + + // Send the initiation message. + let result = mech.next(); + ok(!result.done); + let value = await Promise.resolve(result.value); + + // Check the algorithm. + equal(value.send.attributes.mechanism, "PLAIN"); + // Check the PLAIN content. + equal(value.send.children[0].text, "AGp1bGlldAByMG0zMG15cjBtMzA="); + + // Receive the success. + let response = Stanza.node("success", Stanza.NS.sasl); + result = mech.next(response); + ok(result.done); + // There is no final value. + equal(result.value, undefined); +}); + +/* + * Test SCRAM-SHA-1 using the examples given in section 5 of RFC 5802. + * + * Full test vectors of intermediate values are available at: + * https://wiki.xmpp.org/web/SASL_and_SCRAM-SHA-1 + */ +add_task(async function testScramSha1() { + const username = "user"; + const password = "pencil"; + + // Use a constant value for the nonce. + const nonce = "fyko+d2lbbFgONRv9qkxdawL"; + + let mech = XMPPAuthMechanisms["SCRAM-SHA-1"]( + username, + password, + undefined, + nonce + ); + + // Send the client-first-message. + let result = mech.next(); + ok(!result.done); + let value = await Promise.resolve(result.value); + + // Check the algorithm. + equal(value.send.attributes.mechanism, "SCRAM-SHA-1"); + // Check the SCRAM content. + equal( + atob(value.send.children[0].text), + "n,,n=user,r=fyko+d2lbbFgONRv9qkxdawL" + ); + + // Receive the server-first-message and send the client-final-message. + let response = Stanza.node( + "challenge", + Stanza.NS.sasl, + null, + btoa( + "r=fyko+d2lbbFgONRv9qkxdawL3rfcNHYJY1ZVvWVs7j,s=QSXCR+Q6sek8bf92,i=4096" + ) + ); + result = mech.next(response); + ok(!result.done); + value = await Promise.resolve(result.value); + + // Check the SCRAM content. + equal( + atob(value.send.children[0].text), + "c=biws,r=fyko+d2lbbFgONRv9qkxdawL3rfcNHYJY1ZVvWVs7j,p=v0X8v3Bz2T0CJGbJQyF0X+HI4Ts=" + ); + + // Receive the server-final-message. + response = Stanza.node( + "success", + Stanza.NS.sasl, + null, + btoa("v=rmF9pqV8S7suAoZWja4dJRkFsKQ=") + ); + result = mech.next(response); + ok(result.done); + // There is no final value. + equal(result.value, undefined); +}); + +/* + * Test SCRAM-SHA-256 using the examples given in section 3 of RFC 7677. + */ +add_task(async function testScramSha256() { + const username = "user"; + const password = "pencil"; + + // Use a constant value for the nonce. + const nonce = "rOprNGfwEbeRWgbNEkqO"; + + let mech = XMPPAuthMechanisms["SCRAM-SHA-256"]( + username, + password, + undefined, + nonce + ); + + // Send the client-first-message. + let result = mech.next(); + ok(!result.done); + let value = await Promise.resolve(result.value); + + // Check the algorithm. + equal(value.send.attributes.mechanism, "SCRAM-SHA-256"); + // Check the SCRAM content. + equal(atob(value.send.children[0].text), "n,,n=user,r=rOprNGfwEbeRWgbNEkqO"); + + // Receive the server-first-message and send the client-final-message. + let response = Stanza.node( + "challenge", + Stanza.NS.sasl, + null, + btoa( + "r=rOprNGfwEbeRWgbNEkqO%hvYDpWUa2RaTCAfuxFIlj)hNlF$k0,s=W22ZaJ0SNY7soEsUEjb6gQ==,i=4096" + ) + ); + result = mech.next(response); + ok(!result.done); + value = await Promise.resolve(result.value); + + // Check the SCRAM content. + equal( + atob(value.send.children[0].text), + "c=biws,r=rOprNGfwEbeRWgbNEkqO%hvYDpWUa2RaTCAfuxFIlj)hNlF$k0,p=dHzbZapWIk4jUhN+Ute9ytag9zjfMHgsqmmiz7AndVQ=" + ); + + // Receive the server-final-message. + response = Stanza.node( + "success", + Stanza.NS.sasl, + null, + btoa("v=6rriTRBi23WpRR/wtup+mMhUZUn/dB5nLTJRsjl95G4=") + ); + result = mech.next(response); + ok(result.done); + // There is no final value. + equal(result.value, undefined); +}); diff --git a/comm/chat/protocols/xmpp/test/test_dnsSrv.js b/comm/chat/protocols/xmpp/test/test_dnsSrv.js new file mode 100644 index 0000000000..37f1b6b052 --- /dev/null +++ b/comm/chat/protocols/xmpp/test/test_dnsSrv.js @@ -0,0 +1,112 @@ +/* Any copyright is dedicated to the Public Domain. + * http://creativecommons.org/publicdomain/zero/1.0/ */ + +var { XMPPAccountPrototype } = ChromeUtils.importESModule( + "resource:///modules/xmpp-base.sys.mjs" +); +var { XMPPSession } = ChromeUtils.importESModule( + "resource:///modules/xmpp-session.sys.mjs" +); +var { SRVRecord } = ChromeUtils.import("resource:///modules/DNS.jsm"); + +function FakeXMPPSession() {} +FakeXMPPSession.prototype = { + __proto__: XMPPSession.prototype, + _account: { __proto__: XMPPAccountPrototype }, + _host: null, + _port: 0, + connect( + aHostOrigin, + aPortOrigin, + aSecurity, + aProxy, + aHost = aHostOrigin, + aPort = aPortOrigin + ) {}, + _connectNextRecord() { + this.isConnectNextRecord = true; + }, + + // Used to indicate that method _connectNextRecord is called or not. + isConnectNextRecord: false, + + LOG(aMsg) {}, + WARN(aMsg) {}, +}; + +var TEST_DATA = [ + { + // Test sorting based on priority and weight. + input: [ + new SRVRecord(20, 0, "xmpp.instantbird.com", 5222), + new SRVRecord(5, 0, "xmpp1.instantbird.com", 5222), + new SRVRecord(10, 0, "xmpp2.instantbird.com", 5222), + new SRVRecord(0, 0, "xmpp3.instantbird.com", 5222), + new SRVRecord(15, 0, "xmpp4.instantbird.com", 5222), + ], + output: [ + new SRVRecord(0, 0, "xmpp3.instantbird.com", 5222), + new SRVRecord(5, 0, "xmpp1.instantbird.com", 5222), + new SRVRecord(10, 0, "xmpp2.instantbird.com", 5222), + new SRVRecord(15, 0, "xmpp4.instantbird.com", 5222), + new SRVRecord(20, 0, "xmpp.instantbird.com", 5222), + ], + isConnectNextRecord: true, + }, + { + input: [ + new SRVRecord(5, 30, "xmpp5.instantbird.com", 5222), + new SRVRecord(5, 0, "xmpp1.instantbird.com", 5222), + new SRVRecord(10, 60, "xmpp2.instantbird.com", 5222), + new SRVRecord(5, 10, "xmpp3.instantbird.com", 5222), + new SRVRecord(20, 10, "xmpp.instantbird.com", 5222), + new SRVRecord(15, 0, "xmpp4.instantbird.com", 5222), + ], + output: [ + new SRVRecord(5, 30, "xmpp5.instantbird.com", 5222), + new SRVRecord(5, 10, "xmpp3.instantbird.com", 5222), + new SRVRecord(5, 0, "xmpp1.instantbird.com", 5222), + new SRVRecord(10, 60, "xmpp2.instantbird.com", 5222), + new SRVRecord(15, 0, "xmpp4.instantbird.com", 5222), + new SRVRecord(20, 10, "xmpp.instantbird.com", 5222), + ], + isConnectNextRecord: true, + }, + + // Tests no SRV records are found. + { + input: [], + output: [], + isConnectNextRecord: false, + }, + + // Tests XMPP is not supported if the result is one record with target ".". + { + input: [new SRVRecord(5, 30, ".", 5222)], + output: XMPPSession.prototype.SRV_ERROR_XMPP_NOT_SUPPORTED, + isConnectNextRecord: false, + }, + { + input: [new SRVRecord(5, 30, "xmpp.instantbird.com", 5222)], + output: [new SRVRecord(5, 30, "xmpp.instantbird.com", 5222)], + isConnectNextRecord: true, + }, +]; + +function run_test() { + for (let currentQuery of TEST_DATA) { + let session = new FakeXMPPSession(); + try { + session._handleSrvQuery(currentQuery.input); + equal(session._srvRecords.length, currentQuery.output.length); + for (let index = 0; index < session._srvRecords.length; index++) { + deepEqual(session._srvRecords[index], currentQuery.output[index]); + } + } catch (e) { + equal(e, currentQuery.output); + } + equal(session.isConnectNextRecord, currentQuery.isConnectNextRecord); + } + + run_next_test(); +} diff --git a/comm/chat/protocols/xmpp/test/test_parseJidAndNormalization.js b/comm/chat/protocols/xmpp/test/test_parseJidAndNormalization.js new file mode 100644 index 0000000000..f041d2356b --- /dev/null +++ b/comm/chat/protocols/xmpp/test/test_parseJidAndNormalization.js @@ -0,0 +1,104 @@ +/* Any copyright is dedicated to the Public Domain. + * http://creativecommons.org/publicdomain/zero/1.0/ */ + +var { XMPPAccountPrototype } = ChromeUtils.importESModule( + "resource:///modules/xmpp-base.sys.mjs" +); + +var TEST_DATA = { + "abdelrhman@instantbird": { + node: "abdelrhman", + domain: "instantbird", + jid: "abdelrhman@instantbird", + normalized: "abdelrhman@instantbird", + }, + " room@instantbird/abdelrhman ": { + node: "room", + domain: "instantbird", + resource: "abdelrhman", + jid: "room@instantbird/abdelrhman", + normalized: "room@instantbird", + }, + "room@instantbird/@bdelrhman": { + node: "room", + domain: "instantbird", + resource: "@bdelrhman", + jid: "room@instantbird/@bdelrhman", + normalized: "room@instantbird", + }, + "room@instantbird/abdelrhm\u0061\u0308n": { + node: "room", + domain: "instantbird", + resource: "abdelrhm\u0061\u0308n", + jid: "room@instantbird/abdelrhm\u0061\u0308n", + normalized: "room@instantbird", + }, + "Room@Instantbird/Abdelrhman": { + node: "room", + domain: "instantbird", + resource: "Abdelrhman", + jid: "room@instantbird/Abdelrhman", + normalized: "room@instantbird", + }, + "Abdelrhman@instantbird/Instant bird": { + node: "abdelrhman", + domain: "instantbird", + resource: "Instant bird", + jid: "abdelrhman@instantbird/Instant bird", + normalized: "abdelrhman@instantbird", + }, + "abdelrhman@host/instant/Bird": { + node: "abdelrhman", + domain: "host", + resource: "instant/Bird", + jid: "abdelrhman@host/instant/Bird", + normalized: "abdelrhman@host", + }, + instantbird: { + domain: "instantbird", + jid: "instantbird", + normalized: "instantbird", + }, +}; + +function testParseJID() { + for (let currentJID in TEST_DATA) { + let jid = XMPPAccountPrototype._parseJID(currentJID); + equal(jid.node, TEST_DATA[currentJID].node); + equal(jid.domain, TEST_DATA[currentJID].domain); + equal(jid.resource, TEST_DATA[currentJID].resource); + equal(jid.jid, TEST_DATA[currentJID].jid); + } + + run_next_test(); +} + +function testNormalize() { + for (let currentJID in TEST_DATA) { + equal( + XMPPAccountPrototype.normalize(currentJID), + TEST_DATA[currentJID].normalized + ); + } + + run_next_test(); +} + +function testNormalizeFullJid() { + for (let currentJID in TEST_DATA) { + equal( + XMPPAccountPrototype.normalizeFullJid(currentJID), + TEST_DATA[currentJID].jid + ); + } + + run_next_test(); +} + +function run_test() { + add_test(testParseJID); + add_test(testNormalize); + add_test(testNormalizeFullJid); + + run_next_test(); +} diff --git a/comm/chat/protocols/xmpp/test/test_parseVCard.js b/comm/chat/protocols/xmpp/test/test_parseVCard.js new file mode 100644 index 0000000000..08155218de --- /dev/null +++ b/comm/chat/protocols/xmpp/test/test_parseVCard.js @@ -0,0 +1,139 @@ +/* Any copyright is dedicated to the Public Domain. + * http://creativecommons.org/publicdomain/zero/1.0/ */ + +var { XMPPAccountPrototype } = ChromeUtils.importESModule( + "resource:///modules/xmpp-base.sys.mjs" +); +var { XMPPParser } = ChromeUtils.importESModule( + "resource:///modules/xmpp-xml.sys.mjs" +); + +/* + * Open an input stream, instantiate an XMPP parser, and feed the input string + * into it. Then assert that the resulting vCard matches the expected result. + */ +function _test_vcard(aInput, aExpectedResult) { + let listener = { + onXMLError(aError, aException) { + // Ensure that no errors happen. + ok(false, aError + " - " + aException); + }, + LOG(aString) {}, + onXmppStanza(aStanza) { + // This is a simplified stanza parser that assumes inputs are vCards. + let vCard = aStanza.getElement(["vCard"]); + deepEqual(XMPPAccountPrototype.parseVCard(vCard), aExpectedResult); + }, + }; + let parser = new XMPPParser(listener); + parser.onDataAvailable(aInput); + parser.destroy(); +} + +/* + * Test parsing of the example vCard from XEP-0054 section 3.1, example 2. + */ +function test_standard_vcard() { + const standard_vcard = + "<iq xmlns='jabber:client'\ + id='v1'\ + to='stpeter@jabber.org/roundabout'\ + type='result'>\ + <vCard xmlns='vcard-temp'>\ + <FN>Peter Saint-Andre</FN>\ + <N>\ + <FAMILY>Saint-Andre</FAMILY>\ + <GIVEN>Peter</GIVEN>\ + <MIDDLE/>\ + </N>\ + <NICKNAME>stpeter</NICKNAME>\ + <URL>http://www.xmpp.org/xsf/people/stpeter.shtml</URL>\ + <BDAY>1966-08-06</BDAY>\ + <ORG>\ + <ORGNAME>XMPP Standards Foundation</ORGNAME>\ + <ORGUNIT/>\ + </ORG>\ + <TITLE>Executive Director</TITLE>\ + <ROLE>Patron Saint</ROLE>\ + <TEL><WORK/><VOICE/><NUMBER>303-308-3282</NUMBER></TEL>\ + <TEL><WORK/><FAX/><NUMBER/></TEL>\ + <TEL><WORK/><MSG/><NUMBER/></TEL>\ + <ADR>\ + <WORK/>\ + <EXTADD>Suite 600</EXTADD>\ + <STREET>1899 Wynkoop Street</STREET>\ + <LOCALITY>Denver</LOCALITY>\ + <REGION>CO</REGION>\ + <PCODE>80202</PCODE>\ + <CTRY>USA</CTRY>\ + </ADR>\ + <TEL><HOME/><VOICE/><NUMBER>303-555-1212</NUMBER></TEL>\ + <TEL><HOME/><FAX/><NUMBER/></TEL>\ + <TEL><HOME/><MSG/><NUMBER/></TEL>\ + <ADR>\ + <HOME/>\ + <EXTADD/>\ + <STREET/>\ + <LOCALITY>Denver</LOCALITY>\ + <REGION>CO</REGION>\ + <PCODE>80209</PCODE>\ + <CTRY>USA</CTRY>\ + </ADR>\ + <EMAIL><INTERNET/><PREF/><USERID>stpeter@jabber.org</USERID></EMAIL>\ + <JABBERID>stpeter@jabber.org</JABBERID>\ + <DESC>\ + More information about me is located on my\ + personal website: http://www.saint-andre.com/\ + </DESC>\ + </vCard>\ +</iq>"; + + const expectedResult = { + fullName: "Peter Saint-Andre", + // Name is not parsed. + nickname: "stpeter", + // URL is not parsed. + birthday: "1966-08-06", + organization: "XMPP Standards Foundation", + title: "Executive Director", + // Role is not parsed. + // This only pulls the *last* telephone number. + telephone: "303-555-1212", + // Part of the address is parsed. + locality: "Denver", + country: "USA", + email: "stpeter@jabber.org", + userName: "stpeter@jabber.org", // Jabber ID. + // Description is not parsed. + }; + + _test_vcard(standard_vcard, expectedResult); + + run_next_test(); +} + +/* + * Test parsing of the example empty vCard from XEP-0054 section 3.1, example + * 4. This can be used instead of returning an error stanza. + */ +function test_empty_vcard() { + const empty_vcard = + "<iq xmlns='jabber:client'\ + id='v1'\ + to='stpeter@jabber.org/roundabout'\ + type='result'>\ + <vCard xmlns='vcard-temp'/>\ +</iq>"; + + // There should be no properties. + _test_vcard(empty_vcard, {}); + + run_next_test(); +} + +function run_test() { + add_test(test_standard_vcard); + add_test(test_empty_vcard); + + run_next_test(); +} diff --git a/comm/chat/protocols/xmpp/test/test_saslPrep.js b/comm/chat/protocols/xmpp/test/test_saslPrep.js new file mode 100644 index 0000000000..b2d0a1f147 --- /dev/null +++ b/comm/chat/protocols/xmpp/test/test_saslPrep.js @@ -0,0 +1,66 @@ +/* Any copyright is dedicated to the Public Domain. + * http://creativecommons.org/publicdomain/zero/1.0/ */ + +var { saslPrep } = ChromeUtils.importESModule( + "resource:///modules/xmpp-authmechs.sys.mjs" +); + +// RFC 4013 3.Examples +var TEST_DATA = [ + { + // SOFT HYPHEN mapped to nothing. + input: "I\u00adX", + output: "IX", + isError: false, + }, + { + // No transformation. + input: "user", + output: "user", + isError: false, + }, + { + // Case preserved, will not match #2. + input: "USER", + output: "USER", + isError: false, + }, + { + // Output is NFKC, input in ISO 8859-1. + input: "\u00aa", + output: "a", + isError: false, + }, + { + // Output is NFKC, will match #1. + input: "\u2168", + output: "IX", + isError: false, + }, + { + // Error - prohibited character. + input: "\u0007", + output: "", + isError: true, + }, + { + // Error - bidirectional check. + input: "\u0627\u0031", + output: "", + isError: true, + }, +]; + +function run_test() { + for (let current of TEST_DATA) { + try { + let result = saslPrep(current.input); + equal(current.isError, false); + equal(result, current.output); + } catch (e) { + equal(current.isError, true); + } + } + + run_next_test(); +} diff --git a/comm/chat/protocols/xmpp/test/test_xmppParser.js b/comm/chat/protocols/xmpp/test/test_xmppParser.js new file mode 100644 index 0000000000..c18304a544 --- /dev/null +++ b/comm/chat/protocols/xmpp/test/test_xmppParser.js @@ -0,0 +1,135 @@ +/* Any copyright is dedicated to the Public Domain. + * http://creativecommons.org/publicdomain/zero/1.0/ */ + +var { XMPPParser } = ChromeUtils.importESModule( + "resource:///modules/xmpp-xml.sys.mjs" +); + +let expectedResult = + '<presence xmlns="jabber:client" from="chat@example.com/Étienne" to="user@example.com/Thunderbird" \ +xml:lang="en" id="5ed0ae8b7051fa6169037da4e2a1ded6"><c xmlns="http://jabber.org/protocol/caps" \ +ver="ZyB1liM9c9GvKOnvl61+5ScWcqw=" node="https://example.com" hash="sha-1"/><x \ +xmlns="vcard-temp:x:update"><photo xmlns="vcard-temp:x:update"/></x><idle xmlns="urn:xmpp:idle:1" \ +since="2021-04-13T11:52:16.538713+00:00"/><occupant-id xmlns="urn:xmpp:occupant-id:0" \ +id="wNZPCZIVQ51D/heZQpOHi0ZgHXAEQonNPaLdyzLxHWs="/><x xmlns="http://jabber.org/protocol/muc#user"><item \ +xmlns="http://jabber.org/protocol/muc#user" jid="example@example.com/client" affiliation="member" \ +role="participant"/></x></presence>'; +let byteVersion = new TextEncoder().encode(expectedResult); +let utf8Input = Array.from(byteVersion, byte => String.fromCharCode(byte)).join( + "" +); + +var TEST_DATA = [ + { + input: + '<message xmlns="jabber:client" from="juliet@capulet.example/balcony" \ +to="romeo@montague.example/garden" type="chat">\ +<body>What man art thou that, thus bescreen"d in night, so stumblest on my \ +counsel?</body>\ +</message>', + output: + '<message xmlns="jabber:client" \ +from="juliet@capulet.example/balcony" to="romeo@montague.example/garden" \ +type="chat"><body xmlns="jabber:client">What man art thou that, thus \ +bescreen"d in night, so stumblest on my counsel?</body>\ +</message>', + isError: false, + description: "Message stanza with body element", + }, + { + input: + '<message xmlns="jabber:client" from="romeo@montague.example" \ +to="romeo@montague.example/home" type="chat">\ +<received xmlns="urn:xmpp:carbons:2">\ +<forwarded xmlns="urn:xmpp:forward:0">\ +<message xmlns="jabber:client" from="juliet@capulet.example/balcony" \ +to="romeo@montague.example/garden" type="chat">\ +<body>What man art thou that, thus bescreen"d in night, so stumblest on my \ +counsel?</body>\ +<thread>0e3141cd80894871a68e6fe6b1ec56fa</thread>\ +</message>\ +</forwarded>\ +</received>\ +</message>', + output: + '<message xmlns="jabber:client" from="romeo@montague.example" \ +to="romeo@montague.example/home" type="chat">\ +<received xmlns="urn:xmpp:carbons:2"><forwarded xmlns="urn:xmpp:forward:0">\ +<message xmlns="jabber:client" from="juliet@capulet.example/balcony" \ +to="romeo@montague.example/garden" type="chat">\ +<body xmlns="jabber:client">What man art thou that, thus bescreen"d in night, \ +so stumblest on my counsel?</body>\ +<thread xmlns="jabber:client">0e3141cd80894871a68e6fe6b1ec56fa</thread>\ +</message>\ +</forwarded>\ +</received>\ +</message>', + isError: false, + description: "Forwarded copy of message carbons", + }, + { + input: + '<message xmlns="jabber:client" from="juliet@capulet.example/balcony" \ +to="romeo@montague.example/garden" type="chat">\ +<body>What man art thou that, thus bescreen"d in night, so stumblest on my \ +counsel?\ +</message>', + output: "", + isError: true, + description: "No closing of body tag", + }, + { + input: + '<message xmlns="http://etherx.jabber.org/streams" from="juliet@capulet.example/balcony" \ +to="romeo@montague.example/garden" type="chat">\ +<body>What man art thou that, thus bescreen"d in night, so stumblest on my \ +counsel?</body>\ +</message>', + output: "", + isError: true, + description: "Invalid namespace of top-level element", + }, + { + input: + '<field xmlns="jabber:x:data" type="fixed">\ +<value>What man art thou that, thus bescreen"d in night, so stumblest on my \ +counsel?</value>\ +</field>', + output: "", + isError: true, + description: "Invalid top-level element", + }, + { + input: utf8Input, + output: expectedResult, + isError: false, + description: "UTF-8 encoded content from socket", + }, +]; + +function testXMPPParser() { + for (let current of TEST_DATA) { + let listener = { + onXMLError(aString) { + ok(current.isError, aString + " - " + current.description); + }, + LOG(aString) {}, + startLegacyAuth() {}, + onXmppStanza(aStanza) { + equal(current.output, aStanza.getXML(), current.description); + ok(!current.isError, current.description); + }, + }; + let parser = new XMPPParser(listener); + parser.onDataAvailable(current.input); + parser.destroy(); + } + + run_next_test(); +} + +function run_test() { + add_test(testXMPPParser); + + run_next_test(); +} diff --git a/comm/chat/protocols/xmpp/test/test_xmppXml.js b/comm/chat/protocols/xmpp/test/test_xmppXml.js new file mode 100644 index 0000000000..1b6ea9a175 --- /dev/null +++ b/comm/chat/protocols/xmpp/test/test_xmppXml.js @@ -0,0 +1,103 @@ +/* Any copyright is dedicated to the Public Domain. + * http://creativecommons.org/publicdomain/zero/1.0/ */ + +var { Stanza } = ChromeUtils.importESModule( + "resource:///modules/xmpp-xml.sys.mjs" +); + +var TEST_DATA = [ + { + input: { + name: "message", + namespace: Stanza.NS.client, + attributes: { + jid: "user@domain", + type: null, + }, + data: [], + }, + XmlOutput: '<message xmlns="jabber:client" jid="user@domain"/>', + stringOutput: '<message xmlns="jabber:client" jid="user@domain"/>\n', + isError: false, + description: "Ignore attribute with null value", + }, + { + input: { + name: "message", + namespace: Stanza.NS.client, + attributes: { + jid: "user@domain", + type: undefined, + }, + data: [], + }, + XmlOutput: '<message xmlns="jabber:client" jid="user@domain"/>', + stringOutput: '<message xmlns="jabber:client" jid="user@domain"/>\n', + isError: false, + description: "Ignore attribute with undefined value", + }, + { + input: { + name: "message", + namespace: undefined, + attributes: {}, + data: [], + }, + XmlOutput: "<message/>", + stringOutput: "<message/>\n", + isError: false, + description: "Ignore namespace with undefined value", + }, + { + input: { + name: undefined, + attributes: {}, + data: [], + }, + XmlOutput: "", + stringOutput: "", + isError: true, + description: "Node must have a name", + }, + { + input: { + name: "message", + attributes: {}, + data: "test message", + }, + XmlOutput: "<message>test message</message>", + stringOutput: "<message>\n test message\n</message>\n", + isError: false, + description: "Node with text content", + }, +]; + +function testXMLNode() { + for (let current of TEST_DATA) { + try { + let result = Stanza.node( + current.input.name, + current.input.namespace, + current.input.attributes, + current.input.data + ); + equal(result.getXML(), current.XmlOutput, current.description); + equal( + result.convertToString(), + current.stringOutput, + current.description + ); + equal(current.isError, false); + } catch (e) { + equal(current.isError, true, current.description); + } + } + + run_next_test(); +} + +function run_test() { + add_test(testXMLNode); + + run_next_test(); +} diff --git a/comm/chat/protocols/xmpp/test/xpcshell.ini b/comm/chat/protocols/xmpp/test/xpcshell.ini new file mode 100644 index 0000000000..a4cb9534b8 --- /dev/null +++ b/comm/chat/protocols/xmpp/test/xpcshell.ini @@ -0,0 +1,11 @@ +[DEFAULT] +head = +tail = + +[test_authmechs.js] +[test_dnsSrv.js] +[test_parseJidAndNormalization.js] +[test_saslPrep.js] +[test_parseVCard.js] +[test_xmppParser.js] +[test_xmppXml.js] diff --git a/comm/chat/protocols/xmpp/xmpp-authmechs.sys.mjs b/comm/chat/protocols/xmpp/xmpp-authmechs.sys.mjs new file mode 100644 index 0000000000..7517d6f6aa --- /dev/null +++ b/comm/chat/protocols/xmpp/xmpp-authmechs.sys.mjs @@ -0,0 +1,561 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +// This module exports XMPPAuthMechanisms, an object containing the supported +// SASL authentication mechanisms. Each authentication mechanism is a generator +// function which takes the following parameters: +// +// * The provided username (JID node), +// * The password +// * The user's domain (again from the JID). +// +// The generator should yield objects (or Promises which resolve to objects) +// with two properties: +// +// * send: The next XML stanza to send. +// * log: The plaintext content to log (instead of the stanza, which likely +// contains sensitive information). +// +// Alternately the object can have an error property which causes the account +// to disconnect with an ERROR_AUTHENTICATION_FAILED error. +// +// The response stanza from the server is sent to the generator each time it +// yields. Once the authentication negotiation is complete the generator should +// return. +// +// By default the PLAIN, SCRAM-SHA-1, and SCRAM-SHA-256 mechanisms are supported. +// +// As this is only used by XMPPSession, it may seem like an internal detail of +// the XMPP implementation, but exporting it is valuable for testing purposes. + +import { CommonUtils } from "resource://services-common/utils.sys.mjs"; +import { CryptoUtils } from "resource://services-crypto/utils.sys.mjs"; +import { Stanza } from "resource:///modules/xmpp-xml.sys.mjs"; + +// Handle PLAIN authorization mechanism. +function* PlainAuth(aUsername, aPassword, aDomain) { + let data = "\0" + aUsername + "\0" + aPassword; + + // btoa for Unicode, see https://developer.mozilla.org/en-US/docs/DOM/window.btoa + let base64Data = btoa(unescape(encodeURIComponent(data))); + + let stanza = yield { + send: Stanza.node( + "auth", + Stanza.NS.sasl, + { mechanism: "PLAIN" }, + base64Data + ), + log: '<auth mechanism:="PLAIN"/> (base64 encoded username and password not logged)', + }; + + if (stanza.localName != "success") { + throw new Error("Didn't receive the expected auth success stanza."); + } +} + +// Handle SCRAM-SHA-1 authorization mechanism. +const RFC3454 = { + A1: "\u0221|[\u0234-\u024f]|[\u02ae-\u02af]|[\u02ef-\u02ff]|\ +[\u0350-\u035f]|[\u0370-\u0373]|[\u0376-\u0379]|[\u037b-\u037d]|\ +[\u037f-\u0383]|\u038b|\u038d|\u03a2|\u03cf|[\u03f7-\u03ff]|\u0487|\ +\u04cf|[\u04f6-\u04f7]|[\u04fa-\u04ff]|[\u0510-\u0530]|\ +[\u0557-\u0558]|\u0560|\u0588|[\u058b-\u0590]|\u05a2|\u05ba|\ +[\u05c5-\u05cf]|[\u05eb-\u05ef]|[\u05f5-\u060b]|[\u060d-\u061a]|\ +[\u061c-\u061e]|\u0620|[\u063b-\u063f]|[\u0656-\u065f]|\ +[\u06ee-\u06ef]|\u06ff|\u070e|[\u072d-\u072f]|[\u074b-\u077f]|\ +[\u07b2-\u0900]|\u0904|[\u093a-\u093b]|[\u094e-\u094f]|\ +[\u0955-\u0957]|[\u0971-\u0980]|\u0984|[\u098d-\u098e]|\ +[\u0991-\u0992]|\u09a9|\u09b1|[\u09b3-\u09b5]|[\u09ba-\u09bb]|\u09bd|\ +[\u09c5-\u09c6]|[\u09c9-\u09ca]|[\u09ce-\u09d6]|[\u09d8-\u09db]|\ +\u09de|[\u09e4-\u09e5]|[\u09fb-\u0a01]|[\u0a03-\u0a04]|\ +[\u0a0b-\u0a0e]|[\u0a11-\u0a12]|\u0a29|\u0a31|\u0a34|\u0a37|\ +[\u0a3a-\u0a3b]|\u0a3d|[\u0a43-\u0a46]|[\u0a49-\u0a4a]|\ +[\u0a4e-\u0a58]|\u0a5d|[\u0a5f-\u0a65]|[\u0a75-\u0a80]|\u0a84|\u0a8c|\ +\u0a8e|\u0a92|\u0aa9|\u0ab1|\u0ab4|[\u0aba-\u0abb]|\u0ac6|\u0aca|\ +[\u0ace-\u0acf]|[\u0ad1-\u0adf]|[\u0ae1-\u0ae5]|[\u0af0-\u0b00]|\ +\u0b04|[\u0b0d-\u0b0e]|[\u0b11-\u0b12]|\u0b29|\u0b31|[\u0b34-\u0b35]|\ +[\u0b3a-\u0b3b]|[\u0b44-\u0b46]|[\u0b49-\u0b4a]|[\u0b4e-\u0b55]|\ +[\u0b58-\u0b5b]|\u0b5e|[\u0b62-\u0b65]|[\u0b71-\u0b81]|\u0b84|\ +[\u0b8b-\u0b8d]|\u0b91|[\u0b96-\u0b98]|\u0b9b|\u0b9d|[\u0ba0-\u0ba2]|\ +[\u0ba5-\u0ba7]|[\u0bab-\u0bad]|\u0bb6|[\u0bba-\u0bbd]|\ +[\u0bc3-\u0bc5]|\u0bc9|[\u0bce-\u0bd6]|[\u0bd8-\u0be6]|\ +[\u0bf3-\u0c00]|\u0c04|\u0c0d|\u0c11|\u0c29|\u0c34|[\u0c3a-\u0c3d]|\ +\u0c45|\u0c49|[\u0c4e-\u0c54]|[\u0c57-\u0c5f]|[\u0c62-\u0c65]|\ +[\u0c70-\u0c81]|\u0c84|\u0c8d|\u0c91|\u0ca9|\u0cb4|[\u0cba-\u0cbd]|\ +\u0cc5|\u0cc9|[\u0cce-\u0cd4]|[\u0cd7-\u0cdd]|\u0cdf|[\u0ce2-\u0ce5]|\ +[\u0cf0-\u0d01]|\u0d04|\u0d0d|\u0d11|\u0d29|[\u0d3a-\u0d3d]|\ +[\u0d44-\u0d45]|\u0d49|[\u0d4e-\u0d56]|[\u0d58-\u0d5f]|\ +[\u0d62-\u0d65]|[\u0d70-\u0d81]|\u0d84|[\u0d97-\u0d99]|\u0db2|\u0dbc|\ +[\u0dbe-\u0dbf]|[\u0dc7-\u0dc9]|[\u0dcb-\u0dce]|\u0dd5|\u0dd7|\ +[\u0de0-\u0df1]|[\u0df5-\u0e00]|[\u0e3b-\u0e3e]|[\u0e5c-\u0e80]|\ +\u0e83|[\u0e85-\u0e86]|\u0e89|[\u0e8b-\u0e8c]|[\u0e8e-\u0e93]|\u0e98|\ +\u0ea0|\u0ea4|\u0ea6|[\u0ea8-\u0ea9]|\u0eac|\u0eba|[\u0ebe-\u0ebf]|\ +\u0ec5|\u0ec7|[\u0ece-\u0ecf]|[\u0eda-\u0edb]|[\u0ede-\u0eff]|\u0f48|\ +[\u0f6b-\u0f70]|[\u0f8c-\u0f8f]|\u0f98|\u0fbd|[\u0fcd-\u0fce]|\ +[\u0fd0-\u0fff]|\u1022|\u1028|\u102b|[\u1033-\u1035]|[\u103a-\u103f]|\ +[\u105a-\u109f]|[\u10c6-\u10cf]|[\u10f9-\u10fa]|[\u10fc-\u10ff]|\ +[\u115a-\u115e]|[\u11a3-\u11a7]|[\u11fa-\u11ff]|\u1207|\u1247|\u1249|\ +[\u124e-\u124f]|\u1257|\u1259|[\u125e-\u125f]|\u1287|\u1289|\ +[\u128e-\u128f]|\u12af|\u12b1|[\u12b6-\u12b7]|\u12bf|\u12c1|\ +[\u12c6-\u12c7]|\u12cf|\u12d7|\u12ef|\u130f|\u1311|[\u1316-\u1317]|\ +\u131f|\u1347|[\u135b-\u1360]|[\u137d-\u139f]|[\u13f5-\u1400]|\ +[\u1677-\u167f]|[\u169d-\u169f]|[\u16f1-\u16ff]|\u170d|\ +[\u1715-\u171f]|[\u1737-\u173f]|[\u1754-\u175f]|\u176d|\u1771|\ +[\u1774-\u177f]|[\u17dd-\u17df]|[\u17ea-\u17ff]|\u180f|\ +[\u181a-\u181f]|[\u1878-\u187f]|[\u18aa-\u1dff]|[\u1e9c-\u1e9f]|\ +[\u1efa-\u1eff]|[\u1f16-\u1f17]|[\u1f1e-\u1f1f]|[\u1f46-\u1f47]|\ +[\u1f4e-\u1f4f]|\u1f58|\u1f5a|\u1f5c|\u1f5e|[\u1f7e-\u1f7f]|\u1fb5|\ +\u1fc5|[\u1fd4-\u1fd5]|\u1fdc|[\u1ff0-\u1ff1]|\u1ff5|\u1fff|\ +[\u2053-\u2056]|[\u2058-\u205e]|[\u2064-\u2069]|[\u2072-\u2073]|\ +[\u208f-\u209f]|[\u20b2-\u20cf]|[\u20eb-\u20ff]|[\u213b-\u213c]|\ +[\u214c-\u2152]|[\u2184-\u218f]|[\u23cf-\u23ff]|[\u2427-\u243f]|\ +[\u244b-\u245f]|\u24ff|[\u2614-\u2615]|\u2618|[\u267e-\u267f]|\ +[\u268a-\u2700]|\u2705|[\u270a-\u270b]|\u2728|\u274c|\u274e|\ +[\u2753-\u2755]|\u2757|[\u275f-\u2760]|[\u2795-\u2797]|\u27b0|\ +[\u27bf-\u27cf]|[\u27ec-\u27ef]|[\u2b00-\u2e7f]|\u2e9a|\ +[\u2ef4-\u2eff]|[\u2fd6-\u2fef]|[\u2ffc-\u2fff]|\u3040|\ +[\u3097-\u3098]|[\u3100-\u3104]|[\u312d-\u3130]|\u318f|\ +[\u31b8-\u31ef]|[\u321d-\u321f]|[\u3244-\u3250]|[\u327c-\u327e]|\ +[\u32cc-\u32cf]|\u32ff|[\u3377-\u337a]|[\u33de-\u33df]|\u33ff|\ +[\u4db6-\u4dff]|[\u9fa6-\u9fff]|[\ua48d-\ua48f]|[\ua4c7-\uabff]|\ +[\ud7a4-\ud7ff]|[\ufa2e-\ufa2f]|[\ufa6b-\ufaff]|[\ufb07-\ufb12]|\ +[\ufb18-\ufb1c]|\ufb37|\ufb3d|\ufb3f|\ufb42|\ufb45|[\ufbb2-\ufbd2]|\ +[\ufd40-\ufd4f]|[\ufd90-\ufd91]|[\ufdc8-\ufdcf]|[\ufdfd-\ufdff]|\ +[\ufe10-\ufe1f]|[\ufe24-\ufe2f]|[\ufe47-\ufe48]|\ufe53|\ufe67|\ +[\ufe6c-\ufe6f]|\ufe75|[\ufefd-\ufefe]|\uff00|[\uffbf-\uffc1]|\ +[\uffc8-\uffc9]|[\uffd0-\uffd1]|[\uffd8-\uffd9]|[\uffdd-\uffdf]|\ +\uffe7|[\uffef-\ufff8]|[\u{10000}-\u{102ff}]|\u{1031f}|\ +[\u{10324}-\u{1032f}]|[\u{1034b}-\u{103ff}]|[\u{10426}-\u{10427}]|\ +[\u{1044e}-\u{1cfff}]|[\u{1d0f6}-\u{1d0ff}]|[\u{1d127}-\u{1d129}]|\ +[\u{1d1de}-\u{1d3ff}]|\u{1d455}|\u{1d49d}|[\u{1d4a0}-\u{1d4a1}]|\ +[\u{1d4a3}-\u{1d4a4}]|[\u{1d4a7}-\u{1d4a8}]|\u{1d4ad}|\u{1d4ba}|\ +\u{1d4bc}|\u{1d4c1}|\u{1d4c4}|\u{1d506}|[\u{1d50b}-\u{1d50c}]|\ +\u{1d515}|\u{1d51d}|\u{1d53a}|\u{1d53f}|\u{1d545}|\ +[\u{1d547}-\u{1d549}]|\u{1d551}|[\u{1d6a4}-\u{1d6a7}]|\ +[\u{1d7ca}-\u{1d7cd}]|[\u{1d800}-\u{1fffd}]|[\u{2a6d7}-\u{2f7ff}]|\ +[\u{2fa1e}-\u{2fffd}]|[\u{30000}-\u{3fffd}]|[\u{40000}-\u{4fffd}]|\ +[\u{50000}-\u{5fffd}]|[\u{60000}-\u{6fffd}]|[\u{70000}-\u{7fffd}]|\ +[\u{80000}-\u{8fffd}]|[\u{90000}-\u{9fffd}]|[\u{a0000}-\u{afffd}]|\ +[\u{b0000}-\u{bfffd}]|[\u{c0000}-\u{cfffd}]|[\u{d0000}-\u{dfffd}]|\ +\u{e0000}|[\u{e0002}-\u{e001f}]|[\u{e0080}-\u{efffd}]", + B1: "\u00ad|\u034f|\u1806|[\u180b-\u180d]|[\u200b-\u200d]|\u2060|\ +[\ufe00-\ufe0f]|\ufeff", + C12: "\u00a0|\u1680|[\u2000-\u200b]|\u202f|\u205f|\u3000", + C21: "[\u0000-\u001f]|\u007f", + C22: "[\u0080-\u009f]|\u06dd|\u070f|\u180e|\u200c|\u200d|\u2028|\u2029|\ +[\u2060-\u2063]|[\u206a-\u206f]|\ufeff|[\ufff9-\ufffc]", + C3: "[\ue000-\uf8ff]|[\u{f0000}-\u{ffffd}]|[\u{100000}-\u{10fffd}]", + C4: "[\ufdd0-\ufdef]|[\ufffe-\uffff]|[\u{1fffe}-\u{1ffff}]|\ +[\u{2fffe}-\u{2ffff}]|[\u{3fffe}-\u{3ffff}]|[\u{4fffe}-\u{4ffff}]|\ +[\u{5fffe}-\u{5ffff}]|[\u{6fffe}-\u{6ffff}]|[\u{7fffe}-\u{7ffff}]|\ +[\u{8fffe}-\u{8ffff}]|[\u{9fffe}-\u{9ffff}]|[\u{afffe}-\u{affff}]|\ +[\u{bfffe}-\u{bffff}]|[\u{cfffe}-\u{cffff}]|[\u{dfffe}-\u{dffff}]|\ +[\u{efffe}-\u{effff}]|[\u{ffffe}-\u{fffff}]|[\u{10fffe}-\u{10ffff}]", + C5: "[\ud800-\udfff]", + C6: "\ufff9|[\ufffa-\ufffd]", + C7: "[\u2ff0-\u2ffb]", + C8: "\u0340|\u0341|\u200e|\u200f|[\u202a-\u202e]|[\u206a-\u206f]", + C9: "\u{e0001}|[\u{e0020}-\u{e007f}]", + D1: "\u05be|\u05c0|\u05c3|[\u05d0-\u05ea]|[\u05f0-\u05f4]|\u061b|\u061f|\ +[\u0621-\u063a]|[\u0640-\u064a]|[\u066d-\u066f]|[\u0671-\u06d5]|\ +\u06dd|[\u06e5-\u06e6]|[\u06fa-\u06fe]|[\u0700-\u070d]|\u0710|\ +[\u0712-\u072c]|[\u0780-\u07a5]|\u07b1|\u200f|\ufb1d|[\ufb1f-\ufb28]|\ +[\ufb2a-\ufb36]|[\ufb38-\ufb3c]|\ufb3e|[\ufb40-\ufb41]|\ +[\ufb43-\ufb44]|[\ufb46-\ufbb1]|[\ufbd3-\ufd3d]|[\ufd50-\ufd8f]|\ +[\ufd92-\ufdc7]|[\ufdf0-\ufdfc]|[\ufe70-\ufe74]|[\ufe76-\ufefc]", + D2: "[\u0041-\u005a]|[\u0061-\u007a]|\u00aa|\u00b5|\u00ba|[\u00c0-\u00d6]|\ +[\u00d8-\u00f6]|[\u00f8-\u0220]|[\u0222-\u0233]|[\u0250-\u02ad]|\ +[\u02b0-\u02b8]|[\u02bb-\u02c1]|[\u02d0-\u02d1]|[\u02e0-\u02e4]|\ +\u02ee|\u037a|\u0386|[\u0388-\u038a]|\u038c|[\u038e-\u03a1]|\ +[\u03a3-\u03ce]|[\u03d0-\u03f5]|[\u0400-\u0482]|[\u048a-\u04ce]|\ +[\u04d0-\u04f5]|[\u04f8-\u04f9]|[\u0500-\u050f]|[\u0531-\u0556]|\ +[\u0559-\u055f]|[\u0561-\u0587]|\u0589|\u0903|[\u0905-\u0939]|\ +[\u093d-\u0940]|[\u0949-\u094c]|\u0950|[\u0958-\u0961]|\ +[\u0964-\u0970]|[\u0982-\u0983]|[\u0985-\u098c]|[\u098f-\u0990]|\ +[\u0993-\u09a8]|[\u09aa-\u09b0]|\u09b2|[\u09b6-\u09b9]|\ +[\u09be-\u09c0]|[\u09c7-\u09c8]|[\u09cb-\u09cc]|\u09d7|\ +[\u09dc-\u09dd]|[\u09df-\u09e1]|[\u09e6-\u09f1]|[\u09f4-\u09fa]|\ +[\u0a05-\u0a0a]|[\u0a0f-\u0a10]|[\u0a13-\u0a28]|[\u0a2a-\u0a30]|\ +[\u0a32-\u0a33]|[\u0a35-\u0a36]|[\u0a38-\u0a39]|[\u0a3e-\u0a40]|\ +[\u0a59-\u0a5c]|\u0a5e|[\u0a66-\u0a6f]|[\u0a72-\u0a74]|\u0a83|\ +[\u0a85-\u0a8b]|\u0a8d|[\u0a8f-\u0a91]|[\u0a93-\u0aa8]|\ +[\u0aaa-\u0ab0]|[\u0ab2-\u0ab3]|[\u0ab5-\u0ab9]|[\u0abd-\u0ac0]|\ +\u0ac9|[\u0acb-\u0acc]|\u0ad0|\u0ae0|[\u0ae6-\u0aef]|[\u0b02-\u0b03]|\ +[\u0b05-\u0b0c]|[\u0b0f-\u0b10]|[\u0b13-\u0b28]|[\u0b2a-\u0b30]|\ +[\u0b32-\u0b33]|[\u0b36-\u0b39]|[\u0b3d-\u0b3e]|\u0b40|\ +[\u0b47-\u0b48]|[\u0b4b-\u0b4c]|\u0b57|[\u0b5c-\u0b5d]|\ +[\u0b5f-\u0b61]|[\u0b66-\u0b70]|\u0b83|[\u0b85-\u0b8a]|\ +[\u0b8e-\u0b90]|[\u0b92-\u0b95]|[\u0b99-\u0b9a]|\u0b9c|\ +[\u0b9e-\u0b9f]|[\u0ba3-\u0ba4]|[\u0ba8-\u0baa]|[\u0bae-\u0bb5]|\ +[\u0bb7-\u0bb9]|[\u0bbe-\u0bbf]|[\u0bc1-\u0bc2]|[\u0bc6-\u0bc8]|\ +[\u0bca-\u0bcc]|\u0bd7|[\u0be7-\u0bf2]|[\u0c01-\u0c03]|\ +[\u0c05-\u0c0c]|[\u0c0e-\u0c10]|[\u0c12-\u0c28]|[\u0c2a-\u0c33]|\ +[\u0c35-\u0c39]|[\u0c41-\u0c44]|[\u0c60-\u0c61]|[\u0c66-\u0c6f]|\ +[\u0c82-\u0c83]|[\u0c85-\u0c8c]|[\u0c8e-\u0c90]|[\u0c92-\u0ca8]|\ +[\u0caa-\u0cb3]|[\u0cb5-\u0cb9]|\u0cbe|[\u0cc0-\u0cc4]|\ +[\u0cc7-\u0cc8]|[\u0cca-\u0ccb]|[\u0cd5-\u0cd6]|\u0cde|\ +[\u0ce0-\u0ce1]|[\u0ce6-\u0cef]|[\u0d02-\u0d03]|[\u0d05-\u0d0c]|\ +[\u0d0e-\u0d10]|[\u0d12-\u0d28]|[\u0d2a-\u0d39]|[\u0d3e-\u0d40]|\ +[\u0d46-\u0d48]|[\u0d4a-\u0d4c]|\u0d57|[\u0d60-\u0d61]|\ +[\u0d66-\u0d6f]|[\u0d82-\u0d83]|[\u0d85-\u0d96]|[\u0d9a-\u0db1]|\ +[\u0db3-\u0dbb]|\u0dbd|[\u0dc0-\u0dc6]|[\u0dcf-\u0dd1]|\ +[\u0dd8-\u0ddf]|[\u0df2-\u0df4]|[\u0e01-\u0e30]|[\u0e32-\u0e33]|\ +[\u0e40-\u0e46]|[\u0e4f-\u0e5b]|[\u0e81-\u0e82]|\u0e84|\ +[\u0e87-\u0e88]|\u0e8a|\u0e8d|[\u0e94-\u0e97]|[\u0e99-\u0e9f]|\ +[\u0ea1-\u0ea3]|\u0ea5|\u0ea7|[\u0eaa-\u0eab]|[\u0ead-\u0eb0]|\ +[\u0eb2-\u0eb3]|\u0ebd|[\u0ec0-\u0ec4]|\u0ec6|[\u0ed0-\u0ed9]|\ +[\u0edc-\u0edd]|[\u0f00-\u0f17]|[\u0f1a-\u0f34]|\u0f36|\u0f38|\ +[\u0f3e-\u0f47]|[\u0f49-\u0f6a]|\u0f7f|\u0f85|[\u0f88-\u0f8b]|\ +[\u0fbe-\u0fc5]|[\u0fc7-\u0fcc]|\u0fcf|[\u1000-\u1021]|\ +[\u1023-\u1027]|[\u1029-\u102a]|\u102c|\u1031|\u1038|[\u1040-\u1057]|\ +[\u10a0-\u10c5]|[\u10d0-\u10f8]|\u10fb|[\u1100-\u1159]|\ +[\u115f-\u11a2]|[\u11a8-\u11f9]|[\u1200-\u1206]|[\u1208-\u1246]|\ +\u1248|[\u124a-\u124d]|[\u1250-\u1256]|\u1258|[\u125a-\u125d]|\ +[\u1260-\u1286]|\u1288|[\u128a-\u128d]|[\u1290-\u12ae]|\u12b0|\ +[\u12b2-\u12b5]|[\u12b8-\u12be]|\u12c0|[\u12c2-\u12c5]|\ +[\u12c8-\u12ce]|[\u12d0-\u12d6]|[\u12d8-\u12ee]|[\u12f0-\u130e]|\ +\u1310|[\u1312-\u1315]|[\u1318-\u131e]|[\u1320-\u1346]|\ +[\u1348-\u135a]|[\u1361-\u137c]|[\u13a0-\u13f4]|[\u1401-\u1676]|\ +[\u1681-\u169a]|[\u16a0-\u16f0]|[\u1700-\u170c]|[\u170e-\u1711]|\ +[\u1720-\u1731]|[\u1735-\u1736]|[\u1740-\u1751]|[\u1760-\u176c]|\ +[\u176e-\u1770]|[\u1780-\u17b6]|[\u17be-\u17c5]|[\u17c7-\u17c8]|\ +[\u17d4-\u17da]|\u17dc|[\u17e0-\u17e9]|[\u1810-\u1819]|\ +[\u1820-\u1877]|[\u1880-\u18a8]|[\u1e00-\u1e9b]|[\u1ea0-\u1ef9]|\ +[\u1f00-\u1f15]|[\u1f18-\u1f1d]|[\u1f20-\u1f45]|[\u1f48-\u1f4d]|\ +[\u1f50-\u1f57]|\u1f59|\u1f5b|\u1f5d|[\u1f5f-\u1f7d]|[\u1f80-\u1fb4]|\ +[\u1fb6-\u1fbc]|\u1fbe|[\u1fc2-\u1fc4]|[\u1fc6-\u1fcc]|\ +[\u1fd0-\u1fd3]|[\u1fd6-\u1fdb]|[\u1fe0-\u1fec]|[\u1ff2-\u1ff4]|\ +[\u1ff6-\u1ffc]|\u200e|\u2071|\u207f|\u2102|\u2107|[\u210a-\u2113]|\ +\u2115|[\u2119-\u211d]|\u2124|\u2126|\u2128|[\u212a-\u212d]|\ +[\u212f-\u2131]|[\u2133-\u2139]|[\u213d-\u213f]|[\u2145-\u2149]|\ +[\u2160-\u2183]|[\u2336-\u237a]|\u2395|[\u249c-\u24e9]|\ +[\u3005-\u3007]|[\u3021-\u3029]|[\u3031-\u3035]|[\u3038-\u303c]|\ +[\u3041-\u3096]|[\u309d-\u309f]|[\u30a1-\u30fa]|[\u30fc-\u30ff]|\ +[\u3105-\u312c]|[\u3131-\u318e]|[\u3190-\u31b7]|[\u31f0-\u321c]|\ +[\u3220-\u3243]|[\u3260-\u327b]|[\u327f-\u32b0]|[\u32c0-\u32cb]|\ +[\u32d0-\u32fe]|[\u3300-\u3376]|[\u337b-\u33dd]|[\u33e0-\u33fe]|\ +[\u3400-\u4db5]|[\u4e00-\u9fa5]|[\ua000-\ua48c]|[\uac00-\ud7a3]|\ +[\ud800-\ufa2d]|[\ufa30-\ufa6a]|[\ufb00-\ufb06]|[\ufb13-\ufb17]|\ +[\uff21-\uff3a]|[\uff41-\uff5a]|[\uff66-\uffbe]|[\uffc2-\uffc7]|\ +[\uffca-\uffcf]|[\uffd2-\uffd7]|[\uffda-\uffdc]|[\u{10300}-\u{1031e}]|\ +[\u{10320}-\u{10323}]|[\u{10330}-\u{1034a}]|[\u{10400}-\u{10425}]|\ +[\u{10428}-\u{1044d}]|[\u{1d000}-\u{1d0f5}]|[\u{1d100}-\u{1d126}]|\ +[\u{1d12a}-\u{1d166}]|[\u{1d16a}-\u{1d172}]|[\u{1d183}-\u{1d184}]|\ +[\u{1d18c}-\u{1d1a9}]|[\u{1d1ae}-\u{1d1dd}]|[\u{1d400}-\u{1d454}]|\ +[\u{1d456}-\u{1d49c}]|[\u{1d49e}-\u{1d49f}]|\u{1d4a2}|\ +[\u{1d4a5}-\u{1d4a6}]|[\u{1d4a9}-\u{1d4ac}]|[\u{1d4ae}-\u{1d4b9}]|\ +\u{1d4bb}|[\u{1d4bd}-\u{1d4c0}]|[\u{1d4c2}-\u{1d4c3}]|\ +[\u{1d4c5}-\u{1d505}]|[\u{1d507}-\u{1d50a}]|[\u{1d50d}-\u{1d514}]|\ +[\u{1d516}-\u{1d51c}]|[\u{1d51e}-\u{1d539}]|[\u{1d53b}-\u{1d53e}]|\ +[\u{1d540}-\u{1d544}]|\u{1d546}|[\u{1d54a}-\u{1d550}]|\ +[\u{1d552}-\u{1d6a3}]|[\u{1d6a8}-\u{1d7c9}]|[\u{20000}-\u{2a6d6}]|\ +[\u{2f800}-\u{2fa1d}]|[\u{f0000}-\u{ffffd}]|[\u{100000}-\u{10fffd}]", +}; + +// Generates a random nonce and returns a base64 encoded string. +// aLength in bytes. +function createNonce(aLength) { + // RFC 5802 (5.1): Printable ASCII except ",". + // We guarantee a valid nonce value using base64 encoding. + return btoa(CryptoUtils.generateRandomBytes(aLength)); +} + +// Parses the string of server's response (aChallenge) into an object. +function parseChallenge(aChallenge) { + let attributes = {}; + aChallenge.split(",").forEach(value => { + let match = /^(\w)=([\s\S]*)$/.exec(value); + if (match) { + attributes[match[1]] = match[2]; + } + }); + return attributes; +} + +// RFC 4013 and RFC 3454: Stringprep Profile for User Names and Passwords. +export function saslPrep(aString) { + // RFC 4013 2.1: non-ASCII space characters (RFC 3454 C.1.2) mapped to space. + let retVal = aString.replace(new RegExp(RFC3454.C12, "u"), " "); + + // RFC 4013 2.1: RFC 3454 3.1, B.1: Map certain codepoints to nothing. + retVal = retVal.replace(new RegExp(RFC3454.B1, "u"), ""); + + // RFC 4013 2.2 asks for Unicode normalization form KC, which corresponds to + // RFC 3454 B.2. + retVal = retVal.normalize("NFKC"); + + // RFC 4013 2.3: Prohibited Output and 2.5: Unassigned Code Points. + let matchStr = + RFC3454.C12 + + "|" + + RFC3454.C21 + + "|" + + RFC3454.C22 + + "|" + + RFC3454.C3 + + "|" + + RFC3454.C4 + + "|" + + RFC3454.C5 + + "|" + + RFC3454.C6 + + "|" + + RFC3454.C7 + + "|" + + RFC3454.C8 + + "|" + + RFC3454.C9 + + "|" + + RFC3454.A1; + let match = new RegExp(matchStr, "u").test(retVal); + if (match) { + throw new Error("String contains prohibited characters"); + } + + // RFC 4013 2.4: Bidirectional Characters. + let r = new RegExp(RFC3454.D1, "u").test(retVal); + let l = new RegExp(RFC3454.D2, "u").test(retVal); + if (l && r) { + throw new Error( + "String must not contain LCat and RandALCat characters together" + ); + } else if (r) { + let matchFirst = new RegExp("^(" + RFC3454.D1 + ")", "u").test(retVal); + let matchLast = new RegExp("(" + RFC3454.D1 + ")$", "u").test(retVal); + if (!matchFirst || !matchLast) { + throw new Error( + "A RandALCat character must be the first and the last character" + ); + } + } + + return retVal; +} + +// Converts aName to saslname. +function saslName(aName) { + // RFC 5802 (5.1): the client SHOULD prepare the username using the "SASLprep". + // The characters ’,’ or ’=’ in usernames are sent as ’=2C’ and + // ’=3D’ respectively. + let saslName = saslPrep(aName).replace(/=/g, "=3D").replace(/,/g, "=2C"); + if (!saslName) { + throw new Error("Name is not valid"); + } + + return saslName; +} + +// Converts aMessage to array of bytes then apply hashing. +function bytesAndHash(aMessage, aHash) { + let hasher = Cc["@mozilla.org/security/hash;1"].createInstance( + Ci.nsICryptoHash + ); + hasher.init(hasher[aHash]); + + return CryptoUtils.digestBytes(aMessage, hasher); +} + +/** + * PBKDF2 password stretching with hmac. + * + * This is a copy of CryptoUtils.pbkdf2Generate, but with an additional argument to take the hash type. + * + * @param {string} passphrase Passphrase as an octet string. + * @param {string} salt Salt as an octet string. + * @param {string} iterations Number of iterations, a positive integer. + * @param {string} len Desired output length in bytes. + * @param {string} hash The desired hash algorithm (e.g. SHA-1 or SHA-256). + * @returns {Uint8Array} + */ +async function pbkdf2Generate(passphrase, salt, iterations, len, hash) { + passphrase = CommonUtils.byteStringToArrayBuffer(passphrase); + salt = CommonUtils.byteStringToArrayBuffer(salt); + const key = await crypto.subtle.importKey( + "raw", + passphrase, + { name: "PBKDF2" }, + false, + ["deriveBits"] + ); + const output = await crypto.subtle.deriveBits( + { + name: "PBKDF2", + hash, + salt, + iterations, + }, + key, + len * 8 + ); + return new Uint8Array(output); +} + +/* + * Given hash functions return a generator to be used as an XMPP authentication + * mechanism. + * + * @param {string} aHashFunctionName The name of a hash, e.g. SHA-1 or SHA-256. + * @param {string} aDigestLength The length of a hash digest, e.g. 20 for SHA-1 or 32 for SHA-256. + */ +function generateScramAuth(aHashFunctionName, aDigestLength) { + function* scramAuth(aUsername, aPassword, aDomain, aNonce) { + // The hash function name, without the '-' in it (e.g. convert SHA-1 to SHA1). + const hashFunctionProp = aHashFunctionName.replace("-", ""); + + // RFC 5802 (5): SCRAM Authentication Exchange. + const gs2Header = "n,,"; + // If a hard-coded nonce was given (e.g. for testing), use it. + let cNonce = aNonce ? aNonce : createNonce(32); + + let clientFirstMessageBare = "n=" + saslName(aUsername) + ",r=" + cNonce; + let clientFirstMessage = gs2Header + clientFirstMessageBare; + + let receivedStanza = yield { + send: Stanza.node( + "auth", + Stanza.NS.sasl, + { mechanism: "SCRAM-" + aHashFunctionName }, + btoa(clientFirstMessage) + ), + }; + + if (receivedStanza.localName != "challenge") { + throw new Error("Not authorized"); + } + + // RFC 5802 (3): SCRAM Algorithm Overview. + let decodedChallenge = atob(receivedStanza.innerText); + + // Expected to contain the user’s iteration count (i) and the user’s + // salt (s), and the server appends its own nonce to the client-specified + // one (r). + let attributes = parseChallenge(decodedChallenge); + if (attributes.hasOwnProperty("e")) { + throw new Error("Authentication failed: " + attributes.e); + } else if ( + !attributes.hasOwnProperty("i") || + !attributes.hasOwnProperty("s") || + !attributes.hasOwnProperty("r") + ) { + throw new Error("Unexpected response: " + decodedChallenge); + } + if (!attributes.r.startsWith(cNonce)) { + throw new Error("Nonce is not correct"); + } + + let clientFinalMessageWithoutProof = + "c=" + btoa(gs2Header) + ",r=" + attributes.r; + + // The server signature is calculated below, but needs to escape back to the main scope. + let serverSignature; + + // Once the promise resolves, continue with the handshake. + receivedStanza = yield (async () => { + // SaltedPassword := Hi(Normalize(password), salt, i) + // Normalize using saslPrep. + // dkLen MUST be equal to the SHA digest size. + let saltedPassword = await pbkdf2Generate( + saslPrep(aPassword), + atob(attributes.s), + parseInt(attributes.i), + aDigestLength, + aHashFunctionName + ); + + // Calculate ClientProof. + + // ClientKey := HMAC(SaltedPassword, "Client Key") + let clientKeyBuffer = await CryptoUtils.hmac( + aHashFunctionName, + saltedPassword, + CommonUtils.byteStringToArrayBuffer("Client Key") + ); + let clientKey = CommonUtils.arrayBufferToByteString(clientKeyBuffer); + + // StoredKey := H(ClientKey) + let storedKey = bytesAndHash(clientKey, hashFunctionProp); + + let authMessage = CommonUtils.byteStringToArrayBuffer( + clientFirstMessageBare + + "," + + decodedChallenge + + "," + + clientFinalMessageWithoutProof + ); + + // ClientSignature := HMAC(StoredKey, AuthMessage) + let clientSignatureBuffer = await CryptoUtils.hmac( + aHashFunctionName, + CommonUtils.byteStringToArrayBuffer(storedKey), + authMessage + ); + let clientSignature = CommonUtils.arrayBufferToByteString( + clientSignatureBuffer + ); + // ClientProof := ClientKey XOR ClientSignature + let clientProof = CryptoUtils.xor(clientKey, clientSignature); + + // Calculate ServerSignature. + + // ServerKey := HMAC(SaltedPassword, "Server Key") + let serverKeyBuffer = await CryptoUtils.hmac( + aHashFunctionName, + saltedPassword, + CommonUtils.byteStringToArrayBuffer("Server Key") + ); + + // ServerSignature := HMAC(ServerKey, AuthMessage) + let serverSignatureBuffer = await CryptoUtils.hmac( + aHashFunctionName, + serverKeyBuffer, + authMessage + ); + serverSignature = CommonUtils.arrayBufferToByteString( + serverSignatureBuffer + ); + + let clientFinalMessage = + clientFinalMessageWithoutProof + ",p=" + btoa(clientProof); + + return { + send: Stanza.node( + "response", + Stanza.NS.sasl, + null, + btoa(clientFinalMessage) + ), + log: "<response/> (base64 encoded SCRAM response containing password not logged)", + }; + })(); + + // Only check server signature if we succeed to authenticate. + if (receivedStanza.localName != "success") { + throw new Error("Didn't receive the expected auth success stanza."); + } + + let decodedResponse = atob(receivedStanza.innerText); + + // Expected to contain a base64-encoded ServerSignature (v). + attributes = parseChallenge(decodedResponse); + if (!attributes.hasOwnProperty("v")) { + throw new Error("Unexpected response: " + decodedResponse); + } + + // Compare ServerSignature with our ServerSignature which we calculated in + // _generateResponse. + let serverSignatureResponse = atob(attributes.v); + if (serverSignature != serverSignatureResponse) { + throw new Error("Server signature does not match"); + } + } + + return scramAuth; +} + +export var XMPPAuthMechanisms = { + PLAIN: PlainAuth, + "SCRAM-SHA-1": generateScramAuth("SHA-1", 20), + "SCRAM-SHA-256": generateScramAuth("SHA-256", 32), +}; diff --git a/comm/chat/protocols/xmpp/xmpp-base.sys.mjs b/comm/chat/protocols/xmpp/xmpp-base.sys.mjs new file mode 100644 index 0000000000..f7e0ccd98e --- /dev/null +++ b/comm/chat/protocols/xmpp/xmpp-base.sys.mjs @@ -0,0 +1,3421 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +import { clearTimeout, setTimeout } from "resource://gre/modules/Timer.sys.mjs"; + +import { IMServices } from "resource:///modules/IMServices.sys.mjs"; +import { Status } from "resource:///modules/imStatusUtils.sys.mjs"; +import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs"; +import { + executeSoon, + nsSimpleEnumerator, + EmptyEnumerator, + ClassInfo, + l10nHelper, +} from "resource:///modules/imXPCOMUtils.sys.mjs"; +import { + GenericAccountPrototype, + GenericAccountBuddyPrototype, + GenericConvIMPrototype, + GenericConvChatPrototype, + GenericConversationPrototype, + TooltipInfo, +} from "resource:///modules/jsProtoHelper.sys.mjs"; +import { NormalizedMap } from "resource:///modules/NormalizedMap.sys.mjs"; +import { + Stanza, + SupportedFeatures, +} from "resource:///modules/xmpp-xml.sys.mjs"; +import { XMPPSession } from "resource:///modules/xmpp-session.sys.mjs"; + +const lazy = {}; +ChromeUtils.defineESModuleGetters(lazy, { + DownloadUtils: "resource://gre/modules/DownloadUtils.sys.mjs", + FileUtils: "resource://gre/modules/FileUtils.sys.mjs", +}); +ChromeUtils.defineModuleGetter( + lazy, + "NetUtil", + "resource://gre/modules/NetUtil.jsm" +); +XPCOMUtils.defineLazyServiceGetter( + lazy, + "imgTools", + "@mozilla.org/image/tools;1", + "imgITools" +); + +XPCOMUtils.defineLazyGetter(lazy, "_", () => + l10nHelper("chrome://chat/locale/xmpp.properties") +); + +XPCOMUtils.defineLazyGetter(lazy, "TXTToHTML", function () { + let cs = Cc["@mozilla.org/txttohtmlconv;1"].getService(Ci.mozITXTToHTMLConv); + return aTxt => cs.scanTXT(aTxt, cs.kEntities); +}); + +// Parses the status from a presence stanza into an object of statusType, +// statusText and idleSince. +function parseStatus(aStanza) { + let statusType = Ci.imIStatusInfo.STATUS_AVAILABLE; + let show = aStanza.getElement(["show"]); + if (show) { + show = show.innerText; + if (show == "away") { + statusType = Ci.imIStatusInfo.STATUS_AWAY; + } else if (show == "chat") { + statusType = Ci.imIStatusInfo.STATUS_AVAILABLE; // FIXME + } else if (show == "dnd") { + statusType = Ci.imIStatusInfo.STATUS_UNAVAILABLE; + } else if (show == "xa") { + statusType = Ci.imIStatusInfo.STATUS_IDLE; + } + } + + let idleSince = 0; + let date = _getDelay(aStanza); + if (date) { + idleSince = date.getTime(); + } + + let query = aStanza.getElement(["query"]); + if (query && query.uri == Stanza.NS.last) { + let now = Math.floor(Date.now() / 1000); + idleSince = now - parseInt(query.attributes.seconds, 10); + statusType = Ci.imIStatusInfo.STATUS_IDLE; + } + + let status = aStanza.getElement(["status"]); + status = status ? status.innerText : ""; + + return { statusType, statusText: status, idleSince }; +} + +// Returns a Date object for the delay value (stamp) in aStanza if it exists, +// otherwise returns undefined. +function _getDelay(aStanza) { + // XEP-0203: Delayed Delivery. + let date; + let delay = aStanza.getElement(["delay"]); + if (delay && delay.uri == Stanza.NS.delay) { + if (delay.attributes.stamp) { + date = new Date(delay.attributes.stamp); + } + } + if (date && isNaN(date.getTime())) { + return undefined; + } + + return date; +} + +// Writes aMsg in aConv as an outgoing message with optional date as the +// message may be sent from another client. +function _displaySentMsg(aConv, aMsg, aDate) { + let who; + if (aConv._account._connection) { + who = aConv._account._connection._jid.jid; + } + if (!who) { + who = aConv._account.name; + } + + let flags = { outgoing: true }; + flags._alias = aConv.account.alias || aConv.account.statusInfo.displayName; + + if (aDate) { + flags.time = aDate / 1000; + flags.delayed = true; + } + aConv.writeMessage(who, aMsg, flags); +} + +// The timespan after which we consider roomInfo to be stale. +var kListRefreshInterval = 12 * 60 * 60 * 1000; // 12 hours. + +/* This is an ordered list, used to determine chat buddy flags: + * index = member -> voiced + * moderator -> moderator + * admin -> admin + * owner -> founder + */ +var kRoles = [ + "outcast", + "visitor", + "participant", + "member", + "moderator", + "admin", + "owner", +]; + +function MUCParticipant(aNick, aJid, aPresenceStanza) { + this._jid = aJid; + this.name = aNick; + this.onPresenceStanza(aPresenceStanza); +} +MUCParticipant.prototype = { + __proto__: ClassInfo("prplIConvChatBuddy", "XMPP ConvChatBuddy object"), + + buddy: false, + + // The occupant jid of the participant which is of the form room@domain/nick. + _jid: null, + + // The real jid of the participant which is of the form local@domain/resource. + accountJid: null, + + statusType: null, + statusText: null, + get alias() { + return this.name; + }, + + role: 2, // "participant" by default + + // Called when a presence stanza is received for this participant. + onPresenceStanza(aStanza) { + let statusInfo = parseStatus(aStanza); + this.statusType = statusInfo.statusType; + this.statusText = statusInfo.statusText; + + let x = aStanza.children.filter( + child => child.localName == "x" && child.uri == Stanza.NS.muc_user + ); + if (x.length == 0) { + return; + } + + // XEP-0045 (7.2.3): We only expect a single <x/> element of this namespace, + // so we ignore any others. + x = x[0]; + + let item = x.getElement(["item"]); + if (!item) { + return; + } + + this.role = Math.max( + kRoles.indexOf(item.attributes.role), + kRoles.indexOf(item.attributes.affiliation) + ); + + let accountJid = item.attributes.jid; + if (accountJid) { + this.accountJid = accountJid; + } + }, + + get voiced() { + /* FIXME: The "voiced" role corresponds to users that can send messages to + * the room. If the chat is unmoderated, this should include everyone, not + * just members. */ + return this.role == kRoles.indexOf("member"); + }, + get moderator() { + return this.role == kRoles.indexOf("moderator"); + }, + get admin() { + return this.role == kRoles.indexOf("admin"); + }, + get founder() { + return this.role == kRoles.indexOf("owner"); + }, + typing: false, +}; + +// MUC (Multi-User Chat) +export var XMPPMUCConversationPrototype = { + __proto__: GenericConvChatPrototype, + // By default users are not in a MUC. + _left: true, + + // Tracks all received messages to avoid possible duplication if the server + // sends us the last few messages again when we rejoin a room. + _messageIds: new Set(), + + _init(aAccount, aJID, aNick) { + this._messageIds = new Set(); + GenericConvChatPrototype._init.call(this, aAccount, aJID, aNick); + }, + + _targetResource: "", + + // True while we are rejoining a room previously parted by the user. + _rejoined: false, + + get topic() { + return this._topic; + }, + set topic(aTopic) { + // XEP-0045 (8.1): Modifying the room subject. + let subject = Stanza.node("subject", null, null, aTopic.trim()); + let s = Stanza.message( + this.name, + null, + null, + { type: "groupchat" }, + subject + ); + let notAuthorized = lazy._( + "conversation.error.changeTopicFailedNotAuthorized" + ); + this._account.sendStanza( + s, + this._account.handleErrors( + { + forbidden: notAuthorized, + notAcceptable: notAuthorized, + itemNotFound: notAuthorized, + }, + this + ) + ); + }, + get topicSettable() { + return true; + }, + + /* Called when the user enters a chat message */ + dispatchMessage(aMsg, aAction = false) { + if (aAction) { + // XEP-0245: The /me Command. + // We need to prepend "/me " as the first four characters of the message + // body. + aMsg = "/me " + aMsg; + } + // XEP-0045 (7.4): Sending a message to all occupants in a room. + let s = Stanza.message(this.name, aMsg, null, { type: "groupchat" }); + let notInRoom = lazy._( + "conversation.error.sendFailedAsNotInRoom", + this.name, + aMsg + ); + this._account.sendStanza( + s, + this._account.handleErrors( + { + itemNotFound: notInRoom, + notAcceptable: notInRoom, + }, + this + ) + ); + }, + + /* Called by the account when a presence stanza is received for this muc */ + onPresenceStanza(aStanza) { + let from = aStanza.attributes.from; + let nick = this._account._parseJID(from).resource; + let jid = this._account.normalize(from); + let x = aStanza + .getElements(["x"]) + .find( + e => e.uri == Stanza.NS.muc_user || e.uri == Stanza.NS.vcard_update + ); + + // Check if the join failed. + if (this.left && aStanza.attributes.type == "error") { + let error = this._account.parseError(aStanza); + let message; + switch (error.condition) { + case "not-authorized": + case "registration-required": + // XEP-0045 (7.2.7): Members-Only Rooms. + message = lazy._("conversation.error.joinFailedNotAuthorized"); + break; + case "not-allowed": + message = lazy._("conversation.error.creationFailedNotAllowed"); + break; + case "remote-server-not-found": + message = lazy._( + "conversation.error.joinFailedRemoteServerNotFound", + this.name + ); + break; + case "forbidden": + // XEP-0045 (7.2.8): Banned users. + message = lazy._("conversation.error.joinForbidden", this.name); + break; + default: + message = lazy._("conversation.error.joinFailed", this.name); + this.ERROR("Failed to join MUC: " + aStanza.convertToString()); + break; + } + this.writeMessage(this.name, message, { system: true, error: true }); + this.joining = false; + return; + } + + if (!x) { + this.WARN( + "Received a MUC presence stanza without an x element or " + + "with a namespace we don't handle." + ); + return; + } + // Handle a MUC resource avatar + if ( + x.uri == Stanza.NS.vcard_update && + aStanza.attributes.from == this.normalizedName + ) { + let photo = aStanza.getElement(["x", "photo"]); + if (photo && photo.uri == Stanza.NS.vcard_update) { + let hash = photo.innerText; + if (hash && hash != this._photoHash) { + this._account._addVCardRequest(this.normalizedName); + } else if (!hash && this._photoHash) { + delete this._photoHash; + this.convIconFilename = ""; + } + } + return; + } + let codes = x.getElements(["status"]).map(elt => elt.attributes.code); + let item = x.getElement(["item"]); + + // Changes the nickname of a participant for this muc. + let changeNick = () => { + if (!item || !item.attributes.nick) { + this.WARN( + "Received a MUC presence code 303 or 210 stanza without an " + + "item element or a nick attribute." + ); + return; + } + let newNick = item.attributes.nick; + this.updateNick(nick, newNick, nick == this.nick); + }; + + if (aStanza.attributes.type == "unavailable") { + if (!this._participants.has(nick)) { + this.WARN( + "received unavailable presence for an unknown MUC participant: " + + from + ); + return; + } + if (codes.includes("303")) { + // XEP-0045 (7.6): Changing Nickname. + // Service Updates Nick for user. + changeNick(); + return; + } + if (item && item.attributes.role == "none") { + // XEP-0045: an occupant has left the room. + this.removeParticipant(nick); + + // Who caused the participant to leave the room. + let actor = item.getElement(["actor"]); + let actorNick = actor ? actor.attributes.nick : ""; + let isActor = actorNick ? ".actor" : ""; + + // Why the participant left. + let reasonNode = item.getElement(["reason"]); + let reason = reasonNode ? reasonNode.innerText : ""; + let isReason = reason ? ".reason" : ""; + + let isYou = nick == this.nick ? ".you" : ""; + let affectedNick = isYou ? "" : nick; + if (isYou) { + this.left = true; + } + + let message; + if (codes.includes("301")) { + // XEP-0045 (9.1): Banning a User. + message = "conversation.message.banned"; + } else if (codes.includes("307")) { + // XEP-0045 (8.2): Kicking an Occupant. + message = "conversation.message.kicked"; + } else if (codes.includes("322") || codes.includes("321")) { + // XEP-0045: Inform user that he or she is being removed from the + // room because the room has been changed to members-only and the + // user is not a member. + message = "conversation.message.removedNonMember"; + } else if (codes.includes("332")) { + // XEP-0045: Inform user that he or she is being removed from the + // room because the MUC service is being shut down. + message = "conversation.message.mucShutdown"; + + // The reason here just duplicates what's in the system message. + reason = isReason = ""; + } else { + // XEP-0045 (7.14): Received when the user parts a room. + message = "conversation.message.parted"; + + // The reason is in a status element in this case. + reasonNode = aStanza.getElement(["status"]); + reason = reasonNode ? reasonNode.innerText : ""; + isReason = reason ? ".reason" : ""; + } + + if (message) { + let messageID = message + isYou + isActor + isReason; + let params = [actorNick, affectedNick, reason].filter(s => s); + this.writeMessage(this.name, lazy._(messageID, ...params), { + system: true, + }); + } + } else { + this.WARN("Unhandled type==unavailable MUC presence stanza."); + } + return; + } + + if (codes.includes("201")) { + // XEP-0045 (10.1): Creating room. + // Service Acknowledges Room Creation + // and Room is awaiting configuration. + // XEP-0045 (10.1.2): Instant room. + let query = Stanza.node( + "query", + Stanza.NS.muc_owner, + null, + Stanza.node("x", Stanza.NS.xdata, { type: "submit" }) + ); + let s = Stanza.iq("set", null, jid, query); + this._account.sendStanza(s, aStanzaReceived => { + if (aStanzaReceived.attributes.type != "result") { + return false; + } + + // XEP-0045: Service Informs New Room Owner of Success + // for instant and reserved rooms. + this.left = false; + this.joining = false; + return true; + }); + } else if (codes.includes("210")) { + // XEP-0045 (7.6): Changing Nickname. + // Service modifies this user's nickname in accordance with local service + // policies. + changeNick(); + return; + } else if (codes.includes("110")) { + // XEP-0045: Room exists and joined successfully. + this.left = false; + this.joining = false; + // TODO (Bug 1172350): Implement Service Discovery Extensions (XEP-0128) to obtain + // configuration of this room. + } else if (codes.includes("104") && nick == this.name) { + // https://xmpp.org/extensions/inbox/muc-avatars.html (XEP-XXXX) + this._account._addVCardRequest(this.normalizedName); + } + + if (!this._participants.get(nick)) { + let participant = new MUCParticipant(nick, from, aStanza); + this._participants.set(nick, participant); + this.notifyObservers( + new nsSimpleEnumerator([participant]), + "chat-buddy-add" + ); + if (this.nick != nick && !this.joining) { + this.writeMessage( + this.name, + lazy._("conversation.message.join", nick), + { + system: true, + } + ); + } else if (this.nick == nick && this._rejoined) { + this.writeMessage(this.name, lazy._("conversation.message.rejoined"), { + system: true, + }); + this._rejoined = false; + } + } else { + this._participants.get(nick).onPresenceStanza(aStanza); + this.notifyObservers(this._participants.get(nick), "chat-buddy-update"); + } + }, + + /* Called by the account when a message is received for this muc */ + incomingMessage(aMsg, aStanza, aDate) { + let from = this._account._parseJID(aStanza.attributes.from).resource; + let id = aStanza.attributes.id; + let flags = {}; + if (!from) { + flags.system = true; + from = this.name; + } else if (aStanza.attributes.type == "error") { + aMsg = lazy._("conversation.error.notDelivered", aMsg); + flags.system = true; + flags.error = true; + } else if (from == this._nick) { + flags.outgoing = true; + } else { + flags.incoming = true; + } + if (aDate) { + flags.time = aDate / 1000; + flags.delayed = true; + } + if (id) { + // Checks if a message exists in conversation to avoid duplication. + if (this._messageIds.has(id)) { + return; + } + this._messageIds.add(id); + } + this.writeMessage(from, aMsg, flags); + }, + + getNormalizedChatBuddyName(aNick) { + return this._account.normalizeFullJid(this.name + "/" + aNick); + }, + + // Leaves MUC conversation. + part(aMsg = null) { + let s = Stanza.presence( + { to: this.name + "/" + this._nick, type: "unavailable" }, + aMsg ? Stanza.node("status", null, null, aMsg.trim()) : null + ); + this._account.sendStanza(s); + delete this.chatRoomFields; + }, + + // Invites a user to MUC conversation. + invite(aJID, aMsg = null) { + // XEP-0045 (7.8): Inviting Another User to a Room. + // XEP-0045 (7.8.2): Mediated Invitation. + let invite = Stanza.node( + "invite", + null, + { to: aJID }, + aMsg ? Stanza.node("reason", null, null, aMsg) : null + ); + let x = Stanza.node("x", Stanza.NS.muc_user, null, invite); + let s = Stanza.node("message", null, { to: this.name }, x); + this._account.sendStanza( + s, + this._account.handleErrors( + { + forbidden: lazy._("conversation.error.inviteFailedForbidden"), + // ejabberd uses error not-allowed to indicate that this account does not + // have the required privileges to invite users instead of forbidden error, + // and this is not mentioned in the spec (XEP-0045). + notAllowed: lazy._("conversation.error.inviteFailedForbidden"), + itemNotFound: lazy._("conversation.error.failedJIDNotFound", aJID), + }, + this + ) + ); + }, + + // Bans a participant from MUC conversation. + ban(aNickName, aMsg = null) { + // XEP-0045 (9.1): Banning a User. + let participant = this._participants.get(aNickName); + if (!participant) { + this.writeMessage( + this.name, + lazy._("conversation.error.nickNotInRoom", aNickName), + { system: true } + ); + return; + } + if (!participant.accountJid) { + this.writeMessage( + this.name, + lazy._("conversation.error.banCommandAnonymousRoom"), + { system: true } + ); + return; + } + + let attributes = { affiliation: "outcast", jid: participant.accountJid }; + let item = Stanza.node( + "item", + null, + attributes, + aMsg ? Stanza.node("reason", null, null, aMsg) : null + ); + let s = Stanza.iq( + "set", + null, + this.name, + Stanza.node("query", Stanza.NS.muc_admin, null, item) + ); + this._account.sendStanza(s, this._banKickHandler, this); + }, + + // Kicks a participant from MUC conversation. + kick(aNickName, aMsg = null) { + // XEP-0045 (8.2): Kicking an Occupant. + let attributes = { role: "none", nick: aNickName }; + let item = Stanza.node( + "item", + null, + attributes, + aMsg ? Stanza.node("reason", null, null, aMsg) : null + ); + let s = Stanza.iq( + "set", + null, + this.name, + Stanza.node("query", Stanza.NS.muc_admin, null, item) + ); + this._account.sendStanza(s, this._banKickHandler, this); + }, + + // Callback for ban and kick commands. + _banKickHandler(aStanza) { + return this._account._handleResult( + { + notAllowed: lazy._("conversation.error.banKickCommandNotAllowed"), + conflict: lazy._("conversation.error.banKickCommandConflict"), + }, + this + )(aStanza); + }, + + // Changes nick in MUC conversation to a new one. + setNick(aNewNick) { + // XEP-0045 (7.6): Changing Nickname. + let s = Stanza.presence({ to: this.name + "/" + aNewNick }, null); + this._account.sendStanza( + s, + this._account.handleErrors( + { + // XEP-0045 (7.6): Changing Nickname (example 53). + // TODO: We should discover if the user has a reserved nickname (maybe + // before joining a room), cf. XEP-0045 (7.12). + notAcceptable: lazy._( + "conversation.error.changeNickFailedNotAcceptable", + aNewNick + ), + // XEP-0045 (7.2.9): Nickname Conflict. + conflict: lazy._( + "conversation.error.changeNickFailedConflict", + aNewNick + ), + }, + this + ) + ); + }, + + // Called by the account when a message stanza is received for this muc and + // needs to be handled. + onMessageStanza(aStanza) { + let x = aStanza.getElement(["x"]); + let decline = x.getElement(["decline"]); + if (decline) { + // XEP-0045 (7.8): Inviting Another User to a Room. + // XEP-0045 (7.8.2): Mediated Invitation. + let invitee = decline.attributes.jid; + let reasonNode = decline.getElement(["reason"]); + let reason = reasonNode ? reasonNode.innerText : ""; + let msg; + if (reason) { + msg = lazy._( + "conversation.message.invitationDeclined.reason", + invitee, + reason + ); + } else { + msg = lazy._("conversation.message.invitationDeclined", invitee); + } + + this.writeMessage(this.name, msg, { system: true }); + } else { + this.WARN("Unhandled message stanza."); + } + }, + + /* Called when the user closed the conversation */ + close() { + if (!this.left) { + this.part(); + } + GenericConvChatPrototype.close.call(this); + }, + unInit() { + this._account.removeConversation(this.name); + GenericConvChatPrototype.unInit.call(this); + }, + + _photoHash: null, + _saveIcon(aPhotoNode) { + this._account._saveResourceIcon(aPhotoNode, this).then( + url => { + this.convIconFilename = url; + }, + error => { + this._account.WARN( + "Error while loading conversation icon for " + + this.normalizedName + + ": " + + error.message + ); + } + ); + }, +}; + +function XMPPMUCConversation(aAccount, aJID, aNick) { + this._init(aAccount, aJID, aNick); +} +XMPPMUCConversation.prototype = XMPPMUCConversationPrototype; + +/* Helper class for buddy conversations */ +export var XMPPConversationPrototype = { + __proto__: GenericConvIMPrototype, + + _typingTimer: null, + supportChatStateNotifications: true, + _typingState: "active", + + // Indicates that current conversation is with a MUC participant and the + // recipient jid (stored in the userName) is of the form room@domain/nick. + _isMucParticipant: false, + + get buddy() { + return this._account._buddies.get(this.name); + }, + get title() { + return this.contactDisplayName; + }, + get contactDisplayName() { + return this.buddy ? this.buddy.contactDisplayName : this.name; + }, + get userName() { + return this.buddy ? this.buddy.userName : this.name; + }, + + // Returns jid (room@domain/nick) if it is with a MUC participant, and the + // name of conversation otherwise. + get normalizedName() { + if (this._isMucParticipant) { + return this._account.normalizeFullJid(this.name); + } + return this._account.normalize(this.name); + }, + + // Used to avoid showing full jids in typing notifications. + get shortName() { + if (this.buddy) { + return this.buddy.contactDisplayName; + } + + let jid = this._account._parseJID(this.name); + if (!jid) { + return this.name; + } + + // Returns nick of the recipient if conversation is with a participant of + // a MUC we are in as jid of the recipient is of the form room@domain/nick. + if (this._isMucParticipant) { + return jid.resource; + } + + return jid.node; + }, + + get shouldSendTypingNotifications() { + return ( + this.supportChatStateNotifications && + Services.prefs.getBoolPref("purple.conversations.im.send_typing") + ); + }, + + /* Called when the user is typing a message + * aString - the currently typed message + * Returns the number of characters that can still be typed */ + sendTyping(aString) { + if (!this.shouldSendTypingNotifications) { + return Ci.prplIConversation.NO_TYPING_LIMIT; + } + + this._cancelTypingTimer(); + if (aString.length) { + this._typingTimer = setTimeout(this.finishedComposing.bind(this), 10000); + } + + this._setTypingState(aString.length ? "composing" : "active"); + + return Ci.prplIConversation.NO_TYPING_LIMIT; + }, + + finishedComposing() { + if (!this.shouldSendTypingNotifications) { + return; + } + + this._setTypingState("paused"); + }, + + _setTypingState(aNewState) { + if (this._typingState == aNewState) { + return; + } + + let s = Stanza.message(this.to, null, aNewState); + + // We don't care about errors in response to typing notifications + // (e.g. because the user has left the room when talking to a MUC + // participant). + this._account.sendStanza(s, () => true); + + this._typingState = aNewState; + }, + _cancelTypingTimer() { + if (this._typingTimer) { + clearTimeout(this._typingTimer); + delete this._typingTimer; + } + }, + + // Holds the resource of user that you are currently talking to, but if the + // user is a participant of a MUC we are in, holds the nick of user you are + // talking to. + _targetResource: "", + + get to() { + if (!this._targetResource || this._isMucParticipant) { + return this.userName; + } + return this.userName + "/" + this._targetResource; + }, + + /* Called when the user enters a chat message */ + dispatchMessage(aMsg, aAction = false) { + if (aAction) { + // XEP-0245: The /me Command. + // We need to prepend "/me " as the first four characters of the message + // body. + aMsg = "/me" + aMsg; + } + this._cancelTypingTimer(); + let cs = this.shouldSendTypingNotifications ? "active" : null; + let s = Stanza.message(this.to, aMsg, cs); + this._account.sendStanza(s); + _displaySentMsg(this, aMsg); + delete this._typingState; + }, + + // Invites the contact to a MUC room. + invite(aRoomJid, aPassword = null) { + // XEP-0045 (7.8): Inviting Another User to a Room. + // XEP-0045 (7.8.1) and XEP-0249: Direct Invitation. + let x = Stanza.node("x", Stanza.NS.conference, { + jid: aRoomJid, + password: aPassword, + }); + this._account.sendStanza(Stanza.node("message", null, { to: this.to }, x)); + }, + + // Query the user for its Software Version. + // XEP-0092: Software Version. + getVersion() { + // TODO: Use Service Discovery to determine if the user's client supports + // jabber:iq:version protocol. + + let s = Stanza.iq( + "get", + null, + this.to, + Stanza.node("query", Stanza.NS.version) + ); + this._account.sendStanza(s, aStanza => { + // TODO: handle other errors that can result from querying + // user for its software version. + if ( + this._account.handleErrors( + { + default: lazy._("conversation.error.version.unknown"), + }, + this + )(aStanza) + ) { + return; + } + + let query = aStanza.getElement(["query"]); + if (!query || query.uri != Stanza.NS.version) { + this.WARN( + "Received a response to version query which does not " + + "contain query element or 'jabber:iq:version' namespace." + ); + return; + } + + let name = query.getElement(["name"]); + let version = query.getElement(["version"]); + if (!name || !version) { + // XEP-0092: name and version elements are REQUIRED. + this.WARN( + "Received a response to version query which does not " + + "contain name or version." + ); + return; + } + + let messageID = "conversation.message.version"; + let params = [this.shortName, name.innerText, version.innerText]; + + // XEP-0092: os is OPTIONAL. + let os = query.getElement(["os"]); + if (os) { + params.push(os.innerText); + messageID += "WithOS"; + } + + this.writeMessage(this.name, lazy._(messageID, ...params), { + system: true, + }); + }); + }, + + /* Perform entity escaping before displaying the message. We assume incoming + messages have already been escaped, and will otherwise be filtered. */ + prepareForDisplaying(aMsg) { + if (aMsg.outgoing && !aMsg.system) { + aMsg.displayMessage = lazy.TXTToHTML(aMsg.displayMessage); + } + GenericConversationPrototype.prepareForDisplaying.apply(this, arguments); + }, + + /* Called by the account when a message is received from the buddy */ + incomingMessage(aMsg, aStanza, aDate) { + let from = aStanza.attributes.from; + this._targetResource = this._account._parseJID(from).resource; + let flags = {}; + let error = this._account.parseError(aStanza); + if (error) { + let norm = this._account.normalize(from); + let muc = this._account._mucs.get(norm); + + if (!aMsg) { + // Failed outgoing message. + switch (error.condition) { + case "remote-server-not-found": + aMsg = lazy._("conversation.error.remoteServerNotFound"); + break; + case "service-unavailable": + aMsg = lazy._( + "conversation.error.sendServiceUnavailable", + this.shortName + ); + break; + default: + aMsg = lazy._("conversation.error.unknownSendError"); + break; + } + } else if ( + this._isMucParticipant && + muc && + !muc.left && + error.condition == "item-not-found" + ) { + // XEP-0045 (7.5): MUC private messages. + // If we try to send to participant not in a room we are in. + aMsg = lazy._( + "conversation.error.sendFailedAsRecipientNotInRoom", + this._targetResource, + aMsg + ); + } else if ( + this._isMucParticipant && + (error.condition == "item-not-found" || + error.condition == "not-acceptable") + ) { + // If we left a room and try to send to a participant in it or the + // room is removed. + aMsg = lazy._( + "conversation.error.sendFailedAsNotInRoom", + this._account.normalize(from), + aMsg + ); + } else { + aMsg = lazy._("conversation.error.notDelivered", aMsg); + } + flags.system = true; + flags.error = true; + } else { + flags = { incoming: true, _alias: this.contactDisplayName }; + // XEP-0245: The /me Command. + if (aMsg.startsWith("/me ")) { + flags.action = true; + aMsg = aMsg.slice(4); + } + } + if (aDate) { + flags.time = aDate / 1000; + flags.delayed = true; + } + this.writeMessage(from, aMsg, flags); + }, + + /* Called when the user closed the conversation */ + close() { + // TODO send the stanza indicating we have left the conversation? + GenericConvIMPrototype.close.call(this); + }, + unInit() { + this._account.removeConversation(this.normalizedName); + GenericConvIMPrototype.unInit.call(this); + }, +}; + +// Creates XMPP conversation. +function XMPPConversation(aAccount, aNormalizedName, aMucParticipant) { + this._init(aAccount, aNormalizedName); + if (aMucParticipant) { + this._isMucParticipant = true; + } +} +XMPPConversation.prototype = XMPPConversationPrototype; + +/* Helper class for buddies */ +export var XMPPAccountBuddyPrototype = { + __proto__: GenericAccountBuddyPrototype, + + subscription: "none", + // Returns a list of TooltipInfo objects to be displayed when the user + // hovers over the buddy. + getTooltipInfo() { + if (!this._account.connected) { + return null; + } + + let tooltipInfo = []; + if (this._resources) { + for (let r in this._resources) { + let status = this._resources[r]; + let statusString = Status.toLabel(status.statusType); + if ( + status.statusType == Ci.imIStatusInfo.STATUS_IDLE && + status.idleSince + ) { + let now = Math.floor(Date.now() / 1000); + let valuesAndUnits = lazy.DownloadUtils.convertTimeUnits( + now - status.idleSince + ); + if (!valuesAndUnits[2]) { + valuesAndUnits.splice(2, 2); + } + statusString += " (" + valuesAndUnits.join(" ") + ")"; + } + if (status.statusText) { + statusString += " - " + status.statusText; + } + let label = r + ? lazy._("tooltip.status", r) + : lazy._("tooltip.statusNoResource"); + tooltipInfo.push(new TooltipInfo(label, statusString)); + } + } + + // The subscription value is interesting to display only in unusual cases. + if (this.subscription != "both") { + tooltipInfo.push( + new TooltipInfo(lazy._("tooltip.subscription"), this.subscription) + ); + } + + return tooltipInfo; + }, + + // _rosterAlias is the value stored in the roster on the XMPP + // server. For most servers we will be read/write. + _rosterAlias: "", + set rosterAlias(aNewAlias) { + let old = this.displayName; + this._rosterAlias = aNewAlias; + if (old != this.displayName) { + this._notifyObservers("display-name-changed", old); + } + }, + _vCardReceived: false, + // _vCardFormattedName is the display name the contact has set for + // himself in his vCard. It's read-only from our point of view. + _vCardFormattedName: "", + set vCardFormattedName(aNewFormattedName) { + let old = this.displayName; + this._vCardFormattedName = aNewFormattedName; + if (old != this.displayName) { + this._notifyObservers("display-name-changed", old); + } + }, + + // _serverAlias is set by jsProtoHelper to the value we cached in sqlite. + // Use it only if we have neither of the other two values; usually because + // we haven't connected to the server yet. + get serverAlias() { + return this._rosterAlias || this._vCardFormattedName || this._serverAlias; + }, + set serverAlias(aNewAlias) { + if (!this._rosterItem) { + this.ERROR( + "attempting to update the server alias of an account buddy " + + "for which we haven't received a roster item." + ); + return; + } + + let item = this._rosterItem; + if (aNewAlias) { + item.attributes.name = aNewAlias; + } else if ("name" in item.attributes) { + delete item.attributes.name; + } + + let s = Stanza.iq( + "set", + null, + null, + Stanza.node("query", Stanza.NS.roster, null, item) + ); + this._account.sendStanza(s); + + // If we are going to change the alias on the server, discard the cached + // value that we got from our local sqlite storage at startup. + delete this._serverAlias; + }, + + /* Display name of the buddy */ + get contactDisplayName() { + return this.buddy.contact.displayName || this.displayName; + }, + + get tag() { + return this._tag; + }, + set tag(aNewTag) { + let oldTag = this._tag; + if (oldTag.name == aNewTag.name) { + this.ERROR("attempting to set the tag to the same value"); + return; + } + + this._tag = aNewTag; + IMServices.contacts.accountBuddyMoved(this, oldTag, aNewTag); + + if (!this._rosterItem) { + this.ERROR( + "attempting to change the tag of an account buddy without roster item" + ); + return; + } + + let item = this._rosterItem; + let oldXML = item.getXML(); + // Remove the old tag if it was listed in the roster item. + item.children = item.children.filter( + c => c.qName != "group" || c.innerText != oldTag.name + ); + // Ensure the new tag is listed. + let newTagName = aNewTag.name; + if (!item.getChildren("group").some(g => g.innerText == newTagName)) { + item.addChild(Stanza.node("group", null, null, newTagName)); + } + // Avoid sending anything to the server if the roster item hasn't changed. + // It's possible that the roster item hasn't changed if the roster + // item had several groups and the user moved locally the contact + // to another group where it already was on the server. + if (item.getXML() == oldXML) { + return; + } + + let s = Stanza.iq( + "set", + null, + null, + Stanza.node("query", Stanza.NS.roster, null, item) + ); + this._account.sendStanza(s); + }, + + remove() { + if (!this._account.connected) { + return; + } + + let s = Stanza.iq( + "set", + null, + null, + Stanza.node( + "query", + Stanza.NS.roster, + null, + Stanza.node("item", null, { + jid: this.normalizedName, + subscription: "remove", + }) + ) + ); + this._account.sendStanza(s); + }, + + _photoHash: null, + _saveIcon(aPhotoNode) { + this._account._saveResourceIcon(aPhotoNode, this).then( + url => { + this.buddyIconFilename = url; + }, + error => { + this._account.WARN( + "Error loading buddy icon for " + + this.normalizedName + + ": " + + error.message + ); + } + ); + }, + + _preferredResource: undefined, + _resources: null, + onAccountDisconnected() { + delete this._preferredResource; + delete this._resources; + }, + // Called by the account when a presence stanza is received for this buddy. + onPresenceStanza(aStanza) { + let preferred = this._preferredResource; + + // Facebook chat's XMPP server doesn't send resources, let's + // replace undefined resources with empty resources. + let resource = + this._account._parseJID(aStanza.attributes.from).resource || ""; + + let type = aStanza.attributes.type; + + // Reset typing status if the buddy is in a conversation and becomes unavailable. + let conv = this._account._conv.get(this.normalizedName); + if (type == "unavailable" && conv) { + conv.updateTyping(Ci.prplIConvIM.NOT_TYPING, this.contactDisplayName); + } + + if (type == "unavailable" || type == "error") { + if (!this._resources || !(resource in this._resources)) { + // Ignore for already offline resources. + return; + } + delete this._resources[resource]; + if (preferred == resource) { + preferred = undefined; + } + } else { + let statusInfo = parseStatus(aStanza); + let priority = aStanza.getElement(["priority"]); + priority = priority ? parseInt(priority.innerText, 10) : 0; + + if (!this._resources) { + this._resources = {}; + } + this._resources[resource] = { + statusType: statusInfo.statusType, + statusText: statusInfo.statusText, + idleSince: statusInfo.idleSince, + priority, + stanza: aStanza, + }; + } + + let photo = aStanza.getElement(["x", "photo"]); + if (photo && photo.uri == Stanza.NS.vcard_update) { + let hash = photo.innerText; + if (hash && hash != this._photoHash) { + this._account._addVCardRequest(this.normalizedName); + } else if (!hash && this._photoHash) { + delete this._photoHash; + this.buddyIconFilename = ""; + } + } + + for (let r in this._resources) { + if ( + preferred === undefined || + this._resources[r].statusType > this._resources[preferred].statusType + ) { + // FIXME also compare priorities... + preferred = r; + } + } + if ( + preferred != undefined && + preferred == this._preferredResource && + resource != preferred + ) { + // The presence information change is only for an unused resource, + // only potential buddy tooltips need to be refreshed. + this._notifyObservers("status-detail-changed"); + return; + } + + // Presence info has changed enough that if we are having a + // conversation with one resource of this buddy, we should send + // the next message to all resources. + // FIXME: the test here isn't exactly right... + if ( + this._preferredResource != preferred && + this._account._conv.has(this.normalizedName) + ) { + delete this._account._conv.get(this.normalizedName)._targetResource; + } + + this._preferredResource = preferred; + if (preferred === undefined) { + let statusType = Ci.imIStatusInfo.STATUS_UNKNOWN; + if (type == "unavailable") { + statusType = Ci.imIStatusInfo.STATUS_OFFLINE; + } + this.setStatus(statusType, ""); + } else { + preferred = this._resources[preferred]; + this.setStatus(preferred.statusType, preferred.statusText); + } + }, + + /* Can send messages to buddies who appear offline */ + get canSendMessage() { + return this.account.connected; + }, + + /* Called when the user wants to chat with the buddy */ + createConversation() { + return this._account.createConversation(this.normalizedName); + }, +}; + +function XMPPAccountBuddy(aAccount, aBuddy, aTag, aUserName) { + this._init(aAccount, aBuddy, aTag, aUserName); +} +XMPPAccountBuddy.prototype = XMPPAccountBuddyPrototype; + +var XMPPRoomInfoPrototype = { + __proto__: ClassInfo("prplIRoomInfo", "XMPP RoomInfo Object"), + get topic() { + return ""; + }, + get participantCount() { + return Ci.prplIRoomInfo.NO_PARTICIPANT_COUNT; + }, + get chatRoomFieldValues() { + let roomJid = this._account._roomList.get(this.name); + return this._account.getChatRoomDefaultFieldValues(roomJid); + }, +}; +function XMPPRoomInfo(aName, aAccount) { + this.name = aName; + this._account = aAccount; +} +XMPPRoomInfo.prototype = XMPPRoomInfoPrototype; + +/* Helper class for account */ +export var XMPPAccountPrototype = { + __proto__: GenericAccountPrototype, + + _jid: null, // parsed Jabber ID: node, domain, resource + _connection: null, // XMPPSession socket + authMechanisms: null, // hook to let prpls tweak the list of auth mechanisms + + // Contains the domain of MUC service which is obtained using service + // discovery. + _mucService: null, + + // Maps room names to room jid. + _roomList: new Map(), + + // Callbacks used when roomInfo is available. + _roomInfoCallbacks: new Set(), + + // Determines if roomInfo that we have is expired or not. + _lastListTime: 0, + get isRoomInfoStale() { + return Date.now() - this._lastListTime > kListRefreshInterval; + }, + + // If true, we are waiting for replies. + _pendingList: false, + + // An array of jids for which we still need to request vCards. + _pendingVCardRequests: [], + + // XEP-0280: Message Carbons. + // If true, message carbons are currently enabled. + _isCarbonsEnabled: false, + + /* Generate unique id for a stanza. Using id and unique sid is defined in + * RFC 6120 (Section 8.2.3, 4.7.3). + */ + generateId: () => Services.uuid.generateUUID().toString().slice(1, -1), + + _init(aProtoInstance, aImAccount) { + GenericAccountPrototype._init.call(this, aProtoInstance, aImAccount); + + // Ongoing conversations. + // The keys of this._conv are assumed to be normalized like account@domain + // for normal conversations and like room@domain/nick for MUC participant + // convs. + this._conv = new NormalizedMap(this.normalizeFullJid.bind(this)); + + this._buddies = new NormalizedMap(this.normalize.bind(this)); + this._mucs = new NormalizedMap(this.normalize.bind(this)); + + this._pendingVCardRequests = []; + }, + + get canJoinChat() { + return true; + }, + chatRoomFields: { + room: { + get label() { + return lazy._("chatRoomField.room"); + }, + required: true, + }, + server: { + get label() { + return lazy._("chatRoomField.server"); + }, + required: true, + }, + nick: { + get label() { + return lazy._("chatRoomField.nick"); + }, + required: true, + }, + password: { + get label() { + return lazy._("chatRoomField.password"); + }, + isPassword: true, + }, + }, + parseDefaultChatName(aDefaultChatName) { + if (!aDefaultChatName) { + return { nick: this._jid.node }; + } + + let params = aDefaultChatName.trim().split(/\s+/); + let jid = this._parseJID(params[0]); + + // We swap node and domain as domain is required for parseJID, but node and + // resource are optional. In MUC join command, Node is required as it + // represents a room, but domain and resource are optional as we get muc + // domain from service discovery. + if (!jid.node && jid.domain) { + [jid.node, jid.domain] = [jid.domain, jid.node]; + } + + let chatFields = { + room: jid.node, + server: jid.domain || this._mucService, + nick: jid.resource || this._jid.node, + }; + if (params.length > 1) { + chatFields.password = params[1]; + } + return chatFields; + }, + getChatRoomDefaultFieldValues(aDefaultChatName) { + let rv = GenericAccountPrototype.getChatRoomDefaultFieldValues.call( + this, + aDefaultChatName + ); + if (!rv.values.nick) { + rv.values.nick = this._jid.node; + } + if (!rv.values.server && this._mucService) { + rv.values.server = this._mucService; + } + + return rv; + }, + + // XEP-0045: Requests joining room if it exists or + // creating room if it does not exist. + joinChat(aComponents) { + let jid = + aComponents.getValue("room") + "@" + aComponents.getValue("server"); + let nick = aComponents.getValue("nick"); + + let muc = this._mucs.get(jid); + if (muc) { + if (!muc.left) { + // We are already in this conversation. + return muc; + } else if (!muc.chatRoomFields) { + // We are rejoining a room that was parted by the user. + muc._rejoined = true; + } + } else { + muc = new this._MUCConversationConstructor(this, jid, nick); + this._mucs.set(jid, muc); + } + + // Store the prplIChatRoomFieldValues to enable later reconnections. + muc.chatRoomFields = aComponents; + muc.joining = true; + muc.removeAllParticipants(); + + let password = aComponents.getValue("password"); + let x = Stanza.node( + "x", + Stanza.NS.muc, + null, + password ? Stanza.node("password", null, null, password) : null + ); + let logString; + if (password) { + logString = + "<presence .../> (Stanza containing password to join MUC " + + jid + + "/" + + nick + + " not logged)"; + } + this.sendStanza( + Stanza.presence({ to: jid + "/" + nick }, x), + undefined, + undefined, + logString + ); + return muc; + }, + + _idleSince: 0, + observe(aSubject, aTopic, aData) { + if (aTopic == "idle-time-changed") { + let idleTime = parseInt(aData, 10); + if (idleTime) { + this._idleSince = Math.floor(Date.now() / 1000) - idleTime; + } else { + delete this._idleSince; + } + this._shouldSendPresenceForIdlenessChange = true; + executeSoon( + function () { + if ("_shouldSendPresenceForIdlenessChange" in this) { + this._sendPresence(); + } + }.bind(this) + ); + } else if (aTopic == "status-changed") { + this._sendPresence(); + } else if (aTopic == "user-icon-changed") { + delete this._cachedUserIcon; + this._forceUserIconUpdate = true; + this._sendVCard(); + } else if (aTopic == "user-display-name-changed") { + this._forceUserDisplayNameUpdate = true; + } + this._sendVCard(); + }, + + /* GenericAccountPrototype events */ + /* Connect to the server */ + connect() { + this._jid = this._parseJID(this.name); + + // For the resource, if the user has edited the option, always use that. + if (this.prefs.prefHasUserValue("resource")) { + let resource = this.getString("resource"); + + // this._jid needs to be updated. This value is however never used + // because while connected it's the jid of the session that's + // interesting. + this._jid = this._setJID(this._jid.domain, this._jid.node, resource); + } else if (this._jid.resource) { + // If there is a resource in the account name (inherited from libpurple), + // migrate it to the pref so it appears correctly in the advanced account + // options next time. + this.prefs.setStringPref("resource", this._jid.resource); + } + + this._connection = new XMPPSession( + this.getString("server") || this._jid.domain, + this.getInt("port") || 5222, + this.getString("connection_security"), + this._jid, + this.imAccount.password, + this + ); + }, + + remove() { + this._conv.forEach(conv => conv.close()); + this._mucs.forEach(muc => muc.close()); + this._buddies.forEach((buddy, jid) => this._forgetRosterItem(jid)); + }, + + unInit() { + if (this._connection) { + this._disconnect(undefined, undefined, true); + } + delete this._jid; + delete this._conv; + delete this._buddies; + delete this._mucs; + }, + + /* Disconnect from the server */ + disconnect() { + this._disconnect(); + }, + + addBuddy(aTag, aName) { + if (!this._connection) { + throw new Error("The account isn't connected"); + } + + let jid = this.normalize(aName); + if (!jid || !jid.includes("@")) { + throw new Error("Invalid username"); + } + + if (this._buddies.has(jid)) { + let subscription = this._buddies.get(jid).subscription; + if (subscription && (subscription == "both" || subscription == "to")) { + this.DEBUG("not re-adding an existing buddy"); + return; + } + } else { + let s = Stanza.iq( + "set", + null, + null, + Stanza.node( + "query", + Stanza.NS.roster, + null, + Stanza.node( + "item", + null, + { jid }, + Stanza.node("group", null, null, aTag.name) + ) + ) + ); + this.sendStanza( + s, + this._handleResult({ + default: aError => { + this.WARN( + "Unable to add a roster item due to " + aError + " error." + ); + }, + }) + ); + } + this.sendStanza(Stanza.presence({ to: jid, type: "subscribe" })); + }, + + /* Loads a buddy from the local storage. + * Called for each buddy locally stored before connecting + * to the server. */ + loadBuddy(aBuddy, aTag) { + let buddy = new this._accountBuddyConstructor(this, aBuddy, aTag); + this._buddies.set(buddy.normalizedName, buddy); + return buddy; + }, + + /* Replies to a buddy request in order to accept it or deny it. */ + replyToBuddyRequest(aReply, aRequest) { + if (!this._connection) { + return; + } + let s = Stanza.presence({ to: aRequest.userName, type: aReply }); + this.sendStanza(s); + this.removeBuddyRequest(aRequest); + }, + + requestBuddyInfo(aJid) { + if (!this.connected) { + Services.obs.notifyObservers(EmptyEnumerator, "user-info-received", aJid); + return; + } + + let userName; + let tooltipInfo = []; + let jid = this._parseJID(aJid); + let muc = this._mucs.get(jid.node + "@" + jid.domain); + let participant; + if (muc) { + participant = muc._participants.get(jid.resource); + if (participant) { + if (participant.accountJid) { + userName = participant.accountJid; + } + if (!muc.left) { + let statusType = participant.statusType; + let statusText = participant.statusText; + tooltipInfo.push( + new TooltipInfo(statusType, statusText, Ci.prplITooltipInfo.status) + ); + + if (participant.buddyIconFilename) { + tooltipInfo.push( + new TooltipInfo( + null, + participant.buddyIconFilename, + Ci.prplITooltipInfo.icon + ) + ); + } + } + } + } + Services.obs.notifyObservers( + new nsSimpleEnumerator(tooltipInfo), + "user-info-received", + aJid + ); + + let iq = Stanza.iq( + "get", + null, + aJid, + Stanza.node("vCard", Stanza.NS.vcard) + ); + this.sendStanza(iq, aStanza => { + let vCardInfo = {}; + let vCardNode = aStanza.getElement(["vCard"]); + + // In the case of an error response, we just notify the observers with + // what info we already have. + if (aStanza.attributes.type == "result" && vCardNode) { + vCardInfo = this.parseVCard(vCardNode); + } + + // The real jid of participant which is of the form local@domain/resource. + // We consider the jid is provided by server is more correct than jid is + // set by the user. + if (userName) { + vCardInfo.userName = userName; + } + + // vCard fields we want to display in the tooltip. + const kTooltipFields = [ + "userName", + "fullName", + "nickname", + "title", + "organization", + "email", + "birthday", + "locality", + "country", + "telephone", + ]; + + let tooltipInfo = []; + for (let field of kTooltipFields) { + if (vCardInfo.hasOwnProperty(field)) { + tooltipInfo.push( + new TooltipInfo(lazy._("tooltip." + field), vCardInfo[field]) + ); + } + } + if (vCardInfo.photo) { + let dataURI = this._getPhotoURI(vCardInfo.photo); + + // Store the photo URI for this participant. + if (participant) { + participant.buddyIconFilename = dataURI; + } + + tooltipInfo.push( + new TooltipInfo(null, dataURI, Ci.prplITooltipInfo.icon) + ); + } + Services.obs.notifyObservers( + new nsSimpleEnumerator(tooltipInfo), + "user-info-received", + aJid + ); + }); + }, + + // Parses the photo node of a received vCard if exists and returns string of + // data URI, otherwise returns null. + _getPhotoURI(aPhotoNode) { + if (!aPhotoNode) { + return null; + } + + let type = aPhotoNode.getElement(["TYPE"]); + let value = aPhotoNode.getElement(["BINVAL"]); + if (!type || !value) { + return null; + } + + return "data:" + type.innerText + ";base64," + value.innerText; + }, + + // Parses the vCard into the properties of the returned object. + parseVCard(aVCardNode) { + // XEP-0054: vcard-temp. + let aResult = {}; + for (let node of aVCardNode.children.filter( + child => child.type == "node" + )) { + let localName = node.localName; + let innerText = node.innerText; + if (innerText) { + if (localName == "FN") { + aResult.fullName = innerText; + } else if (localName == "NICKNAME") { + aResult.nickname = innerText; + } else if (localName == "TITLE") { + aResult.title = innerText; + } else if (localName == "BDAY") { + aResult.birthday = innerText; + } else if (localName == "JABBERID") { + aResult.userName = innerText; + } + } + if (localName == "ORG") { + let organization = node.getElement(["ORGNAME"]); + if (organization && organization.innerText) { + aResult.organization = organization.innerText; + } + } else if (localName == "EMAIL") { + let userID = node.getElement(["USERID"]); + if (userID && userID.innerText) { + aResult.email = userID.innerText; + } + } else if (localName == "ADR") { + let locality = node.getElement(["LOCALITY"]); + if (locality && locality.innerText) { + aResult.locality = locality.innerText; + } + + let country = node.getElement(["CTRY"]); + if (country && country.innerText) { + aResult.country = country.innerText; + } + } else if (localName == "PHOTO") { + aResult.photo = node; + } else if (localName == "TEL") { + let number = node.getElement(["NUMBER"]); + if (number && number.innerText) { + aResult.telephone = number.innerText; + } + } + // TODO: Parse the other fields of vCard and display it in system messages + // in response to /whois. + } + return aResult; + }, + + // Returns undefined if not an error stanza, and an object + // describing the error otherwise: + parseError(aStanza) { + if (aStanza.attributes.type != "error") { + return undefined; + } + + let retval = { stanza: aStanza }; + let error = aStanza.getElement(["error"]); + + // RFC 6120 Section 8.3.2: Type must be one of + // auth -- retry after providing credentials + // cancel -- do not retry (the error cannot be remedied) + // continue -- proceed (the condition was only a warning) + // modify -- retry after changing the data sent + // wait -- retry after waiting (the error is temporary). + retval.type = error.attributes.type; + + // RFC 6120 Section 8.3.3. + const kDefinedConditions = [ + "bad-request", + "conflict", + "feature-not-implemented", + "forbidden", + "gone", + "internal-server-error", + "item-not-found", + "jid-malformed", + "not-acceptable", + "not-allowed", + "not-authorized", + "policy-violation", + "recipient-unavailable", + "redirect", + "registration-required", + "remote-server-not-found", + "remote-server-timeout", + "resource-constraint", + "service-unavailable", + "subscription-required", + "undefined-condition", + "unexpected-request", + ]; + let condition = kDefinedConditions.find(c => error.getElement([c])); + if (!condition) { + // RFC 6120 Section 8.3.2. + this.WARN( + "Nonstandard or missing defined-condition element in error stanza." + ); + condition = "undefined-condition"; + } + retval.condition = condition; + + let errortext = error.getElement(["text"]); + if (errortext) { + retval.text = errortext.innerText; + } + + return retval; + }, + + // Returns an error-handling callback for use with sendStanza generated + // from aHandlers, an object defining the error handlers. + // If the stanza passed to the callback is an error stanza, it checks if + // aHandlers contains a property with the name of the defined condition + // of the error. + // * If the property is a function, it is called with the parsed error + // as its argument, bound to aThis (if provided). + // It should return true if the error was handled. + // * If the property is a string, it is displayed as a system message + // in the conversation given by aThis. + handleErrors(aHandlers, aThis) { + return aStanza => { + if (!aHandlers) { + return false; + } + + let error = this.parseError(aStanza); + if (!error) { + return false; + } + + let toCamelCase = aStr => { + // Turn defined condition string into a valid camelcase + // JS property name. + let capitalize = s => s[0].toUpperCase() + s.slice(1); + let uncapitalize = s => s[0].toLowerCase() + s.slice(1); + return uncapitalize(aStr.split("-").map(capitalize).join("")); + }; + let condition = toCamelCase(error.condition); + // Check if we have a handler property for this kind of error or a + // default handler. + if (!(condition in aHandlers) && !("default" in aHandlers)) { + return false; + } + + // Try to get the handler for condition, if we cannot get it, try to get + // the default handler. + let handler = aHandlers[condition]; + if (!handler) { + handler = aHandlers.default; + } + + if (typeof handler == "string") { + // The string is an error message to be displayed in the conversation. + if (!aThis || !aThis.writeMessage) { + this.ERROR( + "HandleErrors was passed an error message string, but " + + "no conversation to display it in:\n" + + handler + ); + return true; + } + aThis.writeMessage(aThis.name, handler, { system: true, error: true }); + return true; + } else if (typeof handler == "function") { + // If we're given a function, call this error handler. + return handler.call(aThis, error); + } + + // If this happens, there's a bug somewhere. + this.ERROR( + "HandleErrors was passed a handler for '" + + condition + + "'' which is neither a function nor a string." + ); + return false; + }; + }, + + // Returns a callback suitable for use in sendStanza, to handle type==result + // responses. aHandlers and aThis are passed on to handleErrors for error + // handling. + _handleResult(aHandlers, aThis) { + return aStanza => { + if (aStanza.attributes.type == "result") { + return true; + } + return this.handleErrors(aHandlers, aThis)(aStanza); + }; + }, + + /* XMPPSession events */ + + /* Called when the XMPP session is started */ + onConnection() { + // Request the roster. The account will be marked as connected when this is + // complete. + this.reportConnecting(lazy._("connection.downloadingRoster")); + let s = Stanza.iq( + "get", + null, + null, + Stanza.node("query", Stanza.NS.roster) + ); + this.sendStanza(s, this.onRoster, this); + + // XEP-0030 and XEP-0045 (6): Service Discovery. + // Queries Server for Associated Services. + let iq = Stanza.iq( + "get", + null, + this._jid.domain, + Stanza.node("query", Stanza.NS.disco_items) + ); + this.sendStanza(iq, this.onServiceDiscovery, this); + + // XEP-0030: Service Discovery Information Features. + iq = Stanza.iq( + "get", + null, + this._jid.domain, + Stanza.node("query", Stanza.NS.disco_info) + ); + this.sendStanza(iq, this.onServiceDiscoveryInfo, this); + }, + + /* Called whenever a stanza is received */ + onXmppStanza(aStanza) {}, + + /* Called when a iq stanza is received */ + onIQStanza(aStanza) { + let type = aStanza.attributes.type; + if (type == "set") { + for (let query of aStanza.getChildren("query")) { + if (query.uri != Stanza.NS.roster) { + continue; + } + + // RFC 6121 2.1.6 (Roster push): + // A receiving client MUST ignore the stanza unless it has no 'from' + // attribute (i.e., implicitly from the bare JID of the user's + // account) or it has a 'from' attribute whose value matches the + // user's bare JID <user@domainpart>. + let from = aStanza.attributes.from; + if (from && from != this._jid.node + "@" + this._jid.domain) { + this.WARN("Ignoring potentially spoofed roster push."); + return; + } + + for (let item of query.getChildren("item")) { + this._onRosterItem(item, true); + } + return; + } + } else if (type == "get") { + let id = aStanza.attributes.id; + let from = aStanza.attributes.from; + + // XEP-0199: XMPP server-to-client ping (XEP-0199) + let ping = aStanza.getElement(["ping"]); + if (ping && ping.uri == Stanza.NS.ping) { + this.sendStanza(Stanza.iq("result", id, from)); + return; + } + + let query = aStanza.getElement(["query"]); + if (query && query.uri == Stanza.NS.version) { + // XEP-0092: Software Version. + let children = []; + children.push(Stanza.node("name", null, null, Services.appinfo.name)); + children.push( + Stanza.node("version", null, null, Services.appinfo.version) + ); + let versionQuery = Stanza.node( + "query", + Stanza.NS.version, + null, + children + ); + this.sendStanza(Stanza.iq("result", id, from, versionQuery)); + return; + } + if (query && query.uri == Stanza.NS.disco_info) { + // XEP-0030: Service Discovery. + let children = []; + if (aStanza.attributes.node == Stanza.NS.muc_rooms) { + // XEP-0045 (6.7): Room query. + // TODO: Currently, we return an empty <query/> element, but we + // should return non-private rooms. + } else { + children = SupportedFeatures.map(feature => + Stanza.node("feature", null, { var: feature }) + ); + children.unshift( + Stanza.node("identity", null, { + category: "client", + type: "pc", + name: Services.appinfo.name, + }) + ); + } + let discoveryQuery = Stanza.node( + "query", + Stanza.NS.disco_info, + null, + children + ); + this.sendStanza(Stanza.iq("result", id, from, discoveryQuery)); + return; + } + } + this.WARN(`Unhandled IQ ${type} stanza.`); + if (type == "get" || type == "set") { + // RFC 6120 (section 8.2.3): An entity that receives an IQ request of + // type "get" or "set" MUST reply with an IQ response of type "result" + // or "error". + let id = aStanza.attributes.id; + let from = aStanza.attributes.from; + let condition = Stanza.node("service-unavailable", Stanza.NS.stanzas, { + type: "cancel", + }); + let error = Stanza.node("error", null, { type: "cancel" }, condition); + this.sendStanza(Stanza.iq("error", id, from, error)); + } + }, + + /* Called when a presence stanza is received */ + onPresenceStanza(aStanza) { + let from = aStanza.attributes.from; + this.DEBUG("Received presence stanza for " + from); + + let jid = this.normalize(from); + let type = aStanza.attributes.type; + if (type == "subscribe") { + this.addBuddyRequest( + jid, + this.replyToBuddyRequest.bind(this, "subscribed"), + this.replyToBuddyRequest.bind(this, "unsubscribed") + ); + } else if ( + type == "unsubscribe" || + type == "unsubscribed" || + type == "subscribed" + ) { + // Nothing useful to do for these presence stanzas, as we will also + // receive a roster push containing more or less the same information + } else if (this._buddies.has(jid)) { + this._buddies.get(jid).onPresenceStanza(aStanza); + } else if (this._mucs.has(jid)) { + this._mucs.get(jid).onPresenceStanza(aStanza); + } else if (jid != this.normalize(this._connection._jid.jid)) { + this.WARN("received presence stanza for unknown buddy " + from); + } else if ( + jid == this._jid.node + "@" + this._jid.domain && + this._connection._resource != this._parseJID(from).resource + ) { + // Ignore presence stanzas for another resource. + } else { + this.WARN("Unhandled presence stanza."); + } + }, + + // XEP-0030: Discovering services and their features that are supported by + // the server. + onServiceDiscovery(aStanza) { + let query = aStanza.getElement(["query"]); + if ( + aStanza.attributes.type != "result" || + !query || + query.uri != Stanza.NS.disco_items + ) { + this.LOG("Could not get services for this server: " + this._jid.domain); + return true; + } + + // Discovering the Features that are Supported by each service. + query.getElements(["item"]).forEach(item => { + let jid = item.attributes.jid; + if (!jid) { + return; + } + let iq = Stanza.iq( + "get", + null, + jid, + Stanza.node("query", Stanza.NS.disco_info) + ); + this.sendStanza(iq, receivedStanza => { + let query = receivedStanza.getElement(["query"]); + let from = receivedStanza.attributes.from; + if ( + aStanza.attributes.type != "result" || + !query || + query.uri != Stanza.NS.disco_info + ) { + this.LOG("Could not get features for this service: " + from); + return true; + } + let features = query + .getElements(["feature"]) + .map(elt => elt.attributes.var); + let identity = query.getElement(["identity"]); + if ( + identity && + identity.attributes.category == "conference" && + identity.attributes.type == "text" && + features.includes(Stanza.NS.muc) + ) { + // XEP-0045 (6.2): this feature is for a MUC Service. + // XEP-0045 (15.2): Service Discovery Category/Type. + this._mucService = from; + } + // TODO: Handle other services that are supported by XMPP through + // their features. + + return true; + }); + }); + return true; + }, + + // XEP-0030: Discovering Service Information and its features that are + // supported by the server. + onServiceDiscoveryInfo(aStanza) { + let query = aStanza.getElement(["query"]); + if ( + aStanza.attributes.type != "result" || + !query || + query.uri != Stanza.NS.disco_info + ) { + this.LOG("Could not get features for this server: " + this._jid.domain); + return true; + } + + let features = query + .getElements(["feature"]) + .map(elt => elt.attributes.var); + if (features.includes(Stanza.NS.carbons)) { + // XEP-0280: Message Carbons. + // Enabling Carbons on server, as it's disabled by default on server. + if (Services.prefs.getBoolPref("chat.xmpp.messageCarbons")) { + let iqStanza = Stanza.iq( + "set", + null, + null, + Stanza.node("enable", Stanza.NS.carbons) + ); + this.sendStanza(iqStanza, aStanza => { + let error = this.parseError(aStanza); + if (error) { + this.WARN( + "Unable to enable message carbons due to " + + error.condition + + " error." + ); + return true; + } + + let type = aStanza.attributes.type; + if (type != "result") { + this.WARN( + "Received unexpected stanza with " + + type + + " type " + + "while enabling message carbons." + ); + return true; + } + + this.LOG("Message carbons enabled."); + this._isCarbonsEnabled = true; + return true; + }); + } + } + // TODO: Handle other features that are supported by the server. + return true; + }, + + requestRoomInfo(aCallback) { + if (this._roomInfoCallbacks.has(aCallback)) { + return; + } + + if (this.isRoomInfoStale && !this._pendingList) { + this._roomList = new Map(); + this._lastListTime = Date.now(); + this._roomInfoCallback = aCallback; + this._pendingList = true; + + // XEP-0045 (6.3): Discovering Rooms. + let iq = Stanza.iq( + "get", + null, + this._mucService, + Stanza.node("query", Stanza.NS.disco_items) + ); + this.sendStanza(iq, this.onRoomDiscovery, this); + } else { + let rooms = [...this._roomList.keys()]; + aCallback.onRoomInfoAvailable(rooms, !this._pendingList); + } + + if (this._pendingList) { + this._roomInfoCallbacks.add(aCallback); + } + }, + + onRoomDiscovery(aStanza) { + let query = aStanza.getElement(["query"]); + if (!query || query.uri != Stanza.NS.disco_items) { + this.LOG("Could not get rooms for this server: " + this._jid.domain); + return; + } + + // XEP-0059: Result Set Management. + let set = query.getElement(["set"]); + let last = set ? set.getElement(["last"]) : null; + if (last) { + let iq = Stanza.iq( + "get", + null, + this._mucService, + Stanza.node("query", Stanza.NS.disco_items) + ); + this.sendStanza(iq, this.onRoomDiscovery, this); + } else { + this._pendingList = false; + } + + let rooms = []; + query.getElements(["item"]).forEach(item => { + let jid = this._parseJID(item.attributes.jid); + if (!jid) { + return; + } + + let name = item.attributes.name; + if (!name) { + name = jid.node ? jid.node : jid.jid; + } + + this._roomList.set(name, jid.jid); + rooms.push(name); + }); + + this._roomInfoCallback.onRoomInfoAvailable(rooms, !this._pendingList); + }, + + getRoomInfo(aName) { + return new XMPPRoomInfo(aName, this); + }, + + // Returns null if not an invitation stanza, and an object + // describing the invitation otherwise. + parseInvitation(aStanza) { + let x = aStanza.getElement(["x"]); + if (!x) { + return null; + } + let retVal = { + shouldDecline: false, + }; + + // XEP-0045. Direct Invitation (7.8.1) + // Described in XEP-0249. + // jid (chatroom) is required. + // Password, reason, continue and thread are optional. + if (x.uri == Stanza.NS.conference) { + if (!x.attributes.jid) { + this.WARN("Received an invitation with missing MUC jid."); + return null; + } + retVal.mucJid = this.normalize(x.attributes.jid); + retVal.from = this.normalize(aStanza.attributes.from); + retVal.password = x.attributes.password; + retVal.reason = x.attributes.reason; + retVal.continue = x.attributes.continue; + retVal.thread = x.attributes.thread; + return retVal; + } + + // XEP-0045. Mediated Invitation (7.8.2) + // Sent by the chatroom on behalf of someone in the chatroom. + // jid (chatroom) and from (inviter) are required. + // password and reason are optional. + if (x.uri == Stanza.NS.muc_user) { + let invite = x.getElement(["invite"]); + if (!invite || !invite.attributes.from) { + this.WARN("Received an invitation with missing MUC invite or from."); + return null; + } + retVal.mucJid = this.normalize(aStanza.attributes.from); + retVal.from = this.normalize(invite.attributes.from); + retVal.shouldDecline = true; + let continueElement = invite.getElement(["continue"]); + retVal.continue = !!continueElement; + if (continueElement) { + retVal.thread = continueElement.attributes.thread; + } + if (x.getElement(["password"])) { + retVal.password = x.getElement(["password"]).innerText; + } + if (invite.getElement(["reason"])) { + retVal.reason = invite.getElement(["reason"]).innerText; + } + return retVal; + } + + return null; + }, + + /* Called when a message stanza is received */ + onMessageStanza(aStanza) { + // XEP-0280: Message Carbons. + // Sending and Receiving Messages. + // Indicates that the forwarded message was sent or received. + let isSent = false; + let carbonStanza = + aStanza.getElement(["sent"]) || aStanza.getElement(["received"]); + if (carbonStanza) { + if (carbonStanza.uri != Stanza.NS.carbons) { + this.WARN( + "Received a forwarded message which does not '" + + Stanza.NS.carbons + + "' namespace." + ); + return; + } + + isSent = carbonStanza.localName == "sent"; + carbonStanza = carbonStanza.getElement(["forwarded", "message"]); + if (this._isCarbonsEnabled) { + aStanza = carbonStanza; + } else { + this.WARN( + "Received an unexpected forwarded message while message " + + "carbons are not enabled." + ); + return; + } + } + + // For forwarded sent messages, we need to use "to" attribute to + // get the right conversation as from in this case is this account. + let convJid = isSent ? aStanza.attributes.to : aStanza.attributes.from; + + let normConvJid = this.normalize(convJid); + let isMuc = this._mucs.has(normConvJid); + + let type = aStanza.attributes.type; + let x = aStanza.getElement(["x"]); + let body; + let b = aStanza.getElement(["body"]); + if (b) { + // If there's a <body> child we have more than just typing notifications. + // Prefer HTML (in <html><body>) and use plain text (<body>) as fallback. + let htmlBody = aStanza.getElement(["html", "body"]); + if (htmlBody) { + body = htmlBody.innerXML; + } else { + // Even if the message is in plain text, the prplIMessage + // should contain a string that's correctly escaped for + // insertion in an HTML document. + body = lazy.TXTToHTML(b.innerText); + } + } + + let subject = aStanza.getElement(["subject"]); + // Ignore subject when !isMuc. We're being permissive about subject changes + // in the comment below, so we need to be careful about where that makes + // sense. Psi+'s OTR plugin includes a subject and body in its message + // stanzas. + if (subject && isMuc) { + // XEP-0045 (7.2.16): Check for a subject element in the stanza and update + // the topic if it exists. + // We are breaking the spec because only a message that contains a + // <subject/> but no <body/> element shall be considered a subject change + // for MUC, but we ignore that to be compatible with ejabberd versions + // before 15.06. + let muc = this._mucs.get(normConvJid); + let nick = this._parseJID(convJid).resource; + // TODO There can be multiple subject elements with different xml:lang + // attributes. + muc.setTopic(subject.innerText, nick); + return; + } + + let invitation = this.parseInvitation(aStanza); + if (invitation) { + let messageID; + if (invitation.reason) { + messageID = "conversation.muc.invitationWithReason2"; + } else { + messageID = "conversation.muc.invitationWithoutReason"; + } + if (invitation.password) { + messageID += ".password"; + } + let params = [ + invitation.from, + invitation.mucJid, + invitation.password, + invitation.reason, + ].filter(s => s); + let message = lazy._(messageID, ...params); + + this.addChatRequest( + invitation.mucJid, + () => { + let chatRoomFields = this.getChatRoomDefaultFieldValues( + invitation.mucJid + ); + if (invitation.password) { + chatRoomFields.setValue("password", invitation.password); + } + let muc = this.joinChat(chatRoomFields); + muc.writeMessage(muc.name, message, { system: true }); + }, + (request, tryToDeny) => { + // Only mediated invitations (XEP-0045) can explicitly decline. + if (invitation.shouldDecline && tryToDeny) { + let decline = Stanza.node( + "decline", + null, + { from: invitation.from }, + null + ); + let x = Stanza.node("x", Stanza.NS.muc_user, null, decline); + let s = Stanza.node("message", null, { to: invitation.mucJid }, x); + this.sendStanza(s); + } + // Always show invite reason or password, even if the invite wasn't + // automatically declined based on the setting. + if (!request || invitation.reason || invitation.password) { + let conv = this.createConversation(invitation.from); + if (conv) { + conv.writeMessage(invitation.from, message, { system: true }); + } + } + } + ); + } + + if (body) { + let date = _getDelay(aStanza); + if ( + type == "groupchat" || + (type == "error" && isMuc && !this._conv.has(convJid)) + ) { + if (!isMuc) { + this.WARN( + "Received a groupchat message for unknown MUC " + normConvJid + ); + return; + } + let muc = this._mucs.get(normConvJid); + muc.incomingMessage(body, aStanza, date); + return; + } + + let conv = this.createConversation(convJid); + if (!conv) { + return; + } + + if (isSent) { + _displaySentMsg(conv, body, date); + return; + } + conv.incomingMessage(body, aStanza, date); + } else if (type == "error") { + let conv = this.createConversation(convJid); + if (conv) { + conv.incomingMessage(null, aStanza); + } + } else if (x && x.uri == Stanza.NS.muc_user) { + let muc = this._mucs.get(normConvJid); + if (!muc) { + this.WARN( + "Received a groupchat message for unknown MUC " + normConvJid + ); + return; + } + muc.onMessageStanza(aStanza); + return; + } + + // If this is a sent message carbon, the user is typing on another client. + if (isSent) { + return; + } + + // Don't create a conversation to only display the typing notifications. + if (!this._conv.has(normConvJid) && !this._conv.has(convJid)) { + return; + } + + // Ignore errors while delivering typing notifications. + if (type == "error") { + return; + } + + let typingState = Ci.prplIConvIM.NOT_TYPING; + let state; + let s = aStanza.getChildrenByNS(Stanza.NS.chatstates); + if (s.length > 0) { + state = s[0].localName; + } + if (state) { + this.DEBUG(state); + if (state == "composing") { + typingState = Ci.prplIConvIM.TYPING; + } else if (state == "paused") { + typingState = Ci.prplIConvIM.TYPED; + } + } + let convName = normConvJid; + + // If the bare JID is a MUC that we have joined, use the full JID as this + // is a private message to a MUC participant. + if (isMuc) { + convName = convJid; + } + + let conv = this._conv.get(convName); + if (!conv) { + return; + } + conv.updateTyping(typingState, conv.shortName); + conv.supportChatStateNotifications = !!state; + }, + + /** Called when there is an error in the XMPP session */ + onError(aError, aException) { + if (aError === null || aError === undefined) { + aError = Ci.prplIAccount.ERROR_OTHER_ERROR; + } + this._disconnect(aError, aException.toString()); + }, + + onVCard(aStanza) { + let jid = this._pendingVCardRequests.shift(); + this._requestNextVCard(); + if (!this._buddies.has(jid) && !this._mucs.has(jid)) { + this.WARN("Received a vCard for unknown buddy " + jid); + return; + } + + let vCard = aStanza.getElement(["vCard"]); + let error = this.parseError(aStanza); + if ( + (error && + (error.condition == "item-not-found" || + error.condition == "service-unavailable")) || + !vCard || + !vCard.children.length + ) { + this.LOG("No vCard exists (or the user does not exist) for " + jid); + return; + } else if (error) { + this.WARN("Received unexpected vCard error " + error.condition); + return; + } + + let stanzaJid = this.normalize(aStanza.attributes.from); + if (jid && jid != stanzaJid) { + this.ERROR( + "Received vCard for a different jid (" + + stanzaJid + + ") " + + "than the requested " + + jid + ); + } + + let foundFormattedName = false; + let vCardInfo = this.parseVCard(vCard); + if (this._mucs.has(jid)) { + const conv = this._mucs.get(jid); + if (vCardInfo.photo) { + conv._saveIcon(vCardInfo.photo); + } + return; + } + let buddy = this._buddies.get(jid); + if (vCardInfo.fullName) { + buddy.vCardFormattedName = vCardInfo.fullName; + foundFormattedName = true; + } + if (vCardInfo.photo) { + buddy._saveIcon(vCardInfo.photo); + } + if (!foundFormattedName && buddy._vCardFormattedName) { + buddy.vCardFormattedName = ""; + } + buddy._vCardReceived = true; + }, + + /** + * Save the icon for a resource to the local file system. + * + * @param photo - The vcard photo node representing the icon. + * @param {prplIChatBuddy|prplIConversation} resource - Resource the icon is for. + * @returns {Promise<string>} Resolves with the file:// URI to the local icon file. + */ + _saveResourceIcon(photo, resource) { + // Some servers seem to send a photo node without a type declared. + let type = photo.getElement(["TYPE"]); + if (!type) { + return Promise.reject(new Error("Missing image type")); + } + type = type.innerText; + const kExt = { + "image/gif": "gif", + "image/jpeg": "jpg", + "image/png": "png", + }; + if (!kExt.hasOwnProperty(type)) { + return Promise.reject(new Error("Unknown image type")); + } + + let content = "", + data = ""; + // Strip all characters not allowed in base64 before parsing. + let parseBase64 = aBase => atob(aBase.replace(/[^A-Za-z0-9\+\/\=]/g, "")); + for (let line of photo.getElement(["BINVAL"]).innerText.split("\n")) { + data += line; + // Mozilla's atob() doesn't handle padding with "=" or "==" + // unless it's at the end of the string, so we have to work around that. + if (line.endsWith("=")) { + content += parseBase64(data); + data = ""; + } + } + content += parseBase64(data); + + // Store a sha1 hash of the photo we have just received. + let ch = Cc["@mozilla.org/security/hash;1"].createInstance( + Ci.nsICryptoHash + ); + ch.init(ch.SHA1); + let dataArray = Object.keys(content).map(i => content.charCodeAt(i)); + ch.update(dataArray, dataArray.length); + let hash = ch.finish(false); + function toHexString(charCode) { + return charCode.toString(16).padStart(2, "0"); + } + resource._photoHash = Object.keys(hash) + .map(i => toHexString(hash.charCodeAt(i))) + .join(""); + + let istream = Cc["@mozilla.org/io/string-input-stream;1"].createInstance( + Ci.nsIStringInputStream + ); + istream.setData(content, content.length); + + let fileName = resource._photoHash + "." + kExt[type]; + let file = lazy.FileUtils.getFile("ProfD", [ + "icons", + this.protocol.normalizedName, + this.normalizedName, + fileName, + ]); + let ostream = lazy.FileUtils.openSafeFileOutputStream(file); + return new Promise(resolve => { + lazy.NetUtil.asyncCopy(istream, ostream, rc => { + if (Components.isSuccessCode(rc)) { + resolve(Services.io.newFileURI(file).spec); + } + }); + }); + }, + + _requestNextVCard() { + if (!this._pendingVCardRequests.length) { + return; + } + let s = Stanza.iq( + "get", + null, + this._pendingVCardRequests[0], + Stanza.node("vCard", Stanza.NS.vcard) + ); + this.sendStanza(s, this.onVCard, this); + }, + + _addVCardRequest(aJID) { + let requestPending = !!this._pendingVCardRequests.length; + this._pendingVCardRequests.push(aJID); + if (!requestPending) { + this._requestNextVCard(); + } + }, + + // XEP-0029 (Section 2) and RFC 6122 (Section 2): The node and domain are + // lowercase, while resources are case sensitive and can contain spaces. + normalizeFullJid(aJID) { + return this._parseJID(aJID.trim()).jid; + }, + + // Standard normalization for XMPP removes the resource part of jids. + normalize(aJID) { + return aJID + .trim() + .split("/", 1)[0] // up to first slash + .toLowerCase(); + }, + + // RFC 6122 (Section 2): [ localpart "@" ] domainpart [ "/" resourcepart ] is + // the form of jid. + // Localpart is parsed as node and optional. + // Domainpart is parsed as domain and required. + // resourcepart is parsed as resource and optional. + _parseJID(aJid) { + let match = /^(?:([^"&'/:<>@]+)@)?([^@/<>'\"]+)(?:\/(.*))?$/.exec( + aJid.trim() + ); + if (!match) { + return null; + } + + let result = { + node: match[1], + domain: match[2].toLowerCase(), + resource: match[3], + }; + return this._setJID(result.domain, result.node, result.resource); + }, + + // Constructs jid as an object from domain, node and resource parts. + // The object has properties (node, domain, resource and jid). + // aDomain is required, but aNode and aResource are optional. + _setJID(aDomain, aNode = null, aResource = null) { + if (!aDomain) { + throw new Error("aDomain must have a value"); + } + + let result = { + node: aNode, + domain: aDomain.toLowerCase(), + resource: aResource, + }; + let jid = result.domain; + if (result.node) { + result.node = result.node.toLowerCase(); + jid = result.node + "@" + jid; + } + if (result.resource) { + jid += "/" + result.resource; + } + result.jid = jid; + return result; + }, + + _onRosterItem(aItem, aNotifyOfUpdates) { + let jid = aItem.attributes.jid; + if (!jid) { + this.WARN("Received a roster item without jid: " + aItem.getXML()); + return ""; + } + jid = this.normalize(jid); + + let subscription = ""; + if ("subscription" in aItem.attributes) { + subscription = aItem.attributes.subscription; + } + if (subscription == "remove") { + this._forgetRosterItem(jid); + return ""; + } + + let buddy; + if (this._buddies.has(jid)) { + buddy = this._buddies.get(jid); + let groups = aItem.getChildren("group"); + if (groups.length) { + // If the server specified at least one group, ensure the group we use + // as the account buddy's tag is still a group on the server... + let tagName = buddy.tag.name; + if (!groups.some(g => g.innerText == tagName)) { + // ... otherwise we need to move our account buddy to a new group. + tagName = groups[0].innerText; + if (tagName) { + // Should always be true, but check just in case... + let oldTag = buddy.tag; + buddy._tag = IMServices.tags.createTag(tagName); + IMServices.contacts.accountBuddyMoved(buddy, oldTag, buddy._tag); + } + } + } + } else { + let tag; + for (let group of aItem.getChildren("group")) { + let name = group.innerText; + if (name) { + tag = IMServices.tags.createTag(name); + break; // TODO we should create an accountBuddy per group, + // but this._buddies would probably not like that... + } + } + buddy = new this._accountBuddyConstructor( + this, + null, + tag || IMServices.tags.defaultTag, + jid + ); + } + + // We request the vCard only if we haven't received it yet and are + // subscribed to presence for that contact. + if ( + (subscription == "both" || subscription == "to") && + !buddy._vCardReceived + ) { + this._addVCardRequest(jid); + } + + let alias = "name" in aItem.attributes ? aItem.attributes.name : ""; + if (alias) { + if (aNotifyOfUpdates && this._buddies.has(jid)) { + buddy.rosterAlias = alias; + } else { + buddy._rosterAlias = alias; + } + } else if (buddy._rosterAlias) { + buddy.rosterAlias = ""; + } + + if (subscription) { + buddy.subscription = subscription; + } + if (!this._buddies.has(jid)) { + this._buddies.set(jid, buddy); + IMServices.contacts.accountBuddyAdded(buddy); + } else if (aNotifyOfUpdates) { + buddy._notifyObservers("status-detail-changed"); + } + + // Keep the xml nodes of the item so that we don't have to + // recreate them when changing something (eg. the alias) in it. + buddy._rosterItem = aItem; + + return jid; + }, + _forgetRosterItem(aJID) { + IMServices.contacts.accountBuddyRemoved(this._buddies.get(aJID)); + this._buddies.delete(aJID); + }, + + /* When the roster is received */ + onRoster(aStanza) { + // For the first element that is a roster stanza. + for (let qe of aStanza.getChildren("query")) { + if (qe.uri != Stanza.NS.roster) { + continue; + } + + // Find all the roster items in the new message. + let newRoster = new Set(); + for (let item of qe.getChildren("item")) { + let jid = this._onRosterItem(item); + if (jid) { + newRoster.add(jid); + } + } + // If an item was in the old roster, but not in the new, forget it. + for (let jid of this._buddies.keys()) { + if (!newRoster.has(jid)) { + this._forgetRosterItem(jid); + } + } + break; + } + + this._sendPresence(); + this._buddies.forEach(b => { + if (b.subscription == "both" || b.subscription == "to") { + b.setStatus(Ci.imIStatusInfo.STATUS_OFFLINE, ""); + } + }); + this.reportConnected(); + this._sendVCard(); + }, + + /* Public methods */ + + sendStanza(aStanza, aCallback, aThis, aLogString) { + return this._connection.sendStanza(aStanza, aCallback, aThis, aLogString); + }, + + // Variations of the XMPP protocol can change these default constructors: + _conversationConstructor: XMPPConversation, + _MUCConversationConstructor: XMPPMUCConversation, + _accountBuddyConstructor: XMPPAccountBuddy, + + /* Create a new conversation */ + createConversation(aName) { + let convName = this.normalize(aName); + + // Checks if conversation is with a participant of a MUC we are in. We do + // not want to strip the resource as it is of the form room@domain/nick. + let isMucParticipant = this._mucs.has(convName); + if (isMucParticipant) { + convName = this.normalizeFullJid(aName); + } + + // Checking that the aName can be parsed and is not broken. + let jid = this._parseJID(convName); + if ( + !jid || + !jid.domain || + (isMucParticipant && (!jid.node || !jid.resource)) + ) { + this.ERROR("Could not create conversation as jid is broken: " + convName); + throw new Error("Invalid JID"); + } + + if (!this._conv.has(convName)) { + this._conv.set( + convName, + new this._conversationConstructor(this, convName, isMucParticipant) + ); + } + + return this._conv.get(convName); + }, + + /* Remove an existing conversation */ + removeConversation(aNormalizedName) { + if (this._conv.has(aNormalizedName)) { + this._conv.delete(aNormalizedName); + } else if (this._mucs.has(aNormalizedName)) { + this._mucs.delete(aNormalizedName); + } + }, + + /* Private methods */ + + /** + * Disconnect from the server + * + * @param {number} aError - The error reason, passed to reportDisconnecting. + * @param {string} aErrorMessage - The error message, passed to reportDisconnecting. + * @param {boolean} aQuiet - True to avoid sending status change notifications + * during the uninitialization of the account. + */ + _disconnect( + aError = Ci.prplIAccount.NO_ERROR, + aErrorMessage = "", + aQuiet = false + ) { + if (!this._connection) { + return; + } + + this.reportDisconnecting(aError, aErrorMessage); + + this._buddies.forEach(b => { + if (!aQuiet) { + b.setStatus(Ci.imIStatusInfo.STATUS_UNKNOWN, ""); + } + b.onAccountDisconnected(); + }); + + this._mucs.forEach(muc => { + muc.joining = false; // In case we never finished joining. + muc.left = true; + }); + + this._connection.disconnect(); + delete this._connection; + + // We won't receive "user-icon-changed" notifications while the + // account isn't connected, so clear the cache to avoid keeping an + // obsolete icon. + delete this._cachedUserIcon; + // Also clear the cached user vCard, as we will want to redownload it + // after reconnecting. + delete this._userVCard; + + // Clear vCard requests. + this._pendingVCardRequests = []; + + this.reportDisconnected(); + }, + + /* Set the user status on the server */ + _sendPresence() { + delete this._shouldSendPresenceForIdlenessChange; + + if (!this._connection) { + return; + } + + let si = this.imAccount.statusInfo; + let statusType = si.statusType; + let show = ""; + if (statusType == Ci.imIStatusInfo.STATUS_UNAVAILABLE) { + show = "dnd"; + } else if ( + statusType == Ci.imIStatusInfo.STATUS_AWAY || + statusType == Ci.imIStatusInfo.STATUS_IDLE + ) { + show = "away"; + } + let children = []; + if (show) { + children.push(Stanza.node("show", null, null, show)); + } + let statusText = si.statusText; + if (statusText) { + children.push(Stanza.node("status", null, null, statusText)); + } + if (this._idleSince) { + let time = Math.floor(Date.now() / 1000) - this._idleSince; + children.push(Stanza.node("query", Stanza.NS.last, { seconds: time })); + } + if (this.prefs.prefHasUserValue("priority")) { + let priority = Math.max(-128, Math.min(127, this.getInt("priority"))); + if (priority) { + children.push(Stanza.node("priority", null, null, priority.toString())); + } + } + this.sendStanza( + Stanza.presence({ "xml:lang": "en" }, children), + aStanza => { + // As we are implicitly subscribed to our own presence (rfc6121#4), we + // will receive the presence stanza mirrored back to us. We don't need + // to do anything with this response. + return true; + } + ); + }, + + _downloadingUserVCard: false, + _downloadUserVCard() { + // If a download is already in progress, don't start another one. + if (this._downloadingUserVCard) { + return; + } + this._downloadingUserVCard = true; + let s = Stanza.iq("get", null, null, Stanza.node("vCard", Stanza.NS.vcard)); + this.sendStanza(s, this.onUserVCard, this); + }, + + onUserVCard(aStanza) { + delete this._downloadingUserVCard; + let userVCard = aStanza.getElement(["vCard"]) || null; + if (userVCard) { + // Strip any server-specific namespace off the incoming vcard + // before storing it. + this._userVCard = Stanza.node( + "vCard", + Stanza.NS.vcard, + null, + userVCard.children + ); + } + + // If a user icon exists in the vCard we received from the server, + // we need to ensure the line breaks in its binval are exactly the + // same as those we would include if we sent the icon, and that + // there isn't any other whitespace. + if (this._userVCard) { + let binval = this._userVCard.getElement(["PHOTO", "BINVAL"]); + if (binval && binval.children.length) { + binval = binval.children[0]; + binval.text = binval.text + .replace(/[^A-Za-z0-9\+\/\=]/g, "") + .replace(/.{74}/g, "$&\n"); + } + } else { + // Downloading the vCard failed. + if ( + this.handleErrors({ + itemNotFound: () => false, // OK, no vCard exists yet. + default: () => true, + })(aStanza) + ) { + this.WARN( + "Unexpected error retrieving the user's vcard, " + + "so we won't attempt to set it either." + ); + return; + } + // Set this so that we don't get into an infinite loop trying to download + // the vcard again. The check in sendVCard is for hasOwnProperty. + this._userVCard = null; + } + this._sendVCard(); + }, + + _cachingUserIcon: false, + _cacheUserIcon() { + if (this._cachingUserIcon) { + return; + } + + let userIcon = this.imAccount.statusInfo.getUserIcon(); + if (!userIcon) { + this._cachedUserIcon = null; + this._sendVCard(); + return; + } + + this._cachingUserIcon = true; + let channel = lazy.NetUtil.newChannel({ + uri: userIcon, + loadingPrincipal: Services.scriptSecurityManager.getSystemPrincipal(), + securityFlags: + Ci.nsILoadInfo.SEC_REQUIRE_SAME_ORIGIN_INHERITS_SEC_CONTEXT, + contentPolicyType: Ci.nsIContentPolicy.TYPE_IMAGE, + }); + lazy.NetUtil.asyncFetch(channel, (inputStream, resultCode) => { + if (!Components.isSuccessCode(resultCode)) { + return; + } + try { + let type = channel.contentType; + let buffer = lazy.NetUtil.readInputStreamToString( + inputStream, + inputStream.available() + ); + let readImage = lazy.imgTools.decodeImageFromBuffer( + buffer, + buffer.length, + type + ); + let scaledImage; + if (readImage.width <= 96 && readImage.height <= 96) { + scaledImage = lazy.imgTools.encodeImage(readImage, type); + } else { + if (type != "image/jpeg") { + type = "image/png"; + } + scaledImage = lazy.imgTools.encodeScaledImage( + readImage, + type, + 64, + 64 + ); + } + + let bstream = Cc["@mozilla.org/binaryinputstream;1"].createInstance( + Ci.nsIBinaryInputStream + ); + bstream.setInputStream(scaledImage); + + let data = bstream.readBytes(bstream.available()); + this._cachedUserIcon = { + type, + binval: btoa(data).replace(/.{74}/g, "$&\n"), + }; + } catch (e) { + console.error(e); + this._cachedUserIcon = null; + } + delete this._cachingUserIcon; + this._sendVCard(); + }); + }, + _sendVCard() { + if (!this._connection) { + return; + } + + // We have to download the user's existing vCard before updating it. + // This lets us preserve the fields that we don't change or don't know. + // Some servers may reject a new vCard if we don't do this first. + if (!this.hasOwnProperty("_userVCard")) { + // The download of the vCard is asynchronous and will call _sendVCard back + // when the user's vCard has been received. + this._downloadUserVCard(); + return; + } + + // Read the local user icon asynchronously from the disk. + // _cacheUserIcon will call _sendVCard back once the icon is ready. + if (!this.hasOwnProperty("_cachedUserIcon")) { + this._cacheUserIcon(); + return; + } + + // If the user currently doesn't have any vCard on the server or + // the download failed, an empty new one. + if (!this._userVCard) { + this._userVCard = Stanza.node("vCard", Stanza.NS.vcard); + } + + // Keep a serialized copy of the existing user vCard so that we + // can avoid resending identical data to the server. + let existingVCard = this._userVCard.getXML(); + + let fn = this._userVCard.getElement(["FN"]); + let displayName = this.imAccount.statusInfo.displayName; + if (displayName) { + // If a display name is set locally, update or add an FN field to the vCard. + if (!fn) { + this._userVCard.addChild( + Stanza.node("FN", Stanza.NS.vcard, null, displayName) + ); + } else if (fn.children.length) { + fn.children[0].text = displayName; + } else { + fn.addText(displayName); + } + } else if ("_forceUserDisplayNameUpdate" in this) { + // We remove a display name stored on the server without replacing + // it with a new value only if this _sendVCard call is the result of + // a user action. This is to avoid removing data from the server each + // time the user connects from a new profile. + this._userVCard.children = this._userVCard.children.filter( + n => n.qName != "FN" + ); + } + delete this._forceUserDisplayNameUpdate; + + if (this._cachedUserIcon) { + // If we have a local user icon, update or add it in the PHOTO field. + let photoChildren = [ + Stanza.node("TYPE", Stanza.NS.vcard, null, this._cachedUserIcon.type), + Stanza.node( + "BINVAL", + Stanza.NS.vcard, + null, + this._cachedUserIcon.binval + ), + ]; + let photo = this._userVCard.getElement(["PHOTO"]); + if (photo) { + photo.children = photoChildren; + } else { + this._userVCard.addChild( + Stanza.node("PHOTO", Stanza.NS.vcard, null, photoChildren) + ); + } + } else if ("_forceUserIconUpdate" in this) { + // Like for the display name, we remove a photo without + // replacing it only if the call is caused by a user action. + this._userVCard.children = this._userVCard.children.filter( + n => n.qName != "PHOTO" + ); + } + delete this._forceUserIconUpdate; + + // Send the vCard only if it has really changed. + // We handle the result response from the server (it does not require + // any further action). + if (this._userVCard.getXML() != existingVCard) { + this.sendStanza( + Stanza.iq("set", null, null, this._userVCard), + this._handleResult() + ); + } else { + this.LOG( + "Not sending the vCard because the server stored vCard is identical." + ); + } + }, +}; diff --git a/comm/chat/protocols/xmpp/xmpp-commands.sys.mjs b/comm/chat/protocols/xmpp/xmpp-commands.sys.mjs new file mode 100644 index 0000000000..fc02f3bc0e --- /dev/null +++ b/comm/chat/protocols/xmpp/xmpp-commands.sys.mjs @@ -0,0 +1,347 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs"; +import { l10nHelper } from "resource:///modules/imXPCOMUtils.sys.mjs"; + +const lazy = {}; + +XPCOMUtils.defineLazyGetter(lazy, "_", () => + l10nHelper("chrome://chat/locale/xmpp.properties") +); + +// Get conversation object. +function getConv(aConv) { + return aConv.wrappedJSObject; +} + +// Get account object. +function getAccount(aConv) { + return getConv(aConv)._account; +} + +function getMUC(aConv) { + let conv = getConv(aConv); + if (conv.left) { + conv.writeMessage( + conv.name, + lazy._("conversation.error.commandFailedNotInRoom"), + { system: true } + ); + return null; + } + return conv; +} + +// Trims the string and splits it in two parts on the first space +// if there is one. Returns the non-empty parts in an array. +function splitInput(aString) { + let params = aString.trim(); + if (!params) { + return []; + } + + let splitParams = []; + let offset = params.indexOf(" "); + if (offset != -1) { + splitParams.push(params.slice(0, offset)); + splitParams.push(params.slice(offset + 1).trimLeft()); + } else { + splitParams.push(params); + } + return splitParams; +} + +// Trims the string and splits it in two parts (The first part is a nickname +// and the second part is the rest of string) based on nicknames of current +// participants. Returns the non-empty parts in an array. +function splitByNick(aString, aConv) { + let params = aString.trim(); + if (!params) { + return []; + } + + // Match trimmed-string with the longest prefix of participant's nickname. + let nickName = ""; + for (let participant of aConv._participants.keys()) { + if ( + params.startsWith(participant + " ") && + participant.length > nickName.length + ) { + nickName = participant; + } + } + if (!nickName) { + let offset = params.indexOf(" "); + let expectedNickName = offset != -1 ? params.slice(0, offset) : params; + aConv.writeMessage( + aConv.name, + lazy._("conversation.error.nickNotInRoom", expectedNickName), + { system: true } + ); + return []; + } + + let splitParams = []; + splitParams.push(nickName); + + let msg = params.substring(nickName.length); + if (msg) { + splitParams.push(msg.trimLeft()); + } + return splitParams; +} + +// Splits aMsg in two entries and checks the first entry is a valid jid, then +// passes it to aConv.invite(). +// Returns false if aMsg is empty, otherwise returns true. +function invite(aMsg, aConv) { + let params = splitInput(aMsg); + if (!params.length) { + return false; + } + + // Check user's jid is valid. + let account = getAccount(aConv); + let jid = account._parseJID(params[0]); + if (!jid) { + aConv.writeMessage( + aConv.name, + lazy._("conversation.error.invalidJID", params[0]), + { system: true } + ); + return true; + } + + aConv.invite(...params); + return true; +} + +export var commands = [ + { + name: "join", + get helpString() { + return lazy._("command.join3", "join"); + }, + run(aMsg, aConv, aReturnedConv) { + let account = getAccount(aConv); + let params = aMsg.trim(); + let conv; + + if (!params) { + conv = getConv(aConv); + if (!conv.isChat) { + return false; + } + if (!conv.left) { + return true; + } + + // Rejoin the current conversation. If the conversation was explicitly + // parted by the user, chatRoomFields will have been deleted. + // Otherwise, make use of it. + if (conv.chatRoomFields) { + account.joinChat(conv.chatRoomFields); + return true; + } + + params = conv.name; + } + let chatRoomFields = account.getChatRoomDefaultFieldValues(params); + conv = account.joinChat(chatRoomFields); + + if (aReturnedConv) { + aReturnedConv.value = conv; + } + return true; + }, + }, + { + name: "part", + get helpString() { + return lazy._("command.part2", "part"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_CHAT, + run(aMsg, aConv) { + let conv = getConv(aConv); + if (!conv.left) { + conv.part(aMsg); + } + return true; + }, + }, + { + name: "topic", + get helpString() { + return lazy._("command.topic", "topic"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_CHAT, + run(aMsg, aConv) { + let conv = getMUC(aConv); + if (!conv) { + return true; + } + conv.topic = aMsg; + return true; + }, + }, + { + name: "ban", + get helpString() { + return lazy._("command.ban", "ban"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_CHAT, + run(aMsg, aConv) { + let params = splitInput(aMsg); + if (!params.length) { + return false; + } + + let conv = getMUC(aConv); + if (conv) { + conv.ban(...params); + } + return true; + }, + }, + { + name: "kick", + get helpString() { + return lazy._("command.kick", "kick"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_CHAT, + run(aMsg, aConv) { + let conv = getMUC(aConv); + if (!conv) { + return true; + } + + let params = splitByNick(aMsg, conv); + if (!params.length) { + return false; + } + conv.kick(...params); + return true; + }, + }, + { + name: "invite", + get helpString() { + return lazy._("command.invite", "invite"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_CHAT, + run(aMsg, aConv) { + let conv = getMUC(aConv); + if (!conv) { + return true; + } + + return invite(aMsg, conv); + }, + }, + { + name: "inviteto", + get helpString() { + return lazy._("command.inviteto", "inviteto"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_IM, + run: (aMsg, aConv) => invite(aMsg, getConv(aConv)), + }, + { + name: "me", + get helpString() { + return lazy._("command.me", "me"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_CHAT, + run(aMsg, aConv) { + let params = aMsg.trim(); + if (!params) { + return false; + } + + let conv = getConv(aConv); + conv.sendMsg(params, true); + + return true; + }, + }, + { + name: "nick", + get helpString() { + return lazy._("command.nick", "nick"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_CHAT, + run(aMsg, aConv) { + let params = aMsg.trim().split(/\s+/); + if (!params[0]) { + return false; + } + + let conv = getMUC(aConv); + if (conv) { + conv.setNick(params[0]); + } + return true; + }, + }, + { + name: "msg", + get helpString() { + return lazy._("command.msg", "msg"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_CHAT, + run(aMsg, aConv, aReturnedConv) { + let conv = getMUC(aConv); + if (!conv) { + return true; + } + + let params = splitByNick(aMsg, conv); + if (params.length != 2) { + return false; + } + let [nickName, msg] = params; + + let account = getAccount(aConv); + let privateConv = account.createConversation(conv.name + "/" + nickName); + if (!privateConv) { + return true; + } + privateConv.sendMsg(msg.trim()); + + if (aReturnedConv) { + aReturnedConv.value = privateConv; + } + return true; + }, + }, + { + name: "version", + get helpString() { + return lazy._("command.version", "version"); + }, + usageContext: Ci.imICommand.CMD_CONTEXT_IM, + run(aMsg, aConv, aReturnedConv) { + let conv = getConv(aConv); + if (conv.left) { + return true; + } + + // We do not have user's resource. + if (!conv._targetResource) { + conv.writeMessage( + conv.name, + lazy._("conversation.error.resourceNotAvailable", conv.shortName), + { + system: true, + } + ); + return true; + } + + conv.getVersion(); + return true; + }, + }, +]; diff --git a/comm/chat/protocols/xmpp/xmpp-session.sys.mjs b/comm/chat/protocols/xmpp/xmpp-session.sys.mjs new file mode 100644 index 0000000000..ca2fd4eebb --- /dev/null +++ b/comm/chat/protocols/xmpp/xmpp-session.sys.mjs @@ -0,0 +1,764 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +const { DNS } = ChromeUtils.import("resource:///modules/DNS.jsm"); +import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs"; +import { l10nHelper } from "resource:///modules/imXPCOMUtils.sys.mjs"; +import { Socket } from "resource:///modules/socket.sys.mjs"; +import { Stanza, XMPPParser } from "resource:///modules/xmpp-xml.sys.mjs"; +import { XMPPAuthMechanisms } from "resource:///modules/xmpp-authmechs.sys.mjs"; + +const lazy = {}; + +XPCOMUtils.defineLazyGetter(lazy, "_", () => + l10nHelper("chrome://chat/locale/xmpp.properties") +); + +export function XMPPSession( + aHost, + aPort, + aSecurity, + aJID, + aPassword, + aAccount +) { + this._host = aHost; + this._port = aPort; + + this._connectionSecurity = aSecurity; + if (this._connectionSecurity == "old_ssl") { + this._security = ["ssl"]; + } else if (this._connectionSecurity != "none") { + this._security = [aPort == 5223 || aPort == 443 ? "ssl" : "starttls"]; + } + + if (!aJID.node) { + aAccount.reportDisconnecting( + Ci.prplIAccount.ERROR_INVALID_USERNAME, + lazy._("connection.error.invalidUsername") + ); + aAccount.reportDisconnected(); + return; + } + this._jid = aJID; + this._domain = aJID.domain; + this._password = aPassword; + this._account = aAccount; + this._resource = aJID.resource; + this._handlers = new Map(); + this._account.reportConnecting(); + + // The User has specified a certain server or port, so we should not do + // DNS SRV lookup or the preference of disabling DNS SRV part and use + // normal connect is set. + // RFC 6120 (Section 3.2.3): When Not to Use SRV. + if ( + Services.prefs.getBoolPref("chat.dns.srv.disable") || + this._account.prefs.prefHasUserValue("server") || + this._account.prefs.prefHasUserValue("port") + ) { + this.connect(this._host, this._port, this._security); + return; + } + + // RFC 6120 (Section 3.2.1): SRV lookup. + this._account.reportConnecting(lazy._("connection.srvLookup")); + DNS.srv("_xmpp-client._tcp." + this._host) + .then(aResult => this._handleSrvQuery(aResult)) + .catch(aError => { + if (aError === this.SRV_ERROR_XMPP_NOT_SUPPORTED) { + this.LOG("SRV: XMPP is not supported on this domain."); + + // RFC 6120 (Section 3.2.1) and RFC 2782 (Usage rules): Abort as the + // service is decidedly not available at this domain. + this._account.reportDisconnecting( + Ci.prplIAccount.ERROR_OTHER_ERROR, + lazy._("connection.error.XMPPNotSupported") + ); + this._account.reportDisconnected(); + return; + } + + this.ERROR("Error during SRV lookup:", aError); + + // Since we don't receive a response to SRV query, we SHOULD attempt the + // fallback process (use normal connect without SRV lookup). + this.connect(this._host, this._port, this._security); + }); +} + +XMPPSession.prototype = { + /* for the socket.jsm helper */ + __proto__: Socket, + connectTimeout: 60, + readWriteTimeout: 300, + + // Contains the remaining SRV records if we failed to connect the current one. + _srvRecords: [], + + sendPing() { + this.sendStanza( + Stanza.iq("get", null, null, Stanza.node("ping", Stanza.NS.ping)), + this.cancelDisconnectTimer, + this + ); + }, + _lastReceiveTime: 0, + _lastSendTime: 0, + checkPingTimer(aJustSentSomething = false) { + // Don't start a ping timer if we're not fully connected yet. + if (this.onXmppStanza != this.stanzaListeners.accountListening) { + return; + } + let now = Date.now(); + if (aJustSentSomething) { + this._lastSendTime = now; + } else { + this._lastReceiveTime = now; + } + // We only cancel the ping timer if we've both received and sent + // something in the last two minutes. This is because Openfire + // servers will disconnect us if we don't send anything for a + // couple of minutes. + if ( + Math.min(this._lastSendTime, this._lastReceiveTime) > + now - this.kTimeBeforePing + ) { + this.resetPingTimer(); + } + }, + + get DEBUG() { + return this._account.DEBUG; + }, + get LOG() { + return this._account.LOG; + }, + get WARN() { + return this._account.WARN; + }, + get ERROR() { + return this._account.ERROR; + }, + + _security: null, + _encrypted: false, + + // DNS SRV errors in XMPP. + SRV_ERROR_XMPP_NOT_SUPPORTED: -2, + + // Handles result of DNS SRV query and saves sorted results if it's OK in _srvRecords, + // otherwise throws error. + _handleSrvQuery(aResult) { + this.LOG("SRV lookup: " + JSON.stringify(aResult)); + if (aResult.length == 0) { + // RFC 6120 (Section 3.2.1) and RFC 2782 (Usage rules): No SRV records, + // try to login with the given domain name. + this.connect(this._host, this._port, this._security); + return; + } else if (aResult.length == 1 && aResult[0].host == ".") { + throw this.SRV_ERROR_XMPP_NOT_SUPPORTED; + } + + // Sort results: Lower priority is more preferred and higher weight is + // more preferred in equal priorities. + aResult.sort(function (a, b) { + return a.prio - b.prio || b.weight - a.weight; + }); + + this._srvRecords = aResult; + this._connectNextRecord(); + }, + + _connectNextRecord() { + if (!this._srvRecords.length) { + this.ERROR( + "_connectNextRecord is called and there are no more records " + + "to connect." + ); + return; + } + + let record = this._srvRecords.shift(); + + // RFC 3920 (Section 5.1): Certificates MUST be checked against the + // hostname as provided by the initiating entity (e.g. user). + this.connect( + this._domain, + this._port, + this._security, + null, + record.host, + record.port + ); + }, + + /* Disconnect from the server */ + disconnect() { + if (this.onXmppStanza == this.stanzaListeners.accountListening) { + this.send("</stream:stream>"); + } + delete this.onXmppStanza; + Socket.disconnect.call(this); + if (this._parser) { + this._parser.destroy(); + delete this._parser; + } + this.cancelDisconnectTimer(); + }, + + /* Report errors to the account */ + onError(aError, aException) { + // If we're trying to connect to SRV entries, then keep trying until a + // successful connection occurs or we run out of SRV entries to try. + if (this._srvRecords.length) { + this._connectNextRecord(); + return; + } + + this._account.onError(aError, aException); + }, + + /* Send a text message to the server */ + send(aMsg, aLogString) { + this.sendString(aMsg, "UTF-8", aLogString); + }, + + /* Send a stanza to the server. + * Can set a callback if required, which will be called when the server + * responds to the stanza with a stanza of the same id. The callback should + * return true if the stanza was handled, false if not. Note that an + * undefined return value is treated as true. + */ + sendStanza(aStanza, aCallback, aThis, aLogString) { + if (!aStanza.attributes.hasOwnProperty("id")) { + aStanza.attributes.id = this._account.generateId(); + } + if (aCallback) { + this._handlers.set(aStanza.attributes.id, aCallback.bind(aThis)); + } + this.send(aStanza.getXML(), aLogString); + this.checkPingTimer(true); + return aStanza.attributes.id; + }, + + /* This method handles callbacks for specific ids. */ + execHandler(aId, aStanza) { + let handler = this._handlers.get(aId); + if (!handler) { + return false; + } + let isHandled = handler(aStanza); + // Treat undefined return values as handled. + if (isHandled === undefined) { + isHandled = true; + } + this._handlers.delete(aId); + return isHandled; + }, + + /* Start the XMPP stream */ + startStream() { + if (this._parser) { + this._parser.destroy(); + } + this._parser = new XMPPParser(this); + this.send( + '<?xml version="1.0"?><stream:stream to="' + + this._domain + + '" xmlns="jabber:client" xmlns:stream="http://etherx.jabber.org/streams" version="1.0">' + ); + }, + + startSession() { + this.sendStanza( + Stanza.iq("set", null, null, Stanza.node("session", Stanza.NS.session)), + aStanza => aStanza.attributes.type == "result" + ); + this.onXmppStanza = this.stanzaListeners.sessionStarted; + }, + + /* XEP-0078: Non-SASL Authentication */ + startLegacyAuth() { + if (!this._encrypted && this._connectionSecurity == "require_tls") { + this.onError( + Ci.prplIAccount.ERROR_ENCRYPTION_ERROR, + lazy._("connection.error.startTLSNotSupported") + ); + return; + } + + this.onXmppStanza = this.stanzaListeners.legacyAuth; + let s = Stanza.iq( + "get", + null, + this._domain, + Stanza.node( + "query", + Stanza.NS.auth, + null, + Stanza.node("username", null, null, this._jid.node) + ) + ); + this.sendStanza(s); + }, + + // If aResource is null, it will request to bind a server-generated + // resourcepart, otherwise request to bind a client-submitted resourcepart. + _requestBind(aResource) { + let resourceNode = aResource + ? Stanza.node("resource", null, null, aResource) + : null; + this.sendStanza( + Stanza.iq( + "set", + null, + null, + Stanza.node("bind", Stanza.NS.bind, null, resourceNode) + ) + ); + }, + + /* Socket events */ + /* The connection is established */ + onConnection() { + if (this._security.includes("ssl")) { + this.onXmppStanza = this.stanzaListeners.startAuth; + this._encrypted = true; + } else { + this.onXmppStanza = this.stanzaListeners.initStream; + } + + // Clear SRV results since we have connected. + this._srvRecords = []; + + this._account.reportConnecting(lazy._("connection.initializingStream")); + this.startStream(); + }, + + /* When incoming data is available to be parsed */ + onDataReceived(aData) { + this.checkPingTimer(); + this._lastReceivedData = aData; + try { + this._parser.onDataAvailable(aData); + } catch (e) { + console.error(e); + this.onXMLError("parser-exception", e); + } + delete this._lastReceivedData; + }, + + /* The connection got disconnected without us closing it. */ + onConnectionClosed() { + this._networkError(lazy._("connection.error.serverClosedConnection")); + }, + onConnectionSecurityError(aTLSError, aNSSErrorMessage) { + let error = this._account.handleConnectionSecurityError(this); + this.onError(error, aNSSErrorMessage); + }, + onConnectionReset() { + this._networkError(lazy._("connection.error.resetByPeer")); + }, + onConnectionTimedOut() { + this._networkError(lazy._("connection.error.timedOut")); + }, + _networkError(aMessage) { + this.onError(Ci.prplIAccount.ERROR_NETWORK_ERROR, aMessage); + }, + + /* Methods called by the XMPPParser instance */ + onXMLError(aError, aException) { + if (aError == "parsing-characters") { + this.WARN(aError + ": " + aException + "\n" + this._lastReceivedData); + } else { + this.ERROR(aError + ": " + aException + "\n" + this._lastReceivedData); + } + if (aError != "parse-warning" && aError != "parsing-characters") { + this._networkError(lazy._("connection.error.receivedUnexpectedData")); + } + }, + + // All the functions in stanzaListeners are used as onXmppStanza + // implementations at various steps of establishing the session. + stanzaListeners: { + initStream(aStanza) { + if (aStanza.localName != "features") { + this.ERROR( + "Unexpected stanza " + aStanza.localName + ", expected 'features'" + ); + this._networkError(lazy._("connection.error.incorrectResponse")); + return; + } + + let starttls = aStanza.getElement(["starttls"]); + if (starttls && this._security.includes("starttls")) { + this._account.reportConnecting( + lazy._("connection.initializingEncryption") + ); + this.sendStanza(Stanza.node("starttls", Stanza.NS.tls)); + this.onXmppStanza = this.stanzaListeners.startTLS; + return; + } + if (starttls && starttls.children.some(c => c.localName == "required")) { + this.onError( + Ci.prplIAccount.ERROR_ENCRYPTION_ERROR, + lazy._("connection.error.startTLSRequired") + ); + return; + } + if (!starttls && this._connectionSecurity == "require_tls") { + this.onError( + Ci.prplIAccount.ERROR_ENCRYPTION_ERROR, + lazy._("connection.error.startTLSNotSupported") + ); + return; + } + + // If we aren't starting TLS, jump to the auth step. + this.onXmppStanza = this.stanzaListeners.startAuth; + this.onXmppStanza(aStanza); + }, + startTLS(aStanza) { + if (aStanza.localName != "proceed") { + this._networkError(lazy._("connection.error.failedToStartTLS")); + return; + } + + this.startTLS(); + this._encrypted = true; + this.startStream(); + this.onXmppStanza = this.stanzaListeners.startAuth; + }, + startAuth(aStanza) { + if (aStanza.localName != "features") { + this.ERROR( + "Unexpected stanza " + aStanza.localName + ", expected 'features'" + ); + this._networkError(lazy._("connection.error.incorrectResponse")); + return; + } + + let mechs = aStanza.getElement(["mechanisms"]); + if (!mechs) { + let auth = aStanza.getElement(["auth"]); + if (auth && auth.uri == Stanza.NS.auth_feature) { + this.startLegacyAuth(); + } else { + this._networkError(lazy._("connection.error.noAuthMec")); + } + return; + } + + // Select the auth mechanism we will use. PLAIN will be treated + // a bit differently as we want to avoid it over an unencrypted + // connection, except if the user has explicitly allowed that + // behavior. + let authMechanisms = this._account.authMechanisms || XMPPAuthMechanisms; + let selectedMech = ""; + let canUsePlain = false; + mechs = mechs.getChildren("mechanism"); + for (let m of mechs) { + let mech = m.innerText; + if (mech == "PLAIN" && !this._encrypted) { + // If PLAIN is proposed over an unencrypted connection, + // remember that it's a possibility but don't bother + // checking if the user allowed it until we have verified + // that nothing more secure is available. + canUsePlain = true; + } else if (authMechanisms.hasOwnProperty(mech)) { + selectedMech = mech; + break; + } + } + if (!selectedMech && canUsePlain) { + if (this._connectionSecurity == "allow_unencrypted_plain_auth") { + selectedMech = "PLAIN"; + } else { + this.onError( + Ci.prplIAccount.ERROR_AUTHENTICATION_IMPOSSIBLE, + lazy._("connection.error.notSendingPasswordInClear") + ); + return; + } + } + if (!selectedMech) { + this.onError( + Ci.prplIAccount.ERROR_AUTHENTICATION_IMPOSSIBLE, + lazy._("connection.error.noCompatibleAuthMec") + ); + return; + } + let authMec = authMechanisms[selectedMech]( + this._jid.node, + this._password, + this._domain + ); + this._password = null; + + this._account.reportConnecting(lazy._("connection.authenticating")); + this.onXmppStanza = this.stanzaListeners.authDialog.bind(this, authMec); + this.onXmppStanza(null); // the first auth step doesn't read anything + }, + authDialog(aAuthMec, aStanza) { + if (aStanza && aStanza.localName == "failure") { + let errorMsg = "authenticationFailure"; + if ( + aStanza.getElement(["not-authorized"]) || + aStanza.getElement(["bad-auth"]) + ) { + errorMsg = "notAuthorized"; + } + this.onError( + Ci.prplIAccount.ERROR_AUTHENTICATION_FAILED, + lazy._("connection.error." + errorMsg) + ); + return; + } + + let result; + try { + result = aAuthMec.next(aStanza); + } catch (e) { + this.ERROR("Error in auth mechanism: " + e); + this.onError( + Ci.prplIAccount.ERROR_AUTHENTICATION_FAILED, + lazy._("connection.error.authenticationFailure") + ); + return; + } + + // The authentication mechanism can yield a promise which must resolve + // before sending data. If it rejects, abort. + if (result.value) { + Promise.resolve(result.value).then( + value => { + // Send the XML stanza that is returned. + if (value.send) { + this.send(value.send.getXML(), value.log); + } + }, + e => { + this.ERROR("Error resolving auth mechanism result: " + e); + this.onError( + Ci.prplIAccount.ERROR_AUTHENTICATION_FAILED, + lazy._("connection.error.authenticationFailure") + ); + } + ); + } + if (result.done) { + this.startStream(); + this.onXmppStanza = this.stanzaListeners.startBind; + } + }, + startBind(aStanza) { + if (!aStanza.getElement(["bind"])) { + this.ERROR("Unexpected lack of the bind feature"); + this._networkError(lazy._("connection.error.incorrectResponse")); + return; + } + + this._account.reportConnecting(lazy._("connection.gettingResource")); + this._requestBind(this._resource); + this.onXmppStanza = this.stanzaListeners.bindResult; + }, + bindResult(aStanza) { + if (aStanza.attributes.type == "error") { + let error = this._account.parseError(aStanza); + let message; + switch (error.condition) { + case "resource-constraint": + // RFC 6120 (7.6.2.1): Resource Constraint. + // The account has reached a limit on the number of simultaneous + // connected resources allowed. + message = "connection.error.failedMaxResourceLimit"; + break; + case "bad-request": + // RFC 6120 (7.7.2.1): Bad Request. + // The provided resourcepart cannot be processed by the server. + message = "connection.error.failedResourceNotValid"; + break; + case "conflict": + // RFC 6120 (7.7.2.2): Conflict. + // The provided resourcepart is already in use and the server + // disallowed the resource binding attempt. + this._requestBind(); + return; + default: + this.WARN(`Unhandled bind result error ${error.condition}.`); + message = "connection.error.failedToGetAResource"; + } + this._networkError(lazy._(message)); + return; + } + + let jid = aStanza.getElement(["bind", "jid"]); + if (!jid) { + this._networkError(lazy._("connection.error.failedToGetAResource")); + return; + } + jid = jid.innerText; + this.DEBUG("jid = " + jid); + this._jid = this._account._parseJID(jid); + this._resource = this._jid.resource; + this.startSession(); + }, + legacyAuth(aStanza) { + if (aStanza.attributes.type == "error") { + let error = aStanza.getElement(["error"]); + if (!error) { + this._networkError(lazy._("connection.error.incorrectResponse")); + return; + } + + let code = parseInt(error.attributes.code, 10); + if (code == 401) { + // Failed Authentication (Incorrect Credentials) + this.onError( + Ci.prplIAccount.ERROR_AUTHENTICATION_FAILED, + lazy._("connection.error.notAuthorized") + ); + return; + } else if (code == 406) { + // Failed Authentication (Required Information Not Provided) + this.onError( + Ci.prplIAccount.ERROR_AUTHENTICATION_FAILED, + lazy._("connection.error.authenticationFailure") + ); + return; + } + // else if (code == 409) { + // Failed Authentication (Resource Conflict) + // XXX Flo The spec in XEP-0078 defines this error code, but + // I've yet to find a server sending it. The server I tested + // with just closed the first connection when a second + // connection was attempted with the same resource. + // libpurple's jabber prpl doesn't support this code either. + // } + } + + if (aStanza.attributes.type != "result") { + this._networkError(lazy._("connection.error.incorrectResponse")); + return; + } + + if (aStanza.children.length == 0) { + // Success! + this._password = null; + this.startSession(); + return; + } + + let query = aStanza.getElement(["query"]); + let values = {}; + for (let c of query.children) { + values[c.qName] = c.innerText; + } + + if (!("username" in values) || !("resource" in values)) { + this._networkError(lazy._("connection.error.incorrectResponse")); + return; + } + + // If the resource is empty, we will fallback to brandShortName as + // resource is REQUIRED. + if (!this._resource) { + this._resource = Services.strings + .createBundle("chrome://branding/locale/brand.properties") + .GetStringFromName("brandShortName"); + this._jid = this._setJID( + this._jid.domain, + this._jid.node, + this._resource + ); + } + + let children = [ + Stanza.node("username", null, null, this._jid.node), + Stanza.node("resource", null, null, this._resource), + ]; + + let logString; + if ("digest" in values && this._streamId) { + let hashBase = this._streamId + this._password; + + let ch = Cc["@mozilla.org/security/hash;1"].createInstance( + Ci.nsICryptoHash + ); + ch.init(ch.SHA1); + // Non-US-ASCII characters MUST be encoded as UTF-8 since the + // SHA-1 hashing algorithm operates on byte arrays. + let data = [...new TextEncoder().encode(hashBase)]; + ch.update(data, data.length); + let hash = ch.finish(false); + let toHexString = charCode => ("0" + charCode.toString(16)).slice(-2); + let digest = Object.keys(hash) + .map(i => toHexString(hash.charCodeAt(i))) + .join(""); + + children.push(Stanza.node("digest", null, null, digest)); + logString = + "legacyAuth stanza containing SHA-1 hash of the password not logged"; + } else if ("password" in values) { + if ( + !this._encrypted && + this._connectionSecurity != "allow_unencrypted_plain_auth" + ) { + this.onError( + Ci.prplIAccount.ERROR_AUTHENTICATION_IMPOSSIBLE, + lazy._("connection.error.notSendingPasswordInClear") + ); + return; + } + children.push(Stanza.node("password", null, null, this._password)); + logString = "legacyAuth stanza containing password not logged"; + } else { + this.onError( + Ci.prplIAccount.ERROR_AUTHENTICATION_IMPOSSIBLE, + lazy._("connection.error.noCompatibleAuthMec") + ); + return; + } + + let s = Stanza.iq( + "set", + null, + this._domain, + Stanza.node("query", Stanza.NS.auth, null, children) + ); + this.sendStanza( + s, + undefined, + undefined, + `<iq type="set".../> (${logString})` + ); + }, + sessionStarted(aStanza) { + this.resetPingTimer(); + this._account.onConnection(); + this.LOG("Account successfully connected."); + this.onXmppStanza = this.stanzaListeners.accountListening; + }, + accountListening(aStanza) { + let id = aStanza.attributes.id; + if (id && this.execHandler(id, aStanza)) { + return; + } + + this._account.onXmppStanza(aStanza); + let name = aStanza.qName; + if (name == "presence") { + this._account.onPresenceStanza(aStanza); + } else if (name == "message") { + this._account.onMessageStanza(aStanza); + } else if (name == "iq") { + this._account.onIQStanza(aStanza); + } + }, + }, + onXmppStanza(aStanza) { + this.ERROR("should not be reached\n"); + }, +}; diff --git a/comm/chat/protocols/xmpp/xmpp-xml.sys.mjs b/comm/chat/protocols/xmpp/xmpp-xml.sys.mjs new file mode 100644 index 0000000000..9d8c4ca523 --- /dev/null +++ b/comm/chat/protocols/xmpp/xmpp-xml.sys.mjs @@ -0,0 +1,508 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +import { SAX } from "resource:///modules/sax.sys.mjs"; + +var NS = { + xml: "http://www.w3.org/XML/1998/namespace", + xhtml: "http://www.w3.org/1999/xhtml", + xhtml_im: "http://jabber.org/protocol/xhtml-im", + + // auth + client: "jabber:client", + streams: "http://etherx.jabber.org/streams", + stream: "urn:ietf:params:xml:ns:xmpp-streams", + sasl: "urn:ietf:params:xml:ns:xmpp-sasl", + tls: "urn:ietf:params:xml:ns:xmpp-tls", + bind: "urn:ietf:params:xml:ns:xmpp-bind", + session: "urn:ietf:params:xml:ns:xmpp-session", + auth: "jabber:iq:auth", + auth_feature: "http://jabber.org/features/iq-auth", + http_bind: "http://jabber.org/protocol/httpbind", + http_auth: "http://jabber.org/protocol/http-auth", + xbosh: "urn:xmpp:xbosh", + + private: "jabber:iq:private", + xdata: "jabber:x:data", + + // roster + roster: "jabber:iq:roster", + roster_versioning: "urn:xmpp:features:rosterver", + roster_delimiter: "roster:delimiter", + + // privacy lists + privacy: "jabber:iq:privacy", + + // discovering + disco_info: "http://jabber.org/protocol/disco#info", + disco_items: "http://jabber.org/protocol/disco#items", + caps: "http://jabber.org/protocol/caps", + + // addressing + address: "http://jabber.org/protocol/address", + + muc_user: "http://jabber.org/protocol/muc#user", + muc_owner: "http://jabber.org/protocol/muc#owner", + muc_admin: "http://jabber.org/protocol/muc#admin", + muc_rooms: "http://jabber.org/protocol/muc#rooms", + conference: "jabber:x:conference", + muc: "http://jabber.org/protocol/muc", + register: "jabber:iq:register", + delay: "urn:xmpp:delay", + delay_legacy: "jabber:x:delay", + bookmarks: "storage:bookmarks", + chatstates: "http://jabber.org/protocol/chatstates", + event: "jabber:x:event", + stanzas: "urn:ietf:params:xml:ns:xmpp-stanzas", + vcard: "vcard-temp", + vcard_update: "vcard-temp:x:update", + ping: "urn:xmpp:ping", + carbons: "urn:xmpp:carbons:2", + + geoloc: "http://jabber.org/protocol/geoloc", + geoloc_notify: "http://jabber.org/protocol/geoloc+notify", + mood: "http://jabber.org/protocol/mood", + tune: "http://jabber.org/protocol/tune", + nick: "http://jabber.org/protocol/nick", + nick_notify: "http://jabber.org/protocol/nick+notify", + activity: "http://jabber.org/protocol/activity", + rsm: "http://jabber.org/protocol/rsm", + last: "jabber:iq:last", + version: "jabber:iq:version", + avatar_data: "urn:xmpp:avatar:data", + avatar_data_notify: "urn:xmpp:avatar:data+notify", + avatar_metadata: "urn:xmpp:avatar:metadata", + avatar_metadata_notify: "urn:xmpp:avatar:metadata+notify", + pubsub: "http://jabber.org/protocol/pubsub", + pubsub_event: "http://jabber.org/protocol/pubsub#event", +}; + +var TOP_LEVEL_ELEMENTS = { + message: "jabber:client", + presence: "jabber:client", + iq: "jabber:client", + "stream:features": "http://etherx.jabber.org/streams", + proceed: "urn:ietf:params:xml:ns:xmpp-tls", + failure: [ + "urn:ietf:params:xml:ns:xmpp-tls", + "urn:ietf:params:xml:ns:xmpp-sasl", + ], + success: "urn:ietf:params:xml:ns:xmpp-sasl", + challenge: "urn:ietf:params:xml:ns:xmpp-sasl", + error: "urn:ietf:params:xml:ns:xmpp-streams", +}; + +// Features that we support in XMPP. +// Don't forget to add your new features here. +export var SupportedFeatures = [ + NS.chatstates, + NS.conference, + NS.disco_info, + NS.last, + NS.muc, + NS.ping, + NS.vcard, + NS.version, +]; + +/* Stanza Builder */ +export var Stanza = { + NS, + + /* Create a presence stanza */ + presence: (aAttr, aData) => Stanza.node("presence", null, aAttr, aData), + + /* Create a message stanza */ + message(aTo, aMsg, aState, aAttr = {}, aData = []) { + aAttr.to = aTo; + if (!("type" in aAttr)) { + aAttr.type = "chat"; + } + + if (aMsg) { + aData.push(Stanza.node("body", null, null, aMsg)); + } + + if (aState) { + aData.push(Stanza.node(aState, Stanza.NS.chatstates)); + } + + return Stanza.node("message", null, aAttr, aData); + }, + + /* Create a iq stanza */ + iq(aType, aId, aTo, aData) { + let attrs = { type: aType }; + if (aId) { + attrs.id = aId; + } + if (aTo) { + attrs.to = aTo; + } + return this.node("iq", null, attrs, aData); + }, + + /* Create a XML node */ + node(aName, aNs, aAttr, aData) { + let node = new XMLNode(null, aNs, aName, aName, aAttr); + if (aData) { + if (!Array.isArray(aData)) { + aData = [aData]; + } + for (let child of aData) { + node[typeof child == "string" ? "addText" : "addChild"](child); + } + } + + return node; + }, +}; + +/* Text node + * Contains a text */ +function TextNode(aText) { + this.text = aText; +} +TextNode.prototype = { + get type() { + return "text"; + }, + + append(aText) { + this.text += aText; + }, + + /* For debug purposes, returns an indented (unencoded) string */ + convertToString(aIndent) { + return aIndent + this.text + "\n"; + }, + + /* Returns the encoded XML */ + getXML() { + return Cc["@mozilla.org/txttohtmlconv;1"] + .getService(Ci.mozITXTToHTMLConv) + .scanTXT(this.text, Ci.mozITXTToHTMLConv.kEntities); + }, + + /* To read the unencoded data. */ + get innerText() { + return this.text; + }, +}; + +/* XML node */ +/* https://www.w3.org/TR/2008/REC-xml-20081126 */ +/* aUri is the namespace. */ +/* aLocalName must have value, otherwise throws. */ +/* aAttr is an object */ +/* Example: <f:a xmlns:f='g' d='1'> is parsed to + uri/namespace='g', localName='a', qName='f:a', attributes={d='1'} */ +function XMLNode( + aParentNode, + aUri, + aLocalName, + aQName = aLocalName, + aAttr = {} +) { + if (!aLocalName) { + throw new Error("aLocalName must have value"); + } + + this._parentNode = aParentNode; // Used only for parsing + this.uri = aUri; + this.localName = aLocalName; + this.qName = aQName; + this.attributes = {}; + this.children = []; + + for (let attributeName in aAttr) { + // Each attribute specification has a name and a value. + if (aAttr[attributeName]) { + this.attributes[attributeName] = aAttr[attributeName]; + } + } +} +XMLNode.prototype = { + get type() { + return "node"; + }, + + /* Add a new child node */ + addChild(aNode) { + this.children.push(aNode); + }, + + /* Add text node */ + addText(aText) { + let lastIndex = this.children.length - 1; + if (lastIndex >= 0 && this.children[lastIndex] instanceof TextNode) { + this.children[lastIndex].append(aText); + } else { + this.children.push(new TextNode(aText)); + } + }, + + /* Get child elements by namespace */ + getChildrenByNS(aNS) { + return this.children.filter(c => c.uri == aNS); + }, + + /* Get the first element anywhere inside the node (including child nodes) + that matches the query. + A query consists of an array of localNames. */ + getElement(aQuery) { + if (aQuery.length == 0) { + return this; + } + + let nq = aQuery.slice(1); + for (let child of this.children) { + if (child.type == "text" || child.localName != aQuery[0]) { + continue; + } + let n = child.getElement(nq); + if (n) { + return n; + } + } + + return null; + }, + + /* Get all elements of the node (including child nodes) that match the query. + A query consists of an array of localNames. */ + getElements(aQuery) { + if (aQuery.length == 0) { + return [this]; + } + + let c = this.getChildren(aQuery[0]); + let nq = aQuery.slice(1); + let res = []; + for (let child of c) { + let n = child.getElements(nq); + res = res.concat(n); + } + + return res; + }, + + /* Get immediate children by the node name */ + getChildren(aName) { + return this.children.filter(c => c.type != "text" && c.localName == aName); + }, + + // Test if the node is a stanza and its namespace is valid. + isXmppStanza() { + if (!TOP_LEVEL_ELEMENTS.hasOwnProperty(this.qName)) { + return false; + } + let ns = TOP_LEVEL_ELEMENTS[this.qName]; + return ns == this.uri || (Array.isArray(ns) && ns.includes(this.uri)); + }, + + /* Returns indented XML */ + convertToString(aIndent = "") { + let s = + aIndent + "<" + this.qName + this._getXmlns() + this._getAttributeText(); + let content = ""; + for (let child of this.children) { + content += child.convertToString(aIndent + " "); + } + return ( + s + + (content ? ">\n" + content + aIndent + "</" + this.qName : "/") + + ">\n" + ); + }, + + /* Returns the XML */ + getXML() { + let s = "<" + this.qName + this._getXmlns() + this._getAttributeText(); + let innerXML = this.innerXML; + return s + (innerXML ? ">" + innerXML + "</" + this.qName : "/") + ">"; + }, + + get innerXML() { + return this.children.map(c => c.getXML()).join(""); + }, + get innerText() { + return this.children.map(c => c.innerText).join(""); + }, + + /* Private methods */ + _getXmlns() { + return this.uri ? ' xmlns="' + this.uri + '"' : ""; + }, + _getAttributeText() { + let s = ""; + for (let name in this.attributes) { + s += " " + name + '="' + this.attributes[name] + '"'; + } + return s; + }, +}; + +export function XMPPParser(aListener) { + this._listener = aListener; + + // We only get tagName from onclosetag callback, but we need more, so save the + // opening tags. + let tagStack = []; + this._parser = SAX.parser(true, { xmlns: true, lowercase: true }); + this._parser.onopentag = node => { + if (this._parser.error) { + // sax-js doesn't stop on error, but we want to. + return; + } + let attrs = {}; + for (let [name, attr] of Object.entries(node.attributes)) { + if (name == "xmlns") { + continue; + } + attrs[name] = attr.value; + } + this.startElement(node.uri, node.local, node.name, attrs); + tagStack.push(node); + }; + this._parser.onclosetag = tagName => { + if (this._parser.error) { + return; + } + let node = tagStack.pop(); + if (tagName == node.name) { + this.endElement(node.uri, node.local, node.name); + } else { + this.error(`Unexpected </${tagName}>, expecting </${node.name}>`); + } + }; + this._parser.ontext = t => { + if (this._parser.error) { + return; + } + this.characters(t); + }; + this._parser.onerror = this.error; +} + +XMPPParser.prototype = { + _decoder: new TextDecoder(), + _destroyPending: false, + destroy() { + delete this._listener; + + try { + this._parser.close(); + } catch (e) {} + delete this._parser; + }, + + _logReceivedData(aData) { + this._listener.LOG("received:\n" + aData); + }, + /** + * Decodes the byte string to UTF-8 (via byte array) before feeding it to the + * SAXML parser. + * + * @param {string} data - Raw XML byte string. + */ + onDataAvailable(data) { + let bytes = new Uint8Array(data.length); + for (let i = 0; i < data.length; i++) { + bytes[i] = data.charCodeAt(i); + } + let utf8Data = this._decoder.decode(bytes); + this._parser.write(utf8Data); + }, + + startElement(aUri, aLocalName, aQName, aAttributes) { + if (aQName == "stream:stream") { + let node = new XMLNode(null, aUri, aLocalName, aQName, aAttributes); + // The node we created doesn't have children, but + // <stream:stream> isn't closed, so avoid displaying /> at the end. + this._logReceivedData(node.convertToString().slice(0, -3) + ">\n"); + + if ("_node" in this) { + this._listener.onXMLError( + "unexpected-stream-start", + "stream:stream inside an already started stream" + ); + return; + } + + this._listener._streamId = node.attributes.id; + if (!("version" in node.attributes)) { + this._listener.startLegacyAuth(); + } + + this._node = null; + return; + } + + let node = new XMLNode(this._node, aUri, aLocalName, aQName, aAttributes); + if (this._node) { + this._node.addChild(node); + } + + this._node = node; + }, + + characters(aCharacters) { + if (!this._node) { + // Ignore whitespace received on the stream to keep the connection alive. + if (aCharacters.trim()) { + this._listener.onXMLError( + "parsing-characters", + "No parent for characters: " + aCharacters + ); + } + return; + } + + this._node.addText(aCharacters); + }, + + endElement(aUri, aLocalName, aQName) { + if (aQName == "stream:stream") { + this._logReceivedData("</stream:stream>"); + delete this._node; + return; + } + + if (!this._node) { + this._listener.onXMLError( + "parsing-node", + "No parent for node : " + aLocalName + ); + return; + } + + // RFC 6120 (8): XML Stanzas. + // Checks if the node is the root and it's valid. + if (!this._node._parentNode) { + if (this._node.isXmppStanza()) { + this._logReceivedData(this._node.convertToString()); + try { + this._listener.onXmppStanza(this._node); + } catch (e) { + console.error(e); + dump(e + "\n"); + } + } else { + this._listener.onXMLError( + "parsing-node", + "Root node " + aLocalName + " is not valid." + ); + } + } + + this._node = this._node._parentNode; + }, + + error(aError) { + if (this._listener) { + this._listener.onXMLError("parse-error", aError); + } + }, +}; diff --git a/comm/chat/protocols/xmpp/xmpp.sys.mjs b/comm/chat/protocols/xmpp/xmpp.sys.mjs new file mode 100644 index 0000000000..08fdcd5629 --- /dev/null +++ b/comm/chat/protocols/xmpp/xmpp.sys.mjs @@ -0,0 +1,106 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ + +import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs"; +import { l10nHelper } from "resource:///modules/imXPCOMUtils.sys.mjs"; +import { GenericProtocolPrototype } from "resource:///modules/jsProtoHelper.sys.mjs"; + +const lazy = {}; + +XPCOMUtils.defineLazyGetter(lazy, "_", () => + l10nHelper("chrome://chat/locale/xmpp.properties") +); +ChromeUtils.defineESModuleGetters(lazy, { + XMPPAccountPrototype: "resource:///modules/xmpp-base.sys.mjs", +}); + +XPCOMUtils.defineLazyGetter(lazy, "XMPPAccount", () => { + function XMPPAccount(aProtoInstance, aImAccount) { + this._init(aProtoInstance, aImAccount); + } + XMPPAccount.prototype = lazy.XMPPAccountPrototype; + return XMPPAccount; +}); + +export function XMPPProtocol() { + this.commands = ChromeUtils.importESModule( + "resource:///modules/xmpp-commands.sys.mjs" + ).commands; + this.registerCommands(); +} + +XMPPProtocol.prototype = { + __proto__: GenericProtocolPrototype, + get normalizedName() { + return "jabber"; + }, + get name() { + return "XMPP"; + }, + get iconBaseURI() { + return "chrome://prpl-jabber/skin/"; + }, + getAccount(aImAccount) { + return new lazy.XMPPAccount(this, aImAccount); + }, + + usernameSplits: [ + { + get label() { + return lazy._("options.domain"); + }, + separator: "@", + defaultValue: "jabber.org", + }, + ], + + options: { + resource: { + get label() { + return lazy._("options.resource"); + }, + default: "", + }, + priority: { + get label() { + return lazy._("options.priority"); + }, + default: 0, + }, + connection_security: { + get label() { + return lazy._("options.connectionSecurity"); + }, + listValues: { + get require_tls() { + return lazy._("options.connectionSecurity.requireEncryption"); + }, + get opportunistic_tls() { + return lazy._("options.connectionSecurity.opportunisticTLS"); + }, + get allow_unencrypted_plain_auth() { + return lazy._("options.connectionSecurity.allowUnencryptedAuth"); + }, + // "old_ssl" and "none" are also supported, but not exposed in the UI. + // Any unknown value will fallback to the opportunistic_tls behavior. + }, + default: "require_tls", + }, + server: { + get label() { + return lazy._("options.connectServer"); + }, + default: "", + }, + port: { + get label() { + return lazy._("options.connectPort"); + }, + default: 5222, + }, + }, + get chatHasTopic() { + return true; + }, +}; |