Converse converse.js

Source: headless/utils/stanza.js

  1. import log from '../log.js';
  2. import { Strophe } from 'strophe.js';
  3. const PARSE_ERROR_NS = 'http://www.w3.org/1999/xhtml';
  4. export function toStanza (string, throwErrorIfInvalidNS) {
  5. const doc = Strophe.xmlHtmlNode(string);
  6. if (doc.getElementsByTagNameNS(PARSE_ERROR_NS, 'parsererror').length) {
  7. throw new Error(`Parser Error: ${string}`);
  8. }
  9. const node = doc.firstElementChild;
  10. if (
  11. ['message', 'iq', 'presence'].includes(node.nodeName.toLowerCase()) &&
  12. node.namespaceURI !== 'jabber:client' &&
  13. node.namespaceURI !== 'jabber:server'
  14. ) {
  15. const err_msg = `Invalid namespaceURI ${node.namespaceURI}`;
  16. log.error(err_msg);
  17. if (throwErrorIfInvalidNS) throw new Error(err_msg);
  18. }
  19. return node;
  20. }
  21. /**
  22. * A Stanza represents a XML element used in XMPP (commonly referred to as
  23. * stanzas).
  24. */
  25. class Stanza {
  26. constructor (strings, values) {
  27. this.strings = strings;
  28. this.values = values;
  29. }
  30. toString () {
  31. this.string = this.string ||
  32. this.strings.reduce((acc, str) => {
  33. const idx = this.strings.indexOf(str);
  34. const value = this.values.length > idx ? this.values[idx].toString() : '';
  35. return acc + str + value;
  36. }, '');
  37. return this.string;
  38. }
  39. tree () {
  40. this.node = this.node ?? toStanza(this.toString(), true);
  41. return this.node;
  42. }
  43. }
  44. /**
  45. * Tagged template literal function which generates {@link Stanza } objects
  46. *
  47. * Similar to the `html` function, from Lit.
  48. *
  49. * @example stx`<presence type="${type}"><show>${show}</show></presence>`
  50. */
  51. export function stx (strings, ...values) {
  52. return new Stanza(strings, values);
  53. }