'+'
or '-'
then the bits to add or remove, e.g. '+R-W'
or '-PS'
.\n * - a new value of access mode\n *\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {number} val - access mode value to update.\n * @param {string} upd - update to apply to val.\n * @returns {number} - updated access mode.\n */\n static update(val, upd) {\n if (!upd || typeof upd != 'string') {\n return val;\n }\n\n let action = upd.charAt(0);\n if (action == '+' || action == '-') {\n let val0 = val;\n // Split delta-string like '+ABC-DEF+Z' into an array of parts including + and -.\n const parts = upd.split(/([-+])/);\n // Starting iteration from 1 because String.split() creates an array with the first empty element.\n // Iterating by 2 because we parse pairs +/- then data.\n for (let i = 1; i < parts.length - 1; i += 2) {\n action = parts[i];\n const m0 = AccessMode.decode(parts[i + 1]);\n if (m0 == AccessMode._INVALID) {\n return val;\n }\n if (m0 == null) {\n continue;\n }\n if (action === '+') {\n val0 |= m0;\n } else if (action === '-') {\n val0 &= ~m0;\n }\n }\n val = val0;\n } else {\n // The string is an explicit new value 'ABC' rather than delta.\n const val0 = AccessMode.decode(upd);\n if (val0 != AccessMode._INVALID) {\n val = val0;\n }\n }\n\n return val;\n }\n /**\n * Bits present in a1 but missing in a2.\n *\n * @static\n * @memberof Tinode\n *\n * @param {number | string} a1 - access mode to subtract from.\n * @param {number | string} a2 - access mode to subtract.\n * @returns {number} access mode with bits present in a1
but missing in a2
.\n */\n static diff(a1, a2) {\n a1 = AccessMode.decode(a1);\n a2 = AccessMode.decode(a2);\n\n if (a1 == AccessMode._INVALID || a2 == AccessMode._INVALID) {\n return AccessMode._INVALID;\n }\n return a1 & ~a2;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Custom formatter\n */\n toString() {\n return '{\"mode\": \"' + AccessMode.encode(this.mode) +\n '\", \"given\": \"' + AccessMode.encode(this.given) +\n '\", \"want\": \"' + AccessMode.encode(this.want) + '\"}';\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Converts numeric values to strings.\n */\n jsonHelper() {\n return {\n mode: AccessMode.encode(this.mode),\n given: AccessMode.encode(this.given),\n want: AccessMode.encode(this.want)\n };\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign value to 'mode'.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} m - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setMode(m) {\n this.mode = AccessMode.decode(m);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update mode
value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateMode(u) {\n this.mode = AccessMode.update(this.mode, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get mode
value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - mode
value.\n */\n getMode() {\n return AccessMode.encode(this.mode);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign given
value.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} g - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setGiven(g) {\n this.given = AccessMode.decode(g);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'given' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateGiven(u) {\n this.given = AccessMode.update(this.given, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get 'given' value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - given value.\n */\n getGiven() {\n return AccessMode.encode(this.given);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign 'want' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} w - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setWant(w) {\n this.want = AccessMode.decode(w);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'want' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateWant(u) {\n this.want = AccessMode.update(this.want, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get 'want' value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - want value.\n */\n getWant() {\n return AccessMode.encode(this.want);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get permissions present in 'want' but missing in 'given'.\n * Inverse of {@link Tinode.AccessMode#getExcessive}\n *\n * @memberof Tinode.AccessMode\n *\n * @returns {string} permissions present in want but missing in given.\n */\n getMissing() {\n return AccessMode.encode(this.want & ~this.given);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get permissions present in 'given' but missing in 'want'.\n * Inverse of {@link Tinode.AccessMode#getMissing}\n * @memberof Tinode.AccessMode\n *\n * @returns {string} permissions present in given but missing in want.\n */\n getExcessive() {\n return AccessMode.encode(this.given & ~this.want);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'want', 'give', and 'mode' values.\n * @memberof Tinode.AccessMode\n *\n * @param {AccessMode} val - new access mode value.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateAll(val) {\n if (val) {\n this.updateGiven(val.given);\n this.updateWant(val.want);\n this.mode = this.given & this.want;\n }\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Owner (O) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isOwner(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._OWNER);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Presence (P) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isPresencer(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._PRES);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Presence (P) flag is NOT set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isMuted(side) {\n return !this.isPresencer(side);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Join (J) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isJoiner(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._JOIN);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Reader (R) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isReader(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._READ);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Writer (W) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isWriter(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._WRITE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Approver (A) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isApprover(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._APPROVE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if either one of Owner (O) or Approver (A) flags is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isAdmin(side) {\n return this.isOwner(side) || this.isApprover(side);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if either one of Owner (O), Approver (A), or Sharer (S) flags is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isSharer(side) {\n return this.isAdmin(side) || AccessMode.#checkFlag(this, side, AccessMode._SHARE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Deleter (D) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isDeleter(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._DELETE);\n }\n}\n\nAccessMode._NONE = 0x00;\nAccessMode._JOIN = 0x01;\nAccessMode._READ = 0x02;\nAccessMode._WRITE = 0x04;\nAccessMode._PRES = 0x08;\nAccessMode._APPROVE = 0x10;\nAccessMode._SHARE = 0x20;\nAccessMode._DELETE = 0x40;\nAccessMode._OWNER = 0x80;\n\nAccessMode._BITMASK = AccessMode._JOIN | AccessMode._READ | AccessMode._WRITE | AccessMode._PRES |\n AccessMode._APPROVE | AccessMode._SHARE | AccessMode._DELETE | AccessMode._OWNER;\nAccessMode._INVALID = 0x100000;\n","/**\n * @file In-memory sorted cache of objects.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n/**\n * In-memory sorted cache of objects.\n *\n * @class CBuffer\n * @memberof Tinode\n * @protected\n *\n * @param {function} compare custom comparator of objects. Takes two parameters a
and b
;\n * returns -1
if a < b
, 0
if a == b
, 1
otherwise.\n * @param {boolean} unique enforce element uniqueness: when true
replace existing element with a new\n * one on conflict; when false
keep both elements.\n */\nexport default class CBuffer {\n #comparator = undefined;\n #unique = false;\n buffer = [];\n\n constructor(compare_, unique_) {\n this.#comparator = compare_ || ((a, b) => {\n return a === b ? 0 : a < b ? -1 : 1;\n });\n this.#unique = unique_;\n }\n\n #findNearest(elem, arr, exact) {\n let start = 0;\n let end = arr.length - 1;\n let pivot = 0;\n let diff = 0;\n let found = false;\n\n while (start <= end) {\n pivot = (start + end) / 2 | 0;\n diff = this.#comparator(arr[pivot], elem);\n if (diff < 0) {\n start = pivot + 1;\n } else if (diff > 0) {\n end = pivot - 1;\n } else {\n found = true;\n break;\n }\n }\n if (found) {\n return {\n idx: pivot,\n exact: true\n };\n }\n if (exact) {\n return {\n idx: -1\n };\n }\n // Not exact - insertion point\n return {\n idx: diff < 0 ? pivot + 1 : pivot\n };\n }\n\n // Insert element into a sorted array.\n #insertSorted(elem, arr) {\n const found = this.#findNearest(elem, arr, false);\n const count = (found.exact && this.#unique) ? 1 : 0;\n arr.splice(found.idx, count, elem);\n return arr;\n }\n\n /**\n * Get an element at the given position.\n * @memberof Tinode.CBuffer#\n * @param {number} at - Position to fetch from.\n * @returns {Object} Element at the given position or undefined
.\n */\n getAt(at) {\n return this.buffer[at];\n }\n\n /**\n * Convenience method for getting the element from the end of the buffer.\n * @memberof Tinode.CBuffer#\n * @param {number} at - position to fetch from, counting from the end;\n * undefined
or null
mean \"last\".\n * @returns {Object} The last element in the buffer or undefined
if buffer is empty.\n */\n getLast(at) {\n at |= 0;\n return this.buffer.length > at ? this.buffer[this.buffer.length - 1 - at] : undefined;\n }\n\n /**\n * Add new element(s) to the buffer. Variadic: takes one or more arguments. If an array is passed as a single\n * argument, its elements are inserted individually.\n * @memberof Tinode.CBuffer#\n *\n * @param {...Object|Array} - One or more objects to insert.\n */\n put() {\n let insert;\n // inspect arguments: if array, insert its elements, if one or more non-array arguments, insert them one by one\n if (arguments.length == 1 && Array.isArray(arguments[0])) {\n insert = arguments[0];\n } else {\n insert = arguments;\n }\n for (let idx in insert) {\n this.#insertSorted(insert[idx], this.buffer);\n }\n }\n\n /**\n * Remove element at the given position.\n * @memberof Tinode.CBuffer#\n * @param {number} at - Position to delete at.\n * @returns {Object} Element at the given position or undefined
.\n */\n delAt(at) {\n at |= 0;\n let r = this.buffer.splice(at, 1);\n if (r && r.length > 0) {\n return r[0];\n }\n return undefined;\n }\n\n /**\n * Remove elements between two positions.\n * @memberof Tinode.CBuffer#\n * @param {number} since - Position to delete from (inclusive).\n * @param {number} before - Position to delete to (exclusive).\n *\n * @returns {Array} array of removed elements (could be zero length).\n */\n delRange(since, before) {\n return this.buffer.splice(since, before - since);\n }\n\n /**\n * Return the number of elements the buffer holds.\n * @memberof Tinode.CBuffer#\n * @return {number} Number of elements in the buffer.\n */\n length() {\n return this.buffer.length;\n }\n\n /**\n * Reset the buffer discarding all elements\n * @memberof Tinode.CBuffer#\n */\n reset() {\n this.buffer = [];\n }\n\n /**\n * Callback for iterating contents of buffer. See {@link Tinode.CBuffer#forEach}.\n * @callback ForEachCallbackType\n * @memberof Tinode.CBuffer#\n * @param {Object} elem - Current element of the buffer.\n * @param {Object} prev - Previous element of the buffer.\n * @param {Object} next - Next element of the buffer.\n * @param {number} index - Index of the current element.\n */\n\n /**\n * Apply given callback
to all elements of the buffer.\n * @memberof Tinode.CBuffer#\n *\n * @param {Tinode.ForEachCallbackType} callback - Function to call for each element.\n * @param {number} startIdx - Optional index to start iterating from (inclusive).\n * @param {number} beforeIdx - Optional index to stop iterating before (exclusive).\n * @param {Object} context - calling context (i.e. value of this
in callback)\n */\n forEach(callback, startIdx, beforeIdx, context) {\n startIdx = startIdx | 0;\n beforeIdx = beforeIdx || this.buffer.length;\n\n for (let i = startIdx; i < beforeIdx; i++) {\n callback.call(context, this.buffer[i],\n (i > startIdx ? this.buffer[i - 1] : undefined),\n (i < beforeIdx - 1 ? this.buffer[i + 1] : undefined), i);\n }\n }\n\n /**\n * Find element in buffer using buffer's comparison function.\n * @memberof Tinode.CBuffer#\n *\n * @param {Object} elem - element to find.\n * @param {boolean=} nearest - when true and exact match is not found, return the nearest element (insertion point).\n * @returns {number} index of the element in the buffer or -1.\n */\n find(elem, nearest) {\n const {\n idx\n } = this.#findNearest(elem, this.buffer, !nearest);\n return idx;\n }\n\n /**\n * Callback for filtering the buffer. See {@link Tinode.CBuffer#filter}.\n * @callback FilterCallbackType\n * @memberof Tinode.CBuffer#\n * @param {Object} elem - Current element of the buffer.\n * @param {number} index - Index of the current element.\n * @returns {boolen} true
to keep the element, false
to remove.\n */\n\n /**\n * Remove all elements that do not pass the test implemented by the provided callback function.\n * @memberof Tinode.CBuffer#\n *\n * @param {Tinode.FilterCallbackType} callback - Function to call for each element.\n * @param {Object} context - calling context (i.e. value of this
in the callback)\n */\n filter(callback, context) {\n let count = 0;\n for (let i = 0; i < this.buffer.length; i++) {\n if (callback.call(context, this.buffer[i], i)) {\n this.buffer[count] = this.buffer[i];\n count++;\n }\n }\n\n this.buffer.splice(count);\n }\n\n /**\n * Check if buffer is empty.\n * @returns {boolean} true
if the buffer is empty, false
otherwise.\n */\n isEmpty() {\n return this.buffer.length == 0;\n }\n}\n","/**\n * @file Throwable error with numeric error code.\n *\n * @copyright 2015-2023 Tinode LLC.\n */\n'use strict';\n\nexport default class CommError extends Error {\n constructor(message, code) {\n super(`${message} (${code})`);\n this.name = 'CommError';\n this.code = code;\n }\n}\n","/**\n * @file Global constants and configuration parameters.\n *\n * @copyright 2015-2023 Tinode LLC\n */\n'use strict';\n\nimport {\n PACKAGE_VERSION\n} from '../version.js';\n\n// Global constants\nexport const PROTOCOL_VERSION = '0'; // Major component of the version, e.g. '0' in '0.17.1'.\nexport const VERSION = PACKAGE_VERSION || '0.21';\nexport const LIBRARY = 'tinodejs/' + VERSION;\n\n// Topic name prefixes.\nexport const TOPIC_NEW = 'new';\nexport const TOPIC_NEW_CHAN = 'nch';\nexport const TOPIC_ME = 'me';\nexport const TOPIC_FND = 'fnd';\nexport const TOPIC_SYS = 'sys';\nexport const TOPIC_CHAN = 'chn';\nexport const TOPIC_GRP = 'grp';\nexport const TOPIC_P2P = 'p2p';\nexport const USER_NEW = 'new';\n\n// Starting value of a locally-generated seqId used for pending messages.\nexport const LOCAL_SEQID = 0xFFFFFFF;\n\n// Status codes.\nexport const MESSAGE_STATUS_NONE = 0; // Status not assigned.\nexport const MESSAGE_STATUS_QUEUED = 10; // Local ID assigned, in progress to be sent.\nexport const MESSAGE_STATUS_SENDING = 20; // Transmission started.\nexport const MESSAGE_STATUS_FAILED = 30; // At least one attempt was made to send the message.\nexport const MESSAGE_STATUS_FATAL = 40; // Message sending failed and it should not be retried.\nexport const MESSAGE_STATUS_SENT = 50; // Delivered to the server.\nexport const MESSAGE_STATUS_RECEIVED = 60; // Received by the client.\nexport const MESSAGE_STATUS_READ = 70; // Read by the user.\nexport const MESSAGE_STATUS_TO_ME = 80; // The message is received from another user.\n\n// Reject unresolved futures after this many milliseconds.\nexport const EXPIRE_PROMISES_TIMEOUT = 5_000;\n// Periodicity of garbage collection of unresolved futures.\nexport const EXPIRE_PROMISES_PERIOD = 1_000;\n\n// Delay before acknowledging that a message was recived.\nexport const RECV_TIMEOUT = 100;\n\n// Default number of messages to pull into memory from persistent cache.\nexport const DEFAULT_MESSAGES_PAGE = 24;\n\n// Unicode DEL character indicating data was deleted.\nexport const DEL_CHAR = '\\u2421';\n","/**\n * @file Abstraction layer for websocket and long polling connections.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport CommError from './comm-error.js';\nimport {\n jsonParseHelper\n} from './utils.js';\n\nlet WebSocketProvider;\nlet XHRProvider;\n\n// Error code to return in case of a network problem.\nconst NETWORK_ERROR = 503;\nconst NETWORK_ERROR_TEXT = \"Connection failed\";\n\n// Error code to return when user disconnected from server.\nconst NETWORK_USER = 418;\nconst NETWORK_USER_TEXT = \"Disconnected by client\";\n\n// Settings for exponential backoff\nconst _BOFF_BASE = 2000; // 2000 milliseconds, minimum delay between reconnects\nconst _BOFF_MAX_ITER = 10; // Maximum delay between reconnects 2^10 * 2000 ~ 34 minutes\nconst _BOFF_JITTER = 0.3; // Add random delay\n\n// Helper function for creating an endpoint URL.\nfunction makeBaseUrl(host, protocol, version, apiKey) {\n let url = null;\n\n if (['http', 'https', 'ws', 'wss'].includes(protocol)) {\n url = `${protocol}://${host}`;\n if (url.charAt(url.length - 1) !== '/') {\n url += '/';\n }\n url += 'v' + version + '/channels';\n if (['http', 'https'].includes(protocol)) {\n // Long polling endpoint ends with \"lp\", i.e.\n // '/v0/channels/lp' vs just '/v0/channels' for ws\n url += '/lp';\n }\n url += '?apikey=' + apiKey;\n }\n return url;\n}\n\n/**\n * An abstraction for a websocket or a long polling connection.\n *\n * @class Connection\n * @memberof Tinode\n\n * @param {Object} config - configuration parameters.\n * @param {string} config.host - Host name and optional port number to connect to.\n * @param {string} config.apiKey - API key generated by keygen
.\n * @param {string} config.transport - Network transport to use, either \"ws\"/\"wss\"
for websocket or\n * lp
for long polling.\n * @param {boolean} config.secure - Use Secure WebSocket if true
.\n * @param {string} version_ - Major value of the protocol version, e.g. '0' in '0.17.1'.\n * @param {boolean} autoreconnect_ - If connection is lost, try to reconnect automatically.\n */\nexport default class Connection {\n // Logger, does nothing by default.\n static #log = _ => {};\n\n #boffTimer = null;\n #boffIteration = 0;\n #boffClosed = false; // Indicator if the socket was manually closed - don't autoreconnect if true.\n\n // Websocket.\n #socket = null;\n\n host;\n secure;\n apiKey;\n\n version;\n autoreconnect;\n\n initialized;\n\n // (config.host, config.apiKey, config.transport, config.secure), PROTOCOL_VERSION, true\n constructor(config, version_, autoreconnect_) {\n this.host = config.host;\n this.secure = config.secure;\n this.apiKey = config.apiKey;\n\n this.version = version_;\n this.autoreconnect = autoreconnect_;\n\n if (config.transport === 'lp') {\n // explicit request to use long polling\n this.#init_lp();\n this.initialized = 'lp';\n } else if (config.transport === 'ws') {\n // explicit request to use web socket\n // if websockets are not available, horrible things will happen\n this.#init_ws();\n this.initialized = 'ws';\n }\n\n if (!this.initialized) {\n // Invalid or undefined network transport.\n Connection.#log(\"Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.\");\n throw new Error(\"Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.\");\n }\n }\n\n /**\n * To use Connection in a non browser context, supply WebSocket and XMLHttpRequest providers.\n * @static\n * @memberof Connection\n * @param wsProvider WebSocket provider, e.g. for nodeJS , require('ws')
.\n * @param xhrProvider XMLHttpRequest provider, e.g. for node require('xhr')
.\n */\n static setNetworkProviders(wsProvider, xhrProvider) {\n WebSocketProvider = wsProvider;\n XHRProvider = xhrProvider;\n }\n\n /**\n * Assign a non-default logger.\n * @static\n * @memberof Connection\n * @param {function} l variadic logging function.\n */\n static set logger(l) {\n Connection.#log = l;\n }\n\n /**\n * Initiate a new connection\n * @memberof Tinode.Connection#\n * @param {string} host_ Host name to connect to; if null
the old host name will be used.\n * @param {boolean} force Force new connection even if one already exists.\n * @return {Promise} Promise resolved/rejected when the connection call completes, resolution is called without\n * parameters, rejection passes the {Error} as parameter.\n */\n connect(host_, force) {\n return Promise.reject(null);\n }\n\n /**\n * Try to restore a network connection, also reset backoff.\n * @memberof Tinode.Connection#\n *\n * @param {boolean} force - reconnect even if there is a live connection already.\n */\n reconnect(force) {}\n\n /**\n * Terminate the network connection\n * @memberof Tinode.Connection#\n */\n disconnect() {}\n\n /**\n * Send a string to the server.\n * @memberof Tinode.Connection#\n *\n * @param {string} msg - String to send.\n * @throws Throws an exception if the underlying connection is not live.\n */\n sendText(msg) {}\n\n /**\n * Check if connection is alive.\n * @memberof Tinode.Connection#\n * @returns {boolean} true
if connection is live, false
otherwise.\n */\n isConnected() {\n return false;\n }\n\n /**\n * Get the name of the current network transport.\n * @memberof Tinode.Connection#\n * @returns {string} name of the transport such as \"ws\"
or \"lp\"
.\n */\n transport() {\n return this.initialized;\n }\n\n /**\n * Send network probe to check if connection is indeed live.\n * @memberof Tinode.Connection#\n */\n probe() {\n this.sendText('1');\n }\n\n /**\n * Reset autoreconnect counter to zero.\n * @memberof Tinode.Connection#\n */\n backoffReset() {\n this.#boffReset();\n }\n\n // Backoff implementation - reconnect after a timeout.\n #boffReconnect() {\n // Clear timer\n clearTimeout(this.#boffTimer);\n // Calculate when to fire the reconnect attempt\n const timeout = _BOFF_BASE * (Math.pow(2, this.#boffIteration) * (1.0 + _BOFF_JITTER * Math.random()));\n // Update iteration counter for future use\n this.#boffIteration = (this.#boffIteration >= _BOFF_MAX_ITER ? this.#boffIteration : this.#boffIteration + 1);\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(timeout);\n }\n\n this.#boffTimer = setTimeout(_ => {\n Connection.#log(`Reconnecting, iter=${this.#boffIteration}, timeout=${timeout}`);\n // Maybe the socket was closed while we waited for the timer?\n if (!this.#boffClosed) {\n const prom = this.connect();\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(0, prom);\n } else {\n // Suppress error if it's not used.\n prom.catch(_ => {\n /* do nothing */\n });\n }\n } else if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(-1);\n }\n }, timeout);\n }\n\n // Terminate auto-reconnect process.\n #boffStop() {\n clearTimeout(this.#boffTimer);\n this.#boffTimer = null;\n }\n\n // Reset auto-reconnect iteration counter.\n #boffReset() {\n this.#boffIteration = 0;\n }\n\n // Initialization for long polling.\n #init_lp() {\n const XDR_UNSENT = 0; // Client has been created. open() not called yet.\n const XDR_OPENED = 1; // open() has been called.\n const XDR_HEADERS_RECEIVED = 2; // send() has been called, and headers and status are available.\n const XDR_LOADING = 3; // Downloading; responseText holds partial data.\n const XDR_DONE = 4; // The operation is complete.\n\n // Fully composed endpoint URL, with API key & SID\n let _lpURL = null;\n\n let _poller = null;\n let _sender = null;\n\n let lp_sender = (url_) => {\n const sender = new XHRProvider();\n sender.onreadystatechange = (evt) => {\n if (sender.readyState == XDR_DONE && sender.status >= 400) {\n // Some sort of error response\n throw new CommError(\"LP sender failed\", sender.status);\n }\n };\n\n sender.open('POST', url_, true);\n return sender;\n }\n\n let lp_poller = (url_, resolve, reject) => {\n let poller = new XHRProvider();\n let promiseCompleted = false;\n\n poller.onreadystatechange = evt => {\n if (poller.readyState == XDR_DONE) {\n if (poller.status == 201) { // 201 == HTTP.Created, get SID\n let pkt = JSON.parse(poller.responseText, jsonParseHelper);\n _lpURL = url_ + '&sid=' + pkt.ctrl.params.sid;\n poller = lp_poller(_lpURL);\n poller.send(null);\n if (this.onOpen) {\n this.onOpen();\n }\n\n if (resolve) {\n promiseCompleted = true;\n resolve();\n }\n\n if (this.autoreconnect) {\n this.#boffStop();\n }\n } else if (poller.status > 0 && poller.status < 400) { // 0 = network error; 400 = HTTP.BadRequest\n if (this.onMessage) {\n this.onMessage(poller.responseText);\n }\n poller = lp_poller(_lpURL);\n poller.send(null);\n } else {\n // Don't throw an error here, gracefully handle server errors\n if (reject && !promiseCompleted) {\n promiseCompleted = true;\n reject(poller.responseText);\n }\n if (this.onMessage && poller.responseText) {\n this.onMessage(poller.responseText);\n }\n if (this.onDisconnect) {\n const code = poller.status || (this.#boffClosed ? NETWORK_USER : NETWORK_ERROR);\n const text = poller.responseText || (this.#boffClosed ? NETWORK_USER_TEXT : NETWORK_ERROR_TEXT);\n this.onDisconnect(new CommError(text, code), code);\n }\n\n // Polling has stopped. Indicate it by setting poller to null.\n poller = null;\n if (!this.#boffClosed && this.autoreconnect) {\n this.#boffReconnect();\n }\n }\n }\n };\n // Using POST to avoid caching response by service worker.\n poller.open('POST', url_, true);\n return poller;\n }\n\n this.connect = (host_, force) => {\n this.#boffClosed = false;\n\n if (_poller) {\n if (!force) {\n return Promise.resolve();\n }\n _poller.onreadystatechange = undefined;\n _poller.abort();\n _poller = null;\n }\n\n if (host_) {\n this.host = host_;\n }\n\n return new Promise((resolve, reject) => {\n const url = makeBaseUrl(this.host, this.secure ? 'https' : 'http', this.version, this.apiKey);\n Connection.#log(\"LP connecting to:\", url);\n _poller = lp_poller(url, resolve, reject);\n _poller.send(null);\n }).catch(err => {\n Connection.#log(\"LP connection failed:\", err);\n });\n };\n\n this.reconnect = force => {\n this.#boffStop();\n this.connect(null, force);\n };\n\n this.disconnect = _ => {\n this.#boffClosed = true;\n this.#boffStop();\n\n if (_sender) {\n _sender.onreadystatechange = undefined;\n _sender.abort();\n _sender = null;\n }\n if (_poller) {\n _poller.onreadystatechange = undefined;\n _poller.abort();\n _poller = null;\n }\n\n if (this.onDisconnect) {\n this.onDisconnect(new CommError(NETWORK_USER_TEXT, NETWORK_USER), NETWORK_USER);\n }\n // Ensure it's reconstructed\n _lpURL = null;\n };\n\n this.sendText = (msg) => {\n _sender = lp_sender(_lpURL);\n if (_sender && (_sender.readyState == XDR_OPENED)) { // 1 == OPENED\n _sender.send(msg);\n } else {\n throw new Error(\"Long poller failed to connect\");\n }\n };\n\n this.isConnected = _ => {\n return (_poller && true);\n };\n }\n\n // Initialization for Websocket\n #init_ws() {\n this.connect = (host_, force) => {\n this.#boffClosed = false;\n\n if (this.#socket) {\n if (!force && this.#socket.readyState == this.#socket.OPEN) {\n return Promise.resolve();\n }\n this.#socket.close();\n this.#socket = null;\n }\n\n if (host_) {\n this.host = host_;\n }\n\n return new Promise((resolve, reject) => {\n const url = makeBaseUrl(this.host, this.secure ? 'wss' : 'ws', this.version, this.apiKey);\n\n Connection.#log(\"WS connecting to: \", url);\n\n // It throws when the server is not accessible but the exception cannot be caught:\n // https://stackoverflow.com/questions/31002592/javascript-doesnt-catch-error-in-websocket-instantiation/31003057\n const conn = new WebSocketProvider(url);\n\n conn.onerror = err => {\n reject(err);\n };\n\n conn.onopen = _ => {\n if (this.autoreconnect) {\n this.#boffStop();\n }\n\n if (this.onOpen) {\n this.onOpen();\n }\n\n resolve();\n };\n\n conn.onclose = _ => {\n this.#socket = null;\n\n if (this.onDisconnect) {\n const code = this.#boffClosed ? NETWORK_USER : NETWORK_ERROR;\n this.onDisconnect(new CommError(this.#boffClosed ? NETWORK_USER_TEXT : NETWORK_ERROR_TEXT, code), code);\n }\n\n if (!this.#boffClosed && this.autoreconnect) {\n this.#boffReconnect();\n }\n };\n\n conn.onmessage = evt => {\n if (this.onMessage) {\n this.onMessage(evt.data);\n }\n };\n\n this.#socket = conn;\n });\n }\n\n this.reconnect = force => {\n this.#boffStop();\n this.connect(null, force);\n };\n\n this.disconnect = _ => {\n this.#boffClosed = true;\n this.#boffStop();\n\n if (!this.#socket) {\n return;\n }\n this.#socket.close();\n this.#socket = null;\n };\n\n this.sendText = msg => {\n if (this.#socket && (this.#socket.readyState == this.#socket.OPEN)) {\n this.#socket.send(msg);\n } else {\n throw new Error(\"Websocket is not connected\");\n }\n };\n\n this.isConnected = _ => {\n return (this.#socket && (this.#socket.readyState == this.#socket.OPEN));\n };\n }\n\n // Callbacks:\n\n /**\n * A callback to pass incoming messages to. See {@link Tinode.Connection#onMessage}.\n * @callback Tinode.Connection.OnMessage\n * @memberof Tinode.Connection\n * @param {string} message - Message to process.\n */\n onMessage = undefined;\n\n /**\n * A callback for reporting a dropped connection.\n * @type {function}\n * @memberof Tinode.Connection#\n */\n onDisconnect = undefined;\n\n /**\n * A callback called when the connection is ready to be used for sending. For websockets it's socket open,\n * for long polling it's readyState=1
(OPENED)\n * @type {function}\n * @memberof Tinode.Connection#\n */\n onOpen = undefined;\n\n /**\n * A callback to notify of reconnection attempts. See {@link Tinode.Connection#onAutoreconnectIteration}.\n * @memberof Tinode.Connection\n * @callback AutoreconnectIterationType\n * @param {string} timeout - time till the next reconnect attempt in milliseconds. -1
means reconnect was skipped.\n * @param {Promise} promise resolved or rejected when the reconnect attemp completes.\n *\n */\n /**\n * A callback to inform when the next attampt to reconnect will happen and to receive connection promise.\n * @memberof Tinode.Connection#\n * @type {Tinode.Connection.AutoreconnectIterationType}\n */\n onAutoreconnectIteration = undefined;\n}\n\nConnection.NETWORK_ERROR = NETWORK_ERROR;\nConnection.NETWORK_ERROR_TEXT = NETWORK_ERROR_TEXT;\nConnection.NETWORK_USER = NETWORK_USER;\nConnection.NETWORK_USER_TEXT = NETWORK_USER_TEXT;\n","/**\n * @file Helper methods for dealing with IndexedDB cache of messages, users, and topics.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nconst DB_VERSION = 1;\nconst DB_NAME = 'tinode-web';\n\nlet IDBProvider;\n\nexport default class DB {\n #onError = _ => {};\n #logger = _ => {};\n\n // Instance of IndexDB.\n db = null;\n // Indicator that the cache is disabled.\n disabled = true;\n\n constructor(onError, logger) {\n this.#onError = onError || this.#onError;\n this.#logger = logger || this.#logger;\n }\n\n #mapObjects(source, callback, context) {\n if (!this.db) {\n return disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction([source]);\n trx.onerror = event => {\n this.#logger('PCache', 'mapObjects', source, event.target.error);\n reject(event.target.error);\n };\n trx.objectStore(source).getAll().onsuccess = event => {\n if (callback) {\n event.target.result.forEach(topic => {\n callback.call(context, topic);\n });\n }\n resolve(event.target.result);\n };\n });\n }\n\n /**\n * Initialize persistent cache: open or create/upgrade if needed.\n * @returns {Promise} promise to be resolved/rejected when the DB is initialized.\n */\n initDatabase() {\n return new Promise((resolve, reject) => {\n // Open the database and initialize callbacks.\n const req = IDBProvider.open(DB_NAME, DB_VERSION);\n req.onsuccess = event => {\n this.db = event.target.result;\n this.disabled = false;\n resolve(this.db);\n };\n req.onerror = event => {\n this.#logger('PCache', \"failed to initialize\", event);\n reject(event.target.error);\n this.#onError(event.target.error);\n };\n req.onupgradeneeded = event => {\n this.db = event.target.result;\n\n this.db.onerror = event => {\n this.#logger('PCache', \"failed to create storage\", event);\n this.#onError(event.target.error);\n };\n\n // Individual object stores.\n // Object store (table) for topics. The primary key is topic name.\n this.db.createObjectStore('topic', {\n keyPath: 'name'\n });\n\n // Users object store. UID is the primary key.\n this.db.createObjectStore('user', {\n keyPath: 'uid'\n });\n\n // Subscriptions object store topic <-> user. Topic name + UID is the primary key.\n this.db.createObjectStore('subscription', {\n keyPath: ['topic', 'uid']\n });\n\n // Messages object store. The primary key is topic name + seq.\n this.db.createObjectStore('message', {\n keyPath: ['topic', 'seq']\n });\n };\n });\n }\n\n /**\n * Delete persistent cache.\n */\n deleteDatabase() {\n // Close connection, otherwise operations will fail with 'onblocked'.\n if (this.db) {\n this.db.close();\n this.db = null;\n }\n return new Promise((resolve, reject) => {\n const req = IDBProvider.deleteDatabase(DB_NAME);\n req.onblocked = _ => {\n if (this.db) {\n this.db.close();\n }\n const err = new Error(\"blocked\");\n this.#logger('PCache', 'deleteDatabase', err);\n reject(err);\n };\n req.onsuccess = _ => {\n this.db = null;\n this.disabled = true;\n resolve(true);\n };\n req.onerror = event => {\n this.#logger('PCache', 'deleteDatabase', event.target.error);\n reject(event.target.error);\n };\n });\n }\n\n /**\n * Check if persistent cache is ready for use.\n * @memberOf DB\n * @returns {boolean} true
if cache is ready, false
otherwise.\n */\n isReady() {\n return !!this.db;\n }\n\n // Topics.\n\n /**\n * Save to cache or update topic in persistent cache.\n * @memberOf DB\n * @param {Topic} topic - topic to be added or updated.\n * @returns {Promise} promise resolved/rejected on operation completion.\n */\n updTopic(topic) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updTopic', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('topic').get(topic.name);\n req.onsuccess = _ => {\n trx.objectStore('topic').put(DB.#serializeTopic(req.result, topic));\n trx.commit();\n };\n });\n }\n\n /**\n * Mark or unmark topic as deleted.\n * @memberOf DB\n * @param {string} name - name of the topic to mark or unmark.\n * @param {boolean} deleted - status\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n markTopicAsDeleted(name, deleted) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'markTopicAsDeleted', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('topic').get(name);\n req.onsuccess = event => {\n const topic = event.target.result;\n if (topic && topic._deleted != deleted) {\n topic._deleted = deleted;\n trx.objectStore('topic').put(topic);\n }\n trx.commit();\n };\n });\n }\n\n /**\n * Remove topic from persistent cache.\n * @memberOf DB\n * @param {string} name - name of the topic to remove from database.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remTopic(name) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic', 'subscription', 'message'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'remTopic', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('topic').delete(IDBKeyRange.only(name));\n trx.objectStore('subscription').delete(IDBKeyRange.bound([name, '-'], [name, '~']));\n trx.objectStore('message').delete(IDBKeyRange.bound([name, 0], [name, Number.MAX_SAFE_INTEGER]));\n trx.commit();\n });\n }\n\n /**\n * Execute a callback for each stored topic.\n * @memberOf DB\n * @param {function} callback - function to call for each topic.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapTopics(callback, context) {\n return this.#mapObjects('topic', callback, context);\n }\n\n /**\n * Copy data from serialized object to topic.\n * @memberOf DB\n * @param {Topic} topic - target to deserialize to.\n * @param {Object} src - serialized data to copy from.\n */\n deserializeTopic(topic, src) {\n DB.#deserializeTopic(topic, src);\n }\n\n // Users.\n /**\n * Add or update user object in the persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to save or update.\n * @param {Object} pub - user's public
information.\n * @returns {Promise} promise resolved/rejected on operation completion.\n */\n updUser(uid, pub) {\n if (arguments.length < 2 || pub === undefined) {\n // No point inupdating user with invalid data.\n return;\n }\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').put({\n uid: uid,\n public: pub\n });\n trx.commit();\n });\n }\n\n /**\n * Remove user from persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to remove from the cache.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remUser(uid) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'remUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').delete(IDBKeyRange.only(uid));\n trx.commit();\n });\n }\n\n /**\n * Execute a callback for each stored user.\n * @memberOf DB\n * @param {function} callback - function to call for each topic.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapUsers(callback, context) {\n return this.#mapObjects('user', callback, context);\n }\n\n /**\n * Read a single user from persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to fetch from cache.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n getUser(uid) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user']);\n trx.oncomplete = event => {\n const user = event.target.result;\n resolve({\n user: user.uid,\n public: user.public\n });\n };\n trx.onerror = event => {\n this.#logger('PCache', 'getUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').get(uid);\n });\n }\n\n // Subscriptions.\n /**\n * Add or update subscription in persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {string} uid - ID of the subscribed user.\n * @param {Object} sub - subscription to save.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n updSubscription(topicName, uid, sub) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['subscription'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updSubscription', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('subscription').get([topicName, uid]).onsuccess = (event) => {\n trx.objectStore('subscription').put(DB.#serializeSubscription(event.target.result, topicName, uid, sub));\n trx.commit();\n };\n });\n }\n\n /**\n * Execute a callback for each cached subscription in a given topic.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the subscriptions.\n * @param {function} callback - function to call for each subscription.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapSubscriptions(topicName, callback, context) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['subscription']);\n trx.onerror = (event) => {\n this.#logger('PCache', 'mapSubscriptions', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('subscription').getAll(IDBKeyRange.bound([topicName, '-'], [topicName, '~'])).onsuccess = (event) => {\n if (callback) {\n event.target.result.forEach((topic) => {\n callback.call(context, topic);\n });\n }\n resolve(event.target.result);\n };\n });\n }\n\n // Messages.\n\n /**\n * Save message to persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {Object} msg - message to save.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n addMessage(msg) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'addMessage', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('message').add(DB.#serializeMessage(null, msg));\n trx.commit();\n });\n }\n\n /**\n * Update delivery status of a message stored in persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {number} seq - ID of the message to update\n * @param {number} status - new delivery status of the message.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n updMessageStatus(topicName, seq, status) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updMessageStatus', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('message').get(IDBKeyRange.only([topicName, seq]));\n req.onsuccess = event => {\n const src = req.result || event.target.result;\n if (!src || src._status == status) {\n trx.commit();\n return;\n }\n trx.objectStore('message').put(DB.#serializeMessage(src, {\n topic: topicName,\n seq: seq,\n _status: status\n }));\n trx.commit();\n };\n });\n }\n\n /**\n * Remove one or more messages from persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {number} from - id of the message to remove or lower boundary when removing range (inclusive).\n * @param {number=} to - upper boundary (exclusive) when removing a range of messages.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remMessages(topicName, from, to) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n if (!from && !to) {\n from = 0;\n to = Number.MAX_SAFE_INTEGER;\n }\n const range = to > 0 ? IDBKeyRange.bound([topicName, from], [topicName, to], false, true) :\n IDBKeyRange.only([topicName, from]);\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = (event) => {\n resolve(event.target.result);\n };\n trx.onerror = (event) => {\n this.#logger('PCache', 'remMessages', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('message').delete(range);\n trx.commit();\n });\n }\n\n /**\n * Retrieve messages from persistent store.\n * @memberOf DB\n * @param {string} topicName - name of the topic to retrieve messages from.\n * @param {function} callback to call for each retrieved message.\n * @param {Object} query - parameters of the message range to retrieve.\n * @param {number=} query.since - the least message ID to retrieve (inclusive).\n * @param {number=} query.before - the greatest message ID to retrieve (exclusive).\n * @param {number=} query.limit - the maximum number of messages to retrieve.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n readMessages(topicName, query, callback, context) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n query = query || {};\n const since = query.since > 0 ? query.since : 0;\n const before = query.before > 0 ? query.before : Number.MAX_SAFE_INTEGER;\n const limit = query.limit | 0;\n\n const result = [];\n const range = IDBKeyRange.bound([topicName, since], [topicName, before], false, true);\n const trx = this.db.transaction(['message']);\n trx.onerror = (event) => {\n this.#logger('PCache', 'readMessages', event.target.error);\n reject(event.target.error);\n };\n // Iterate in descending order.\n trx.objectStore('message').openCursor(range, 'prev').onsuccess = (event) => {\n const cursor = event.target.result;\n if (cursor) {\n if (callback) {\n callback.call(context, cursor.value);\n }\n result.push(cursor.value);\n if (limit <= 0 || result.length < limit) {\n cursor.continue();\n } else {\n resolve(result);\n }\n } else {\n resolve(result);\n }\n };\n });\n }\n\n // Private methods.\n\n // Serializable topic fields.\n static #topic_fields = ['created', 'updated', 'deleted', 'read', 'recv', 'seq', 'clear', 'defacs',\n 'creds', 'public', 'trusted', 'private', 'touched', '_deleted'\n ];\n\n // Copy data from src to Topic object.\n static #deserializeTopic(topic, src) {\n DB.#topic_fields.forEach((f) => {\n if (src.hasOwnProperty(f)) {\n topic[f] = src[f];\n }\n });\n if (Array.isArray(src.tags)) {\n topic._tags = src.tags;\n }\n if (src.acs) {\n topic.setAccessMode(src.acs);\n }\n topic.seq |= 0;\n topic.read |= 0;\n topic.unread = Math.max(0, topic.seq - topic.read);\n }\n\n // Copy values from 'src' to 'dst'. Allocate dst if it's null or undefined.\n static #serializeTopic(dst, src) {\n const res = dst || {\n name: src.name\n };\n DB.#topic_fields.forEach((f) => {\n if (src.hasOwnProperty(f)) {\n res[f] = src[f];\n }\n });\n if (Array.isArray(src._tags)) {\n res.tags = src._tags;\n }\n if (src.acs) {\n res.acs = src.getAccessMode().jsonHelper();\n }\n return res;\n }\n\n static #serializeSubscription(dst, topicName, uid, sub) {\n const fields = ['updated', 'mode', 'read', 'recv', 'clear', 'lastSeen', 'userAgent'];\n const res = dst || {\n topic: topicName,\n uid: uid\n };\n\n fields.forEach((f) => {\n if (sub.hasOwnProperty(f)) {\n res[f] = sub[f];\n }\n });\n\n return res;\n }\n\n static #serializeMessage(dst, msg) {\n // Serializable fields.\n const fields = ['topic', 'seq', 'ts', '_status', 'from', 'head', 'content'];\n const res = dst || {};\n fields.forEach((f) => {\n if (msg.hasOwnProperty(f)) {\n res[f] = msg[f];\n }\n });\n return res;\n }\n\n /**\n * To use DB in a non browser context, supply indexedDB provider.\n * @static\n * @memberof DB\n * @param idbProvider indexedDB provider, e.g. for node require('fake-indexeddb')
.\n */\n static setDatabaseProvider(idbProvider) {\n IDBProvider = idbProvider;\n }\n}\n","/**\n * @copyright 2015-2022 Tinode LLC.\n * @summary Minimally rich text representation and formatting for Tinode.\n * @license Apache 2.0\n *\n * @file Basic parser and formatter for very simple text markup. Mostly targeted at\n * mobile use cases similar to Telegram, WhatsApp, and FB Messenger.\n *\n * Supports conversion of user keyboard input to formatted text:
\n * \n * - *abc* → abc
\n * - _abc_ → abc
\n * - ~abc~ →
abc \n * - `abc` → abc
\n *
\n * Also supports forms and buttons.\n *\n * Nested formatting is supported, e.g. *abc _def_* -> abc def\n * URLs, @mentions, and #hashtags are extracted and converted into links.\n * Forms and buttons can be added procedurally.\n * JSON data representation is inspired by Draft.js raw formatting.\n *\n *\n * @example\n * Text:\n * \n * this is *bold*, `code` and _italic_, ~strike~\n * combined *bold and _italic_*\n * an url: https://www.example.com/abc#fragment and another _www.tinode.co_\n * this is a @mention and a #hashtag in a string\n * second #hashtag\n *
\n *\n * Sample JSON representation of the text above:\n * {\n * \"txt\": \"this is bold, code and italic, strike combined bold and italic an url: https://www.example.com/abc#fragment \" +\n * \"and another www.tinode.co this is a @mention and a #hashtag in a string second #hashtag\",\n * \"fmt\": [\n * { \"at\":8, \"len\":4,\"tp\":\"ST\" },{ \"at\":14, \"len\":4, \"tp\":\"CO\" },{ \"at\":23, \"len\":6, \"tp\":\"EM\"},\n * { \"at\":31, \"len\":6, \"tp\":\"DL\" },{ \"tp\":\"BR\", \"len\":1, \"at\":37 },{ \"at\":56, \"len\":6, \"tp\":\"EM\" },\n * { \"at\":47, \"len\":15, \"tp\":\"ST\" },{ \"tp\":\"BR\", \"len\":1, \"at\":62 },{ \"at\":120, \"len\":13, \"tp\":\"EM\" },\n * { \"at\":71, \"len\":36, \"key\":0 },{ \"at\":120, \"len\":13, \"key\":1 },{ \"tp\":\"BR\", \"len\":1, \"at\":133 },\n * { \"at\":144, \"len\":8, \"key\":2 },{ \"at\":159, \"len\":8, \"key\":3 },{ \"tp\":\"BR\", \"len\":1, \"at\":179 },\n * { \"at\":187, \"len\":8, \"key\":3 },{ \"tp\":\"BR\", \"len\":1, \"at\":195 }\n * ],\n * \"ent\": [\n * { \"tp\":\"LN\", \"data\":{ \"url\":\"https://www.example.com/abc#fragment\" } },\n * { \"tp\":\"LN\", \"data\":{ \"url\":\"http://www.tinode.co\" } },\n * { \"tp\":\"MN\", \"data\":{ \"val\":\"mention\" } },\n * { \"tp\":\"HT\", \"data\":{ \"val\":\"hashtag\" } }\n * ]\n * }\n */\n\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nconst MAX_FORM_ELEMENTS = 8;\nconst MAX_PREVIEW_ATTACHMENTS = 3;\nconst MAX_PREVIEW_DATA_SIZE = 64;\nconst JSON_MIME_TYPE = 'application/json';\nconst DRAFTY_MIME_TYPE = 'text/x-drafty';\nconst ALLOWED_ENT_FIELDS = ['act', 'height', 'duration', 'incoming', 'mime', 'name', 'premime', 'preref', 'preview',\n 'ref', 'size', 'state', 'url', 'val', 'width'\n];\n\n// Regular expressions for parsing inline formats. Javascript does not support lookbehind,\n// so it's a bit messy.\nconst INLINE_STYLES = [\n // Strong = bold, *bold text*\n {\n name: 'ST',\n start: /(?:^|[\\W_])(\\*)[^\\s*]/,\n end: /[^\\s*](\\*)(?=$|[\\W_])/\n },\n // Emphesized = italic, _italic text_\n {\n name: 'EM',\n start: /(?:^|\\W)(_)[^\\s_]/,\n end: /[^\\s_](_)(?=$|\\W)/\n },\n // Deleted, ~strike this though~\n {\n name: 'DL',\n start: /(?:^|[\\W_])(~)[^\\s~]/,\n end: /[^\\s~](~)(?=$|[\\W_])/\n },\n // Code block `this is monospace`\n {\n name: 'CO',\n start: /(?:^|\\W)(`)[^`]/,\n end: /[^`](`)(?=$|\\W)/\n }\n];\n\n// Relative weights of formatting spans. Greater index in array means greater weight.\nconst FMT_WEIGHT = ['QQ'];\n\n// RegExps for entity extraction (RF = reference)\nconst ENTITY_TYPES = [\n // URLs\n {\n name: 'LN',\n dataName: 'url',\n pack: function(val) {\n // Check if the protocol is specified, if not use http\n if (!/^[a-z]+:\\/\\//i.test(val)) {\n val = 'http://' + val;\n }\n return {\n url: val\n };\n },\n re: /(?:(?:https?|ftp):\\/\\/|www\\.|ftp\\.)[-A-Z0-9+&@#\\/%=~_|$?!:,.]*[A-Z0-9+&@#\\/%=~_|$]/ig\n },\n // Mentions @user (must be 2 or more characters)\n {\n name: 'MN',\n dataName: 'val',\n pack: function(val) {\n return {\n val: val.slice(1)\n };\n },\n re: /\\B@([\\p{L}\\p{N}][._\\p{L}\\p{N}]*[\\p{L}\\p{N}])/ug\n },\n // Hashtags #hashtag, like metion 2 or more characters.\n {\n name: 'HT',\n dataName: 'val',\n pack: function(val) {\n return {\n val: val.slice(1)\n };\n },\n re: /\\B#([\\p{L}\\p{N}][._\\p{L}\\p{N}]*[\\p{L}\\p{N}])/ug\n }\n];\n\n// HTML tag name suggestions\nconst FORMAT_TAGS = {\n AU: {\n html_tag: 'audio',\n md_tag: undefined,\n isVoid: false\n },\n BN: {\n html_tag: 'button',\n md_tag: undefined,\n isVoid: false\n },\n BR: {\n html_tag: 'br',\n md_tag: '\\n',\n isVoid: true\n },\n CO: {\n html_tag: 'tt',\n md_tag: '`',\n isVoid: false\n },\n DL: {\n html_tag: 'del',\n md_tag: '~',\n isVoid: false\n },\n EM: {\n html_tag: 'i',\n md_tag: '_',\n isVoid: false\n },\n EX: {\n html_tag: '',\n md_tag: undefined,\n isVoid: true\n },\n FM: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n HD: {\n html_tag: '',\n md_tag: undefined,\n isVoid: false\n },\n HL: {\n html_tag: 'span',\n md_tag: undefined,\n isVoid: false\n },\n HT: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n IM: {\n html_tag: 'img',\n md_tag: undefined,\n isVoid: false\n },\n LN: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n MN: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n RW: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false,\n },\n QQ: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n ST: {\n html_tag: 'b',\n md_tag: '*',\n isVoid: false\n },\n VC: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n VD: {\n html_tag: 'video',\n md_tag: undefined,\n isVoid: false\n }\n};\n\n// Convert base64-encoded string into Blob.\nfunction base64toObjectUrl(b64, contentType, logger) {\n if (!b64) {\n return null;\n }\n\n try {\n const bin = atob(b64);\n const length = bin.length;\n const buf = new ArrayBuffer(length);\n const arr = new Uint8Array(buf);\n for (let i = 0; i < length; i++) {\n arr[i] = bin.charCodeAt(i);\n }\n\n return URL.createObjectURL(new Blob([buf], {\n type: contentType\n }));\n } catch (err) {\n if (logger) {\n logger(\"Drafty: failed to convert object.\", err.message);\n }\n }\n\n return null;\n}\n\nfunction base64toDataUrl(b64, contentType) {\n if (!b64) {\n return null;\n }\n contentType = contentType || 'image/jpeg';\n return 'data:' + contentType + ';base64,' + b64;\n}\n\n// Helpers for converting Drafty to HTML.\nconst DECORATORS = {\n // Visial styles\n ST: {\n open: _ => '',\n close: _ => ''\n },\n EM: {\n open: _ => '',\n close: _ => ''\n },\n DL: {\n open: _ => '',\n close: _ => ''\n },\n CO: {\n open: _ => '',\n close: _ => ''\n },\n // Line break\n BR: {\n open: _ => '
',\n close: _ => ''\n },\n // Hidden element\n HD: {\n open: _ => '',\n close: _ => ''\n },\n // Highlighted element.\n HL: {\n open: _ => '',\n close: _ => ''\n },\n // Link (URL)\n LN: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n href: data.url,\n target: '_blank'\n } : null;\n },\n },\n // Mention\n MN: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n id: data.val\n } : null;\n },\n },\n // Hashtag\n HT: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n id: data.val\n } : null;\n },\n },\n // Button\n BN: {\n open: _ => '',\n props: (data) => {\n return data ? {\n 'data-act': data.act,\n 'data-val': data.val,\n 'data-name': data.name,\n 'data-ref': data.ref\n } : null;\n },\n },\n // Audio recording\n AU: {\n open: (data) => {\n const url = data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger);\n return '',\n props: (data) => {\n if (!data) return null;\n return {\n // Embedded data or external link.\n src: data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n 'data-preload': data.ref ? 'metadata' : 'auto',\n 'data-duration': data.duration,\n 'data-name': data.name,\n 'data-size': data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0),\n 'data-mime': data.mime,\n };\n }\n },\n // Image\n IM: {\n open: data => {\n // Don't use data.ref for preview: it's a security risk.\n const tmpPreviewUrl = base64toDataUrl(data._tempPreview, data.mime);\n const previewUrl = base64toObjectUrl(data.val, data.mime, Drafty.logger);\n const downloadUrl = data.ref || previewUrl;\n return (data.name ? '' : '') +\n '';\n },\n close: data => {\n return (data.name ? '' : '');\n },\n props: data => {\n if (!data) return null;\n return {\n // Temporary preview, or permanent preview, or external link.\n src: base64toDataUrl(data._tempPreview, data.mime) ||\n data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n title: data.name,\n alt: data.name,\n 'data-width': data.width,\n 'data-height': data.height,\n 'data-name': data.name,\n 'data-size': data.ref ? (data.size | 0) : (data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0)),\n 'data-mime': data.mime,\n };\n },\n },\n // Form - structured layout of elements.\n FM: {\n open: _ => '',\n close: _ => ''\n },\n // Row: logic grouping of elements\n RW: {\n open: _ => '',\n close: _ => ''\n },\n // Quoted block.\n QQ: {\n open: _ => '',\n close: _ => '',\n props: (data) => {\n return data ? {} : null;\n },\n },\n // Video call\n VC: {\n open: _ => '',\n close: _ => '',\n props: data => {\n if (!data) return {};\n return {\n 'data-duration': data.duration,\n 'data-state': data.state,\n };\n }\n },\n // Video.\n VD: {\n open: data => {\n const tmpPreviewUrl = base64toDataUrl(data._tempPreview, data.mime);\n const previewUrl = data.ref || base64toObjectUrl(data.preview, data.premime || 'image/jpeg', Drafty.logger);\n return '';\n },\n close: _ => '',\n props: data => {\n if (!data) return null;\n const poster = data.preref || base64toObjectUrl(data.preview, data.premime || 'image/jpeg', Drafty.logger);\n return {\n // Embedded data or external link.\n src: poster,\n 'data-src': data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n 'data-width': data.width,\n 'data-height': data.height,\n 'data-preload': data.ref ? 'metadata' : 'auto',\n 'data-preview': poster,\n 'data-duration': data.duration | 0,\n 'data-name': data.name,\n 'data-size': data.ref ? (data.size | 0) : (data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0)),\n 'data-mime': data.mime,\n };\n }\n },\n};\n\n/**\n * The main object which performs all the formatting actions.\n * @class Drafty\n * @constructor\n */\nconst Drafty = function() {\n this.txt = '';\n this.fmt = [];\n this.ent = [];\n}\n\n/**\n * Initialize Drafty document to a plain text string.\n *\n * @param {string} plainText - string to use as Drafty content.\n *\n * @returns new Drafty document or null is plainText is not a string or undefined.\n */\nDrafty.init = function(plainText) {\n if (typeof plainText == 'undefined') {\n plainText = '';\n } else if (typeof plainText != 'string') {\n return null;\n }\n\n return {\n txt: plainText\n };\n}\n\n/**\n * Parse plain text into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {string} content - plain-text content to parse.\n * @return {Drafty} parsed document or null if the source is not plain text.\n */\nDrafty.parse = function(content) {\n // Make sure we are parsing strings only.\n if (typeof content != 'string') {\n return null;\n }\n\n // Split text into lines. It makes further processing easier.\n const lines = content.split(/\\r?\\n/);\n\n // Holds entities referenced from text\n const entityMap = [];\n const entityIndex = {};\n\n // Processing lines one by one, hold intermediate result in blx.\n const blx = [];\n lines.forEach((line) => {\n let spans = [];\n let entities;\n\n // Find formatted spans in the string.\n // Try to match each style.\n INLINE_STYLES.forEach((tag) => {\n // Each style could be matched multiple times.\n spans = spans.concat(spannify(line, tag.start, tag.end, tag.name));\n });\n\n let block;\n if (spans.length == 0) {\n block = {\n txt: line\n };\n } else {\n // Sort spans by style occurence early -> late, then by length: first long then short.\n spans.sort((a, b) => {\n const diff = a.at - b.at;\n return diff != 0 ? diff : b.end - a.end;\n });\n\n // Convert an array of possibly overlapping spans into a tree.\n spans = toSpanTree(spans);\n\n // Build a tree representation of the entire string, not\n // just the formatted parts.\n const chunks = chunkify(line, 0, line.length, spans);\n\n const drafty = draftify(chunks, 0);\n\n block = {\n txt: drafty.txt,\n fmt: drafty.fmt\n };\n }\n\n // Extract entities from the cleaned up string.\n entities = extractEntities(block.txt);\n if (entities.length > 0) {\n const ranges = [];\n for (let i in entities) {\n // {offset: match['index'], unique: match[0], len: match[0].length, data: ent.packer(), type: ent.name}\n const entity = entities[i];\n let index = entityIndex[entity.unique];\n if (!index) {\n index = entityMap.length;\n entityIndex[entity.unique] = index;\n entityMap.push({\n tp: entity.type,\n data: entity.data\n });\n }\n ranges.push({\n at: entity.offset,\n len: entity.len,\n key: index\n });\n }\n block.ent = ranges;\n }\n\n blx.push(block);\n });\n\n const result = {\n txt: ''\n };\n\n // Merge lines and save line breaks as BR inline formatting.\n if (blx.length > 0) {\n result.txt = blx[0].txt;\n result.fmt = (blx[0].fmt || []).concat(blx[0].ent || []);\n\n for (let i = 1; i < blx.length; i++) {\n const block = blx[i];\n const offset = result.txt.length + 1;\n\n result.fmt.push({\n tp: 'BR',\n len: 1,\n at: offset - 1\n });\n\n result.txt += ' ' + block.txt;\n if (block.fmt) {\n result.fmt = result.fmt.concat(block.fmt.map((s) => {\n s.at += offset;\n return s;\n }));\n }\n if (block.ent) {\n result.fmt = result.fmt.concat(block.ent.map((s) => {\n s.at += offset;\n return s;\n }));\n }\n }\n\n if (result.fmt.length == 0) {\n delete result.fmt;\n }\n\n if (entityMap.length > 0) {\n result.ent = entityMap;\n }\n }\n return result;\n}\n\n/**\n * Append one Drafty document to another.\n *\n * @param {Drafty} first - Drafty document to append to.\n * @param {Drafty|string} second - Drafty document or string being appended.\n *\n * @return {Drafty} first document with the second appended to it.\n */\nDrafty.append = function(first, second) {\n if (!first) {\n return second;\n }\n if (!second) {\n return first;\n }\n\n first.txt = first.txt || '';\n const len = first.txt.length;\n\n if (typeof second == 'string') {\n first.txt += second;\n } else if (second.txt) {\n first.txt += second.txt;\n }\n\n if (Array.isArray(second.fmt)) {\n first.fmt = first.fmt || [];\n if (Array.isArray(second.ent)) {\n first.ent = first.ent || [];\n }\n second.fmt.forEach(src => {\n const fmt = {\n at: (src.at | 0) + len,\n len: src.len | 0\n };\n // Special case for the outside of the normal rendering flow styles.\n if (src.at == -1) {\n fmt.at = -1;\n fmt.len = 0;\n }\n if (src.tp) {\n fmt.tp = src.tp;\n } else {\n fmt.key = first.ent.length;\n first.ent.push(second.ent[src.key || 0]);\n }\n first.fmt.push(fmt);\n });\n }\n\n return first;\n}\n\n/**\n * Description of an image to attach.\n * @typedef {Object} ImageDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the image, e.g. \"image/png\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - base64-encoded image content. Could be null/undefined.\n * @property {string} preview - base64-encoded thumbnail of the image.\n * @property {integer} width - width of the image.\n * @property {integer} height - height of the image.\n * @property {string} filename - file name suggestion for downloading the image.\n * @property {integer} size - size of the image in bytes. Treat is as an untrusted hint.\n * @property {string} _tempPreview - base64-encoded image preview used during upload process; not serializable.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Insert inline image into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add image to.\n * @param {integer} at - index where the object is inserted. The length of the image is always 1.\n * @param {ImageDesc} imageDesc - object with image paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertImage = function(content, at, imageDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'IM',\n data: {\n mime: imageDesc.mime,\n ref: imageDesc.refurl,\n val: imageDesc.bits || imageDesc.preview,\n width: imageDesc.width,\n height: imageDesc.height,\n name: imageDesc.filename,\n size: imageDesc.size | 0,\n }\n };\n\n if (imageDesc.urlPromise) {\n ex.data._tempPreview = imageDesc._tempPreview;\n ex.data._processing = true;\n imageDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._tempPreview = undefined;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Description of a video to attach.\n * @typedef {Object} VideoDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the video, e.g. \"video/mpeg\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - in-band base64-encoded image data. Could be null/undefined.\n * @property {string} preview - base64-encoded screencapture from the video. Could be null/undefined.\n * @property {string} preref - reference to screencapture from the video. Could be null/undefined.\n * @property {integer} width - width of the video.\n * @property {integer} height - height of the video.\n * @property {integer} duration - duration of the video.\n * @property {string} filename - file name suggestion for downloading the video.\n * @property {integer} size - size of the video in bytes. Treat is as an untrusted hint.\n * @property {string} _tempPreview - base64-encoded screencapture used during upload process; not serializable.\n * @property {Promise} urlPromise - array of two promises, which return URLs of video and preview uploads correspondingly\n * (either could be null).\n */\n\n/**\n * Insert inline image into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add video to.\n * @param {integer} at - index where the object is inserted. The length of the video is always 1.\n * @param {VideoDesc} videoDesc - object with video paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertVideo = function(content, at, videoDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'VD',\n data: {\n mime: videoDesc.mime,\n ref: videoDesc.refurl,\n val: videoDesc.bits,\n preref: videoDesc.preref,\n preview: videoDesc.preview,\n width: videoDesc.width,\n height: videoDesc.height,\n duration: videoDesc.duration | 0,\n name: videoDesc.filename,\n size: videoDesc.size | 0,\n }\n };\n\n if (videoDesc.urlPromise) {\n ex.data._tempPreview = videoDesc._tempPreview;\n ex.data._processing = true;\n videoDesc.urlPromise.then(\n urls => {\n ex.data.ref = urls[0];\n ex.data.preref = urls[1];\n ex.data._tempPreview = undefined;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Description of an audio recording to attach.\n * @typedef {Object} AudioDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the audio, e.g. \"audio/ogg\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - base64-encoded audio content. Could be null/undefined.\n * @property {integer} duration - duration of the record in milliseconds.\n * @property {string} preview - base64 encoded short array of amplitude values 0..100.\n * @property {string} filename - file name suggestion for downloading the audio.\n * @property {integer} size - size of the recording in bytes. Treat is as an untrusted hint.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Insert audio recording into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add audio record to.\n * @param {integer} at - index where the object is inserted. The length of the record is always 1.\n * @param {AudioDesc} audioDesc - object with the audio paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertAudio = function(content, at, audioDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'AU',\n data: {\n mime: audioDesc.mime,\n val: audioDesc.bits,\n duration: audioDesc.duration | 0,\n preview: audioDesc.preview,\n name: audioDesc.filename,\n size: audioDesc.size | 0,\n ref: audioDesc.refurl\n }\n };\n\n if (audioDesc.urlPromise) {\n ex.data._processing = true;\n audioDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Create a (self-contained) video call Drafty document.\n * @memberof Drafty\n * @static\n * @param {boolean} audioOnly true
if the call is initially audio-only.\n * @returns Video Call drafty document.\n */\nDrafty.videoCall = function(audioOnly) {\n const content = {\n txt: ' ',\n fmt: [{\n at: 0,\n len: 1,\n key: 0\n }],\n ent: [{\n tp: 'VC',\n data: {\n aonly: audioOnly\n },\n }]\n };\n return content;\n}\n\n/**\n * Update video call (VC) entity with the new status and duration.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - VC document to update.\n * @param {object} params - new video call parameters.\n * @param {string} params.state - state of video call.\n * @param {number} params.duration - duration of the video call in milliseconds.\n *\n * @returns the same document with update applied.\n */\nDrafty.updateVideoCall = function(content, params) {\n // The video element could be just a format or a format + entity.\n // Must ensure it's the latter first.\n const fmt = ((content || {}).fmt || [])[0];\n if (!fmt) {\n // Unrecognized content.\n return content;\n }\n\n let ent;\n if (fmt.tp == 'VC') {\n // Just a format, convert to format + entity.\n delete fmt.tp;\n fmt.key = 0;\n ent = {\n tp: 'VC'\n };\n content.ent = [ent];\n } else {\n ent = (content.ent || [])[fmt.key | 0];\n if (!ent || ent.tp != 'VC') {\n // Not a VC entity.\n return content;\n }\n }\n ent.data = ent.data || {};\n Object.assign(ent.data, params);\n return content;\n}\n\n/**\n * Create a quote to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {string} header - Quote header (title, etc.).\n * @param {string} uid - UID of the author to mention.\n * @param {Drafty} body - Body of the quoted message.\n *\n * @returns Reply quote Drafty doc with the quote formatting.\n */\nDrafty.quote = function(header, uid, body) {\n const quote = Drafty.append(Drafty.appendLineBreak(Drafty.mention(header, uid)), body);\n\n // Wrap into a quote.\n quote.fmt.push({\n at: 0,\n len: quote.txt.length,\n tp: 'QQ'\n });\n\n return quote;\n}\n\n/**\n * Create a Drafty document with a mention.\n *\n * @param {string} name - mentioned name.\n * @param {string} uid - mentioned user ID.\n *\n * @returns {Drafty} document with the mention.\n */\nDrafty.mention = function(name, uid) {\n return {\n txt: name || '',\n fmt: [{\n at: 0,\n len: (name || '').length,\n key: 0\n }],\n ent: [{\n tp: 'MN',\n data: {\n val: uid\n }\n }]\n };\n}\n\n/**\n * Append a link to a Drafty document.\n *\n * @param {Drafty} content - Drafty document to append link to.\n * @param {Object} linkData - Link info in format {txt: 'ankor text', url: 'http://...'}
.\n *\n * @returns {Drafty} the same document as content
.\n */\nDrafty.appendLink = function(content, linkData) {\n content = content || {\n txt: ''\n };\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: content.txt.length,\n len: linkData.txt.length,\n key: content.ent.length\n });\n content.txt += linkData.txt;\n\n const ex = {\n tp: 'LN',\n data: {\n url: linkData.url\n }\n }\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Append image to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add image to.\n * @param {ImageDesc} imageDesc - object with image paramenets.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendImage = function(content, imageDesc) {\n content = content || {\n txt: ''\n };\n content.txt += ' ';\n return Drafty.insertImage(content, content.txt.length - 1, imageDesc);\n}\n\n/**\n * Append audio recodring to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add recording to.\n * @param {AudioDesc} audioDesc - object with audio data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendAudio = function(content, audioDesc) {\n content = content || {\n txt: ''\n };\n content.txt += ' ';\n return Drafty.insertAudio(content, content.txt.length - 1, audioDesc);\n}\n\n/**\n * Description of a file to attach.\n * @typedef {Object} AttachmentDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the attachment, e.g. \"application/octet-stream\"\n * @property {string} data - base64-encoded in-band content of small attachments. Could be null/undefined.\n * @property {string} filename - file name suggestion for downloading the attachment.\n * @property {integer} size - size of the file in bytes. Treat is as an untrusted hint.\n * @property {string} refurl - reference to the out-of-band content. Could be null/undefined.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Attach file to Drafty content. Either as a blob or as a reference.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to attach file to.\n * @param {AttachmentDesc} object - containing attachment description and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.attachFile = function(content, attachmentDesc) {\n content = content || {\n txt: ''\n };\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: -1,\n len: 0,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'EX',\n data: {\n mime: attachmentDesc.mime,\n val: attachmentDesc.data,\n name: attachmentDesc.filename,\n ref: attachmentDesc.refurl,\n size: attachmentDesc.size | 0\n }\n }\n if (attachmentDesc.urlPromise) {\n ex.data._processing = true;\n attachmentDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._processing = undefined;\n },\n _ => {\n /* catch the error, otherwise it will appear in the console. */\n ex.data._processing = undefined;\n }\n );\n }\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Wraps drafty document into a simple formatting style.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - document or string to wrap into a style.\n * @param {string} style - two-letter style to wrap into.\n * @param {number} at - index where the style starts, default 0.\n * @param {number} len - length of the form content, default all of it.\n *\n * @return {Drafty} updated document.\n */\nDrafty.wrapInto = function(content, style, at, len) {\n if (typeof content == 'string') {\n content = {\n txt: content\n };\n }\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at || 0,\n len: len || content.txt.length,\n tp: style,\n });\n\n return content;\n}\n\n/**\n * Wraps content into an interactive form.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - to wrap into a form.\n * @param {number} at - index where the forms starts.\n * @param {number} len - length of the form content.\n *\n * @return {Drafty} updated document.\n */\nDrafty.wrapAsForm = function(content, at, len) {\n return Drafty.wrapInto(content, 'FM', at, len);\n}\n\n/**\n * Insert clickable button into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - Drafty document to insert button to or a string to be used as button text.\n * @param {number} at - location where the button is inserted.\n * @param {number} len - the length of the text to be used as button title.\n * @param {string} name - the button. Client should return it to the server when the button is clicked.\n * @param {string} actionType - the type of the button, one of 'url' or 'pub'.\n * @param {string} actionValue - the value to return on click:\n * @param {string} refUrl - the URL to go to when the 'url' button is clicked.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertButton = function(content, at, len, name, actionType, actionValue, refUrl) {\n if (typeof content == 'string') {\n content = {\n txt: content\n };\n }\n\n if (!content || !content.txt || content.txt.length < at + len) {\n return null;\n }\n\n if (len <= 0 || ['url', 'pub'].indexOf(actionType) == -1) {\n return null;\n }\n // Ensure refUrl is a string.\n if (actionType == 'url' && !refUrl) {\n return null;\n }\n refUrl = '' + refUrl;\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: len,\n key: content.ent.length\n });\n content.ent.push({\n tp: 'BN',\n data: {\n act: actionType,\n val: actionValue,\n ref: refUrl,\n name: name\n }\n });\n\n return content;\n}\n\n/**\n * Append clickable button to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - Drafty document to insert button to or a string to be used as button text.\n * @param {string} title - the text to be used as button title.\n * @param {string} name - the button. Client should return it to the server when the button is clicked.\n * @param {string} actionType - the type of the button, one of 'url' or 'pub'.\n * @param {string} actionValue - the value to return on click:\n * @param {string} refUrl - the URL to go to when the 'url' button is clicked.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendButton = function(content, title, name, actionType, actionValue, refUrl) {\n content = content || {\n txt: ''\n };\n const at = content.txt.length;\n content.txt += title;\n return Drafty.insertButton(content, at, title.length, name, actionType, actionValue, refUrl);\n}\n\n/**\n * Attach a generic JS object. The object is attached as a json string.\n * Intended for representing a form response.\n *\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - Drafty document to attach file to.\n * @param {Object} data - data to convert to json string and attach.\n * @returns {Drafty} the same document as content
.\n */\nDrafty.attachJSON = function(content, data) {\n content = content || {\n txt: ''\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: -1,\n len: 0,\n key: content.ent.length\n });\n\n content.ent.push({\n tp: 'EX',\n data: {\n mime: JSON_MIME_TYPE,\n val: data\n }\n });\n\n return content;\n}\n/**\n * Append line break to a Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - Drafty document to append linebreak to.\n * @returns {Drafty} the same document as content
.\n */\nDrafty.appendLineBreak = function(content) {\n content = content || {\n txt: ''\n };\n content.fmt = content.fmt || [];\n content.fmt.push({\n at: content.txt.length,\n len: 1,\n tp: 'BR'\n });\n content.txt += ' ';\n\n return content;\n}\n/**\n * Given Drafty document, convert it to HTML.\n * No attempt is made to strip pre-existing html markup.\n * This is potentially unsafe because content.txt
may contain malicious HTML\n * markup.\n * @memberof Tinode.Drafty\n * @static\n *\n * @param {Drafty} doc - document to convert.\n *\n * @returns {string} HTML-representation of content.\n */\nDrafty.UNSAFE_toHTML = function(doc) {\n const tree = draftyToTree(doc);\n const htmlFormatter = function(type, data, values) {\n const tag = DECORATORS[type];\n let result = values ? values.join('') : '';\n if (tag) {\n result = tag.open(data) + result + tag.close(data);\n }\n return result;\n };\n return treeBottomUp(tree, htmlFormatter, 0);\n}\n\n/**\n * Callback for applying custom formatting to a Drafty document.\n * Called once for each style span.\n * @memberof Drafty\n * @static\n *\n * @callback Formatter\n * @param {string} style - style code such as \"ST\" or \"IM\".\n * @param {Object} data - entity's data.\n * @param {Object} values - possibly styled subspans contained in this style span.\n * @param {number} index - index of the element guaranteed to be unique.\n */\n\n/**\n * Convert Drafty document to a representation suitable for display.\n * The context
may expose a function getFormatter(style)
. If it's available\n * it will call it to obtain a formatter
for a subtree of styles under the style
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|Object} content - Drafty document to transform.\n * @param {Formatter} formatter - callback which formats individual elements.\n * @param {Object} context - context provided to formatter as this
.\n *\n * @return {Object} transformed object\n */\nDrafty.format = function(original, formatter, context) {\n return treeBottomUp(draftyToTree(original), formatter, 0, [], context);\n}\n\n/**\n * Shorten Drafty document making the drafty text no longer than the limit.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characrets to shorten to.\n * @param {boolean} light - remove heavy data from entities.\n * @returns new shortened Drafty object leaving the original intact.\n */\nDrafty.shorten = function(original, limit, light) {\n let tree = draftyToTree(original);\n tree = shortenTree(tree, limit, '…');\n if (tree && light) {\n tree = lightEntity(tree);\n }\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Transform Drafty doc for forwarding: strip leading @mention and any leading line breaks or whitespace.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @returns converted Drafty object leaving the original intact.\n */\nDrafty.forwardedContent = function(original) {\n let tree = draftyToTree(original);\n const rmMention = function(node) {\n if (node.type == 'MN') {\n if (!node.parent || !node.parent.type) {\n return null;\n }\n }\n return node;\n }\n // Strip leading mention.\n tree = treeTopDown(tree, rmMention);\n // Remove leading whitespace.\n tree = lTrim(tree);\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Prepare Drafty doc for wrapping into QQ as a reply:\n * - Replace forwarding mention with symbol '➦' and remove data (UID).\n * - Remove quoted text completely.\n * - Replace line breaks with spaces.\n * - Strip entities of heavy content.\n * - Move attachments to the end of the document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characters to shorten to.\n * @returns converted Drafty object leaving the original intact.\n */\nDrafty.replyContent = function(original, limit) {\n const convMNnQQnBR = function(node) {\n if (node.type == 'QQ') {\n return null;\n } else if (node.type == 'MN') {\n if ((!node.parent || !node.parent.type) && (node.text || '').startsWith('➦')) {\n node.text = '➦';\n delete node.children;\n delete node.data;\n }\n } else if (node.type == 'BR') {\n node.text = ' ';\n delete node.type;\n delete node.children;\n }\n return node;\n }\n\n let tree = draftyToTree(original);\n if (!tree) {\n return original;\n }\n\n // Strip leading mention.\n tree = treeTopDown(tree, convMNnQQnBR);\n // Move attachments to the end of the doc.\n tree = attachmentsToEnd(tree, MAX_PREVIEW_ATTACHMENTS);\n // Shorten the doc.\n tree = shortenTree(tree, limit, '…');\n // Strip heavy elements except IM.data['val'] and VD.data['preview'] (have to keep them to generate previews later).\n const filter = node => {\n switch (node.type) {\n case 'IM':\n return ['val'];\n case 'VD':\n return ['preview'];\n }\n return null;\n };\n tree = lightEntity(tree, filter);\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n\n/**\n * Generate drafty preview:\n * - Shorten the document.\n * - Strip all heavy entity data leaving just inline styles and entity references.\n * - Replace line breaks with spaces.\n * - Replace content of QQ with a space.\n * - Replace forwarding mention with symbol '➦'.\n * move all attachments to the end of the document and make them visible.\n * The context
may expose a function getFormatter(style)
. If it's available\n * it will call it to obtain a formatter
for a subtree of styles under the style
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characters to shorten to.\n * @param {boolean} forwarding - this a forwarding message preview.\n * @returns new shortened Drafty object leaving the original intact.\n */\nDrafty.preview = function(original, limit, forwarding) {\n let tree = draftyToTree(original);\n\n // Move attachments to the end.\n tree = attachmentsToEnd(tree, MAX_PREVIEW_ATTACHMENTS);\n\n // Convert leading mention to '➦' and replace QQ and BR with a space ' '.\n const convMNnQQnBR = function(node) {\n if (node.type == 'MN') {\n if ((!node.parent || !node.parent.type) && (node.text || '').startsWith('➦')) {\n node.text = '➦';\n delete node.children;\n }\n } else if (node.type == 'QQ') {\n node.text = ' ';\n delete node.children;\n } else if (node.type == 'BR') {\n node.text = ' ';\n delete node.children;\n delete node.type;\n }\n return node;\n }\n tree = treeTopDown(tree, convMNnQQnBR);\n\n tree = shortenTree(tree, limit, '…');\n if (forwarding) {\n // Keep some IM and VD data for preview.\n const filter = {\n IM: ['val'],\n VD: ['preview']\n };\n tree = lightEntity(tree, node => {\n return filter[node.type];\n });\n } else {\n tree = lightEntity(tree);\n }\n\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Given Drafty document, convert it to plain text.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to convert to plain text.\n * @returns {string} plain-text representation of the drafty document.\n */\nDrafty.toPlainText = function(content) {\n return typeof content == 'string' ? content : content.txt;\n}\n\n/**\n * Check if the document has no markup and no entities.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - content to check for presence of markup.\n * @returns true
is content is plain text, false
otherwise.\n */\nDrafty.isPlainText = function(content) {\n return typeof content == 'string' || !(content.fmt || content.ent);\n}\n\n/**\n * Convert document to plain text with markdown. All elements which cannot\n * be represented in markdown are stripped.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to convert to plain text with markdown.\n */\nDrafty.toMarkdown = function(content) {\n let tree = draftyToTree(content);\n const mdFormatter = function(type, _, values) {\n const def = FORMAT_TAGS[type];\n let result = (values ? values.join('') : '');\n if (def) {\n if (def.isVoid) {\n result = def.md_tag || '';\n } else if (def.md_tag) {\n result = def.md_tag + result + def.md_tag;\n }\n }\n return result;\n };\n return treeBottomUp(tree, mdFormatter, 0);\n}\n\n/**\n * Checks if the object represets is a valid Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - content to check for validity.\n * @returns true
is content is valid, false
otherwise.\n */\nDrafty.isValid = function(content) {\n if (!content) {\n return false;\n }\n\n const {\n txt,\n fmt,\n ent\n } = content;\n\n if (!txt && txt !== '' && !fmt && !ent) {\n return false;\n }\n\n const txt_type = typeof txt;\n if (txt_type != 'string' && txt_type != 'undefined' && txt !== null) {\n return false;\n }\n\n if (typeof fmt != 'undefined' && !Array.isArray(fmt) && fmt !== null) {\n return false;\n }\n\n if (typeof ent != 'undefined' && !Array.isArray(ent) && ent !== null) {\n return false;\n }\n return true;\n}\n\n/**\n * Check if the drafty document has attachments: style EX and outside of normal rendering flow,\n * i.e. at = -1
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to check for attachments.\n * @returns true
if there are attachments.\n */\nDrafty.hasAttachments = function(content) {\n if (!Array.isArray(content.fmt)) {\n return false;\n }\n for (let i in content.fmt) {\n const fmt = content.fmt[i];\n if (fmt && fmt.at < 0) {\n const ent = content.ent[fmt.key | 0];\n return ent && ent.tp == 'EX' && ent.data;\n }\n }\n return false;\n}\n\n/**\n * Callback for enumerating entities in a Drafty document.\n * Called once for each entity.\n * @memberof Drafty\n * @static\n *\n * @callback EntityCallback\n * @param {Object} data entity data.\n * @param {string} entity type.\n * @param {number} index entity's index in `content.ent`.\n *\n * @return 'true-ish' to stop processing, 'false-ish' otherwise.\n */\n\n/**\n * Enumerate attachments: style EX and outside of normal rendering flow, i.e. at = -1
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to process for attachments.\n * @param {EntityCallback} callback - callback to call for each attachment.\n * @param {Object} context - value of \"this\" for callback.\n */\nDrafty.attachments = function(content, callback, context) {\n if (!Array.isArray(content.fmt)) {\n return;\n }\n let count = 0;\n for (let i in content.fmt) {\n let fmt = content.fmt[i];\n if (fmt && fmt.at < 0) {\n const ent = content.ent[fmt.key | 0];\n if (ent && ent.tp == 'EX' && ent.data) {\n if (callback.call(context, ent.data, count++, 'EX')) {\n break;\n }\n }\n }\n };\n}\n\n/**\n * Check if the drafty document has entities.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to check for entities.\n * @returns true
if there are entities.\n */\nDrafty.hasEntities = function(content) {\n return content.ent && content.ent.length > 0;\n}\n\n/**\n * Enumerate entities. Enumeration stops if callback returns 'true'.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with entities to enumerate.\n * @param {EntityCallback} callback - callback to call for each entity.\n * @param {Object} context - value of \"this\" for callback.\n *\n */\nDrafty.entities = function(content, callback, context) {\n if (content.ent && content.ent.length > 0) {\n for (let i in content.ent) {\n if (content.ent[i]) {\n if (callback.call(context, content.ent[i].data, i, content.ent[i].tp)) {\n break;\n }\n }\n }\n }\n}\n\n/**\n * Callback for enumerating styles (inline formats) in a Drafty document.\n * Called once for each style.\n * @memberof Drafty\n * @static\n *\n * @callback StyleCallback\n * @param {string} tp - format type.\n * @param {number} at - starting position of the format in text.\n * @param {number} len - extent of the format in characters.\n * @param {number} key - index of the entity if format is a reference.\n * @param {number} index - style's index in `content.fmt`.\n *\n * @return 'true-ish' to stop processing, 'false-ish' otherwise.\n */\n\n/**\n * Enumerate styles (inline formats). Enumeration stops if callback returns 'true'.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with styles (formats) to enumerate.\n * @param {StyleCallback} callback - callback to call for each format.\n * @param {Object} context - value of \"this\" for callback.\n */\nDrafty.styles = function(content, callback, context) {\n if (content.fmt && content.fmt.length > 0) {\n for (let i in content.fmt) {\n const fmt = content.fmt[i];\n if (fmt) {\n if (callback.call(context, fmt.tp, fmt.at, fmt.len, fmt.key, i)) {\n break;\n }\n }\n }\n }\n}\n\n/**\n * Remove unrecognized fields from entity data\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with entities to enumerate.\n * @returns content.\n */\nDrafty.sanitizeEntities = function(content) {\n if (content && content.ent && content.ent.length > 0) {\n for (let i in content.ent) {\n const ent = content.ent[i];\n if (ent && ent.data) {\n const data = copyEntData(ent.data);\n if (data) {\n content.ent[i].data = data;\n } else {\n delete content.ent[i].data;\n }\n }\n }\n }\n return content;\n}\n\n/**\n * Given the entity, get URL which can be used for downloading\n * entity data.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the URl from.\n * @returns {string} URL to download entity data or null
.\n */\nDrafty.getDownloadUrl = function(entData) {\n let url = null;\n if (entData.mime != JSON_MIME_TYPE && entData.val) {\n url = base64toObjectUrl(entData.val, entData.mime, Drafty.logger);\n } else if (typeof entData.ref == 'string') {\n url = entData.ref;\n }\n return url;\n}\n\n/**\n * Check if the entity data is not ready for sending, such as being uploaded to the server.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entity.data to get the URl from.\n * @returns {boolean} true if upload is in progress, false otherwise.\n */\nDrafty.isProcessing = function(entData) {\n return !!entData._processing;\n}\n\n/**\n * Given the entity, get URL which can be used for previewing\n * the entity.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entity.data to get the URl from.\n *\n * @returns {string} url for previewing or null if no such url is available.\n */\nDrafty.getPreviewUrl = function(entData) {\n return entData.val ? base64toObjectUrl(entData.val, entData.mime, Drafty.logger) : null;\n}\n\n/**\n * Get approximate size of the entity.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the size for.\n * @returns {number} size of entity data in bytes.\n */\nDrafty.getEntitySize = function(entData) {\n // Either size hint or length of value. The value is base64 encoded,\n // the actual object size is smaller than the encoded length.\n return entData.size ? entData.size : entData.val ? (entData.val.length * 0.75) | 0 : 0;\n}\n\n/**\n * Get entity mime type.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the type for.\n * @returns {string} mime type of entity.\n */\nDrafty.getEntityMimeType = function(entData) {\n return entData.mime || 'text/plain';\n}\n\n/**\n * Get HTML tag for a given two-letter style name.\n * @memberof Drafty\n * @static\n *\n * @param {string} style - two-letter style, like ST or LN.\n *\n * @returns {string} HTML tag name if style is found, {code: undefined} if style is falsish or not found.\n */\nDrafty.tagName = function(style) {\n return FORMAT_TAGS[style] && FORMAT_TAGS[style].html_tag;\n}\n\n/**\n * For a given data bundle generate an object with HTML attributes,\n * for instance, given {url: \"http://www.example.com/\"} return\n * {href: \"http://www.example.com/\"}\n * @memberof Drafty\n * @static\n *\n * @param {string} style - two-letter style to generate attributes for.\n * @param {Object} data - data bundle to convert to attributes\n *\n * @returns {Object} object with HTML attributes.\n */\nDrafty.attrValue = function(style, data) {\n if (data && DECORATORS[style] && DECORATORS[style].props) {\n return DECORATORS[style].props(data);\n }\n\n return undefined;\n}\n\n/**\n * Drafty MIME type.\n * @memberof Drafty\n * @static\n *\n * @returns {string} content-Type \"text/x-drafty\".\n */\nDrafty.getContentType = function() {\n return DRAFTY_MIME_TYPE;\n}\n\n// =================\n// Utility methods.\n// =================\n\n// Take a string and defined earlier style spans, re-compose them into a tree where each leaf is\n// a same-style (including unstyled) string. I.e. 'hello *bold _italic_* and ~more~ world' ->\n// ('hello ', (b: 'bold ', (i: 'italic')), ' and ', (s: 'more'), ' world');\n//\n// This is needed in order to clear markup, i.e. 'hello *world*' -> 'hello world' and convert\n// ranges from markup-ed offsets to plain text offsets.\nfunction chunkify(line, start, end, spans) {\n const chunks = [];\n\n if (spans.length == 0) {\n return [];\n }\n\n for (let i in spans) {\n // Get the next chunk from the queue\n const span = spans[i];\n\n // Grab the initial unstyled chunk\n if (span.at > start) {\n chunks.push({\n txt: line.slice(start, span.at)\n });\n }\n\n // Grab the styled chunk. It may include subchunks.\n const chunk = {\n tp: span.tp\n };\n const chld = chunkify(line, span.at + 1, span.end, span.children);\n if (chld.length > 0) {\n chunk.children = chld;\n } else {\n chunk.txt = span.txt;\n }\n chunks.push(chunk);\n start = span.end + 1; // '+1' is to skip the formatting character\n }\n\n // Grab the remaining unstyled chunk, after the last span\n if (start < end) {\n chunks.push({\n txt: line.slice(start, end)\n });\n }\n\n return chunks;\n}\n\n// Detect starts and ends of formatting spans. Unformatted spans are\n// ignored at this stage.\nfunction spannify(original, re_start, re_end, type) {\n const result = [];\n let index = 0;\n let line = original.slice(0); // make a copy;\n\n while (line.length > 0) {\n // match[0]; // match, like '*abc*'\n // match[1]; // match captured in parenthesis, like 'abc'\n // match['index']; // offset where the match started.\n\n // Find the opening token.\n const start = re_start.exec(line);\n if (start == null) {\n break;\n }\n\n // Because javascript RegExp does not support lookbehind, the actual offset may not point\n // at the markup character. Find it in the matched string.\n let start_offset = start['index'] + start[0].lastIndexOf(start[1]);\n // Clip the processed part of the string.\n line = line.slice(start_offset + 1);\n // start_offset is an offset within the clipped string. Convert to original index.\n start_offset += index;\n // Index now point to the beginning of 'line' within the 'original' string.\n index = start_offset + 1;\n\n // Find the matching closing token.\n const end = re_end ? re_end.exec(line) : null;\n if (end == null) {\n break;\n }\n let end_offset = end['index'] + end[0].indexOf(end[1]);\n // Clip the processed part of the string.\n line = line.slice(end_offset + 1);\n // Update offsets\n end_offset += index;\n // Index now points to the beginning of 'line' within the 'original' string.\n index = end_offset + 1;\n\n result.push({\n txt: original.slice(start_offset + 1, end_offset),\n children: [],\n at: start_offset,\n end: end_offset,\n tp: type\n });\n }\n\n return result;\n}\n\n// Convert linear array or spans into a tree representation.\n// Keep standalone and nested spans, throw away partially overlapping spans.\nfunction toSpanTree(spans) {\n if (spans.length == 0) {\n return [];\n }\n\n const tree = [spans[0]];\n let last = spans[0];\n for (let i = 1; i < spans.length; i++) {\n // Keep spans which start after the end of the previous span or those which\n // are complete within the previous span.\n if (spans[i].at > last.end) {\n // Span is completely outside of the previous span.\n tree.push(spans[i]);\n last = spans[i];\n } else if (spans[i].end <= last.end) {\n // Span is fully inside of the previous span. Push to subnode.\n last.children.push(spans[i]);\n }\n // Span could partially overlap, ignoring it as invalid.\n }\n\n // Recursively rearrange the subnodes.\n for (let i in tree) {\n tree[i].children = toSpanTree(tree[i].children);\n }\n\n return tree;\n}\n\n// Convert drafty document to a tree.\nfunction draftyToTree(doc) {\n if (!doc) {\n return null;\n }\n\n doc = (typeof doc == 'string') ? {\n txt: doc\n } : doc;\n let {\n txt,\n fmt,\n ent\n } = doc;\n\n txt = txt || '';\n if (!Array.isArray(ent)) {\n ent = [];\n }\n\n if (!Array.isArray(fmt) || fmt.length == 0) {\n if (ent.length == 0) {\n return {\n text: txt\n };\n }\n\n // Handle special case when all values in fmt are 0 and fmt therefore is skipped.\n fmt = [{\n at: 0,\n len: 0,\n key: 0\n }];\n }\n\n // Sanitize spans.\n const spans = [];\n const attachments = [];\n fmt.forEach((span) => {\n if (!span || typeof span != 'object') {\n return;\n }\n\n if (!['undefined', 'number'].includes(typeof span.at)) {\n // Present, but non-numeric 'at'.\n return;\n }\n if (!['undefined', 'number'].includes(typeof span.len)) {\n // Present, but non-numeric 'len'.\n return;\n }\n let at = span.at | 0;\n let len = span.len | 0;\n if (len < 0) {\n // Invalid span length.\n return;\n }\n\n let key = span.key || 0;\n if (ent.length > 0 && (typeof key != 'number' || key < 0 || key >= ent.length)) {\n // Invalid key value.\n return;\n }\n\n if (at <= -1) {\n // Attachment. Store attachments separately.\n attachments.push({\n start: -1,\n end: 0,\n key: key\n });\n return;\n } else if (at + len > txt.length) {\n // Span is out of bounds.\n return;\n }\n\n if (!span.tp) {\n if (ent.length > 0 && (typeof ent[key] == 'object')) {\n spans.push({\n start: at,\n end: at + len,\n key: key\n });\n }\n } else {\n spans.push({\n type: span.tp,\n start: at,\n end: at + len\n });\n }\n });\n\n // Sort spans first by start index (asc) then by length (desc), then by weight.\n spans.sort((a, b) => {\n let diff = a.start - b.start;\n if (diff != 0) {\n return diff;\n }\n diff = b.end - a.end;\n if (diff != 0) {\n return diff;\n }\n return FMT_WEIGHT.indexOf(b.type) - FMT_WEIGHT.indexOf(a.type);\n });\n\n // Move attachments to the end of the list.\n if (attachments.length > 0) {\n spans.push(...attachments);\n }\n\n spans.forEach((span) => {\n if (ent.length > 0 && !span.type && ent[span.key] && typeof ent[span.key] == 'object') {\n span.type = ent[span.key].tp;\n span.data = ent[span.key].data;\n }\n\n // Is type still undefined? Hide the invalid element!\n if (!span.type) {\n span.type = 'HD';\n }\n });\n\n let tree = spansToTree({}, txt, 0, txt.length, spans);\n\n // Flatten tree nodes.\n const flatten = function(node) {\n if (Array.isArray(node.children) && node.children.length == 1) {\n // Unwrap.\n const child = node.children[0];\n if (!node.type) {\n const parent = node.parent;\n node = child;\n node.parent = parent;\n } else if (!child.type && !child.children) {\n node.text = child.text;\n delete node.children;\n }\n }\n return node;\n }\n tree = treeTopDown(tree, flatten);\n\n return tree;\n}\n\n// Add tree node to a parent tree.\nfunction addNode(parent, n) {\n if (!n) {\n return parent;\n }\n\n if (!parent.children) {\n parent.children = [];\n }\n\n // If text is present, move it to a subnode.\n if (parent.text) {\n parent.children.push({\n text: parent.text,\n parent: parent\n });\n delete parent.text;\n }\n\n n.parent = parent;\n parent.children.push(n);\n\n return parent;\n}\n\n// Returns a tree of nodes.\nfunction spansToTree(parent, text, start, end, spans) {\n if (!spans || spans.length == 0) {\n if (start < end) {\n addNode(parent, {\n text: text.substring(start, end)\n });\n }\n return parent;\n }\n\n // Process subspans.\n for (let i = 0; i < spans.length; i++) {\n const span = spans[i];\n if (span.start < 0 && span.type == 'EX') {\n addNode(parent, {\n type: span.type,\n data: span.data,\n key: span.key,\n att: true\n });\n continue;\n }\n\n // Add un-styled range before the styled span starts.\n if (start < span.start) {\n addNode(parent, {\n text: text.substring(start, span.start)\n });\n start = span.start;\n }\n\n // Get all spans which are within the current span.\n const subspans = [];\n while (i < spans.length - 1) {\n const inner = spans[i + 1];\n if (inner.start < 0) {\n // Attachments are in the end. Stop.\n break;\n } else if (inner.start < span.end) {\n if (inner.end <= span.end) {\n const tag = FORMAT_TAGS[inner.tp] || {};\n if (inner.start < inner.end || tag.isVoid) {\n // Valid subspan: completely within the current span and\n // either non-zero length or zero length is acceptable.\n subspans.push(inner);\n }\n }\n i++;\n // Overlapping subspans are ignored.\n } else {\n // Past the end of the current span. Stop.\n break;\n }\n }\n\n addNode(parent, spansToTree({\n type: span.type,\n data: span.data,\n key: span.key\n }, text, start, span.end, subspans));\n start = span.end;\n }\n\n // Add the last unformatted range.\n if (start < end) {\n addNode(parent, {\n text: text.substring(start, end)\n });\n }\n\n return parent;\n}\n\n// Append a tree to a Drafty doc.\nfunction treeToDrafty(doc, tree, keymap) {\n if (!tree) {\n return doc;\n }\n\n doc.txt = doc.txt || '';\n\n // Checkpoint to measure length of the current tree node.\n const start = doc.txt.length;\n\n if (tree.text) {\n doc.txt += tree.text;\n } else if (Array.isArray(tree.children)) {\n tree.children.forEach((c) => {\n treeToDrafty(doc, c, keymap);\n });\n }\n\n if (tree.type) {\n const len = doc.txt.length - start;\n doc.fmt = doc.fmt || [];\n if (Object.keys(tree.data || {}).length > 0) {\n doc.ent = doc.ent || [];\n const newKey = (typeof keymap[tree.key] == 'undefined') ? doc.ent.length : keymap[tree.key];\n keymap[tree.key] = newKey;\n doc.ent[newKey] = {\n tp: tree.type,\n data: tree.data\n };\n if (tree.att) {\n // Attachment.\n doc.fmt.push({\n at: -1,\n len: 0,\n key: newKey\n });\n } else {\n doc.fmt.push({\n at: start,\n len: len,\n key: newKey\n });\n }\n } else {\n doc.fmt.push({\n tp: tree.type,\n at: start,\n len: len\n });\n }\n }\n return doc;\n}\n\n// Traverse the tree top down transforming the nodes: apply transformer to every tree node.\nfunction treeTopDown(src, transformer, context) {\n if (!src) {\n return null;\n }\n\n let dst = transformer.call(context, src);\n if (!dst || !dst.children) {\n return dst;\n }\n\n const children = [];\n for (let i in dst.children) {\n let n = dst.children[i];\n if (n) {\n n = treeTopDown(n, transformer, context);\n if (n) {\n children.push(n);\n }\n }\n }\n\n if (children.length == 0) {\n dst.children = null;\n } else {\n dst.children = children;\n }\n\n return dst;\n}\n\n// Traverse the tree bottom-up: apply formatter to every node.\n// The formatter must maintain its state through context.\nfunction treeBottomUp(src, formatter, index, stack, context) {\n if (!src) {\n return null;\n }\n\n if (stack && src.type) {\n stack.push(src.type);\n }\n\n let values = [];\n for (let i in src.children) {\n const n = treeBottomUp(src.children[i], formatter, i, stack, context);\n if (n) {\n values.push(n);\n }\n }\n if (values.length == 0) {\n if (src.text) {\n values = [src.text];\n } else {\n values = null;\n }\n }\n\n if (stack && src.type) {\n stack.pop();\n }\n\n return formatter.call(context, src.type, src.data, values, index, stack);\n}\n\n// Clip tree to the provided limit.\nfunction shortenTree(tree, limit, tail) {\n if (!tree) {\n return null;\n }\n\n if (tail) {\n limit -= tail.length;\n }\n\n const shortener = function(node) {\n if (limit <= -1) {\n // Limit -1 means the doc was already clipped.\n return null;\n }\n\n if (node.att) {\n // Attachments are unchanged.\n return node;\n }\n if (limit == 0) {\n node.text = tail;\n limit = -1;\n } else if (node.text) {\n const len = node.text.length;\n if (len > limit) {\n node.text = node.text.substring(0, limit) + tail;\n limit = -1;\n } else {\n limit -= len;\n }\n }\n return node;\n }\n\n return treeTopDown(tree, shortener);\n}\n\n// Strip heavy entities from a tree.\nfunction lightEntity(tree, allow) {\n const lightCopy = node => {\n const data = copyEntData(node.data, true, allow ? allow(node) : null);\n if (data) {\n node.data = data;\n } else {\n delete node.data;\n }\n return node;\n }\n return treeTopDown(tree, lightCopy);\n}\n\n// Remove spaces and breaks on the left.\nfunction lTrim(tree) {\n if (tree.type == 'BR') {\n tree = null;\n } else if (tree.text) {\n if (!tree.type) {\n tree.text = tree.text.trimStart();\n if (!tree.text) {\n tree = null;\n }\n }\n } else if (!tree.type && tree.children && tree.children.length > 0) {\n const c = lTrim(tree.children[0]);\n if (c) {\n tree.children[0] = c;\n } else {\n tree.children.shift();\n if (!tree.type && tree.children.length == 0) {\n tree = null;\n }\n }\n }\n return tree;\n}\n\n// Move attachments to the end. Attachments must be at the top level, no need to traverse the tree.\nfunction attachmentsToEnd(tree, limit) {\n if (!tree) {\n return null;\n }\n\n if (tree.att) {\n tree.text = ' ';\n delete tree.att;\n delete tree.children;\n } else if (tree.children) {\n const attachments = [];\n const children = [];\n for (let i in tree.children) {\n const c = tree.children[i];\n if (c.att) {\n if (attachments.length == limit) {\n // Too many attachments to preview;\n continue;\n }\n if (c.data['mime'] == JSON_MIME_TYPE) {\n // JSON attachments are not shown in preview.\n continue;\n }\n\n delete c.att;\n delete c.children;\n c.text = ' ';\n attachments.push(c);\n } else {\n children.push(c);\n }\n }\n tree.children = children.concat(attachments);\n }\n return tree;\n}\n\n// Get a list of entities from a text.\nfunction extractEntities(line) {\n let match;\n let extracted = [];\n ENTITY_TYPES.forEach((entity) => {\n while ((match = entity.re.exec(line)) !== null) {\n extracted.push({\n offset: match['index'],\n len: match[0].length,\n unique: match[0],\n data: entity.pack(match[0]),\n type: entity.name\n });\n }\n });\n\n if (extracted.length == 0) {\n return extracted;\n }\n\n // Remove entities detected inside other entities, like #hashtag in a URL.\n extracted.sort((a, b) => {\n return a.offset - b.offset;\n });\n\n let idx = -1;\n extracted = extracted.filter((el) => {\n const result = (el.offset > idx);\n idx = el.offset + el.len;\n return result;\n });\n\n return extracted;\n}\n\n// Convert the chunks into format suitable for serialization.\nfunction draftify(chunks, startAt) {\n let plain = '';\n let ranges = [];\n for (let i in chunks) {\n const chunk = chunks[i];\n if (!chunk.txt) {\n const drafty = draftify(chunk.children, plain.length + startAt);\n chunk.txt = drafty.txt;\n ranges = ranges.concat(drafty.fmt);\n }\n\n if (chunk.tp) {\n ranges.push({\n at: plain.length + startAt,\n len: chunk.txt.length,\n tp: chunk.tp\n });\n }\n\n plain += chunk.txt;\n }\n return {\n txt: plain,\n fmt: ranges\n };\n}\n\n// Create a copy of entity data with (light=false) or without (light=true) the large payload.\n// The array 'allow' contains a list of fields exempt from stripping.\nfunction copyEntData(data, light, allow) {\n if (data && Object.entries(data).length > 0) {\n allow = allow || [];\n const dc = {};\n ALLOWED_ENT_FIELDS.forEach(key => {\n if (data[key]) {\n if (light && !allow.includes(key) &&\n (typeof data[key] == 'string' || Array.isArray(data[key])) &&\n data[key].length > MAX_PREVIEW_DATA_SIZE) {\n return;\n }\n if (typeof data[key] == 'object') {\n return;\n }\n dc[key] = data[key];\n }\n });\n\n if (Object.entries(dc).length != 0) {\n return dc;\n }\n }\n return null;\n}\n\nif (typeof module != 'undefined') {\n module.exports = Drafty;\n}\n","/**\n * @file Utilities for uploading and downloading files.\n *\n * @copyright 2015-2023 Tinode LLC.\n */\n'use strict';\n\nimport CommError from './comm-error.js';\nimport {\n isUrlRelative,\n jsonParseHelper\n} from './utils.js';\n\nlet XHRProvider;\n\nfunction addURLParam(relUrl, key, value) {\n const url = new URL(relUrl, window.location.origin);\n url.searchParams.append(key, value);\n return url.toString().substring(window.location.origin.length);\n}\n\n/**\n * @class LargeFileHelper - utilities for uploading and downloading files out of band.\n * Don't instantiate this class directly. Use {Tinode.getLargeFileHelper} instead.\n * @memberof Tinode\n *\n * @param {Tinode} tinode - the main Tinode object.\n * @param {string} version - protocol version, i.e. '0'.\n */\nexport default class LargeFileHelper {\n constructor(tinode, version) {\n this._tinode = tinode;\n this._version = version;\n\n this._apiKey = tinode._apiKey;\n this._authToken = tinode.getAuthToken();\n\n // Ongoing requests.\n this.xhr = [];\n }\n\n /**\n * Start uploading the file to an endpoint at baseUrl.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {string} baseUrl base URL of upload server.\n * @param {File|Blob} data data to upload.\n * @param {string} avatarFor topic name if the upload represents an avatar.\n * @param {Callback} onProgress callback. Takes one {float} parameter 0..1\n * @param {Callback} onSuccess callback. Called when the file is successfully uploaded.\n * @param {Callback} onFailure callback. Called in case of a failure.\n *\n * @returns {Promise} resolved/rejected when the upload is completed/failed.\n */\n uploadWithBaseUrl(baseUrl, data, avatarFor, onProgress, onSuccess, onFailure) {\n let url = `/v${this._version}/file/u/`;\n if (baseUrl) {\n let base = baseUrl;\n if (base.endsWith('/')) {\n // Removing trailing slash.\n base = base.slice(0, -1);\n }\n if (base.startsWith('http://') || base.startsWith('https://')) {\n url = base + url;\n } else {\n throw new Error(`Invalid base URL '${baseUrl}'`);\n }\n }\n\n const instance = this;\n const xhr = new XHRProvider();\n this.xhr.push(xhr);\n\n xhr.open('POST', url, true);\n xhr.setRequestHeader('X-Tinode-APIKey', this._apiKey);\n if (this._authToken) {\n xhr.setRequestHeader('X-Tinode-Auth', `Token ${this._authToken.token}`);\n }\n\n let toResolve = null;\n let toReject = null;\n\n const result = new Promise((resolve, reject) => {\n toResolve = resolve;\n toReject = reject;\n });\n\n xhr.upload.onprogress = e => {\n if (e.lengthComputable) {\n if (onProgress) {\n onProgress(e.loaded / e.total);\n }\n if (this.onProgress) {\n this.onProgress(e.loaded / e.total);\n }\n }\n };\n\n xhr.onload = function() {\n let pkt;\n try {\n pkt = JSON.parse(this.response, jsonParseHelper);\n } catch (err) {\n instance._tinode.logger(\"ERROR: Invalid server response in LargeFileHelper\", this.response);\n pkt = {\n ctrl: {\n code: this.status,\n text: this.statusText\n }\n };\n }\n\n if (this.status >= 200 && this.status < 300) {\n if (toResolve) {\n toResolve(pkt.ctrl.params.url);\n }\n if (onSuccess) {\n onSuccess(pkt.ctrl);\n }\n } else if (this.status >= 400) {\n if (toReject) {\n toReject(new CommError(pkt.ctrl.text, pkt.ctrl.code));\n }\n if (onFailure) {\n onFailure(pkt.ctrl);\n }\n } else {\n instance._tinode.logger(\"ERROR: Unexpected server response status\", this.status, this.response);\n }\n };\n\n xhr.onerror = function(e) {\n if (toReject) {\n toReject(e || new Error(\"failed\"));\n }\n if (onFailure) {\n onFailure(null);\n }\n };\n\n xhr.onabort = function(e) {\n if (toReject) {\n toReject(new Error(\"upload cancelled by user\"));\n }\n if (onFailure) {\n onFailure(null);\n }\n };\n\n try {\n const form = new FormData();\n form.append('file', data);\n form.set('id', this._tinode.getNextUniqueId());\n if (avatarFor) {\n form.set('topic', avatarFor);\n }\n xhr.send(form);\n } catch (err) {\n if (toReject) {\n toReject(err);\n }\n if (onFailure) {\n onFailure(null);\n }\n }\n\n return result;\n }\n /**\n * Start uploading the file to default endpoint.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {File|Blob} data to upload\n * @param {string} avatarFor topic name if the upload represents an avatar.\n * @param {Callback} onProgress callback. Takes one {float} parameter 0..1\n * @param {Callback} onSuccess callback. Called when the file is successfully uploaded.\n * @param {Callback} onFailure callback. Called in case of a failure.\n *\n * @returns {Promise} resolved/rejected when the upload is completed/failed.\n */\n upload(data, avatarFor, onProgress, onSuccess, onFailure) {\n const baseUrl = (this._tinode._secure ? 'https://' : 'http://') + this._tinode._host;\n return this.uploadWithBaseUrl(baseUrl, data, avatarFor, onProgress, onSuccess, onFailure);\n }\n /**\n * Download the file from a given URL using GET request. This method works with the Tinode server only.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {string} relativeUrl - URL to download the file from. Must be relative url, i.e. must not contain the host.\n * @param {string=} filename - file name to use for the downloaded file.\n *\n * @returns {Promise} resolved/rejected when the download is completed/failed.\n */\n download(relativeUrl, filename, mimetype, onProgress, onError) {\n if (!isUrlRelative(relativeUrl)) {\n // As a security measure refuse to download from an absolute URL.\n if (onError) {\n onError(`The URL '${relativeUrl}' must be relative, not absolute`);\n }\n return;\n }\n if (!this._authToken) {\n if (onError) {\n onError(\"Must authenticate first\");\n }\n return;\n }\n const instance = this;\n\n const xhr = new XHRProvider();\n this.xhr.push(xhr);\n\n // Add '&asatt=1' to URL to request 'Content-Disposition: attachment' response header.\n relativeUrl = addURLParam(relativeUrl, 'asatt', '1');\n\n // Get data as blob (stored by the browser as a temporary file).\n xhr.open('GET', relativeUrl, true);\n xhr.setRequestHeader('X-Tinode-APIKey', this._apiKey);\n xhr.setRequestHeader('X-Tinode-Auth', 'Token ' + this._authToken.token);\n xhr.responseType = 'blob';\n\n xhr.onprogress = function(e) {\n if (onProgress) {\n // Passing e.loaded instead of e.loaded/e.total because e.total\n // is always 0 with gzip compression enabled by the server.\n onProgress(e.loaded);\n }\n };\n\n let toResolve = null;\n let toReject = null;\n\n const result = new Promise((resolve, reject) => {\n toResolve = resolve;\n toReject = reject;\n });\n\n // The blob needs to be saved as file. There is no known way to\n // save the blob as file other than to fake a click on an .\n xhr.onload = function() {\n if (this.status == 200) {\n const link = document.createElement('a');\n // URL.createObjectURL is not available in non-browser environment. This call will fail.\n link.href = window.URL.createObjectURL(new Blob([this.response], {\n type: mimetype\n }));\n link.style.display = 'none';\n link.setAttribute('download', filename);\n document.body.appendChild(link);\n link.click();\n document.body.removeChild(link);\n window.URL.revokeObjectURL(link.href);\n if (toResolve) {\n toResolve();\n }\n } else if (this.status >= 400 && toReject) {\n // The this.responseText is undefined, must use this.response which is a blob.\n // Need to convert this.response to JSON. The blob can only be accessed by the\n // FileReader.\n const reader = new FileReader();\n reader.onload = function() {\n try {\n const pkt = JSON.parse(this.result, jsonParseHelper);\n toReject(new CommError(pkt.ctrl.text, pkt.ctrl.code));\n } catch (err) {\n instance._tinode.logger(\"ERROR: Invalid server response in LargeFileHelper\", this.result);\n toReject(err);\n }\n };\n reader.readAsText(this.response);\n }\n };\n\n xhr.onerror = function(e) {\n if (toReject) {\n toReject(new Error(\"failed\"));\n }\n if (onError) {\n onError(e);\n }\n };\n\n xhr.onabort = function() {\n if (toReject) {\n toReject(null);\n }\n };\n\n try {\n xhr.send();\n } catch (err) {\n if (toReject) {\n toReject(err);\n }\n if (onError) {\n onError(err);\n }\n }\n\n return result;\n }\n /**\n * Try to cancel all ongoing uploads or downloads.\n * @memberof Tinode.LargeFileHelper#\n */\n cancel() {\n this.xhr.forEach(req => {\n if (req.readyState < 4) {\n req.abort();\n }\n });\n }\n /**\n * To use LargeFileHelper in a non browser context, supply XMLHttpRequest provider.\n * @static\n * @memberof LargeFileHelper\n * @param xhrProvider XMLHttpRequest provider, e.g. for node require('xhr')
.\n */\n static setNetworkProvider(xhrProvider) {\n XHRProvider = xhrProvider;\n }\n}\n","/**\n * @file Helper class for constructing {@link Tinode.GetQuery}.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n/**\n * Helper class for constructing {@link Tinode.GetQuery}.\n *\n * @class MetaGetBuilder\n * @memberof Tinode\n *\n * @param {Tinode.Topic} parent topic which instantiated this builder.\n */\nexport default class MetaGetBuilder {\n constructor(parent) {\n this.topic = parent;\n this.what = {};\n }\n\n // Get timestamp of the most recent desc update.\n #get_desc_ims() {\n return this.topic._deleted ? undefined : this.topic.updated;\n }\n\n // Get timestamp of the most recent subs update.\n #get_subs_ims() {\n if (this.topic.isP2PType()) {\n return this.#get_desc_ims();\n }\n return this.topic._deleted ? undefined : this.topic._lastSubsUpdate;\n }\n /**\n * Add query parameters to fetch messages within explicit limits.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} since - messages newer than this (inclusive);\n * @param {number=} before - older than this (exclusive)\n * @param {number=} limit - number of messages to fetch\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withData(since, before, limit) {\n this.what['data'] = {\n since: since,\n before: before,\n limit: limit\n };\n return this;\n }\n /**\n * Add query parameters to fetch messages newer than the latest saved message.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - number of messages to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterData(limit) {\n return this.withData(this.topic._maxSeq > 0 ? this.topic._maxSeq + 1 : undefined, undefined, limit);\n }\n /**\n * Add query parameters to fetch messages older than the earliest saved message.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - maximum number of messages to fetch.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withEarlierData(limit) {\n return this.withData(undefined, this.topic._minSeq > 0 ? this.topic._minSeq : undefined, limit);\n }\n /**\n * Add query parameters to fetch topic description if it's newer than the given timestamp.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch messages newer than this timestamp.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withDesc(ims) {\n this.what['desc'] = {\n ims: ims\n };\n return this;\n }\n /**\n * Add query parameters to fetch topic description if it's newer than the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterDesc() {\n return this.withDesc(this.#get_desc_ims());\n }\n /**\n * Add query parameters to fetch subscriptions.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch subscriptions modified more recently than this timestamp\n * @param {number=} limit - maximum number of subscriptions to fetch.\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withSub(ims, limit, userOrTopic) {\n const opts = {\n ims: ims,\n limit: limit\n };\n if (this.topic.getType() == 'me') {\n opts.topic = userOrTopic;\n } else {\n opts.user = userOrTopic;\n }\n this.what['sub'] = opts;\n return this;\n }\n /**\n * Add query parameters to fetch a single subscription.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch subscriptions modified more recently than this timestamp\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withOneSub(ims, userOrTopic) {\n return this.withSub(ims, undefined, userOrTopic);\n }\n /**\n * Add query parameters to fetch a single subscription if it's been updated since the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterOneSub(userOrTopic) {\n return this.withOneSub(this.topic._lastSubsUpdate, userOrTopic);\n }\n /**\n * Add query parameters to fetch subscriptions updated since the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - maximum number of subscriptions to fetch.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterSub(limit) {\n return this.withSub(this.#get_subs_ims(), limit);\n }\n /**\n * Add query parameters to fetch topic tags.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withTags() {\n this.what['tags'] = true;\n return this;\n }\n /**\n * Add query parameters to fetch user's credentials. 'me'
topic only.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withCred() {\n if (this.topic.getType() == 'me') {\n this.what['cred'] = true;\n } else {\n this.topic._tinode.logger(\"ERROR: Invalid topic type for MetaGetBuilder:withCreds\", this.topic.getType());\n }\n return this;\n }\n /**\n * Add query parameters to fetch deleted messages within explicit limits. Any/all parameters can be null.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} since - ids of messages deleted since this 'del' id (inclusive)\n * @param {number=} limit - number of deleted message ids to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withDel(since, limit) {\n if (since || limit) {\n this.what['del'] = {\n since: since,\n limit: limit\n };\n }\n return this;\n }\n /**\n * Add query parameters to fetch messages deleted after the saved 'del'
id.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - number of deleted message ids to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterDel(limit) {\n // Specify 'since' only if we have already received some messages. If\n // we have no locally cached messages then we don't care if any messages were deleted.\n return this.withDel(this.topic._maxSeq > 0 ? this.topic._maxDel + 1 : undefined, limit);\n }\n\n /**\n * Extract subquery: get an object that contains specified subquery.\n * @memberof Tinode.MetaGetBuilder#\n * @param {string} what - subquery to return: one of 'data', 'sub', 'desc', 'tags', 'cred', 'del'.\n * @returns {Object} requested subquery or undefined
.\n */\n extract(what) {\n return this.what[what];\n }\n\n /**\n * Construct parameters.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.GetQuery} Get query\n */\n build() {\n const what = [];\n let params = {};\n ['data', 'sub', 'desc', 'tags', 'cred', 'del'].forEach((key) => {\n if (this.what.hasOwnProperty(key)) {\n what.push(key);\n if (Object.getOwnPropertyNames(this.what[key]).length > 0) {\n params[key] = this.what[key];\n }\n }\n });\n if (what.length > 0) {\n params.what = what.join(' ');\n } else {\n params = undefined;\n }\n return params;\n }\n}\n","/**\n * @file Topic management.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport AccessMode from './access-mode.js';\nimport CBuffer from './cbuffer.js';\nimport CommError from './comm-error.js';\nimport * as Const from './config.js';\nimport Drafty from './drafty.js';\nimport MetaGetBuilder from './meta-builder.js';\nimport {\n mergeObj,\n mergeToCache,\n normalizeArray\n} from './utils.js';\n\n/**\n * Topic is a class representing a logical communication channel.\n */\nexport class Topic {\n /**\n * @callback onData\n * @param {Data} data - Data packet\n */\n\n /**\n * Create topic.\n * @param {string} name - Name of the topic to create.\n * @param {Object=} callbacks - Object with various event callbacks.\n * @param {onData} callbacks.onData - Callback which receives a {data}
message.\n * @param {callback} callbacks.onMeta - Callback which receives a {meta}
message.\n * @param {callback} callbacks.onPres - Callback which receives a {pres}
message.\n * @param {callback} callbacks.onInfo - Callback which receives an {info}
message.\n * @param {callback} callbacks.onMetaDesc - Callback which receives changes to topic desctioption {@link desc}.\n * @param {callback} callbacks.onMetaSub - Called for a single subscription record change.\n * @param {callback} callbacks.onSubsUpdated - Called after a batch of subscription changes have been recieved and cached.\n * @param {callback} callbacks.onDeleteTopic - Called after the topic is deleted.\n * @param {callback} callbacls.onAllMessagesReceived - Called when all requested {data}
messages have been recived.\n */\n constructor(name, callbacks) {\n // Parent Tinode object.\n this._tinode = null;\n\n // Server-provided data, locally immutable.\n // topic name\n this.name = name;\n // Timestamp when the topic was created.\n this.created = null;\n // Timestamp when the topic was last updated.\n this.updated = null;\n // Timestamp of the last messages\n this.touched = new Date(0);\n // Access mode, see AccessMode\n this.acs = new AccessMode(null);\n // Per-topic private data (accessible by current user only).\n this.private = null;\n // Per-topic public data (accessible by all users).\n this.public = null;\n // Per-topic system-provided data (accessible by all users).\n this.trusted = null;\n\n // Locally cached data\n // Subscribed users, for tracking read/recv/msg notifications.\n this._users = {};\n\n // Current value of locally issued seqId, used for pending messages.\n this._queuedSeqId = Const.LOCAL_SEQID;\n\n // The maximum known {data.seq} value.\n this._maxSeq = 0;\n // The minimum known {data.seq} value.\n this._minSeq = 0;\n // Indicator that the last request for earlier messages returned 0.\n this._noEarlierMsgs = false;\n // The maximum known deletion ID.\n this._maxDel = 0;\n // Timer object used to send 'recv' notifications.\n this._recvNotificationTimer = null;\n\n // User discovery tags\n this._tags = [];\n // Credentials such as email or phone number.\n this._credentials = [];\n // Message versions cache (e.g. for edited messages).\n // Keys: original message seq ID.\n // Values: CBuffers containing newer versions of the original message\n // ordered by seq id.\n this._messageVersions = {};\n // Message cache, sorted by message seq values, from old to new.\n this._messages = new CBuffer((a, b) => {\n return a.seq - b.seq;\n }, true);\n // Boolean, true if the topic is currently live\n this._attached = false;\n // Timestap of the most recently updated subscription.\n this._lastSubsUpdate = new Date(0);\n // Topic created but not yet synced with the server. Used only during initialization.\n this._new = true;\n // The topic is deleted at the server, this is a local copy.\n this._deleted = false;\n\n // Timer used to trgger {leave} request after a delay.\n this._delayedLeaveTimer = null;\n\n // Callbacks\n if (callbacks) {\n this.onData = callbacks.onData;\n this.onMeta = callbacks.onMeta;\n this.onPres = callbacks.onPres;\n this.onInfo = callbacks.onInfo;\n // A single desc update;\n this.onMetaDesc = callbacks.onMetaDesc;\n // A single subscription record;\n this.onMetaSub = callbacks.onMetaSub;\n // All subscription records received;\n this.onSubsUpdated = callbacks.onSubsUpdated;\n this.onTagsUpdated = callbacks.onTagsUpdated;\n this.onCredsUpdated = callbacks.onCredsUpdated;\n this.onDeleteTopic = callbacks.onDeleteTopic;\n this.onAllMessagesReceived = callbacks.onAllMessagesReceived;\n }\n }\n\n // Static methods.\n\n /**\n * Determine topic type from topic's name: grp, p2p, me, fnd, sys.\n *\n * @param {string} name - Name of the topic to test.\n * @returns {string} One of \"me\"
, \"fnd\"
, \"sys\"
, \"grp\"
,\n * \"p2p\"
or undefined
.\n */\n static topicType(name) {\n const types = {\n 'me': Const.TOPIC_ME,\n 'fnd': Const.TOPIC_FND,\n 'grp': Const.TOPIC_GRP,\n 'new': Const.TOPIC_GRP,\n 'nch': Const.TOPIC_GRP,\n 'chn': Const.TOPIC_GRP,\n 'usr': Const.TOPIC_P2P,\n 'sys': Const.TOPIC_SYS\n };\n return types[(typeof name == 'string') ? name.substring(0, 3) : 'xxx'];\n }\n\n /**\n * Check if the given topic name is a name of a 'me' topic.\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a 'me' topic, false
otherwise.\n */\n static isMeTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_ME;\n }\n\n /**\n * Check if the given topic name is a name of a group topic.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a group topic, false
otherwise.\n */\n static isGroupTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_GRP;\n }\n\n /**\n * Check if the given topic name is a name of a p2p topic.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p topic, false
otherwise.\n */\n static isP2PTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_P2P;\n }\n\n /**\n * Check if the given topic name is a name of a communication topic, i.e. P2P or group.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p or group topic, false
otherwise.\n */\n static isCommTopicName(name) {\n return Topic.isP2PTopicName(name) || Topic.isGroupTopicName(name);\n }\n\n /**\n * Check if the topic name is a name of a new topic.\n * @static\n *\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a new topic, false
otherwise.\n */\n static isNewGroupTopicName(name) {\n return (typeof name == 'string') &&\n (name.substring(0, 3) == Const.TOPIC_NEW || name.substring(0, 3) == Const.TOPIC_NEW_CHAN);\n }\n\n /**\n * Check if the topic name is a name of a channel.\n * @static\n *\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a channel, false
otherwise.\n */\n static isChannelTopicName(name) {\n return (typeof name == 'string') &&\n (name.substring(0, 3) == Const.TOPIC_CHAN || name.substring(0, 3) == Const.TOPIC_NEW_CHAN);\n }\n\n /**\n * Check if the topic is subscribed.\n * @returns {boolean} True is topic is attached/subscribed, false otherwise.\n */\n isSubscribed() {\n return this._attached;\n }\n\n /**\n * Request topic to subscribe. Wrapper for {@link Tinode#subscribe}.\n *\n * @param {Tinode.GetQuery=} getParams - get query parameters.\n * @param {Tinode.SetParams=} setParams - set parameters.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n subscribe(getParams, setParams) {\n // Clear request to leave topic.\n clearTimeout(this._delayedLeaveTimer);\n this._delayedLeaveTimer = null;\n\n // If the topic is already subscribed, return resolved promise\n if (this._attached) {\n return Promise.resolve(this);\n }\n\n // Send subscribe message, handle async response.\n // If topic name is explicitly provided, use it. If no name, then it's a new group topic,\n // use \"new\".\n return this._tinode.subscribe(this.name || Const.TOPIC_NEW, getParams, setParams).then(ctrl => {\n if (ctrl.code >= 300) {\n // Do nothing if subscription status has not changed.\n return ctrl;\n }\n\n this._attached = true;\n this._deleted = false;\n this.acs = (ctrl.params && ctrl.params.acs) ? ctrl.params.acs : this.acs;\n\n // Set topic name for new topics and add it to cache.\n if (this._new) {\n delete this._new;\n\n if (this.name != ctrl.topic) {\n // Name may change new123456 -> grpAbCdEf. Remove from cache under the old name.\n this._cacheDelSelf();\n this.name = ctrl.topic;\n }\n this._cachePutSelf();\n\n this.created = ctrl.ts;\n this.updated = ctrl.ts;\n\n if (this.name != Const.TOPIC_ME && this.name != Const.TOPIC_FND) {\n // Add the new topic to the list of contacts maintained by the 'me' topic.\n const me = this._tinode.getMeTopic();\n if (me.onMetaSub) {\n me.onMetaSub(this);\n }\n if (me.onSubsUpdated) {\n me.onSubsUpdated([this.name], 1);\n }\n }\n\n if (setParams && setParams.desc) {\n setParams.desc._noForwarding = true;\n this._processMetaDesc(setParams.desc);\n }\n }\n return ctrl;\n });\n }\n\n /**\n * Create a draft of a message without sending it to the server.\n * @memberof Tinode.Topic#\n *\n * @param {string | Object} data - Content to wrap in a draft.\n * @param {boolean=} noEcho - If true
server will not echo message back to originating\n * session. Otherwise the server will send a copy of the message to sender.\n *\n * @returns {Object} message draft.\n */\n createMessage(data, noEcho) {\n return this._tinode.createMessage(this.name, data, noEcho);\n }\n\n /**\n * Immediately publish data to topic. Wrapper for {@link Tinode#publish}.\n * @memberof Tinode.Topic#\n *\n * @param {string | Object} data - Message to publish, either plain string or a Drafty object.\n * @param {boolean=} noEcho - If true
server will not echo message back to originating\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n publish(data, noEcho) {\n return this.publishMessage(this.createMessage(data, noEcho));\n }\n\n /**\n * Publish message created by {@link Tinode.Topic#createMessage}.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub - {data} object to publish. Must be created by {@link Tinode.Topic#createMessage}\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n publishMessage(pub) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot publish on inactive topic\"));\n }\n if (this._sending) {\n return Promise.reject(new Error(\"The message is already being sent\"));\n }\n\n // Send data.\n pub._sending = true;\n pub._failed = false;\n\n // Extract refereces to attachments and out of band image records.\n let attachments = null;\n if (Drafty.hasEntities(pub.content)) {\n attachments = [];\n Drafty.entities(pub.content, data => {\n if (data) {\n if (data.ref) {\n attachments.push(data.ref);\n }\n if (data.preref) {\n attachments.push(data.preref);\n }\n }\n });\n if (attachments.length == 0) {\n attachments = null;\n }\n }\n\n return this._tinode.publishMessage(pub, attachments).then(ctrl => {\n pub._sending = false;\n pub.ts = ctrl.ts;\n this.swapMessageId(pub, ctrl.params.seq);\n this._maybeUpdateMessageVersionsCache(pub);\n this._routeData(pub);\n return ctrl;\n }).catch(err => {\n this._tinode.logger(\"WARNING: Message rejected by the server\", err);\n pub._sending = false;\n pub._failed = true;\n if (this.onData) {\n this.onData();\n }\n });\n }\n\n /**\n * Add message to local message cache, send to the server when the promise is resolved.\n * If promise is null or undefined, the message will be sent immediately.\n * The message is sent when the\n * The message should be created by {@link Tinode.Topic#createMessage}.\n * This is probably not the final API.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub - Message to use as a draft.\n * @param {Promise} prom - Message will be sent when this promise is resolved, discarded if rejected.\n *\n * @returns {Promise} derived promise.\n */\n publishDraft(pub, prom) {\n const seq = pub.seq || this._getQueuedSeqId();\n if (!pub._noForwarding) {\n // The 'seq', 'ts', and 'from' are added to mimic {data}. They are removed later\n // before the message is sent.\n pub._noForwarding = true;\n pub.seq = seq;\n pub.ts = new Date();\n pub.from = this._tinode.getCurrentUserID();\n\n // Don't need an echo message because the message is added to local cache right away.\n pub.noecho = true;\n // Add to cache.\n this._messages.put(pub);\n this._tinode._db.addMessage(pub);\n\n if (this.onData) {\n this.onData(pub);\n }\n }\n // If promise is provided, send the queued message when it's resolved.\n // If no promise is provided, create a resolved one and send immediately.\n return (prom || Promise.resolve())\n .then(_ => {\n if (pub._cancelled) {\n return {\n code: 300,\n text: \"cancelled\"\n };\n }\n return this.publishMessage(pub);\n }).catch(err => {\n this._tinode.logger(\"WARNING: Message draft rejected\", err);\n pub._sending = false;\n pub._failed = true;\n pub._fatal = err instanceof CommError ? (err.code >= 400 && err.code < 500) : false;\n if (this.onData) {\n this.onData();\n }\n // Rethrow to let caller know that the operation failed.\n throw err;\n });\n }\n\n /**\n * Leave the topic, optionally unsibscribe. Leaving the topic means the topic will stop\n * receiving updates from the server. Unsubscribing will terminate user's relationship with the topic.\n * Wrapper for {@link Tinode#leave}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean=} unsub - If true, unsubscribe, otherwise just leave.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n leave(unsub) {\n // It's possible to unsubscribe (unsub==true) from inactive topic.\n if (!this._attached && !unsub) {\n return Promise.reject(new Error(\"Cannot leave inactive topic\"));\n }\n\n // Send a 'leave' message, handle async response\n return this._tinode.leave(this.name, unsub).then(ctrl => {\n this._resetSub();\n if (unsub) {\n this._gone();\n }\n return ctrl;\n });\n }\n\n /**\n * Leave the topic, optionally unsibscribe after a delay. Leaving the topic means the topic will stop\n * receiving updates from the server. Unsubscribing will terminate user's relationship with the topic.\n * Wrapper for {@link Tinode#leave}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} unsub - If true, unsubscribe, otherwise just leave.\n * @param {number} delay - time in milliseconds to delay leave request.\n */\n leaveDelayed(unsub, delay) {\n clearTimeout(this._delayedLeaveTimer);\n this._delayedLeaveTimer = setTimeout(_ => {\n this._delayedLeaveTimer = null;\n this.leave(unsub)\n }, delay);\n }\n\n /**\n * Request topic metadata from the server.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.GetQuery} request parameters\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n getMeta(params) {\n // Send {get} message, return promise.\n return this._tinode.getMeta(this.name, params);\n }\n\n /**\n * Request more messages from the server\n * @memberof Tinode.Topic#\n *\n * @param {number} limit number of messages to get.\n * @param {boolean} forward if true, request newer messages.\n */\n getMessagesPage(limit, forward) {\n let query = forward ?\n this.startMetaQuery().withLaterData(limit) :\n this.startMetaQuery().withEarlierData(limit);\n\n // First try fetching from DB, then from the server.\n return this._loadMessages(this._tinode._db, query.extract('data'))\n .then((count) => {\n if (count == limit) {\n // Got enough messages from local cache.\n return Promise.resolve({\n topic: this.name,\n code: 200,\n params: {\n count: count\n }\n });\n }\n\n // Reduce the count of requested messages.\n limit -= count;\n // Update query with new values loaded from DB.\n query = forward ? this.startMetaQuery().withLaterData(limit) :\n this.startMetaQuery().withEarlierData(limit);\n let promise = this.getMeta(query.build());\n if (!forward) {\n promise = promise.then(ctrl => {\n if (ctrl && ctrl.params && !ctrl.params.count) {\n this._noEarlierMsgs = true;\n }\n });\n }\n return promise;\n });\n }\n /**\n * Update topic metadata.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.SetParams} params parameters to update.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n setMeta(params) {\n if (params.tags) {\n params.tags = normalizeArray(params.tags);\n }\n // Send Set message, handle async response.\n return this._tinode.setMeta(this.name, params)\n .then(ctrl => {\n if (ctrl && ctrl.code >= 300) {\n // Not modified\n return ctrl;\n }\n\n if (params.sub) {\n params.sub.topic = this.name;\n if (ctrl.params && ctrl.params.acs) {\n params.sub.acs = ctrl.params.acs;\n params.sub.updated = ctrl.ts;\n }\n if (!params.sub.user) {\n // This is a subscription update of the current user.\n // Assign user ID otherwise the update will be ignored by _processMetaSubs.\n params.sub.user = this._tinode.getCurrentUserID();\n if (!params.desc) {\n // Force update to topic's asc.\n params.desc = {};\n }\n }\n params.sub._noForwarding = true;\n this._processMetaSubs([params.sub]);\n }\n\n if (params.desc) {\n if (ctrl.params && ctrl.params.acs) {\n params.desc.acs = ctrl.params.acs;\n params.desc.updated = ctrl.ts;\n }\n this._processMetaDesc(params.desc);\n }\n\n if (params.tags) {\n this._processMetaTags(params.tags);\n }\n if (params.cred) {\n this._processMetaCreds([params.cred], true);\n }\n\n return ctrl;\n });\n }\n /**\n * Update access mode of the current user or of another topic subsriber.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - UID of the user to update or null to update current user.\n * @param {string} update - the update value, full or delta.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n updateMode(uid, update) {\n const user = uid ? this.subscriber(uid) : null;\n const am = user ?\n user.acs.updateGiven(update).getGiven() :\n this.getAccessMode().updateWant(update).getWant();\n\n return this.setMeta({\n sub: {\n user: uid,\n mode: am\n }\n });\n }\n /**\n * Create new topic subscription. Wrapper for {@link Tinode#setMeta}.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - ID of the user to invite\n * @param {string=} mode - Access mode. null
means to use default.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n invite(uid, mode) {\n return this.setMeta({\n sub: {\n user: uid,\n mode: mode\n }\n });\n }\n /**\n * Archive or un-archive the topic. Wrapper for {@link Tinode#setMeta}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} arch - true to archive the topic, false otherwise.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n archive(arch) {\n if (this.private && (!this.private.arch == !arch)) {\n return Promise.resolve(arch);\n }\n return this.setMeta({\n desc: {\n private: {\n arch: arch ? true : Const.DEL_CHAR\n }\n }\n });\n }\n /**\n * Delete messages. Hard-deleting messages requires Owner permission.\n * Wrapper for {@link Tinode#delMessages}.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.DelRange[]} ranges - Ranges of message IDs to delete.\n * @param {boolean=} hard - Hard or soft delete\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessages(ranges, hard) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete messages in inactive topic\"));\n }\n\n // Sort ranges in accending order by low, the descending by hi.\n ranges.sort((r1, r2) => {\n if (r1.low < r2.low) {\n return true;\n }\n if (r1.low == r2.low) {\n return !r2.hi || (r1.hi >= r2.hi);\n }\n return false;\n });\n\n // Remove pending messages from ranges possibly clipping some ranges.\n let tosend = ranges.reduce((out, r) => {\n if (r.low < Const.LOCAL_SEQID) {\n if (!r.hi || r.hi < Const.LOCAL_SEQID) {\n out.push(r);\n } else {\n // Clip hi to max allowed value.\n out.push({\n low: r.low,\n hi: this._maxSeq + 1\n });\n }\n }\n return out;\n }, []);\n\n // Send {del} message, return promise\n let result;\n if (tosend.length > 0) {\n result = this._tinode.delMessages(this.name, tosend, hard);\n } else {\n result = Promise.resolve({\n params: {\n del: 0\n }\n });\n }\n // Update local cache.\n return result.then(ctrl => {\n if (ctrl.params.del > this._maxDel) {\n this._maxDel = ctrl.params.del;\n }\n\n ranges.forEach((r) => {\n if (r.hi) {\n this.flushMessageRange(r.low, r.hi);\n } else {\n this.flushMessage(r.low);\n }\n });\n\n if (this.onData) {\n // Calling with no parameters to indicate the messages were deleted.\n this.onData();\n }\n return ctrl;\n });\n }\n /**\n * Delete all messages. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessagesAll(hardDel) {\n if (!this._maxSeq || this._maxSeq <= 0) {\n // There are no messages to delete.\n return Promise.resolve();\n }\n return this.delMessages([{\n low: 1,\n hi: this._maxSeq + 1,\n _all: true\n }], hardDel);\n }\n\n /**\n * Delete multiple messages defined by their IDs. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {Array.} list - list of seq IDs to delete.\n * @param {boolean=} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessagesList(list, hardDel) {\n // Sort the list in ascending order\n list.sort((a, b) => a - b);\n // Convert the array of IDs to ranges.\n let ranges = list.reduce((out, id) => {\n if (out.length == 0) {\n // First element.\n out.push({\n low: id\n });\n } else {\n let prev = out[out.length - 1];\n if ((!prev.hi && (id != prev.low + 1)) || (id > prev.hi)) {\n // New range.\n out.push({\n low: id\n });\n } else {\n // Expand existing range.\n prev.hi = prev.hi ? Math.max(prev.hi, id + 1) : id + 1;\n }\n }\n return out;\n }, []);\n // Send {del} message, return promise\n return this.delMessages(ranges, hardDel);\n }\n\n /**\n * Delete original message and edited variants. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - original seq ID of the message to delete.\n * @param {boolean=} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n delMessagesEdits(seq, hardDel) {\n const list = [seq];\n this.messageVersions(seq, msg => list.push(msg.seq));\n // Send {del} message, return promise\n return this.delMessagesList(list, hardDel);\n }\n\n /**\n * Delete topic. Requires Owner permission. Wrapper for {@link Tinode#delTopic}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} hard - had-delete topic.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n delTopic(hard) {\n if (this._deleted) {\n // The topic is already deleted at the server, just remove from DB.\n this._gone();\n return Promise.resolve(null);\n }\n\n return this._tinode.delTopic(this.name, hard).then(ctrl => {\n this._deleted = true;\n this._resetSub();\n this._gone();\n return ctrl;\n });\n }\n /**\n * Delete subscription. Requires Share permission. Wrapper for {@link Tinode#delSubscription}.\n * @memberof Tinode.Topic#\n *\n * @param {string} user - ID of the user to remove subscription for.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delSubscription(user) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete subscription in inactive topic\"));\n }\n // Send {del} message, return promise\n return this._tinode.delSubscription(this.name, user).then(ctrl => {\n // Remove the object from the subscription cache;\n delete this._users[user];\n // Notify listeners\n if (this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._users));\n }\n return ctrl;\n });\n }\n /**\n * Send a read/recv notification.\n * @memberof Tinode.Topic#\n *\n * @param {string} what - what notification to send: recv
, read
.\n * @param {number} seq - ID or the message read or received.\n */\n note(what, seq) {\n if (!this._attached) {\n // Cannot sending {note} on an inactive topic\".\n return;\n }\n\n // Update local cache with the new count.\n const user = this._users[this._tinode.getCurrentUserID()];\n let update = false;\n if (user) {\n // Self-subscription is found.\n if (!user[what] || user[what] < seq) {\n user[what] = seq;\n update = true;\n }\n } else {\n // Self-subscription is not found.\n update = (this[what] | 0) < seq;\n }\n\n if (update) {\n // Send notification to the server.\n this._tinode.note(this.name, what, seq);\n // Update locally cached contact with the new count.\n this._updateMyReadRecv(what, seq);\n\n if (this.acs != null && !this.acs.isMuted()) {\n const me = this._tinode.getMeTopic();\n // Sent a notification to 'me' listeners.\n me._refreshContact(what, this);\n }\n }\n }\n\n /**\n * Send a 'recv' receipt. Wrapper for {@link Tinode#noteRecv}.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - ID of the message to aknowledge.\n */\n noteRecv(seq) {\n this.note('recv', seq);\n }\n /**\n * Send a 'read' receipt. Wrapper for {@link Tinode#noteRead}.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - ID of the message to aknowledge or 0/undefined to acknowledge the latest messages.\n */\n noteRead(seq) {\n seq = seq || this._maxSeq;\n if (seq > 0) {\n this.note('read', seq);\n }\n }\n /**\n * Send a key-press notification. Wrapper for {@link Tinode#noteKeyPress}.\n * @memberof Tinode.Topic#\n */\n noteKeyPress() {\n if (this._attached) {\n this._tinode.noteKeyPress(this.name);\n } else {\n this._tinode.logger(\"INFO: Cannot send notification in inactive topic\");\n }\n }\n /**\n * Send a notification than a video or audio message is . Wrapper for {@link Tinode#noteKeyPress}.\n * @memberof Tinode.Topic#\n * @param audioOnly - true if the recording is audio-only, false if it's a video recording.\n */\n noteRecording(audioOnly) {\n if (this._attached) {\n this._tinode.noteKeyPress(this.name, audioOnly ? 'kpa' : 'kpv');\n } else {\n this._tinode.logger(\"INFO: Cannot send notification in inactive topic\");\n }\n }\n\n /**\n * Send a {note what='call'}. Wrapper for {@link Tinode#videoCall}.\n * @memberof Tinode#\n *\n * @param {string} evt - Call event.\n * @param {int} seq - ID of the call message the event pertains to.\n * @param {string} payload - Payload associated with this event (e.g. SDP string).\n *\n * @returns {Promise} Promise (for some call events) which will\n * be resolved/rejected on receiving server reply\n */\n videoCall(evt, seq, payload) {\n if (!this._attached && !['ringing', 'hang-up'].includes(evt)) {\n // Cannot {call} on an inactive topic\".\n return;\n }\n return this._tinode.videoCall(this.name, seq, evt, payload);\n }\n\n // Update cached read/recv/unread counts for the current user.\n _updateMyReadRecv(what, seq, ts) {\n let oldVal, doUpdate = false;\n\n seq = seq | 0;\n this.seq = this.seq | 0;\n this.read = this.read | 0;\n this.recv = this.recv | 0;\n switch (what) {\n case 'recv':\n oldVal = this.recv;\n this.recv = Math.max(this.recv, seq);\n doUpdate = (oldVal != this.recv);\n break;\n case 'read':\n oldVal = this.read;\n this.read = Math.max(this.read, seq);\n doUpdate = (oldVal != this.read);\n break;\n case 'msg':\n oldVal = this.seq;\n this.seq = Math.max(this.seq, seq);\n if (!this.touched || this.touched < ts) {\n this.touched = ts;\n }\n doUpdate = (oldVal != this.seq);\n break;\n }\n\n // Sanity checks.\n if (this.recv < this.read) {\n this.recv = this.read;\n doUpdate = true;\n }\n if (this.seq < this.recv) {\n this.seq = this.recv;\n if (!this.touched || this.touched < ts) {\n this.touched = ts;\n }\n doUpdate = true;\n }\n this.unread = this.seq - this.read;\n return doUpdate;\n }\n /**\n * Get user description from global cache. The user does not need to be a\n * subscriber of this topic.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - ID of the user to fetch.\n * @return {Object} user description or undefined.\n */\n userDesc(uid) {\n // TODO: handle asynchronous requests\n const user = this._cacheGetUser(uid);\n if (user) {\n return user; // Promise.resolve(user)\n }\n }\n /**\n * Get description of the p2p peer from subscription cache.\n * @memberof Tinode.Topic#\n *\n * @return {Object} peer's description or undefined.\n */\n p2pPeerDesc() {\n if (!this.isP2PType()) {\n return undefined;\n }\n return this._users[this.name];\n }\n /**\n * Iterate over cached subscribers. If callback is undefined, use this.onMetaSub.\n * @memberof Tinode.Topic#\n *\n * @param {function} callback - Callback which will receive subscribers one by one.\n * @param {Object=} context - Value of `this` inside the `callback`.\n */\n subscribers(callback, context) {\n const cb = (callback || this.onMetaSub);\n if (cb) {\n for (let idx in this._users) {\n cb.call(context, this._users[idx], idx, this._users);\n }\n }\n }\n /**\n * Get a copy of cached tags.\n * @memberof Tinode.Topic#\n *\n * @return {Array.} a copy of tags\n */\n tags() {\n // Return a copy.\n return this._tags.slice(0);\n }\n /**\n * Get cached subscription for the given user ID.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - id of the user to query for\n * @return user description or undefined.\n */\n subscriber(uid) {\n return this._users[uid];\n }\n /**\n * Iterate over versions of a message: call callback
for each version (excluding original).\n * If callback
is undefined, does nothing.\n * @memberof Tinode.Topic#\n *\n * @param {number} origSeq - seq ID of the original message.\n * @param {Tinode.ForEachCallbackType} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {Object} context - Value of `this` inside the `callback`.\n */\n messageVersions(origSeq, callback, context) {\n if (!callback) {\n // No callback? We are done then.\n return;\n }\n const versions = this._messageVersions[origSeq];\n if (!versions) {\n return;\n }\n versions.forEach(callback, undefined, undefined, context);\n }\n /**\n * Iterate over cached messages: call callback
for each message in the range [sinceIdx, beforeIdx).\n * If callback
is undefined, use this.onData
.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.ForEachCallbackType} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {number} sinceId - Optional seqId to start iterating from (inclusive).\n * @param {number} beforeId - Optional seqId to stop iterating before it is reached (exclusive).\n * @param {Object} context - Value of `this` inside the `callback`.\n */\n messages(callback, sinceId, beforeId, context) {\n const cb = (callback || this.onData);\n if (cb) {\n const startIdx = typeof sinceId == 'number' ? this._messages.find({\n seq: sinceId\n }, true) : undefined;\n const beforeIdx = typeof beforeId == 'number' ? this._messages.find({\n seq: beforeId\n }, true) : undefined;\n if (startIdx != -1 && beforeIdx != -1) {\n // Step 1. Filter out all replacement messages and\n // save displayable messages in a temporary buffer.\n let msgs = [];\n this._messages.forEach((msg, unused1, unused2, i) => {\n if (this._isReplacementMsg(msg)) {\n // Skip replacements.\n return;\n }\n // In case the massage was edited, replace timestamp of the version with the original's timestamp.\n const latest = this.latestMsgVersion(msg.seq) || msg;\n if (!latest._origTs) {\n latest._origTs = latest.ts;\n latest._origSeq = latest.seq;\n latest.ts = msg.ts;\n latest.seq = msg.seq;\n }\n msgs.push({\n data: latest,\n idx: i\n });\n }, startIdx, beforeIdx, {});\n // Step 2. Loop over displayble messages invoking cb on each of them.\n msgs.forEach((val, i) => {\n cb.call(context, val.data,\n (i > 0 ? msgs[i - 1].data : undefined),\n (i < msgs.length - 1 ? msgs[i + 1].data : undefined), val.idx);\n });\n }\n }\n }\n /**\n * Get the message from cache by seq
.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - message seqId to search for.\n * @returns {Object} the message with the given seq
or undefined
, if no such message is found.\n */\n findMessage(seq) {\n const idx = this._messages.find({\n seq: seq\n });\n if (idx >= 0) {\n return this._messages.getAt(idx);\n }\n return undefined;\n }\n /**\n * Get the most recent message from cache. This method counts all messages, including deleted ranges.\n * @memberof Tinode.Topic#\n *\n * @returns {Object} the most recent cached message or undefined
, if no messages are cached.\n */\n latestMessage() {\n return this._messages.getLast();\n }\n /**\n * Get the latest version for message.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - original seq ID of the message.\n * @returns {Object} the latest version of the message or null if message not found.\n */\n latestMsgVersion(seq) {\n const versions = this._messageVersions[seq];\n return versions ? versions.getLast() : null;\n }\n /**\n * Get the maximum cached seq ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the greatest seq ID in cache.\n */\n maxMsgSeq() {\n return this._maxSeq;\n }\n /**\n * Get the minimum cached seq ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the smallest seq ID in cache or 0.\n */\n minMsgSeq() {\n return this._minSeq;\n }\n /**\n * Get the maximum deletion ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the greatest deletion ID.\n */\n maxClearId() {\n return this._maxDel;\n }\n /**\n * Get the number of messages in the cache.\n * @memberof Tinode.Topic#\n *\n * @returns {number} count of cached messages.\n */\n messageCount() {\n return this._messages.length();\n }\n /**\n * Iterate over cached unsent messages. Wraps {@link Tinode.Topic#messages}.\n * @memberof Tinode.Topic#\n *\n * @param {function} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {Object} context - Value of this
inside the callback
.\n */\n queuedMessages(callback, context) {\n if (!callback) {\n throw new Error(\"Callback must be provided\");\n }\n this.messages(callback, Const.LOCAL_SEQID, undefined, context);\n }\n /**\n * Get the number of topic subscribers who marked this message as either recv or read\n * Current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {string} what - what action to consider: received \"recv\"
or read \"read\"
.\n * @param {number} seq - ID or the message read or received.\n *\n * @returns {number} the number of subscribers who marked the message with the given ID as read or received.\n */\n msgReceiptCount(what, seq) {\n let count = 0;\n if (seq > 0) {\n const me = this._tinode.getCurrentUserID();\n for (let idx in this._users) {\n const user = this._users[idx];\n if (user.user !== me && user[what] >= seq) {\n count++;\n }\n }\n }\n return count;\n }\n /**\n * Get the number of topic subscribers who marked this message (and all older messages) as read.\n * The current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - message id to check.\n * @returns {number} number of subscribers who claim to have received the message.\n */\n msgReadCount(seq) {\n return this.msgReceiptCount('read', seq);\n }\n /**\n * Get the number of topic subscribers who marked this message (and all older messages) as received.\n * The current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - Message id to check.\n * @returns {number} Number of subscribers who claim to have received the message.\n */\n msgRecvCount(seq) {\n return this.msgReceiptCount('recv', seq);\n }\n /**\n * Check if cached message IDs indicate that the server may have more messages.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} newer - if true
, check for newer messages only.\n */\n msgHasMoreMessages(newer) {\n return newer ? this.seq > this._maxSeq :\n // _minSeq could be more than 1, but earlier messages could have been deleted.\n (this._minSeq > 1 && !this._noEarlierMsgs);\n }\n /**\n * Check if the given seq Id is id of the most recent message.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to check\n */\n isNewMessage(seqId) {\n return this._maxSeq <= seqId;\n }\n /**\n * Remove one message from local cache.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to remove from cache.\n * @returns {Message} removed message or undefined if such message was not found.\n */\n flushMessage(seqId) {\n const idx = this._messages.find({\n seq: seqId\n });\n delete this._messageVersions[seqId];\n if (idx >= 0) {\n this._tinode._db.remMessages(this.name, seqId);\n return this._messages.delAt(idx);\n }\n return undefined;\n }\n /**\n * Remove a range of messages from the local cache.\n * @memberof Tinode.Topic#\n *\n * @param {number} fromId seq ID of the first message to remove (inclusive).\n * @param {number} untilId seqID of the last message to remove (exclusive).\n *\n * @returns {Message[]} array of removed messages (could be empty).\n */\n flushMessageRange(fromId, untilId) {\n // Remove range from persistent cache.\n this._tinode._db.remMessages(this.name, fromId, untilId);\n\n // Remove all versions keyed by IDs in the range.\n for (let i = fromId; i < untilId; i++) {\n delete this._messageVersions[i];\n }\n\n // start, end: find insertion points (nearest == true).\n const since = this._messages.find({\n seq: fromId\n }, true);\n return since >= 0 ? this._messages.delRange(since, this._messages.find({\n seq: untilId\n }, true)) : [];\n }\n /**\n * Update message's seqId.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub message object.\n * @param {number} newSeqId new seq id for pub.\n */\n swapMessageId(pub, newSeqId) {\n const idx = this._messages.find(pub);\n const numMessages = this._messages.length();\n if (0 <= idx && idx < numMessages) {\n // Remove message with the old seq ID.\n this._messages.delAt(idx);\n this._tinode._db.remMessages(this.name, pub.seq);\n // Add message with the new seq ID.\n pub.seq = newSeqId;\n this._messages.put(pub);\n this._tinode._db.addMessage(pub);\n }\n }\n /**\n * Attempt to stop message from being sent.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to stop sending and remove from cache.\n *\n * @returns {boolean} true
if message was cancelled, false
otherwise.\n */\n cancelSend(seqId) {\n const idx = this._messages.find({\n seq: seqId\n });\n if (idx >= 0) {\n const msg = this._messages.getAt(idx);\n const status = this.msgStatus(msg);\n if (status == Const.MESSAGE_STATUS_QUEUED ||\n status == Const.MESSAGE_STATUS_FAILED ||\n status == Const.MESSAGE_STATUS_FATAL) {\n this._tinode._db.remMessages(this.name, seqId);\n msg._cancelled = true;\n this._messages.delAt(idx);\n if (this.onData) {\n // Calling with no parameters to indicate the message was deleted.\n this.onData();\n }\n return true;\n }\n }\n return false;\n }\n /**\n * Get type of the topic: me, p2p, grp, fnd...\n * @memberof Tinode.Topic#\n *\n * @returns {string} One of 'me', 'p2p', 'grp', 'fnd', 'sys' or undefined
.\n */\n getType() {\n return Topic.topicType(this.name);\n }\n /**\n * Get current user's access mode of the topic.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.AccessMode} - user's access mode\n */\n getAccessMode() {\n return this.acs;\n }\n /**\n * Set current user's access mode of the topic.\n * @memberof Tinode.Topic#\n *\n * @param {AccessMode | Object} acs - access mode to set.\n */\n setAccessMode(acs) {\n return this.acs = new AccessMode(acs);\n }\n /**\n * Get topic's default access mode.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.DefAcs} - access mode, such as {auth: `RWP`, anon: `N`}.\n */\n getDefaultAccess() {\n return this.defacs;\n }\n /**\n * Initialize new meta {@link Tinode.GetQuery} builder. The query is attched to the current topic.\n * It will not work correctly if used with a different topic.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.MetaGetBuilder} query attached to the current topic.\n */\n startMetaQuery() {\n return new MetaGetBuilder(this);\n }\n /**\n * Check if topic is archived, i.e. private.arch == true.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is archived, false
otherwise.\n */\n isArchived() {\n return this.private && !!this.private.arch;\n }\n /**\n * Check if topic is a 'me' topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a 'me' topic, false
otherwise.\n */\n isMeType() {\n return Topic.isMeTopicName(this.name);\n }\n /**\n * Check if topic is a channel.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a channel, false
otherwise.\n */\n isChannelType() {\n return Topic.isChannelTopicName(this.name);\n }\n /**\n * Check if topic is a group topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a group, false
otherwise.\n */\n isGroupType() {\n return Topic.isGroupTopicName(this.name);\n }\n /**\n * Check if topic is a p2p topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a p2p topic, false
otherwise.\n */\n isP2PType() {\n return Topic.isP2PTopicName(this.name);\n }\n /**\n * Check if topic is a communication topic, i.e. a group or p2p topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a p2p or group topic, false
otherwise.\n */\n isCommType() {\n return Topic.isCommTopicName(this.name);\n }\n /**\n * Get status (queued, sent, received etc) of a given message in the context\n * of this topic.\n * @memberof Tinode.Topic#\n *\n * @param {Message} msg - message to check for status.\n * @param {boolean} upd - update chached message status.\n *\n * @returns message status constant.\n */\n msgStatus(msg, upd) {\n let status = Const.MESSAGE_STATUS_NONE;\n if (this._tinode.isMe(msg.from)) {\n if (msg._sending) {\n status = Const.MESSAGE_STATUS_SENDING;\n } else if (msg._fatal || msg._cancelled) {\n status = Const.MESSAGE_STATUS_FATAL;\n } else if (msg._failed) {\n status = Const.MESSAGE_STATUS_FAILED;\n } else if (msg.seq >= Const.LOCAL_SEQID) {\n status = Const.MESSAGE_STATUS_QUEUED;\n } else if (this.msgReadCount(msg.seq) > 0) {\n status = Const.MESSAGE_STATUS_READ;\n } else if (this.msgRecvCount(msg.seq) > 0) {\n status = Const.MESSAGE_STATUS_RECEIVED;\n } else if (msg.seq > 0) {\n status = Const.MESSAGE_STATUS_SENT;\n }\n } else {\n status = Const.MESSAGE_STATUS_TO_ME;\n }\n\n if (upd && msg._status != status) {\n msg._status = status;\n this._tinode._db.updMessageStatus(this.name, msg.seq, status);\n }\n\n return status;\n }\n\n // Returns true if pub is meant to replace another message (e.g. original message was edited).\n _isReplacementMsg(pub) {\n return pub.head && pub.head.replace;\n }\n\n // If msg is a replacement for another message, save the msg in the message versions cache\n // as a newer version for the message it's supposed to replace.\n _maybeUpdateMessageVersionsCache(msg) {\n if (!this._isReplacementMsg(msg)) {\n // Check if this message is the original in the chain of edits and if so\n // ensure all version have the same sender.\n if (this._messageVersions[msg.seq]) {\n // Remove versions with different 'from'.\n this._messageVersions[msg.seq].filter(version => version.from == msg.from);\n if (this._messageVersions[msg.seq].isEmpty()) {\n delete this._messageVersions[msg.seq];\n }\n }\n return;\n }\n\n const targetSeq = parseInt(msg.head.replace.split(':')[1]);\n if (targetSeq > msg.seq) {\n // Substitutes are supposed to have higher seq ids.\n return;\n }\n const targetMsg = this.findMessage(targetSeq);\n if (targetMsg && targetMsg.from != msg.from) {\n // Substitute cannot change the sender.\n return;\n }\n const versions = this._messageVersions[targetSeq] || new CBuffer((a, b) => {\n return a.seq - b.seq;\n }, true);\n versions.put(msg);\n this._messageVersions[targetSeq] = versions;\n }\n\n // Process data message\n _routeData(data) {\n if (data.content) {\n if (!this.touched || this.touched < data.ts) {\n this.touched = data.ts;\n this._tinode._db.updTopic(this);\n }\n }\n\n if (data.seq > this._maxSeq) {\n this._maxSeq = data.seq;\n this.msgStatus(data, true);\n // Ackn receiving the message.\n clearTimeout(this._recvNotificationTimer);\n this._recvNotificationTimer = setTimeout(_ => {\n this._recvNotificationTimer = null;\n this.noteRecv(this._maxSeq);\n }, Const.RECV_TIMEOUT);\n }\n\n if (data.seq < this._minSeq || this._minSeq == 0) {\n this._minSeq = data.seq;\n }\n\n const outgoing = ((!this.isChannelType() && !data.from) || this._tinode.isMe(data.from));\n\n if (data.head && data.head.webrtc && data.head.mime == Drafty.getContentType() && data.content) {\n // Rewrite VC body with info from the headers.\n data.content = Drafty.updateVideoCall(data.content, {\n state: data.head.webrtc,\n duration: data.head['webrtc-duration'],\n incoming: !outgoing,\n });\n }\n\n if (!data._noForwarding) {\n this._messages.put(data);\n this._tinode._db.addMessage(data);\n this._maybeUpdateMessageVersionsCache(data);\n }\n\n if (this.onData) {\n this.onData(data);\n }\n\n // Update locally cached contact with the new message count.\n const what = outgoing ? 'read' : 'msg';\n this._updateMyReadRecv(what, data.seq, data.ts);\n\n if (!outgoing && data.from) {\n // Mark messages as read by the sender.\n this._routeInfo({\n what: 'read',\n from: data.from,\n seq: data.seq,\n _noForwarding: true\n });\n }\n\n // Notify 'me' listeners of the change.\n this._tinode.getMeTopic()._refreshContact(what, this);\n }\n\n // Process metadata message\n _routeMeta(meta) {\n if (meta.desc) {\n this._processMetaDesc(meta.desc);\n }\n if (meta.sub && meta.sub.length > 0) {\n this._processMetaSubs(meta.sub);\n }\n if (meta.del) {\n this._processDelMessages(meta.del.clear, meta.del.delseq);\n }\n if (meta.tags) {\n this._processMetaTags(meta.tags);\n }\n if (meta.cred) {\n this._processMetaCreds(meta.cred);\n }\n if (this.onMeta) {\n this.onMeta(meta);\n }\n }\n // Process presence change message\n _routePres(pres) {\n let user, uid;\n switch (pres.what) {\n case 'del':\n // Delete cached messages.\n this._processDelMessages(pres.clear, pres.delseq);\n break;\n case 'on':\n case 'off':\n // Update online status of a subscription.\n user = this._users[pres.src];\n if (user) {\n user.online = pres.what == 'on';\n } else {\n this._tinode.logger(\"WARNING: Presence update for an unknown user\", this.name, pres.src);\n }\n break;\n case 'term':\n // Attachment to topic is terminated probably due to cluster rehashing.\n this._resetSub();\n break;\n case 'upd':\n // A topic subscriber has updated his description.\n // Issue {get sub} only if the current user has no p2p topics with the updated user (p2p name is not in cache).\n // Otherwise 'me' will issue a {get desc} request.\n if (pres.src && !this._tinode.isTopicCached(pres.src)) {\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n }\n break;\n case 'acs':\n uid = pres.src || this._tinode.getCurrentUserID();\n user = this._users[uid];\n if (!user) {\n // Update for an unknown user: notification of a new subscription.\n const acs = new AccessMode().updateAll(pres.dacs);\n if (acs && acs.mode != AccessMode._NONE) {\n user = this._cacheGetUser(uid);\n if (!user) {\n user = {\n user: uid,\n acs: acs\n };\n this.getMeta(this.startMetaQuery().withOneSub(undefined, uid).build());\n } else {\n user.acs = acs;\n }\n user.updated = new Date();\n this._processMetaSubs([user]);\n }\n } else {\n // Known user\n user.acs.updateAll(pres.dacs);\n // Update user's access mode.\n this._processMetaSubs([{\n user: uid,\n updated: new Date(),\n acs: user.acs\n }]);\n }\n break;\n default:\n this._tinode.logger(\"INFO: Ignored presence update\", pres.what);\n }\n\n if (this.onPres) {\n this.onPres(pres);\n }\n }\n // Process {info} message\n _routeInfo(info) {\n switch (info.what) {\n case 'recv':\n case 'read':\n const user = this._users[info.from];\n if (user) {\n user[info.what] = info.seq;\n if (user.recv < user.read) {\n user.recv = user.read;\n }\n }\n const msg = this.latestMessage();\n if (msg) {\n this.msgStatus(msg, true);\n }\n\n // If this is an update from the current user, update the cache with the new count.\n if (this._tinode.isMe(info.from) && !info._noForwarding) {\n this._updateMyReadRecv(info.what, info.seq);\n }\n\n // Notify 'me' listener of the status change.\n this._tinode.getMeTopic()._refreshContact(info.what, this);\n break;\n case 'kp':\n case 'kpa':\n case 'kpv':\n // Typing or audio/video recording notification. Do nothing.\n break;\n case 'call':\n // Do nothing here.\n break;\n default:\n this._tinode.logger(\"INFO: Ignored info update\", info.what);\n }\n\n if (this.onInfo) {\n this.onInfo(info);\n }\n }\n // Called by Tinode when meta.desc packet is received.\n // Called by 'me' topic on contact update (desc._noForwarding is true).\n _processMetaDesc(desc) {\n if (this.isP2PType()) {\n // Synthetic desc may include defacs for p2p topics which is useless.\n // Remove it.\n delete desc.defacs;\n\n // Update to p2p desc is the same as user update. Update cached user.\n this._tinode._db.updUser(this.name, desc.public);\n }\n\n // Copy parameters from desc object to this topic.\n mergeObj(this, desc);\n // Update persistent cache.\n this._tinode._db.updTopic(this);\n\n // Notify 'me' listener, if available:\n if (this.name !== Const.TOPIC_ME && !desc._noForwarding) {\n const me = this._tinode.getMeTopic();\n if (me.onMetaSub) {\n me.onMetaSub(this);\n }\n if (me.onSubsUpdated) {\n me.onSubsUpdated([this.name], 1);\n }\n }\n\n if (this.onMetaDesc) {\n this.onMetaDesc(this);\n }\n }\n // Called by Tinode when meta.sub is recived or in response to received\n // {ctrl} after setMeta-sub.\n _processMetaSubs(subs) {\n for (let idx in subs) {\n const sub = subs[idx];\n\n // Fill defaults.\n sub.online = !!sub.online;\n // Update timestamp of the most recent subscription update.\n this._lastSubsUpdate = new Date(Math.max(this._lastSubsUpdate, sub.updated));\n\n let user = null;\n if (!sub.deleted) {\n // If this is a change to user's own permissions, update them in topic too.\n // Desc will update 'me' topic.\n if (this._tinode.isMe(sub.user) && sub.acs) {\n this._processMetaDesc({\n updated: sub.updated,\n touched: sub.touched,\n acs: sub.acs\n });\n }\n user = this._updateCachedUser(sub.user, sub);\n } else {\n // Subscription is deleted, remove it from topic (but leave in Users cache)\n delete this._users[sub.user];\n user = sub;\n }\n\n if (this.onMetaSub) {\n this.onMetaSub(user);\n }\n }\n\n if (this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._users));\n }\n }\n // Called by Tinode when meta.tags is recived.\n _processMetaTags(tags) {\n if (tags.length == 1 && tags[0] == Const.DEL_CHAR) {\n tags = [];\n }\n this._tags = tags;\n if (this.onTagsUpdated) {\n this.onTagsUpdated(tags);\n }\n }\n // Do nothing for topics other than 'me'\n _processMetaCreds(creds) {}\n // Delete cached messages and update cached transaction IDs\n _processDelMessages(clear, delseq) {\n this._maxDel = Math.max(clear, this._maxDel);\n this.clear = Math.max(clear, this.clear);\n const topic = this;\n let count = 0;\n if (Array.isArray(delseq)) {\n delseq.forEach(function(range) {\n if (!range.hi) {\n count++;\n topic.flushMessage(range.low);\n } else {\n for (let i = range.low; i < range.hi; i++) {\n count++;\n topic.flushMessage(i);\n }\n }\n });\n }\n\n if (count > 0) {\n // this._updateDeletedRanges();\n\n if (this.onData) {\n this.onData();\n }\n }\n }\n // Topic is informed that the entire response to {get what=data} has been received.\n _allMessagesReceived(count) {\n\n if (this.onAllMessagesReceived) {\n this.onAllMessagesReceived(count);\n }\n }\n // Reset subscribed state\n _resetSub() {\n this._attached = false;\n }\n // This topic is either deleted or unsubscribed from.\n _gone() {\n this._messages.reset();\n this._tinode._db.remMessages(this.name);\n this._users = {};\n this.acs = new AccessMode(null);\n this.private = null;\n this.public = null;\n this.trusted = null;\n this._maxSeq = 0;\n this._minSeq = 0;\n this._attached = false;\n\n const me = this._tinode.getMeTopic();\n if (me) {\n me._routePres({\n _noForwarding: true,\n what: 'gone',\n topic: Const.TOPIC_ME,\n src: this.name\n });\n }\n if (this.onDeleteTopic) {\n this.onDeleteTopic();\n }\n }\n // Update global user cache and local subscribers cache.\n // Don't call this method for non-subscribers.\n _updateCachedUser(uid, obj) {\n // Fetch user object from the global cache.\n // This is a clone of the stored object\n let cached = this._cacheGetUser(uid);\n cached = mergeObj(cached || {}, obj);\n // Save to global cache\n this._cachePutUser(uid, cached);\n // Save to the list of topic subsribers.\n return mergeToCache(this._users, uid, cached);\n }\n // Get local seqId for a queued message.\n _getQueuedSeqId() {\n return this._queuedSeqId++;\n }\n\n // Load most recent messages from persistent cache.\n _loadMessages(db, params) {\n const {\n since,\n before,\n limit\n } = params || {};\n return db.readMessages(this.name, {\n since: since,\n before: before,\n limit: limit || Const.DEFAULT_MESSAGES_PAGE\n })\n .then(msgs => {\n msgs.forEach((data) => {\n if (data.seq > this._maxSeq) {\n this._maxSeq = data.seq;\n }\n if (data.seq < this._minSeq || this._minSeq == 0) {\n this._minSeq = data.seq;\n }\n this._messages.put(data);\n this._maybeUpdateMessageVersionsCache(data);\n });\n return msgs.length;\n });\n }\n // Push or {pres}: message received.\n _updateReceived(seq, act) {\n this.touched = new Date();\n this.seq = seq | 0;\n // Check if message is sent by the current user. If so it's been read already.\n if (!act || this._tinode.isMe(act)) {\n this.read = this.read ? Math.max(this.read, this.seq) : this.seq;\n this.recv = this.recv ? Math.max(this.read, this.recv) : this.read;\n }\n this.unread = this.seq - (this.read | 0);\n this._tinode._db.updTopic(this);\n }\n}\n\n/**\n * @class TopicMe - special case of {@link Tinode.Topic} for\n * managing data of the current user, including contact list.\n * @extends Tinode.Topic\n * @memberof Tinode\n *\n * @param {TopicMe.Callbacks} callbacks - Callbacks to receive various events.\n */\nexport class TopicMe extends Topic {\n onContactUpdate;\n\n constructor(callbacks) {\n super(Const.TOPIC_ME, callbacks);\n\n // me-specific callbacks\n if (callbacks) {\n this.onContactUpdate = callbacks.onContactUpdate;\n }\n }\n\n // Override the original Topic._processMetaDesc.\n _processMetaDesc(desc) {\n // Check if online contacts need to be turned off because P permission was removed.\n const turnOff = (desc.acs && !desc.acs.isPresencer()) && (this.acs && this.acs.isPresencer());\n\n // Copy parameters from desc object to this topic.\n mergeObj(this, desc);\n this._tinode._db.updTopic(this);\n // Update current user's record in the global cache.\n this._updateCachedUser(this._tinode._myUID, desc);\n\n // 'P' permission was removed. All topics are offline now.\n if (turnOff) {\n this._tinode.mapTopics((cont) => {\n if (cont.online) {\n cont.online = false;\n cont.seen = Object.assign(cont.seen || {}, {\n when: new Date()\n });\n this._refreshContact('off', cont);\n }\n });\n }\n\n if (this.onMetaDesc) {\n this.onMetaDesc(this);\n }\n }\n\n // Override the original Topic._processMetaSubs\n _processMetaSubs(subs) {\n let updateCount = 0;\n subs.forEach((sub) => {\n const topicName = sub.topic;\n // Don't show 'me' and 'fnd' topics in the list of contacts.\n if (topicName == Const.TOPIC_FND || topicName == Const.TOPIC_ME) {\n return;\n }\n sub.online = !!sub.online;\n\n let cont = null;\n if (sub.deleted) {\n cont = sub;\n this._tinode.cacheRemTopic(topicName);\n this._tinode._db.remTopic(topicName);\n } else {\n // Ensure the values are defined and are integers.\n if (typeof sub.seq != 'undefined') {\n sub.seq = sub.seq | 0;\n sub.recv = sub.recv | 0;\n sub.read = sub.read | 0;\n sub.unread = sub.seq - sub.read;\n }\n\n const topic = this._tinode.getTopic(topicName);\n if (topic._new) {\n delete topic._new;\n }\n\n cont = mergeObj(topic, sub);\n this._tinode._db.updTopic(cont);\n\n if (Topic.isP2PTopicName(topicName)) {\n this._cachePutUser(topicName, cont);\n this._tinode._db.updUser(topicName, cont.public);\n }\n // Notify topic of the update if it's an external update.\n if (!sub._noForwarding && topic) {\n sub._noForwarding = true;\n topic._processMetaDesc(sub);\n }\n }\n\n updateCount++;\n\n if (this.onMetaSub) {\n this.onMetaSub(cont);\n }\n });\n\n if (this.onSubsUpdated && updateCount > 0) {\n const keys = [];\n subs.forEach((s) => {\n keys.push(s.topic);\n });\n this.onSubsUpdated(keys, updateCount);\n }\n }\n\n // Called by Tinode when meta.sub is recived.\n _processMetaCreds(creds, upd) {\n if (creds.length == 1 && creds[0] == Const.DEL_CHAR) {\n creds = [];\n }\n if (upd) {\n creds.forEach((cr) => {\n if (cr.val) {\n // Adding a credential.\n let idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && el.val == cr.val;\n });\n if (idx < 0) {\n // Not found.\n if (!cr.done) {\n // Unconfirmed credential replaces previous unconfirmed credential of the same method.\n idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && !el.done;\n });\n if (idx >= 0) {\n // Remove previous unconfirmed credential.\n this._credentials.splice(idx, 1);\n }\n }\n this._credentials.push(cr);\n } else {\n // Found. Maybe change 'done' status.\n this._credentials[idx].done = cr.done;\n }\n } else if (cr.resp) {\n // Handle credential confirmation.\n const idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && !el.done;\n });\n if (idx >= 0) {\n this._credentials[idx].done = true;\n }\n }\n });\n } else {\n this._credentials = creds;\n }\n if (this.onCredsUpdated) {\n this.onCredsUpdated(this._credentials);\n }\n }\n\n // Process presence change message\n _routePres(pres) {\n if (pres.what == 'term') {\n // The 'me' topic itself is detached. Mark as unsubscribed.\n this._resetSub();\n return;\n }\n\n if (pres.what == 'upd' && pres.src == Const.TOPIC_ME) {\n // Update to me's description. Request updated value.\n this.getMeta(this.startMetaQuery().withDesc().build());\n return;\n }\n\n const cont = this._tinode.cacheGetTopic(pres.src);\n if (cont) {\n switch (pres.what) {\n case 'on': // topic came online\n cont.online = true;\n break;\n case 'off': // topic went offline\n if (cont.online) {\n cont.online = false;\n cont.seen = Object.assign(cont.seen || {}, {\n when: new Date()\n });\n }\n break;\n case 'msg': // new message received\n cont._updateReceived(pres.seq, pres.act);\n break;\n case 'upd': // desc updated\n // Request updated subscription.\n this.getMeta(this.startMetaQuery().withLaterOneSub(pres.src).build());\n break;\n case 'acs': // access mode changed\n // If 'tgt' is not set then this is an update to the permissions of the current user.\n // Otherwise it's an update to group topic subscriber permissions while the topic is offline.\n // Just gnore it then.\n if (!pres.tgt) {\n if (cont.acs) {\n cont.acs.updateAll(pres.dacs);\n } else {\n cont.acs = new AccessMode().updateAll(pres.dacs);\n }\n cont.touched = new Date();\n }\n break;\n case 'ua':\n // user agent changed.\n cont.seen = {\n when: new Date(),\n ua: pres.ua\n };\n break;\n case 'recv':\n // user's other session marked some messges as received.\n pres.seq = pres.seq | 0;\n cont.recv = cont.recv ? Math.max(cont.recv, pres.seq) : pres.seq;\n break;\n case 'read':\n // user's other session marked some messages as read.\n pres.seq = pres.seq | 0;\n cont.read = cont.read ? Math.max(cont.read, pres.seq) : pres.seq;\n cont.recv = cont.recv ? Math.max(cont.read, cont.recv) : cont.recv;\n cont.unread = cont.seq - cont.read;\n break;\n case 'gone':\n // topic deleted or unsubscribed from.\n this._tinode.cacheRemTopic(pres.src);\n if (!cont._deleted) {\n cont._deleted = true;\n cont._attached = false;\n this._tinode._db.markTopicAsDeleted(pres.src, true);\n } else {\n this._tinode._db.remTopic(pres.src);\n }\n break;\n case 'del':\n // Update topic.del value.\n break;\n default:\n this._tinode.logger(\"INFO: Unsupported presence update in 'me'\", pres.what);\n }\n\n this._refreshContact(pres.what, cont);\n } else {\n if (pres.what == 'acs') {\n // New subscriptions and deleted/banned subscriptions have full\n // access mode (no + or - in the dacs string). Changes to known subscriptions are sent as\n // deltas, but they should not happen here.\n const acs = new AccessMode(pres.dacs);\n if (!acs || acs.mode == AccessMode._INVALID) {\n this._tinode.logger(\"ERROR: Invalid access mode update\", pres.src, pres.dacs);\n return;\n } else if (acs.mode == AccessMode._NONE) {\n this._tinode.logger(\"WARNING: Removing non-existent subscription\", pres.src, pres.dacs);\n return;\n } else {\n // New subscription. Send request for the full description.\n // Using .withOneSub (not .withLaterOneSub) to make sure IfModifiedSince is not set.\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n // Create a dummy entry to catch online status update.\n const dummy = this._tinode.getTopic(pres.src);\n dummy.online = false;\n dummy.acs = acs;\n this._tinode._db.updTopic(dummy);\n }\n } else if (pres.what == 'tags') {\n this.getMeta(this.startMetaQuery().withTags().build());\n } else if (pres.what == 'msg') {\n // Message received for un unknown (previously deleted) topic.\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n // Create an entry to catch updates and messages.\n const dummy = this._tinode.getTopic(pres.src);\n dummy._deleted = false;\n this._tinode._db.updTopic(dummy);\n }\n\n this._refreshContact(pres.what, cont);\n }\n\n if (this.onPres) {\n this.onPres(pres);\n }\n }\n\n // Contact is updated, execute callbacks.\n _refreshContact(what, cont) {\n if (this.onContactUpdate) {\n this.onContactUpdate(what, cont);\n }\n }\n\n /**\n * Publishing to TopicMe is not supported. {@link Topic#publish} is overriden and thows an {Error} if called.\n * @memberof Tinode.TopicMe#\n * @throws {Error} Always throws an error.\n */\n publish() {\n return Promise.reject(new Error(\"Publishing to 'me' is not supported\"));\n }\n\n /**\n * Delete validation credential.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} topic - Name of the topic to delete\n * @param {string} user - User ID to remove.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCredential(method, value) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete credential in inactive 'me' topic\"));\n }\n // Send {del} message, return promise\n return this._tinode.delCredential(method, value).then(ctrl => {\n // Remove deleted credential from the cache.\n const index = this._credentials.findIndex((el) => {\n return el.meth == method && el.val == value;\n });\n if (index > -1) {\n this._credentials.splice(index, 1);\n }\n // Notify listeners\n if (this.onCredsUpdated) {\n this.onCredsUpdated(this._credentials);\n }\n return ctrl;\n });\n }\n\n /**\n * @callback contactFilter\n * @param {Object} contact to check for inclusion.\n * @returns {boolean} true
if contact should be processed, false
to exclude it.\n */\n /**\n * Iterate over cached contacts.\n *\n * @function\n * @memberof Tinode.TopicMe#\n * @param {TopicMe.ContactCallback} callback - Callback to call for each contact.\n * @param {contactFilter=} filter - Optionally filter contacts; include all if filter is false-ish, otherwise\n * include those for which filter returns true-ish.\n * @param {Object=} context - Context to use for calling the `callback`, i.e. the value of `this` inside the callback.\n */\n contacts(callback, filter, context) {\n this._tinode.mapTopics((c, idx) => {\n if (c.isCommType() && (!filter || filter(c))) {\n callback.call(context, c, idx);\n }\n });\n }\n\n /**\n * Get a contact from cache.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to get, either a UID (for p2p topics) or a topic name.\n * @returns {Tinode.Contact} - Contact or `undefined`.\n */\n getContact(name) {\n return this._tinode.cacheGetTopic(name);\n }\n\n /**\n * Get access mode of a given contact from cache.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to get access mode for, either a UID (for p2p topics)\n * or a topic name; if missing, access mode for the 'me' topic itself.\n * @returns {string} - access mode, such as `RWP`.\n */\n getAccessMode(name) {\n if (name) {\n const cont = this._tinode.cacheGetTopic(name);\n return cont ? cont.acs : null;\n }\n return this.acs;\n }\n\n /**\n * Check if contact is archived, i.e. contact.private.arch == true.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to check archived status, either a UID (for p2p topics) or a topic name.\n * @returns {boolean} - true if contact is archived, false otherwise.\n */\n isArchived(name) {\n const cont = this._tinode.cacheGetTopic(name);\n return cont && cont.private && !!cont.private.arch;\n }\n\n /**\n * @typedef Tinode.Credential\n * @memberof Tinode\n * @type Object\n * @property {string} meth - validation method such as 'email' or 'tel'.\n * @property {string} val - credential value, i.e. 'jdoe@example.com' or '+17025551234'\n * @property {boolean} done - true if credential is validated.\n */\n /**\n * Get the user's credentials: email, phone, etc.\n * @memberof Tinode.TopicMe#\n *\n * @returns {Tinode.Credential[]} - array of credentials.\n */\n getCredentials() {\n return this._credentials;\n }\n}\n\n/**\n * Special case of {@link Tinode.Topic} for searching for contacts and group topics\n * @extends Tinode.Topic\n *\n */\nexport class TopicFnd extends Topic {\n // List of users and topics uid or topic_name -> Contact object)\n _contacts = {};\n\n /**\n * Create TopicFnd.\n *\n * @param {TopicFnd.Callbacks} callbacks - Callbacks to receive various events.\n */\n constructor(callbacks) {\n super(Const.TOPIC_FND, callbacks);\n }\n\n // Override the original Topic._processMetaSubs\n _processMetaSubs(subs) {\n let updateCount = Object.getOwnPropertyNames(this._contacts).length;\n // Reset contact list.\n this._contacts = {};\n for (let idx in subs) {\n let sub = subs[idx];\n const indexBy = sub.topic ? sub.topic : sub.user;\n\n sub = mergeToCache(this._contacts, indexBy, sub);\n updateCount++;\n\n if (this.onMetaSub) {\n this.onMetaSub(sub);\n }\n }\n\n if (updateCount > 0 && this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._contacts));\n }\n }\n\n /**\n * Publishing to TopicFnd is not supported. {@link Topic#publish} is overriden and thows an {Error} if called.\n * @memberof Tinode.TopicFnd#\n * @throws {Error} Always throws an error.\n */\n publish() {\n return Promise.reject(new Error(\"Publishing to 'fnd' is not supported\"));\n }\n\n /**\n * setMeta to TopicFnd resets contact list in addition to sending the message.\n * @memberof Tinode.TopicFnd#\n * @param {Tinode.SetParams} params parameters to update.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n setMeta(params) {\n return Object.getPrototypeOf(TopicFnd.prototype).setMeta.call(this, params).then(_ => {\n if (Object.keys(this._contacts).length > 0) {\n this._contacts = {};\n if (this.onSubsUpdated) {\n this.onSubsUpdated([]);\n }\n }\n });\n }\n\n /**\n * Iterate over found contacts. If callback is undefined, use {@link this.onMetaSub}.\n * @function\n * @memberof Tinode.TopicFnd#\n * @param {TopicFnd.ContactCallback} callback - Callback to call for each contact.\n * @param {Object} context - Context to use for calling the `callback`, i.e. the value of `this` inside the callback.\n */\n contacts(callback, context) {\n const cb = (callback || this.onMetaSub);\n if (cb) {\n for (let idx in this._contacts) {\n cb.call(context, this._contacts[idx], idx, this._contacts);\n }\n }\n }\n}\n","/**\n * @file Utilities used in multiple places.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport AccessMode from './access-mode.js';\nimport {\n DEL_CHAR\n} from './config.js';\n\n// Attempt to convert date and AccessMode strings to objects.\nexport function jsonParseHelper(key, val) {\n // Try to convert string timestamps with optional milliseconds to Date,\n // e.g. 2015-09-02T01:45:43[.123]Z\n if (typeof val == 'string' && val.length >= 20 && val.length <= 24 && ['ts', 'touched', 'updated', 'created', 'when', 'deleted', 'expires'].includes(key)) {\n const date = new Date(val);\n if (!isNaN(date)) {\n return date;\n }\n } else if (key === 'acs' && typeof val === 'object') {\n return new AccessMode(val);\n }\n return val;\n}\n\n// Checks if URL is a relative url, i.e. has no 'scheme://', including the case of missing scheme '//'.\n// The scheme is expected to be RFC-compliant, e.g. [a-z][a-z0-9+.-]*\n// example.html - ok\n// https:example.com - not ok.\n// http:/example.com - not ok.\n// ' ↲ https://example.com' - not ok. (↲ means carriage return)\nexport function isUrlRelative(url) {\n return url && !/^\\s*([a-z][a-z0-9+.-]*:|\\/\\/)/im.test(url);\n}\n\nfunction isValidDate(d) {\n return (d instanceof Date) && !isNaN(d) && (d.getTime() != 0);\n}\n\n// RFC3339 formater of Date\nexport function rfc3339DateString(d) {\n if (!isValidDate(d)) {\n return undefined;\n }\n\n const pad = function(val, sp) {\n sp = sp || 2;\n return '0'.repeat(sp - ('' + val).length) + val;\n };\n\n const millis = d.getUTCMilliseconds();\n return d.getUTCFullYear() + '-' + pad(d.getUTCMonth() + 1) + '-' + pad(d.getUTCDate()) +\n 'T' + pad(d.getUTCHours()) + ':' + pad(d.getUTCMinutes()) + ':' + pad(d.getUTCSeconds()) +\n (millis ? '.' + pad(millis, 3) : '') + 'Z';\n}\n\n// Recursively merge src's own properties to dst.\n// Ignore properties where ignore[property] is true.\n// Array and Date objects are shallow-copied.\nexport function mergeObj(dst, src, ignore) {\n if (typeof src != 'object') {\n if (src === undefined) {\n return dst;\n }\n if (src === DEL_CHAR) {\n return undefined;\n }\n return src;\n }\n // JS is crazy: typeof null is 'object'.\n if (src === null) {\n return src;\n }\n\n // Handle Date\n if (src instanceof Date && !isNaN(src)) {\n return (!dst || !(dst instanceof Date) || isNaN(dst) || dst < src) ? src : dst;\n }\n\n // Access mode\n if (src instanceof AccessMode) {\n return new AccessMode(src);\n }\n\n // Handle Array\n if (src instanceof Array) {\n return src;\n }\n\n if (!dst || dst === DEL_CHAR) {\n dst = src.constructor();\n }\n\n for (let prop in src) {\n if (src.hasOwnProperty(prop) && (!ignore || !ignore[prop]) && (prop != '_noForwarding')) {\n try {\n dst[prop] = mergeObj(dst[prop], src[prop]);\n } catch (err) {\n // FIXME: probably need to log something here.\n }\n }\n }\n return dst;\n}\n\n// Update object stored in a cache. Returns updated value.\nexport function mergeToCache(cache, key, newval, ignore) {\n cache[key] = mergeObj(cache[key], newval, ignore);\n return cache[key];\n}\n\n// Strips all values from an object of they evaluate to false or if their name starts with '_'.\n// Used on all outgoing object before serialization to string.\nexport function simplify(obj) {\n Object.keys(obj).forEach((key) => {\n if (key[0] == '_') {\n // Strip fields like \"obj._key\".\n delete obj[key];\n } else if (!obj[key]) {\n // Strip fields which evaluate to false.\n delete obj[key];\n } else if (Array.isArray(obj[key]) && obj[key].length == 0) {\n // Strip empty arrays.\n delete obj[key];\n } else if (!obj[key]) {\n // Strip fields which evaluate to false.\n delete obj[key];\n } else if (obj[key] instanceof Date) {\n // Strip invalid or zero date.\n if (!isValidDate(obj[key])) {\n delete obj[key];\n }\n } else if (typeof obj[key] == 'object') {\n simplify(obj[key]);\n // Strip empty objects.\n if (Object.getOwnPropertyNames(obj[key]).length == 0) {\n delete obj[key];\n }\n }\n });\n return obj;\n};\n\n\n// Trim whitespace, strip empty and duplicate elements elements.\n// If the result is an empty array, add a single element \"\\u2421\" (Unicode Del character).\nexport function normalizeArray(arr) {\n let out = [];\n if (Array.isArray(arr)) {\n // Trim, throw away very short and empty tags.\n for (let i = 0, l = arr.length; i < l; i++) {\n let t = arr[i];\n if (t) {\n t = t.trim().toLowerCase();\n if (t.length > 1) {\n out.push(t);\n }\n }\n }\n out.sort().filter(function(item, pos, ary) {\n return !pos || item != ary[pos - 1];\n });\n }\n if (out.length == 0) {\n // Add single tag with a Unicode Del character, otherwise an ampty array\n // is ambiguos. The Del tag will be stripped by the server.\n out.push(DEL_CHAR);\n }\n return out;\n}\n","export const PACKAGE_VERSION = \"0.22.11\";\n","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","/**\n * @module tinode-sdk\n *\n * @copyright 2015-2022 Tinode LLC.\n * @summary Javascript bindings for Tinode.\n * @license Apache 2.0\n * @version 0.20\n *\n * See https://github.com/tinode/webapp for real-life usage.\n *\n * @example\n * \n * \n * \n *\n * \n * ...\n * \n * \n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nimport AccessMode from './access-mode.js';\nimport * as Const from './config.js';\nimport CommError from './comm-error.js';\nimport Connection from './connection.js';\nimport DBCache from './db.js';\nimport Drafty from './drafty.js';\nimport LargeFileHelper from './large-file.js';\nimport MetaGetBuilder from './meta-builder.js';\nimport {\n Topic,\n TopicMe,\n TopicFnd\n} from './topic.js';\n\nimport {\n isUrlRelative,\n jsonParseHelper,\n mergeObj,\n rfc3339DateString,\n simplify\n} from './utils.js';\n\n// Re-export AccessMode\nexport {\n AccessMode\n};\n\nlet WebSocketProvider;\nif (typeof WebSocket != 'undefined') {\n WebSocketProvider = WebSocket;\n}\n\nlet XHRProvider;\nif (typeof XMLHttpRequest != 'undefined') {\n XHRProvider = XMLHttpRequest;\n}\n\nlet IndexedDBProvider;\nif (typeof indexedDB != 'undefined') {\n IndexedDBProvider = indexedDB;\n}\n\n// Re-export Drafty.\nexport {\n Drafty\n}\n\ninitForNonBrowserApp();\n\n// Utility functions\n\n// Polyfill for non-browser context, e.g. NodeJs.\nfunction initForNonBrowserApp() {\n // Tinode requirement in native mode because react native doesn't provide Base64 method\n const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';\n\n if (typeof btoa == 'undefined') {\n global.btoa = function(input = '') {\n let str = input;\n let output = '';\n\n for (let block = 0, charCode, i = 0, map = chars; str.charAt(i | 0) || (map = '=', i % 1); output += map.charAt(63 & block >> 8 - i % 1 * 8)) {\n\n charCode = str.charCodeAt(i += 3 / 4);\n\n if (charCode > 0xFF) {\n throw new Error(\"'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.\");\n }\n block = block << 8 | charCode;\n }\n\n return output;\n };\n }\n\n if (typeof atob == 'undefined') {\n global.atob = function(input = '') {\n let str = input.replace(/=+$/, '');\n let output = '';\n\n if (str.length % 4 == 1) {\n throw new Error(\"'atob' failed: The string to be decoded is not correctly encoded.\");\n }\n for (let bc = 0, bs = 0, buffer, i = 0; buffer = str.charAt(i++);\n\n ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,\n bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0\n ) {\n buffer = chars.indexOf(buffer);\n }\n\n return output;\n };\n }\n\n if (typeof window == 'undefined') {\n global.window = {\n WebSocket: WebSocketProvider,\n XMLHttpRequest: XHRProvider,\n indexedDB: IndexedDBProvider,\n URL: {\n createObjectURL: function() {\n throw new Error(\"Unable to use URL.createObjectURL in a non-browser application\");\n }\n }\n }\n }\n\n Connection.setNetworkProviders(WebSocketProvider, XHRProvider);\n LargeFileHelper.setNetworkProvider(XHRProvider);\n DBCache.setDatabaseProvider(IndexedDBProvider);\n}\n\n// Detect find most useful network transport.\nfunction detectTransport() {\n if (typeof window == 'object') {\n if (window['WebSocket']) {\n return 'ws';\n } else if (window['XMLHttpRequest']) {\n // The browser or node has no websockets, using long polling.\n return 'lp';\n }\n }\n return null;\n}\n\n// btoa replacement. Stock btoa fails on on non-Latin1 strings.\nfunction b64EncodeUnicode(str) {\n // The encodeURIComponent percent-encodes UTF-8 string,\n // then the percent encoding is converted into raw bytes which\n // can be fed into btoa.\n return btoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g,\n function toSolidBytes(match, p1) {\n return String.fromCharCode('0x' + p1);\n }));\n}\n\n// JSON stringify helper - pre-processor for JSON.stringify\nfunction jsonBuildHelper(key, val) {\n if (val instanceof Date) {\n // Convert javascript Date objects to rfc3339 strings\n val = rfc3339DateString(val);\n } else if (val instanceof AccessMode) {\n val = val.jsonHelper();\n } else if (val === undefined || val === null || val === false ||\n (Array.isArray(val) && val.length == 0) ||\n ((typeof val == 'object') && (Object.keys(val).length == 0))) {\n // strip out empty elements while serializing objects to JSON\n return undefined;\n }\n\n return val;\n};\n\n// Trims very long strings (encoded images) to make logged packets more readable.\nfunction jsonLoggerHelper(key, val) {\n if (typeof val == 'string' && val.length > 128) {\n return '<' + val.length + ', bytes: ' + val.substring(0, 12) + '...' + val.substring(val.length - 12) + '>';\n }\n return jsonBuildHelper(key, val);\n};\n\n// Parse browser user agent to extract browser name and version.\nfunction getBrowserInfo(ua, product) {\n ua = ua || '';\n let reactnative = '';\n // Check if this is a ReactNative app.\n if (/reactnative/i.test(product)) {\n reactnative = 'ReactNative; ';\n }\n let result;\n // Remove useless string.\n ua = ua.replace(' (KHTML, like Gecko)', '');\n // Test for WebKit-based browser.\n let m = ua.match(/(AppleWebKit\\/[.\\d]+)/i);\n if (m) {\n // List of common strings, from more useful to less useful.\n // All unknown strings get the highest (-1) priority.\n const priority = ['edg', 'chrome', 'safari', 'mobile', 'version'];\n let tmp = ua.substr(m.index + m[0].length).split(' ');\n let tokens = [];\n let version; // 1.0 in Version/1.0 or undefined;\n // Split string like 'Name/0.0.0' into ['Name', '0.0.0', 3] where the last element is the priority.\n for (let i = 0; i < tmp.length; i++) {\n let m2 = /([\\w.]+)[\\/]([\\.\\d]+)/.exec(tmp[i]);\n if (m2) {\n // Unknown values are highest priority (-1).\n tokens.push([m2[1], m2[2], priority.findIndex((e) => {\n return m2[1].toLowerCase().startsWith(e);\n })]);\n if (m2[1] == 'Version') {\n version = m2[2];\n }\n }\n }\n // Sort by priority: more interesting is earlier than less interesting.\n tokens.sort((a, b) => {\n return a[2] - b[2];\n });\n if (tokens.length > 0) {\n // Return the least common browser string and version.\n if (tokens[0][0].toLowerCase().startsWith('edg')) {\n tokens[0][0] = 'Edge';\n } else if (tokens[0][0] == 'OPR') {\n tokens[0][0] = 'Opera';\n } else if (tokens[0][0] == 'Safari' && version) {\n tokens[0][1] = version;\n }\n result = tokens[0][0] + '/' + tokens[0][1];\n } else {\n // Failed to ID the browser. Return the webkit version.\n result = m[1];\n }\n } else if (/firefox/i.test(ua)) {\n m = /Firefox\\/([.\\d]+)/g.exec(ua);\n if (m) {\n result = 'Firefox/' + m[1];\n } else {\n result = 'Firefox/?';\n }\n } else {\n // Neither AppleWebKit nor Firefox. Try the last resort.\n m = /([\\w.]+)\\/([.\\d]+)/.exec(ua);\n if (m) {\n result = m[1] + '/' + m[2];\n } else {\n m = ua.split(' ');\n result = m[0];\n }\n }\n\n // Shorten the version to one dot 'a.bb.ccc.d -> a.bb' at most.\n m = result.split('/');\n if (m.length > 1) {\n const v = m[1].split('.');\n const minor = v[1] ? '.' + v[1].substr(0, 2) : '';\n result = `${m[0]}/${v[0]}${minor}`;\n }\n return reactnative + result;\n}\n\n/**\n * The main class for interacting with Tinode server.\n */\nexport class Tinode {\n _host;\n _secure;\n\n _appName;\n\n // API Key.\n _apiKey;\n\n // Name and version of the browser.\n _browser = '';\n _platform;\n // Hardware\n _hwos = 'undefined';\n _humanLanguage = 'xx';\n\n // Logging to console enabled\n _loggingEnabled = false;\n // When logging, trip long strings (base64-encoded images) for readability\n _trimLongStrings = false;\n // UID of the currently authenticated user.\n _myUID = null;\n // Status of connection: authenticated or not.\n _authenticated = false;\n // Login used in the last successful basic authentication\n _login = null;\n // Token which can be used for login instead of login/password.\n _authToken = null;\n // Counter of received packets\n _inPacketCount = 0;\n // Counter for generating unique message IDs\n _messageId = Math.floor((Math.random() * 0xFFFF) + 0xFFFF);\n // Information about the server, if connected\n _serverInfo = null;\n // Push notification token. Called deviceToken for consistency with the Android SDK.\n _deviceToken = null;\n\n // Cache of pending promises by message id.\n _pendingPromises = {};\n // The Timeout object returned by the reject expired promises setInterval.\n _expirePromises = null;\n\n // Websocket or long polling connection.\n _connection = null;\n\n // Use indexDB for caching topics and messages.\n _persist = false;\n // IndexedDB wrapper object.\n _db = null;\n\n // Tinode's cache of objects\n _cache = {};\n\n /**\n * Create Tinode object.\n *\n * @param {Object} config - configuration parameters.\n * @param {string} config.appName - Name of the calling application to be reported in the User Agent.\n * @param {string} config.host - Host name and optional port number to connect to.\n * @param {string} config.apiKey - API key generated by keygen
.\n * @param {string} config.transport - See {@link Tinode.Connection#transport}.\n * @param {boolean} config.secure - Use Secure WebSocket if true
.\n * @param {string} config.platform - Optional platform identifier, one of \"ios\"
, \"web\"
, \"android\"
.\n * @param {boolen} config.persist - Use IndexedDB persistent storage.\n * @param {function} onComplete - callback to call when initialization is completed.\n */\n constructor(config, onComplete) {\n this._host = config.host;\n this._secure = config.secure;\n\n // Client-provided application name, format /\n this._appName = config.appName || \"Undefined\";\n\n // API Key.\n this._apiKey = config.apiKey;\n\n // Name and version of the browser.\n this._platform = config.platform || 'web';\n // Underlying OS.\n if (typeof navigator != 'undefined') {\n this._browser = getBrowserInfo(navigator.userAgent, navigator.product);\n this._hwos = navigator.platform;\n // This is the default language. It could be changed by client.\n this._humanLanguage = navigator.language || 'en-US';\n }\n\n Connection.logger = this.logger;\n Drafty.logger = this.logger;\n\n // WebSocket or long polling network connection.\n if (config.transport != 'lp' && config.transport != 'ws') {\n config.transport = detectTransport();\n }\n this._connection = new Connection(config, Const.PROTOCOL_VERSION, /* autoreconnect */ true);\n this._connection.onMessage = (data) => {\n // Call the main message dispatcher.\n this.#dispatchMessage(data);\n }\n\n // Ready to start sending.\n this._connection.onOpen = _ => this.#connectionOpen();\n this._connection.onDisconnect = (err, code) => this.#disconnected(err, code);\n\n // Wrapper for the reconnect iterator callback.\n this._connection.onAutoreconnectIteration = (timeout, promise) => {\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(timeout, promise);\n }\n }\n\n this._persist = config.persist;\n // Initialize object regardless. It simplifies the code.\n this._db = new DBCache(err => {\n this.logger('DB', err);\n }, this.logger);\n\n if (this._persist) {\n // Create the persistent cache.\n // Store promises to be resolved when messages load into memory.\n const prom = [];\n this._db.initDatabase().then(_ => {\n // First load topics into memory.\n return this._db.mapTopics((data) => {\n let topic = this.#cacheGet('topic', data.name);\n if (topic) {\n return;\n }\n if (data.name == Const.TOPIC_ME) {\n topic = new TopicMe();\n } else if (data.name == Const.TOPIC_FND) {\n topic = new TopicFnd();\n } else {\n topic = new Topic(data.name);\n }\n this._db.deserializeTopic(topic, data);\n this.#attachCacheToTopic(topic);\n topic._cachePutSelf();\n // Topic loaded from DB is not new.\n delete topic._new;\n // Request to load messages and save the promise.\n prom.push(topic._loadMessages(this._db));\n });\n }).then(_ => {\n // Then load users.\n return this._db.mapUsers((data) => {\n this.#cachePut('user', data.uid, mergeObj({}, data.public));\n });\n }).then(_ => {\n // Now wait for all messages to finish loading.\n return Promise.all(prom);\n }).then(_ => {\n if (onComplete) {\n onComplete();\n }\n this.logger(\"Persistent cache initialized.\");\n }).catch(err => {\n if (onComplete) {\n onComplete(err);\n }\n this.logger(\"Failed to initialize persistent cache:\", err);\n });\n } else {\n this._db.deleteDatabase().then(_ => {\n if (onComplete) {\n onComplete();\n }\n });\n }\n }\n\n // Private methods.\n\n // Console logger. Babel somehow fails to parse '...rest' parameter.\n logger(str, ...args) {\n if (this._loggingEnabled) {\n const d = new Date();\n const dateString = ('0' + d.getUTCHours()).slice(-2) + ':' +\n ('0' + d.getUTCMinutes()).slice(-2) + ':' +\n ('0' + d.getUTCSeconds()).slice(-2) + '.' +\n ('00' + d.getUTCMilliseconds()).slice(-3);\n\n console.log('[' + dateString + ']', str, args.join(' '));\n }\n }\n\n // Generator of default promises for sent packets.\n #makePromise(id) {\n let promise = null;\n if (id) {\n promise = new Promise((resolve, reject) => {\n // Stored callbacks will be called when the response packet with this Id arrives\n this._pendingPromises[id] = {\n 'resolve': resolve,\n 'reject': reject,\n 'ts': new Date()\n };\n });\n }\n return promise;\n };\n\n // Resolve or reject a pending promise.\n // Unresolved promises are stored in _pendingPromises.\n #execPromise(id, code, onOK, errorText) {\n const callbacks = this._pendingPromises[id];\n if (callbacks) {\n delete this._pendingPromises[id];\n if (code >= 200 && code < 400) {\n if (callbacks.resolve) {\n callbacks.resolve(onOK);\n }\n } else if (callbacks.reject) {\n callbacks.reject(new CommError(errorText, code));\n }\n }\n }\n\n // Send a packet. If packet id is provided return a promise.\n #send(pkt, id) {\n let promise;\n if (id) {\n promise = this.#makePromise(id);\n }\n pkt = simplify(pkt);\n let msg = JSON.stringify(pkt);\n this.logger(\"out: \" + (this._trimLongStrings ? JSON.stringify(pkt, jsonLoggerHelper) : msg));\n try {\n this._connection.sendText(msg);\n } catch (err) {\n // If sendText throws, wrap the error in a promise or rethrow.\n if (id) {\n this.#execPromise(id, Connection.NETWORK_ERROR, null, err.message);\n } else {\n throw err;\n }\n }\n return promise;\n }\n\n // The main message dispatcher.\n #dispatchMessage(data) {\n // Skip empty response. This happens when LP times out.\n if (!data)\n return;\n\n this._inPacketCount++;\n\n // Send raw message to listener\n if (this.onRawMessage) {\n this.onRawMessage(data);\n }\n\n if (data === '0') {\n // Server response to a network probe.\n if (this.onNetworkProbe) {\n this.onNetworkProbe();\n }\n // No processing is necessary.\n return;\n }\n\n let pkt = JSON.parse(data, jsonParseHelper);\n if (!pkt) {\n this.logger(\"in: \" + data);\n this.logger(\"ERROR: failed to parse data\");\n } else {\n this.logger(\"in: \" + (this._trimLongStrings ? JSON.stringify(pkt, jsonLoggerHelper) : data));\n\n // Send complete packet to listener\n if (this.onMessage) {\n this.onMessage(pkt);\n }\n\n if (pkt.ctrl) {\n // Handling {ctrl} message\n if (this.onCtrlMessage) {\n this.onCtrlMessage(pkt.ctrl);\n }\n\n // Resolve or reject a pending promise, if any\n if (pkt.ctrl.id) {\n this.#execPromise(pkt.ctrl.id, pkt.ctrl.code, pkt.ctrl, pkt.ctrl.text);\n }\n setTimeout(_ => {\n if (pkt.ctrl.code == 205 && pkt.ctrl.text == 'evicted') {\n // User evicted from topic.\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n topic._resetSub();\n if (pkt.ctrl.params && pkt.ctrl.params.unsub) {\n topic._gone();\n }\n }\n } else if (pkt.ctrl.code < 300 && pkt.ctrl.params) {\n if (pkt.ctrl.params.what == 'data') {\n // code=208, all messages received: \"params\":{\"count\":11,\"what\":\"data\"},\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n topic._allMessagesReceived(pkt.ctrl.params.count);\n }\n } else if (pkt.ctrl.params.what == 'sub') {\n // code=204, the topic has no (refreshed) subscriptions.\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n // Trigger topic.onSubsUpdated.\n topic._processMetaSubs([]);\n }\n }\n }\n }, 0);\n } else {\n setTimeout(_ => {\n if (pkt.meta) {\n // Handling a {meta} message.\n // Preferred API: Route meta to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.meta.topic);\n if (topic) {\n topic._routeMeta(pkt.meta);\n }\n\n if (pkt.meta.id) {\n this.#execPromise(pkt.meta.id, 200, pkt.meta, 'META');\n }\n\n // Secondary API: callback\n if (this.onMetaMessage) {\n this.onMetaMessage(pkt.meta);\n }\n } else if (pkt.data) {\n // Handling {data} message\n // Preferred API: Route data to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.data.topic);\n if (topic) {\n topic._routeData(pkt.data);\n }\n\n // Secondary API: Call callback\n if (this.onDataMessage) {\n this.onDataMessage(pkt.data);\n }\n } else if (pkt.pres) {\n // Handling {pres} message\n // Preferred API: Route presence to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.pres.topic);\n if (topic) {\n topic._routePres(pkt.pres);\n }\n\n // Secondary API - callback\n if (this.onPresMessage) {\n this.onPresMessage(pkt.pres);\n }\n } else if (pkt.info) {\n // {info} message - read/received notifications and key presses\n // Preferred API: Route {info}} to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.info.topic);\n if (topic) {\n topic._routeInfo(pkt.info);\n }\n\n // Secondary API - callback\n if (this.onInfoMessage) {\n this.onInfoMessage(pkt.info);\n }\n } else {\n this.logger(\"ERROR: Unknown packet received.\");\n }\n }, 0);\n }\n }\n }\n\n // Connection open, ready to start sending.\n #connectionOpen() {\n if (!this._expirePromises) {\n // Reject promises which have not been resolved for too long.\n this._expirePromises = setInterval(_ => {\n const err = new CommError(\"timeout\", 504);\n const expires = new Date(new Date().getTime() - Const.EXPIRE_PROMISES_TIMEOUT);\n for (let id in this._pendingPromises) {\n let callbacks = this._pendingPromises[id];\n if (callbacks && callbacks.ts < expires) {\n this.logger(\"Promise expired\", id);\n delete this._pendingPromises[id];\n if (callbacks.reject) {\n callbacks.reject(err);\n }\n }\n }\n }, Const.EXPIRE_PROMISES_PERIOD);\n }\n this.hello();\n }\n\n #disconnected(err, code) {\n this._inPacketCount = 0;\n this._serverInfo = null;\n this._authenticated = false;\n\n if (this._expirePromises) {\n clearInterval(this._expirePromises);\n this._expirePromises = null;\n }\n\n // Mark all topics as unsubscribed\n this.#cacheMap('topic', (topic, key) => {\n topic._resetSub();\n });\n\n // Reject all pending promises\n for (let key in this._pendingPromises) {\n const callbacks = this._pendingPromises[key];\n if (callbacks && callbacks.reject) {\n callbacks.reject(err);\n }\n }\n this._pendingPromises = {};\n\n if (this.onDisconnect) {\n this.onDisconnect(err);\n }\n }\n\n // Get User Agent string\n #getUserAgent() {\n return this._appName + ' (' + (this._browser ? this._browser + '; ' : '') + this._hwos + '); ' + Const.LIBRARY;\n }\n\n // Generator of packets stubs\n #initPacket(type, topic) {\n switch (type) {\n case 'hi':\n return {\n 'hi': {\n 'id': this.getNextUniqueId(),\n 'ver': Const.VERSION,\n 'ua': this.#getUserAgent(),\n 'dev': this._deviceToken,\n 'lang': this._humanLanguage,\n 'platf': this._platform\n }\n };\n\n case 'acc':\n return {\n 'acc': {\n 'id': this.getNextUniqueId(),\n 'user': null,\n 'scheme': null,\n 'secret': null,\n 'tmpscheme': null,\n 'tmpsecret': null,\n 'login': false,\n 'tags': null,\n 'desc': {},\n 'cred': {}\n }\n };\n\n case 'login':\n return {\n 'login': {\n 'id': this.getNextUniqueId(),\n 'scheme': null,\n 'secret': null\n }\n };\n\n case 'sub':\n return {\n 'sub': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'set': {},\n 'get': {}\n }\n };\n\n case 'leave':\n return {\n 'leave': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'unsub': false\n }\n };\n\n case 'pub':\n return {\n 'pub': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'noecho': false,\n 'head': null,\n 'content': {}\n }\n };\n\n case 'get':\n return {\n 'get': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'what': null,\n 'desc': {},\n 'sub': {},\n 'data': {}\n }\n };\n\n case 'set':\n return {\n 'set': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'desc': {},\n 'sub': {},\n 'tags': [],\n 'ephemeral': {}\n }\n };\n\n case 'del':\n return {\n 'del': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'what': null,\n 'delseq': null,\n 'user': null,\n 'hard': false\n }\n };\n\n case 'note':\n return {\n 'note': {\n // no id by design (except calls).\n 'topic': topic,\n 'what': null, // one of \"recv\", \"read\", \"kp\", \"call\"\n 'seq': undefined // the server-side message id acknowledged as received or read.\n }\n };\n\n default:\n throw new Error(`Unknown packet type requested: ${type}`);\n }\n }\n\n // Cache management\n #cachePut(type, name, obj) {\n this._cache[type + ':' + name] = obj;\n }\n #cacheGet(type, name) {\n return this._cache[type + ':' + name];\n }\n #cacheDel(type, name) {\n delete this._cache[type + ':' + name];\n }\n\n // Enumerate all items in cache, call func for each item.\n // Enumeration stops if func returns true.\n #cacheMap(type, func, context) {\n const key = type ? type + ':' : undefined;\n for (let idx in this._cache) {\n if (!key || idx.indexOf(key) == 0) {\n if (func.call(context, this._cache[idx], idx)) {\n break;\n }\n }\n }\n }\n\n // Make limited cache management available to topic.\n // Caching user.public only. Everything else is per-topic.\n #attachCacheToTopic(topic) {\n topic._tinode = this;\n\n topic._cacheGetUser = (uid) => {\n const pub = this.#cacheGet('user', uid);\n if (pub) {\n return {\n user: uid,\n public: mergeObj({}, pub)\n };\n }\n return undefined;\n };\n topic._cachePutUser = (uid, user) => {\n this.#cachePut('user', uid, mergeObj({}, user.public));\n };\n topic._cacheDelUser = (uid) => {\n this.#cacheDel('user', uid);\n };\n topic._cachePutSelf = _ => {\n this.#cachePut('topic', topic.name, topic);\n };\n topic._cacheDelSelf = _ => {\n this.#cacheDel('topic', topic.name);\n };\n }\n\n // On successful login save server-provided data.\n #loginSuccessful(ctrl) {\n if (!ctrl.params || !ctrl.params.user) {\n return ctrl;\n }\n // This is a response to a successful login,\n // extract UID and security token, save it in Tinode module\n this._myUID = ctrl.params.user;\n this._authenticated = (ctrl && ctrl.code >= 200 && ctrl.code < 300);\n if (ctrl.params && ctrl.params.token && ctrl.params.expires) {\n this._authToken = {\n token: ctrl.params.token,\n expires: ctrl.params.expires\n };\n } else {\n this._authToken = null;\n }\n\n if (this.onLogin) {\n this.onLogin(ctrl.code, ctrl.text);\n }\n\n return ctrl;\n }\n\n // Static methods.\n /**\n * Helper method to package account credential.\n *\n * @param {string | Credential} meth - validation method or object with validation data.\n * @param {string=} val - validation value (e.g. email or phone number).\n * @param {Object=} params - validation parameters.\n * @param {string=} resp - validation response.\n *\n * @returns {Array.} array with a single credential or null
if no valid credentials were given.\n */\n static credential(meth, val, params, resp) {\n if (typeof meth == 'object') {\n ({\n val,\n params,\n resp,\n meth\n } = meth);\n }\n if (meth && (val || resp)) {\n return [{\n 'meth': meth,\n 'val': val,\n 'resp': resp,\n 'params': params\n }];\n }\n return null;\n }\n\n /**\n * Determine topic type from topic's name: grp, p2p, me, fnd, sys.\n * @param {string} name - Name of the topic to test.\n * @returns {string} One of \"me\"
, \"fnd\"
, \"sys\"
, \"grp\"
,\n * \"p2p\"
or undefined
.\n */\n static topicType(name) {\n return Topic.topicType(name);\n }\n\n /**\n * Check if the given topic name is a name of a 'me' topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a 'me' topic, false
otherwise.\n */\n static isMeTopicName(name) {\n return Topic.isMeTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a group topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a group topic, false
otherwise.\n */\n static isGroupTopicName(name) {\n return Topic.isGroupTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a p2p topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p topic, false
otherwise.\n */\n static isP2PTopicName(name) {\n return Topic.isP2PTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a communication topic, i.e. P2P or group.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p or group topic, false
otherwise.\n */\n static isCommTopicName(name) {\n return Topic.isCommTopicName(name);\n }\n /**\n * Check if the topic name is a name of a new topic.\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a new topic, false
otherwise.\n */\n static isNewGroupTopicName(name) {\n return Topic.isNewGroupTopicName(name);\n }\n /**\n * Check if the topic name is a name of a channel.\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a channel, false
otherwise.\n */\n static isChannelTopicName(name) {\n return Topic.isChannelTopicName(name);\n }\n /**\n * Get information about the current version of this Tinode client library.\n * @returns {string} semantic version of the library, e.g. \"0.15.5-rc1\"
.\n */\n static getVersion() {\n return Const.VERSION;\n }\n /**\n * To use Tinode in a non browser context, supply WebSocket and XMLHttpRequest providers.\n * @static\n *\n * @param wsProvider WebSocket
provider, e.g. for nodeJS , require('ws')
.\n * @param xhrProvider XMLHttpRequest
provider, e.g. for node require('xhr')
.\n */\n static setNetworkProviders(wsProvider, xhrProvider) {\n WebSocketProvider = wsProvider;\n XHRProvider = xhrProvider;\n\n Connection.setNetworkProviders(WebSocketProvider, XHRProvider);\n LargeFileHelper.setNetworkProvider(XHRProvider);\n }\n /**\n * To use Tinode in a non browser context, supply indexedDB
provider.\n * @static\n *\n * @param idbProvider indexedDB
provider, e.g. for nodeJS , require('fake-indexeddb')
.\n */\n static setDatabaseProvider(idbProvider) {\n IndexedDBProvider = idbProvider;\n\n DBCache.setDatabaseProvider(IndexedDBProvider);\n }\n /**\n * Return information about the current name and version of this Tinode library.\n * @static\n *\n * @returns {string} the name of the library and it's version.\n */\n static getLibrary() {\n return Const.LIBRARY;\n }\n /**\n * Check if the given string represents NULL
value as defined by Tinode ('\\u2421'
).\n * @param {string} str - string to check for NULL
value.\n * @returns {boolean} true
if string represents NULL
value, false
otherwise.\n */\n static isNullValue(str) {\n return str === Const.DEL_CHAR;\n }\n\n // Instance methods.\n\n // Generates unique message IDs\n getNextUniqueId() {\n return (this._messageId != 0) ? '' + this._messageId++ : undefined;\n };\n\n /**\n * Connect to the server.\n *\n * @param {string} host_ - name of the host to connect to.\n * @return {Promise} Promise resolved/rejected when the connection call completes:\n * resolve()
is called without parameters, reject()
receives the\n * Error
as a single parameter.\n */\n connect(host_) {\n return this._connection.connect(host_);\n }\n\n /**\n * Attempt to reconnect to the server immediately.\n *\n * @param {string} force - if true
, reconnect even if there is a connection already.\n */\n reconnect(force) {\n this._connection.reconnect(force);\n }\n\n /**\n * Disconnect from the server.\n */\n disconnect() {\n this._connection.disconnect();\n }\n\n /**\n * Clear persistent cache: remove IndexedDB.\n *\n * @return {Promise} Promise resolved/rejected when the operation is completed.\n */\n clearStorage() {\n if (this._db.isReady()) {\n return this._db.deleteDatabase();\n }\n return Promise.resolve();\n }\n\n /**\n * Initialize persistent cache: create IndexedDB cache.\n *\n * @return {Promise} Promise resolved/rejected when the operation is completed.\n */\n initStorage() {\n if (!this._db.isReady()) {\n return this._db.initDatabase();\n }\n return Promise.resolve();\n }\n\n /**\n * Send a network probe message to make sure the connection is alive.\n */\n networkProbe() {\n this._connection.probe();\n }\n\n /**\n * Check for live connection to server.\n *\n * @returns {boolean} true
if there is a live connection, false
otherwise.\n */\n isConnected() {\n return this._connection.isConnected();\n }\n\n /**\n * Check if connection is authenticated (last login was successful).\n *\n * @returns {boolean} true
if authenticated, false
otherwise.\n */\n isAuthenticated() {\n return this._authenticated;\n }\n\n /**\n * Add API key and auth token to the relative URL making it usable for getting data\n * from the server in a simple HTTP GET
request.\n *\n * @param {string} URL - URL to wrap.\n * @returns {string} URL with appended API key and token, if valid token is present.\n */\n authorizeURL(url) {\n if (typeof url != 'string') {\n return url;\n }\n\n if (isUrlRelative(url)) {\n // Fake base to make the relative URL parseable.\n const base = 'scheme://host/';\n const parsed = new URL(url, base);\n if (this._apiKey) {\n parsed.searchParams.append('apikey', this._apiKey);\n }\n if (this._authToken && this._authToken.token) {\n parsed.searchParams.append('auth', 'token');\n parsed.searchParams.append('secret', this._authToken.token);\n }\n // Convert back to string and strip fake base URL except for the root slash.\n url = parsed.toString().substring(base.length - 1);\n }\n return url;\n }\n\n /**\n * @typedef AccountParams\n * @type {Object}\n * @property {DefAcs=} defacs - Default access parameters for user's me
topic.\n * @property {Object=} public - Public application-defined data exposed on me
topic.\n * @property {Object=} private - Private application-defined data accessible on me
topic.\n * @property {Object=} trusted - Trusted user data which can be set by a root user only.\n * @property {Array.} tags - array of string tags for user discovery.\n * @property {string} scheme - Temporary authentication scheme for password reset.\n * @property {string} secret - Temporary authentication secret for password reset.\n * @property {Array.=} attachments - Array of references to out of band attachments used in account description.\n */\n /**\n * @typedef DefAcs\n * @type {Object}\n * @property {string=} auth - Access mode for me
for authenticated users.\n * @property {string=} anon - Access mode for me
for anonymous users.\n */\n\n /**\n * Create or update an account.\n *\n * @param {string} uid - User id to update\n * @param {string} scheme - Authentication scheme; \"basic\"
and \"anonymous\"
are the currently supported schemes.\n * @param {string} secret - Authentication secret, assumed to be already base64 encoded.\n * @param {boolean=} login - Use new account to authenticate current session\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n account(uid, scheme, secret, login, params) {\n const pkt = this.#initPacket('acc');\n pkt.acc.user = uid;\n pkt.acc.scheme = scheme;\n pkt.acc.secret = secret;\n // Log in to the new account using selected scheme\n pkt.acc.login = login;\n\n if (params) {\n pkt.acc.desc.defacs = params.defacs;\n pkt.acc.desc.public = params.public;\n pkt.acc.desc.private = params.private;\n pkt.acc.desc.trusted = params.trusted;\n\n pkt.acc.tags = params.tags;\n pkt.acc.cred = params.cred;\n\n pkt.acc.tmpscheme = params.scheme;\n pkt.acc.tmpsecret = params.secret;\n\n if (Array.isArray(params.attachments) && params.attachments.length > 0) {\n pkt.extra = {\n attachments: params.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n }\n\n return this.#send(pkt, pkt.acc.id);\n }\n\n /**\n * Create a new user. Wrapper for {@link Tinode#account}.\n *\n * @param {string} scheme - Authentication scheme; \"basic\"
is the only currently supported scheme.\n * @param {string} secret - Authentication.\n * @param {boolean=} login - Use new account to authenticate current session\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n createAccount(scheme, secret, login, params) {\n let promise = this.account(Const.USER_NEW, scheme, secret, login, params);\n if (login) {\n promise = promise.then(ctrl => this.#loginSuccessful(ctrl));\n }\n return promise;\n }\n\n /**\n * Create user with 'basic'
authentication scheme and immediately\n * use it for authentication. Wrapper for {@link Tinode#account}.\n *\n * @param {string} username - Login to use for the new account.\n * @param {string} password - User's password.\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n createAccountBasic(username, password, params) {\n // Make sure we are not using 'null' or 'undefined';\n username = username || '';\n password = password || '';\n return this.createAccount('basic',\n b64EncodeUnicode(username + ':' + password), true, params);\n }\n\n /**\n * Update user's credentials for 'basic'
authentication scheme. Wrapper for {@link Tinode#account}.\n *\n * @param {string} uid - User ID to update.\n * @param {string} username - Login to use for the new account.\n * @param {string} password - User's password.\n * @param {AccountParams=} params - data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n updateAccountBasic(uid, username, password, params) {\n // Make sure we are not using 'null' or 'undefined';\n username = username || '';\n password = password || '';\n return this.account(uid, 'basic',\n b64EncodeUnicode(username + ':' + password), false, params);\n }\n\n /**\n * Send handshake to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n hello() {\n const pkt = this.#initPacket('hi');\n\n return this.#send(pkt, pkt.hi.id)\n .then(ctrl => {\n // Reset backoff counter on successful connection.\n this._connection.backoffReset();\n\n // Server response contains server protocol version, build, constraints,\n // session ID for long polling. Save them.\n if (ctrl.params) {\n this._serverInfo = ctrl.params;\n }\n\n if (this.onConnect) {\n this.onConnect();\n }\n\n return ctrl;\n }).catch(err => {\n this._connection.reconnect(true);\n\n if (this.onDisconnect) {\n this.onDisconnect(err);\n }\n });\n }\n\n /**\n * Set or refresh the push notifications/device token. If the client is connected,\n * the deviceToken can be sent to the server.\n *\n * @param {string} dt - token obtained from the provider or false
,\n * null
or undefined
to clear the token.\n *\n * @returns true
if attempt was made to send the update to the server.\n */\n setDeviceToken(dt) {\n let sent = false;\n // Convert any falsish value to null.\n dt = dt || null;\n if (dt != this._deviceToken) {\n this._deviceToken = dt;\n if (this.isConnected() && this.isAuthenticated()) {\n this.#send({\n 'hi': {\n 'dev': dt || Tinode.DEL_CHAR\n }\n });\n sent = true;\n }\n }\n return sent;\n }\n\n /**\n * @typedef Credential\n * @type {Object}\n * @property {string} meth - validation method.\n * @property {string} val - value to validate (e.g. email or phone number).\n * @property {string} resp - validation response.\n * @property {Object} params - validation parameters.\n */\n /**\n * Authenticate current session.\n *\n * @param {string} scheme - Authentication scheme; \"basic\"
is the only currently supported scheme.\n * @param {string} secret - Authentication secret, assumed to be already base64 encoded.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n login(scheme, secret, cred) {\n const pkt = this.#initPacket('login');\n pkt.login.scheme = scheme;\n pkt.login.secret = secret;\n pkt.login.cred = cred;\n\n return this.#send(pkt, pkt.login.id)\n .then(ctrl => this.#loginSuccessful(ctrl));\n }\n\n /**\n * Wrapper for {@link Tinode#login} with basic authentication\n *\n * @param {string} uname - User name.\n * @param {string} password - Password.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n loginBasic(uname, password, cred) {\n return this.login('basic', b64EncodeUnicode(uname + ':' + password), cred)\n .then(ctrl => {\n this._login = uname;\n return ctrl;\n });\n }\n\n /**\n * Wrapper for {@link Tinode#login} with token authentication\n *\n * @param {string} token - Token received in response to earlier login.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n loginToken(token, cred) {\n return this.login('token', token, cred);\n }\n\n /**\n * Send a request for resetting an authentication secret.\n *\n * @param {string} scheme - authentication scheme to reset.\n * @param {string} method - method to use for resetting the secret, such as \"email\" or \"tel\".\n * @param {string} value - value of the credential to use, a specific email address or a phone number.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving the server reply.\n */\n requestResetAuthSecret(scheme, method, value) {\n return this.login('reset', b64EncodeUnicode(scheme + ':' + method + ':' + value));\n }\n\n /**\n * @typedef AuthToken\n * @type {Object}\n * @property {string} token - Token value.\n * @property {Date} expires - Token expiration time.\n */\n /**\n * Get stored authentication token.\n *\n * @returns {AuthToken} authentication token.\n */\n getAuthToken() {\n if (this._authToken && (this._authToken.expires.getTime() > Date.now())) {\n return this._authToken;\n } else {\n this._authToken = null;\n }\n return null;\n }\n\n /**\n * Application may provide a saved authentication token.\n *\n * @param {AuthToken} token - authentication token.\n */\n setAuthToken(token) {\n this._authToken = token;\n }\n\n /**\n * @typedef SetParams\n * @type {Object}\n * @property {SetDesc=} desc - Topic initialization parameters when creating a new topic or a new subscription.\n * @property {SetSub=} sub - Subscription initialization parameters.\n * @property {Array.=} attachments - URLs of out of band attachments used in parameters.\n */\n /**\n * @typedef SetDesc\n * @type {Object}\n * @property {DefAcs=} defacs - Default access mode.\n * @property {Object=} public - Free-form topic description, publically accessible.\n * @property {Object=} private - Free-form topic description accessible only to the owner.\n * @property {Object=} trusted - Trusted user data which can be set by a root user only.\n */\n /**\n * @typedef SetSub\n * @type {Object}\n * @property {string=} user - UID of the user affected by the request. Default (empty) - current user.\n * @property {string=} mode - User access mode, either requested or assigned dependent on context.\n */\n /**\n * Send a topic subscription request.\n *\n * @param {string} topic - Name of the topic to subscribe to.\n * @param {GetQuery=} getParams - Optional subscription metadata query\n * @param {SetParams=} setParams - Optional initialization parameters\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n subscribe(topicName, getParams, setParams) {\n const pkt = this.#initPacket('sub', topicName)\n if (!topicName) {\n topicName = Const.TOPIC_NEW;\n }\n\n pkt.sub.get = getParams;\n\n if (setParams) {\n if (setParams.sub) {\n pkt.sub.set.sub = setParams.sub;\n }\n\n if (setParams.desc) {\n const desc = setParams.desc;\n if (Tinode.isNewGroupTopicName(topicName)) {\n // Full set.desc params are used for new topics only\n pkt.sub.set.desc = desc;\n } else if (Tinode.isP2PTopicName(topicName) && desc.defacs) {\n // Use optional default permissions only.\n pkt.sub.set.desc = {\n defacs: desc.defacs\n };\n }\n }\n\n // See if external objects were used in topic description.\n if (Array.isArray(setParams.attachments) && setParams.attachments.length > 0) {\n pkt.extra = {\n attachments: setParams.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n\n if (setParams.tags) {\n pkt.sub.set.tags = setParams.tags;\n }\n }\n return this.#send(pkt, pkt.sub.id);\n }\n\n /**\n * Detach and optionally unsubscribe from the topic\n *\n * @param {string} topic - Topic to detach from.\n * @param {boolean} unsub - If true
, detach and unsubscribe, otherwise just detach.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n leave(topic, unsub) {\n const pkt = this.#initPacket('leave', topic);\n pkt.leave.unsub = unsub;\n\n return this.#send(pkt, pkt.leave.id);\n }\n\n /**\n * Create message draft without sending it to the server.\n *\n * @param {string} topic - Name of the topic to publish to.\n * @param {Object} content - Payload to publish.\n * @param {boolean=} noEcho - If true
, tell the server not to echo the message to the original session.\n *\n * @returns {Object} new message which can be sent to the server or otherwise used.\n */\n createMessage(topic, content, noEcho) {\n const pkt = this.#initPacket('pub', topic);\n\n let dft = typeof content == 'string' ? Drafty.parse(content) : content;\n if (dft && !Drafty.isPlainText(dft)) {\n pkt.pub.head = {\n mime: Drafty.getContentType()\n };\n content = dft;\n }\n pkt.pub.noecho = noEcho;\n pkt.pub.content = content;\n\n return pkt.pub;\n }\n\n /**\n * Publish {data} message to topic.\n *\n * @param {string} topicName - Name of the topic to publish to.\n * @param {Object} content - Payload to publish.\n * @param {boolean=} noEcho - If true
, tell the server not to echo the message to the original session.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n publish(topicName, content, noEcho) {\n return this.publishMessage(\n this.createMessage(topicName, content, noEcho)\n );\n }\n\n /**\n * Publish message to topic. The message should be created by {@link Tinode#createMessage}.\n *\n * @param {Object} pub - Message to publish.\n * @param {Array.=} attachments - array of URLs with attachments.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n publishMessage(pub, attachments) {\n // Make a shallow copy. Needed in order to clear locally-assigned temp values;\n pub = Object.assign({}, pub);\n pub.seq = undefined;\n pub.from = undefined;\n pub.ts = undefined;\n const msg = {\n pub: pub,\n };\n if (attachments) {\n msg.extra = {\n attachments: attachments.filter(ref => isUrlRelative(ref))\n };\n }\n return this.#send(msg, pub.id);\n }\n\n /**\n * Out of band notification: notify topic that an external (push) notification was recived by the client.\n *\n * @param {object} data - notification payload.\n * @param {string} data.what - notification type, 'msg', 'read', 'sub'.\n * @param {string} data.topic - name of the updated topic.\n * @param {number=} data.seq - seq ID of the affected message.\n * @param {string=} data.xfrom - UID of the sender.\n * @param {object=} data.given - new subscription 'given', e.g. 'ASWP...'.\n * @param {object=} data.want - new subscription 'want', e.g. 'RWJ...'.\n */\n oobNotification(data) {\n this.logger('oob: ' + (this._trimLongStrings ? JSON.stringify(data, jsonLoggerHelper) : data));\n\n switch (data.what) {\n case 'msg':\n if (!data.seq || data.seq < 1 || !data.topic) {\n // Server sent invalid data.\n break;\n }\n\n if (!this.isConnected()) {\n // Let's ignore the message if there is no connection: no connection means there are no open\n // tabs with Tinode.\n break;\n }\n\n const topic = this.#cacheGet('topic', data.topic);\n if (!topic) {\n // TODO: check if there is a case when a message can arrive from an unknown topic.\n break;\n }\n\n if (topic.isSubscribed()) {\n // No need to fetch: topic is already subscribed and got data through normal channel.\n break;\n }\n\n if (topic.maxMsgSeq() < data.seq) {\n if (topic.isChannelType()) {\n topic._updateReceived(data.seq, 'fake-uid');\n }\n\n // New message.\n if (data.xfrom && !this.#cacheGet('user', data.xfrom)) {\n // Message from unknown sender, fetch description from the server.\n // Sending asynchronously without a subscription.\n this.getMeta(data.xfrom, new MetaGetBuilder().withDesc().build()).catch(err => {\n this.logger(\"Failed to get the name of a new sender\", err);\n });\n }\n\n topic.subscribe(null).then(_ => {\n return topic.getMeta(new MetaGetBuilder(topic).withLaterData(24).withLaterDel(24).build());\n }).then(_ => {\n // Allow data fetch to complete and get processed successfully.\n topic.leaveDelayed(false, 1000);\n }).catch(err => {\n this.logger(\"On push data fetch failed\", err);\n }).finally(_ => {\n this.getMeTopic()._refreshContact('msg', topic);\n });\n }\n break;\n\n case 'read':\n this.getMeTopic()._routePres({\n what: 'read',\n seq: data.seq\n });\n break;\n\n case 'sub':\n if (!this.isMe(data.xfrom)) {\n // TODO: handle updates from other users.\n break;\n }\n\n const mode = {\n given: data.modeGiven,\n want: data.modeWant\n };\n const acs = new AccessMode(mode);\n const pres = (!acs.mode || acs.mode == AccessMode._NONE) ?\n // Subscription deleted.\n {\n what: 'gone',\n src: data.topic\n } :\n // New subscription or subscription updated.\n {\n what: 'acs',\n src: data.topic,\n dacs: mode\n };\n this.getMeTopic()._routePres(pres);\n break;\n\n default:\n this.logger(\"Unknown push type ignored\", data.what);\n }\n }\n\n /**\n * @typedef GetQuery\n * @type {Object}\n * @property {GetOptsType=} desc - If provided (even if empty), fetch topic description.\n * @property {GetOptsType=} sub - If provided (even if empty), fetch topic subscriptions.\n * @property {GetDataType=} data - If provided (even if empty), get messages.\n */\n\n /**\n * @typedef GetOptsType\n * @type {Object}\n * @property {Date=} ims - \"If modified since\", fetch data only it was was modified since stated date.\n * @property {number=} limit - Maximum number of results to return. Ignored when querying topic description.\n */\n\n /**\n * @typedef GetDataType\n * @type {Object}\n * @property {number=} since - Load messages with seq id equal or greater than this value.\n * @property {number=} before - Load messages with seq id lower than this number.\n * @property {number=} limit - Maximum number of results to return.\n */\n\n /**\n * Request topic metadata\n *\n * @param {string} topic - Name of the topic to query.\n * @param {GetQuery} params - Parameters of the query. Use {@link Tinode.MetaGetBuilder} to generate.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n getMeta(topic, params) {\n const pkt = this.#initPacket('get', topic);\n\n pkt.get = mergeObj(pkt.get, params);\n\n return this.#send(pkt, pkt.get.id);\n }\n\n /**\n * Update topic's metadata: description, subscribtions.\n *\n * @param {string} topic - Topic to update.\n * @param {SetParams} params - topic metadata to update.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n setMeta(topic, params) {\n const pkt = this.#initPacket('set', topic);\n const what = [];\n\n if (params) {\n ['desc', 'sub', 'tags', 'cred', 'ephemeral'].forEach(function(key) {\n if (params.hasOwnProperty(key)) {\n what.push(key);\n pkt.set[key] = params[key];\n }\n });\n\n if (Array.isArray(params.attachments) && params.attachments.length > 0) {\n pkt.extra = {\n attachments: params.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n }\n\n if (what.length == 0) {\n return Promise.reject(new Error(\"Invalid {set} parameters\"));\n }\n\n return this.#send(pkt, pkt.set.id);\n }\n\n /**\n * Range of message IDs to delete.\n *\n * @typedef DelRange\n * @type {Object}\n * @property {number} low - low end of the range, inclusive (closed).\n * @property {number=} hi - high end of the range, exclusive (open).\n */\n /**\n * Delete some or all messages in a topic.\n *\n * @param {string} topic - Topic name to delete messages from.\n * @param {DelRange[]} list - Ranges of message IDs to delete.\n * @param {boolean=} hard - Hard or soft delete\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delMessages(topic, ranges, hard) {\n const pkt = this.#initPacket('del', topic);\n\n pkt.del.what = 'msg';\n pkt.del.delseq = ranges;\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete the topic alltogether. Requires Owner permission.\n *\n * @param {string} topicName - Name of the topic to delete\n * @param {boolean} hard - hard-delete topic.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delTopic(topicName, hard) {\n const pkt = this.#initPacket('del', topicName);\n pkt.del.what = 'topic';\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete subscription. Requires Share permission.\n *\n * @param {string} topicName - Name of the topic to delete\n * @param {string} user - User ID to remove.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delSubscription(topicName, user) {\n const pkt = this.#initPacket('del', topicName);\n pkt.del.what = 'sub';\n pkt.del.user = user;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete credential. Always sent on 'me'
topic.\n *\n * @param {string} method - validation method such as 'email'
or 'tel'
.\n * @param {string} value - validation value, i.e. 'alice@example.com'
.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCredential(method, value) {\n const pkt = this.#initPacket('del', Const.TOPIC_ME);\n pkt.del.what = 'cred';\n pkt.del.cred = {\n meth: method,\n val: value\n };\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Request to delete account of the current user.\n *\n * @param {boolean} hard - hard-delete user.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCurrentUser(hard) {\n const pkt = this.#initPacket('del', null);\n pkt.del.what = 'user';\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id).then(_ => {\n this._myUID = null;\n });\n }\n\n /**\n * Notify server that a message or messages were read or received. Does NOT return promise.\n *\n * @param {string} topicName - Name of the topic where the mesage is being aknowledged.\n * @param {string} what - Action being aknowledged, either \"read\"
or \"recv\"
.\n * @param {number} seq - Maximum id of the message being acknowledged.\n */\n note(topicName, what, seq) {\n if (seq <= 0 || seq >= Const.LOCAL_SEQID) {\n throw new Error(`Invalid message id ${seq}`);\n }\n\n const pkt = this.#initPacket('note', topicName);\n pkt.note.what = what;\n pkt.note.seq = seq;\n this.#send(pkt);\n }\n\n /**\n * Broadcast a key-press notification to topic subscribers. Used to show\n * typing notifications \"user X is typing...\".\n *\n * @param {string} topicName - Name of the topic to broadcast to.\n * @param {string=} type - notification to send, default is 'kp'.\n */\n noteKeyPress(topicName, type) {\n const pkt = this.#initPacket('note', topicName);\n pkt.note.what = type || 'kp';\n this.#send(pkt);\n }\n\n /**\n * Send a video call notification to topic subscribers (including dialing,\n * hangup, etc.).\n *\n * @param {string} topicName - Name of the topic to broadcast to.\n * @param {int} seq - ID of the call message the event pertains to.\n * @param {string} evt - Call event.\n * @param {string} payload - Payload associated with this event (e.g. SDP string).\n *\n * @returns {Promise} Promise (for some call events) which will\n * be resolved/rejected on receiving server reply\n */\n videoCall(topicName, seq, evt, payload) {\n const pkt = this.#initPacket('note', topicName);\n pkt.note.seq = seq;\n pkt.note.what = 'call';\n pkt.note.event = evt;\n pkt.note.payload = payload;\n this.#send(pkt, pkt.note.id);\n }\n\n /**\n * Get a named topic, either pull it from cache or create a new instance.\n * There is a single instance of topic for each name.\n *\n * @param {string} topicName - Name of the topic to get.\n *\n * @returns {Topic} Requested or newly created topic or undefined
if topic name is invalid.\n */\n getTopic(topicName) {\n let topic = this.#cacheGet('topic', topicName);\n if (!topic && topicName) {\n if (topicName == Const.TOPIC_ME) {\n topic = new TopicMe();\n } else if (topicName == Const.TOPIC_FND) {\n topic = new TopicFnd();\n } else {\n topic = new Topic(topicName);\n }\n // Cache management.\n this.#attachCacheToTopic(topic);\n topic._cachePutSelf();\n // Don't save to DB here: a record will be added when the topic is subscribed.\n }\n return topic;\n }\n\n /**\n * Get a named topic from cache.\n *\n * @param {string} topicName - Name of the topic to get.\n *\n * @returns {Topic} Requested topic or undefined
if topic is not found in cache.\n */\n cacheGetTopic(topicName) {\n return this.#cacheGet('topic', topicName);\n }\n\n /**\n * Remove named topic from cache.\n *\n * @param {string} topicName - Name of the topic to remove from cache.\n */\n cacheRemTopic(topicName) {\n this.#cacheDel('topic', topicName);\n }\n\n /**\n * Iterate over cached topics.\n *\n * @param {Function} func - callback to call for each topic.\n * @param {Object} context - 'this' inside the 'func'.\n */\n mapTopics(func, context) {\n this.#cacheMap('topic', func, context);\n }\n\n /**\n * Check if named topic is already present in cache.\n *\n * @param {string} topicName - Name of the topic to check.\n * @returns {boolean} true if topic is found in cache, false otherwise.\n */\n isTopicCached(topicName) {\n return !!this.#cacheGet('topic', topicName);\n }\n\n /**\n * Generate unique name like 'new123456'
suitable for creating a new group topic.\n *\n * @param {boolean} isChan - if the topic is channel-enabled.\n * @returns {string} name which can be used for creating a new group topic.\n */\n newGroupTopicName(isChan) {\n return (isChan ? Const.TOPIC_NEW_CHAN : Const.TOPIC_NEW) + this.getNextUniqueId();\n }\n\n /**\n * Instantiate 'me'
topic or get it from cache.\n *\n * @returns {TopicMe} Instance of 'me'
topic.\n */\n getMeTopic() {\n return this.getTopic(Const.TOPIC_ME);\n }\n\n /**\n * Instantiate 'fnd'
(find) topic or get it from cache.\n *\n * @returns {Topic} Instance of 'fnd'
topic.\n */\n getFndTopic() {\n return this.getTopic(Const.TOPIC_FND);\n }\n\n /**\n * Create a new {@link LargeFileHelper} instance\n *\n * @returns {LargeFileHelper} instance of a {@link Tinode.LargeFileHelper}.\n */\n getLargeFileHelper() {\n return new LargeFileHelper(this, Const.PROTOCOL_VERSION);\n }\n\n /**\n * Get the UID of the the current authenticated user.\n *\n * @returns {string} UID of the current user or undefined
if the session is not yet\n * authenticated or if there is no session.\n */\n getCurrentUserID() {\n return this._myUID;\n }\n\n /**\n * Check if the given user ID is equal to the current user's UID.\n *\n * @param {string} uid - UID to check.\n *\n * @returns {boolean} true if the given UID belongs to the current logged in user.\n */\n isMe(uid) {\n return this._myUID === uid;\n }\n\n /**\n * Get login used for last successful authentication.\n *\n * @returns {string} login last used successfully or undefined
.\n */\n getCurrentLogin() {\n return this._login;\n }\n\n /**\n * Return information about the server: protocol version and build timestamp.\n *\n * @returns {Object} build and version of the server or null
if there is no connection or\n * if the first server response has not been received yet.\n */\n getServerInfo() {\n return this._serverInfo;\n }\n\n /**\n * Report a topic for abuse. Wrapper for {@link Tinode#publish}.\n *\n * @param {string} action - the only supported action is 'report'.\n * @param {string} target - name of the topic being reported.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n report(action, target) {\n return this.publish(Const.TOPIC_SYS, Drafty.attachJSON(null, {\n 'action': action,\n 'target': target\n }));\n }\n\n /**\n * Return server-provided configuration value.\n *\n * @param {string} name of the value to return.\n * @param {Object} defaultValue to return in case the parameter is not set or not found.\n *\n * @returns {Object} named value.\n */\n getServerParam(name, defaultValue) {\n return this._serverInfo && this._serverInfo[name] || defaultValue;\n }\n\n /**\n * Toggle console logging. Logging is off by default.\n *\n * @param {boolean} enabled - Set to true
to enable logging to console.\n * @param {boolean} trimLongStrings - Set to true
to trim long strings.\n */\n enableLogging(enabled, trimLongStrings) {\n this._loggingEnabled = enabled;\n this._trimLongStrings = enabled && trimLongStrings;\n }\n\n /**\n * Set UI language to report to the server. Must be called before 'hi'
is sent, otherwise it will not be used.\n *\n * @param {string} hl - human (UI) language, like \"en_US\"
or \"zh-Hans\"
.\n */\n setHumanLanguage(hl) {\n if (hl) {\n this._humanLanguage = hl;\n }\n }\n\n /**\n * Check if given topic is online.\n *\n * @param {string} name of the topic to test.\n * @returns {boolean} true if topic is online, false otherwise.\n */\n isTopicOnline(name) {\n const topic = this.#cacheGet('topic', name);\n return topic && topic.online;\n }\n\n /**\n * Get access mode for the given contact.\n *\n * @param {string} name of the topic to query.\n * @returns {AccessMode} access mode if topic is found, null otherwise.\n */\n getTopicAccessMode(name) {\n const topic = this.#cacheGet('topic', name);\n return topic ? topic.acs : null;\n }\n\n /**\n * Include message ID into all subsequest messages to server instructin it to send aknowledgemens.\n * Required for promises to function. Default is \"on\"
.\n *\n * @param {boolean} status - Turn aknowledgemens on or off.\n * @deprecated\n */\n wantAkn(status) {\n if (status) {\n this._messageId = Math.floor((Math.random() * 0xFFFFFF) + 0xFFFFFF);\n } else {\n this._messageId = 0;\n }\n }\n\n // Callbacks:\n /**\n * Callback to report when the websocket is opened. The callback has no parameters.\n *\n * @type {onWebsocketOpen}\n */\n onWebsocketOpen = undefined;\n\n /**\n * @typedef ServerParams\n *\n * @type {Object}\n * @property {string} ver - Server version\n * @property {string} build - Server build\n * @property {string=} sid - Session ID, long polling connections only.\n */\n\n /**\n * @callback onConnect\n * @param {number} code - Result code\n * @param {string} text - Text epxplaining the completion, i.e \"OK\" or an error message.\n * @param {ServerParams} params - Parameters returned by the server.\n */\n /**\n * Callback to report when connection with Tinode server is established.\n * @type {onConnect}\n */\n onConnect = undefined;\n\n /**\n * Callback to report when connection is lost. The callback has no parameters.\n * @type {onDisconnect}\n */\n onDisconnect = undefined;\n\n /**\n * @callback onLogin\n * @param {number} code - NUmeric completion code, same as HTTP status codes.\n * @param {string} text - Explanation of the completion code.\n */\n /**\n * Callback to report login completion.\n * @type {onLogin}\n */\n onLogin = undefined;\n\n /**\n * Callback to receive {ctrl}
(control) messages.\n * @type {onCtrlMessage}\n */\n onCtrlMessage = undefined;\n\n /**\n * Callback to recieve {data}
(content) messages.\n * @type {onDataMessage}\n */\n onDataMessage = undefined;\n\n /**\n * Callback to receive {pres}
(presence) messages.\n * @type {onPresMessage}\n */\n onPresMessage = undefined;\n\n /**\n * Callback to receive all messages as objects.\n * @type {onMessage}\n */\n onMessage = undefined;\n\n /**\n * Callback to receive all messages as unparsed text.\n * @type {onRawMessage}\n */\n onRawMessage = undefined;\n\n /**\n * Callback to receive server responses to network probes. See {@link Tinode#networkProbe}\n * @type {onNetworkProbe}\n */\n onNetworkProbe = undefined;\n\n /**\n * Callback to be notified when exponential backoff is iterating.\n * @type {onAutoreconnectIteration}\n */\n onAutoreconnectIteration = undefined;\n};\n\n// Exported constants\nTinode.MESSAGE_STATUS_NONE = Const.MESSAGE_STATUS_NONE;\nTinode.MESSAGE_STATUS_QUEUED = Const.MESSAGE_STATUS_QUEUED;\nTinode.MESSAGE_STATUS_SENDING = Const.MESSAGE_STATUS_SENDING;\nTinode.MESSAGE_STATUS_FAILED = Const.MESSAGE_STATUS_FAILED;\nTinode.MESSAGE_STATUS_FATAL = Const.MESSAGE_STATUS_FATAL;\nTinode.MESSAGE_STATUS_SENT = Const.MESSAGE_STATUS_SENT;\nTinode.MESSAGE_STATUS_RECEIVED = Const.MESSAGE_STATUS_RECEIVED;\nTinode.MESSAGE_STATUS_READ = Const.MESSAGE_STATUS_READ;\nTinode.MESSAGE_STATUS_TO_ME = Const.MESSAGE_STATUS_TO_ME;\n\n// Unicode [del] symbol.\nTinode.DEL_CHAR = Const.DEL_CHAR;\n\n// Names of keys to server-provided configuration limits.\nTinode.MAX_MESSAGE_SIZE = 'maxMessageSize';\nTinode.MAX_SUBSCRIBER_COUNT = 'maxSubscriberCount';\nTinode.MAX_TAG_COUNT = 'maxTagCount';\nTinode.MAX_FILE_UPLOAD_SIZE = 'maxFileUploadSize';\nTinode.REQ_CRED_VALIDATORS = 'reqCred';\n\n// Tinode URI topic ID prefix, 'scheme:path/'.\nTinode.URI_TOPIC_ID_PREFIX = 'tinode:topic/';\n"],"names":["AccessMode","constructor","acs","given","decode","want","mode","checkFlag","#checkFlag","val","side","flag","includes","Error","str","_BITMASK","_NONE","bitmask","_JOIN","_READ","_WRITE","_PRES","_APPROVE","_SHARE","_DELETE","_OWNER","m0","i","length","bit","charAt","toUpperCase","encode","_INVALID","res","update","upd","action","val0","parts","split","diff","a1","a2","toString","jsonHelper","setMode","m","updateMode","u","getMode","setGiven","g","updateGiven","getGiven","setWant","w","updateWant","getWant","getMissing","getExcessive","updateAll","isOwner","isPresencer","isMuted","isJoiner","isReader","isWriter","isApprover","isAdmin","isSharer","isDeleter","CBuffer","comparator","undefined","unique","buffer","compare_","unique_","a","b","findNearest","#findNearest","elem","arr","exact","start","end","pivot","found","idx","insertSorted","#insertSorted","count","splice","getAt","at","getLast","put","insert","arguments","Array","isArray","delAt","r","delRange","since","before","reset","forEach","callback","startIdx","beforeIdx","context","call","find","nearest","filter","isEmpty","CommError","message","code","name","PACKAGE_VERSION","PROTOCOL_VERSION","VERSION","LIBRARY","TOPIC_NEW","TOPIC_NEW_CHAN","TOPIC_ME","TOPIC_FND","TOPIC_SYS","TOPIC_CHAN","TOPIC_GRP","TOPIC_P2P","USER_NEW","LOCAL_SEQID","MESSAGE_STATUS_NONE","MESSAGE_STATUS_QUEUED","MESSAGE_STATUS_SENDING","MESSAGE_STATUS_FAILED","MESSAGE_STATUS_FATAL","MESSAGE_STATUS_SENT","MESSAGE_STATUS_RECEIVED","MESSAGE_STATUS_READ","MESSAGE_STATUS_TO_ME","EXPIRE_PROMISES_TIMEOUT","EXPIRE_PROMISES_PERIOD","RECV_TIMEOUT","DEFAULT_MESSAGES_PAGE","DEL_CHAR","jsonParseHelper","WebSocketProvider","XHRProvider","NETWORK_ERROR","NETWORK_ERROR_TEXT","NETWORK_USER","NETWORK_USER_TEXT","_BOFF_BASE","_BOFF_MAX_ITER","_BOFF_JITTER","makeBaseUrl","host","protocol","version","apiKey","url","Connection","log","_","boffTimer","boffIteration","boffClosed","socket","secure","autoreconnect","initialized","config","version_","autoreconnect_","transport","init_lp","init_ws","setNetworkProviders","wsProvider","xhrProvider","logger","l","connect","host_","force","Promise","reject","reconnect","disconnect","sendText","msg","isConnected","probe","backoffReset","boffReset","boffReconnect","#boffReconnect","clearTimeout","timeout","Math","pow","random","onAutoreconnectIteration","setTimeout","prom","catch","boffStop","#boffStop","#boffReset","#init_lp","XDR_UNSENT","XDR_OPENED","XDR_HEADERS_RECEIVED","XDR_LOADING","XDR_DONE","_lpURL","_poller","_sender","lp_sender","url_","sender","onreadystatechange","evt","readyState","status","open","lp_poller","resolve","poller","promiseCompleted","pkt","JSON","parse","responseText","ctrl","params","sid","send","onOpen","onMessage","onDisconnect","text","abort","err","#init_ws","OPEN","close","conn","onerror","onopen","onclose","onmessage","data","DB_VERSION","DB_NAME","IDBProvider","DB","onError","db","disabled","mapObjects","#mapObjects","source","trx","transaction","event","target","error","objectStore","getAll","onsuccess","result","topic","initDatabase","req","onupgradeneeded","createObjectStore","keyPath","deleteDatabase","onblocked","isReady","updTopic","oncomplete","get","serializeTopic","commit","markTopicAsDeleted","deleted","_deleted","remTopic","delete","IDBKeyRange","only","bound","Number","MAX_SAFE_INTEGER","mapTopics","deserializeTopic","src","updUser","uid","pub","public","remUser","mapUsers","getUser","user","updSubscription","topicName","sub","serializeSubscription","mapSubscriptions","addMessage","add","serializeMessage","updMessageStatus","seq","_status","remMessages","from","to","range","readMessages","query","limit","openCursor","cursor","value","push","continue","topic_fields","#deserializeTopic","f","hasOwnProperty","tags","_tags","setAccessMode","read","unread","max","#serializeTopic","dst","getAccessMode","#serializeSubscription","fields","#serializeMessage","setDatabaseProvider","idbProvider","MAX_FORM_ELEMENTS","MAX_PREVIEW_ATTACHMENTS","MAX_PREVIEW_DATA_SIZE","JSON_MIME_TYPE","DRAFTY_MIME_TYPE","ALLOWED_ENT_FIELDS","INLINE_STYLES","FMT_WEIGHT","ENTITY_TYPES","dataName","pack","test","re","slice","FORMAT_TAGS","AU","html_tag","md_tag","isVoid","BN","BR","CO","DL","EM","EX","FM","HD","HL","HT","IM","LN","MN","RW","QQ","ST","VC","VD","base64toObjectUrl","b64","contentType","bin","atob","buf","ArrayBuffer","Uint8Array","charCodeAt","URL","createObjectURL","Blob","type","base64toDataUrl","DECORATORS","props","href","id","act","ref","mime","Drafty","duration","size","tmpPreviewUrl","_tempPreview","previewUrl","downloadUrl","width","height","title","alt","state","preview","premime","poster","preref","txt","fmt","ent","init","plainText","content","lines","entityMap","entityIndex","blx","line","spans","entities","tag","concat","spannify","block","sort","toSpanTree","chunks","chunkify","drafty","draftify","extractEntities","ranges","entity","index","tp","offset","len","key","map","s","append","first","second","insertImage","imageDesc","ex","refurl","bits","filename","urlPromise","_processing","then","insertVideo","videoDesc","urls","insertAudio","audioDesc","videoCall","audioOnly","aonly","updateVideoCall","Object","assign","quote","header","body","appendLineBreak","mention","appendLink","linkData","appendImage","appendAudio","attachFile","attachmentDesc","wrapInto","style","wrapAsForm","insertButton","actionType","actionValue","refUrl","indexOf","appendButton","attachJSON","UNSAFE_toHTML","doc","tree","draftyToTree","htmlFormatter","values","join","treeBottomUp","format","original","formatter","shorten","light","shortenTree","lightEntity","treeToDrafty","forwardedContent","rmMention","node","parent","treeTopDown","lTrim","replyContent","convMNnQQnBR","startsWith","children","attachmentsToEnd","forwarding","toPlainText","isPlainText","toMarkdown","mdFormatter","def","isValid","txt_type","hasAttachments","attachments","hasEntities","styles","sanitizeEntities","copyEntData","getDownloadUrl","entData","isProcessing","getPreviewUrl","getEntitySize","getEntityMimeType","tagName","attrValue","getContentType","span","chunk","chld","re_start","re_end","exec","start_offset","lastIndexOf","end_offset","last","spansToTree","flatten","child","addNode","n","substring","att","subspans","inner","keymap","c","keys","newKey","transformer","stack","pop","tail","shortener","allow","lightCopy","trimStart","shift","match","extracted","el","startAt","plain","entries","dc","module","exports","isUrlRelative","addURLParam","relUrl","window","location","origin","searchParams","LargeFileHelper","tinode","_tinode","_version","_apiKey","_authToken","getAuthToken","xhr","uploadWithBaseUrl","baseUrl","avatarFor","onProgress","onSuccess","onFailure","base","endsWith","instance","setRequestHeader","token","toResolve","toReject","upload","onprogress","e","lengthComputable","loaded","total","onload","response","statusText","onabort","form","FormData","set","getNextUniqueId","_secure","_host","download","relativeUrl","mimetype","responseType","link","document","createElement","display","setAttribute","appendChild","click","removeChild","revokeObjectURL","reader","FileReader","readAsText","cancel","setNetworkProvider","MetaGetBuilder","what","get_desc_ims","#get_desc_ims","updated","get_subs_ims","#get_subs_ims","isP2PType","_lastSubsUpdate","withData","withLaterData","_maxSeq","withEarlierData","_minSeq","withDesc","ims","withLaterDesc","withSub","userOrTopic","opts","getType","withOneSub","withLaterOneSub","withLaterSub","withTags","withCred","withDel","withLaterDel","_maxDel","extract","build","getOwnPropertyNames","Const","mergeObj","mergeToCache","normalizeArray","Topic","callbacks","created","touched","Date","private","trusted","_users","_queuedSeqId","_noEarlierMsgs","_recvNotificationTimer","_credentials","_messageVersions","_messages","_attached","_new","_delayedLeaveTimer","onData","onMeta","onPres","onInfo","onMetaDesc","onMetaSub","onSubsUpdated","onTagsUpdated","onCredsUpdated","onDeleteTopic","onAllMessagesReceived","topicType","types","isMeTopicName","isGroupTopicName","isP2PTopicName","isCommTopicName","isNewGroupTopicName","isChannelTopicName","isSubscribed","subscribe","getParams","setParams","_cacheDelSelf","_cachePutSelf","ts","me","getMeTopic","desc","_noForwarding","_processMetaDesc","createMessage","noEcho","publish","publishMessage","_sending","_failed","swapMessageId","_maybeUpdateMessageVersionsCache","_routeData","publishDraft","_getQueuedSeqId","getCurrentUserID","noecho","_db","_cancelled","_fatal","leave","unsub","_resetSub","_gone","leaveDelayed","delay","getMeta","getMessagesPage","forward","startMetaQuery","_loadMessages","promise","setMeta","_processMetaSubs","_processMetaTags","cred","_processMetaCreds","subscriber","am","invite","archive","arch","delMessages","hard","r1","r2","low","hi","tosend","reduce","out","del","flushMessageRange","flushMessage","delMessagesAll","hardDel","_all","delMessagesList","list","prev","delMessagesEdits","messageVersions","delTopic","delSubscription","note","_updateMyReadRecv","_refreshContact","noteRecv","noteRead","noteKeyPress","noteRecording","payload","oldVal","doUpdate","recv","userDesc","_cacheGetUser","p2pPeerDesc","subscribers","cb","origSeq","versions","messages","sinceId","beforeId","msgs","unused1","unused2","_isReplacementMsg","latest","latestMsgVersion","_origTs","_origSeq","findMessage","latestMessage","maxMsgSeq","minMsgSeq","maxClearId","messageCount","queuedMessages","msgReceiptCount","msgReadCount","msgRecvCount","msgHasMoreMessages","newer","isNewMessage","seqId","fromId","untilId","newSeqId","numMessages","cancelSend","msgStatus","getDefaultAccess","defacs","isArchived","isMeType","isChannelType","isGroupType","isCommType","isMe","head","replace","targetSeq","parseInt","targetMsg","outgoing","webrtc","incoming","_routeInfo","_routeMeta","meta","_processDelMessages","clear","delseq","_routePres","pres","online","isTopicCached","dacs","info","subs","_updateCachedUser","creds","_allMessagesReceived","obj","cached","_cachePutUser","_updateReceived","TopicMe","onContactUpdate","turnOff","_myUID","cont","seen","when","updateCount","cacheRemTopic","getTopic","cr","findIndex","meth","done","resp","cacheGetTopic","tgt","ua","dummy","delCredential","method","contacts","getContact","getCredentials","TopicFnd","_contacts","indexBy","getPrototypeOf","prototype","date","isNaN","isValidDate","d","getTime","rfc3339DateString","pad","sp","repeat","millis","getUTCMilliseconds","getUTCFullYear","getUTCMonth","getUTCDate","getUTCHours","getUTCMinutes","getUTCSeconds","ignore","prop","cache","newval","simplify","t","trim","toLowerCase","item","pos","ary","DBCache","WebSocket","XMLHttpRequest","IndexedDBProvider","indexedDB","initForNonBrowserApp","chars","btoa","global","input","output","charCode","bc","bs","String","fromCharCode","detectTransport","b64EncodeUnicode","encodeURIComponent","toSolidBytes","p1","jsonBuildHelper","jsonLoggerHelper","getBrowserInfo","product","reactnative","priority","tmp","substr","tokens","m2","v","minor","Tinode","_appName","_browser","_platform","_hwos","_humanLanguage","_loggingEnabled","_trimLongStrings","_authenticated","_login","_inPacketCount","_messageId","floor","_serverInfo","_deviceToken","_pendingPromises","_expirePromises","_connection","_persist","_cache","onComplete","appName","platform","navigator","userAgent","language","dispatchMessage","connectionOpen","disconnected","persist","cacheGet","attachCacheToTopic","cachePut","all","dateString","_len","args","_key","console","makePromise","#makePromise","execPromise","#execPromise","onOK","errorText","#send","stringify","#dispatchMessage","onRawMessage","onNetworkProbe","onCtrlMessage","onMetaMessage","onDataMessage","onPresMessage","onInfoMessage","#connectionOpen","setInterval","expires","hello","#disconnected","clearInterval","cacheMap","getUserAgent","#getUserAgent","initPacket","#initPacket","#cachePut","#cacheGet","cacheDel","#cacheDel","#cacheMap","func","#attachCacheToTopic","_cacheDelUser","loginSuccessful","#loginSuccessful","onLogin","credential","getVersion","getLibrary","isNullValue","clearStorage","initStorage","networkProbe","isAuthenticated","authorizeURL","parsed","account","scheme","secret","login","acc","tmpscheme","tmpsecret","extra","createAccount","createAccountBasic","username","password","updateAccountBasic","onConnect","setDeviceToken","dt","sent","loginBasic","uname","loginToken","requestResetAuthSecret","now","setAuthToken","dft","oobNotification","xfrom","finally","modeGiven","modeWant","delCurrentUser","newGroupTopicName","isChan","getFndTopic","getLargeFileHelper","getCurrentLogin","getServerInfo","report","getServerParam","defaultValue","enableLogging","enabled","trimLongStrings","setHumanLanguage","hl","isTopicOnline","getTopicAccessMode","wantAkn","onWebsocketOpen","MAX_MESSAGE_SIZE","MAX_SUBSCRIBER_COUNT","MAX_TAG_COUNT","MAX_FILE_UPLOAD_SIZE","REQ_CRED_VALIDATORS","URI_TOPIC_ID_PREFIX"],"sourceRoot":""}
\ No newline at end of file
+{"version":3,"file":"tinode.dev.js","mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;AACD,O;;;;;;;;;;;;;;ACLa;AAcE,MAAMA,UAAU,CAAC;EAC9BC,WAAWA,CAACC,GAAG,EAAE;IACf,IAAIA,GAAG,EAAE;MACP,IAAI,CAACC,KAAK,GAAG,OAAOD,GAAG,CAACC,KAAK,IAAI,QAAQ,GAAGD,GAAG,CAACC,KAAK,GAAGH,UAAU,CAACI,MAAM,CAACF,GAAG,CAACC,KAAK,CAAC;MACpF,IAAI,CAACE,IAAI,GAAG,OAAOH,GAAG,CAACG,IAAI,IAAI,QAAQ,GAAGH,GAAG,CAACG,IAAI,GAAGL,UAAU,CAACI,MAAM,CAACF,GAAG,CAACG,IAAI,CAAC;MAChF,IAAI,CAACC,IAAI,GAAGJ,GAAG,CAACI,IAAI,GAAI,OAAOJ,GAAG,CAACI,IAAI,IAAI,QAAQ,GAAGJ,GAAG,CAACI,IAAI,GAAGN,UAAU,CAACI,MAAM,CAACF,GAAG,CAACI,IAAI,CAAC,GACzF,IAAI,CAACH,KAAK,GAAG,IAAI,CAACE,IAAK;IAC5B;EACF;EAEA,OAAO,CAACE,SAASC,CAACC,GAAG,EAAEC,IAAI,EAAEC,IAAI,EAAE;IACjCD,IAAI,GAAGA,IAAI,IAAI,MAAM;IACrB,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,CAAC,CAACE,QAAQ,CAACF,IAAI,CAAC,EAAE;MAC5C,OAAQ,CAACD,GAAG,CAACC,IAAI,CAAC,GAAGC,IAAI,KAAK,CAAC;IACjC;IACA,MAAM,IAAIE,KAAK,CAAE,iCAAgCH,IAAK,GAAE,CAAC;EAC3D;EASA,OAAON,MAAMA,CAACU,GAAG,EAAE;IACjB,IAAI,CAACA,GAAG,EAAE;MACR,OAAO,IAAI;IACb,CAAC,MAAM,IAAI,OAAOA,GAAG,IAAI,QAAQ,EAAE;MACjC,OAAOA,GAAG,GAAGd,UAAU,CAACe,QAAQ;IAClC,CAAC,MAAM,IAAID,GAAG,KAAK,GAAG,IAAIA,GAAG,KAAK,GAAG,EAAE;MACrC,OAAOd,UAAU,CAACgB,KAAK;IACzB;IAEA,MAAMC,OAAO,GAAG;MACd,GAAG,EAAEjB,UAAU,CAACkB,KAAK;MACrB,GAAG,EAAElB,UAAU,CAACmB,KAAK;MACrB,GAAG,EAAEnB,UAAU,CAACoB,MAAM;MACtB,GAAG,EAAEpB,UAAU,CAACqB,KAAK;MACrB,GAAG,EAAErB,UAAU,CAACsB,QAAQ;MACxB,GAAG,EAAEtB,UAAU,CAACuB,MAAM;MACtB,GAAG,EAAEvB,UAAU,CAACwB,OAAO;MACvB,GAAG,EAAExB,UAAU,CAACyB;IAClB,CAAC;IAED,IAAIC,EAAE,GAAG1B,UAAU,CAACgB,KAAK;IAEzB,KAAK,IAAIW,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGb,GAAG,CAACc,MAAM,EAAED,CAAC,EAAE,EAAE;MACnC,MAAME,GAAG,GAAGZ,OAAO,CAACH,GAAG,CAACgB,MAAM,CAACH,CAAC,CAAC,CAACI,WAAW,CAAC,CAAC,CAAC;MAChD,IAAI,CAACF,GAAG,EAAE;QAER;MACF;MACAH,EAAE,IAAIG,GAAG;IACX;IACA,OAAOH,EAAE;EACX;EAUA,OAAOM,MAAMA,CAACvB,GAAG,EAAE;IACjB,IAAIA,GAAG,KAAK,IAAI,IAAIA,GAAG,KAAKT,UAAU,CAACiC,QAAQ,EAAE;MAC/C,OAAO,IAAI;IACb,CAAC,MAAM,IAAIxB,GAAG,KAAKT,UAAU,CAACgB,KAAK,EAAE;MACnC,OAAO,GAAG;IACZ;IAEA,MAAMC,OAAO,GAAG,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC;IACxD,IAAIiB,GAAG,GAAG,EAAE;IACZ,KAAK,IAAIP,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGV,OAAO,CAACW,MAAM,EAAED,CAAC,EAAE,EAAE;MACvC,IAAI,CAAClB,GAAG,GAAI,CAAC,IAAIkB,CAAE,KAAK,CAAC,EAAE;QACzBO,GAAG,GAAGA,GAAG,GAAGjB,OAAO,CAACU,CAAC,CAAC;MACxB;IACF;IACA,OAAOO,GAAG;EACZ;EAcA,OAAOC,MAAMA,CAAC1B,GAAG,EAAE2B,GAAG,EAAE;IACtB,IAAI,CAACA,GAAG,IAAI,OAAOA,GAAG,IAAI,QAAQ,EAAE;MAClC,OAAO3B,GAAG;IACZ;IAEA,IAAI4B,MAAM,GAAGD,GAAG,CAACN,MAAM,CAAC,CAAC,CAAC;IAC1B,IAAIO,MAAM,IAAI,GAAG,IAAIA,MAAM,IAAI,GAAG,EAAE;MAClC,IAAIC,IAAI,GAAG7B,GAAG;MAEd,MAAM8B,KAAK,GAAGH,GAAG,CAACI,KAAK,CAAC,QAAQ,CAAC;MAGjC,KAAK,IAAIb,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGY,KAAK,CAACX,MAAM,GAAG,CAAC,EAAED,CAAC,IAAI,CAAC,EAAE;QAC5CU,MAAM,GAAGE,KAAK,CAACZ,CAAC,CAAC;QACjB,MAAMD,EAAE,GAAG1B,UAAU,CAACI,MAAM,CAACmC,KAAK,CAACZ,CAAC,GAAG,CAAC,CAAC,CAAC;QAC1C,IAAID,EAAE,IAAI1B,UAAU,CAACiC,QAAQ,EAAE;UAC7B,OAAOxB,GAAG;QACZ;QACA,IAAIiB,EAAE,IAAI,IAAI,EAAE;UACd;QACF;QACA,IAAIW,MAAM,KAAK,GAAG,EAAE;UAClBC,IAAI,IAAIZ,EAAE;QACZ,CAAC,MAAM,IAAIW,MAAM,KAAK,GAAG,EAAE;UACzBC,IAAI,IAAI,CAACZ,EAAE;QACb;MACF;MACAjB,GAAG,GAAG6B,IAAI;IACZ,CAAC,MAAM;MAEL,MAAMA,IAAI,GAAGtC,UAAU,CAACI,MAAM,CAACgC,GAAG,CAAC;MACnC,IAAIE,IAAI,IAAItC,UAAU,CAACiC,QAAQ,EAAE;QAC/BxB,GAAG,GAAG6B,IAAI;MACZ;IACF;IAEA,OAAO7B,GAAG;EACZ;EAWA,OAAOgC,IAAIA,CAACC,EAAE,EAAEC,EAAE,EAAE;IAClBD,EAAE,GAAG1C,UAAU,CAACI,MAAM,CAACsC,EAAE,CAAC;IAC1BC,EAAE,GAAG3C,UAAU,CAACI,MAAM,CAACuC,EAAE,CAAC;IAE1B,IAAID,EAAE,IAAI1C,UAAU,CAACiC,QAAQ,IAAIU,EAAE,IAAI3C,UAAU,CAACiC,QAAQ,EAAE;MAC1D,OAAOjC,UAAU,CAACiC,QAAQ;IAC5B;IACA,OAAOS,EAAE,GAAG,CAACC,EAAE;EACjB;EAUAC,QAAQA,CAAA,EAAG;IACT,OAAO,YAAY,GAAG5C,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC1B,IAAI,CAAC,GAChD,eAAe,GAAGN,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC7B,KAAK,CAAC,GAC/C,cAAc,GAAGH,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC3B,IAAI,CAAC,GAAG,IAAI;EACxD;EAUAwC,UAAUA,CAAA,EAAG;IACX,OAAO;MACLvC,IAAI,EAAEN,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC1B,IAAI,CAAC;MAClCH,KAAK,EAAEH,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC7B,KAAK,CAAC;MACpCE,IAAI,EAAEL,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC3B,IAAI;IACnC,CAAC;EACH;EAcAyC,OAAOA,CAACC,CAAC,EAAE;IACT,IAAI,CAACzC,IAAI,GAAGN,UAAU,CAACI,MAAM,CAAC2C,CAAC,CAAC;IAChC,OAAO,IAAI;EACb;EAcAC,UAAUA,CAACC,CAAC,EAAE;IACZ,IAAI,CAAC3C,IAAI,GAAGN,UAAU,CAACmC,MAAM,CAAC,IAAI,CAAC7B,IAAI,EAAE2C,CAAC,CAAC;IAC3C,OAAO,IAAI;EACb;EAaAC,OAAOA,CAAA,EAAG;IACR,OAAOlD,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC1B,IAAI,CAAC;EACrC;EAcA6C,QAAQA,CAACC,CAAC,EAAE;IACV,IAAI,CAACjD,KAAK,GAAGH,UAAU,CAACI,MAAM,CAACgD,CAAC,CAAC;IACjC,OAAO,IAAI;EACb;EAcAC,WAAWA,CAACJ,CAAC,EAAE;IACb,IAAI,CAAC9C,KAAK,GAAGH,UAAU,CAACmC,MAAM,CAAC,IAAI,CAAChC,KAAK,EAAE8C,CAAC,CAAC;IAC7C,OAAO,IAAI;EACb;EAaAK,QAAQA,CAAA,EAAG;IACT,OAAOtD,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC7B,KAAK,CAAC;EACtC;EAcAoD,OAAOA,CAACC,CAAC,EAAE;IACT,IAAI,CAACnD,IAAI,GAAGL,UAAU,CAACI,MAAM,CAACoD,CAAC,CAAC;IAChC,OAAO,IAAI;EACb;EAcAC,UAAUA,CAACR,CAAC,EAAE;IACZ,IAAI,CAAC5C,IAAI,GAAGL,UAAU,CAACmC,MAAM,CAAC,IAAI,CAAC9B,IAAI,EAAE4C,CAAC,CAAC;IAC3C,OAAO,IAAI;EACb;EAaAS,OAAOA,CAAA,EAAG;IACR,OAAO1D,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC3B,IAAI,CAAC;EACrC;EAeAsD,UAAUA,CAAA,EAAG;IACX,OAAO3D,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC3B,IAAI,GAAG,CAAC,IAAI,CAACF,KAAK,CAAC;EACnD;EAcAyD,YAAYA,CAAA,EAAG;IACb,OAAO5D,UAAU,CAACgC,MAAM,CAAC,IAAI,CAAC7B,KAAK,GAAG,CAAC,IAAI,CAACE,IAAI,CAAC;EACnD;EAcAwD,SAASA,CAACpD,GAAG,EAAE;IACb,IAAIA,GAAG,EAAE;MACP,IAAI,CAAC4C,WAAW,CAAC5C,GAAG,CAACN,KAAK,CAAC;MAC3B,IAAI,CAACsD,UAAU,CAAChD,GAAG,CAACJ,IAAI,CAAC;MACzB,IAAI,CAACC,IAAI,GAAG,IAAI,CAACH,KAAK,GAAG,IAAI,CAACE,IAAI;IACpC;IACA,OAAO,IAAI;EACb;EAaAyD,OAAOA,CAACpD,IAAI,EAAE;IACZ,OAAOV,UAAU,CAAC,CAACO,SAAS,CAAC,IAAI,EAAEG,IAAI,EAAEV,UAAU,CAACyB,MAAM,CAAC;EAC7D;EAaAsC,WAAWA,CAACrD,IAAI,EAAE;IAChB,OAAOV,UAAU,CAAC,CAACO,SAAS,CAAC,IAAI,EAAEG,IAAI,EAAEV,UAAU,CAACqB,KAAK,CAAC;EAC5D;EAaA2C,OAAOA,CAACtD,IAAI,EAAE;IACZ,OAAO,CAAC,IAAI,CAACqD,WAAW,CAACrD,IAAI,CAAC;EAChC;EAaAuD,QAAQA,CAACvD,IAAI,EAAE;IACb,OAAOV,UAAU,CAAC,CAACO,SAAS,CAAC,IAAI,EAAEG,IAAI,EAAEV,UAAU,CAACkB,KAAK,CAAC;EAC5D;EAaAgD,QAAQA,CAACxD,IAAI,EAAE;IACb,OAAOV,UAAU,CAAC,CAACO,SAAS,CAAC,IAAI,EAAEG,IAAI,EAAEV,UAAU,CAACmB,KAAK,CAAC;EAC5D;EAaAgD,QAAQA,CAACzD,IAAI,EAAE;IACb,OAAOV,UAAU,CAAC,CAACO,SAAS,CAAC,IAAI,EAAEG,IAAI,EAAEV,UAAU,CAACoB,MAAM,CAAC;EAC7D;EAaAgD,UAAUA,CAAC1D,IAAI,EAAE;IACf,OAAOV,UAAU,CAAC,CAACO,SAAS,CAAC,IAAI,EAAEG,IAAI,EAAEV,UAAU,CAACsB,QAAQ,CAAC;EAC/D;EAaA+C,OAAOA,CAAC3D,IAAI,EAAE;IACZ,OAAO,IAAI,CAACoD,OAAO,CAACpD,IAAI,CAAC,IAAI,IAAI,CAAC0D,UAAU,CAAC1D,IAAI,CAAC;EACpD;EAaA4D,QAAQA,CAAC5D,IAAI,EAAE;IACb,OAAO,IAAI,CAAC2D,OAAO,CAAC3D,IAAI,CAAC,IAAIV,UAAU,CAAC,CAACO,SAAS,CAAC,IAAI,EAAEG,IAAI,EAAEV,UAAU,CAACuB,MAAM,CAAC;EACnF;EAaAgD,SAASA,CAAC7D,IAAI,EAAE;IACd,OAAOV,UAAU,CAAC,CAACO,SAAS,CAAC,IAAI,EAAEG,IAAI,EAAEV,UAAU,CAACwB,OAAO,CAAC;EAC9D;AACF;AAEAxB,UAAU,CAACgB,KAAK,GAAG,IAAI;AACvBhB,UAAU,CAACkB,KAAK,GAAG,IAAI;AACvBlB,UAAU,CAACmB,KAAK,GAAG,IAAI;AACvBnB,UAAU,CAACoB,MAAM,GAAG,IAAI;AACxBpB,UAAU,CAACqB,KAAK,GAAG,IAAI;AACvBrB,UAAU,CAACsB,QAAQ,GAAG,IAAI;AAC1BtB,UAAU,CAACuB,MAAM,GAAG,IAAI;AACxBvB,UAAU,CAACwB,OAAO,GAAG,IAAI;AACzBxB,UAAU,CAACyB,MAAM,GAAG,IAAI;AAExBzB,UAAU,CAACe,QAAQ,GAAGf,UAAU,CAACkB,KAAK,GAAGlB,UAAU,CAACmB,KAAK,GAAGnB,UAAU,CAACoB,MAAM,GAAGpB,UAAU,CAACqB,KAAK,GAC9FrB,UAAU,CAACsB,QAAQ,GAAGtB,UAAU,CAACuB,MAAM,GAAGvB,UAAU,CAACwB,OAAO,GAAGxB,UAAU,CAACyB,MAAM;AAClFzB,UAAU,CAACiC,QAAQ,GAAG,QAAQ;;;;;;;;;;;;;;ACjjBjB;AAcE,MAAMuC,OAAO,CAAC;EAC3B,CAACC,UAAU,GAAGC,SAAS;EACvB,CAACC,MAAM,GAAG,KAAK;EACfC,MAAM,GAAG,EAAE;EAEX3E,WAAWA,CAAC4E,QAAQ,EAAEC,OAAO,EAAE;IAC7B,IAAI,CAAC,CAACL,UAAU,GAAGI,QAAQ,KAAK,CAACE,CAAC,EAAEC,CAAC,KAAK;MACxC,OAAOD,CAAC,KAAKC,CAAC,GAAG,CAAC,GAAGD,CAAC,GAAGC,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC;IACrC,CAAC,CAAC;IACF,IAAI,CAAC,CAACL,MAAM,GAAGG,OAAO;EACxB;EAEA,CAACG,WAAWC,CAACC,IAAI,EAAEC,GAAG,EAAEC,KAAK,EAAE;IAC7B,IAAIC,KAAK,GAAG,CAAC;IACb,IAAIC,GAAG,GAAGH,GAAG,CAACxD,MAAM,GAAG,CAAC;IACxB,IAAI4D,KAAK,GAAG,CAAC;IACb,IAAI/C,IAAI,GAAG,CAAC;IACZ,IAAIgD,KAAK,GAAG,KAAK;IAEjB,OAAOH,KAAK,IAAIC,GAAG,EAAE;MACnBC,KAAK,GAAG,CAACF,KAAK,GAAGC,GAAG,IAAI,CAAC,GAAG,CAAC;MAC7B9C,IAAI,GAAG,IAAI,CAAC,CAACgC,UAAU,CAACW,GAAG,CAACI,KAAK,CAAC,EAAEL,IAAI,CAAC;MACzC,IAAI1C,IAAI,GAAG,CAAC,EAAE;QACZ6C,KAAK,GAAGE,KAAK,GAAG,CAAC;MACnB,CAAC,MAAM,IAAI/C,IAAI,GAAG,CAAC,EAAE;QACnB8C,GAAG,GAAGC,KAAK,GAAG,CAAC;MACjB,CAAC,MAAM;QACLC,KAAK,GAAG,IAAI;QACZ;MACF;IACF;IACA,IAAIA,KAAK,EAAE;MACT,OAAO;QACLC,GAAG,EAAEF,KAAK;QACVH,KAAK,EAAE;MACT,CAAC;IACH;IACA,IAAIA,KAAK,EAAE;MACT,OAAO;QACLK,GAAG,EAAE,CAAC;MACR,CAAC;IACH;IAEA,OAAO;MACLA,GAAG,EAAEjD,IAAI,GAAG,CAAC,GAAG+C,KAAK,GAAG,CAAC,GAAGA;IAC9B,CAAC;EACH;EAGA,CAACG,YAAYC,CAACT,IAAI,EAAEC,GAAG,EAAE;IACvB,MAAMK,KAAK,GAAG,IAAI,CAAC,CAACR,WAAW,CAACE,IAAI,EAAEC,GAAG,EAAE,KAAK,CAAC;IACjD,MAAMS,KAAK,GAAIJ,KAAK,CAACJ,KAAK,IAAI,IAAI,CAAC,CAACV,MAAM,GAAI,CAAC,GAAG,CAAC;IACnDS,GAAG,CAACU,MAAM,CAACL,KAAK,CAACC,GAAG,EAAEG,KAAK,EAAEV,IAAI,CAAC;IAClC,OAAOC,GAAG;EACZ;EAQAW,KAAKA,CAACC,EAAE,EAAE;IACR,OAAO,IAAI,CAACpB,MAAM,CAACoB,EAAE,CAAC;EACxB;EASAC,OAAOA,CAACD,EAAE,EAAE;IACVA,EAAE,IAAI,CAAC;IACP,OAAO,IAAI,CAACpB,MAAM,CAAChD,MAAM,GAAGoE,EAAE,GAAG,IAAI,CAACpB,MAAM,CAAC,IAAI,CAACA,MAAM,CAAChD,MAAM,GAAG,CAAC,GAAGoE,EAAE,CAAC,GAAGtB,SAAS;EACvF;EASAwB,GAAGA,CAAA,EAAG;IACJ,IAAIC,MAAM;IAEV,IAAIC,SAAS,CAACxE,MAAM,IAAI,CAAC,IAAIyE,KAAK,CAACC,OAAO,CAACF,SAAS,CAAC,CAAC,CAAC,CAAC,EAAE;MACxDD,MAAM,GAAGC,SAAS,CAAC,CAAC,CAAC;IACvB,CAAC,MAAM;MACLD,MAAM,GAAGC,SAAS;IACpB;IACA,KAAK,IAAIV,GAAG,IAAIS,MAAM,EAAE;MACtB,IAAI,CAAC,CAACR,YAAY,CAACQ,MAAM,CAACT,GAAG,CAAC,EAAE,IAAI,CAACd,MAAM,CAAC;IAC9C;EACF;EAQA2B,KAAKA,CAACP,EAAE,EAAE;IACRA,EAAE,IAAI,CAAC;IACP,IAAIQ,CAAC,GAAG,IAAI,CAAC5B,MAAM,CAACkB,MAAM,CAACE,EAAE,EAAE,CAAC,CAAC;IACjC,IAAIQ,CAAC,IAAIA,CAAC,CAAC5E,MAAM,GAAG,CAAC,EAAE;MACrB,OAAO4E,CAAC,CAAC,CAAC,CAAC;IACb;IACA,OAAO9B,SAAS;EAClB;EAUA+B,QAAQA,CAACC,KAAK,EAAEC,MAAM,EAAE;IACtB,OAAO,IAAI,CAAC/B,MAAM,CAACkB,MAAM,CAACY,KAAK,EAAEC,MAAM,GAAGD,KAAK,CAAC;EAClD;EAOA9E,MAAMA,CAAA,EAAG;IACP,OAAO,IAAI,CAACgD,MAAM,CAAChD,MAAM;EAC3B;EAMAgF,KAAKA,CAAA,EAAG;IACN,IAAI,CAAChC,MAAM,GAAG,EAAE;EAClB;EAqBAiC,OAAOA,CAACC,QAAQ,EAAEC,QAAQ,EAAEC,SAAS,EAAEC,OAAO,EAAE;IAC9CF,QAAQ,GAAGA,QAAQ,GAAG,CAAC;IACvBC,SAAS,GAAGA,SAAS,IAAI,IAAI,CAACpC,MAAM,CAAChD,MAAM;IAE3C,KAAK,IAAID,CAAC,GAAGoF,QAAQ,EAAEpF,CAAC,GAAGqF,SAAS,EAAErF,CAAC,EAAE,EAAE;MACzCmF,QAAQ,CAACI,IAAI,CAACD,OAAO,EAAE,IAAI,CAACrC,MAAM,CAACjD,CAAC,CAAC,EAClCA,CAAC,GAAGoF,QAAQ,GAAG,IAAI,CAACnC,MAAM,CAACjD,CAAC,GAAG,CAAC,CAAC,GAAG+C,SAAS,EAC7C/C,CAAC,GAAGqF,SAAS,GAAG,CAAC,GAAG,IAAI,CAACpC,MAAM,CAACjD,CAAC,GAAG,CAAC,CAAC,GAAG+C,SAAS,EAAG/C,CAAC,CAAC;IAC5D;EACF;EAUAwF,IAAIA,CAAChC,IAAI,EAAEiC,OAAO,EAAE;IAClB,MAAM;MACJ1B;IACF,CAAC,GAAG,IAAI,CAAC,CAACT,WAAW,CAACE,IAAI,EAAE,IAAI,CAACP,MAAM,EAAE,CAACwC,OAAO,CAAC;IAClD,OAAO1B,GAAG;EACZ;EAkBA2B,MAAMA,CAACP,QAAQ,EAAEG,OAAO,EAAE;IACxB,IAAIpB,KAAK,GAAG,CAAC;IACb,KAAK,IAAIlE,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAG,IAAI,CAACiD,MAAM,CAAChD,MAAM,EAAED,CAAC,EAAE,EAAE;MAC3C,IAAImF,QAAQ,CAACI,IAAI,CAACD,OAAO,EAAE,IAAI,CAACrC,MAAM,CAACjD,CAAC,CAAC,EAAEA,CAAC,CAAC,EAAE;QAC7C,IAAI,CAACiD,MAAM,CAACiB,KAAK,CAAC,GAAG,IAAI,CAACjB,MAAM,CAACjD,CAAC,CAAC;QACnCkE,KAAK,EAAE;MACT;IACF;IAEA,IAAI,CAACjB,MAAM,CAACkB,MAAM,CAACD,KAAK,CAAC;EAC3B;EAMAyB,OAAOA,CAAA,EAAG;IACR,OAAO,IAAI,CAAC1C,MAAM,CAAChD,MAAM,IAAI,CAAC;EAChC;AACF;;;;;;;;;;;;;;AC5Oa;;AAEE,MAAM2F,SAAS,SAAS1G,KAAK,CAAC;EAC3CZ,WAAWA,CAACuH,OAAO,EAAEC,IAAI,EAAE;IACzB,KAAK,CAAE,GAAED,OAAQ,KAAIC,IAAK,GAAE,CAAC;IAC7B,IAAI,CAACC,IAAI,GAAG,WAAW;IACvB,IAAI,CAACD,IAAI,GAAGA,IAAI;EAClB;AACF;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACRa;;AAIU;AAGhB,MAAMG,gBAAgB,GAAG,GAAG;AAC5B,MAAMC,OAAO,GAAGF,wDAAe,IAAI,MAAM;AACzC,MAAMG,OAAO,GAAG,WAAW,GAAGD,OAAO;AAGrC,MAAME,SAAS,GAAG,KAAK;AACvB,MAAMC,cAAc,GAAG,KAAK;AAC5B,MAAMC,QAAQ,GAAG,IAAI;AACrB,MAAMC,SAAS,GAAG,KAAK;AACvB,MAAMC,SAAS,GAAG,KAAK;AACvB,MAAMC,UAAU,GAAG,KAAK;AACxB,MAAMC,SAAS,GAAG,KAAK;AACvB,MAAMC,SAAS,GAAG,KAAK;AACvB,MAAMC,QAAQ,GAAG,KAAK;AAGtB,MAAMC,WAAW,GAAG,SAAS;AAG7B,MAAMC,mBAAmB,GAAG,CAAC;AAC7B,MAAMC,qBAAqB,GAAG,EAAE;AAChC,MAAMC,sBAAsB,GAAG,EAAE;AACjC,MAAMC,qBAAqB,GAAG,EAAE;AAChC,MAAMC,oBAAoB,GAAG,EAAE;AAC/B,MAAMC,mBAAmB,GAAG,EAAE;AAC9B,MAAMC,uBAAuB,GAAG,EAAE;AAClC,MAAMC,mBAAmB,GAAG,EAAE;AAC9B,MAAMC,oBAAoB,GAAG,EAAE;AAG/B,MAAMC,uBAAuB,GAAG,IAAK;AAErC,MAAMC,sBAAsB,GAAG,IAAK;AAGpC,MAAMC,YAAY,GAAG,GAAG;AAGxB,MAAMC,qBAAqB,GAAG,EAAE;AAGhC,MAAMC,QAAQ,GAAG,QAAQ;;;;;;;;;;;;;;;;AChDnB;;AAE2B;AAGpB;AAEpB,IAAIE,iBAAiB;AACrB,IAAIC,WAAW;AAGf,MAAMC,aAAa,GAAG,GAAG;AACzB,MAAMC,kBAAkB,GAAG,mBAAmB;AAG9C,MAAMC,YAAY,GAAG,GAAG;AACxB,MAAMC,iBAAiB,GAAG,wBAAwB;AAGlD,MAAMC,UAAU,GAAG,IAAI;AACvB,MAAMC,cAAc,GAAG,EAAE;AACzB,MAAMC,YAAY,GAAG,GAAG;AAGxB,SAASC,WAAWA,CAACC,IAAI,EAAEC,QAAQ,EAAEC,OAAO,EAAEC,MAAM,EAAE;EACpD,IAAIC,GAAG,GAAG,IAAI;EAEd,IAAI,CAAC,MAAM,EAAE,OAAO,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC1J,QAAQ,CAACuJ,QAAQ,CAAC,EAAE;IACrDG,GAAG,GAAI,GAAEH,QAAS,MAAKD,IAAK,EAAC;IAC7B,IAAII,GAAG,CAACxI,MAAM,CAACwI,GAAG,CAAC1I,MAAM,GAAG,CAAC,CAAC,KAAK,GAAG,EAAE;MACtC0I,GAAG,IAAI,GAAG;IACZ;IACAA,GAAG,IAAI,GAAG,GAAGF,OAAO,GAAG,WAAW;IAClC,IAAI,CAAC,MAAM,EAAE,OAAO,CAAC,CAACxJ,QAAQ,CAACuJ,QAAQ,CAAC,EAAE;MAGxCG,GAAG,IAAI,KAAK;IACd;IACAA,GAAG,IAAI,UAAU,GAAGD,MAAM;EAC5B;EACA,OAAOC,GAAG;AACZ;AAiBe,MAAMC,UAAU,CAAC;EAE9B,OAAO,CAACC,GAAG,GAAGC,CAAC,IAAI,CAAC,CAAC;EAErB,CAACC,SAAS,GAAG,IAAI;EACjB,CAACC,aAAa,GAAG,CAAC;EAClB,CAACC,UAAU,GAAG,KAAK;EAGnB,CAACC,MAAM,GAAG,IAAI;EAEdX,IAAI;EACJY,MAAM;EACNT,MAAM;EAEND,OAAO;EACPW,aAAa;EAEbC,WAAW;EAGX/K,WAAWA,CAACgL,MAAM,EAAEC,QAAQ,EAAEC,cAAc,EAAE;IAC5C,IAAI,CAACjB,IAAI,GAAGe,MAAM,CAACf,IAAI;IACvB,IAAI,CAACY,MAAM,GAAGG,MAAM,CAACH,MAAM;IAC3B,IAAI,CAACT,MAAM,GAAGY,MAAM,CAACZ,MAAM;IAE3B,IAAI,CAACD,OAAO,GAAGc,QAAQ;IACvB,IAAI,CAACH,aAAa,GAAGI,cAAc;IAEnC,IAAIF,MAAM,CAACG,SAAS,KAAK,IAAI,EAAE;MAE7B,IAAI,CAAC,CAACC,OAAO,CAAC,CAAC;MACf,IAAI,CAACL,WAAW,GAAG,IAAI;IACzB,CAAC,MAAM,IAAIC,MAAM,CAACG,SAAS,KAAK,IAAI,EAAE;MAGpC,IAAI,CAAC,CAACE,OAAO,CAAC,CAAC;MACf,IAAI,CAACN,WAAW,GAAG,IAAI;IACzB;IAEA,IAAI,CAAC,IAAI,CAACA,WAAW,EAAE;MAErBT,UAAU,CAAC,CAACC,GAAG,CAAC,gGAAgG,CAAC;MACjH,MAAM,IAAI3J,KAAK,CAAC,gGAAgG,CAAC;IACnH;EACF;EASA,OAAO0K,mBAAmBA,CAACC,UAAU,EAAEC,WAAW,EAAE;IAClDjC,iBAAiB,GAAGgC,UAAU;IAC9B/B,WAAW,GAAGgC,WAAW;EAC3B;EAQA,WAAWC,MAAMA,CAACC,CAAC,EAAE;IACnBpB,UAAU,CAAC,CAACC,GAAG,GAAGmB,CAAC;EACrB;EAUAC,OAAOA,CAACC,KAAK,EAAEC,KAAK,EAAE;IACpB,OAAOC,OAAO,CAACC,MAAM,CAAC,IAAI,CAAC;EAC7B;EAQAC,SAASA,CAACH,KAAK,EAAE,CAAC;EAMlBI,UAAUA,CAAA,EAAG,CAAC;EASdC,QAAQA,CAACC,GAAG,EAAE,CAAC;EAOfC,WAAWA,CAAA,EAAG;IACZ,OAAO,KAAK;EACd;EAOAjB,SAASA,CAAA,EAAG;IACV,OAAO,IAAI,CAACJ,WAAW;EACzB;EAMAsB,KAAKA,CAAA,EAAG;IACN,IAAI,CAACH,QAAQ,CAAC,GAAG,CAAC;EACpB;EAMAI,YAAYA,CAAA,EAAG;IACb,IAAI,CAAC,CAACC,SAAS,CAAC,CAAC;EACnB;EAGA,CAACC,aAAaC,CAAA,EAAG;IAEfC,YAAY,CAAC,IAAI,CAAC,CAACjC,SAAS,CAAC;IAE7B,MAAMkC,OAAO,GAAG9C,UAAU,IAAI+C,IAAI,CAACC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,CAACnC,aAAa,CAAC,IAAI,GAAG,GAAGX,YAAY,GAAG6C,IAAI,CAACE,MAAM,CAAC,CAAC,CAAC,CAAC;IAEtG,IAAI,CAAC,CAACpC,aAAa,GAAI,IAAI,CAAC,CAACA,aAAa,IAAIZ,cAAc,GAAG,IAAI,CAAC,CAACY,aAAa,GAAG,IAAI,CAAC,CAACA,aAAa,GAAG,CAAE;IAC7G,IAAI,IAAI,CAACqC,wBAAwB,EAAE;MACjC,IAAI,CAACA,wBAAwB,CAACJ,OAAO,CAAC;IACxC;IAEA,IAAI,CAAC,CAAClC,SAAS,GAAGuC,UAAU,CAACxC,CAAC,IAAI;MAChCF,UAAU,CAAC,CAACC,GAAG,CAAE,sBAAqB,IAAI,CAAC,CAACG,aAAc,aAAYiC,OAAQ,EAAC,CAAC;MAEhF,IAAI,CAAC,IAAI,CAAC,CAAChC,UAAU,EAAE;QACrB,MAAMsC,IAAI,GAAG,IAAI,CAACtB,OAAO,CAAC,CAAC;QAC3B,IAAI,IAAI,CAACoB,wBAAwB,EAAE;UACjC,IAAI,CAACA,wBAAwB,CAAC,CAAC,EAAEE,IAAI,CAAC;QACxC,CAAC,MAAM;UAELA,IAAI,CAACC,KAAK,CAAC1C,CAAC,IAAI,CAEhB,CAAC,CAAC;QACJ;MACF,CAAC,MAAM,IAAI,IAAI,CAACuC,wBAAwB,EAAE;QACxC,IAAI,CAACA,wBAAwB,CAAC,CAAC,CAAC,CAAC;MACnC;IACF,CAAC,EAAEJ,OAAO,CAAC;EACb;EAGA,CAACQ,QAAQC,CAAA,EAAG;IACVV,YAAY,CAAC,IAAI,CAAC,CAACjC,SAAS,CAAC;IAC7B,IAAI,CAAC,CAACA,SAAS,GAAG,IAAI;EACxB;EAGA,CAAC8B,SAASc,CAAA,EAAG;IACX,IAAI,CAAC,CAAC3C,aAAa,GAAG,CAAC;EACzB;EAGA,CAACU,OAAOkC,CAAA,EAAG;IACT,MAAMC,UAAU,GAAG,CAAC;IACpB,MAAMC,UAAU,GAAG,CAAC;IACpB,MAAMC,oBAAoB,GAAG,CAAC;IAC9B,MAAMC,WAAW,GAAG,CAAC;IACrB,MAAMC,QAAQ,GAAG,CAAC;IAGlB,IAAIC,MAAM,GAAG,IAAI;IAEjB,IAAIC,OAAO,GAAG,IAAI;IAClB,IAAIC,OAAO,GAAG,IAAI;IAElB,IAAIC,SAAS,GAAIC,IAAI,IAAK;MACxB,MAAMC,MAAM,GAAG,IAAIzE,WAAW,CAAC,CAAC;MAChCyE,MAAM,CAACC,kBAAkB,GAAIC,GAAG,IAAK;QACnC,IAAIF,MAAM,CAACG,UAAU,IAAIT,QAAQ,IAAIM,MAAM,CAACI,MAAM,IAAI,GAAG,EAAE;UAEzD,MAAM,IAAI/G,sDAAS,CAAC,kBAAkB,EAAE2G,MAAM,CAACI,MAAM,CAAC;QACxD;MACF,CAAC;MAEDJ,MAAM,CAACK,IAAI,CAAC,MAAM,EAAEN,IAAI,EAAE,IAAI,CAAC;MAC/B,OAAOC,MAAM;IACf,CAAC;IAED,IAAIM,SAAS,GAAGA,CAACP,IAAI,EAAEQ,OAAO,EAAEzC,MAAM,KAAK;MACzC,IAAI0C,MAAM,GAAG,IAAIjF,WAAW,CAAC,CAAC;MAC9B,IAAIkF,gBAAgB,GAAG,KAAK;MAE5BD,MAAM,CAACP,kBAAkB,GAAGC,GAAG,IAAI;QACjC,IAAIM,MAAM,CAACL,UAAU,IAAIT,QAAQ,EAAE;UACjC,IAAIc,MAAM,CAACJ,MAAM,IAAI,GAAG,EAAE;YACxB,IAAIM,GAAG,GAAGC,IAAI,CAACC,KAAK,CAACJ,MAAM,CAACK,YAAY,EAAExF,sDAAe,CAAC;YAC1DsE,MAAM,GAAGI,IAAI,GAAG,OAAO,GAAGW,GAAG,CAACI,IAAI,CAACC,MAAM,CAACC,GAAG;YAC7CR,MAAM,GAAGF,SAAS,CAACX,MAAM,CAAC;YAC1Ba,MAAM,CAACS,IAAI,CAAC,IAAI,CAAC;YACjB,IAAI,IAAI,CAACC,MAAM,EAAE;cACf,IAAI,CAACA,MAAM,CAAC,CAAC;YACf;YAEA,IAAIX,OAAO,EAAE;cACXE,gBAAgB,GAAG,IAAI;cACvBF,OAAO,CAAC,CAAC;YACX;YAEA,IAAI,IAAI,CAAC1D,aAAa,EAAE;cACtB,IAAI,CAAC,CAACqC,QAAQ,CAAC,CAAC;YAClB;UACF,CAAC,MAAM,IAAIsB,MAAM,CAACJ,MAAM,GAAG,CAAC,IAAII,MAAM,CAACJ,MAAM,GAAG,GAAG,EAAE;YACnD,IAAI,IAAI,CAACe,SAAS,EAAE;cAClB,IAAI,CAACA,SAAS,CAACX,MAAM,CAACK,YAAY,CAAC;YACrC;YACAL,MAAM,GAAGF,SAAS,CAACX,MAAM,CAAC;YAC1Ba,MAAM,CAACS,IAAI,CAAC,IAAI,CAAC;UACnB,CAAC,MAAM;YAEL,IAAInD,MAAM,IAAI,CAAC2C,gBAAgB,EAAE;cAC/BA,gBAAgB,GAAG,IAAI;cACvB3C,MAAM,CAAC0C,MAAM,CAACK,YAAY,CAAC;YAC7B;YACA,IAAI,IAAI,CAACM,SAAS,IAAIX,MAAM,CAACK,YAAY,EAAE;cACzC,IAAI,CAACM,SAAS,CAACX,MAAM,CAACK,YAAY,CAAC;YACrC;YACA,IAAI,IAAI,CAACO,YAAY,EAAE;cACrB,MAAM7H,IAAI,GAAGiH,MAAM,CAACJ,MAAM,KAAK,IAAI,CAAC,CAAC1D,UAAU,GAAGhB,YAAY,GAAGF,aAAa,CAAC;cAC/E,MAAM6F,IAAI,GAAGb,MAAM,CAACK,YAAY,KAAK,IAAI,CAAC,CAACnE,UAAU,GAAGf,iBAAiB,GAAGF,kBAAkB,CAAC;cAC/F,IAAI,CAAC2F,YAAY,CAAC,IAAI/H,sDAAS,CAACgI,IAAI,EAAE9H,IAAI,CAAC,EAAEA,IAAI,CAAC;YACpD;YAGAiH,MAAM,GAAG,IAAI;YACb,IAAI,CAAC,IAAI,CAAC,CAAC9D,UAAU,IAAI,IAAI,CAACG,aAAa,EAAE;cAC3C,IAAI,CAAC,CAAC0B,aAAa,CAAC,CAAC;YACvB;UACF;QACF;MACF,CAAC;MAEDiC,MAAM,CAACH,IAAI,CAAC,MAAM,EAAEN,IAAI,EAAE,IAAI,CAAC;MAC/B,OAAOS,MAAM;IACf,CAAC;IAED,IAAI,CAAC9C,OAAO,GAAG,CAACC,KAAK,EAAEC,KAAK,KAAK;MAC/B,IAAI,CAAC,CAAClB,UAAU,GAAG,KAAK;MAExB,IAAIkD,OAAO,EAAE;QACX,IAAI,CAAChC,KAAK,EAAE;UACV,OAAOC,OAAO,CAAC0C,OAAO,CAAC,CAAC;QAC1B;QACAX,OAAO,CAACK,kBAAkB,GAAGzJ,SAAS;QACtCoJ,OAAO,CAAC0B,KAAK,CAAC,CAAC;QACf1B,OAAO,GAAG,IAAI;MAChB;MAEA,IAAIjC,KAAK,EAAE;QACT,IAAI,CAAC3B,IAAI,GAAG2B,KAAK;MACnB;MAEA,OAAO,IAAIE,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;QACtC,MAAM1B,GAAG,GAAGL,WAAW,CAAC,IAAI,CAACC,IAAI,EAAE,IAAI,CAACY,MAAM,GAAG,OAAO,GAAG,MAAM,EAAE,IAAI,CAACV,OAAO,EAAE,IAAI,CAACC,MAAM,CAAC;QAC7FE,UAAU,CAAC,CAACC,GAAG,CAAC,mBAAmB,EAAEF,GAAG,CAAC;QACzCwD,OAAO,GAAGU,SAAS,CAAClE,GAAG,EAAEmE,OAAO,EAAEzC,MAAM,CAAC;QACzC8B,OAAO,CAACqB,IAAI,CAAC,IAAI,CAAC;MACpB,CAAC,CAAC,CAAChC,KAAK,CAACsC,GAAG,IAAI;QACdlF,UAAU,CAAC,CAACC,GAAG,CAAC,uBAAuB,EAAEiF,GAAG,CAAC;MAC/C,CAAC,CAAC;IACJ,CAAC;IAED,IAAI,CAACxD,SAAS,GAAGH,KAAK,IAAI;MACxB,IAAI,CAAC,CAACsB,QAAQ,CAAC,CAAC;MAChB,IAAI,CAACxB,OAAO,CAAC,IAAI,EAAEE,KAAK,CAAC;IAC3B,CAAC;IAED,IAAI,CAACI,UAAU,GAAGzB,CAAC,IAAI;MACrB,IAAI,CAAC,CAACG,UAAU,GAAG,IAAI;MACvB,IAAI,CAAC,CAACwC,QAAQ,CAAC,CAAC;MAEhB,IAAIW,OAAO,EAAE;QACXA,OAAO,CAACI,kBAAkB,GAAGzJ,SAAS;QACtCqJ,OAAO,CAACyB,KAAK,CAAC,CAAC;QACfzB,OAAO,GAAG,IAAI;MAChB;MACA,IAAID,OAAO,EAAE;QACXA,OAAO,CAACK,kBAAkB,GAAGzJ,SAAS;QACtCoJ,OAAO,CAAC0B,KAAK,CAAC,CAAC;QACf1B,OAAO,GAAG,IAAI;MAChB;MAEA,IAAI,IAAI,CAACwB,YAAY,EAAE;QACrB,IAAI,CAACA,YAAY,CAAC,IAAI/H,sDAAS,CAACsC,iBAAiB,EAAED,YAAY,CAAC,EAAEA,YAAY,CAAC;MACjF;MAEAiE,MAAM,GAAG,IAAI;IACf,CAAC;IAED,IAAI,CAAC1B,QAAQ,GAAIC,GAAG,IAAK;MACvB2B,OAAO,GAAGC,SAAS,CAACH,MAAM,CAAC;MAC3B,IAAIE,OAAO,IAAKA,OAAO,CAACM,UAAU,IAAIZ,UAAW,EAAE;QACjDM,OAAO,CAACoB,IAAI,CAAC/C,GAAG,CAAC;MACnB,CAAC,MAAM;QACL,MAAM,IAAIvL,KAAK,CAAC,+BAA+B,CAAC;MAClD;IACF,CAAC;IAED,IAAI,CAACwL,WAAW,GAAG5B,CAAC,IAAI;MACtB,OAAQqD,OAAO,IAAI,IAAI;IACzB,CAAC;EACH;EAGA,CAACxC,OAAOoE,CAAA,EAAG;IACT,IAAI,CAAC9D,OAAO,GAAG,CAACC,KAAK,EAAEC,KAAK,KAAK;MAC/B,IAAI,CAAC,CAAClB,UAAU,GAAG,KAAK;MAExB,IAAI,IAAI,CAAC,CAACC,MAAM,EAAE;QAChB,IAAI,CAACiB,KAAK,IAAI,IAAI,CAAC,CAACjB,MAAM,CAACwD,UAAU,IAAI,IAAI,CAAC,CAACxD,MAAM,CAAC8E,IAAI,EAAE;UAC1D,OAAO5D,OAAO,CAAC0C,OAAO,CAAC,CAAC;QAC1B;QACA,IAAI,CAAC,CAAC5D,MAAM,CAAC+E,KAAK,CAAC,CAAC;QACpB,IAAI,CAAC,CAAC/E,MAAM,GAAG,IAAI;MACrB;MAEA,IAAIgB,KAAK,EAAE;QACT,IAAI,CAAC3B,IAAI,GAAG2B,KAAK;MACnB;MAEA,OAAO,IAAIE,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;QACtC,MAAM1B,GAAG,GAAGL,WAAW,CAAC,IAAI,CAACC,IAAI,EAAE,IAAI,CAACY,MAAM,GAAG,KAAK,GAAG,IAAI,EAAE,IAAI,CAACV,OAAO,EAAE,IAAI,CAACC,MAAM,CAAC;QAEzFE,UAAU,CAAC,CAACC,GAAG,CAAC,oBAAoB,EAAEF,GAAG,CAAC;QAI1C,MAAMuF,IAAI,GAAG,IAAIrG,iBAAiB,CAACc,GAAG,CAAC;QAEvCuF,IAAI,CAACC,OAAO,GAAGL,GAAG,IAAI;UACpBzD,MAAM,CAACyD,GAAG,CAAC;QACb,CAAC;QAEDI,IAAI,CAACE,MAAM,GAAGtF,CAAC,IAAI;UACjB,IAAI,IAAI,CAACM,aAAa,EAAE;YACtB,IAAI,CAAC,CAACqC,QAAQ,CAAC,CAAC;UAClB;UAEA,IAAI,IAAI,CAACgC,MAAM,EAAE;YACf,IAAI,CAACA,MAAM,CAAC,CAAC;UACf;UAEAX,OAAO,CAAC,CAAC;QACX,CAAC;QAEDoB,IAAI,CAACG,OAAO,GAAGvF,CAAC,IAAI;UAClB,IAAI,CAAC,CAACI,MAAM,GAAG,IAAI;UAEnB,IAAI,IAAI,CAACyE,YAAY,EAAE;YACrB,MAAM7H,IAAI,GAAG,IAAI,CAAC,CAACmD,UAAU,GAAGhB,YAAY,GAAGF,aAAa;YAC5D,IAAI,CAAC4F,YAAY,CAAC,IAAI/H,sDAAS,CAAC,IAAI,CAAC,CAACqD,UAAU,GAAGf,iBAAiB,GAAGF,kBAAkB,EAAElC,IAAI,CAAC,EAAEA,IAAI,CAAC;UACzG;UAEA,IAAI,CAAC,IAAI,CAAC,CAACmD,UAAU,IAAI,IAAI,CAACG,aAAa,EAAE;YAC3C,IAAI,CAAC,CAAC0B,aAAa,CAAC,CAAC;UACvB;QACF,CAAC;QAEDoD,IAAI,CAACI,SAAS,GAAG7B,GAAG,IAAI;UACtB,IAAI,IAAI,CAACiB,SAAS,EAAE;YAClB,IAAI,CAACA,SAAS,CAACjB,GAAG,CAAC8B,IAAI,CAAC;UAC1B;QACF,CAAC;QAED,IAAI,CAAC,CAACrF,MAAM,GAAGgF,IAAI;MACrB,CAAC,CAAC;IACJ,CAAC;IAED,IAAI,CAAC5D,SAAS,GAAGH,KAAK,IAAI;MACxB,IAAI,CAAC,CAACsB,QAAQ,CAAC,CAAC;MAChB,IAAI,CAACxB,OAAO,CAAC,IAAI,EAAEE,KAAK,CAAC;IAC3B,CAAC;IAED,IAAI,CAACI,UAAU,GAAGzB,CAAC,IAAI;MACrB,IAAI,CAAC,CAACG,UAAU,GAAG,IAAI;MACvB,IAAI,CAAC,CAACwC,QAAQ,CAAC,CAAC;MAEhB,IAAI,CAAC,IAAI,CAAC,CAACvC,MAAM,EAAE;QACjB;MACF;MACA,IAAI,CAAC,CAACA,MAAM,CAAC+E,KAAK,CAAC,CAAC;MACpB,IAAI,CAAC,CAAC/E,MAAM,GAAG,IAAI;IACrB,CAAC;IAED,IAAI,CAACsB,QAAQ,GAAGC,GAAG,IAAI;MACrB,IAAI,IAAI,CAAC,CAACvB,MAAM,IAAK,IAAI,CAAC,CAACA,MAAM,CAACwD,UAAU,IAAI,IAAI,CAAC,CAACxD,MAAM,CAAC8E,IAAK,EAAE;QAClE,IAAI,CAAC,CAAC9E,MAAM,CAACsE,IAAI,CAAC/C,GAAG,CAAC;MACxB,CAAC,MAAM;QACL,MAAM,IAAIvL,KAAK,CAAC,4BAA4B,CAAC;MAC/C;IACF,CAAC;IAED,IAAI,CAACwL,WAAW,GAAG5B,CAAC,IAAI;MACtB,OAAQ,IAAI,CAAC,CAACI,MAAM,IAAK,IAAI,CAAC,CAACA,MAAM,CAACwD,UAAU,IAAI,IAAI,CAAC,CAACxD,MAAM,CAAC8E,IAAK;IACxE,CAAC;EACH;EAUAN,SAAS,GAAG3K,SAAS;EAOrB4K,YAAY,GAAG5K,SAAS;EAQxB0K,MAAM,GAAG1K,SAAS;EAelBsI,wBAAwB,GAAGtI,SAAS;AACtC;AAEA6F,UAAU,CAACb,aAAa,GAAGA,aAAa;AACxCa,UAAU,CAACZ,kBAAkB,GAAGA,kBAAkB;AAClDY,UAAU,CAACX,YAAY,GAAGA,YAAY;AACtCW,UAAU,CAACV,iBAAiB,GAAGA,iBAAiB;;;;;;;;;;;;;;AC/gBnC;AAMb,MAAMsG,UAAU,GAAG,CAAC;AACpB,MAAMC,OAAO,GAAG,YAAY;AAE5B,IAAIC,WAAW;AAEA,MAAMC,EAAE,CAAC;EACtB,CAACC,OAAO,GAAG9F,CAAC,IAAI,CAAC,CAAC;EAClB,CAACiB,MAAM,GAAGjB,CAAC,IAAI,CAAC,CAAC;EAGjB+F,EAAE,GAAG,IAAI;EAETC,QAAQ,GAAG,IAAI;EAEfxQ,WAAWA,CAACsQ,OAAO,EAAE7E,MAAM,EAAE;IAC3B,IAAI,CAAC,CAAC6E,OAAO,GAAGA,OAAO,IAAI,IAAI,CAAC,CAACA,OAAO;IACxC,IAAI,CAAC,CAAC7E,MAAM,GAAGA,MAAM,IAAI,IAAI,CAAC,CAACA,MAAM;EACvC;EAEA,CAACgF,UAAUC,CAACC,MAAM,EAAE9J,QAAQ,EAAEG,OAAO,EAAE;IACrC,IAAI,CAAC,IAAI,CAACuJ,EAAE,EAAE;MACZ,OAAOC,QAAQ,GACb1E,OAAO,CAAC0C,OAAO,CAAC,EAAE,CAAC,GACnB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IAEA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAACF,MAAM,CAAC,CAAC;MACzCC,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,YAAY,EAAEkF,MAAM,EAAEG,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QAChEjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACDJ,GAAG,CAACK,WAAW,CAACN,MAAM,CAAC,CAACO,MAAM,CAAC,CAAC,CAACC,SAAS,GAAGL,KAAK,IAAI;QACpD,IAAIjK,QAAQ,EAAE;UACZiK,KAAK,CAACC,MAAM,CAACK,MAAM,CAACxK,OAAO,CAACyK,KAAK,IAAI;YACnCxK,QAAQ,CAACI,IAAI,CAACD,OAAO,EAAEqK,KAAK,CAAC;UAC/B,CAAC,CAAC;QACJ;QACA7C,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;IACH,CAAC,CAAC;EACJ;EAMAE,YAAYA,CAAA,EAAG;IACb,OAAO,IAAIxF,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MAEtC,MAAMwF,GAAG,GAAGnB,WAAW,CAAC9B,IAAI,CAAC6B,OAAO,EAAED,UAAU,CAAC;MACjDqB,GAAG,CAACJ,SAAS,GAAGL,KAAK,IAAI;QACvB,IAAI,CAACP,EAAE,GAAGO,KAAK,CAACC,MAAM,CAACK,MAAM;QAC7B,IAAI,CAACZ,QAAQ,GAAG,KAAK;QACrBhC,OAAO,CAAC,IAAI,CAAC+B,EAAE,CAAC;MAClB,CAAC;MACDgB,GAAG,CAAC1B,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,sBAAsB,EAAEqF,KAAK,CAAC;QACrD/E,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QAC1B,IAAI,CAAC,CAACV,OAAO,CAACQ,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MACnC,CAAC;MACDO,GAAG,CAACC,eAAe,GAAGV,KAAK,IAAI;QAC7B,IAAI,CAACP,EAAE,GAAGO,KAAK,CAACC,MAAM,CAACK,MAAM;QAE7B,IAAI,CAACb,EAAE,CAACV,OAAO,GAAGiB,KAAK,IAAI;UACzB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,0BAA0B,EAAEqF,KAAK,CAAC;UACzD,IAAI,CAAC,CAACR,OAAO,CAACQ,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QACnC,CAAC;QAID,IAAI,CAACT,EAAE,CAACkB,iBAAiB,CAAC,OAAO,EAAE;UACjCC,OAAO,EAAE;QACX,CAAC,CAAC;QAGF,IAAI,CAACnB,EAAE,CAACkB,iBAAiB,CAAC,MAAM,EAAE;UAChCC,OAAO,EAAE;QACX,CAAC,CAAC;QAGF,IAAI,CAACnB,EAAE,CAACkB,iBAAiB,CAAC,cAAc,EAAE;UACxCC,OAAO,EAAE,CAAC,OAAO,EAAE,KAAK;QAC1B,CAAC,CAAC;QAGF,IAAI,CAACnB,EAAE,CAACkB,iBAAiB,CAAC,SAAS,EAAE;UACnCC,OAAO,EAAE,CAAC,OAAO,EAAE,KAAK;QAC1B,CAAC,CAAC;MACJ,CAAC;IACH,CAAC,CAAC;EACJ;EAKAC,cAAcA,CAAA,EAAG;IAEf,IAAI,IAAI,CAACpB,EAAE,EAAE;MACX,IAAI,CAACA,EAAE,CAACZ,KAAK,CAAC,CAAC;MACf,IAAI,CAACY,EAAE,GAAG,IAAI;IAChB;IACA,OAAO,IAAIzE,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAMwF,GAAG,GAAGnB,WAAW,CAACuB,cAAc,CAACxB,OAAO,CAAC;MAC/CoB,GAAG,CAACK,SAAS,GAAGpH,CAAC,IAAI;QACnB,IAAI,IAAI,CAAC+F,EAAE,EAAE;UACX,IAAI,CAACA,EAAE,CAACZ,KAAK,CAAC,CAAC;QACjB;QACA,MAAMH,GAAG,GAAG,IAAI5O,KAAK,CAAC,SAAS,CAAC;QAChC,IAAI,CAAC,CAAC6K,MAAM,CAAC,QAAQ,EAAE,gBAAgB,EAAE+D,GAAG,CAAC;QAC7CzD,MAAM,CAACyD,GAAG,CAAC;MACb,CAAC;MACD+B,GAAG,CAACJ,SAAS,GAAG3G,CAAC,IAAI;QACnB,IAAI,CAAC+F,EAAE,GAAG,IAAI;QACd,IAAI,CAACC,QAAQ,GAAG,IAAI;QACpBhC,OAAO,CAAC,IAAI,CAAC;MACf,CAAC;MACD+C,GAAG,CAAC1B,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,gBAAgB,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QAC5DjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;IACH,CAAC,CAAC;EACJ;EAOAa,OAAOA,CAAA,EAAG;IACR,OAAO,CAAC,CAAC,IAAI,CAACtB,EAAE;EAClB;EAUAuB,QAAQA,CAACT,KAAK,EAAE;IACd,IAAI,CAAC,IAAI,CAACQ,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,OAAO,CAAC,EAAE,WAAW,CAAC;MACvDD,GAAG,CAACmB,UAAU,GAAGjB,KAAK,IAAI;QACxBtC,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;MACDR,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,UAAU,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QACtDjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACD,MAAMO,GAAG,GAAGX,GAAG,CAACK,WAAW,CAAC,OAAO,CAAC,CAACe,GAAG,CAACX,KAAK,CAAC5J,IAAI,CAAC;MACpD8J,GAAG,CAACJ,SAAS,GAAG3G,CAAC,IAAI;QACnBoG,GAAG,CAACK,WAAW,CAAC,OAAO,CAAC,CAAChL,GAAG,CAACoK,EAAE,CAAC,CAAC4B,cAAc,CAACV,GAAG,CAACH,MAAM,EAAEC,KAAK,CAAC,CAAC;QACnET,GAAG,CAACsB,MAAM,CAAC,CAAC;MACd,CAAC;IACH,CAAC,CAAC;EACJ;EASAC,kBAAkBA,CAAC1K,IAAI,EAAE2K,OAAO,EAAE;IAChC,IAAI,CAAC,IAAI,CAACP,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,OAAO,CAAC,EAAE,WAAW,CAAC;MACvDD,GAAG,CAACmB,UAAU,GAAGjB,KAAK,IAAI;QACxBtC,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;MACDR,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,oBAAoB,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QAChEjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACD,MAAMO,GAAG,GAAGX,GAAG,CAACK,WAAW,CAAC,OAAO,CAAC,CAACe,GAAG,CAACvK,IAAI,CAAC;MAC9C8J,GAAG,CAACJ,SAAS,GAAGL,KAAK,IAAI;QACvB,MAAMO,KAAK,GAAGP,KAAK,CAACC,MAAM,CAACK,MAAM;QACjC,IAAIC,KAAK,IAAIA,KAAK,CAACgB,QAAQ,IAAID,OAAO,EAAE;UACtCf,KAAK,CAACgB,QAAQ,GAAGD,OAAO;UACxBxB,GAAG,CAACK,WAAW,CAAC,OAAO,CAAC,CAAChL,GAAG,CAACoL,KAAK,CAAC;QACrC;QACAT,GAAG,CAACsB,MAAM,CAAC,CAAC;MACd,CAAC;IACH,CAAC,CAAC;EACJ;EAQAI,QAAQA,CAAC7K,IAAI,EAAE;IACb,IAAI,CAAC,IAAI,CAACoK,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,OAAO,EAAE,cAAc,EAAE,SAAS,CAAC,EAAE,WAAW,CAAC;MAClFD,GAAG,CAACmB,UAAU,GAAGjB,KAAK,IAAI;QACxBtC,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;MACDR,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,UAAU,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QACtDjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACDJ,GAAG,CAACK,WAAW,CAAC,OAAO,CAAC,CAACsB,MAAM,CAACC,WAAW,CAACC,IAAI,CAAChL,IAAI,CAAC,CAAC;MACvDmJ,GAAG,CAACK,WAAW,CAAC,cAAc,CAAC,CAACsB,MAAM,CAACC,WAAW,CAACE,KAAK,CAAC,CAACjL,IAAI,EAAE,GAAG,CAAC,EAAE,CAACA,IAAI,EAAE,GAAG,CAAC,CAAC,CAAC;MACnFmJ,GAAG,CAACK,WAAW,CAAC,SAAS,CAAC,CAACsB,MAAM,CAACC,WAAW,CAACE,KAAK,CAAC,CAACjL,IAAI,EAAE,CAAC,CAAC,EAAE,CAACA,IAAI,EAAEkL,MAAM,CAACC,gBAAgB,CAAC,CAAC,CAAC;MAChGhC,GAAG,CAACsB,MAAM,CAAC,CAAC;IACd,CAAC,CAAC;EACJ;EASAW,SAASA,CAAChM,QAAQ,EAAEG,OAAO,EAAE;IAC3B,OAAO,IAAI,CAAC,CAACyJ,UAAU,CAAC,OAAO,EAAE5J,QAAQ,EAAEG,OAAO,CAAC;EACrD;EAQA8L,gBAAgBA,CAACzB,KAAK,EAAE0B,GAAG,EAAE;IAC3B1C,EAAE,CAAC,CAACyC,gBAAgB,CAACzB,KAAK,EAAE0B,GAAG,CAAC;EAClC;EAUAC,OAAOA,CAACC,GAAG,EAAEC,GAAG,EAAE;IAChB,IAAI/M,SAAS,CAACxE,MAAM,GAAG,CAAC,IAAIuR,GAAG,KAAKzO,SAAS,EAAE;MAE7C;IACF;IACA,IAAI,CAAC,IAAI,CAACoN,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,MAAM,CAAC,EAAE,WAAW,CAAC;MACtDD,GAAG,CAACmB,UAAU,GAAGjB,KAAK,IAAI;QACxBtC,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;MACDR,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,SAAS,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QACrDjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACDJ,GAAG,CAACK,WAAW,CAAC,MAAM,CAAC,CAAChL,GAAG,CAAC;QAC1BgN,GAAG,EAAEA,GAAG;QACRE,MAAM,EAAED;MACV,CAAC,CAAC;MACFtC,GAAG,CAACsB,MAAM,CAAC,CAAC;IACd,CAAC,CAAC;EACJ;EAQAkB,OAAOA,CAACH,GAAG,EAAE;IACX,IAAI,CAAC,IAAI,CAACpB,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,MAAM,CAAC,EAAE,WAAW,CAAC;MACtDD,GAAG,CAACmB,UAAU,GAAGjB,KAAK,IAAI;QACxBtC,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;MACDR,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,SAAS,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QACrDjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACDJ,GAAG,CAACK,WAAW,CAAC,MAAM,CAAC,CAACsB,MAAM,CAACC,WAAW,CAACC,IAAI,CAACQ,GAAG,CAAC,CAAC;MACrDrC,GAAG,CAACsB,MAAM,CAAC,CAAC;IACd,CAAC,CAAC;EACJ;EASAmB,QAAQA,CAACxM,QAAQ,EAAEG,OAAO,EAAE;IAC1B,OAAO,IAAI,CAAC,CAACyJ,UAAU,CAAC,MAAM,EAAE5J,QAAQ,EAAEG,OAAO,CAAC;EACpD;EAQAsM,OAAOA,CAACL,GAAG,EAAE;IACX,IAAI,CAAC,IAAI,CAACpB,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,MAAM,CAAC,CAAC;MACzCD,GAAG,CAACmB,UAAU,GAAGjB,KAAK,IAAI;QACxB,MAAMyC,IAAI,GAAGzC,KAAK,CAACC,MAAM,CAACK,MAAM;QAChC5C,OAAO,CAAC;UACN+E,IAAI,EAAEA,IAAI,CAACN,GAAG;UACdE,MAAM,EAAEI,IAAI,CAACJ;QACf,CAAC,CAAC;MACJ,CAAC;MACDvC,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,SAAS,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QACrDjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACDJ,GAAG,CAACK,WAAW,CAAC,MAAM,CAAC,CAACe,GAAG,CAACiB,GAAG,CAAC;IAClC,CAAC,CAAC;EACJ;EAWAO,eAAeA,CAACC,SAAS,EAAER,GAAG,EAAES,GAAG,EAAE;IACnC,IAAI,CAAC,IAAI,CAAC7B,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,cAAc,CAAC,EAAE,WAAW,CAAC;MAC9DD,GAAG,CAACmB,UAAU,GAAGjB,KAAK,IAAI;QACxBtC,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;MACDR,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,iBAAiB,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QAC7DjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACDJ,GAAG,CAACK,WAAW,CAAC,cAAc,CAAC,CAACe,GAAG,CAAC,CAACyB,SAAS,EAAER,GAAG,CAAC,CAAC,CAAC9B,SAAS,GAAIL,KAAK,IAAK;QAC3EF,GAAG,CAACK,WAAW,CAAC,cAAc,CAAC,CAAChL,GAAG,CAACoK,EAAE,CAAC,CAACsD,qBAAqB,CAAC7C,KAAK,CAACC,MAAM,CAACK,MAAM,EAAEqC,SAAS,EAAER,GAAG,EAAES,GAAG,CAAC,CAAC;QACxG9C,GAAG,CAACsB,MAAM,CAAC,CAAC;MACd,CAAC;IACH,CAAC,CAAC;EACJ;EAUA0B,gBAAgBA,CAACH,SAAS,EAAE5M,QAAQ,EAAEG,OAAO,EAAE;IAC7C,IAAI,CAAC,IAAI,CAAC6K,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,EAAE,CAAC,GACnB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,cAAc,CAAC,CAAC;MACjDD,GAAG,CAACf,OAAO,GAAIiB,KAAK,IAAK;QACvB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,kBAAkB,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QAC9DjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACDJ,GAAG,CAACK,WAAW,CAAC,cAAc,CAAC,CAACC,MAAM,CAACsB,WAAW,CAACE,KAAK,CAAC,CAACe,SAAS,EAAE,GAAG,CAAC,EAAE,CAACA,SAAS,EAAE,GAAG,CAAC,CAAC,CAAC,CAACtC,SAAS,GAAIL,KAAK,IAAK;QACnH,IAAIjK,QAAQ,EAAE;UACZiK,KAAK,CAACC,MAAM,CAACK,MAAM,CAACxK,OAAO,CAAEyK,KAAK,IAAK;YACrCxK,QAAQ,CAACI,IAAI,CAACD,OAAO,EAAEqK,KAAK,CAAC;UAC/B,CAAC,CAAC;QACJ;QACA7C,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;IACH,CAAC,CAAC;EACJ;EAWAyC,UAAUA,CAAC1H,GAAG,EAAE;IACd,IAAI,CAAC,IAAI,CAAC0F,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,SAAS,CAAC,EAAE,WAAW,CAAC;MACzDD,GAAG,CAACO,SAAS,GAAGL,KAAK,IAAI;QACvBtC,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;MACDR,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,YAAY,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QACxDjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACDJ,GAAG,CAACK,WAAW,CAAC,SAAS,CAAC,CAAC6C,GAAG,CAACzD,EAAE,CAAC,CAAC0D,gBAAgB,CAAC,IAAI,EAAE5H,GAAG,CAAC,CAAC;MAC/DyE,GAAG,CAACsB,MAAM,CAAC,CAAC;IACd,CAAC,CAAC;EACJ;EAUA8B,gBAAgBA,CAACP,SAAS,EAAEQ,GAAG,EAAE5F,MAAM,EAAE;IACvC,IAAI,CAAC,IAAI,CAACwD,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,MAAM6E,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,SAAS,CAAC,EAAE,WAAW,CAAC;MACzDD,GAAG,CAACO,SAAS,GAAGL,KAAK,IAAI;QACvBtC,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;MACDR,GAAG,CAACf,OAAO,GAAGiB,KAAK,IAAI;QACrB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,kBAAkB,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QAC9DjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACD,MAAMO,GAAG,GAAGX,GAAG,CAACK,WAAW,CAAC,SAAS,CAAC,CAACe,GAAG,CAACQ,WAAW,CAACC,IAAI,CAAC,CAACgB,SAAS,EAAEQ,GAAG,CAAC,CAAC,CAAC;MAC9E1C,GAAG,CAACJ,SAAS,GAAGL,KAAK,IAAI;QACvB,MAAMiC,GAAG,GAAGxB,GAAG,CAACH,MAAM,IAAIN,KAAK,CAACC,MAAM,CAACK,MAAM;QAC7C,IAAI,CAAC2B,GAAG,IAAIA,GAAG,CAACmB,OAAO,IAAI7F,MAAM,EAAE;UACjCuC,GAAG,CAACsB,MAAM,CAAC,CAAC;UACZ;QACF;QACAtB,GAAG,CAACK,WAAW,CAAC,SAAS,CAAC,CAAChL,GAAG,CAACoK,EAAE,CAAC,CAAC0D,gBAAgB,CAAChB,GAAG,EAAE;UACvD1B,KAAK,EAAEoC,SAAS;UAChBQ,GAAG,EAAEA,GAAG;UACRC,OAAO,EAAE7F;QACX,CAAC,CAAC,CAAC;QACHuC,GAAG,CAACsB,MAAM,CAAC,CAAC;MACd,CAAC;IACH,CAAC,CAAC;EACJ;EAUAiC,WAAWA,CAACV,SAAS,EAAEW,IAAI,EAAEC,EAAE,EAAE;IAC/B,IAAI,CAAC,IAAI,CAACxC,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,CAAC,GACjB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtC,IAAI,CAACqI,IAAI,IAAI,CAACC,EAAE,EAAE;QAChBD,IAAI,GAAG,CAAC;QACRC,EAAE,GAAG1B,MAAM,CAACC,gBAAgB;MAC9B;MACA,MAAM0B,KAAK,GAAGD,EAAE,GAAG,CAAC,GAAG7B,WAAW,CAACE,KAAK,CAAC,CAACe,SAAS,EAAEW,IAAI,CAAC,EAAE,CAACX,SAAS,EAAEY,EAAE,CAAC,EAAE,KAAK,EAAE,IAAI,CAAC,GACvF7B,WAAW,CAACC,IAAI,CAAC,CAACgB,SAAS,EAAEW,IAAI,CAAC,CAAC;MACrC,MAAMxD,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,SAAS,CAAC,EAAE,WAAW,CAAC;MACzDD,GAAG,CAACO,SAAS,GAAIL,KAAK,IAAK;QACzBtC,OAAO,CAACsC,KAAK,CAACC,MAAM,CAACK,MAAM,CAAC;MAC9B,CAAC;MACDR,GAAG,CAACf,OAAO,GAAIiB,KAAK,IAAK;QACvB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,aAAa,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QACzDjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MACDJ,GAAG,CAACK,WAAW,CAAC,SAAS,CAAC,CAACsB,MAAM,CAAC+B,KAAK,CAAC;MACxC1D,GAAG,CAACsB,MAAM,CAAC,CAAC;IACd,CAAC,CAAC;EACJ;EAaAqC,YAAYA,CAACd,SAAS,EAAEe,KAAK,EAAE3N,QAAQ,EAAEG,OAAO,EAAE;IAChD,IAAI,CAAC,IAAI,CAAC6K,OAAO,CAAC,CAAC,EAAE;MACnB,OAAO,IAAI,CAACrB,QAAQ,GAClB1E,OAAO,CAAC0C,OAAO,CAAC,EAAE,CAAC,GACnB1C,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iBAAiB,CAAC,CAAC;IAChD;IACA,OAAO,IAAIkL,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MACtCyI,KAAK,GAAGA,KAAK,IAAI,CAAC,CAAC;MACnB,MAAM/N,KAAK,GAAG+N,KAAK,CAAC/N,KAAK,GAAG,CAAC,GAAG+N,KAAK,CAAC/N,KAAK,GAAG,CAAC;MAC/C,MAAMC,MAAM,GAAG8N,KAAK,CAAC9N,MAAM,GAAG,CAAC,GAAG8N,KAAK,CAAC9N,MAAM,GAAGiM,MAAM,CAACC,gBAAgB;MACxE,MAAM6B,KAAK,GAAGD,KAAK,CAACC,KAAK,GAAG,CAAC;MAE7B,MAAMrD,MAAM,GAAG,EAAE;MACjB,MAAMkD,KAAK,GAAG9B,WAAW,CAACE,KAAK,CAAC,CAACe,SAAS,EAAEhN,KAAK,CAAC,EAAE,CAACgN,SAAS,EAAE/M,MAAM,CAAC,EAAE,KAAK,EAAE,IAAI,CAAC;MACrF,MAAMkK,GAAG,GAAG,IAAI,CAACL,EAAE,CAACM,WAAW,CAAC,CAAC,SAAS,CAAC,CAAC;MAC5CD,GAAG,CAACf,OAAO,GAAIiB,KAAK,IAAK;QACvB,IAAI,CAAC,CAACrF,MAAM,CAAC,QAAQ,EAAE,cAAc,EAAEqF,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;QAC1DjF,MAAM,CAAC+E,KAAK,CAACC,MAAM,CAACC,KAAK,CAAC;MAC5B,CAAC;MAEDJ,GAAG,CAACK,WAAW,CAAC,SAAS,CAAC,CAACyD,UAAU,CAACJ,KAAK,EAAE,MAAM,CAAC,CAACnD,SAAS,GAAIL,KAAK,IAAK;QAC1E,MAAM6D,MAAM,GAAG7D,KAAK,CAACC,MAAM,CAACK,MAAM;QAClC,IAAIuD,MAAM,EAAE;UACV,IAAI9N,QAAQ,EAAE;YACZA,QAAQ,CAACI,IAAI,CAACD,OAAO,EAAE2N,MAAM,CAACC,KAAK,CAAC;UACtC;UACAxD,MAAM,CAACyD,IAAI,CAACF,MAAM,CAACC,KAAK,CAAC;UACzB,IAAIH,KAAK,IAAI,CAAC,IAAIrD,MAAM,CAACzP,MAAM,GAAG8S,KAAK,EAAE;YACvCE,MAAM,CAACG,QAAQ,CAAC,CAAC;UACnB,CAAC,MAAM;YACLtG,OAAO,CAAC4C,MAAM,CAAC;UACjB;QACF,CAAC,MAAM;UACL5C,OAAO,CAAC4C,MAAM,CAAC;QACjB;MACF,CAAC;IACH,CAAC,CAAC;EACJ;EAKA,OAAO,CAAC2D,YAAY,GAAG,CAAC,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,QAAQ,EAC/F,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,UAAU,CAC/D;EAGD,OAAO,CAACjC,gBAAgBkC,CAAC3D,KAAK,EAAE0B,GAAG,EAAE;IACnC1C,EAAE,CAAC,CAAC0E,YAAY,CAACnO,OAAO,CAAEqO,CAAC,IAAK;MAC9B,IAAIlC,GAAG,CAACmC,cAAc,CAACD,CAAC,CAAC,EAAE;QACzB5D,KAAK,CAAC4D,CAAC,CAAC,GAAGlC,GAAG,CAACkC,CAAC,CAAC;MACnB;IACF,CAAC,CAAC;IACF,IAAI7O,KAAK,CAACC,OAAO,CAAC0M,GAAG,CAACoC,IAAI,CAAC,EAAE;MAC3B9D,KAAK,CAAC+D,KAAK,GAAGrC,GAAG,CAACoC,IAAI;IACxB;IACA,IAAIpC,GAAG,CAAC9S,GAAG,EAAE;MACXoR,KAAK,CAACgE,aAAa,CAACtC,GAAG,CAAC9S,GAAG,CAAC;IAC9B;IACAoR,KAAK,CAAC4C,GAAG,IAAI,CAAC;IACd5C,KAAK,CAACiE,IAAI,IAAI,CAAC;IACfjE,KAAK,CAACkE,MAAM,GAAG3I,IAAI,CAAC4I,GAAG,CAAC,CAAC,EAAEnE,KAAK,CAAC4C,GAAG,GAAG5C,KAAK,CAACiE,IAAI,CAAC;EACpD;EAGA,OAAO,CAACrD,cAAcwD,CAACC,GAAG,EAAE3C,GAAG,EAAE;IAC/B,MAAM9Q,GAAG,GAAGyT,GAAG,IAAI;MACjBjO,IAAI,EAAEsL,GAAG,CAACtL;IACZ,CAAC;IACD4I,EAAE,CAAC,CAAC0E,YAAY,CAACnO,OAAO,CAAEqO,CAAC,IAAK;MAC9B,IAAIlC,GAAG,CAACmC,cAAc,CAACD,CAAC,CAAC,EAAE;QACzBhT,GAAG,CAACgT,CAAC,CAAC,GAAGlC,GAAG,CAACkC,CAAC,CAAC;MACjB;IACF,CAAC,CAAC;IACF,IAAI7O,KAAK,CAACC,OAAO,CAAC0M,GAAG,CAACqC,KAAK,CAAC,EAAE;MAC5BnT,GAAG,CAACkT,IAAI,GAAGpC,GAAG,CAACqC,KAAK;IACtB;IACA,IAAIrC,GAAG,CAAC9S,GAAG,EAAE;MACXgC,GAAG,CAAChC,GAAG,GAAG8S,GAAG,CAAC4C,aAAa,CAAC,CAAC,CAAC/S,UAAU,CAAC,CAAC;IAC5C;IACA,OAAOX,GAAG;EACZ;EAEA,OAAO,CAAC0R,qBAAqBiC,CAACF,GAAG,EAAEjC,SAAS,EAAER,GAAG,EAAES,GAAG,EAAE;IACtD,MAAMmC,MAAM,GAAG,CAAC,SAAS,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,UAAU,EAAE,WAAW,CAAC;IACpF,MAAM5T,GAAG,GAAGyT,GAAG,IAAI;MACjBrE,KAAK,EAAEoC,SAAS;MAChBR,GAAG,EAAEA;IACP,CAAC;IAED4C,MAAM,CAACjP,OAAO,CAAEqO,CAAC,IAAK;MACpB,IAAIvB,GAAG,CAACwB,cAAc,CAACD,CAAC,CAAC,EAAE;QACzBhT,GAAG,CAACgT,CAAC,CAAC,GAAGvB,GAAG,CAACuB,CAAC,CAAC;MACjB;IACF,CAAC,CAAC;IAEF,OAAOhT,GAAG;EACZ;EAEA,OAAO,CAAC8R,gBAAgB+B,CAACJ,GAAG,EAAEvJ,GAAG,EAAE;IAEjC,MAAM0J,MAAM,GAAG,CAAC,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,CAAC;IAC3E,MAAM5T,GAAG,GAAGyT,GAAG,IAAI,CAAC,CAAC;IACrBG,MAAM,CAACjP,OAAO,CAAEqO,CAAC,IAAK;MACpB,IAAI9I,GAAG,CAAC+I,cAAc,CAACD,CAAC,CAAC,EAAE;QACzBhT,GAAG,CAACgT,CAAC,CAAC,GAAG9I,GAAG,CAAC8I,CAAC,CAAC;MACjB;IACF,CAAC,CAAC;IACF,OAAOhT,GAAG;EACZ;EAQA,OAAO8T,mBAAmBA,CAACC,WAAW,EAAE;IACtC5F,WAAW,GAAG4F,WAAW;EAC3B;AACF;;;;;;;;;;AC/oBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEa;AAMb,MAAMC,iBAAiB,GAAG,CAAC;AAC3B,MAAMC,uBAAuB,GAAG,CAAC;AACjC,MAAMC,qBAAqB,GAAG,EAAE;AAChC,MAAMC,cAAc,GAAG,kBAAkB;AACzC,MAAMC,gBAAgB,GAAG,eAAe;AACxC,MAAMC,kBAAkB,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,UAAU,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,SAAS,EACjH,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,OAAO,CAC9C;AAID,MAAMC,aAAa,GAAG,CAEpB;EACE9O,IAAI,EAAE,IAAI;EACVpC,KAAK,EAAE,uBAAuB;EAC9BC,GAAG,EAAE;AACP,CAAC,EAED;EACEmC,IAAI,EAAE,IAAI;EACVpC,KAAK,EAAE,mBAAmB;EAC1BC,GAAG,EAAE;AACP,CAAC,EAED;EACEmC,IAAI,EAAE,IAAI;EACVpC,KAAK,EAAE,sBAAsB;EAC7BC,GAAG,EAAE;AACP,CAAC,EAED;EACEmC,IAAI,EAAE,IAAI;EACVpC,KAAK,EAAE,iBAAiB;EACxBC,GAAG,EAAE;AACP,CAAC,CACF;AAGD,MAAMkR,UAAU,GAAG,CAAC,IAAI,CAAC;AAGzB,MAAMC,YAAY,GAAG,CAEnB;EACEhP,IAAI,EAAE,IAAI;EACViP,QAAQ,EAAE,KAAK;EACfC,IAAI,EAAE,SAAAA,CAASnW,GAAG,EAAE;IAElB,IAAI,CAAC,eAAe,CAACoW,IAAI,CAACpW,GAAG,CAAC,EAAE;MAC9BA,GAAG,GAAG,SAAS,GAAGA,GAAG;IACvB;IACA,OAAO;MACL6J,GAAG,EAAE7J;IACP,CAAC;EACH,CAAC;EACDqW,EAAE,EAAE;AACN,CAAC,EAED;EACEpP,IAAI,EAAE,IAAI;EACViP,QAAQ,EAAE,KAAK;EACfC,IAAI,EAAE,SAAAA,CAASnW,GAAG,EAAE;IAClB,OAAO;MACLA,GAAG,EAAEA,GAAG,CAACsW,KAAK,CAAC,CAAC;IAClB,CAAC;EACH,CAAC;EACDD,EAAE,EAAE;AACN,CAAC,EAED;EACEpP,IAAI,EAAE,IAAI;EACViP,QAAQ,EAAE,KAAK;EACfC,IAAI,EAAE,SAAAA,CAASnW,GAAG,EAAE;IAClB,OAAO;MACLA,GAAG,EAAEA,GAAG,CAACsW,KAAK,CAAC,CAAC;IAClB,CAAC;EACH,CAAC;EACDD,EAAE,EAAE;AACN,CAAC,CACF;AAGD,MAAME,WAAW,GAAG;EAClBC,EAAE,EAAE;IACFC,QAAQ,EAAE,OAAO;IACjBC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDC,EAAE,EAAE;IACFH,QAAQ,EAAE,QAAQ;IAClBC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDE,EAAE,EAAE;IACFJ,QAAQ,EAAE,IAAI;IACdC,MAAM,EAAE,IAAI;IACZC,MAAM,EAAE;EACV,CAAC;EACDG,EAAE,EAAE;IACFL,QAAQ,EAAE,IAAI;IACdC,MAAM,EAAE,GAAG;IACXC,MAAM,EAAE;EACV,CAAC;EACDI,EAAE,EAAE;IACFN,QAAQ,EAAE,KAAK;IACfC,MAAM,EAAE,GAAG;IACXC,MAAM,EAAE;EACV,CAAC;EACDK,EAAE,EAAE;IACFP,QAAQ,EAAE,GAAG;IACbC,MAAM,EAAE,GAAG;IACXC,MAAM,EAAE;EACV,CAAC;EACDM,EAAE,EAAE;IACFR,QAAQ,EAAE,EAAE;IACZC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDO,EAAE,EAAE;IACFT,QAAQ,EAAE,KAAK;IACfC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDQ,EAAE,EAAE;IACFV,QAAQ,EAAE,EAAE;IACZC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDS,EAAE,EAAE;IACFX,QAAQ,EAAE,MAAM;IAChBC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDU,EAAE,EAAE;IACFZ,QAAQ,EAAE,GAAG;IACbC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDW,EAAE,EAAE;IACFb,QAAQ,EAAE,KAAK;IACfC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDY,EAAE,EAAE;IACFd,QAAQ,EAAE,GAAG;IACbC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDa,EAAE,EAAE;IACFf,QAAQ,EAAE,GAAG;IACbC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDc,EAAE,EAAE;IACFhB,QAAQ,EAAE,KAAK;IACfC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDe,EAAE,EAAE;IACFjB,QAAQ,EAAE,KAAK;IACfC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDgB,EAAE,EAAE;IACFlB,QAAQ,EAAE,GAAG;IACbC,MAAM,EAAE,GAAG;IACXC,MAAM,EAAE;EACV,CAAC;EACDiB,EAAE,EAAE;IACFnB,QAAQ,EAAE,KAAK;IACfC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV,CAAC;EACDkB,EAAE,EAAE;IACFpB,QAAQ,EAAE,OAAO;IACjBC,MAAM,EAAEzS,SAAS;IACjB0S,MAAM,EAAE;EACV;AACF,CAAC;AAGD,SAASmB,iBAAiBA,CAACC,GAAG,EAAEC,WAAW,EAAE/M,MAAM,EAAE;EACnD,IAAI,CAAC8M,GAAG,EAAE;IACR,OAAO,IAAI;EACb;EAEA,IAAI;IACF,MAAME,GAAG,GAAGC,IAAI,CAACH,GAAG,CAAC;IACrB,MAAM5W,MAAM,GAAG8W,GAAG,CAAC9W,MAAM;IACzB,MAAMgX,GAAG,GAAG,IAAIC,WAAW,CAACjX,MAAM,CAAC;IACnC,MAAMwD,GAAG,GAAG,IAAI0T,UAAU,CAACF,GAAG,CAAC;IAC/B,KAAK,IAAIjX,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGC,MAAM,EAAED,CAAC,EAAE,EAAE;MAC/ByD,GAAG,CAACzD,CAAC,CAAC,GAAG+W,GAAG,CAACK,UAAU,CAACpX,CAAC,CAAC;IAC5B;IAEA,OAAOqX,GAAG,CAACC,eAAe,CAAC,IAAIC,IAAI,CAAC,CAACN,GAAG,CAAC,EAAE;MACzCO,IAAI,EAAEV;IACR,CAAC,CAAC,CAAC;EACL,CAAC,CAAC,OAAOhJ,GAAG,EAAE;IACZ,IAAI/D,MAAM,EAAE;MACVA,MAAM,CAAC,mCAAmC,EAAE+D,GAAG,CAACjI,OAAO,CAAC;IAC1D;EACF;EAEA,OAAO,IAAI;AACb;AAEA,SAAS4R,eAAeA,CAACZ,GAAG,EAAEC,WAAW,EAAE;EACzC,IAAI,CAACD,GAAG,EAAE;IACR,OAAO,IAAI;EACb;EACAC,WAAW,GAAGA,WAAW,IAAI,YAAY;EACzC,OAAO,OAAO,GAAGA,WAAW,GAAG,UAAU,GAAGD,GAAG;AACjD;AAGA,MAAMa,UAAU,GAAG;EAEjBjB,EAAE,EAAE;IACF7J,IAAI,EAAE9D,CAAC,IAAI,KAAK;IAChBmF,KAAK,EAAEnF,CAAC,IAAI;EACd,CAAC;EACDgN,EAAE,EAAE;IACFlJ,IAAI,EAAE9D,CAAC,IAAI,KAAK;IAChBmF,KAAK,EAAEnF,CAAC,IAAI;EACd,CAAC;EACD+M,EAAE,EAAE;IACFjJ,IAAI,EAAE9D,CAAC,IAAI,OAAO;IAClBmF,KAAK,EAAEnF,CAAC,IAAI;EACd,CAAC;EACD8M,EAAE,EAAE;IACFhJ,IAAI,EAAE9D,CAAC,IAAI,MAAM;IACjBmF,KAAK,EAAEnF,CAAC,IAAI;EACd,CAAC;EAED6M,EAAE,EAAE;IACF/I,IAAI,EAAE9D,CAAC,IAAI,OAAO;IAClBmF,KAAK,EAAEnF,CAAC,IAAI;EACd,CAAC;EAEDmN,EAAE,EAAE;IACFrJ,IAAI,EAAE9D,CAAC,IAAI,EAAE;IACbmF,KAAK,EAAEnF,CAAC,IAAI;EACd,CAAC;EAEDoN,EAAE,EAAE;IACFtJ,IAAI,EAAE9D,CAAC,IAAI,2BAA2B;IACtCmF,KAAK,EAAEnF,CAAC,IAAI;EACd,CAAC;EAEDuN,EAAE,EAAE;IACFzJ,IAAI,EAAG2B,IAAI,IAAK;MACd,OAAO,WAAW,GAAGA,IAAI,CAAC5F,GAAG,GAAG,IAAI;IACtC,CAAC;IACDsF,KAAK,EAAEnF,CAAC,IAAI,MAAM;IAClB6O,KAAK,EAAGpJ,IAAI,IAAK;MACf,OAAOA,IAAI,GAAG;QACZqJ,IAAI,EAAErJ,IAAI,CAAC5F,GAAG;QACd0G,MAAM,EAAE;MACV,CAAC,GAAG,IAAI;IACV;EACF,CAAC;EAEDiH,EAAE,EAAE;IACF1J,IAAI,EAAG2B,IAAI,IAAK;MACd,OAAO,YAAY,GAAGA,IAAI,CAACzP,GAAG,GAAG,IAAI;IACvC,CAAC;IACDmP,KAAK,EAAEnF,CAAC,IAAI,MAAM;IAClB6O,KAAK,EAAGpJ,IAAI,IAAK;MACf,OAAOA,IAAI,GAAG;QACZsJ,EAAE,EAAEtJ,IAAI,CAACzP;MACX,CAAC,GAAG,IAAI;IACV;EACF,CAAC;EAEDqX,EAAE,EAAE;IACFvJ,IAAI,EAAG2B,IAAI,IAAK;MACd,OAAO,YAAY,GAAGA,IAAI,CAACzP,GAAG,GAAG,IAAI;IACvC,CAAC;IACDmP,KAAK,EAAEnF,CAAC,IAAI,MAAM;IAClB6O,KAAK,EAAGpJ,IAAI,IAAK;MACf,OAAOA,IAAI,GAAG;QACZsJ,EAAE,EAAEtJ,IAAI,CAACzP;MACX,CAAC,GAAG,IAAI;IACV;EACF,CAAC;EAED4W,EAAE,EAAE;IACF9I,IAAI,EAAE9D,CAAC,IAAI,UAAU;IACrBmF,KAAK,EAAEnF,CAAC,IAAI,WAAW;IACvB6O,KAAK,EAAGpJ,IAAI,IAAK;MACf,OAAOA,IAAI,GAAG;QACZ,UAAU,EAAEA,IAAI,CAACuJ,GAAG;QACpB,UAAU,EAAEvJ,IAAI,CAACzP,GAAG;QACpB,WAAW,EAAEyP,IAAI,CAACxI,IAAI;QACtB,UAAU,EAAEwI,IAAI,CAACwJ;MACnB,CAAC,GAAG,IAAI;IACV;EACF,CAAC;EAEDzC,EAAE,EAAE;IACF1I,IAAI,EAAG2B,IAAI,IAAK;MACd,MAAM5F,GAAG,GAAG4F,IAAI,CAACwJ,GAAG,IAAInB,iBAAiB,CAACrI,IAAI,CAACzP,GAAG,EAAEyP,IAAI,CAACyJ,IAAI,EAAEC,MAAM,CAAClO,MAAM,CAAC;MAC7E,OAAO,uBAAuB,GAAGpB,GAAG,GAAG,IAAI;IAC7C,CAAC;IACDsF,KAAK,EAAEnF,CAAC,IAAI,UAAU;IACtB6O,KAAK,EAAGpJ,IAAI,IAAK;MACf,IAAI,CAACA,IAAI,EAAE,OAAO,IAAI;MACtB,OAAO;QAEL8C,GAAG,EAAE9C,IAAI,CAACwJ,GAAG,IAAInB,iBAAiB,CAACrI,IAAI,CAACzP,GAAG,EAAEyP,IAAI,CAACyJ,IAAI,EAAEC,MAAM,CAAClO,MAAM,CAAC;QACtE,cAAc,EAAEwE,IAAI,CAACwJ,GAAG,GAAG,UAAU,GAAG,MAAM;QAC9C,eAAe,EAAExJ,IAAI,CAAC2J,QAAQ;QAC9B,WAAW,EAAE3J,IAAI,CAACxI,IAAI;QACtB,WAAW,EAAEwI,IAAI,CAACzP,GAAG,GAAKyP,IAAI,CAACzP,GAAG,CAACmB,MAAM,GAAG,IAAI,GAAI,CAAC,GAAKsO,IAAI,CAAC4J,IAAI,GAAG,CAAE;QACxE,WAAW,EAAE5J,IAAI,CAACyJ;MACpB,CAAC;IACH;EACF,CAAC;EAED5B,EAAE,EAAE;IACFxJ,IAAI,EAAE2B,IAAI,IAAI;MAEZ,MAAM6J,aAAa,GAAGX,eAAe,CAAClJ,IAAI,CAAC8J,YAAY,EAAE9J,IAAI,CAACyJ,IAAI,CAAC;MACnE,MAAMM,UAAU,GAAG1B,iBAAiB,CAACrI,IAAI,CAACzP,GAAG,EAAEyP,IAAI,CAACyJ,IAAI,EAAEC,MAAM,CAAClO,MAAM,CAAC;MACxE,MAAMwO,WAAW,GAAGhK,IAAI,CAACwJ,GAAG,IAAIO,UAAU;MAC1C,OAAO,CAAC/J,IAAI,CAACxI,IAAI,GAAG,WAAW,GAAGwS,WAAW,GAAG,cAAc,GAAGhK,IAAI,CAACxI,IAAI,GAAG,IAAI,GAAG,EAAE,IACpF,YAAY,IAAIqS,aAAa,IAAIE,UAAU,CAAC,GAAG,GAAG,IACjD/J,IAAI,CAACiK,KAAK,GAAG,UAAU,GAAGjK,IAAI,CAACiK,KAAK,GAAG,GAAG,GAAG,EAAE,CAAC,IAChDjK,IAAI,CAACkK,MAAM,GAAG,WAAW,GAAGlK,IAAI,CAACkK,MAAM,GAAG,GAAG,GAAG,EAAE,CAAC,GAAG,gBAAgB;IAC3E,CAAC;IACDxK,KAAK,EAAEM,IAAI,IAAI;MACb,OAAQA,IAAI,CAACxI,IAAI,GAAG,MAAM,GAAG,EAAE;IACjC,CAAC;IACD4R,KAAK,EAAEpJ,IAAI,IAAI;MACb,IAAI,CAACA,IAAI,EAAE,OAAO,IAAI;MACtB,OAAO;QAEL8C,GAAG,EAAEoG,eAAe,CAAClJ,IAAI,CAAC8J,YAAY,EAAE9J,IAAI,CAACyJ,IAAI,CAAC,IAChDzJ,IAAI,CAACwJ,GAAG,IAAInB,iBAAiB,CAACrI,IAAI,CAACzP,GAAG,EAAEyP,IAAI,CAACyJ,IAAI,EAAEC,MAAM,CAAClO,MAAM,CAAC;QACnE2O,KAAK,EAAEnK,IAAI,CAACxI,IAAI;QAChB4S,GAAG,EAAEpK,IAAI,CAACxI,IAAI;QACd,YAAY,EAAEwI,IAAI,CAACiK,KAAK;QACxB,aAAa,EAAEjK,IAAI,CAACkK,MAAM;QAC1B,WAAW,EAAElK,IAAI,CAACxI,IAAI;QACtB,WAAW,EAAEwI,IAAI,CAACwJ,GAAG,GAAIxJ,IAAI,CAAC4J,IAAI,GAAG,CAAC,GAAK5J,IAAI,CAACzP,GAAG,GAAKyP,IAAI,CAACzP,GAAG,CAACmB,MAAM,GAAG,IAAI,GAAI,CAAC,GAAKsO,IAAI,CAAC4J,IAAI,GAAG,CAAG;QACvG,WAAW,EAAE5J,IAAI,CAACyJ;MACpB,CAAC;IACH;EACF,CAAC;EAEDhC,EAAE,EAAE;IACFpJ,IAAI,EAAE9D,CAAC,IAAI,OAAO;IAClBmF,KAAK,EAAEnF,CAAC,IAAI;EACd,CAAC;EAEDyN,EAAE,EAAE;IACF3J,IAAI,EAAE9D,CAAC,IAAI,OAAO;IAClBmF,KAAK,EAAEnF,CAAC,IAAI;EACd,CAAC;EAED0N,EAAE,EAAE;IACF5J,IAAI,EAAE9D,CAAC,IAAI,OAAO;IAClBmF,KAAK,EAAEnF,CAAC,IAAI,QAAQ;IACpB6O,KAAK,EAAGpJ,IAAI,IAAK;MACf,OAAOA,IAAI,GAAG,CAAC,CAAC,GAAG,IAAI;IACzB;EACF,CAAC;EAEDmI,EAAE,EAAE;IACF9J,IAAI,EAAE9D,CAAC,IAAI,OAAO;IAClBmF,KAAK,EAAEnF,CAAC,IAAI,QAAQ;IACpB6O,KAAK,EAAEpJ,IAAI,IAAI;MACb,IAAI,CAACA,IAAI,EAAE,OAAO,CAAC,CAAC;MACpB,OAAO;QACL,eAAe,EAAEA,IAAI,CAAC2J,QAAQ;QAC9B,YAAY,EAAE3J,IAAI,CAACqK;MACrB,CAAC;IACH;EACF,CAAC;EAEDjC,EAAE,EAAE;IACF/J,IAAI,EAAE2B,IAAI,IAAI;MACZ,MAAM6J,aAAa,GAAGX,eAAe,CAAClJ,IAAI,CAAC8J,YAAY,EAAE9J,IAAI,CAACyJ,IAAI,CAAC;MACnE,MAAMM,UAAU,GAAG/J,IAAI,CAACwJ,GAAG,IAAInB,iBAAiB,CAACrI,IAAI,CAACsK,OAAO,EAAEtK,IAAI,CAACuK,OAAO,IAAI,YAAY,EAAEb,MAAM,CAAClO,MAAM,CAAC;MAC3G,OAAO,YAAY,IAAIqO,aAAa,IAAIE,UAAU,CAAC,GAAG,GAAG,IACtD/J,IAAI,CAACiK,KAAK,GAAG,UAAU,GAAGjK,IAAI,CAACiK,KAAK,GAAG,GAAG,GAAG,EAAE,CAAC,IAChDjK,IAAI,CAACkK,MAAM,GAAG,WAAW,GAAGlK,IAAI,CAACkK,MAAM,GAAG,GAAG,GAAG,EAAE,CAAC,GAAG,gBAAgB;IAC3E,CAAC;IACDxK,KAAK,EAAEnF,CAAC,IAAI,EAAE;IACd6O,KAAK,EAAEpJ,IAAI,IAAI;MACb,IAAI,CAACA,IAAI,EAAE,OAAO,IAAI;MACtB,MAAMwK,MAAM,GAAGxK,IAAI,CAACyK,MAAM,IAAIpC,iBAAiB,CAACrI,IAAI,CAACsK,OAAO,EAAEtK,IAAI,CAACuK,OAAO,IAAI,YAAY,EAAEb,MAAM,CAAClO,MAAM,CAAC;MAC1G,OAAO;QAELsH,GAAG,EAAE0H,MAAM;QACX,UAAU,EAAExK,IAAI,CAACwJ,GAAG,IAAInB,iBAAiB,CAACrI,IAAI,CAACzP,GAAG,EAAEyP,IAAI,CAACyJ,IAAI,EAAEC,MAAM,CAAClO,MAAM,CAAC;QAC7E,YAAY,EAAEwE,IAAI,CAACiK,KAAK;QACxB,aAAa,EAAEjK,IAAI,CAACkK,MAAM;QAC1B,cAAc,EAAElK,IAAI,CAACwJ,GAAG,GAAG,UAAU,GAAG,MAAM;QAC9C,cAAc,EAAEgB,MAAM;QACtB,eAAe,EAAExK,IAAI,CAAC2J,QAAQ,GAAG,CAAC;QAClC,WAAW,EAAE3J,IAAI,CAACxI,IAAI;QACtB,WAAW,EAAEwI,IAAI,CAACwJ,GAAG,GAAIxJ,IAAI,CAAC4J,IAAI,GAAG,CAAC,GAAK5J,IAAI,CAACzP,GAAG,GAAKyP,IAAI,CAACzP,GAAG,CAACmB,MAAM,GAAG,IAAI,GAAI,CAAC,GAAKsO,IAAI,CAAC4J,IAAI,GAAG,CAAG;QACvG,WAAW,EAAE5J,IAAI,CAACyJ;MACpB,CAAC;IACH;EACF;AACF,CAAC;AAOD,MAAMC,MAAM,GAAG,SAAAA,CAAA,EAAW;EACxB,IAAI,CAACgB,GAAG,GAAG,EAAE;EACb,IAAI,CAACC,GAAG,GAAG,EAAE;EACb,IAAI,CAACC,GAAG,GAAG,EAAE;AACf,CAAC;AASDlB,MAAM,CAACmB,IAAI,GAAG,UAASC,SAAS,EAAE;EAChC,IAAI,OAAOA,SAAS,IAAI,WAAW,EAAE;IACnCA,SAAS,GAAG,EAAE;EAChB,CAAC,MAAM,IAAI,OAAOA,SAAS,IAAI,QAAQ,EAAE;IACvC,OAAO,IAAI;EACb;EAEA,OAAO;IACLJ,GAAG,EAAEI;EACP,CAAC;AACH,CAAC;AAUDpB,MAAM,CAAC9K,KAAK,GAAG,UAASmM,OAAO,EAAE;EAE/B,IAAI,OAAOA,OAAO,IAAI,QAAQ,EAAE;IAC9B,OAAO,IAAI;EACb;EAGA,MAAMC,KAAK,GAAGD,OAAO,CAACzY,KAAK,CAAC,OAAO,CAAC;EAGpC,MAAM2Y,SAAS,GAAG,EAAE;EACpB,MAAMC,WAAW,GAAG,CAAC,CAAC;EAGtB,MAAMC,GAAG,GAAG,EAAE;EACdH,KAAK,CAACrU,OAAO,CAAEyU,IAAI,IAAK;IACtB,IAAIC,KAAK,GAAG,EAAE;IACd,IAAIC,QAAQ;IAIZhF,aAAa,CAAC3P,OAAO,CAAE4U,GAAG,IAAK;MAE7BF,KAAK,GAAGA,KAAK,CAACG,MAAM,CAACC,QAAQ,CAACL,IAAI,EAAEG,GAAG,CAACnW,KAAK,EAAEmW,GAAG,CAAClW,GAAG,EAAEkW,GAAG,CAAC/T,IAAI,CAAC,CAAC;IACpE,CAAC,CAAC;IAEF,IAAIkU,KAAK;IACT,IAAIL,KAAK,CAAC3Z,MAAM,IAAI,CAAC,EAAE;MACrBga,KAAK,GAAG;QACNhB,GAAG,EAAEU;MACP,CAAC;IACH,CAAC,MAAM;MAELC,KAAK,CAACM,IAAI,CAAC,CAAC9W,CAAC,EAAEC,CAAC,KAAK;QACnB,MAAMvC,IAAI,GAAGsC,CAAC,CAACiB,EAAE,GAAGhB,CAAC,CAACgB,EAAE;QACxB,OAAOvD,IAAI,IAAI,CAAC,GAAGA,IAAI,GAAGuC,CAAC,CAACO,GAAG,GAAGR,CAAC,CAACQ,GAAG;MACzC,CAAC,CAAC;MAGFgW,KAAK,GAAGO,UAAU,CAACP,KAAK,CAAC;MAIzB,MAAMQ,MAAM,GAAGC,QAAQ,CAACV,IAAI,EAAE,CAAC,EAAEA,IAAI,CAAC1Z,MAAM,EAAE2Z,KAAK,CAAC;MAEpD,MAAMU,MAAM,GAAGC,QAAQ,CAACH,MAAM,EAAE,CAAC,CAAC;MAElCH,KAAK,GAAG;QACNhB,GAAG,EAAEqB,MAAM,CAACrB,GAAG;QACfC,GAAG,EAAEoB,MAAM,CAACpB;MACd,CAAC;IACH;IAGAW,QAAQ,GAAGW,eAAe,CAACP,KAAK,CAAChB,GAAG,CAAC;IACrC,IAAIY,QAAQ,CAAC5Z,MAAM,GAAG,CAAC,EAAE;MACvB,MAAMwa,MAAM,GAAG,EAAE;MACjB,KAAK,IAAIza,CAAC,IAAI6Z,QAAQ,EAAE;QAEtB,MAAMa,MAAM,GAAGb,QAAQ,CAAC7Z,CAAC,CAAC;QAC1B,IAAI2a,KAAK,GAAGlB,WAAW,CAACiB,MAAM,CAAC1X,MAAM,CAAC;QACtC,IAAI,CAAC2X,KAAK,EAAE;UACVA,KAAK,GAAGnB,SAAS,CAACvZ,MAAM;UACxBwZ,WAAW,CAACiB,MAAM,CAAC1X,MAAM,CAAC,GAAG2X,KAAK;UAClCnB,SAAS,CAACrG,IAAI,CAAC;YACbyH,EAAE,EAAEF,MAAM,CAAClD,IAAI;YACfjJ,IAAI,EAAEmM,MAAM,CAACnM;UACf,CAAC,CAAC;QACJ;QACAkM,MAAM,CAACtH,IAAI,CAAC;UACV9O,EAAE,EAAEqW,MAAM,CAACG,MAAM;UACjBC,GAAG,EAAEJ,MAAM,CAACI,GAAG;UACfC,GAAG,EAAEJ;QACP,CAAC,CAAC;MACJ;MACAV,KAAK,CAACd,GAAG,GAAGsB,MAAM;IACpB;IAEAf,GAAG,CAACvG,IAAI,CAAC8G,KAAK,CAAC;EACjB,CAAC,CAAC;EAEF,MAAMvK,MAAM,GAAG;IACbuJ,GAAG,EAAE;EACP,CAAC;EAGD,IAAIS,GAAG,CAACzZ,MAAM,GAAG,CAAC,EAAE;IAClByP,MAAM,CAACuJ,GAAG,GAAGS,GAAG,CAAC,CAAC,CAAC,CAACT,GAAG;IACvBvJ,MAAM,CAACwJ,GAAG,GAAG,CAACQ,GAAG,CAAC,CAAC,CAAC,CAACR,GAAG,IAAI,EAAE,EAAEa,MAAM,CAACL,GAAG,CAAC,CAAC,CAAC,CAACP,GAAG,IAAI,EAAE,CAAC;IAExD,KAAK,IAAInZ,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAG0Z,GAAG,CAACzZ,MAAM,EAAED,CAAC,EAAE,EAAE;MACnC,MAAMia,KAAK,GAAGP,GAAG,CAAC1Z,CAAC,CAAC;MACpB,MAAM6a,MAAM,GAAGnL,MAAM,CAACuJ,GAAG,CAAChZ,MAAM,GAAG,CAAC;MAEpCyP,MAAM,CAACwJ,GAAG,CAAC/F,IAAI,CAAC;QACdyH,EAAE,EAAE,IAAI;QACRE,GAAG,EAAE,CAAC;QACNzW,EAAE,EAAEwW,MAAM,GAAG;MACf,CAAC,CAAC;MAEFnL,MAAM,CAACuJ,GAAG,IAAI,GAAG,GAAGgB,KAAK,CAAChB,GAAG;MAC7B,IAAIgB,KAAK,CAACf,GAAG,EAAE;QACbxJ,MAAM,CAACwJ,GAAG,GAAGxJ,MAAM,CAACwJ,GAAG,CAACa,MAAM,CAACE,KAAK,CAACf,GAAG,CAAC8B,GAAG,CAAEC,CAAC,IAAK;UAClDA,CAAC,CAAC5W,EAAE,IAAIwW,MAAM;UACd,OAAOI,CAAC;QACV,CAAC,CAAC,CAAC;MACL;MACA,IAAIhB,KAAK,CAACd,GAAG,EAAE;QACbzJ,MAAM,CAACwJ,GAAG,GAAGxJ,MAAM,CAACwJ,GAAG,CAACa,MAAM,CAACE,KAAK,CAACd,GAAG,CAAC6B,GAAG,CAAEC,CAAC,IAAK;UAClDA,CAAC,CAAC5W,EAAE,IAAIwW,MAAM;UACd,OAAOI,CAAC;QACV,CAAC,CAAC,CAAC;MACL;IACF;IAEA,IAAIvL,MAAM,CAACwJ,GAAG,CAACjZ,MAAM,IAAI,CAAC,EAAE;MAC1B,OAAOyP,MAAM,CAACwJ,GAAG;IACnB;IAEA,IAAIM,SAAS,CAACvZ,MAAM,GAAG,CAAC,EAAE;MACxByP,MAAM,CAACyJ,GAAG,GAAGK,SAAS;IACxB;EACF;EACA,OAAO9J,MAAM;AACf,CAAC;AAUDuI,MAAM,CAACiD,MAAM,GAAG,UAASC,KAAK,EAAEC,MAAM,EAAE;EACtC,IAAI,CAACD,KAAK,EAAE;IACV,OAAOC,MAAM;EACf;EACA,IAAI,CAACA,MAAM,EAAE;IACX,OAAOD,KAAK;EACd;EAEAA,KAAK,CAAClC,GAAG,GAAGkC,KAAK,CAAClC,GAAG,IAAI,EAAE;EAC3B,MAAM6B,GAAG,GAAGK,KAAK,CAAClC,GAAG,CAAChZ,MAAM;EAE5B,IAAI,OAAOmb,MAAM,IAAI,QAAQ,EAAE;IAC7BD,KAAK,CAAClC,GAAG,IAAImC,MAAM;EACrB,CAAC,MAAM,IAAIA,MAAM,CAACnC,GAAG,EAAE;IACrBkC,KAAK,CAAClC,GAAG,IAAImC,MAAM,CAACnC,GAAG;EACzB;EAEA,IAAIvU,KAAK,CAACC,OAAO,CAACyW,MAAM,CAAClC,GAAG,CAAC,EAAE;IAC7BiC,KAAK,CAACjC,GAAG,GAAGiC,KAAK,CAACjC,GAAG,IAAI,EAAE;IAC3B,IAAIxU,KAAK,CAACC,OAAO,CAACyW,MAAM,CAACjC,GAAG,CAAC,EAAE;MAC7BgC,KAAK,CAAChC,GAAG,GAAGgC,KAAK,CAAChC,GAAG,IAAI,EAAE;IAC7B;IACAiC,MAAM,CAAClC,GAAG,CAAChU,OAAO,CAACmM,GAAG,IAAI;MACxB,MAAM6H,GAAG,GAAG;QACV7U,EAAE,EAAE,CAACgN,GAAG,CAAChN,EAAE,GAAG,CAAC,IAAIyW,GAAG;QACtBA,GAAG,EAAEzJ,GAAG,CAACyJ,GAAG,GAAG;MACjB,CAAC;MAED,IAAIzJ,GAAG,CAAChN,EAAE,IAAI,CAAC,CAAC,EAAE;QAChB6U,GAAG,CAAC7U,EAAE,GAAG,CAAC,CAAC;QACX6U,GAAG,CAAC4B,GAAG,GAAG,CAAC;MACb;MACA,IAAIzJ,GAAG,CAACuJ,EAAE,EAAE;QACV1B,GAAG,CAAC0B,EAAE,GAAGvJ,GAAG,CAACuJ,EAAE;MACjB,CAAC,MAAM;QACL1B,GAAG,CAAC6B,GAAG,GAAGI,KAAK,CAAChC,GAAG,CAAClZ,MAAM;QAC1Bkb,KAAK,CAAChC,GAAG,CAAChG,IAAI,CAACiI,MAAM,CAACjC,GAAG,CAAC9H,GAAG,CAAC0J,GAAG,IAAI,CAAC,CAAC,CAAC;MAC1C;MACAI,KAAK,CAACjC,GAAG,CAAC/F,IAAI,CAAC+F,GAAG,CAAC;IACrB,CAAC,CAAC;EACJ;EAEA,OAAOiC,KAAK;AACd,CAAC;AA8BDlD,MAAM,CAACoD,WAAW,GAAG,UAAS/B,OAAO,EAAEjV,EAAE,EAAEiX,SAAS,EAAE;EACpDhC,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EACDK,OAAO,CAACH,GAAG,GAAGG,OAAO,CAACH,GAAG,IAAI,EAAE;EAC/BG,OAAO,CAACJ,GAAG,GAAGI,OAAO,CAACJ,GAAG,IAAI,EAAE;EAE/BI,OAAO,CAACJ,GAAG,CAAC/F,IAAI,CAAC;IACf9O,EAAE,EAAEA,EAAE,GAAG,CAAC;IACVyW,GAAG,EAAE,CAAC;IACNC,GAAG,EAAEzB,OAAO,CAACH,GAAG,CAAClZ;EACnB,CAAC,CAAC;EAEF,MAAMsb,EAAE,GAAG;IACTX,EAAE,EAAE,IAAI;IACRrM,IAAI,EAAE;MACJyJ,IAAI,EAAEsD,SAAS,CAACtD,IAAI;MACpBD,GAAG,EAAEuD,SAAS,CAACE,MAAM;MACrB1c,GAAG,EAAEwc,SAAS,CAACG,IAAI,IAAIH,SAAS,CAACzC,OAAO;MACxCL,KAAK,EAAE8C,SAAS,CAAC9C,KAAK;MACtBC,MAAM,EAAE6C,SAAS,CAAC7C,MAAM;MACxB1S,IAAI,EAAEuV,SAAS,CAACI,QAAQ;MACxBvD,IAAI,EAAEmD,SAAS,CAACnD,IAAI,GAAG;IACzB;EACF,CAAC;EAED,IAAImD,SAAS,CAACK,UAAU,EAAE;IACxBJ,EAAE,CAAChN,IAAI,CAAC8J,YAAY,GAAGiD,SAAS,CAACjD,YAAY;IAC7CkD,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG,IAAI;IAC1BN,SAAS,CAACK,UAAU,CAACE,IAAI,CACvBlT,GAAG,IAAI;MACL4S,EAAE,CAAChN,IAAI,CAACwJ,GAAG,GAAGpP,GAAG;MACjB4S,EAAE,CAAChN,IAAI,CAAC8J,YAAY,GAAGtV,SAAS;MAChCwY,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG7Y,SAAS;IACjC,CAAC,EACD+F,CAAC,IAAI;MAEHyS,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG7Y,SAAS;IACjC,CACF,CAAC;EACH;EAEAuW,OAAO,CAACH,GAAG,CAAChG,IAAI,CAACoI,EAAE,CAAC;EAEpB,OAAOjC,OAAO;AAChB,CAAC;AAiCDrB,MAAM,CAAC6D,WAAW,GAAG,UAASxC,OAAO,EAAEjV,EAAE,EAAE0X,SAAS,EAAE;EACpDzC,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EACDK,OAAO,CAACH,GAAG,GAAGG,OAAO,CAACH,GAAG,IAAI,EAAE;EAC/BG,OAAO,CAACJ,GAAG,GAAGI,OAAO,CAACJ,GAAG,IAAI,EAAE;EAE/BI,OAAO,CAACJ,GAAG,CAAC/F,IAAI,CAAC;IACf9O,EAAE,EAAEA,EAAE,GAAG,CAAC;IACVyW,GAAG,EAAE,CAAC;IACNC,GAAG,EAAEzB,OAAO,CAACH,GAAG,CAAClZ;EACnB,CAAC,CAAC;EAEF,MAAMsb,EAAE,GAAG;IACTX,EAAE,EAAE,IAAI;IACRrM,IAAI,EAAE;MACJyJ,IAAI,EAAE+D,SAAS,CAAC/D,IAAI;MACpBD,GAAG,EAAEgE,SAAS,CAACP,MAAM;MACrB1c,GAAG,EAAEid,SAAS,CAACN,IAAI;MACnBzC,MAAM,EAAE+C,SAAS,CAAC/C,MAAM;MACxBH,OAAO,EAAEkD,SAAS,CAAClD,OAAO;MAC1BL,KAAK,EAAEuD,SAAS,CAACvD,KAAK;MACtBC,MAAM,EAAEsD,SAAS,CAACtD,MAAM;MACxBP,QAAQ,EAAE6D,SAAS,CAAC7D,QAAQ,GAAG,CAAC;MAChCnS,IAAI,EAAEgW,SAAS,CAACL,QAAQ;MACxBvD,IAAI,EAAE4D,SAAS,CAAC5D,IAAI,GAAG;IACzB;EACF,CAAC;EAED,IAAI4D,SAAS,CAACJ,UAAU,EAAE;IACxBJ,EAAE,CAAChN,IAAI,CAAC8J,YAAY,GAAG0D,SAAS,CAAC1D,YAAY;IAC7CkD,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG,IAAI;IAC1BG,SAAS,CAACJ,UAAU,CAACE,IAAI,CACvBG,IAAI,IAAI;MACNT,EAAE,CAAChN,IAAI,CAACwJ,GAAG,GAAGiE,IAAI,CAAC,CAAC,CAAC;MACrBT,EAAE,CAAChN,IAAI,CAACyK,MAAM,GAAGgD,IAAI,CAAC,CAAC,CAAC;MACxBT,EAAE,CAAChN,IAAI,CAAC8J,YAAY,GAAGtV,SAAS;MAChCwY,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG7Y,SAAS;IACjC,CAAC,EACD+F,CAAC,IAAI;MAEHyS,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG7Y,SAAS;IACjC,CACF,CAAC;EACH;EAEAuW,OAAO,CAACH,GAAG,CAAChG,IAAI,CAACoI,EAAE,CAAC;EAEpB,OAAOjC,OAAO;AAChB,CAAC;AA4BDrB,MAAM,CAACgE,WAAW,GAAG,UAAS3C,OAAO,EAAEjV,EAAE,EAAE6X,SAAS,EAAE;EACpD5C,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EACDK,OAAO,CAACH,GAAG,GAAGG,OAAO,CAACH,GAAG,IAAI,EAAE;EAC/BG,OAAO,CAACJ,GAAG,GAAGI,OAAO,CAACJ,GAAG,IAAI,EAAE;EAE/BI,OAAO,CAACJ,GAAG,CAAC/F,IAAI,CAAC;IACf9O,EAAE,EAAEA,EAAE,GAAG,CAAC;IACVyW,GAAG,EAAE,CAAC;IACNC,GAAG,EAAEzB,OAAO,CAACH,GAAG,CAAClZ;EACnB,CAAC,CAAC;EAEF,MAAMsb,EAAE,GAAG;IACTX,EAAE,EAAE,IAAI;IACRrM,IAAI,EAAE;MACJyJ,IAAI,EAAEkE,SAAS,CAAClE,IAAI;MACpBlZ,GAAG,EAAEod,SAAS,CAACT,IAAI;MACnBvD,QAAQ,EAAEgE,SAAS,CAAChE,QAAQ,GAAG,CAAC;MAChCW,OAAO,EAAEqD,SAAS,CAACrD,OAAO;MAC1B9S,IAAI,EAAEmW,SAAS,CAACR,QAAQ;MACxBvD,IAAI,EAAE+D,SAAS,CAAC/D,IAAI,GAAG,CAAC;MACxBJ,GAAG,EAAEmE,SAAS,CAACV;IACjB;EACF,CAAC;EAED,IAAIU,SAAS,CAACP,UAAU,EAAE;IACxBJ,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG,IAAI;IAC1BM,SAAS,CAACP,UAAU,CAACE,IAAI,CACvBlT,GAAG,IAAI;MACL4S,EAAE,CAAChN,IAAI,CAACwJ,GAAG,GAAGpP,GAAG;MACjB4S,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG7Y,SAAS;IACjC,CAAC,EACD+F,CAAC,IAAI;MAEHyS,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG7Y,SAAS;IACjC,CACF,CAAC;EACH;EAEAuW,OAAO,CAACH,GAAG,CAAChG,IAAI,CAACoI,EAAE,CAAC;EAEpB,OAAOjC,OAAO;AAChB,CAAC;AASDrB,MAAM,CAACkE,SAAS,GAAG,UAASC,SAAS,EAAE;EACrC,MAAM9C,OAAO,GAAG;IACdL,GAAG,EAAE,GAAG;IACRC,GAAG,EAAE,CAAC;MACJ7U,EAAE,EAAE,CAAC;MACLyW,GAAG,EAAE,CAAC;MACNC,GAAG,EAAE;IACP,CAAC,CAAC;IACF5B,GAAG,EAAE,CAAC;MACJyB,EAAE,EAAE,IAAI;MACRrM,IAAI,EAAE;QACJ8N,KAAK,EAAED;MACT;IACF,CAAC;EACH,CAAC;EACD,OAAO9C,OAAO;AAChB,CAAC;AAcDrB,MAAM,CAACqE,eAAe,GAAG,UAAShD,OAAO,EAAEhM,MAAM,EAAE;EAGjD,MAAM4L,GAAG,GAAG,CAAC,CAACI,OAAO,IAAI,CAAC,CAAC,EAAEJ,GAAG,IAAI,EAAE,EAAE,CAAC,CAAC;EAC1C,IAAI,CAACA,GAAG,EAAE;IAER,OAAOI,OAAO;EAChB;EAEA,IAAIH,GAAG;EACP,IAAID,GAAG,CAAC0B,EAAE,IAAI,IAAI,EAAE;IAElB,OAAO1B,GAAG,CAAC0B,EAAE;IACb1B,GAAG,CAAC6B,GAAG,GAAG,CAAC;IACX5B,GAAG,GAAG;MACJyB,EAAE,EAAE;IACN,CAAC;IACDtB,OAAO,CAACH,GAAG,GAAG,CAACA,GAAG,CAAC;EACrB,CAAC,MAAM;IACLA,GAAG,GAAG,CAACG,OAAO,CAACH,GAAG,IAAI,EAAE,EAAED,GAAG,CAAC6B,GAAG,GAAG,CAAC,CAAC;IACtC,IAAI,CAAC5B,GAAG,IAAIA,GAAG,CAACyB,EAAE,IAAI,IAAI,EAAE;MAE1B,OAAOtB,OAAO;IAChB;EACF;EACAH,GAAG,CAAC5K,IAAI,GAAG4K,GAAG,CAAC5K,IAAI,IAAI,CAAC,CAAC;EACzBgO,MAAM,CAACC,MAAM,CAACrD,GAAG,CAAC5K,IAAI,EAAEjB,MAAM,CAAC;EAC/B,OAAOgM,OAAO;AAChB,CAAC;AAaDrB,MAAM,CAACwE,KAAK,GAAG,UAASC,MAAM,EAAEnL,GAAG,EAAEoL,IAAI,EAAE;EACzC,MAAMF,KAAK,GAAGxE,MAAM,CAACiD,MAAM,CAACjD,MAAM,CAAC2E,eAAe,CAAC3E,MAAM,CAAC4E,OAAO,CAACH,MAAM,EAAEnL,GAAG,CAAC,CAAC,EAAEoL,IAAI,CAAC;EAGtFF,KAAK,CAACvD,GAAG,CAAC/F,IAAI,CAAC;IACb9O,EAAE,EAAE,CAAC;IACLyW,GAAG,EAAE2B,KAAK,CAACxD,GAAG,CAAChZ,MAAM;IACrB2a,EAAE,EAAE;EACN,CAAC,CAAC;EAEF,OAAO6B,KAAK;AACd,CAAC;AAUDxE,MAAM,CAAC4E,OAAO,GAAG,UAAS9W,IAAI,EAAEwL,GAAG,EAAE;EACnC,OAAO;IACL0H,GAAG,EAAElT,IAAI,IAAI,EAAE;IACfmT,GAAG,EAAE,CAAC;MACJ7U,EAAE,EAAE,CAAC;MACLyW,GAAG,EAAE,CAAC/U,IAAI,IAAI,EAAE,EAAE9F,MAAM;MACxB8a,GAAG,EAAE;IACP,CAAC,CAAC;IACF5B,GAAG,EAAE,CAAC;MACJyB,EAAE,EAAE,IAAI;MACRrM,IAAI,EAAE;QACJzP,GAAG,EAAEyS;MACP;IACF,CAAC;EACH,CAAC;AACH,CAAC;AAUD0G,MAAM,CAAC6E,UAAU,GAAG,UAASxD,OAAO,EAAEyD,QAAQ,EAAE;EAC9CzD,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EAEDK,OAAO,CAACH,GAAG,GAAGG,OAAO,CAACH,GAAG,IAAI,EAAE;EAC/BG,OAAO,CAACJ,GAAG,GAAGI,OAAO,CAACJ,GAAG,IAAI,EAAE;EAE/BI,OAAO,CAACJ,GAAG,CAAC/F,IAAI,CAAC;IACf9O,EAAE,EAAEiV,OAAO,CAACL,GAAG,CAAChZ,MAAM;IACtB6a,GAAG,EAAEiC,QAAQ,CAAC9D,GAAG,CAAChZ,MAAM;IACxB8a,GAAG,EAAEzB,OAAO,CAACH,GAAG,CAAClZ;EACnB,CAAC,CAAC;EACFqZ,OAAO,CAACL,GAAG,IAAI8D,QAAQ,CAAC9D,GAAG;EAE3B,MAAMsC,EAAE,GAAG;IACTX,EAAE,EAAE,IAAI;IACRrM,IAAI,EAAE;MACJ5F,GAAG,EAAEoU,QAAQ,CAACpU;IAChB;EACF,CAAC;EACD2Q,OAAO,CAACH,GAAG,CAAChG,IAAI,CAACoI,EAAE,CAAC;EAEpB,OAAOjC,OAAO;AAChB,CAAC;AAYDrB,MAAM,CAAC+E,WAAW,GAAG,UAAS1D,OAAO,EAAEgC,SAAS,EAAE;EAChDhC,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EACDK,OAAO,CAACL,GAAG,IAAI,GAAG;EAClB,OAAOhB,MAAM,CAACoD,WAAW,CAAC/B,OAAO,EAAEA,OAAO,CAACL,GAAG,CAAChZ,MAAM,GAAG,CAAC,EAAEqb,SAAS,CAAC;AACvE,CAAC;AAYDrD,MAAM,CAACgF,WAAW,GAAG,UAAS3D,OAAO,EAAE4C,SAAS,EAAE;EAChD5C,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EACDK,OAAO,CAACL,GAAG,IAAI,GAAG;EAClB,OAAOhB,MAAM,CAACgE,WAAW,CAAC3C,OAAO,EAAEA,OAAO,CAACL,GAAG,CAAChZ,MAAM,GAAG,CAAC,EAAEic,SAAS,CAAC;AACvE,CAAC;AAyBDjE,MAAM,CAACiF,UAAU,GAAG,UAAS5D,OAAO,EAAE6D,cAAc,EAAE;EACpD7D,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EAEDK,OAAO,CAACH,GAAG,GAAGG,OAAO,CAACH,GAAG,IAAI,EAAE;EAC/BG,OAAO,CAACJ,GAAG,GAAGI,OAAO,CAACJ,GAAG,IAAI,EAAE;EAE/BI,OAAO,CAACJ,GAAG,CAAC/F,IAAI,CAAC;IACf9O,EAAE,EAAE,CAAC,CAAC;IACNyW,GAAG,EAAE,CAAC;IACNC,GAAG,EAAEzB,OAAO,CAACH,GAAG,CAAClZ;EACnB,CAAC,CAAC;EAEF,MAAMsb,EAAE,GAAG;IACTX,EAAE,EAAE,IAAI;IACRrM,IAAI,EAAE;MACJyJ,IAAI,EAAEmF,cAAc,CAACnF,IAAI;MACzBlZ,GAAG,EAAEqe,cAAc,CAAC5O,IAAI;MACxBxI,IAAI,EAAEoX,cAAc,CAACzB,QAAQ;MAC7B3D,GAAG,EAAEoF,cAAc,CAAC3B,MAAM;MAC1BrD,IAAI,EAAEgF,cAAc,CAAChF,IAAI,GAAG;IAC9B;EACF,CAAC;EACD,IAAIgF,cAAc,CAACxB,UAAU,EAAE;IAC7BJ,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG,IAAI;IAC1BuB,cAAc,CAACxB,UAAU,CAACE,IAAI,CAC5BlT,GAAG,IAAI;MACL4S,EAAE,CAAChN,IAAI,CAACwJ,GAAG,GAAGpP,GAAG;MACjB4S,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG7Y,SAAS;IACjC,CAAC,EACD+F,CAAC,IAAI;MAEHyS,EAAE,CAAChN,IAAI,CAACqN,WAAW,GAAG7Y,SAAS;IACjC,CACF,CAAC;EACH;EACAuW,OAAO,CAACH,GAAG,CAAChG,IAAI,CAACoI,EAAE,CAAC;EAEpB,OAAOjC,OAAO;AAChB,CAAC;AAcDrB,MAAM,CAACmF,QAAQ,GAAG,UAAS9D,OAAO,EAAE+D,KAAK,EAAEhZ,EAAE,EAAEyW,GAAG,EAAE;EAClD,IAAI,OAAOxB,OAAO,IAAI,QAAQ,EAAE;IAC9BA,OAAO,GAAG;MACRL,GAAG,EAAEK;IACP,CAAC;EACH;EACAA,OAAO,CAACJ,GAAG,GAAGI,OAAO,CAACJ,GAAG,IAAI,EAAE;EAE/BI,OAAO,CAACJ,GAAG,CAAC/F,IAAI,CAAC;IACf9O,EAAE,EAAEA,EAAE,IAAI,CAAC;IACXyW,GAAG,EAAEA,GAAG,IAAIxB,OAAO,CAACL,GAAG,CAAChZ,MAAM;IAC9B2a,EAAE,EAAEyC;EACN,CAAC,CAAC;EAEF,OAAO/D,OAAO;AAChB,CAAC;AAaDrB,MAAM,CAACqF,UAAU,GAAG,UAAShE,OAAO,EAAEjV,EAAE,EAAEyW,GAAG,EAAE;EAC7C,OAAO7C,MAAM,CAACmF,QAAQ,CAAC9D,OAAO,EAAE,IAAI,EAAEjV,EAAE,EAAEyW,GAAG,CAAC;AAChD,CAAC;AAiBD7C,MAAM,CAACsF,YAAY,GAAG,UAASjE,OAAO,EAAEjV,EAAE,EAAEyW,GAAG,EAAE/U,IAAI,EAAEyX,UAAU,EAAEC,WAAW,EAAEC,MAAM,EAAE;EACtF,IAAI,OAAOpE,OAAO,IAAI,QAAQ,EAAE;IAC9BA,OAAO,GAAG;MACRL,GAAG,EAAEK;IACP,CAAC;EACH;EAEA,IAAI,CAACA,OAAO,IAAI,CAACA,OAAO,CAACL,GAAG,IAAIK,OAAO,CAACL,GAAG,CAAChZ,MAAM,GAAGoE,EAAE,GAAGyW,GAAG,EAAE;IAC7D,OAAO,IAAI;EACb;EAEA,IAAIA,GAAG,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC6C,OAAO,CAACH,UAAU,CAAC,IAAI,CAAC,CAAC,EAAE;IACxD,OAAO,IAAI;EACb;EAEA,IAAIA,UAAU,IAAI,KAAK,IAAI,CAACE,MAAM,EAAE;IAClC,OAAO,IAAI;EACb;EACAA,MAAM,GAAG,EAAE,GAAGA,MAAM;EAEpBpE,OAAO,CAACH,GAAG,GAAGG,OAAO,CAACH,GAAG,IAAI,EAAE;EAC/BG,OAAO,CAACJ,GAAG,GAAGI,OAAO,CAACJ,GAAG,IAAI,EAAE;EAE/BI,OAAO,CAACJ,GAAG,CAAC/F,IAAI,CAAC;IACf9O,EAAE,EAAEA,EAAE,GAAG,CAAC;IACVyW,GAAG,EAAEA,GAAG;IACRC,GAAG,EAAEzB,OAAO,CAACH,GAAG,CAAClZ;EACnB,CAAC,CAAC;EACFqZ,OAAO,CAACH,GAAG,CAAChG,IAAI,CAAC;IACfyH,EAAE,EAAE,IAAI;IACRrM,IAAI,EAAE;MACJuJ,GAAG,EAAE0F,UAAU;MACf1e,GAAG,EAAE2e,WAAW;MAChB1F,GAAG,EAAE2F,MAAM;MACX3X,IAAI,EAAEA;IACR;EACF,CAAC,CAAC;EAEF,OAAOuT,OAAO;AAChB,CAAC;AAgBDrB,MAAM,CAAC2F,YAAY,GAAG,UAAStE,OAAO,EAAEZ,KAAK,EAAE3S,IAAI,EAAEyX,UAAU,EAAEC,WAAW,EAAEC,MAAM,EAAE;EACpFpE,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EACD,MAAM5U,EAAE,GAAGiV,OAAO,CAACL,GAAG,CAAChZ,MAAM;EAC7BqZ,OAAO,CAACL,GAAG,IAAIP,KAAK;EACpB,OAAOT,MAAM,CAACsF,YAAY,CAACjE,OAAO,EAAEjV,EAAE,EAAEqU,KAAK,CAACzY,MAAM,EAAE8F,IAAI,EAAEyX,UAAU,EAAEC,WAAW,EAAEC,MAAM,CAAC;AAC9F,CAAC;AAaDzF,MAAM,CAAC4F,UAAU,GAAG,UAASvE,OAAO,EAAE/K,IAAI,EAAE;EAC1C+K,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EACDK,OAAO,CAACH,GAAG,GAAGG,OAAO,CAACH,GAAG,IAAI,EAAE;EAC/BG,OAAO,CAACJ,GAAG,GAAGI,OAAO,CAACJ,GAAG,IAAI,EAAE;EAE/BI,OAAO,CAACJ,GAAG,CAAC/F,IAAI,CAAC;IACf9O,EAAE,EAAE,CAAC,CAAC;IACNyW,GAAG,EAAE,CAAC;IACNC,GAAG,EAAEzB,OAAO,CAACH,GAAG,CAAClZ;EACnB,CAAC,CAAC;EAEFqZ,OAAO,CAACH,GAAG,CAAChG,IAAI,CAAC;IACfyH,EAAE,EAAE,IAAI;IACRrM,IAAI,EAAE;MACJyJ,IAAI,EAAEtD,cAAc;MACpB5V,GAAG,EAAEyP;IACP;EACF,CAAC,CAAC;EAEF,OAAO+K,OAAO;AAChB,CAAC;AASDrB,MAAM,CAAC2E,eAAe,GAAG,UAAStD,OAAO,EAAE;EACzCA,OAAO,GAAGA,OAAO,IAAI;IACnBL,GAAG,EAAE;EACP,CAAC;EACDK,OAAO,CAACJ,GAAG,GAAGI,OAAO,CAACJ,GAAG,IAAI,EAAE;EAC/BI,OAAO,CAACJ,GAAG,CAAC/F,IAAI,CAAC;IACf9O,EAAE,EAAEiV,OAAO,CAACL,GAAG,CAAChZ,MAAM;IACtB6a,GAAG,EAAE,CAAC;IACNF,EAAE,EAAE;EACN,CAAC,CAAC;EACFtB,OAAO,CAACL,GAAG,IAAI,GAAG;EAElB,OAAOK,OAAO;AAChB,CAAC;AAaDrB,MAAM,CAAC6F,aAAa,GAAG,UAASC,GAAG,EAAE;EACnC,MAAMC,IAAI,GAAGC,YAAY,CAACF,GAAG,CAAC;EAC9B,MAAMG,aAAa,GAAG,SAAAA,CAAS1G,IAAI,EAAEjJ,IAAI,EAAE4P,MAAM,EAAE;IACjD,MAAMrE,GAAG,GAAGpC,UAAU,CAACF,IAAI,CAAC;IAC5B,IAAI9H,MAAM,GAAGyO,MAAM,GAAGA,MAAM,CAACC,IAAI,CAAC,EAAE,CAAC,GAAG,EAAE;IAC1C,IAAItE,GAAG,EAAE;MACPpK,MAAM,GAAGoK,GAAG,CAAClN,IAAI,CAAC2B,IAAI,CAAC,GAAGmB,MAAM,GAAGoK,GAAG,CAAC7L,KAAK,CAACM,IAAI,CAAC;IACpD;IACA,OAAOmB,MAAM;EACf,CAAC;EACD,OAAO2O,YAAY,CAACL,IAAI,EAAEE,aAAa,EAAE,CAAC,CAAC;AAC7C,CAAC;AA4BDjG,MAAM,CAACqG,MAAM,GAAG,UAASC,QAAQ,EAAEC,SAAS,EAAElZ,OAAO,EAAE;EACrD,OAAO+Y,YAAY,CAACJ,YAAY,CAACM,QAAQ,CAAC,EAAEC,SAAS,EAAE,CAAC,EAAE,EAAE,EAAElZ,OAAO,CAAC;AACxE,CAAC;AAYD2S,MAAM,CAACwG,OAAO,GAAG,UAASF,QAAQ,EAAExL,KAAK,EAAE2L,KAAK,EAAE;EAChD,IAAIV,IAAI,GAAGC,YAAY,CAACM,QAAQ,CAAC;EACjCP,IAAI,GAAGW,WAAW,CAACX,IAAI,EAAEjL,KAAK,EAAE,GAAG,CAAC;EACpC,IAAIiL,IAAI,IAAIU,KAAK,EAAE;IACjBV,IAAI,GAAGY,WAAW,CAACZ,IAAI,CAAC;EAC1B;EACA,OAAOa,YAAY,CAAC,CAAC,CAAC,EAAEb,IAAI,EAAE,EAAE,CAAC;AACnC,CAAC;AAUD/F,MAAM,CAAC6G,gBAAgB,GAAG,UAASP,QAAQ,EAAE;EAC3C,IAAIP,IAAI,GAAGC,YAAY,CAACM,QAAQ,CAAC;EACjC,MAAMQ,SAAS,GAAG,SAAAA,CAASC,IAAI,EAAE;IAC/B,IAAIA,IAAI,CAACxH,IAAI,IAAI,IAAI,EAAE;MACrB,IAAI,CAACwH,IAAI,CAACC,MAAM,IAAI,CAACD,IAAI,CAACC,MAAM,CAACzH,IAAI,EAAE;QACrC,OAAO,IAAI;MACb;IACF;IACA,OAAOwH,IAAI;EACb,CAAC;EAEDhB,IAAI,GAAGkB,WAAW,CAAClB,IAAI,EAAEe,SAAS,CAAC;EAEnCf,IAAI,GAAGmB,KAAK,CAACnB,IAAI,CAAC;EAElB,OAAOa,YAAY,CAAC,CAAC,CAAC,EAAEb,IAAI,EAAE,EAAE,CAAC;AACnC,CAAC;AAgBD/F,MAAM,CAACmH,YAAY,GAAG,UAASb,QAAQ,EAAExL,KAAK,EAAE;EAC9C,MAAMsM,YAAY,GAAG,SAAAA,CAASL,IAAI,EAAE;IAClC,IAAIA,IAAI,CAACxH,IAAI,IAAI,IAAI,EAAE;MACrB,OAAO,IAAI;IACb,CAAC,MAAM,IAAIwH,IAAI,CAACxH,IAAI,IAAI,IAAI,EAAE;MAC5B,IAAI,CAAC,CAACwH,IAAI,CAACC,MAAM,IAAI,CAACD,IAAI,CAACC,MAAM,CAACzH,IAAI,KAAK,CAACwH,IAAI,CAACpR,IAAI,IAAI,EAAE,EAAE0R,UAAU,CAAC,GAAG,CAAC,EAAE;QAC5EN,IAAI,CAACpR,IAAI,GAAG,GAAG;QACf,OAAOoR,IAAI,CAACO,QAAQ;QACpB,OAAOP,IAAI,CAACzQ,IAAI;MAClB;IACF,CAAC,MAAM,IAAIyQ,IAAI,CAACxH,IAAI,IAAI,IAAI,EAAE;MAC5BwH,IAAI,CAACpR,IAAI,GAAG,GAAG;MACf,OAAOoR,IAAI,CAACxH,IAAI;MAChB,OAAOwH,IAAI,CAACO,QAAQ;IACtB;IACA,OAAOP,IAAI;EACb,CAAC;EAED,IAAIhB,IAAI,GAAGC,YAAY,CAACM,QAAQ,CAAC;EACjC,IAAI,CAACP,IAAI,EAAE;IACT,OAAOO,QAAQ;EACjB;EAGAP,IAAI,GAAGkB,WAAW,CAAClB,IAAI,EAAEqB,YAAY,CAAC;EAEtCrB,IAAI,GAAGwB,gBAAgB,CAACxB,IAAI,EAAExJ,uBAAuB,CAAC;EAEtDwJ,IAAI,GAAGW,WAAW,CAACX,IAAI,EAAEjL,KAAK,EAAE,GAAG,CAAC;EAEpC,MAAMrN,MAAM,GAAGsZ,IAAI,IAAI;IACrB,QAAQA,IAAI,CAACxH,IAAI;MACf,KAAK,IAAI;QACP,OAAO,CAAC,KAAK,CAAC;MAChB,KAAK,IAAI;QACP,OAAO,CAAC,SAAS,CAAC;IACtB;IACA,OAAO,IAAI;EACb,CAAC;EACDwG,IAAI,GAAGY,WAAW,CAACZ,IAAI,EAAEtY,MAAM,CAAC;EAEhC,OAAOmZ,YAAY,CAAC,CAAC,CAAC,EAAEb,IAAI,EAAE,EAAE,CAAC;AACnC,CAAC;AAqBD/F,MAAM,CAACY,OAAO,GAAG,UAAS0F,QAAQ,EAAExL,KAAK,EAAE0M,UAAU,EAAE;EACrD,IAAIzB,IAAI,GAAGC,YAAY,CAACM,QAAQ,CAAC;EAGjCP,IAAI,GAAGwB,gBAAgB,CAACxB,IAAI,EAAExJ,uBAAuB,CAAC;EAGtD,MAAM6K,YAAY,GAAG,SAAAA,CAASL,IAAI,EAAE;IAClC,IAAIA,IAAI,CAACxH,IAAI,IAAI,IAAI,EAAE;MACrB,IAAI,CAAC,CAACwH,IAAI,CAACC,MAAM,IAAI,CAACD,IAAI,CAACC,MAAM,CAACzH,IAAI,KAAK,CAACwH,IAAI,CAACpR,IAAI,IAAI,EAAE,EAAE0R,UAAU,CAAC,GAAG,CAAC,EAAE;QAC5EN,IAAI,CAACpR,IAAI,GAAG,GAAG;QACf,OAAOoR,IAAI,CAACO,QAAQ;MACtB;IACF,CAAC,MAAM,IAAIP,IAAI,CAACxH,IAAI,IAAI,IAAI,EAAE;MAC5BwH,IAAI,CAACpR,IAAI,GAAG,GAAG;MACf,OAAOoR,IAAI,CAACO,QAAQ;IACtB,CAAC,MAAM,IAAIP,IAAI,CAACxH,IAAI,IAAI,IAAI,EAAE;MAC5BwH,IAAI,CAACpR,IAAI,GAAG,GAAG;MACf,OAAOoR,IAAI,CAACO,QAAQ;MACpB,OAAOP,IAAI,CAACxH,IAAI;IAClB;IACA,OAAOwH,IAAI;EACb,CAAC;EACDhB,IAAI,GAAGkB,WAAW,CAAClB,IAAI,EAAEqB,YAAY,CAAC;EAEtCrB,IAAI,GAAGW,WAAW,CAACX,IAAI,EAAEjL,KAAK,EAAE,GAAG,CAAC;EACpC,IAAI0M,UAAU,EAAE;IAEd,MAAM/Z,MAAM,GAAG;MACb0Q,EAAE,EAAE,CAAC,KAAK,CAAC;MACXO,EAAE,EAAE,CAAC,SAAS;IAChB,CAAC;IACDqH,IAAI,GAAGY,WAAW,CAACZ,IAAI,EAAEgB,IAAI,IAAI;MAC/B,OAAOtZ,MAAM,CAACsZ,IAAI,CAACxH,IAAI,CAAC;IAC1B,CAAC,CAAC;EACJ,CAAC,MAAM;IACLwG,IAAI,GAAGY,WAAW,CAACZ,IAAI,CAAC;EAC1B;EAGA,OAAOa,YAAY,CAAC,CAAC,CAAC,EAAEb,IAAI,EAAE,EAAE,CAAC;AACnC,CAAC;AAUD/F,MAAM,CAACyH,WAAW,GAAG,UAASpG,OAAO,EAAE;EACrC,OAAO,OAAOA,OAAO,IAAI,QAAQ,GAAGA,OAAO,GAAGA,OAAO,CAACL,GAAG;AAC3D,CAAC;AAUDhB,MAAM,CAAC0H,WAAW,GAAG,UAASrG,OAAO,EAAE;EACrC,OAAO,OAAOA,OAAO,IAAI,QAAQ,IAAI,EAAEA,OAAO,CAACJ,GAAG,IAAII,OAAO,CAACH,GAAG,CAAC;AACpE,CAAC;AAUDlB,MAAM,CAAC2H,UAAU,GAAG,UAAStG,OAAO,EAAE;EACpC,IAAI0E,IAAI,GAAGC,YAAY,CAAC3E,OAAO,CAAC;EAChC,MAAMuG,WAAW,GAAG,SAAAA,CAASrI,IAAI,EAAE1O,CAAC,EAAEqV,MAAM,EAAE;IAC5C,MAAM2B,GAAG,GAAGzK,WAAW,CAACmC,IAAI,CAAC;IAC7B,IAAI9H,MAAM,GAAIyO,MAAM,GAAGA,MAAM,CAACC,IAAI,CAAC,EAAE,CAAC,GAAG,EAAG;IAC5C,IAAI0B,GAAG,EAAE;MACP,IAAIA,GAAG,CAACrK,MAAM,EAAE;QACd/F,MAAM,GAAGoQ,GAAG,CAACtK,MAAM,IAAI,EAAE;MAC3B,CAAC,MAAM,IAAIsK,GAAG,CAACtK,MAAM,EAAE;QACrB9F,MAAM,GAAGoQ,GAAG,CAACtK,MAAM,GAAG9F,MAAM,GAAGoQ,GAAG,CAACtK,MAAM;MAC3C;IACF;IACA,OAAO9F,MAAM;EACf,CAAC;EACD,OAAO2O,YAAY,CAACL,IAAI,EAAE6B,WAAW,EAAE,CAAC,CAAC;AAC3C,CAAC;AAUD5H,MAAM,CAAC8H,OAAO,GAAG,UAASzG,OAAO,EAAE;EACjC,IAAI,CAACA,OAAO,EAAE;IACZ,OAAO,KAAK;EACd;EAEA,MAAM;IACJL,GAAG;IACHC,GAAG;IACHC;EACF,CAAC,GAAGG,OAAO;EAEX,IAAI,CAACL,GAAG,IAAIA,GAAG,KAAK,EAAE,IAAI,CAACC,GAAG,IAAI,CAACC,GAAG,EAAE;IACtC,OAAO,KAAK;EACd;EAEA,MAAM6G,QAAQ,GAAG,OAAO/G,GAAG;EAC3B,IAAI+G,QAAQ,IAAI,QAAQ,IAAIA,QAAQ,IAAI,WAAW,IAAI/G,GAAG,KAAK,IAAI,EAAE;IACnE,OAAO,KAAK;EACd;EAEA,IAAI,OAAOC,GAAG,IAAI,WAAW,IAAI,CAACxU,KAAK,CAACC,OAAO,CAACuU,GAAG,CAAC,IAAIA,GAAG,KAAK,IAAI,EAAE;IACpE,OAAO,KAAK;EACd;EAEA,IAAI,OAAOC,GAAG,IAAI,WAAW,IAAI,CAACzU,KAAK,CAACC,OAAO,CAACwU,GAAG,CAAC,IAAIA,GAAG,KAAK,IAAI,EAAE;IACpE,OAAO,KAAK;EACd;EACA,OAAO,IAAI;AACb,CAAC;AAWDlB,MAAM,CAACgI,cAAc,GAAG,UAAS3G,OAAO,EAAE;EACxC,IAAI,CAAC5U,KAAK,CAACC,OAAO,CAAC2U,OAAO,CAACJ,GAAG,CAAC,EAAE;IAC/B,OAAO,KAAK;EACd;EACA,KAAK,IAAIlZ,CAAC,IAAIsZ,OAAO,CAACJ,GAAG,EAAE;IACzB,MAAMA,GAAG,GAAGI,OAAO,CAACJ,GAAG,CAAClZ,CAAC,CAAC;IAC1B,IAAIkZ,GAAG,IAAIA,GAAG,CAAC7U,EAAE,GAAG,CAAC,EAAE;MACrB,MAAM8U,GAAG,GAAGG,OAAO,CAACH,GAAG,CAACD,GAAG,CAAC6B,GAAG,GAAG,CAAC,CAAC;MACpC,OAAO5B,GAAG,IAAIA,GAAG,CAACyB,EAAE,IAAI,IAAI,IAAIzB,GAAG,CAAC5K,IAAI;IAC1C;EACF;EACA,OAAO,KAAK;AACd,CAAC;AAyBD0J,MAAM,CAACiI,WAAW,GAAG,UAAS5G,OAAO,EAAEnU,QAAQ,EAAEG,OAAO,EAAE;EACxD,IAAI,CAACZ,KAAK,CAACC,OAAO,CAAC2U,OAAO,CAACJ,GAAG,CAAC,EAAE;IAC/B;EACF;EACA,IAAIhV,KAAK,GAAG,CAAC;EACb,KAAK,IAAIlE,CAAC,IAAIsZ,OAAO,CAACJ,GAAG,EAAE;IACzB,IAAIA,GAAG,GAAGI,OAAO,CAACJ,GAAG,CAAClZ,CAAC,CAAC;IACxB,IAAIkZ,GAAG,IAAIA,GAAG,CAAC7U,EAAE,GAAG,CAAC,EAAE;MACrB,MAAM8U,GAAG,GAAGG,OAAO,CAACH,GAAG,CAACD,GAAG,CAAC6B,GAAG,GAAG,CAAC,CAAC;MACpC,IAAI5B,GAAG,IAAIA,GAAG,CAACyB,EAAE,IAAI,IAAI,IAAIzB,GAAG,CAAC5K,IAAI,EAAE;QACrC,IAAIpJ,QAAQ,CAACI,IAAI,CAACD,OAAO,EAAE6T,GAAG,CAAC5K,IAAI,EAAErK,KAAK,EAAE,EAAE,IAAI,CAAC,EAAE;UACnD;QACF;MACF;IACF;EACF;EAAC;AACH,CAAC;AAUD+T,MAAM,CAACkI,WAAW,GAAG,UAAS7G,OAAO,EAAE;EACrC,OAAOA,OAAO,CAACH,GAAG,IAAIG,OAAO,CAACH,GAAG,CAAClZ,MAAM,GAAG,CAAC;AAC9C,CAAC;AAYDgY,MAAM,CAAC4B,QAAQ,GAAG,UAASP,OAAO,EAAEnU,QAAQ,EAAEG,OAAO,EAAE;EACrD,IAAIgU,OAAO,CAACH,GAAG,IAAIG,OAAO,CAACH,GAAG,CAAClZ,MAAM,GAAG,CAAC,EAAE;IACzC,KAAK,IAAID,CAAC,IAAIsZ,OAAO,CAACH,GAAG,EAAE;MACzB,IAAIG,OAAO,CAACH,GAAG,CAACnZ,CAAC,CAAC,EAAE;QAClB,IAAImF,QAAQ,CAACI,IAAI,CAACD,OAAO,EAAEgU,OAAO,CAACH,GAAG,CAACnZ,CAAC,CAAC,CAACuO,IAAI,EAAEvO,CAAC,EAAEsZ,OAAO,CAACH,GAAG,CAACnZ,CAAC,CAAC,CAAC4a,EAAE,CAAC,EAAE;UACrE;QACF;MACF;IACF;EACF;AACF,CAAC;AA2BD3C,MAAM,CAACmI,MAAM,GAAG,UAAS9G,OAAO,EAAEnU,QAAQ,EAAEG,OAAO,EAAE;EACnD,IAAIgU,OAAO,CAACJ,GAAG,IAAII,OAAO,CAACJ,GAAG,CAACjZ,MAAM,GAAG,CAAC,EAAE;IACzC,KAAK,IAAID,CAAC,IAAIsZ,OAAO,CAACJ,GAAG,EAAE;MACzB,MAAMA,GAAG,GAAGI,OAAO,CAACJ,GAAG,CAAClZ,CAAC,CAAC;MAC1B,IAAIkZ,GAAG,EAAE;QACP,IAAI/T,QAAQ,CAACI,IAAI,CAACD,OAAO,EAAE4T,GAAG,CAAC0B,EAAE,EAAE1B,GAAG,CAAC7U,EAAE,EAAE6U,GAAG,CAAC4B,GAAG,EAAE5B,GAAG,CAAC6B,GAAG,EAAE/a,CAAC,CAAC,EAAE;UAC/D;QACF;MACF;IACF;EACF;AACF,CAAC;AAUDiY,MAAM,CAACoI,gBAAgB,GAAG,UAAS/G,OAAO,EAAE;EAC1C,IAAIA,OAAO,IAAIA,OAAO,CAACH,GAAG,IAAIG,OAAO,CAACH,GAAG,CAAClZ,MAAM,GAAG,CAAC,EAAE;IACpD,KAAK,IAAID,CAAC,IAAIsZ,OAAO,CAACH,GAAG,EAAE;MACzB,MAAMA,GAAG,GAAGG,OAAO,CAACH,GAAG,CAACnZ,CAAC,CAAC;MAC1B,IAAImZ,GAAG,IAAIA,GAAG,CAAC5K,IAAI,EAAE;QACnB,MAAMA,IAAI,GAAG+R,WAAW,CAACnH,GAAG,CAAC5K,IAAI,CAAC;QAClC,IAAIA,IAAI,EAAE;UACR+K,OAAO,CAACH,GAAG,CAACnZ,CAAC,CAAC,CAACuO,IAAI,GAAGA,IAAI;QAC5B,CAAC,MAAM;UACL,OAAO+K,OAAO,CAACH,GAAG,CAACnZ,CAAC,CAAC,CAACuO,IAAI;QAC5B;MACF;IACF;EACF;EACA,OAAO+K,OAAO;AAChB,CAAC;AAWDrB,MAAM,CAACsI,cAAc,GAAG,UAASC,OAAO,EAAE;EACxC,IAAI7X,GAAG,GAAG,IAAI;EACd,IAAI6X,OAAO,CAACxI,IAAI,IAAItD,cAAc,IAAI8L,OAAO,CAAC1hB,GAAG,EAAE;IACjD6J,GAAG,GAAGiO,iBAAiB,CAAC4J,OAAO,CAAC1hB,GAAG,EAAE0hB,OAAO,CAACxI,IAAI,EAAEC,MAAM,CAAClO,MAAM,CAAC;EACnE,CAAC,MAAM,IAAI,OAAOyW,OAAO,CAACzI,GAAG,IAAI,QAAQ,EAAE;IACzCpP,GAAG,GAAG6X,OAAO,CAACzI,GAAG;EACnB;EACA,OAAOpP,GAAG;AACZ,CAAC;AAUDsP,MAAM,CAACwI,YAAY,GAAG,UAASD,OAAO,EAAE;EACtC,OAAO,CAAC,CAACA,OAAO,CAAC5E,WAAW;AAC9B,CAAC;AAYD3D,MAAM,CAACyI,aAAa,GAAG,UAASF,OAAO,EAAE;EACvC,OAAOA,OAAO,CAAC1hB,GAAG,GAAG8X,iBAAiB,CAAC4J,OAAO,CAAC1hB,GAAG,EAAE0hB,OAAO,CAACxI,IAAI,EAAEC,MAAM,CAAClO,MAAM,CAAC,GAAG,IAAI;AACzF,CAAC;AAUDkO,MAAM,CAAC0I,aAAa,GAAG,UAASH,OAAO,EAAE;EAGvC,OAAOA,OAAO,CAACrI,IAAI,GAAGqI,OAAO,CAACrI,IAAI,GAAGqI,OAAO,CAAC1hB,GAAG,GAAI0hB,OAAO,CAAC1hB,GAAG,CAACmB,MAAM,GAAG,IAAI,GAAI,CAAC,GAAG,CAAC;AACxF,CAAC;AAUDgY,MAAM,CAAC2I,iBAAiB,GAAG,UAASJ,OAAO,EAAE;EAC3C,OAAOA,OAAO,CAACxI,IAAI,IAAI,YAAY;AACrC,CAAC;AAWDC,MAAM,CAAC4I,OAAO,GAAG,UAASxD,KAAK,EAAE;EAC/B,OAAOhI,WAAW,CAACgI,KAAK,CAAC,IAAIhI,WAAW,CAACgI,KAAK,CAAC,CAAC9H,QAAQ;AAC1D,CAAC;AAcD0C,MAAM,CAAC6I,SAAS,GAAG,UAASzD,KAAK,EAAE9O,IAAI,EAAE;EACvC,IAAIA,IAAI,IAAImJ,UAAU,CAAC2F,KAAK,CAAC,IAAI3F,UAAU,CAAC2F,KAAK,CAAC,CAAC1F,KAAK,EAAE;IACxD,OAAOD,UAAU,CAAC2F,KAAK,CAAC,CAAC1F,KAAK,CAACpJ,IAAI,CAAC;EACtC;EAEA,OAAOxL,SAAS;AAClB,CAAC;AASDkV,MAAM,CAAC8I,cAAc,GAAG,YAAW;EACjC,OAAOpM,gBAAgB;AACzB,CAAC;AAYD,SAAS0F,QAAQA,CAACV,IAAI,EAAEhW,KAAK,EAAEC,GAAG,EAAEgW,KAAK,EAAE;EACzC,MAAMQ,MAAM,GAAG,EAAE;EAEjB,IAAIR,KAAK,CAAC3Z,MAAM,IAAI,CAAC,EAAE;IACrB,OAAO,EAAE;EACX;EAEA,KAAK,IAAID,CAAC,IAAI4Z,KAAK,EAAE;IAEnB,MAAMoH,IAAI,GAAGpH,KAAK,CAAC5Z,CAAC,CAAC;IAGrB,IAAIghB,IAAI,CAAC3c,EAAE,GAAGV,KAAK,EAAE;MACnByW,MAAM,CAACjH,IAAI,CAAC;QACV8F,GAAG,EAAEU,IAAI,CAACvE,KAAK,CAACzR,KAAK,EAAEqd,IAAI,CAAC3c,EAAE;MAChC,CAAC,CAAC;IACJ;IAGA,MAAM4c,KAAK,GAAG;MACZrG,EAAE,EAAEoG,IAAI,CAACpG;IACX,CAAC;IACD,MAAMsG,IAAI,GAAG7G,QAAQ,CAACV,IAAI,EAAEqH,IAAI,CAAC3c,EAAE,GAAG,CAAC,EAAE2c,IAAI,CAACpd,GAAG,EAAEod,IAAI,CAACzB,QAAQ,CAAC;IACjE,IAAI2B,IAAI,CAACjhB,MAAM,GAAG,CAAC,EAAE;MACnBghB,KAAK,CAAC1B,QAAQ,GAAG2B,IAAI;IACvB,CAAC,MAAM;MACLD,KAAK,CAAChI,GAAG,GAAG+H,IAAI,CAAC/H,GAAG;IACtB;IACAmB,MAAM,CAACjH,IAAI,CAAC8N,KAAK,CAAC;IAClBtd,KAAK,GAAGqd,IAAI,CAACpd,GAAG,GAAG,CAAC;EACtB;EAGA,IAAID,KAAK,GAAGC,GAAG,EAAE;IACfwW,MAAM,CAACjH,IAAI,CAAC;MACV8F,GAAG,EAAEU,IAAI,CAACvE,KAAK,CAACzR,KAAK,EAAEC,GAAG;IAC5B,CAAC,CAAC;EACJ;EAEA,OAAOwW,MAAM;AACf;AAIA,SAASJ,QAAQA,CAACuE,QAAQ,EAAE4C,QAAQ,EAAEC,MAAM,EAAE5J,IAAI,EAAE;EAClD,MAAM9H,MAAM,GAAG,EAAE;EACjB,IAAIiL,KAAK,GAAG,CAAC;EACb,IAAIhB,IAAI,GAAG4E,QAAQ,CAACnJ,KAAK,CAAC,CAAC,CAAC;EAE5B,OAAOuE,IAAI,CAAC1Z,MAAM,GAAG,CAAC,EAAE;IAMtB,MAAM0D,KAAK,GAAGwd,QAAQ,CAACE,IAAI,CAAC1H,IAAI,CAAC;IACjC,IAAIhW,KAAK,IAAI,IAAI,EAAE;MACjB;IACF;IAIA,IAAI2d,YAAY,GAAG3d,KAAK,CAAC,OAAO,CAAC,GAAGA,KAAK,CAAC,CAAC,CAAC,CAAC4d,WAAW,CAAC5d,KAAK,CAAC,CAAC,CAAC,CAAC;IAElEgW,IAAI,GAAGA,IAAI,CAACvE,KAAK,CAACkM,YAAY,GAAG,CAAC,CAAC;IAEnCA,YAAY,IAAI3G,KAAK;IAErBA,KAAK,GAAG2G,YAAY,GAAG,CAAC;IAGxB,MAAM1d,GAAG,GAAGwd,MAAM,GAAGA,MAAM,CAACC,IAAI,CAAC1H,IAAI,CAAC,GAAG,IAAI;IAC7C,IAAI/V,GAAG,IAAI,IAAI,EAAE;MACf;IACF;IACA,IAAI4d,UAAU,GAAG5d,GAAG,CAAC,OAAO,CAAC,GAAGA,GAAG,CAAC,CAAC,CAAC,CAAC+Z,OAAO,CAAC/Z,GAAG,CAAC,CAAC,CAAC,CAAC;IAEtD+V,IAAI,GAAGA,IAAI,CAACvE,KAAK,CAACoM,UAAU,GAAG,CAAC,CAAC;IAEjCA,UAAU,IAAI7G,KAAK;IAEnBA,KAAK,GAAG6G,UAAU,GAAG,CAAC;IAEtB9R,MAAM,CAACyD,IAAI,CAAC;MACV8F,GAAG,EAAEsF,QAAQ,CAACnJ,KAAK,CAACkM,YAAY,GAAG,CAAC,EAAEE,UAAU,CAAC;MACjDjC,QAAQ,EAAE,EAAE;MACZlb,EAAE,EAAEid,YAAY;MAChB1d,GAAG,EAAE4d,UAAU;MACf5G,EAAE,EAAEpD;IACN,CAAC,CAAC;EACJ;EAEA,OAAO9H,MAAM;AACf;AAIA,SAASyK,UAAUA,CAACP,KAAK,EAAE;EACzB,IAAIA,KAAK,CAAC3Z,MAAM,IAAI,CAAC,EAAE;IACrB,OAAO,EAAE;EACX;EAEA,MAAM+d,IAAI,GAAG,CAACpE,KAAK,CAAC,CAAC,CAAC,CAAC;EACvB,IAAI6H,IAAI,GAAG7H,KAAK,CAAC,CAAC,CAAC;EACnB,KAAK,IAAI5Z,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAG4Z,KAAK,CAAC3Z,MAAM,EAAED,CAAC,EAAE,EAAE;IAGrC,IAAI4Z,KAAK,CAAC5Z,CAAC,CAAC,CAACqE,EAAE,GAAGod,IAAI,CAAC7d,GAAG,EAAE;MAE1Boa,IAAI,CAAC7K,IAAI,CAACyG,KAAK,CAAC5Z,CAAC,CAAC,CAAC;MACnByhB,IAAI,GAAG7H,KAAK,CAAC5Z,CAAC,CAAC;IACjB,CAAC,MAAM,IAAI4Z,KAAK,CAAC5Z,CAAC,CAAC,CAAC4D,GAAG,IAAI6d,IAAI,CAAC7d,GAAG,EAAE;MAEnC6d,IAAI,CAAClC,QAAQ,CAACpM,IAAI,CAACyG,KAAK,CAAC5Z,CAAC,CAAC,CAAC;IAC9B;EAEF;EAGA,KAAK,IAAIA,CAAC,IAAIge,IAAI,EAAE;IAClBA,IAAI,CAAChe,CAAC,CAAC,CAACuf,QAAQ,GAAGpF,UAAU,CAAC6D,IAAI,CAAChe,CAAC,CAAC,CAACuf,QAAQ,CAAC;EACjD;EAEA,OAAOvB,IAAI;AACb;AAGA,SAASC,YAAYA,CAACF,GAAG,EAAE;EACzB,IAAI,CAACA,GAAG,EAAE;IACR,OAAO,IAAI;EACb;EAEAA,GAAG,GAAI,OAAOA,GAAG,IAAI,QAAQ,GAAI;IAC/B9E,GAAG,EAAE8E;EACP,CAAC,GAAGA,GAAG;EACP,IAAI;IACF9E,GAAG;IACHC,GAAG;IACHC;EACF,CAAC,GAAG4E,GAAG;EAEP9E,GAAG,GAAGA,GAAG,IAAI,EAAE;EACf,IAAI,CAACvU,KAAK,CAACC,OAAO,CAACwU,GAAG,CAAC,EAAE;IACvBA,GAAG,GAAG,EAAE;EACV;EAEA,IAAI,CAACzU,KAAK,CAACC,OAAO,CAACuU,GAAG,CAAC,IAAIA,GAAG,CAACjZ,MAAM,IAAI,CAAC,EAAE;IAC1C,IAAIkZ,GAAG,CAAClZ,MAAM,IAAI,CAAC,EAAE;MACnB,OAAO;QACL2N,IAAI,EAAEqL;MACR,CAAC;IACH;IAGAC,GAAG,GAAG,CAAC;MACL7U,EAAE,EAAE,CAAC;MACLyW,GAAG,EAAE,CAAC;MACNC,GAAG,EAAE;IACP,CAAC,CAAC;EACJ;EAGA,MAAMnB,KAAK,GAAG,EAAE;EAChB,MAAMsG,WAAW,GAAG,EAAE;EACtBhH,GAAG,CAAChU,OAAO,CAAE8b,IAAI,IAAK;IACpB,IAAI,CAACA,IAAI,IAAI,OAAOA,IAAI,IAAI,QAAQ,EAAE;MACpC;IACF;IAEA,IAAI,CAAC,CAAC,WAAW,EAAE,QAAQ,CAAC,CAAC/hB,QAAQ,CAAC,OAAO+hB,IAAI,CAAC3c,EAAE,CAAC,EAAE;MAErD;IACF;IACA,IAAI,CAAC,CAAC,WAAW,EAAE,QAAQ,CAAC,CAACpF,QAAQ,CAAC,OAAO+hB,IAAI,CAAClG,GAAG,CAAC,EAAE;MAEtD;IACF;IACA,IAAIzW,EAAE,GAAG2c,IAAI,CAAC3c,EAAE,GAAG,CAAC;IACpB,IAAIyW,GAAG,GAAGkG,IAAI,CAAClG,GAAG,GAAG,CAAC;IACtB,IAAIA,GAAG,GAAG,CAAC,EAAE;MAEX;IACF;IAEA,IAAIC,GAAG,GAAGiG,IAAI,CAACjG,GAAG,IAAI,CAAC;IACvB,IAAI5B,GAAG,CAAClZ,MAAM,GAAG,CAAC,KAAK,OAAO8a,GAAG,IAAI,QAAQ,IAAIA,GAAG,GAAG,CAAC,IAAIA,GAAG,IAAI5B,GAAG,CAAClZ,MAAM,CAAC,EAAE;MAE9E;IACF;IAEA,IAAIoE,EAAE,IAAI,CAAC,CAAC,EAAE;MAEZ6b,WAAW,CAAC/M,IAAI,CAAC;QACfxP,KAAK,EAAE,CAAC,CAAC;QACTC,GAAG,EAAE,CAAC;QACNmX,GAAG,EAAEA;MACP,CAAC,CAAC;MACF;IACF,CAAC,MAAM,IAAI1W,EAAE,GAAGyW,GAAG,GAAG7B,GAAG,CAAChZ,MAAM,EAAE;MAEhC;IACF;IAEA,IAAI,CAAC+gB,IAAI,CAACpG,EAAE,EAAE;MACZ,IAAIzB,GAAG,CAAClZ,MAAM,GAAG,CAAC,IAAK,OAAOkZ,GAAG,CAAC4B,GAAG,CAAC,IAAI,QAAS,EAAE;QACnDnB,KAAK,CAACzG,IAAI,CAAC;UACTxP,KAAK,EAAEU,EAAE;UACTT,GAAG,EAAES,EAAE,GAAGyW,GAAG;UACbC,GAAG,EAAEA;QACP,CAAC,CAAC;MACJ;IACF,CAAC,MAAM;MACLnB,KAAK,CAACzG,IAAI,CAAC;QACTqE,IAAI,EAAEwJ,IAAI,CAACpG,EAAE;QACbjX,KAAK,EAAEU,EAAE;QACTT,GAAG,EAAES,EAAE,GAAGyW;MACZ,CAAC,CAAC;IACJ;EACF,CAAC,CAAC;EAGFlB,KAAK,CAACM,IAAI,CAAC,CAAC9W,CAAC,EAAEC,CAAC,KAAK;IACnB,IAAIvC,IAAI,GAAGsC,CAAC,CAACO,KAAK,GAAGN,CAAC,CAACM,KAAK;IAC5B,IAAI7C,IAAI,IAAI,CAAC,EAAE;MACb,OAAOA,IAAI;IACb;IACAA,IAAI,GAAGuC,CAAC,CAACO,GAAG,GAAGR,CAAC,CAACQ,GAAG;IACpB,IAAI9C,IAAI,IAAI,CAAC,EAAE;MACb,OAAOA,IAAI;IACb;IACA,OAAOgU,UAAU,CAAC6I,OAAO,CAACta,CAAC,CAACmU,IAAI,CAAC,GAAG1C,UAAU,CAAC6I,OAAO,CAACva,CAAC,CAACoU,IAAI,CAAC;EAChE,CAAC,CAAC;EAGF,IAAI0I,WAAW,CAACjgB,MAAM,GAAG,CAAC,EAAE;IAC1B2Z,KAAK,CAACzG,IAAI,CAAC,GAAG+M,WAAW,CAAC;EAC5B;EAEAtG,KAAK,CAAC1U,OAAO,CAAE8b,IAAI,IAAK;IACtB,IAAI7H,GAAG,CAAClZ,MAAM,GAAG,CAAC,IAAI,CAAC+gB,IAAI,CAACxJ,IAAI,IAAI2B,GAAG,CAAC6H,IAAI,CAACjG,GAAG,CAAC,IAAI,OAAO5B,GAAG,CAAC6H,IAAI,CAACjG,GAAG,CAAC,IAAI,QAAQ,EAAE;MACrFiG,IAAI,CAACxJ,IAAI,GAAG2B,GAAG,CAAC6H,IAAI,CAACjG,GAAG,CAAC,CAACH,EAAE;MAC5BoG,IAAI,CAACzS,IAAI,GAAG4K,GAAG,CAAC6H,IAAI,CAACjG,GAAG,CAAC,CAACxM,IAAI;IAChC;IAGA,IAAI,CAACyS,IAAI,CAACxJ,IAAI,EAAE;MACdwJ,IAAI,CAACxJ,IAAI,GAAG,IAAI;IAClB;EACF,CAAC,CAAC;EAEF,IAAIwG,IAAI,GAAG0D,WAAW,CAAC,CAAC,CAAC,EAAEzI,GAAG,EAAE,CAAC,EAAEA,GAAG,CAAChZ,MAAM,EAAE2Z,KAAK,CAAC;EAGrD,MAAM+H,OAAO,GAAG,SAAAA,CAAS3C,IAAI,EAAE;IAC7B,IAAIta,KAAK,CAACC,OAAO,CAACqa,IAAI,CAACO,QAAQ,CAAC,IAAIP,IAAI,CAACO,QAAQ,CAACtf,MAAM,IAAI,CAAC,EAAE;MAE7D,MAAM2hB,KAAK,GAAG5C,IAAI,CAACO,QAAQ,CAAC,CAAC,CAAC;MAC9B,IAAI,CAACP,IAAI,CAACxH,IAAI,EAAE;QACd,MAAMyH,MAAM,GAAGD,IAAI,CAACC,MAAM;QAC1BD,IAAI,GAAG4C,KAAK;QACZ5C,IAAI,CAACC,MAAM,GAAGA,MAAM;MACtB,CAAC,MAAM,IAAI,CAAC2C,KAAK,CAACpK,IAAI,IAAI,CAACoK,KAAK,CAACrC,QAAQ,EAAE;QACzCP,IAAI,CAACpR,IAAI,GAAGgU,KAAK,CAAChU,IAAI;QACtB,OAAOoR,IAAI,CAACO,QAAQ;MACtB;IACF;IACA,OAAOP,IAAI;EACb,CAAC;EACDhB,IAAI,GAAGkB,WAAW,CAAClB,IAAI,EAAE2D,OAAO,CAAC;EAEjC,OAAO3D,IAAI;AACb;AAGA,SAAS6D,OAAOA,CAAC5C,MAAM,EAAE6C,CAAC,EAAE;EAC1B,IAAI,CAACA,CAAC,EAAE;IACN,OAAO7C,MAAM;EACf;EAEA,IAAI,CAACA,MAAM,CAACM,QAAQ,EAAE;IACpBN,MAAM,CAACM,QAAQ,GAAG,EAAE;EACtB;EAGA,IAAIN,MAAM,CAACrR,IAAI,EAAE;IACfqR,MAAM,CAACM,QAAQ,CAACpM,IAAI,CAAC;MACnBvF,IAAI,EAAEqR,MAAM,CAACrR,IAAI;MACjBqR,MAAM,EAAEA;IACV,CAAC,CAAC;IACF,OAAOA,MAAM,CAACrR,IAAI;EACpB;EAEAkU,CAAC,CAAC7C,MAAM,GAAGA,MAAM;EACjBA,MAAM,CAACM,QAAQ,CAACpM,IAAI,CAAC2O,CAAC,CAAC;EAEvB,OAAO7C,MAAM;AACf;AAGA,SAASyC,WAAWA,CAACzC,MAAM,EAAErR,IAAI,EAAEjK,KAAK,EAAEC,GAAG,EAAEgW,KAAK,EAAE;EACpD,IAAI,CAACA,KAAK,IAAIA,KAAK,CAAC3Z,MAAM,IAAI,CAAC,EAAE;IAC/B,IAAI0D,KAAK,GAAGC,GAAG,EAAE;MACfie,OAAO,CAAC5C,MAAM,EAAE;QACdrR,IAAI,EAAEA,IAAI,CAACmU,SAAS,CAACpe,KAAK,EAAEC,GAAG;MACjC,CAAC,CAAC;IACJ;IACA,OAAOqb,MAAM;EACf;EAGA,KAAK,IAAIjf,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAG4Z,KAAK,CAAC3Z,MAAM,EAAED,CAAC,EAAE,EAAE;IACrC,MAAMghB,IAAI,GAAGpH,KAAK,CAAC5Z,CAAC,CAAC;IACrB,IAAIghB,IAAI,CAACrd,KAAK,GAAG,CAAC,IAAIqd,IAAI,CAACxJ,IAAI,IAAI,IAAI,EAAE;MACvCqK,OAAO,CAAC5C,MAAM,EAAE;QACdzH,IAAI,EAAEwJ,IAAI,CAACxJ,IAAI;QACfjJ,IAAI,EAAEyS,IAAI,CAACzS,IAAI;QACfwM,GAAG,EAAEiG,IAAI,CAACjG,GAAG;QACbiH,GAAG,EAAE;MACP,CAAC,CAAC;MACF;IACF;IAGA,IAAIre,KAAK,GAAGqd,IAAI,CAACrd,KAAK,EAAE;MACtBke,OAAO,CAAC5C,MAAM,EAAE;QACdrR,IAAI,EAAEA,IAAI,CAACmU,SAAS,CAACpe,KAAK,EAAEqd,IAAI,CAACrd,KAAK;MACxC,CAAC,CAAC;MACFA,KAAK,GAAGqd,IAAI,CAACrd,KAAK;IACpB;IAGA,MAAMse,QAAQ,GAAG,EAAE;IACnB,OAAOjiB,CAAC,GAAG4Z,KAAK,CAAC3Z,MAAM,GAAG,CAAC,EAAE;MAC3B,MAAMiiB,KAAK,GAAGtI,KAAK,CAAC5Z,CAAC,GAAG,CAAC,CAAC;MAC1B,IAAIkiB,KAAK,CAACve,KAAK,GAAG,CAAC,EAAE;QAEnB;MACF,CAAC,MAAM,IAAIue,KAAK,CAACve,KAAK,GAAGqd,IAAI,CAACpd,GAAG,EAAE;QACjC,IAAIse,KAAK,CAACte,GAAG,IAAIod,IAAI,CAACpd,GAAG,EAAE;UACzB,MAAMkW,GAAG,GAAGzE,WAAW,CAAC6M,KAAK,CAACtH,EAAE,CAAC,IAAI,CAAC,CAAC;UACvC,IAAIsH,KAAK,CAACve,KAAK,GAAGue,KAAK,CAACte,GAAG,IAAIkW,GAAG,CAACrE,MAAM,EAAE;YAGzCwM,QAAQ,CAAC9O,IAAI,CAAC+O,KAAK,CAAC;UACtB;QACF;QACAliB,CAAC,EAAE;MAEL,CAAC,MAAM;QAEL;MACF;IACF;IAEA6hB,OAAO,CAAC5C,MAAM,EAAEyC,WAAW,CAAC;MAC1BlK,IAAI,EAAEwJ,IAAI,CAACxJ,IAAI;MACfjJ,IAAI,EAAEyS,IAAI,CAACzS,IAAI;MACfwM,GAAG,EAAEiG,IAAI,CAACjG;IACZ,CAAC,EAAEnN,IAAI,EAAEjK,KAAK,EAAEqd,IAAI,CAACpd,GAAG,EAAEqe,QAAQ,CAAC,CAAC;IACpCte,KAAK,GAAGqd,IAAI,CAACpd,GAAG;EAClB;EAGA,IAAID,KAAK,GAAGC,GAAG,EAAE;IACfie,OAAO,CAAC5C,MAAM,EAAE;MACdrR,IAAI,EAAEA,IAAI,CAACmU,SAAS,CAACpe,KAAK,EAAEC,GAAG;IACjC,CAAC,CAAC;EACJ;EAEA,OAAOqb,MAAM;AACf;AAGA,SAASJ,YAAYA,CAACd,GAAG,EAAEC,IAAI,EAAEmE,MAAM,EAAE;EACvC,IAAI,CAACnE,IAAI,EAAE;IACT,OAAOD,GAAG;EACZ;EAEAA,GAAG,CAAC9E,GAAG,GAAG8E,GAAG,CAAC9E,GAAG,IAAI,EAAE;EAGvB,MAAMtV,KAAK,GAAGoa,GAAG,CAAC9E,GAAG,CAAChZ,MAAM;EAE5B,IAAI+d,IAAI,CAACpQ,IAAI,EAAE;IACbmQ,GAAG,CAAC9E,GAAG,IAAI+E,IAAI,CAACpQ,IAAI;EACtB,CAAC,MAAM,IAAIlJ,KAAK,CAACC,OAAO,CAACqZ,IAAI,CAACuB,QAAQ,CAAC,EAAE;IACvCvB,IAAI,CAACuB,QAAQ,CAACra,OAAO,CAAEkd,CAAC,IAAK;MAC3BvD,YAAY,CAACd,GAAG,EAAEqE,CAAC,EAAED,MAAM,CAAC;IAC9B,CAAC,CAAC;EACJ;EAEA,IAAInE,IAAI,CAACxG,IAAI,EAAE;IACb,MAAMsD,GAAG,GAAGiD,GAAG,CAAC9E,GAAG,CAAChZ,MAAM,GAAG0D,KAAK;IAClCoa,GAAG,CAAC7E,GAAG,GAAG6E,GAAG,CAAC7E,GAAG,IAAI,EAAE;IACvB,IAAIqD,MAAM,CAAC8F,IAAI,CAACrE,IAAI,CAACzP,IAAI,IAAI,CAAC,CAAC,CAAC,CAACtO,MAAM,GAAG,CAAC,EAAE;MAC3C8d,GAAG,CAAC5E,GAAG,GAAG4E,GAAG,CAAC5E,GAAG,IAAI,EAAE;MACvB,MAAMmJ,MAAM,GAAI,OAAOH,MAAM,CAACnE,IAAI,CAACjD,GAAG,CAAC,IAAI,WAAW,GAAIgD,GAAG,CAAC5E,GAAG,CAAClZ,MAAM,GAAGkiB,MAAM,CAACnE,IAAI,CAACjD,GAAG,CAAC;MAC3FoH,MAAM,CAACnE,IAAI,CAACjD,GAAG,CAAC,GAAGuH,MAAM;MACzBvE,GAAG,CAAC5E,GAAG,CAACmJ,MAAM,CAAC,GAAG;QAChB1H,EAAE,EAAEoD,IAAI,CAACxG,IAAI;QACbjJ,IAAI,EAAEyP,IAAI,CAACzP;MACb,CAAC;MACD,IAAIyP,IAAI,CAACgE,GAAG,EAAE;QAEZjE,GAAG,CAAC7E,GAAG,CAAC/F,IAAI,CAAC;UACX9O,EAAE,EAAE,CAAC,CAAC;UACNyW,GAAG,EAAE,CAAC;UACNC,GAAG,EAAEuH;QACP,CAAC,CAAC;MACJ,CAAC,MAAM;QACLvE,GAAG,CAAC7E,GAAG,CAAC/F,IAAI,CAAC;UACX9O,EAAE,EAAEV,KAAK;UACTmX,GAAG,EAAEA,GAAG;UACRC,GAAG,EAAEuH;QACP,CAAC,CAAC;MACJ;IACF,CAAC,MAAM;MACLvE,GAAG,CAAC7E,GAAG,CAAC/F,IAAI,CAAC;QACXyH,EAAE,EAAEoD,IAAI,CAACxG,IAAI;QACbnT,EAAE,EAAEV,KAAK;QACTmX,GAAG,EAAEA;MACP,CAAC,CAAC;IACJ;EACF;EACA,OAAOiD,GAAG;AACZ;AAGA,SAASmB,WAAWA,CAAC7N,GAAG,EAAEkR,WAAW,EAAEjd,OAAO,EAAE;EAC9C,IAAI,CAAC+L,GAAG,EAAE;IACR,OAAO,IAAI;EACb;EAEA,IAAI2C,GAAG,GAAGuO,WAAW,CAAChd,IAAI,CAACD,OAAO,EAAE+L,GAAG,CAAC;EACxC,IAAI,CAAC2C,GAAG,IAAI,CAACA,GAAG,CAACuL,QAAQ,EAAE;IACzB,OAAOvL,GAAG;EACZ;EAEA,MAAMuL,QAAQ,GAAG,EAAE;EACnB,KAAK,IAAIvf,CAAC,IAAIgU,GAAG,CAACuL,QAAQ,EAAE;IAC1B,IAAIuC,CAAC,GAAG9N,GAAG,CAACuL,QAAQ,CAACvf,CAAC,CAAC;IACvB,IAAI8hB,CAAC,EAAE;MACLA,CAAC,GAAG5C,WAAW,CAAC4C,CAAC,EAAES,WAAW,EAAEjd,OAAO,CAAC;MACxC,IAAIwc,CAAC,EAAE;QACLvC,QAAQ,CAACpM,IAAI,CAAC2O,CAAC,CAAC;MAClB;IACF;EACF;EAEA,IAAIvC,QAAQ,CAACtf,MAAM,IAAI,CAAC,EAAE;IACxB+T,GAAG,CAACuL,QAAQ,GAAG,IAAI;EACrB,CAAC,MAAM;IACLvL,GAAG,CAACuL,QAAQ,GAAGA,QAAQ;EACzB;EAEA,OAAOvL,GAAG;AACZ;AAIA,SAASqK,YAAYA,CAAChN,GAAG,EAAEmN,SAAS,EAAE7D,KAAK,EAAE6H,KAAK,EAAEld,OAAO,EAAE;EAC3D,IAAI,CAAC+L,GAAG,EAAE;IACR,OAAO,IAAI;EACb;EAEA,IAAImR,KAAK,IAAInR,GAAG,CAACmG,IAAI,EAAE;IACrBgL,KAAK,CAACrP,IAAI,CAAC9B,GAAG,CAACmG,IAAI,CAAC;EACtB;EAEA,IAAI2G,MAAM,GAAG,EAAE;EACf,KAAK,IAAIne,CAAC,IAAIqR,GAAG,CAACkO,QAAQ,EAAE;IAC1B,MAAMuC,CAAC,GAAGzD,YAAY,CAAChN,GAAG,CAACkO,QAAQ,CAACvf,CAAC,CAAC,EAAEwe,SAAS,EAAExe,CAAC,EAAEwiB,KAAK,EAAEld,OAAO,CAAC;IACrE,IAAIwc,CAAC,EAAE;MACL3D,MAAM,CAAChL,IAAI,CAAC2O,CAAC,CAAC;IAChB;EACF;EACA,IAAI3D,MAAM,CAACle,MAAM,IAAI,CAAC,EAAE;IACtB,IAAIoR,GAAG,CAACzD,IAAI,EAAE;MACZuQ,MAAM,GAAG,CAAC9M,GAAG,CAACzD,IAAI,CAAC;IACrB,CAAC,MAAM;MACLuQ,MAAM,GAAG,IAAI;IACf;EACF;EAEA,IAAIqE,KAAK,IAAInR,GAAG,CAACmG,IAAI,EAAE;IACrBgL,KAAK,CAACC,GAAG,CAAC,CAAC;EACb;EAEA,OAAOjE,SAAS,CAACjZ,IAAI,CAACD,OAAO,EAAE+L,GAAG,CAACmG,IAAI,EAAEnG,GAAG,CAAC9C,IAAI,EAAE4P,MAAM,EAAExD,KAAK,EAAE6H,KAAK,CAAC;AAC1E;AAGA,SAAS7D,WAAWA,CAACX,IAAI,EAAEjL,KAAK,EAAE2P,IAAI,EAAE;EACtC,IAAI,CAAC1E,IAAI,EAAE;IACT,OAAO,IAAI;EACb;EAEA,IAAI0E,IAAI,EAAE;IACR3P,KAAK,IAAI2P,IAAI,CAACziB,MAAM;EACtB;EAEA,MAAM0iB,SAAS,GAAG,SAAAA,CAAS3D,IAAI,EAAE;IAC/B,IAAIjM,KAAK,IAAI,CAAC,CAAC,EAAE;MAEf,OAAO,IAAI;IACb;IAEA,IAAIiM,IAAI,CAACgD,GAAG,EAAE;MAEZ,OAAOhD,IAAI;IACb;IACA,IAAIjM,KAAK,IAAI,CAAC,EAAE;MACdiM,IAAI,CAACpR,IAAI,GAAG8U,IAAI;MAChB3P,KAAK,GAAG,CAAC,CAAC;IACZ,CAAC,MAAM,IAAIiM,IAAI,CAACpR,IAAI,EAAE;MACpB,MAAMkN,GAAG,GAAGkE,IAAI,CAACpR,IAAI,CAAC3N,MAAM;MAC5B,IAAI6a,GAAG,GAAG/H,KAAK,EAAE;QACfiM,IAAI,CAACpR,IAAI,GAAGoR,IAAI,CAACpR,IAAI,CAACmU,SAAS,CAAC,CAAC,EAAEhP,KAAK,CAAC,GAAG2P,IAAI;QAChD3P,KAAK,GAAG,CAAC,CAAC;MACZ,CAAC,MAAM;QACLA,KAAK,IAAI+H,GAAG;MACd;IACF;IACA,OAAOkE,IAAI;EACb,CAAC;EAED,OAAOE,WAAW,CAAClB,IAAI,EAAE2E,SAAS,CAAC;AACrC;AAGA,SAAS/D,WAAWA,CAACZ,IAAI,EAAE4E,KAAK,EAAE;EAChC,MAAMC,SAAS,GAAG7D,IAAI,IAAI;IACxB,MAAMzQ,IAAI,GAAG+R,WAAW,CAACtB,IAAI,CAACzQ,IAAI,EAAE,IAAI,EAAEqU,KAAK,GAAGA,KAAK,CAAC5D,IAAI,CAAC,GAAG,IAAI,CAAC;IACrE,IAAIzQ,IAAI,EAAE;MACRyQ,IAAI,CAACzQ,IAAI,GAAGA,IAAI;IAClB,CAAC,MAAM;MACL,OAAOyQ,IAAI,CAACzQ,IAAI;IAClB;IACA,OAAOyQ,IAAI;EACb,CAAC;EACD,OAAOE,WAAW,CAAClB,IAAI,EAAE6E,SAAS,CAAC;AACrC;AAGA,SAAS1D,KAAKA,CAACnB,IAAI,EAAE;EACnB,IAAIA,IAAI,CAACxG,IAAI,IAAI,IAAI,EAAE;IACrBwG,IAAI,GAAG,IAAI;EACb,CAAC,MAAM,IAAIA,IAAI,CAACpQ,IAAI,EAAE;IACpB,IAAI,CAACoQ,IAAI,CAACxG,IAAI,EAAE;MACdwG,IAAI,CAACpQ,IAAI,GAAGoQ,IAAI,CAACpQ,IAAI,CAACkV,SAAS,CAAC,CAAC;MACjC,IAAI,CAAC9E,IAAI,CAACpQ,IAAI,EAAE;QACdoQ,IAAI,GAAG,IAAI;MACb;IACF;EACF,CAAC,MAAM,IAAI,CAACA,IAAI,CAACxG,IAAI,IAAIwG,IAAI,CAACuB,QAAQ,IAAIvB,IAAI,CAACuB,QAAQ,CAACtf,MAAM,GAAG,CAAC,EAAE;IAClE,MAAMmiB,CAAC,GAAGjD,KAAK,CAACnB,IAAI,CAACuB,QAAQ,CAAC,CAAC,CAAC,CAAC;IACjC,IAAI6C,CAAC,EAAE;MACLpE,IAAI,CAACuB,QAAQ,CAAC,CAAC,CAAC,GAAG6C,CAAC;IACtB,CAAC,MAAM;MACLpE,IAAI,CAACuB,QAAQ,CAACwD,KAAK,CAAC,CAAC;MACrB,IAAI,CAAC/E,IAAI,CAACxG,IAAI,IAAIwG,IAAI,CAACuB,QAAQ,CAACtf,MAAM,IAAI,CAAC,EAAE;QAC3C+d,IAAI,GAAG,IAAI;MACb;IACF;EACF;EACA,OAAOA,IAAI;AACb;AAGA,SAASwB,gBAAgBA,CAACxB,IAAI,EAAEjL,KAAK,EAAE;EACrC,IAAI,CAACiL,IAAI,EAAE;IACT,OAAO,IAAI;EACb;EAEA,IAAIA,IAAI,CAACgE,GAAG,EAAE;IACZhE,IAAI,CAACpQ,IAAI,GAAG,GAAG;IACf,OAAOoQ,IAAI,CAACgE,GAAG;IACf,OAAOhE,IAAI,CAACuB,QAAQ;EACtB,CAAC,MAAM,IAAIvB,IAAI,CAACuB,QAAQ,EAAE;IACxB,MAAMW,WAAW,GAAG,EAAE;IACtB,MAAMX,QAAQ,GAAG,EAAE;IACnB,KAAK,IAAIvf,CAAC,IAAIge,IAAI,CAACuB,QAAQ,EAAE;MAC3B,MAAM6C,CAAC,GAAGpE,IAAI,CAACuB,QAAQ,CAACvf,CAAC,CAAC;MAC1B,IAAIoiB,CAAC,CAACJ,GAAG,EAAE;QACT,IAAI9B,WAAW,CAACjgB,MAAM,IAAI8S,KAAK,EAAE;UAE/B;QACF;QACA,IAAIqP,CAAC,CAAC7T,IAAI,CAAC,MAAM,CAAC,IAAImG,cAAc,EAAE;UAEpC;QACF;QAEA,OAAO0N,CAAC,CAACJ,GAAG;QACZ,OAAOI,CAAC,CAAC7C,QAAQ;QACjB6C,CAAC,CAACxU,IAAI,GAAG,GAAG;QACZsS,WAAW,CAAC/M,IAAI,CAACiP,CAAC,CAAC;MACrB,CAAC,MAAM;QACL7C,QAAQ,CAACpM,IAAI,CAACiP,CAAC,CAAC;MAClB;IACF;IACApE,IAAI,CAACuB,QAAQ,GAAGA,QAAQ,CAACxF,MAAM,CAACmG,WAAW,CAAC;EAC9C;EACA,OAAOlC,IAAI;AACb;AAGA,SAASxD,eAAeA,CAACb,IAAI,EAAE;EAC7B,IAAIqJ,KAAK;EACT,IAAIC,SAAS,GAAG,EAAE;EAClBlO,YAAY,CAAC7P,OAAO,CAAEwV,MAAM,IAAK;IAC/B,OAAO,CAACsI,KAAK,GAAGtI,MAAM,CAACvF,EAAE,CAACkM,IAAI,CAAC1H,IAAI,CAAC,MAAM,IAAI,EAAE;MAC9CsJ,SAAS,CAAC9P,IAAI,CAAC;QACb0H,MAAM,EAAEmI,KAAK,CAAC,OAAO,CAAC;QACtBlI,GAAG,EAAEkI,KAAK,CAAC,CAAC,CAAC,CAAC/iB,MAAM;QACpB+C,MAAM,EAAEggB,KAAK,CAAC,CAAC,CAAC;QAChBzU,IAAI,EAAEmM,MAAM,CAACzF,IAAI,CAAC+N,KAAK,CAAC,CAAC,CAAC,CAAC;QAC3BxL,IAAI,EAAEkD,MAAM,CAAC3U;MACf,CAAC,CAAC;IACJ;EACF,CAAC,CAAC;EAEF,IAAIkd,SAAS,CAAChjB,MAAM,IAAI,CAAC,EAAE;IACzB,OAAOgjB,SAAS;EAClB;EAGAA,SAAS,CAAC/I,IAAI,CAAC,CAAC9W,CAAC,EAAEC,CAAC,KAAK;IACvB,OAAOD,CAAC,CAACyX,MAAM,GAAGxX,CAAC,CAACwX,MAAM;EAC5B,CAAC,CAAC;EAEF,IAAI9W,GAAG,GAAG,CAAC,CAAC;EACZkf,SAAS,GAAGA,SAAS,CAACvd,MAAM,CAAEwd,EAAE,IAAK;IACnC,MAAMxT,MAAM,GAAIwT,EAAE,CAACrI,MAAM,GAAG9W,GAAI;IAChCA,GAAG,GAAGmf,EAAE,CAACrI,MAAM,GAAGqI,EAAE,CAACpI,GAAG;IACxB,OAAOpL,MAAM;EACf,CAAC,CAAC;EAEF,OAAOuT,SAAS;AAClB;AAGA,SAAS1I,QAAQA,CAACH,MAAM,EAAE+I,OAAO,EAAE;EACjC,IAAIC,KAAK,GAAG,EAAE;EACd,IAAI3I,MAAM,GAAG,EAAE;EACf,KAAK,IAAIza,CAAC,IAAIoa,MAAM,EAAE;IACpB,MAAM6G,KAAK,GAAG7G,MAAM,CAACpa,CAAC,CAAC;IACvB,IAAI,CAACihB,KAAK,CAAChI,GAAG,EAAE;MACd,MAAMqB,MAAM,GAAGC,QAAQ,CAAC0G,KAAK,CAAC1B,QAAQ,EAAE6D,KAAK,CAACnjB,MAAM,GAAGkjB,OAAO,CAAC;MAC/DlC,KAAK,CAAChI,GAAG,GAAGqB,MAAM,CAACrB,GAAG;MACtBwB,MAAM,GAAGA,MAAM,CAACV,MAAM,CAACO,MAAM,CAACpB,GAAG,CAAC;IACpC;IAEA,IAAI+H,KAAK,CAACrG,EAAE,EAAE;MACZH,MAAM,CAACtH,IAAI,CAAC;QACV9O,EAAE,EAAE+e,KAAK,CAACnjB,MAAM,GAAGkjB,OAAO;QAC1BrI,GAAG,EAAEmG,KAAK,CAAChI,GAAG,CAAChZ,MAAM;QACrB2a,EAAE,EAAEqG,KAAK,CAACrG;MACZ,CAAC,CAAC;IACJ;IAEAwI,KAAK,IAAInC,KAAK,CAAChI,GAAG;EACpB;EACA,OAAO;IACLA,GAAG,EAAEmK,KAAK;IACVlK,GAAG,EAAEuB;EACP,CAAC;AACH;AAIA,SAAS6F,WAAWA,CAAC/R,IAAI,EAAEmQ,KAAK,EAAEkE,KAAK,EAAE;EACvC,IAAIrU,IAAI,IAAIgO,MAAM,CAAC8G,OAAO,CAAC9U,IAAI,CAAC,CAACtO,MAAM,GAAG,CAAC,EAAE;IAC3C2iB,KAAK,GAAGA,KAAK,IAAI,EAAE;IACnB,MAAMU,EAAE,GAAG,CAAC,CAAC;IACb1O,kBAAkB,CAAC1P,OAAO,CAAC6V,GAAG,IAAI;MAChC,IAAIxM,IAAI,CAACwM,GAAG,CAAC,EAAE;QACb,IAAI2D,KAAK,IAAI,CAACkE,KAAK,CAAC3jB,QAAQ,CAAC8b,GAAG,CAAC,KAC9B,OAAOxM,IAAI,CAACwM,GAAG,CAAC,IAAI,QAAQ,IAAIrW,KAAK,CAACC,OAAO,CAAC4J,IAAI,CAACwM,GAAG,CAAC,CAAC,CAAC,IAC1DxM,IAAI,CAACwM,GAAG,CAAC,CAAC9a,MAAM,GAAGwU,qBAAqB,EAAE;UAC1C;QACF;QACA,IAAI,OAAOlG,IAAI,CAACwM,GAAG,CAAC,IAAI,QAAQ,EAAE;UAChC;QACF;QACAuI,EAAE,CAACvI,GAAG,CAAC,GAAGxM,IAAI,CAACwM,GAAG,CAAC;MACrB;IACF,CAAC,CAAC;IAEF,IAAIwB,MAAM,CAAC8G,OAAO,CAACC,EAAE,CAAC,CAACrjB,MAAM,IAAI,CAAC,EAAE;MAClC,OAAOqjB,EAAE;IACX;EACF;EACA,OAAO,IAAI;AACb;AAEA,IAAI,IAA4B,EAAE;EAChCC,MAAM,CAACC,OAAO,GAAGvL,MAAM;AACzB;;;;;;;;;;;;;;;;ACxkFa;;AAE2B;AAIpB;AAEpB,IAAInQ,WAAW;AAEf,SAAS4b,WAAWA,CAACC,MAAM,EAAE5I,GAAG,EAAE7H,KAAK,EAAE;EACvC,MAAMvK,GAAG,GAAG,IAAI0O,GAAG,CAACsM,MAAM,EAAEC,MAAM,CAACC,QAAQ,CAACC,MAAM,CAAC;EACnDnb,GAAG,CAACob,YAAY,CAAC7I,MAAM,CAACH,GAAG,EAAE7H,KAAK,CAAC;EACnC,OAAOvK,GAAG,CAAC1H,QAAQ,CAAC,CAAC,CAAC8gB,SAAS,CAAC6B,MAAM,CAACC,QAAQ,CAACC,MAAM,CAAC7jB,MAAM,CAAC;AAChE;AAUe,MAAM+jB,eAAe,CAAC;EACnC1lB,WAAWA,CAAC2lB,MAAM,EAAExb,OAAO,EAAE;IAC3B,IAAI,CAACyb,OAAO,GAAGD,MAAM;IACrB,IAAI,CAACE,QAAQ,GAAG1b,OAAO;IAEvB,IAAI,CAAC2b,OAAO,GAAGH,MAAM,CAACG,OAAO;IAC7B,IAAI,CAACC,UAAU,GAAGJ,MAAM,CAACK,YAAY,CAAC,CAAC;IAGvC,IAAI,CAACC,GAAG,GAAG,EAAE;EACf;EAgBAC,iBAAiBA,CAACC,OAAO,EAAElW,IAAI,EAAEmW,SAAS,EAAEC,UAAU,EAAEC,SAAS,EAAEC,SAAS,EAAE;IAC5E,IAAIlc,GAAG,GAAI,KAAI,IAAI,CAACwb,QAAS,UAAS;IACtC,IAAIM,OAAO,EAAE;MACX,IAAIK,IAAI,GAAGL,OAAO;MAClB,IAAIK,IAAI,CAACC,QAAQ,CAAC,GAAG,CAAC,EAAE;QAEtBD,IAAI,GAAGA,IAAI,CAAC1P,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;MAC1B;MACA,IAAI0P,IAAI,CAACxF,UAAU,CAAC,SAAS,CAAC,IAAIwF,IAAI,CAACxF,UAAU,CAAC,UAAU,CAAC,EAAE;QAC7D3W,GAAG,GAAGmc,IAAI,GAAGnc,GAAG;MAClB,CAAC,MAAM;QACL,MAAM,IAAIzJ,KAAK,CAAE,qBAAoBulB,OAAQ,GAAE,CAAC;MAClD;IACF;IAEA,MAAMO,QAAQ,GAAG,IAAI;IACrB,MAAMT,GAAG,GAAG,IAAIzc,WAAW,CAAC,CAAC;IAC7B,IAAI,CAACyc,GAAG,CAACpR,IAAI,CAACoR,GAAG,CAAC;IAElBA,GAAG,CAAC3X,IAAI,CAAC,MAAM,EAAEjE,GAAG,EAAE,IAAI,CAAC;IAC3B4b,GAAG,CAACU,gBAAgB,CAAC,iBAAiB,EAAE,IAAI,CAACb,OAAO,CAAC;IACrD,IAAI,IAAI,CAACC,UAAU,EAAE;MACnBE,GAAG,CAACU,gBAAgB,CAAC,eAAe,EAAG,SAAQ,IAAI,CAACZ,UAAU,CAACa,KAAM,EAAC,CAAC;IACzE;IAEA,IAAIC,SAAS,GAAG,IAAI;IACpB,IAAIC,QAAQ,GAAG,IAAI;IAEnB,MAAM1V,MAAM,GAAG,IAAItF,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MAC9C8a,SAAS,GAAGrY,OAAO;MACnBsY,QAAQ,GAAG/a,MAAM;IACnB,CAAC,CAAC;IAEFka,GAAG,CAACc,MAAM,CAACC,UAAU,GAAGC,CAAC,IAAI;MAC3B,IAAIA,CAAC,CAACC,gBAAgB,EAAE;QACtB,IAAIb,UAAU,EAAE;UACdA,UAAU,CAACY,CAAC,CAACE,MAAM,GAAGF,CAAC,CAACG,KAAK,CAAC;QAChC;QACA,IAAI,IAAI,CAACf,UAAU,EAAE;UACnB,IAAI,CAACA,UAAU,CAACY,CAAC,CAACE,MAAM,GAAGF,CAAC,CAACG,KAAK,CAAC;QACrC;MACF;IACF,CAAC;IAEDnB,GAAG,CAACoB,MAAM,GAAG,YAAW;MACtB,IAAI1Y,GAAG;MACP,IAAI;QACFA,GAAG,GAAGC,IAAI,CAACC,KAAK,CAAC,IAAI,CAACyY,QAAQ,EAAEhe,sDAAe,CAAC;MAClD,CAAC,CAAC,OAAOkG,GAAG,EAAE;QACZkX,QAAQ,CAACd,OAAO,CAACna,MAAM,CAAC,mDAAmD,EAAE,IAAI,CAAC6b,QAAQ,CAAC;QAC3F3Y,GAAG,GAAG;UACJI,IAAI,EAAE;YACJvH,IAAI,EAAE,IAAI,CAAC6G,MAAM;YACjBiB,IAAI,EAAE,IAAI,CAACiY;UACb;QACF,CAAC;MACH;MAEA,IAAI,IAAI,CAAClZ,MAAM,IAAI,GAAG,IAAI,IAAI,CAACA,MAAM,GAAG,GAAG,EAAE;QAC3C,IAAIwY,SAAS,EAAE;UACbA,SAAS,CAAClY,GAAG,CAACI,IAAI,CAACC,MAAM,CAAC3E,GAAG,CAAC;QAChC;QACA,IAAIic,SAAS,EAAE;UACbA,SAAS,CAAC3X,GAAG,CAACI,IAAI,CAAC;QACrB;MACF,CAAC,MAAM,IAAI,IAAI,CAACV,MAAM,IAAI,GAAG,EAAE;QAC7B,IAAIyY,QAAQ,EAAE;UACZA,QAAQ,CAAC,IAAIxf,sDAAS,CAACqH,GAAG,CAACI,IAAI,CAACO,IAAI,EAAEX,GAAG,CAACI,IAAI,CAACvH,IAAI,CAAC,CAAC;QACvD;QACA,IAAI+e,SAAS,EAAE;UACbA,SAAS,CAAC5X,GAAG,CAACI,IAAI,CAAC;QACrB;MACF,CAAC,MAAM;QACL2X,QAAQ,CAACd,OAAO,CAACna,MAAM,CAAC,0CAA0C,EAAE,IAAI,CAAC4C,MAAM,EAAE,IAAI,CAACiZ,QAAQ,CAAC;MACjG;IACF,CAAC;IAEDrB,GAAG,CAACpW,OAAO,GAAG,UAASoX,CAAC,EAAE;MACxB,IAAIH,QAAQ,EAAE;QACZA,QAAQ,CAACG,CAAC,IAAI,IAAIrmB,KAAK,CAAC,QAAQ,CAAC,CAAC;MACpC;MACA,IAAI2lB,SAAS,EAAE;QACbA,SAAS,CAAC,IAAI,CAAC;MACjB;IACF,CAAC;IAEDN,GAAG,CAACuB,OAAO,GAAG,UAASP,CAAC,EAAE;MACxB,IAAIH,QAAQ,EAAE;QACZA,QAAQ,CAAC,IAAIlmB,KAAK,CAAC,0BAA0B,CAAC,CAAC;MACjD;MACA,IAAI2lB,SAAS,EAAE;QACbA,SAAS,CAAC,IAAI,CAAC;MACjB;IACF,CAAC;IAED,IAAI;MACF,MAAMkB,IAAI,GAAG,IAAIC,QAAQ,CAAC,CAAC;MAC3BD,IAAI,CAAC7K,MAAM,CAAC,MAAM,EAAE3M,IAAI,CAAC;MACzBwX,IAAI,CAACE,GAAG,CAAC,IAAI,EAAE,IAAI,CAAC/B,OAAO,CAACgC,eAAe,CAAC,CAAC,CAAC;MAC9C,IAAIxB,SAAS,EAAE;QACbqB,IAAI,CAACE,GAAG,CAAC,OAAO,EAAEvB,SAAS,CAAC;MAC9B;MACAH,GAAG,CAAC/W,IAAI,CAACuY,IAAI,CAAC;IAChB,CAAC,CAAC,OAAOjY,GAAG,EAAE;MACZ,IAAIsX,QAAQ,EAAE;QACZA,QAAQ,CAACtX,GAAG,CAAC;MACf;MACA,IAAI+W,SAAS,EAAE;QACbA,SAAS,CAAC,IAAI,CAAC;MACjB;IACF;IAEA,OAAOnV,MAAM;EACf;EAcA2V,MAAMA,CAAC9W,IAAI,EAAEmW,SAAS,EAAEC,UAAU,EAAEC,SAAS,EAAEC,SAAS,EAAE;IACxD,MAAMJ,OAAO,GAAG,CAAC,IAAI,CAACP,OAAO,CAACiC,OAAO,GAAG,UAAU,GAAG,SAAS,IAAI,IAAI,CAACjC,OAAO,CAACkC,KAAK;IACpF,OAAO,IAAI,CAAC5B,iBAAiB,CAACC,OAAO,EAAElW,IAAI,EAAEmW,SAAS,EAAEC,UAAU,EAAEC,SAAS,EAAEC,SAAS,CAAC;EAC3F;EAWAwB,QAAQA,CAACC,WAAW,EAAE5K,QAAQ,EAAE6K,QAAQ,EAAE5B,UAAU,EAAE/V,OAAO,EAAE;IAC7D,IAAI,CAAC6U,wDAAa,CAAC6C,WAAW,CAAC,EAAE;MAE/B,IAAI1X,OAAO,EAAE;QACXA,OAAO,CAAE,YAAW0X,WAAY,kCAAiC,CAAC;MACpE;MACA;IACF;IACA,IAAI,CAAC,IAAI,CAACjC,UAAU,EAAE;MACpB,IAAIzV,OAAO,EAAE;QACXA,OAAO,CAAC,yBAAyB,CAAC;MACpC;MACA;IACF;IACA,MAAMoW,QAAQ,GAAG,IAAI;IAErB,MAAMT,GAAG,GAAG,IAAIzc,WAAW,CAAC,CAAC;IAC7B,IAAI,CAACyc,GAAG,CAACpR,IAAI,CAACoR,GAAG,CAAC;IAGlB+B,WAAW,GAAG5C,WAAW,CAAC4C,WAAW,EAAE,OAAO,EAAE,GAAG,CAAC;IAGpD/B,GAAG,CAAC3X,IAAI,CAAC,KAAK,EAAE0Z,WAAW,EAAE,IAAI,CAAC;IAClC/B,GAAG,CAACU,gBAAgB,CAAC,iBAAiB,EAAE,IAAI,CAACb,OAAO,CAAC;IACrDG,GAAG,CAACU,gBAAgB,CAAC,eAAe,EAAE,QAAQ,GAAG,IAAI,CAACZ,UAAU,CAACa,KAAK,CAAC;IACvEX,GAAG,CAACiC,YAAY,GAAG,MAAM;IAEzBjC,GAAG,CAACe,UAAU,GAAG,UAASC,CAAC,EAAE;MAC3B,IAAIZ,UAAU,EAAE;QAGdA,UAAU,CAACY,CAAC,CAACE,MAAM,CAAC;MACtB;IACF,CAAC;IAED,IAAIN,SAAS,GAAG,IAAI;IACpB,IAAIC,QAAQ,GAAG,IAAI;IAEnB,MAAM1V,MAAM,GAAG,IAAItF,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;MAC9C8a,SAAS,GAAGrY,OAAO;MACnBsY,QAAQ,GAAG/a,MAAM;IACnB,CAAC,CAAC;IAIFka,GAAG,CAACoB,MAAM,GAAG,YAAW;MACtB,IAAI,IAAI,CAAChZ,MAAM,IAAI,GAAG,EAAE;QACtB,MAAM8Z,IAAI,GAAGC,QAAQ,CAACC,aAAa,CAAC,GAAG,CAAC;QAExCF,IAAI,CAAC7O,IAAI,GAAGgM,MAAM,CAACvM,GAAG,CAACC,eAAe,CAAC,IAAIC,IAAI,CAAC,CAAC,IAAI,CAACqO,QAAQ,CAAC,EAAE;UAC/DpO,IAAI,EAAE+O;QACR,CAAC,CAAC,CAAC;QACHE,IAAI,CAACpJ,KAAK,CAACuJ,OAAO,GAAG,MAAM;QAC3BH,IAAI,CAACI,YAAY,CAAC,UAAU,EAAEnL,QAAQ,CAAC;QACvCgL,QAAQ,CAAC/J,IAAI,CAACmK,WAAW,CAACL,IAAI,CAAC;QAC/BA,IAAI,CAACM,KAAK,CAAC,CAAC;QACZL,QAAQ,CAAC/J,IAAI,CAACqK,WAAW,CAACP,IAAI,CAAC;QAC/B7C,MAAM,CAACvM,GAAG,CAAC4P,eAAe,CAACR,IAAI,CAAC7O,IAAI,CAAC;QACrC,IAAIuN,SAAS,EAAE;UACbA,SAAS,CAAC,CAAC;QACb;MACF,CAAC,MAAM,IAAI,IAAI,CAACxY,MAAM,IAAI,GAAG,IAAIyY,QAAQ,EAAE;QAIzC,MAAM8B,MAAM,GAAG,IAAIC,UAAU,CAAC,CAAC;QAC/BD,MAAM,CAACvB,MAAM,GAAG,YAAW;UACzB,IAAI;YACF,MAAM1Y,GAAG,GAAGC,IAAI,CAACC,KAAK,CAAC,IAAI,CAACuC,MAAM,EAAE9H,sDAAe,CAAC;YACpDwd,QAAQ,CAAC,IAAIxf,sDAAS,CAACqH,GAAG,CAACI,IAAI,CAACO,IAAI,EAAEX,GAAG,CAACI,IAAI,CAACvH,IAAI,CAAC,CAAC;UACvD,CAAC,CAAC,OAAOgI,GAAG,EAAE;YACZkX,QAAQ,CAACd,OAAO,CAACna,MAAM,CAAC,mDAAmD,EAAE,IAAI,CAAC2F,MAAM,CAAC;YACzF0V,QAAQ,CAACtX,GAAG,CAAC;UACf;QACF,CAAC;QACDoZ,MAAM,CAACE,UAAU,CAAC,IAAI,CAACxB,QAAQ,CAAC;MAClC;IACF,CAAC;IAEDrB,GAAG,CAACpW,OAAO,GAAG,UAASoX,CAAC,EAAE;MACxB,IAAIH,QAAQ,EAAE;QACZA,QAAQ,CAAC,IAAIlmB,KAAK,CAAC,QAAQ,CAAC,CAAC;MAC/B;MACA,IAAI0P,OAAO,EAAE;QACXA,OAAO,CAAC2W,CAAC,CAAC;MACZ;IACF,CAAC;IAEDhB,GAAG,CAACuB,OAAO,GAAG,YAAW;MACvB,IAAIV,QAAQ,EAAE;QACZA,QAAQ,CAAC,IAAI,CAAC;MAChB;IACF,CAAC;IAED,IAAI;MACFb,GAAG,CAAC/W,IAAI,CAAC,CAAC;IACZ,CAAC,CAAC,OAAOM,GAAG,EAAE;MACZ,IAAIsX,QAAQ,EAAE;QACZA,QAAQ,CAACtX,GAAG,CAAC;MACf;MACA,IAAIc,OAAO,EAAE;QACXA,OAAO,CAACd,GAAG,CAAC;MACd;IACF;IAEA,OAAO4B,MAAM;EACf;EAKA2X,MAAMA,CAAA,EAAG;IACP,IAAI,CAAC9C,GAAG,CAACrf,OAAO,CAAC2K,GAAG,IAAI;MACtB,IAAIA,GAAG,CAACnD,UAAU,GAAG,CAAC,EAAE;QACtBmD,GAAG,CAAChC,KAAK,CAAC,CAAC;MACb;IACF,CAAC,CAAC;EACJ;EAOA,OAAOyZ,kBAAkBA,CAACxd,WAAW,EAAE;IACrChC,WAAW,GAAGgC,WAAW;EAC3B;AACF;;;;;;;;;;;;;;AC/Ta;AAUE,MAAMyd,cAAc,CAAC;EAClCjpB,WAAWA,CAAC2gB,MAAM,EAAE;IAClB,IAAI,CAACtP,KAAK,GAAGsP,MAAM;IACnB,IAAI,CAACuI,IAAI,GAAG,CAAC,CAAC;EAChB;EAGA,CAACC,YAAYC,CAAA,EAAG;IACd,OAAO,IAAI,CAAC/X,KAAK,CAACgB,QAAQ,GAAG5N,SAAS,GAAG,IAAI,CAAC4M,KAAK,CAACgY,OAAO;EAC7D;EAGA,CAACC,YAAYC,CAAA,EAAG;IACd,IAAI,IAAI,CAAClY,KAAK,CAACmY,SAAS,CAAC,CAAC,EAAE;MAC1B,OAAO,IAAI,CAAC,CAACL,YAAY,CAAC,CAAC;IAC7B;IACA,OAAO,IAAI,CAAC9X,KAAK,CAACgB,QAAQ,GAAG5N,SAAS,GAAG,IAAI,CAAC4M,KAAK,CAACoY,eAAe;EACrE;EAUAC,QAAQA,CAACjjB,KAAK,EAAEC,MAAM,EAAE+N,KAAK,EAAE;IAC7B,IAAI,CAACyU,IAAI,CAAC,MAAM,CAAC,GAAG;MAClBziB,KAAK,EAAEA,KAAK;MACZC,MAAM,EAAEA,MAAM;MACd+N,KAAK,EAAEA;IACT,CAAC;IACD,OAAO,IAAI;EACb;EASAkV,aAAaA,CAAClV,KAAK,EAAE;IACnB,OAAO,IAAI,CAACiV,QAAQ,CAAC,IAAI,CAACrY,KAAK,CAACuY,OAAO,GAAG,CAAC,GAAG,IAAI,CAACvY,KAAK,CAACuY,OAAO,GAAG,CAAC,GAAGnlB,SAAS,EAAEA,SAAS,EAAEgQ,KAAK,CAAC;EACrG;EASAoV,eAAeA,CAACpV,KAAK,EAAE;IACrB,OAAO,IAAI,CAACiV,QAAQ,CAACjlB,SAAS,EAAE,IAAI,CAAC4M,KAAK,CAACyY,OAAO,GAAG,CAAC,GAAG,IAAI,CAACzY,KAAK,CAACyY,OAAO,GAAGrlB,SAAS,EAAEgQ,KAAK,CAAC;EACjG;EASAsV,QAAQA,CAACC,GAAG,EAAE;IACZ,IAAI,CAACd,IAAI,CAAC,MAAM,CAAC,GAAG;MAClBc,GAAG,EAAEA;IACP,CAAC;IACD,OAAO,IAAI;EACb;EAOAC,aAAaA,CAAA,EAAG;IACd,OAAO,IAAI,CAACF,QAAQ,CAAC,IAAI,CAAC,CAACZ,YAAY,CAAC,CAAC,CAAC;EAC5C;EAWAe,OAAOA,CAACF,GAAG,EAAEvV,KAAK,EAAE0V,WAAW,EAAE;IAC/B,MAAMC,IAAI,GAAG;MACXJ,GAAG,EAAEA,GAAG;MACRvV,KAAK,EAAEA;IACT,CAAC;IACD,IAAI,IAAI,CAACpD,KAAK,CAACgZ,OAAO,CAAC,CAAC,IAAI,IAAI,EAAE;MAChCD,IAAI,CAAC/Y,KAAK,GAAG8Y,WAAW;IAC1B,CAAC,MAAM;MACLC,IAAI,CAAC7W,IAAI,GAAG4W,WAAW;IACzB;IACA,IAAI,CAACjB,IAAI,CAAC,KAAK,CAAC,GAAGkB,IAAI;IACvB,OAAO,IAAI;EACb;EAUAE,UAAUA,CAACN,GAAG,EAAEG,WAAW,EAAE;IAC3B,OAAO,IAAI,CAACD,OAAO,CAACF,GAAG,EAAEvlB,SAAS,EAAE0lB,WAAW,CAAC;EAClD;EASAI,eAAeA,CAACJ,WAAW,EAAE;IAC3B,OAAO,IAAI,CAACG,UAAU,CAAC,IAAI,CAACjZ,KAAK,CAACoY,eAAe,EAAEU,WAAW,CAAC;EACjE;EASAK,YAAYA,CAAC/V,KAAK,EAAE;IAClB,OAAO,IAAI,CAACyV,OAAO,CAAC,IAAI,CAAC,CAACZ,YAAY,CAAC,CAAC,EAAE7U,KAAK,CAAC;EAClD;EAOAgW,QAAQA,CAAA,EAAG;IACT,IAAI,CAACvB,IAAI,CAAC,MAAM,CAAC,GAAG,IAAI;IACxB,OAAO,IAAI;EACb;EAOAwB,QAAQA,CAAA,EAAG;IACT,IAAI,IAAI,CAACrZ,KAAK,CAACgZ,OAAO,CAAC,CAAC,IAAI,IAAI,EAAE;MAChC,IAAI,CAACnB,IAAI,CAAC,MAAM,CAAC,GAAG,IAAI;IAC1B,CAAC,MAAM;MACL,IAAI,CAAC7X,KAAK,CAACuU,OAAO,CAACna,MAAM,CAAC,wDAAwD,EAAE,IAAI,CAAC4F,KAAK,CAACgZ,OAAO,CAAC,CAAC,CAAC;IAC3G;IACA,OAAO,IAAI;EACb;EAUAM,OAAOA,CAAClkB,KAAK,EAAEgO,KAAK,EAAE;IACpB,IAAIhO,KAAK,IAAIgO,KAAK,EAAE;MAClB,IAAI,CAACyU,IAAI,CAAC,KAAK,CAAC,GAAG;QACjBziB,KAAK,EAAEA,KAAK;QACZgO,KAAK,EAAEA;MACT,CAAC;IACH;IACA,OAAO,IAAI;EACb;EASAmW,YAAYA,CAACnW,KAAK,EAAE;IAGlB,OAAO,IAAI,CAACkW,OAAO,CAAC,IAAI,CAACtZ,KAAK,CAACuY,OAAO,GAAG,CAAC,GAAG,IAAI,CAACvY,KAAK,CAACwZ,OAAO,GAAG,CAAC,GAAGpmB,SAAS,EAAEgQ,KAAK,CAAC;EACzF;EAQAqW,OAAOA,CAAC5B,IAAI,EAAE;IACZ,OAAO,IAAI,CAACA,IAAI,CAACA,IAAI,CAAC;EACxB;EAQA6B,KAAKA,CAAA,EAAG;IACN,MAAM7B,IAAI,GAAG,EAAE;IACf,IAAIla,MAAM,GAAG,CAAC,CAAC;IACf,CAAC,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,EAAE,KAAK,CAAC,CAACpI,OAAO,CAAE6V,GAAG,IAAK;MAC9D,IAAI,IAAI,CAACyM,IAAI,CAAChU,cAAc,CAACuH,GAAG,CAAC,EAAE;QACjCyM,IAAI,CAACrU,IAAI,CAAC4H,GAAG,CAAC;QACd,IAAIwB,MAAM,CAAC+M,mBAAmB,CAAC,IAAI,CAAC9B,IAAI,CAACzM,GAAG,CAAC,CAAC,CAAC9a,MAAM,GAAG,CAAC,EAAE;UACzDqN,MAAM,CAACyN,GAAG,CAAC,GAAG,IAAI,CAACyM,IAAI,CAACzM,GAAG,CAAC;QAC9B;MACF;IACF,CAAC,CAAC;IACF,IAAIyM,IAAI,CAACvnB,MAAM,GAAG,CAAC,EAAE;MACnBqN,MAAM,CAACka,IAAI,GAAGA,IAAI,CAACpJ,IAAI,CAAC,GAAG,CAAC;IAC9B,CAAC,MAAM;MACL9Q,MAAM,GAAGvK,SAAS;IACpB;IACA,OAAOuK,MAAM;EACf;AACF;;;;;;;;;;;;;;;;;;;;;;;;AC7Oa;;AAE6B;AACP;AACK;AACH;AACJ;AACc;AAK3B;AAKb,MAAMqc,KAAK,CAAC;EAoBjBrrB,WAAWA,CAACyH,IAAI,EAAE6jB,SAAS,EAAE;IAE3B,IAAI,CAAC1F,OAAO,GAAG,IAAI;IAInB,IAAI,CAACne,IAAI,GAAGA,IAAI;IAEhB,IAAI,CAAC8jB,OAAO,GAAG,IAAI;IAEnB,IAAI,CAAClC,OAAO,GAAG,IAAI;IAEnB,IAAI,CAACmC,OAAO,GAAG,IAAIC,IAAI,CAAC,CAAC,CAAC;IAE1B,IAAI,CAACxrB,GAAG,GAAG,IAAIF,uDAAU,CAAC,IAAI,CAAC;IAE/B,IAAI,CAAC2rB,OAAO,GAAG,IAAI;IAEnB,IAAI,CAACvY,MAAM,GAAG,IAAI;IAElB,IAAI,CAACwY,OAAO,GAAG,IAAI;IAInB,IAAI,CAACC,MAAM,GAAG,CAAC,CAAC;IAGhB,IAAI,CAACC,YAAY,GAAGZ,mDAAiB;IAGrC,IAAI,CAACrB,OAAO,GAAG,CAAC;IAEhB,IAAI,CAACE,OAAO,GAAG,CAAC;IAEhB,IAAI,CAACgC,cAAc,GAAG,KAAK;IAE3B,IAAI,CAACjB,OAAO,GAAG,CAAC;IAEhB,IAAI,CAACkB,sBAAsB,GAAG,IAAI;IAGlC,IAAI,CAAC3W,KAAK,GAAG,EAAE;IAEf,IAAI,CAAC4W,YAAY,GAAG,EAAE;IAKtB,IAAI,CAACC,gBAAgB,GAAG,CAAC,CAAC;IAE1B,IAAI,CAACC,SAAS,GAAG,IAAI3nB,mDAAO,CAAC,CAACO,CAAC,EAAEC,CAAC,KAAK;MACrC,OAAOD,CAAC,CAACmP,GAAG,GAAGlP,CAAC,CAACkP,GAAG;IACtB,CAAC,EAAE,IAAI,CAAC;IAER,IAAI,CAACkY,SAAS,GAAG,KAAK;IAEtB,IAAI,CAAC1C,eAAe,GAAG,IAAIgC,IAAI,CAAC,CAAC,CAAC;IAElC,IAAI,CAACW,IAAI,GAAG,IAAI;IAEhB,IAAI,CAAC/Z,QAAQ,GAAG,KAAK;IAGrB,IAAI,CAACga,kBAAkB,GAAG,IAAI;IAG9B,IAAIf,SAAS,EAAE;MACb,IAAI,CAACgB,MAAM,GAAGhB,SAAS,CAACgB,MAAM;MAC9B,IAAI,CAACC,MAAM,GAAGjB,SAAS,CAACiB,MAAM;MAC9B,IAAI,CAACC,MAAM,GAAGlB,SAAS,CAACkB,MAAM;MAC9B,IAAI,CAACC,MAAM,GAAGnB,SAAS,CAACmB,MAAM;MAE9B,IAAI,CAACC,UAAU,GAAGpB,SAAS,CAACoB,UAAU;MAEtC,IAAI,CAACC,SAAS,GAAGrB,SAAS,CAACqB,SAAS;MAEpC,IAAI,CAACC,aAAa,GAAGtB,SAAS,CAACsB,aAAa;MAC5C,IAAI,CAACC,aAAa,GAAGvB,SAAS,CAACuB,aAAa;MAC5C,IAAI,CAACC,cAAc,GAAGxB,SAAS,CAACwB,cAAc;MAC9C,IAAI,CAACC,aAAa,GAAGzB,SAAS,CAACyB,aAAa;MAC5C,IAAI,CAACC,qBAAqB,GAAG1B,SAAS,CAAC0B,qBAAqB;IAC9D;EACF;EAWA,OAAOC,SAASA,CAACxlB,IAAI,EAAE;IACrB,MAAMylB,KAAK,GAAG;MACZ,IAAI,EAAEjC,gDAAc;MACpB,KAAK,EAAEA,iDAAe;MACtB,KAAK,EAAEA,iDAAe;MACtB,KAAK,EAAEA,iDAAe;MACtB,KAAK,EAAEA,iDAAe;MACtB,KAAK,EAAEA,iDAAe;MACtB,KAAK,EAAEA,iDAAe;MACtB,KAAK,EAAEA,iDAAe/iB;IACxB,CAAC;IACD,OAAOglB,KAAK,CAAE,OAAOzlB,IAAI,IAAI,QAAQ,GAAIA,IAAI,CAACgc,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,KAAK,CAAC;EACxE;EAQA,OAAO0J,aAAaA,CAAC1lB,IAAI,EAAE;IACzB,OAAO4jB,KAAK,CAAC4B,SAAS,CAACxlB,IAAI,CAAC,IAAIwjB,gDAAc;EAChD;EASA,OAAOmC,gBAAgBA,CAAC3lB,IAAI,EAAE;IAC5B,OAAO4jB,KAAK,CAAC4B,SAAS,CAACxlB,IAAI,CAAC,IAAIwjB,iDAAe;EACjD;EASA,OAAOoC,cAAcA,CAAC5lB,IAAI,EAAE;IAC1B,OAAO4jB,KAAK,CAAC4B,SAAS,CAACxlB,IAAI,CAAC,IAAIwjB,iDAAe;EACjD;EASA,OAAOqC,eAAeA,CAAC7lB,IAAI,EAAE;IAC3B,OAAO4jB,KAAK,CAACgC,cAAc,CAAC5lB,IAAI,CAAC,IAAI4jB,KAAK,CAAC+B,gBAAgB,CAAC3lB,IAAI,CAAC;EACnE;EASA,OAAO8lB,mBAAmBA,CAAC9lB,IAAI,EAAE;IAC/B,OAAQ,OAAOA,IAAI,IAAI,QAAQ,KAC5BA,IAAI,CAACgc,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,IAAIwH,iDAAe,IAAIxjB,IAAI,CAACgc,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,IAAIwH,sDAAoB,CAAC;EAC7F;EASA,OAAOuC,kBAAkBA,CAAC/lB,IAAI,EAAE;IAC9B,OAAQ,OAAOA,IAAI,IAAI,QAAQ,KAC5BA,IAAI,CAACgc,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,IAAIwH,kDAAgB,IAAIxjB,IAAI,CAACgc,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,IAAIwH,sDAAoB,CAAC;EAC9F;EAMAwC,YAAYA,CAAA,EAAG;IACb,OAAO,IAAI,CAACtB,SAAS;EACvB;EASAuB,SAASA,CAACC,SAAS,EAAEC,SAAS,EAAE;IAE9BlhB,YAAY,CAAC,IAAI,CAAC2f,kBAAkB,CAAC;IACrC,IAAI,CAACA,kBAAkB,GAAG,IAAI;IAG9B,IAAI,IAAI,CAACF,SAAS,EAAE;MAClB,OAAOrgB,OAAO,CAAC0C,OAAO,CAAC,IAAI,CAAC;IAC9B;IAKA,OAAO,IAAI,CAACoX,OAAO,CAAC8H,SAAS,CAAC,IAAI,CAACjmB,IAAI,IAAIwjB,iDAAe,EAAE0C,SAAS,EAAEC,SAAS,CAAC,CAACrQ,IAAI,CAACxO,IAAI,IAAI;MAC7F,IAAIA,IAAI,CAACvH,IAAI,IAAI,GAAG,EAAE;QAEpB,OAAOuH,IAAI;MACb;MAEA,IAAI,CAACod,SAAS,GAAG,IAAI;MACrB,IAAI,CAAC9Z,QAAQ,GAAG,KAAK;MACrB,IAAI,CAACpS,GAAG,GAAI8O,IAAI,CAACC,MAAM,IAAID,IAAI,CAACC,MAAM,CAAC/O,GAAG,GAAI8O,IAAI,CAACC,MAAM,CAAC/O,GAAG,GAAG,IAAI,CAACA,GAAG;MAGxE,IAAI,IAAI,CAACmsB,IAAI,EAAE;QACb,OAAO,IAAI,CAACA,IAAI;QAEhB,IAAI,IAAI,CAAC3kB,IAAI,IAAIsH,IAAI,CAACsC,KAAK,EAAE;UAE3B,IAAI,CAACwc,aAAa,CAAC,CAAC;UACpB,IAAI,CAACpmB,IAAI,GAAGsH,IAAI,CAACsC,KAAK;QACxB;QACA,IAAI,CAACyc,aAAa,CAAC,CAAC;QAEpB,IAAI,CAACvC,OAAO,GAAGxc,IAAI,CAACgf,EAAE;QACtB,IAAI,CAAC1E,OAAO,GAAGta,IAAI,CAACgf,EAAE;QAEtB,IAAI,IAAI,CAACtmB,IAAI,IAAIwjB,gDAAc,IAAI,IAAI,CAACxjB,IAAI,IAAIwjB,iDAAe,EAAE;UAE/D,MAAM+C,EAAE,GAAG,IAAI,CAACpI,OAAO,CAACqI,UAAU,CAAC,CAAC;UACpC,IAAID,EAAE,CAACrB,SAAS,EAAE;YAChBqB,EAAE,CAACrB,SAAS,CAAC,IAAI,CAAC;UACpB;UACA,IAAIqB,EAAE,CAACpB,aAAa,EAAE;YACpBoB,EAAE,CAACpB,aAAa,CAAC,CAAC,IAAI,CAACnlB,IAAI,CAAC,EAAE,CAAC,CAAC;UAClC;QACF;QAEA,IAAImmB,SAAS,IAAIA,SAAS,CAACM,IAAI,EAAE;UAC/BN,SAAS,CAACM,IAAI,CAACC,aAAa,GAAG,IAAI;UACnC,IAAI,CAACC,gBAAgB,CAACR,SAAS,CAACM,IAAI,CAAC;QACvC;MACF;MACA,OAAOnf,IAAI;IACb,CAAC,CAAC;EACJ;EAYAsf,aAAaA,CAACpe,IAAI,EAAEqe,MAAM,EAAE;IAC1B,OAAO,IAAI,CAAC1I,OAAO,CAACyI,aAAa,CAAC,IAAI,CAAC5mB,IAAI,EAAEwI,IAAI,EAAEqe,MAAM,CAAC;EAC5D;EAUAC,OAAOA,CAACte,IAAI,EAAEqe,MAAM,EAAE;IACpB,OAAO,IAAI,CAACE,cAAc,CAAC,IAAI,CAACH,aAAa,CAACpe,IAAI,EAAEqe,MAAM,CAAC,CAAC;EAC9D;EAUAE,cAAcA,CAACtb,GAAG,EAAE;IAClB,IAAI,CAAC,IAAI,CAACiZ,SAAS,EAAE;MACnB,OAAOrgB,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,kCAAkC,CAAC,CAAC;IACtE;IACA,IAAI,IAAI,CAAC6tB,QAAQ,EAAE;MACjB,OAAO3iB,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,mCAAmC,CAAC,CAAC;IACvE;IAGAsS,GAAG,CAACub,QAAQ,GAAG,IAAI;IACnBvb,GAAG,CAACwb,OAAO,GAAG,KAAK;IAGnB,IAAI9M,WAAW,GAAG,IAAI;IACtB,IAAIjI,6DAAkB,CAACzG,GAAG,CAAC8H,OAAO,CAAC,EAAE;MACnC4G,WAAW,GAAG,EAAE;MAChBjI,0DAAe,CAACzG,GAAG,CAAC8H,OAAO,EAAE/K,IAAI,IAAI;QACnC,IAAIA,IAAI,EAAE;UACR,IAAIA,IAAI,CAACwJ,GAAG,EAAE;YACZmI,WAAW,CAAC/M,IAAI,CAAC5E,IAAI,CAACwJ,GAAG,CAAC;UAC5B;UACA,IAAIxJ,IAAI,CAACyK,MAAM,EAAE;YACfkH,WAAW,CAAC/M,IAAI,CAAC5E,IAAI,CAACyK,MAAM,CAAC;UAC/B;QACF;MACF,CAAC,CAAC;MACF,IAAIkH,WAAW,CAACjgB,MAAM,IAAI,CAAC,EAAE;QAC3BigB,WAAW,GAAG,IAAI;MACpB;IACF;IAEA,OAAO,IAAI,CAACgE,OAAO,CAAC4I,cAAc,CAACtb,GAAG,EAAE0O,WAAW,CAAC,CAACrE,IAAI,CAACxO,IAAI,IAAI;MAChEmE,GAAG,CAACub,QAAQ,GAAG,KAAK;MACpBvb,GAAG,CAAC6a,EAAE,GAAGhf,IAAI,CAACgf,EAAE;MAChB,IAAI,CAACY,aAAa,CAACzb,GAAG,EAAEnE,IAAI,CAACC,MAAM,CAACiF,GAAG,CAAC;MACxC,IAAI,CAAC2a,gCAAgC,CAAC1b,GAAG,CAAC;MAC1C,IAAI,CAAC2b,UAAU,CAAC3b,GAAG,CAAC;MACpB,OAAOnE,IAAI;IACb,CAAC,CAAC,CAAC7B,KAAK,CAACsC,GAAG,IAAI;MACd,IAAI,CAACoW,OAAO,CAACna,MAAM,CAAC,yCAAyC,EAAE+D,GAAG,CAAC;MACnE0D,GAAG,CAACub,QAAQ,GAAG,KAAK;MACpBvb,GAAG,CAACwb,OAAO,GAAG,IAAI;MAClB,IAAI,IAAI,CAACpC,MAAM,EAAE;QACf,IAAI,CAACA,MAAM,CAAC,CAAC;MACf;IACF,CAAC,CAAC;EACJ;EAeAwC,YAAYA,CAAC5b,GAAG,EAAEjG,IAAI,EAAE;IACtB,MAAMgH,GAAG,GAAGf,GAAG,CAACe,GAAG,IAAI,IAAI,CAAC8a,eAAe,CAAC,CAAC;IAC7C,IAAI,CAAC7b,GAAG,CAACib,aAAa,EAAE;MAGtBjb,GAAG,CAACib,aAAa,GAAG,IAAI;MACxBjb,GAAG,CAACe,GAAG,GAAGA,GAAG;MACbf,GAAG,CAAC6a,EAAE,GAAG,IAAItC,IAAI,CAAC,CAAC;MACnBvY,GAAG,CAACkB,IAAI,GAAG,IAAI,CAACwR,OAAO,CAACoJ,gBAAgB,CAAC,CAAC;MAG1C9b,GAAG,CAAC+b,MAAM,GAAG,IAAI;MAEjB,IAAI,CAAC/C,SAAS,CAACjmB,GAAG,CAACiN,GAAG,CAAC;MACvB,IAAI,CAAC0S,OAAO,CAACsJ,GAAG,CAACrb,UAAU,CAACX,GAAG,CAAC;MAEhC,IAAI,IAAI,CAACoZ,MAAM,EAAE;QACf,IAAI,CAACA,MAAM,CAACpZ,GAAG,CAAC;MAClB;IACF;IAGA,OAAO,CAACjG,IAAI,IAAInB,OAAO,CAAC0C,OAAO,CAAC,CAAC,EAC9B+O,IAAI,CAAC/S,CAAC,IAAI;MACT,IAAI0I,GAAG,CAACic,UAAU,EAAE;QAClB,OAAO;UACL3nB,IAAI,EAAE,GAAG;UACT8H,IAAI,EAAE;QACR,CAAC;MACH;MACA,OAAO,IAAI,CAACkf,cAAc,CAACtb,GAAG,CAAC;IACjC,CAAC,CAAC,CAAChG,KAAK,CAACsC,GAAG,IAAI;MACd,IAAI,CAACoW,OAAO,CAACna,MAAM,CAAC,iCAAiC,EAAE+D,GAAG,CAAC;MAC3D0D,GAAG,CAACub,QAAQ,GAAG,KAAK;MACpBvb,GAAG,CAACwb,OAAO,GAAG,IAAI;MAClBxb,GAAG,CAACkc,MAAM,GAAG5f,GAAG,YAAYlI,sDAAS,GAAIkI,GAAG,CAAChI,IAAI,IAAI,GAAG,IAAIgI,GAAG,CAAChI,IAAI,GAAG,GAAG,GAAI,KAAK;MACnF,IAAI,IAAI,CAAC8kB,MAAM,EAAE;QACf,IAAI,CAACA,MAAM,CAAC,CAAC;MACf;MAEA,MAAM9c,GAAG;IACX,CAAC,CAAC;EACN;EAWA6f,KAAKA,CAACC,KAAK,EAAE;IAEX,IAAI,CAAC,IAAI,CAACnD,SAAS,IAAI,CAACmD,KAAK,EAAE;MAC7B,OAAOxjB,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,6BAA6B,CAAC,CAAC;IACjE;IAGA,OAAO,IAAI,CAACglB,OAAO,CAACyJ,KAAK,CAAC,IAAI,CAAC5nB,IAAI,EAAE6nB,KAAK,CAAC,CAAC/R,IAAI,CAACxO,IAAI,IAAI;MACvD,IAAI,CAACwgB,SAAS,CAAC,CAAC;MAChB,IAAID,KAAK,EAAE;QACT,IAAI,CAACE,KAAK,CAAC,CAAC;MACd;MACA,OAAOzgB,IAAI;IACb,CAAC,CAAC;EACJ;EAWA0gB,YAAYA,CAACH,KAAK,EAAEI,KAAK,EAAE;IACzBhjB,YAAY,CAAC,IAAI,CAAC2f,kBAAkB,CAAC;IACrC,IAAI,CAACA,kBAAkB,GAAGrf,UAAU,CAACxC,CAAC,IAAI;MACxC,IAAI,CAAC6hB,kBAAkB,GAAG,IAAI;MAC9B,IAAI,CAACgD,KAAK,CAACC,KAAK,CAAC;IACnB,CAAC,EAAEI,KAAK,CAAC;EACX;EAUAC,OAAOA,CAAC3gB,MAAM,EAAE;IAEd,OAAO,IAAI,CAAC4W,OAAO,CAAC+J,OAAO,CAAC,IAAI,CAACloB,IAAI,EAAEuH,MAAM,CAAC;EAChD;EASA4gB,eAAeA,CAACnb,KAAK,EAAEob,OAAO,EAAE;IAC9B,IAAIrb,KAAK,GAAGqb,OAAO,GACjB,IAAI,CAACC,cAAc,CAAC,CAAC,CAACnG,aAAa,CAAClV,KAAK,CAAC,GAC1C,IAAI,CAACqb,cAAc,CAAC,CAAC,CAACjG,eAAe,CAACpV,KAAK,CAAC;IAG9C,OAAO,IAAI,CAACsb,aAAa,CAAC,IAAI,CAACnK,OAAO,CAACsJ,GAAG,EAAE1a,KAAK,CAACsW,OAAO,CAAC,MAAM,CAAC,CAAC,CAC/DvN,IAAI,CAAE3X,KAAK,IAAK;MACf,IAAIA,KAAK,IAAI6O,KAAK,EAAE;QAElB,OAAO3I,OAAO,CAAC0C,OAAO,CAAC;UACrB6C,KAAK,EAAE,IAAI,CAAC5J,IAAI;UAChBD,IAAI,EAAE,GAAG;UACTwH,MAAM,EAAE;YACNpJ,KAAK,EAAEA;UACT;QACF,CAAC,CAAC;MACJ;MAGA6O,KAAK,IAAI7O,KAAK;MAEd4O,KAAK,GAAGqb,OAAO,GAAG,IAAI,CAACC,cAAc,CAAC,CAAC,CAACnG,aAAa,CAAClV,KAAK,CAAC,GAC1D,IAAI,CAACqb,cAAc,CAAC,CAAC,CAACjG,eAAe,CAACpV,KAAK,CAAC;MAC9C,IAAIub,OAAO,GAAG,IAAI,CAACL,OAAO,CAACnb,KAAK,CAACuW,KAAK,CAAC,CAAC,CAAC;MACzC,IAAI,CAAC8E,OAAO,EAAE;QACZG,OAAO,GAAGA,OAAO,CAACzS,IAAI,CAACxO,IAAI,IAAI;UAC7B,IAAIA,IAAI,IAAIA,IAAI,CAACC,MAAM,IAAI,CAACD,IAAI,CAACC,MAAM,CAACpJ,KAAK,EAAE;YAC7C,IAAI,CAACkmB,cAAc,GAAG,IAAI;UAC5B;QACF,CAAC,CAAC;MACJ;MACA,OAAOkE,OAAO;IAChB,CAAC,CAAC;EACN;EAQAC,OAAOA,CAACjhB,MAAM,EAAE;IACd,IAAIA,MAAM,CAACmG,IAAI,EAAE;MACfnG,MAAM,CAACmG,IAAI,GAAGiW,yDAAc,CAACpc,MAAM,CAACmG,IAAI,CAAC;IAC3C;IAEA,OAAO,IAAI,CAACyQ,OAAO,CAACqK,OAAO,CAAC,IAAI,CAACxoB,IAAI,EAAEuH,MAAM,CAAC,CAC3CuO,IAAI,CAACxO,IAAI,IAAI;MACZ,IAAIA,IAAI,IAAIA,IAAI,CAACvH,IAAI,IAAI,GAAG,EAAE;QAE5B,OAAOuH,IAAI;MACb;MAEA,IAAIC,MAAM,CAAC0E,GAAG,EAAE;QACd1E,MAAM,CAAC0E,GAAG,CAACrC,KAAK,GAAG,IAAI,CAAC5J,IAAI;QAC5B,IAAIsH,IAAI,CAACC,MAAM,IAAID,IAAI,CAACC,MAAM,CAAC/O,GAAG,EAAE;UAClC+O,MAAM,CAAC0E,GAAG,CAACzT,GAAG,GAAG8O,IAAI,CAACC,MAAM,CAAC/O,GAAG;UAChC+O,MAAM,CAAC0E,GAAG,CAAC2V,OAAO,GAAGta,IAAI,CAACgf,EAAE;QAC9B;QACA,IAAI,CAAC/e,MAAM,CAAC0E,GAAG,CAACH,IAAI,EAAE;UAGpBvE,MAAM,CAAC0E,GAAG,CAACH,IAAI,GAAG,IAAI,CAACqS,OAAO,CAACoJ,gBAAgB,CAAC,CAAC;UACjD,IAAI,CAAChgB,MAAM,CAACkf,IAAI,EAAE;YAEhBlf,MAAM,CAACkf,IAAI,GAAG,CAAC,CAAC;UAClB;QACF;QACAlf,MAAM,CAAC0E,GAAG,CAACya,aAAa,GAAG,IAAI;QAC/B,IAAI,CAAC+B,gBAAgB,CAAC,CAAClhB,MAAM,CAAC0E,GAAG,CAAC,CAAC;MACrC;MAEA,IAAI1E,MAAM,CAACkf,IAAI,EAAE;QACf,IAAInf,IAAI,CAACC,MAAM,IAAID,IAAI,CAACC,MAAM,CAAC/O,GAAG,EAAE;UAClC+O,MAAM,CAACkf,IAAI,CAACjuB,GAAG,GAAG8O,IAAI,CAACC,MAAM,CAAC/O,GAAG;UACjC+O,MAAM,CAACkf,IAAI,CAAC7E,OAAO,GAAGta,IAAI,CAACgf,EAAE;QAC/B;QACA,IAAI,CAACK,gBAAgB,CAACpf,MAAM,CAACkf,IAAI,CAAC;MACpC;MAEA,IAAIlf,MAAM,CAACmG,IAAI,EAAE;QACf,IAAI,CAACgb,gBAAgB,CAACnhB,MAAM,CAACmG,IAAI,CAAC;MACpC;MACA,IAAInG,MAAM,CAACohB,IAAI,EAAE;QACf,IAAI,CAACC,iBAAiB,CAAC,CAACrhB,MAAM,CAACohB,IAAI,CAAC,EAAE,IAAI,CAAC;MAC7C;MAEA,OAAOrhB,IAAI;IACb,CAAC,CAAC;EACN;EASAhM,UAAUA,CAACkQ,GAAG,EAAE/Q,MAAM,EAAE;IACtB,MAAMqR,IAAI,GAAGN,GAAG,GAAG,IAAI,CAACqd,UAAU,CAACrd,GAAG,CAAC,GAAG,IAAI;IAC9C,MAAMsd,EAAE,GAAGhd,IAAI,GACbA,IAAI,CAACtT,GAAG,CAACmD,WAAW,CAAClB,MAAM,CAAC,CAACmB,QAAQ,CAAC,CAAC,GACvC,IAAI,CAACsS,aAAa,CAAC,CAAC,CAACnS,UAAU,CAACtB,MAAM,CAAC,CAACuB,OAAO,CAAC,CAAC;IAEnD,OAAO,IAAI,CAACwsB,OAAO,CAAC;MAClBvc,GAAG,EAAE;QACHH,IAAI,EAAEN,GAAG;QACT5S,IAAI,EAAEkwB;MACR;IACF,CAAC,CAAC;EACJ;EAUAC,MAAMA,CAACvd,GAAG,EAAE5S,IAAI,EAAE;IAChB,OAAO,IAAI,CAAC4vB,OAAO,CAAC;MAClBvc,GAAG,EAAE;QACHH,IAAI,EAAEN,GAAG;QACT5S,IAAI,EAAEA;MACR;IACF,CAAC,CAAC;EACJ;EASAowB,OAAOA,CAACC,IAAI,EAAE;IACZ,IAAI,IAAI,CAAChF,OAAO,IAAK,CAAC,IAAI,CAACA,OAAO,CAACgF,IAAI,IAAI,CAACA,IAAK,EAAE;MACjD,OAAO5kB,OAAO,CAAC0C,OAAO,CAACkiB,IAAI,CAAC;IAC9B;IACA,OAAO,IAAI,CAACT,OAAO,CAAC;MAClB/B,IAAI,EAAE;QACJxC,OAAO,EAAE;UACPgF,IAAI,EAAEA,IAAI,GAAG,IAAI,GAAGzF,gDAAc5hB;QACpC;MACF;IACF,CAAC,CAAC;EACJ;EAUAsnB,WAAWA,CAACxU,MAAM,EAAEyU,IAAI,EAAE;IACxB,IAAI,CAAC,IAAI,CAACzE,SAAS,EAAE;MACnB,OAAOrgB,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,0CAA0C,CAAC,CAAC;IAC9E;IAGAub,MAAM,CAACP,IAAI,CAAC,CAACiV,EAAE,EAAEC,EAAE,KAAK;MACtB,IAAID,EAAE,CAACE,GAAG,GAAGD,EAAE,CAACC,GAAG,EAAE;QACnB,OAAO,IAAI;MACb;MACA,IAAIF,EAAE,CAACE,GAAG,IAAID,EAAE,CAACC,GAAG,EAAE;QACpB,OAAO,CAACD,EAAE,CAACE,EAAE,IAAKH,EAAE,CAACG,EAAE,IAAIF,EAAE,CAACE,EAAG;MACnC;MACA,OAAO,KAAK;IACd,CAAC,CAAC;IAGF,IAAIC,MAAM,GAAG9U,MAAM,CAAC+U,MAAM,CAAC,CAACC,GAAG,EAAE5qB,CAAC,KAAK;MACrC,IAAIA,CAAC,CAACwqB,GAAG,GAAG9F,mDAAiB,EAAE;QAC7B,IAAI,CAAC1kB,CAAC,CAACyqB,EAAE,IAAIzqB,CAAC,CAACyqB,EAAE,GAAG/F,mDAAiB,EAAE;UACrCkG,GAAG,CAACtc,IAAI,CAACtO,CAAC,CAAC;QACb,CAAC,MAAM;UAEL4qB,GAAG,CAACtc,IAAI,CAAC;YACPkc,GAAG,EAAExqB,CAAC,CAACwqB,GAAG;YACVC,EAAE,EAAE,IAAI,CAACpH,OAAO,GAAG;UACrB,CAAC,CAAC;QACJ;MACF;MACA,OAAOuH,GAAG;IACZ,CAAC,EAAE,EAAE,CAAC;IAGN,IAAI/f,MAAM;IACV,IAAI6f,MAAM,CAACtvB,MAAM,GAAG,CAAC,EAAE;MACrByP,MAAM,GAAG,IAAI,CAACwU,OAAO,CAAC+K,WAAW,CAAC,IAAI,CAAClpB,IAAI,EAAEwpB,MAAM,EAAEL,IAAI,CAAC;IAC5D,CAAC,MAAM;MACLxf,MAAM,GAAGtF,OAAO,CAAC0C,OAAO,CAAC;QACvBQ,MAAM,EAAE;UACNoiB,GAAG,EAAE;QACP;MACF,CAAC,CAAC;IACJ;IAEA,OAAOhgB,MAAM,CAACmM,IAAI,CAACxO,IAAI,IAAI;MACzB,IAAIA,IAAI,CAACC,MAAM,CAACoiB,GAAG,GAAG,IAAI,CAACvG,OAAO,EAAE;QAClC,IAAI,CAACA,OAAO,GAAG9b,IAAI,CAACC,MAAM,CAACoiB,GAAG;MAChC;MAEAjV,MAAM,CAACvV,OAAO,CAAEL,CAAC,IAAK;QACpB,IAAIA,CAAC,CAACyqB,EAAE,EAAE;UACR,IAAI,CAACK,iBAAiB,CAAC9qB,CAAC,CAACwqB,GAAG,EAAExqB,CAAC,CAACyqB,EAAE,CAAC;QACrC,CAAC,MAAM;UACL,IAAI,CAACM,YAAY,CAAC/qB,CAAC,CAACwqB,GAAG,CAAC;QAC1B;MACF,CAAC,CAAC;MAEF,IAAI,IAAI,CAACzE,MAAM,EAAE;QAEf,IAAI,CAACA,MAAM,CAAC,CAAC;MACf;MACA,OAAOvd,IAAI;IACb,CAAC,CAAC;EACJ;EASAwiB,cAAcA,CAACC,OAAO,EAAE;IACtB,IAAI,CAAC,IAAI,CAAC5H,OAAO,IAAI,IAAI,CAACA,OAAO,IAAI,CAAC,EAAE;MAEtC,OAAO9d,OAAO,CAAC0C,OAAO,CAAC,CAAC;IAC1B;IACA,OAAO,IAAI,CAACmiB,WAAW,CAAC,CAAC;MACvBI,GAAG,EAAE,CAAC;MACNC,EAAE,EAAE,IAAI,CAACpH,OAAO,GAAG,CAAC;MACpB6H,IAAI,EAAE;IACR,CAAC,CAAC,EAAED,OAAO,CAAC;EACd;EAWAE,eAAeA,CAACC,IAAI,EAAEH,OAAO,EAAE;IAE7BG,IAAI,CAAC/V,IAAI,CAAC,CAAC9W,CAAC,EAAEC,CAAC,KAAKD,CAAC,GAAGC,CAAC,CAAC;IAE1B,IAAIoX,MAAM,GAAGwV,IAAI,CAACT,MAAM,CAAC,CAACC,GAAG,EAAE5X,EAAE,KAAK;MACpC,IAAI4X,GAAG,CAACxvB,MAAM,IAAI,CAAC,EAAE;QAEnBwvB,GAAG,CAACtc,IAAI,CAAC;UACPkc,GAAG,EAAExX;QACP,CAAC,CAAC;MACJ,CAAC,MAAM;QACL,IAAIqY,IAAI,GAAGT,GAAG,CAACA,GAAG,CAACxvB,MAAM,GAAG,CAAC,CAAC;QAC9B,IAAK,CAACiwB,IAAI,CAACZ,EAAE,IAAKzX,EAAE,IAAIqY,IAAI,CAACb,GAAG,GAAG,CAAE,IAAMxX,EAAE,GAAGqY,IAAI,CAACZ,EAAG,EAAE;UAExDG,GAAG,CAACtc,IAAI,CAAC;YACPkc,GAAG,EAAExX;UACP,CAAC,CAAC;QACJ,CAAC,MAAM;UAELqY,IAAI,CAACZ,EAAE,GAAGY,IAAI,CAACZ,EAAE,GAAGpkB,IAAI,CAAC4I,GAAG,CAACoc,IAAI,CAACZ,EAAE,EAAEzX,EAAE,GAAG,CAAC,CAAC,GAAGA,EAAE,GAAG,CAAC;QACxD;MACF;MACA,OAAO4X,GAAG;IACZ,CAAC,EAAE,EAAE,CAAC;IAEN,OAAO,IAAI,CAACR,WAAW,CAACxU,MAAM,EAAEqV,OAAO,CAAC;EAC1C;EAWAK,gBAAgBA,CAAC5d,GAAG,EAAEud,OAAO,EAAE;IAC7B,MAAMG,IAAI,GAAG,CAAC1d,GAAG,CAAC;IAClB,IAAI,CAAC6d,eAAe,CAAC7d,GAAG,EAAE9H,GAAG,IAAIwlB,IAAI,CAAC9c,IAAI,CAAC1I,GAAG,CAAC8H,GAAG,CAAC,CAAC;IAEpD,OAAO,IAAI,CAACyd,eAAe,CAACC,IAAI,EAAEH,OAAO,CAAC;EAC5C;EASAO,QAAQA,CAACnB,IAAI,EAAE;IACb,IAAI,IAAI,CAACve,QAAQ,EAAE;MAEjB,IAAI,CAACmd,KAAK,CAAC,CAAC;MACZ,OAAO1jB,OAAO,CAAC0C,OAAO,CAAC,IAAI,CAAC;IAC9B;IAEA,OAAO,IAAI,CAACoX,OAAO,CAACmM,QAAQ,CAAC,IAAI,CAACtqB,IAAI,EAAEmpB,IAAI,CAAC,CAACrT,IAAI,CAACxO,IAAI,IAAI;MACzD,IAAI,CAACsD,QAAQ,GAAG,IAAI;MACpB,IAAI,CAACkd,SAAS,CAAC,CAAC;MAChB,IAAI,CAACC,KAAK,CAAC,CAAC;MACZ,OAAOzgB,IAAI;IACb,CAAC,CAAC;EACJ;EAQAijB,eAAeA,CAACze,IAAI,EAAE;IACpB,IAAI,CAAC,IAAI,CAAC4Y,SAAS,EAAE;MACnB,OAAOrgB,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,8CAA8C,CAAC,CAAC;IAClF;IAEA,OAAO,IAAI,CAACglB,OAAO,CAACoM,eAAe,CAAC,IAAI,CAACvqB,IAAI,EAAE8L,IAAI,CAAC,CAACgK,IAAI,CAACxO,IAAI,IAAI;MAEhE,OAAO,IAAI,CAAC6c,MAAM,CAACrY,IAAI,CAAC;MAExB,IAAI,IAAI,CAACqZ,aAAa,EAAE;QACtB,IAAI,CAACA,aAAa,CAAC3O,MAAM,CAAC8F,IAAI,CAAC,IAAI,CAAC6H,MAAM,CAAC,CAAC;MAC9C;MACA,OAAO7c,IAAI;IACb,CAAC,CAAC;EACJ;EAQAkjB,IAAIA,CAAC/I,IAAI,EAAEjV,GAAG,EAAE;IACd,IAAI,CAAC,IAAI,CAACkY,SAAS,EAAE;MAEnB;IACF;IAGA,MAAM5Y,IAAI,GAAG,IAAI,CAACqY,MAAM,CAAC,IAAI,CAAChG,OAAO,CAACoJ,gBAAgB,CAAC,CAAC,CAAC;IACzD,IAAI9sB,MAAM,GAAG,KAAK;IAClB,IAAIqR,IAAI,EAAE;MAER,IAAI,CAACA,IAAI,CAAC2V,IAAI,CAAC,IAAI3V,IAAI,CAAC2V,IAAI,CAAC,GAAGjV,GAAG,EAAE;QACnCV,IAAI,CAAC2V,IAAI,CAAC,GAAGjV,GAAG;QAChB/R,MAAM,GAAG,IAAI;MACf;IACF,CAAC,MAAM;MAELA,MAAM,GAAG,CAAC,IAAI,CAACgnB,IAAI,CAAC,GAAG,CAAC,IAAIjV,GAAG;IACjC;IAEA,IAAI/R,MAAM,EAAE;MAEV,IAAI,CAAC0jB,OAAO,CAACqM,IAAI,CAAC,IAAI,CAACxqB,IAAI,EAAEyhB,IAAI,EAAEjV,GAAG,CAAC;MAEvC,IAAI,CAACie,iBAAiB,CAAChJ,IAAI,EAAEjV,GAAG,CAAC;MAEjC,IAAI,IAAI,CAAChU,GAAG,IAAI,IAAI,IAAI,CAAC,IAAI,CAACA,GAAG,CAAC8D,OAAO,CAAC,CAAC,EAAE;QAC3C,MAAMiqB,EAAE,GAAG,IAAI,CAACpI,OAAO,CAACqI,UAAU,CAAC,CAAC;QAEpCD,EAAE,CAACmE,eAAe,CAACjJ,IAAI,EAAE,IAAI,CAAC;MAChC;IACF;EACF;EAQAkJ,QAAQA,CAACne,GAAG,EAAE;IACZ,IAAI,CAACge,IAAI,CAAC,MAAM,EAAEhe,GAAG,CAAC;EACxB;EAOAoe,QAAQA,CAACpe,GAAG,EAAE;IACZA,GAAG,GAAGA,GAAG,IAAI,IAAI,CAAC2V,OAAO;IACzB,IAAI3V,GAAG,GAAG,CAAC,EAAE;MACX,IAAI,CAACge,IAAI,CAAC,MAAM,EAAEhe,GAAG,CAAC;IACxB;EACF;EAKAqe,YAAYA,CAAA,EAAG;IACb,IAAI,IAAI,CAACnG,SAAS,EAAE;MAClB,IAAI,CAACvG,OAAO,CAAC0M,YAAY,CAAC,IAAI,CAAC7qB,IAAI,CAAC;IACtC,CAAC,MAAM;MACL,IAAI,CAACme,OAAO,CAACna,MAAM,CAAC,kDAAkD,CAAC;IACzE;EACF;EAMA8mB,aAAaA,CAACzU,SAAS,EAAE;IACvB,IAAI,IAAI,CAACqO,SAAS,EAAE;MAClB,IAAI,CAACvG,OAAO,CAAC0M,YAAY,CAAC,IAAI,CAAC7qB,IAAI,EAAEqW,SAAS,GAAG,KAAK,GAAG,KAAK,CAAC;IACjE,CAAC,MAAM;MACL,IAAI,CAAC8H,OAAO,CAACna,MAAM,CAAC,kDAAkD,CAAC;IACzE;EACF;EAaAoS,SAASA,CAAC1P,GAAG,EAAE8F,GAAG,EAAEue,OAAO,EAAE;IAC3B,IAAI,CAAC,IAAI,CAACrG,SAAS,IAAI,CAAC,CAAC,SAAS,EAAE,SAAS,CAAC,CAACxrB,QAAQ,CAACwN,GAAG,CAAC,EAAE;MAE5D;IACF;IACA,OAAO,IAAI,CAACyX,OAAO,CAAC/H,SAAS,CAAC,IAAI,CAACpW,IAAI,EAAEwM,GAAG,EAAE9F,GAAG,EAAEqkB,OAAO,CAAC;EAC7D;EAGAN,iBAAiBA,CAAChJ,IAAI,EAAEjV,GAAG,EAAE8Z,EAAE,EAAE;IAC/B,IAAI0E,MAAM;MAAEC,QAAQ,GAAG,KAAK;IAE5Bze,GAAG,GAAGA,GAAG,GAAG,CAAC;IACb,IAAI,CAACA,GAAG,GAAG,IAAI,CAACA,GAAG,GAAG,CAAC;IACvB,IAAI,CAACqB,IAAI,GAAG,IAAI,CAACA,IAAI,GAAG,CAAC;IACzB,IAAI,CAACqd,IAAI,GAAG,IAAI,CAACA,IAAI,GAAG,CAAC;IACzB,QAAQzJ,IAAI;MACV,KAAK,MAAM;QACTuJ,MAAM,GAAG,IAAI,CAACE,IAAI;QAClB,IAAI,CAACA,IAAI,GAAG/lB,IAAI,CAAC4I,GAAG,CAAC,IAAI,CAACmd,IAAI,EAAE1e,GAAG,CAAC;QACpCye,QAAQ,GAAID,MAAM,IAAI,IAAI,CAACE,IAAK;QAChC;MACF,KAAK,MAAM;QACTF,MAAM,GAAG,IAAI,CAACnd,IAAI;QAClB,IAAI,CAACA,IAAI,GAAG1I,IAAI,CAAC4I,GAAG,CAAC,IAAI,CAACF,IAAI,EAAErB,GAAG,CAAC;QACpCye,QAAQ,GAAID,MAAM,IAAI,IAAI,CAACnd,IAAK;QAChC;MACF,KAAK,KAAK;QACRmd,MAAM,GAAG,IAAI,CAACxe,GAAG;QACjB,IAAI,CAACA,GAAG,GAAGrH,IAAI,CAAC4I,GAAG,CAAC,IAAI,CAACvB,GAAG,EAAEA,GAAG,CAAC;QAClC,IAAI,CAAC,IAAI,CAACuX,OAAO,IAAI,IAAI,CAACA,OAAO,GAAGuC,EAAE,EAAE;UACtC,IAAI,CAACvC,OAAO,GAAGuC,EAAE;QACnB;QACA2E,QAAQ,GAAID,MAAM,IAAI,IAAI,CAACxe,GAAI;QAC/B;IACJ;IAGA,IAAI,IAAI,CAAC0e,IAAI,GAAG,IAAI,CAACrd,IAAI,EAAE;MACzB,IAAI,CAACqd,IAAI,GAAG,IAAI,CAACrd,IAAI;MACrBod,QAAQ,GAAG,IAAI;IACjB;IACA,IAAI,IAAI,CAACze,GAAG,GAAG,IAAI,CAAC0e,IAAI,EAAE;MACxB,IAAI,CAAC1e,GAAG,GAAG,IAAI,CAAC0e,IAAI;MACpB,IAAI,CAAC,IAAI,CAACnH,OAAO,IAAI,IAAI,CAACA,OAAO,GAAGuC,EAAE,EAAE;QACtC,IAAI,CAACvC,OAAO,GAAGuC,EAAE;MACnB;MACA2E,QAAQ,GAAG,IAAI;IACjB;IACA,IAAI,CAACnd,MAAM,GAAG,IAAI,CAACtB,GAAG,GAAG,IAAI,CAACqB,IAAI;IAClC,OAAOod,QAAQ;EACjB;EASAE,QAAQA,CAAC3f,GAAG,EAAE;IAEZ,MAAMM,IAAI,GAAG,IAAI,CAACsf,aAAa,CAAC5f,GAAG,CAAC;IACpC,IAAIM,IAAI,EAAE;MACR,OAAOA,IAAI;IACb;EACF;EAOAuf,WAAWA,CAAA,EAAG;IACZ,IAAI,CAAC,IAAI,CAACtJ,SAAS,CAAC,CAAC,EAAE;MACrB,OAAO/kB,SAAS;IAClB;IACA,OAAO,IAAI,CAACmnB,MAAM,CAAC,IAAI,CAACnkB,IAAI,CAAC;EAC/B;EAQAsrB,WAAWA,CAAClsB,QAAQ,EAAEG,OAAO,EAAE;IAC7B,MAAMgsB,EAAE,GAAInsB,QAAQ,IAAI,IAAI,CAAC8lB,SAAU;IACvC,IAAIqG,EAAE,EAAE;MACN,KAAK,IAAIvtB,GAAG,IAAI,IAAI,CAACmmB,MAAM,EAAE;QAC3BoH,EAAE,CAAC/rB,IAAI,CAACD,OAAO,EAAE,IAAI,CAAC4kB,MAAM,CAACnmB,GAAG,CAAC,EAAEA,GAAG,EAAE,IAAI,CAACmmB,MAAM,CAAC;MACtD;IACF;EACF;EAOAzW,IAAIA,CAAA,EAAG;IAEL,OAAO,IAAI,CAACC,KAAK,CAAC0B,KAAK,CAAC,CAAC,CAAC;EAC5B;EAQAwZ,UAAUA,CAACrd,GAAG,EAAE;IACd,OAAO,IAAI,CAAC2Y,MAAM,CAAC3Y,GAAG,CAAC;EACzB;EAUA6e,eAAeA,CAACmB,OAAO,EAAEpsB,QAAQ,EAAEG,OAAO,EAAE;IAC1C,IAAI,CAACH,QAAQ,EAAE;MAEb;IACF;IACA,MAAMqsB,QAAQ,GAAG,IAAI,CAACjH,gBAAgB,CAACgH,OAAO,CAAC;IAC/C,IAAI,CAACC,QAAQ,EAAE;MACb;IACF;IACAA,QAAQ,CAACtsB,OAAO,CAACC,QAAQ,EAAEpC,SAAS,EAAEA,SAAS,EAAEuC,OAAO,CAAC;EAC3D;EAWAmsB,QAAQA,CAACtsB,QAAQ,EAAEusB,OAAO,EAAEC,QAAQ,EAAErsB,OAAO,EAAE;IAC7C,MAAMgsB,EAAE,GAAInsB,QAAQ,IAAI,IAAI,CAACylB,MAAO;IACpC,IAAI0G,EAAE,EAAE;MACN,MAAMlsB,QAAQ,GAAG,OAAOssB,OAAO,IAAI,QAAQ,GAAG,IAAI,CAAClH,SAAS,CAAChlB,IAAI,CAAC;QAChE+M,GAAG,EAAEmf;MACP,CAAC,EAAE,IAAI,CAAC,GAAG3uB,SAAS;MACpB,MAAMsC,SAAS,GAAG,OAAOssB,QAAQ,IAAI,QAAQ,GAAG,IAAI,CAACnH,SAAS,CAAChlB,IAAI,CAAC;QAClE+M,GAAG,EAAEof;MACP,CAAC,EAAE,IAAI,CAAC,GAAG5uB,SAAS;MACpB,IAAIqC,QAAQ,IAAI,CAAC,CAAC,IAAIC,SAAS,IAAI,CAAC,CAAC,EAAE;QAGrC,IAAIusB,IAAI,GAAG,EAAE;QACb,IAAI,CAACpH,SAAS,CAACtlB,OAAO,CAAC,CAACuF,GAAG,EAAEonB,OAAO,EAAEC,OAAO,EAAE9xB,CAAC,KAAK;UACnD,IAAI,IAAI,CAAC+xB,iBAAiB,CAACtnB,GAAG,CAAC,EAAE;YAE/B;UACF;UAEA,MAAMunB,MAAM,GAAG,IAAI,CAACC,gBAAgB,CAACxnB,GAAG,CAAC8H,GAAG,CAAC,IAAI9H,GAAG;UACpD,IAAI,CAACunB,MAAM,CAACE,OAAO,EAAE;YACnBF,MAAM,CAACE,OAAO,GAAGF,MAAM,CAAC3F,EAAE;YAC1B2F,MAAM,CAACG,QAAQ,GAAGH,MAAM,CAACzf,GAAG;YAC5Byf,MAAM,CAAC3F,EAAE,GAAG5hB,GAAG,CAAC4hB,EAAE;YAClB2F,MAAM,CAACzf,GAAG,GAAG9H,GAAG,CAAC8H,GAAG;UACtB;UACAqf,IAAI,CAACze,IAAI,CAAC;YACR5E,IAAI,EAAEyjB,MAAM;YACZjuB,GAAG,EAAE/D;UACP,CAAC,CAAC;QACJ,CAAC,EAAEoF,QAAQ,EAAEC,SAAS,EAAE,CAAC,CAAC,CAAC;QAE3BusB,IAAI,CAAC1sB,OAAO,CAAC,CAACpG,GAAG,EAAEkB,CAAC,KAAK;UACvBsxB,EAAE,CAAC/rB,IAAI,CAACD,OAAO,EAAExG,GAAG,CAACyP,IAAI,EACtBvO,CAAC,GAAG,CAAC,GAAG4xB,IAAI,CAAC5xB,CAAC,GAAG,CAAC,CAAC,CAACuO,IAAI,GAAGxL,SAAS,EACpC/C,CAAC,GAAG4xB,IAAI,CAAC3xB,MAAM,GAAG,CAAC,GAAG2xB,IAAI,CAAC5xB,CAAC,GAAG,CAAC,CAAC,CAACuO,IAAI,GAAGxL,SAAS,EAAGjE,GAAG,CAACiF,GAAG,CAAC;QAClE,CAAC,CAAC;MACJ;IACF;EACF;EAQAquB,WAAWA,CAAC7f,GAAG,EAAE;IACf,MAAMxO,GAAG,GAAG,IAAI,CAACymB,SAAS,CAAChlB,IAAI,CAAC;MAC9B+M,GAAG,EAAEA;IACP,CAAC,CAAC;IACF,IAAIxO,GAAG,IAAI,CAAC,EAAE;MACZ,OAAO,IAAI,CAACymB,SAAS,CAACpmB,KAAK,CAACL,GAAG,CAAC;IAClC;IACA,OAAOhB,SAAS;EAClB;EAOAsvB,aAAaA,CAAA,EAAG;IACd,OAAO,IAAI,CAAC7H,SAAS,CAAClmB,OAAO,CAAC,CAAC;EACjC;EAQA2tB,gBAAgBA,CAAC1f,GAAG,EAAE;IACpB,MAAMif,QAAQ,GAAG,IAAI,CAACjH,gBAAgB,CAAChY,GAAG,CAAC;IAC3C,OAAOif,QAAQ,GAAGA,QAAQ,CAACltB,OAAO,CAAC,CAAC,GAAG,IAAI;EAC7C;EAOAguB,SAASA,CAAA,EAAG;IACV,OAAO,IAAI,CAACpK,OAAO;EACrB;EAOAqK,SAASA,CAAA,EAAG;IACV,OAAO,IAAI,CAACnK,OAAO;EACrB;EAOAoK,UAAUA,CAAA,EAAG;IACX,OAAO,IAAI,CAACrJ,OAAO;EACrB;EAOAsJ,YAAYA,CAAA,EAAG;IACb,OAAO,IAAI,CAACjI,SAAS,CAACvqB,MAAM,CAAC,CAAC;EAChC;EAQAyyB,cAAcA,CAACvtB,QAAQ,EAAEG,OAAO,EAAE;IAChC,IAAI,CAACH,QAAQ,EAAE;MACb,MAAM,IAAIjG,KAAK,CAAC,2BAA2B,CAAC;IAC9C;IACA,IAAI,CAACuyB,QAAQ,CAACtsB,QAAQ,EAAEokB,mDAAiB,EAAExmB,SAAS,EAAEuC,OAAO,CAAC;EAChE;EAWAqtB,eAAeA,CAACnL,IAAI,EAAEjV,GAAG,EAAE;IACzB,IAAIrO,KAAK,GAAG,CAAC;IACb,IAAIqO,GAAG,GAAG,CAAC,EAAE;MACX,MAAM+Z,EAAE,GAAG,IAAI,CAACpI,OAAO,CAACoJ,gBAAgB,CAAC,CAAC;MAC1C,KAAK,IAAIvpB,GAAG,IAAI,IAAI,CAACmmB,MAAM,EAAE;QAC3B,MAAMrY,IAAI,GAAG,IAAI,CAACqY,MAAM,CAACnmB,GAAG,CAAC;QAC7B,IAAI8N,IAAI,CAACA,IAAI,KAAKya,EAAE,IAAIza,IAAI,CAAC2V,IAAI,CAAC,IAAIjV,GAAG,EAAE;UACzCrO,KAAK,EAAE;QACT;MACF;IACF;IACA,OAAOA,KAAK;EACd;EASA0uB,YAAYA,CAACrgB,GAAG,EAAE;IAChB,OAAO,IAAI,CAACogB,eAAe,CAAC,MAAM,EAAEpgB,GAAG,CAAC;EAC1C;EASAsgB,YAAYA,CAACtgB,GAAG,EAAE;IAChB,OAAO,IAAI,CAACogB,eAAe,CAAC,MAAM,EAAEpgB,GAAG,CAAC;EAC1C;EAOAugB,kBAAkBA,CAACC,KAAK,EAAE;IACxB,OAAOA,KAAK,GAAG,IAAI,CAACxgB,GAAG,GAAG,IAAI,CAAC2V,OAAO,GAEnC,IAAI,CAACE,OAAO,GAAG,CAAC,IAAI,CAAC,IAAI,CAACgC,cAAe;EAC9C;EAOA4I,YAAYA,CAACC,KAAK,EAAE;IAClB,OAAO,IAAI,CAAC/K,OAAO,IAAI+K,KAAK;EAC9B;EAQArD,YAAYA,CAACqD,KAAK,EAAE;IAClB,MAAMlvB,GAAG,GAAG,IAAI,CAACymB,SAAS,CAAChlB,IAAI,CAAC;MAC9B+M,GAAG,EAAE0gB;IACP,CAAC,CAAC;IACF,OAAO,IAAI,CAAC1I,gBAAgB,CAAC0I,KAAK,CAAC;IACnC,IAAIlvB,GAAG,IAAI,CAAC,EAAE;MACZ,IAAI,CAACmgB,OAAO,CAACsJ,GAAG,CAAC/a,WAAW,CAAC,IAAI,CAAC1M,IAAI,EAAEktB,KAAK,CAAC;MAC9C,OAAO,IAAI,CAACzI,SAAS,CAAC5lB,KAAK,CAACb,GAAG,CAAC;IAClC;IACA,OAAOhB,SAAS;EAClB;EAUA4sB,iBAAiBA,CAACuD,MAAM,EAAEC,OAAO,EAAE;IAEjC,IAAI,CAACjP,OAAO,CAACsJ,GAAG,CAAC/a,WAAW,CAAC,IAAI,CAAC1M,IAAI,EAAEmtB,MAAM,EAAEC,OAAO,CAAC;IAGxD,KAAK,IAAInzB,CAAC,GAAGkzB,MAAM,EAAElzB,CAAC,GAAGmzB,OAAO,EAAEnzB,CAAC,EAAE,EAAE;MACrC,OAAO,IAAI,CAACuqB,gBAAgB,CAACvqB,CAAC,CAAC;IACjC;IAGA,MAAM+E,KAAK,GAAG,IAAI,CAACylB,SAAS,CAAChlB,IAAI,CAAC;MAChC+M,GAAG,EAAE2gB;IACP,CAAC,EAAE,IAAI,CAAC;IACR,OAAOnuB,KAAK,IAAI,CAAC,GAAG,IAAI,CAACylB,SAAS,CAAC1lB,QAAQ,CAACC,KAAK,EAAE,IAAI,CAACylB,SAAS,CAAChlB,IAAI,CAAC;MACrE+M,GAAG,EAAE4gB;IACP,CAAC,EAAE,IAAI,CAAC,CAAC,GAAG,EAAE;EAChB;EAQAlG,aAAaA,CAACzb,GAAG,EAAE4hB,QAAQ,EAAE;IAC3B,MAAMrvB,GAAG,GAAG,IAAI,CAACymB,SAAS,CAAChlB,IAAI,CAACgM,GAAG,CAAC;IACpC,MAAM6hB,WAAW,GAAG,IAAI,CAAC7I,SAAS,CAACvqB,MAAM,CAAC,CAAC;IAC3C,IAAI,CAAC,IAAI8D,GAAG,IAAIA,GAAG,GAAGsvB,WAAW,EAAE;MAEjC,IAAI,CAAC7I,SAAS,CAAC5lB,KAAK,CAACb,GAAG,CAAC;MACzB,IAAI,CAACmgB,OAAO,CAACsJ,GAAG,CAAC/a,WAAW,CAAC,IAAI,CAAC1M,IAAI,EAAEyL,GAAG,CAACe,GAAG,CAAC;MAEhDf,GAAG,CAACe,GAAG,GAAG6gB,QAAQ;MAClB,IAAI,CAAC5I,SAAS,CAACjmB,GAAG,CAACiN,GAAG,CAAC;MACvB,IAAI,CAAC0S,OAAO,CAACsJ,GAAG,CAACrb,UAAU,CAACX,GAAG,CAAC;IAClC;EACF;EASA8hB,UAAUA,CAACL,KAAK,EAAE;IAChB,MAAMlvB,GAAG,GAAG,IAAI,CAACymB,SAAS,CAAChlB,IAAI,CAAC;MAC9B+M,GAAG,EAAE0gB;IACP,CAAC,CAAC;IACF,IAAIlvB,GAAG,IAAI,CAAC,EAAE;MACZ,MAAM0G,GAAG,GAAG,IAAI,CAAC+f,SAAS,CAACpmB,KAAK,CAACL,GAAG,CAAC;MACrC,MAAM4I,MAAM,GAAG,IAAI,CAAC4mB,SAAS,CAAC9oB,GAAG,CAAC;MAClC,IAAIkC,MAAM,IAAI4c,6DAA2B,IACvC5c,MAAM,IAAI4c,6DAA2B,IACrC5c,MAAM,IAAI4c,4DAA0B,EAAE;QACtC,IAAI,CAACrF,OAAO,CAACsJ,GAAG,CAAC/a,WAAW,CAAC,IAAI,CAAC1M,IAAI,EAAEktB,KAAK,CAAC;QAC9CxoB,GAAG,CAACgjB,UAAU,GAAG,IAAI;QACrB,IAAI,CAACjD,SAAS,CAAC5lB,KAAK,CAACb,GAAG,CAAC;QACzB,IAAI,IAAI,CAAC6mB,MAAM,EAAE;UAEf,IAAI,CAACA,MAAM,CAAC,CAAC;QACf;QACA,OAAO,IAAI;MACb;IACF;IACA,OAAO,KAAK;EACd;EAOAjC,OAAOA,CAAA,EAAG;IACR,OAAOgB,KAAK,CAAC4B,SAAS,CAAC,IAAI,CAACxlB,IAAI,CAAC;EACnC;EAOAkO,aAAaA,CAAA,EAAG;IACd,OAAO,IAAI,CAAC1V,GAAG;EACjB;EAOAoV,aAAaA,CAACpV,GAAG,EAAE;IACjB,OAAO,IAAI,CAACA,GAAG,GAAG,IAAIF,uDAAU,CAACE,GAAG,CAAC;EACvC;EAOAi1B,gBAAgBA,CAAA,EAAG;IACjB,OAAO,IAAI,CAACC,MAAM;EACpB;EAQArF,cAAcA,CAAA,EAAG;IACf,OAAO,IAAI7G,wDAAc,CAAC,IAAI,CAAC;EACjC;EAOAmM,UAAUA,CAAA,EAAG;IACX,OAAO,IAAI,CAAC1J,OAAO,IAAI,CAAC,CAAC,IAAI,CAACA,OAAO,CAACgF,IAAI;EAC5C;EAOA2E,QAAQA,CAAA,EAAG;IACT,OAAOhK,KAAK,CAAC8B,aAAa,CAAC,IAAI,CAAC1lB,IAAI,CAAC;EACvC;EAOA6tB,aAAaA,CAAA,EAAG;IACd,OAAOjK,KAAK,CAACmC,kBAAkB,CAAC,IAAI,CAAC/lB,IAAI,CAAC;EAC5C;EAOA8tB,WAAWA,CAAA,EAAG;IACZ,OAAOlK,KAAK,CAAC+B,gBAAgB,CAAC,IAAI,CAAC3lB,IAAI,CAAC;EAC1C;EAOA+hB,SAASA,CAAA,EAAG;IACV,OAAO6B,KAAK,CAACgC,cAAc,CAAC,IAAI,CAAC5lB,IAAI,CAAC;EACxC;EAOA+tB,UAAUA,CAAA,EAAG;IACX,OAAOnK,KAAK,CAACiC,eAAe,CAAC,IAAI,CAAC7lB,IAAI,CAAC;EACzC;EAWAwtB,SAASA,CAAC9oB,GAAG,EAAEhK,GAAG,EAAE;IAClB,IAAIkM,MAAM,GAAG4c,2DAAyB;IACtC,IAAI,IAAI,CAACrF,OAAO,CAAC6P,IAAI,CAACtpB,GAAG,CAACiI,IAAI,CAAC,EAAE;MAC/B,IAAIjI,GAAG,CAACsiB,QAAQ,EAAE;QAChBpgB,MAAM,GAAG4c,8DAA4B;MACvC,CAAC,MAAM,IAAI9e,GAAG,CAACijB,MAAM,IAAIjjB,GAAG,CAACgjB,UAAU,EAAE;QACvC9gB,MAAM,GAAG4c,4DAA0B;MACrC,CAAC,MAAM,IAAI9e,GAAG,CAACuiB,OAAO,EAAE;QACtBrgB,MAAM,GAAG4c,6DAA2B;MACtC,CAAC,MAAM,IAAI9e,GAAG,CAAC8H,GAAG,IAAIgX,mDAAiB,EAAE;QACvC5c,MAAM,GAAG4c,6DAA2B;MACtC,CAAC,MAAM,IAAI,IAAI,CAACqJ,YAAY,CAACnoB,GAAG,CAAC8H,GAAG,CAAC,GAAG,CAAC,EAAE;QACzC5F,MAAM,GAAG4c,2DAAyB;MACpC,CAAC,MAAM,IAAI,IAAI,CAACsJ,YAAY,CAACpoB,GAAG,CAAC8H,GAAG,CAAC,GAAG,CAAC,EAAE;QACzC5F,MAAM,GAAG4c,+DAA6B;MACxC,CAAC,MAAM,IAAI9e,GAAG,CAAC8H,GAAG,GAAG,CAAC,EAAE;QACtB5F,MAAM,GAAG4c,2DAAyB;MACpC;IACF,CAAC,MAAM;MACL5c,MAAM,GAAG4c,4DAA0B;IACrC;IAEA,IAAI9oB,GAAG,IAAIgK,GAAG,CAAC+H,OAAO,IAAI7F,MAAM,EAAE;MAChClC,GAAG,CAAC+H,OAAO,GAAG7F,MAAM;MACpB,IAAI,CAACuX,OAAO,CAACsJ,GAAG,CAAClb,gBAAgB,CAAC,IAAI,CAACvM,IAAI,EAAE0E,GAAG,CAAC8H,GAAG,EAAE5F,MAAM,CAAC;IAC/D;IAEA,OAAOA,MAAM;EACf;EAGAolB,iBAAiBA,CAACvgB,GAAG,EAAE;IACrB,OAAOA,GAAG,CAACwiB,IAAI,IAAIxiB,GAAG,CAACwiB,IAAI,CAACC,OAAO;EACrC;EAIA/G,gCAAgCA,CAACziB,GAAG,EAAE;IACpC,IAAI,CAAC,IAAI,CAACsnB,iBAAiB,CAACtnB,GAAG,CAAC,EAAE;MAGhC,IAAI,IAAI,CAAC8f,gBAAgB,CAAC9f,GAAG,CAAC8H,GAAG,CAAC,EAAE;QAElC,IAAI,CAACgY,gBAAgB,CAAC9f,GAAG,CAAC8H,GAAG,CAAC,CAAC7M,MAAM,CAAC+C,OAAO,IAAIA,OAAO,CAACiK,IAAI,IAAIjI,GAAG,CAACiI,IAAI,CAAC;QAC1E,IAAI,IAAI,CAAC6X,gBAAgB,CAAC9f,GAAG,CAAC8H,GAAG,CAAC,CAAC5M,OAAO,CAAC,CAAC,EAAE;UAC5C,OAAO,IAAI,CAAC4kB,gBAAgB,CAAC9f,GAAG,CAAC8H,GAAG,CAAC;QACvC;MACF;MACA;IACF;IAEA,MAAM2hB,SAAS,GAAGC,QAAQ,CAAC1pB,GAAG,CAACupB,IAAI,CAACC,OAAO,CAACpzB,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1D,IAAIqzB,SAAS,GAAGzpB,GAAG,CAAC8H,GAAG,EAAE;MAEvB;IACF;IACA,MAAM6hB,SAAS,GAAG,IAAI,CAAChC,WAAW,CAAC8B,SAAS,CAAC;IAC7C,IAAIE,SAAS,IAAIA,SAAS,CAAC1hB,IAAI,IAAIjI,GAAG,CAACiI,IAAI,EAAE;MAE3C;IACF;IACA,MAAM8e,QAAQ,GAAG,IAAI,CAACjH,gBAAgB,CAAC2J,SAAS,CAAC,IAAI,IAAIrxB,mDAAO,CAAC,CAACO,CAAC,EAAEC,CAAC,KAAK;MACzE,OAAOD,CAAC,CAACmP,GAAG,GAAGlP,CAAC,CAACkP,GAAG;IACtB,CAAC,EAAE,IAAI,CAAC;IACRif,QAAQ,CAACjtB,GAAG,CAACkG,GAAG,CAAC;IACjB,IAAI,CAAC8f,gBAAgB,CAAC2J,SAAS,CAAC,GAAG1C,QAAQ;EAC7C;EAGArE,UAAUA,CAAC5e,IAAI,EAAE;IACf,IAAIA,IAAI,CAAC+K,OAAO,EAAE;MAChB,IAAI,CAAC,IAAI,CAACwQ,OAAO,IAAI,IAAI,CAACA,OAAO,GAAGvb,IAAI,CAAC8d,EAAE,EAAE;QAC3C,IAAI,CAACvC,OAAO,GAAGvb,IAAI,CAAC8d,EAAE;QACtB,IAAI,CAACnI,OAAO,CAACsJ,GAAG,CAACpd,QAAQ,CAAC,IAAI,CAAC;MACjC;IACF;IAEA,IAAI7B,IAAI,CAACgE,GAAG,GAAG,IAAI,CAAC2V,OAAO,EAAE;MAC3B,IAAI,CAACA,OAAO,GAAG3Z,IAAI,CAACgE,GAAG;MACvB,IAAI,CAACghB,SAAS,CAAChlB,IAAI,EAAE,IAAI,CAAC;MAE1BvD,YAAY,CAAC,IAAI,CAACqf,sBAAsB,CAAC;MACzC,IAAI,CAACA,sBAAsB,GAAG/e,UAAU,CAACxC,CAAC,IAAI;QAC5C,IAAI,CAACuhB,sBAAsB,GAAG,IAAI;QAClC,IAAI,CAACqG,QAAQ,CAAC,IAAI,CAACxI,OAAO,CAAC;MAC7B,CAAC,EAAEqB,oDAAkB,CAAC;IACxB;IAEA,IAAIhb,IAAI,CAACgE,GAAG,GAAG,IAAI,CAAC6V,OAAO,IAAI,IAAI,CAACA,OAAO,IAAI,CAAC,EAAE;MAChD,IAAI,CAACA,OAAO,GAAG7Z,IAAI,CAACgE,GAAG;IACzB;IAEA,MAAM8hB,QAAQ,GAAK,CAAC,IAAI,CAACT,aAAa,CAAC,CAAC,IAAI,CAACrlB,IAAI,CAACmE,IAAI,IAAK,IAAI,CAACwR,OAAO,CAAC6P,IAAI,CAACxlB,IAAI,CAACmE,IAAI,CAAE;IAExF,IAAInE,IAAI,CAACylB,IAAI,IAAIzlB,IAAI,CAACylB,IAAI,CAACM,MAAM,IAAI/lB,IAAI,CAACylB,IAAI,CAAChc,IAAI,IAAIC,gEAAqB,CAAC,CAAC,IAAI1J,IAAI,CAAC+K,OAAO,EAAE;MAE9F/K,IAAI,CAAC+K,OAAO,GAAGrB,iEAAsB,CAAC1J,IAAI,CAAC+K,OAAO,EAAE;QAClDV,KAAK,EAAErK,IAAI,CAACylB,IAAI,CAACM,MAAM;QACvBpc,QAAQ,EAAE3J,IAAI,CAACylB,IAAI,CAAC,iBAAiB,CAAC;QACtCO,QAAQ,EAAE,CAACF;MACb,CAAC,CAAC;IACJ;IAEA,IAAI,CAAC9lB,IAAI,CAACke,aAAa,EAAE;MACvB,IAAI,CAACjC,SAAS,CAACjmB,GAAG,CAACgK,IAAI,CAAC;MACxB,IAAI,CAAC2V,OAAO,CAACsJ,GAAG,CAACrb,UAAU,CAAC5D,IAAI,CAAC;MACjC,IAAI,CAAC2e,gCAAgC,CAAC3e,IAAI,CAAC;IAC7C;IAEA,IAAI,IAAI,CAACqc,MAAM,EAAE;MACf,IAAI,CAACA,MAAM,CAACrc,IAAI,CAAC;IACnB;IAGA,MAAMiZ,IAAI,GAAG6M,QAAQ,GAAG,MAAM,GAAG,KAAK;IACtC,IAAI,CAAC7D,iBAAiB,CAAChJ,IAAI,EAAEjZ,IAAI,CAACgE,GAAG,EAAEhE,IAAI,CAAC8d,EAAE,CAAC;IAE/C,IAAI,CAACgI,QAAQ,IAAI9lB,IAAI,CAACmE,IAAI,EAAE;MAE1B,IAAI,CAAC8hB,UAAU,CAAC;QACdhN,IAAI,EAAE,MAAM;QACZ9U,IAAI,EAAEnE,IAAI,CAACmE,IAAI;QACfH,GAAG,EAAEhE,IAAI,CAACgE,GAAG;QACbka,aAAa,EAAE;MACjB,CAAC,CAAC;IACJ;IAGA,IAAI,CAACvI,OAAO,CAACqI,UAAU,CAAC,CAAC,CAACkE,eAAe,CAACjJ,IAAI,EAAE,IAAI,CAAC;EACvD;EAGAiN,UAAUA,CAACC,IAAI,EAAE;IACf,IAAIA,IAAI,CAAClI,IAAI,EAAE;MACb,IAAI,CAACE,gBAAgB,CAACgI,IAAI,CAAClI,IAAI,CAAC;IAClC;IACA,IAAIkI,IAAI,CAAC1iB,GAAG,IAAI0iB,IAAI,CAAC1iB,GAAG,CAAC/R,MAAM,GAAG,CAAC,EAAE;MACnC,IAAI,CAACuuB,gBAAgB,CAACkG,IAAI,CAAC1iB,GAAG,CAAC;IACjC;IACA,IAAI0iB,IAAI,CAAChF,GAAG,EAAE;MACZ,IAAI,CAACiF,mBAAmB,CAACD,IAAI,CAAChF,GAAG,CAACkF,KAAK,EAAEF,IAAI,CAAChF,GAAG,CAACmF,MAAM,CAAC;IAC3D;IACA,IAAIH,IAAI,CAACjhB,IAAI,EAAE;MACb,IAAI,CAACgb,gBAAgB,CAACiG,IAAI,CAACjhB,IAAI,CAAC;IAClC;IACA,IAAIihB,IAAI,CAAChG,IAAI,EAAE;MACb,IAAI,CAACC,iBAAiB,CAAC+F,IAAI,CAAChG,IAAI,CAAC;IACnC;IACA,IAAI,IAAI,CAAC7D,MAAM,EAAE;MACf,IAAI,CAACA,MAAM,CAAC6J,IAAI,CAAC;IACnB;EACF;EAEAI,UAAUA,CAACC,IAAI,EAAE;IACf,IAAIljB,IAAI,EAAEN,GAAG;IACb,QAAQwjB,IAAI,CAACvN,IAAI;MACf,KAAK,KAAK;QAER,IAAI,CAACmN,mBAAmB,CAACI,IAAI,CAACH,KAAK,EAAEG,IAAI,CAACF,MAAM,CAAC;QACjD;MACF,KAAK,IAAI;MACT,KAAK,KAAK;QAERhjB,IAAI,GAAG,IAAI,CAACqY,MAAM,CAAC6K,IAAI,CAAC1jB,GAAG,CAAC;QAC5B,IAAIQ,IAAI,EAAE;UACRA,IAAI,CAACmjB,MAAM,GAAGD,IAAI,CAACvN,IAAI,IAAI,IAAI;QACjC,CAAC,MAAM;UACL,IAAI,CAACtD,OAAO,CAACna,MAAM,CAAC,8CAA8C,EAAE,IAAI,CAAChE,IAAI,EAAEgvB,IAAI,CAAC1jB,GAAG,CAAC;QAC1F;QACA;MACF,KAAK,MAAM;QAET,IAAI,CAACwc,SAAS,CAAC,CAAC;QAChB;MACF,KAAK,KAAK;QAIR,IAAIkH,IAAI,CAAC1jB,GAAG,IAAI,CAAC,IAAI,CAAC6S,OAAO,CAAC+Q,aAAa,CAACF,IAAI,CAAC1jB,GAAG,CAAC,EAAE;UACrD,IAAI,CAAC4c,OAAO,CAAC,IAAI,CAACG,cAAc,CAAC,CAAC,CAACxF,UAAU,CAAC7lB,SAAS,EAAEgyB,IAAI,CAAC1jB,GAAG,CAAC,CAACgY,KAAK,CAAC,CAAC,CAAC;QAC7E;QACA;MACF,KAAK,KAAK;QACR9X,GAAG,GAAGwjB,IAAI,CAAC1jB,GAAG,IAAI,IAAI,CAAC6S,OAAO,CAACoJ,gBAAgB,CAAC,CAAC;QACjDzb,IAAI,GAAG,IAAI,CAACqY,MAAM,CAAC3Y,GAAG,CAAC;QACvB,IAAI,CAACM,IAAI,EAAE;UAET,MAAMtT,GAAG,GAAG,IAAIF,uDAAU,CAAC,CAAC,CAAC6D,SAAS,CAAC6yB,IAAI,CAACG,IAAI,CAAC;UACjD,IAAI32B,GAAG,IAAIA,GAAG,CAACI,IAAI,IAAIN,uDAAU,CAACgB,KAAK,EAAE;YACvCwS,IAAI,GAAG,IAAI,CAACsf,aAAa,CAAC5f,GAAG,CAAC;YAC9B,IAAI,CAACM,IAAI,EAAE;cACTA,IAAI,GAAG;gBACLA,IAAI,EAAEN,GAAG;gBACThT,GAAG,EAAEA;cACP,CAAC;cACD,IAAI,CAAC0vB,OAAO,CAAC,IAAI,CAACG,cAAc,CAAC,CAAC,CAACxF,UAAU,CAAC7lB,SAAS,EAAEwO,GAAG,CAAC,CAAC8X,KAAK,CAAC,CAAC,CAAC;YACxE,CAAC,MAAM;cACLxX,IAAI,CAACtT,GAAG,GAAGA,GAAG;YAChB;YACAsT,IAAI,CAAC8V,OAAO,GAAG,IAAIoC,IAAI,CAAC,CAAC;YACzB,IAAI,CAACyE,gBAAgB,CAAC,CAAC3c,IAAI,CAAC,CAAC;UAC/B;QACF,CAAC,MAAM;UAELA,IAAI,CAACtT,GAAG,CAAC2D,SAAS,CAAC6yB,IAAI,CAACG,IAAI,CAAC;UAE7B,IAAI,CAAC1G,gBAAgB,CAAC,CAAC;YACrB3c,IAAI,EAAEN,GAAG;YACToW,OAAO,EAAE,IAAIoC,IAAI,CAAC,CAAC;YACnBxrB,GAAG,EAAEsT,IAAI,CAACtT;UACZ,CAAC,CAAC,CAAC;QACL;QACA;MACF;QACE,IAAI,CAAC2lB,OAAO,CAACna,MAAM,CAAC,+BAA+B,EAAEgrB,IAAI,CAACvN,IAAI,CAAC;IACnE;IAEA,IAAI,IAAI,CAACsD,MAAM,EAAE;MACf,IAAI,CAACA,MAAM,CAACiK,IAAI,CAAC;IACnB;EACF;EAEAP,UAAUA,CAACW,IAAI,EAAE;IACf,QAAQA,IAAI,CAAC3N,IAAI;MACf,KAAK,MAAM;MACX,KAAK,MAAM;QACT,MAAM3V,IAAI,GAAG,IAAI,CAACqY,MAAM,CAACiL,IAAI,CAACziB,IAAI,CAAC;QACnC,IAAIb,IAAI,EAAE;UACRA,IAAI,CAACsjB,IAAI,CAAC3N,IAAI,CAAC,GAAG2N,IAAI,CAAC5iB,GAAG;UAC1B,IAAIV,IAAI,CAACof,IAAI,GAAGpf,IAAI,CAAC+B,IAAI,EAAE;YACzB/B,IAAI,CAACof,IAAI,GAAGpf,IAAI,CAAC+B,IAAI;UACvB;QACF;QACA,MAAMnJ,GAAG,GAAG,IAAI,CAAC4nB,aAAa,CAAC,CAAC;QAChC,IAAI5nB,GAAG,EAAE;UACP,IAAI,CAAC8oB,SAAS,CAAC9oB,GAAG,EAAE,IAAI,CAAC;QAC3B;QAGA,IAAI,IAAI,CAACyZ,OAAO,CAAC6P,IAAI,CAACoB,IAAI,CAACziB,IAAI,CAAC,IAAI,CAACyiB,IAAI,CAAC1I,aAAa,EAAE;UACvD,IAAI,CAAC+D,iBAAiB,CAAC2E,IAAI,CAAC3N,IAAI,EAAE2N,IAAI,CAAC5iB,GAAG,CAAC;QAC7C;QAGA,IAAI,CAAC2R,OAAO,CAACqI,UAAU,CAAC,CAAC,CAACkE,eAAe,CAAC0E,IAAI,CAAC3N,IAAI,EAAE,IAAI,CAAC;QAC1D;MACF,KAAK,IAAI;MACT,KAAK,KAAK;MACV,KAAK,KAAK;QAER;MACF,KAAK,MAAM;QAET;MACF;QACE,IAAI,CAACtD,OAAO,CAACna,MAAM,CAAC,2BAA2B,EAAEorB,IAAI,CAAC3N,IAAI,CAAC;IAC/D;IAEA,IAAI,IAAI,CAACuD,MAAM,EAAE;MACf,IAAI,CAACA,MAAM,CAACoK,IAAI,CAAC;IACnB;EACF;EAGAzI,gBAAgBA,CAACF,IAAI,EAAE;IACrB,IAAI,IAAI,CAAC1E,SAAS,CAAC,CAAC,EAAE;MAGpB,OAAO0E,IAAI,CAACiH,MAAM;MAGlB,IAAI,CAACvP,OAAO,CAACsJ,GAAG,CAAClc,OAAO,CAAC,IAAI,CAACvL,IAAI,EAAEymB,IAAI,CAAC/a,MAAM,CAAC;IAClD;IAGA+X,mDAAQ,CAAC,IAAI,EAAEgD,IAAI,CAAC;IAEpB,IAAI,CAACtI,OAAO,CAACsJ,GAAG,CAACpd,QAAQ,CAAC,IAAI,CAAC;IAG/B,IAAI,IAAI,CAACrK,IAAI,KAAKwjB,gDAAc,IAAI,CAACiD,IAAI,CAACC,aAAa,EAAE;MACvD,MAAMH,EAAE,GAAG,IAAI,CAACpI,OAAO,CAACqI,UAAU,CAAC,CAAC;MACpC,IAAID,EAAE,CAACrB,SAAS,EAAE;QAChBqB,EAAE,CAACrB,SAAS,CAAC,IAAI,CAAC;MACpB;MACA,IAAIqB,EAAE,CAACpB,aAAa,EAAE;QACpBoB,EAAE,CAACpB,aAAa,CAAC,CAAC,IAAI,CAACnlB,IAAI,CAAC,EAAE,CAAC,CAAC;MAClC;IACF;IAEA,IAAI,IAAI,CAACilB,UAAU,EAAE;MACnB,IAAI,CAACA,UAAU,CAAC,IAAI,CAAC;IACvB;EACF;EAGAwD,gBAAgBA,CAAC4G,IAAI,EAAE;IACrB,KAAK,IAAIrxB,GAAG,IAAIqxB,IAAI,EAAE;MACpB,MAAMpjB,GAAG,GAAGojB,IAAI,CAACrxB,GAAG,CAAC;MAGrBiO,GAAG,CAACgjB,MAAM,GAAG,CAAC,CAAChjB,GAAG,CAACgjB,MAAM;MAEzB,IAAI,CAACjN,eAAe,GAAG,IAAIgC,IAAI,CAAC7e,IAAI,CAAC4I,GAAG,CAAC,IAAI,CAACiU,eAAe,EAAE/V,GAAG,CAAC2V,OAAO,CAAC,CAAC;MAE5E,IAAI9V,IAAI,GAAG,IAAI;MACf,IAAI,CAACG,GAAG,CAACtB,OAAO,EAAE;QAGhB,IAAI,IAAI,CAACwT,OAAO,CAAC6P,IAAI,CAAC/hB,GAAG,CAACH,IAAI,CAAC,IAAIG,GAAG,CAACzT,GAAG,EAAE;UAC1C,IAAI,CAACmuB,gBAAgB,CAAC;YACpB/E,OAAO,EAAE3V,GAAG,CAAC2V,OAAO;YACpBmC,OAAO,EAAE9X,GAAG,CAAC8X,OAAO;YACpBvrB,GAAG,EAAEyT,GAAG,CAACzT;UACX,CAAC,CAAC;QACJ;QACAsT,IAAI,GAAG,IAAI,CAACwjB,iBAAiB,CAACrjB,GAAG,CAACH,IAAI,EAAEG,GAAG,CAAC;MAC9C,CAAC,MAAM;QAEL,OAAO,IAAI,CAACkY,MAAM,CAAClY,GAAG,CAACH,IAAI,CAAC;QAC5BA,IAAI,GAAGG,GAAG;MACZ;MAEA,IAAI,IAAI,CAACiZ,SAAS,EAAE;QAClB,IAAI,CAACA,SAAS,CAACpZ,IAAI,CAAC;MACtB;IACF;IAEA,IAAI,IAAI,CAACqZ,aAAa,EAAE;MACtB,IAAI,CAACA,aAAa,CAAC3O,MAAM,CAAC8F,IAAI,CAAC,IAAI,CAAC6H,MAAM,CAAC,CAAC;IAC9C;EACF;EAEAuE,gBAAgBA,CAAChb,IAAI,EAAE;IACrB,IAAIA,IAAI,CAACxT,MAAM,IAAI,CAAC,IAAIwT,IAAI,CAAC,CAAC,CAAC,IAAI8V,gDAAc,EAAE;MACjD9V,IAAI,GAAG,EAAE;IACX;IACA,IAAI,CAACC,KAAK,GAAGD,IAAI;IACjB,IAAI,IAAI,CAAC0X,aAAa,EAAE;MACtB,IAAI,CAACA,aAAa,CAAC1X,IAAI,CAAC;IAC1B;EACF;EAEAkb,iBAAiBA,CAAC2G,KAAK,EAAE,CAAC;EAE1BX,mBAAmBA,CAACC,KAAK,EAAEC,MAAM,EAAE;IACjC,IAAI,CAAC1L,OAAO,GAAGje,IAAI,CAAC4I,GAAG,CAAC8gB,KAAK,EAAE,IAAI,CAACzL,OAAO,CAAC;IAC5C,IAAI,CAACyL,KAAK,GAAG1pB,IAAI,CAAC4I,GAAG,CAAC8gB,KAAK,EAAE,IAAI,CAACA,KAAK,CAAC;IACxC,MAAMjlB,KAAK,GAAG,IAAI;IAClB,IAAIzL,KAAK,GAAG,CAAC;IACb,IAAIQ,KAAK,CAACC,OAAO,CAACkwB,MAAM,CAAC,EAAE;MACzBA,MAAM,CAAC3vB,OAAO,CAAC,UAAS0N,KAAK,EAAE;QAC7B,IAAI,CAACA,KAAK,CAAC0c,EAAE,EAAE;UACbprB,KAAK,EAAE;UACPyL,KAAK,CAACigB,YAAY,CAAChd,KAAK,CAACyc,GAAG,CAAC;QAC/B,CAAC,MAAM;UACL,KAAK,IAAIrvB,CAAC,GAAG4S,KAAK,CAACyc,GAAG,EAAErvB,CAAC,GAAG4S,KAAK,CAAC0c,EAAE,EAAEtvB,CAAC,EAAE,EAAE;YACzCkE,KAAK,EAAE;YACPyL,KAAK,CAACigB,YAAY,CAAC5vB,CAAC,CAAC;UACvB;QACF;MACF,CAAC,CAAC;IACJ;IAEA,IAAIkE,KAAK,GAAG,CAAC,EAAE;MAGb,IAAI,IAAI,CAAC0mB,MAAM,EAAE;QACf,IAAI,CAACA,MAAM,CAAC,CAAC;MACf;IACF;EACF;EAEA2K,oBAAoBA,CAACrxB,KAAK,EAAE;IAE1B,IAAI,IAAI,CAAConB,qBAAqB,EAAE;MAC9B,IAAI,CAACA,qBAAqB,CAACpnB,KAAK,CAAC;IACnC;EACF;EAEA2pB,SAASA,CAAA,EAAG;IACV,IAAI,CAACpD,SAAS,GAAG,KAAK;EACxB;EAEAqD,KAAKA,CAAA,EAAG;IACN,IAAI,CAACtD,SAAS,CAACvlB,KAAK,CAAC,CAAC;IACtB,IAAI,CAACif,OAAO,CAACsJ,GAAG,CAAC/a,WAAW,CAAC,IAAI,CAAC1M,IAAI,CAAC;IACvC,IAAI,CAACmkB,MAAM,GAAG,CAAC,CAAC;IAChB,IAAI,CAAC3rB,GAAG,GAAG,IAAIF,uDAAU,CAAC,IAAI,CAAC;IAC/B,IAAI,CAAC2rB,OAAO,GAAG,IAAI;IACnB,IAAI,CAACvY,MAAM,GAAG,IAAI;IAClB,IAAI,CAACwY,OAAO,GAAG,IAAI;IACnB,IAAI,CAAC/B,OAAO,GAAG,CAAC;IAChB,IAAI,CAACE,OAAO,GAAG,CAAC;IAChB,IAAI,CAACqC,SAAS,GAAG,KAAK;IAEtB,MAAM6B,EAAE,GAAG,IAAI,CAACpI,OAAO,CAACqI,UAAU,CAAC,CAAC;IACpC,IAAID,EAAE,EAAE;MACNA,EAAE,CAACwI,UAAU,CAAC;QACZrI,aAAa,EAAE,IAAI;QACnBjF,IAAI,EAAE,MAAM;QACZ7X,KAAK,EAAE4Z,gDAAc;QACrBlY,GAAG,EAAE,IAAI,CAACtL;MACZ,CAAC,CAAC;IACJ;IACA,IAAI,IAAI,CAACslB,aAAa,EAAE;MACtB,IAAI,CAACA,aAAa,CAAC,CAAC;IACtB;EACF;EAGAgK,iBAAiBA,CAAC9jB,GAAG,EAAEikB,GAAG,EAAE;IAG1B,IAAIC,MAAM,GAAG,IAAI,CAACtE,aAAa,CAAC5f,GAAG,CAAC;IACpCkkB,MAAM,GAAGjM,mDAAQ,CAACiM,MAAM,IAAI,CAAC,CAAC,EAAED,GAAG,CAAC;IAEpC,IAAI,CAACE,aAAa,CAACnkB,GAAG,EAAEkkB,MAAM,CAAC;IAE/B,OAAOhM,uDAAY,CAAC,IAAI,CAACS,MAAM,EAAE3Y,GAAG,EAAEkkB,MAAM,CAAC;EAC/C;EAEApI,eAAeA,CAAA,EAAG;IAChB,OAAO,IAAI,CAAClD,YAAY,EAAE;EAC5B;EAGAkE,aAAaA,CAACxf,EAAE,EAAEvB,MAAM,EAAE;IACxB,MAAM;MACJvI,KAAK;MACLC,MAAM;MACN+N;IACF,CAAC,GAAGzF,MAAM,IAAI,CAAC,CAAC;IAChB,OAAOuB,EAAE,CAACgE,YAAY,CAAC,IAAI,CAAC9M,IAAI,EAAE;MAC9BhB,KAAK,EAAEA,KAAK;MACZC,MAAM,EAAEA,MAAM;MACd+N,KAAK,EAAEA,KAAK,IAAIwW,6DAA2B7hB;IAC7C,CAAC,CAAC,CACDmU,IAAI,CAAC+V,IAAI,IAAI;MACZA,IAAI,CAAC1sB,OAAO,CAAEqJ,IAAI,IAAK;QACrB,IAAIA,IAAI,CAACgE,GAAG,GAAG,IAAI,CAAC2V,OAAO,EAAE;UAC3B,IAAI,CAACA,OAAO,GAAG3Z,IAAI,CAACgE,GAAG;QACzB;QACA,IAAIhE,IAAI,CAACgE,GAAG,GAAG,IAAI,CAAC6V,OAAO,IAAI,IAAI,CAACA,OAAO,IAAI,CAAC,EAAE;UAChD,IAAI,CAACA,OAAO,GAAG7Z,IAAI,CAACgE,GAAG;QACzB;QACA,IAAI,CAACiY,SAAS,CAACjmB,GAAG,CAACgK,IAAI,CAAC;QACxB,IAAI,CAAC2e,gCAAgC,CAAC3e,IAAI,CAAC;MAC7C,CAAC,CAAC;MACF,OAAOqjB,IAAI,CAAC3xB,MAAM;IACpB,CAAC,CAAC;EACN;EAEA01B,eAAeA,CAACpjB,GAAG,EAAEuF,GAAG,EAAE;IACxB,IAAI,CAACgS,OAAO,GAAG,IAAIC,IAAI,CAAC,CAAC;IACzB,IAAI,CAACxX,GAAG,GAAGA,GAAG,GAAG,CAAC;IAElB,IAAI,CAACuF,GAAG,IAAI,IAAI,CAACoM,OAAO,CAAC6P,IAAI,CAACjc,GAAG,CAAC,EAAE;MAClC,IAAI,CAAClE,IAAI,GAAG,IAAI,CAACA,IAAI,GAAG1I,IAAI,CAAC4I,GAAG,CAAC,IAAI,CAACF,IAAI,EAAE,IAAI,CAACrB,GAAG,CAAC,GAAG,IAAI,CAACA,GAAG;MAChE,IAAI,CAAC0e,IAAI,GAAG,IAAI,CAACA,IAAI,GAAG/lB,IAAI,CAAC4I,GAAG,CAAC,IAAI,CAACF,IAAI,EAAE,IAAI,CAACqd,IAAI,CAAC,GAAG,IAAI,CAACrd,IAAI;IACpE;IACA,IAAI,CAACC,MAAM,GAAG,IAAI,CAACtB,GAAG,IAAI,IAAI,CAACqB,IAAI,GAAG,CAAC,CAAC;IACxC,IAAI,CAACsQ,OAAO,CAACsJ,GAAG,CAACpd,QAAQ,CAAC,IAAI,CAAC;EACjC;AACF;AAUO,MAAMwlB,OAAO,SAASjM,KAAK,CAAC;EACjCkM,eAAe;EAEfv3B,WAAWA,CAACsrB,SAAS,EAAE;IACrB,KAAK,CAACL,gDAAc,EAAEK,SAAS,CAAC;IAGhC,IAAIA,SAAS,EAAE;MACb,IAAI,CAACiM,eAAe,GAAGjM,SAAS,CAACiM,eAAe;IAClD;EACF;EAGAnJ,gBAAgBA,CAACF,IAAI,EAAE;IAErB,MAAMsJ,OAAO,GAAItJ,IAAI,CAACjuB,GAAG,IAAI,CAACiuB,IAAI,CAACjuB,GAAG,CAAC6D,WAAW,CAAC,CAAC,IAAM,IAAI,CAAC7D,GAAG,IAAI,IAAI,CAACA,GAAG,CAAC6D,WAAW,CAAC,CAAE;IAG7FonB,mDAAQ,CAAC,IAAI,EAAEgD,IAAI,CAAC;IACpB,IAAI,CAACtI,OAAO,CAACsJ,GAAG,CAACpd,QAAQ,CAAC,IAAI,CAAC;IAE/B,IAAI,CAACilB,iBAAiB,CAAC,IAAI,CAACnR,OAAO,CAAC6R,MAAM,EAAEvJ,IAAI,CAAC;IAGjD,IAAIsJ,OAAO,EAAE;MACX,IAAI,CAAC5R,OAAO,CAAC/S,SAAS,CAAE6kB,IAAI,IAAK;QAC/B,IAAIA,IAAI,CAAChB,MAAM,EAAE;UACfgB,IAAI,CAAChB,MAAM,GAAG,KAAK;UACnBgB,IAAI,CAACC,IAAI,GAAG1Z,MAAM,CAACC,MAAM,CAACwZ,IAAI,CAACC,IAAI,IAAI,CAAC,CAAC,EAAE;YACzCC,IAAI,EAAE,IAAInM,IAAI,CAAC;UACjB,CAAC,CAAC;UACF,IAAI,CAAC0G,eAAe,CAAC,KAAK,EAAEuF,IAAI,CAAC;QACnC;MACF,CAAC,CAAC;IACJ;IAEA,IAAI,IAAI,CAAChL,UAAU,EAAE;MACnB,IAAI,CAACA,UAAU,CAAC,IAAI,CAAC;IACvB;EACF;EAGAwD,gBAAgBA,CAAC4G,IAAI,EAAE;IACrB,IAAIe,WAAW,GAAG,CAAC;IACnBf,IAAI,CAAClwB,OAAO,CAAE8M,GAAG,IAAK;MACpB,MAAMD,SAAS,GAAGC,GAAG,CAACrC,KAAK;MAE3B,IAAIoC,SAAS,IAAIwX,iDAAe,IAAIxX,SAAS,IAAIwX,gDAAc,EAAE;QAC/D;MACF;MACAvX,GAAG,CAACgjB,MAAM,GAAG,CAAC,CAAChjB,GAAG,CAACgjB,MAAM;MAEzB,IAAIgB,IAAI,GAAG,IAAI;MACf,IAAIhkB,GAAG,CAACtB,OAAO,EAAE;QACfslB,IAAI,GAAGhkB,GAAG;QACV,IAAI,CAACkS,OAAO,CAACkS,aAAa,CAACrkB,SAAS,CAAC;QACrC,IAAI,CAACmS,OAAO,CAACsJ,GAAG,CAAC5c,QAAQ,CAACmB,SAAS,CAAC;MACtC,CAAC,MAAM;QAEL,IAAI,OAAOC,GAAG,CAACO,GAAG,IAAI,WAAW,EAAE;UACjCP,GAAG,CAACO,GAAG,GAAGP,GAAG,CAACO,GAAG,GAAG,CAAC;UACrBP,GAAG,CAACif,IAAI,GAAGjf,GAAG,CAACif,IAAI,GAAG,CAAC;UACvBjf,GAAG,CAAC4B,IAAI,GAAG5B,GAAG,CAAC4B,IAAI,GAAG,CAAC;UACvB5B,GAAG,CAAC6B,MAAM,GAAG7B,GAAG,CAACO,GAAG,GAAGP,GAAG,CAAC4B,IAAI;QACjC;QAEA,MAAMjE,KAAK,GAAG,IAAI,CAACuU,OAAO,CAACmS,QAAQ,CAACtkB,SAAS,CAAC;QAC9C,IAAIpC,KAAK,CAAC+a,IAAI,EAAE;UACd,OAAO/a,KAAK,CAAC+a,IAAI;QACnB;QAEAsL,IAAI,GAAGxM,mDAAQ,CAAC7Z,KAAK,EAAEqC,GAAG,CAAC;QAC3B,IAAI,CAACkS,OAAO,CAACsJ,GAAG,CAACpd,QAAQ,CAAC4lB,IAAI,CAAC;QAE/B,IAAIrM,KAAK,CAACgC,cAAc,CAAC5Z,SAAS,CAAC,EAAE;UACnC,IAAI,CAAC2jB,aAAa,CAAC3jB,SAAS,EAAEikB,IAAI,CAAC;UACnC,IAAI,CAAC9R,OAAO,CAACsJ,GAAG,CAAClc,OAAO,CAACS,SAAS,EAAEikB,IAAI,CAACvkB,MAAM,CAAC;QAClD;QAEA,IAAI,CAACO,GAAG,CAACya,aAAa,IAAI9c,KAAK,EAAE;UAC/BqC,GAAG,CAACya,aAAa,GAAG,IAAI;UACxB9c,KAAK,CAAC+c,gBAAgB,CAAC1a,GAAG,CAAC;QAC7B;MACF;MAEAmkB,WAAW,EAAE;MAEb,IAAI,IAAI,CAAClL,SAAS,EAAE;QAClB,IAAI,CAACA,SAAS,CAAC+K,IAAI,CAAC;MACtB;IACF,CAAC,CAAC;IAEF,IAAI,IAAI,CAAC9K,aAAa,IAAIiL,WAAW,GAAG,CAAC,EAAE;MACzC,MAAM9T,IAAI,GAAG,EAAE;MACf+S,IAAI,CAAClwB,OAAO,CAAE+V,CAAC,IAAK;QAClBoH,IAAI,CAAClP,IAAI,CAAC8H,CAAC,CAACtL,KAAK,CAAC;MACpB,CAAC,CAAC;MACF,IAAI,CAACub,aAAa,CAAC7I,IAAI,EAAE8T,WAAW,CAAC;IACvC;EACF;EAGAxH,iBAAiBA,CAAC2G,KAAK,EAAE70B,GAAG,EAAE;IAC5B,IAAI60B,KAAK,CAACr1B,MAAM,IAAI,CAAC,IAAIq1B,KAAK,CAAC,CAAC,CAAC,IAAI/L,gDAAc,EAAE;MACnD+L,KAAK,GAAG,EAAE;IACZ;IACA,IAAI70B,GAAG,EAAE;MACP60B,KAAK,CAACpwB,OAAO,CAAEoxB,EAAE,IAAK;QACpB,IAAIA,EAAE,CAACx3B,GAAG,EAAE;UAEV,IAAIiF,GAAG,GAAG,IAAI,CAACumB,YAAY,CAACiM,SAAS,CAAErT,EAAE,IAAK;YAC5C,OAAOA,EAAE,CAACsT,IAAI,IAAIF,EAAE,CAACE,IAAI,IAAItT,EAAE,CAACpkB,GAAG,IAAIw3B,EAAE,CAACx3B,GAAG;UAC/C,CAAC,CAAC;UACF,IAAIiF,GAAG,GAAG,CAAC,EAAE;YAEX,IAAI,CAACuyB,EAAE,CAACG,IAAI,EAAE;cAEZ1yB,GAAG,GAAG,IAAI,CAACumB,YAAY,CAACiM,SAAS,CAAErT,EAAE,IAAK;gBACxC,OAAOA,EAAE,CAACsT,IAAI,IAAIF,EAAE,CAACE,IAAI,IAAI,CAACtT,EAAE,CAACuT,IAAI;cACvC,CAAC,CAAC;cACF,IAAI1yB,GAAG,IAAI,CAAC,EAAE;gBAEZ,IAAI,CAACumB,YAAY,CAACnmB,MAAM,CAACJ,GAAG,EAAE,CAAC,CAAC;cAClC;YACF;YACA,IAAI,CAACumB,YAAY,CAACnX,IAAI,CAACmjB,EAAE,CAAC;UAC5B,CAAC,MAAM;YAEL,IAAI,CAAChM,YAAY,CAACvmB,GAAG,CAAC,CAAC0yB,IAAI,GAAGH,EAAE,CAACG,IAAI;UACvC;QACF,CAAC,MAAM,IAAIH,EAAE,CAACI,IAAI,EAAE;UAElB,MAAM3yB,GAAG,GAAG,IAAI,CAACumB,YAAY,CAACiM,SAAS,CAAErT,EAAE,IAAK;YAC9C,OAAOA,EAAE,CAACsT,IAAI,IAAIF,EAAE,CAACE,IAAI,IAAI,CAACtT,EAAE,CAACuT,IAAI;UACvC,CAAC,CAAC;UACF,IAAI1yB,GAAG,IAAI,CAAC,EAAE;YACZ,IAAI,CAACumB,YAAY,CAACvmB,GAAG,CAAC,CAAC0yB,IAAI,GAAG,IAAI;UACpC;QACF;MACF,CAAC,CAAC;IACJ,CAAC,MAAM;MACL,IAAI,CAACnM,YAAY,GAAGgL,KAAK;IAC3B;IACA,IAAI,IAAI,CAAClK,cAAc,EAAE;MACvB,IAAI,CAACA,cAAc,CAAC,IAAI,CAACd,YAAY,CAAC;IACxC;EACF;EAGAwK,UAAUA,CAACC,IAAI,EAAE;IACf,IAAIA,IAAI,CAACvN,IAAI,IAAI,MAAM,EAAE;MAEvB,IAAI,CAACqG,SAAS,CAAC,CAAC;MAChB;IACF;IAEA,IAAIkH,IAAI,CAACvN,IAAI,IAAI,KAAK,IAAIuN,IAAI,CAAC1jB,GAAG,IAAIkY,gDAAc,EAAE;MAEpD,IAAI,CAAC0E,OAAO,CAAC,IAAI,CAACG,cAAc,CAAC,CAAC,CAAC/F,QAAQ,CAAC,CAAC,CAACgB,KAAK,CAAC,CAAC,CAAC;MACtD;IACF;IAEA,MAAM2M,IAAI,GAAG,IAAI,CAAC9R,OAAO,CAACyS,aAAa,CAAC5B,IAAI,CAAC1jB,GAAG,CAAC;IACjD,IAAI2kB,IAAI,EAAE;MACR,QAAQjB,IAAI,CAACvN,IAAI;QACf,KAAK,IAAI;UACPwO,IAAI,CAAChB,MAAM,GAAG,IAAI;UAClB;QACF,KAAK,KAAK;UACR,IAAIgB,IAAI,CAAChB,MAAM,EAAE;YACfgB,IAAI,CAAChB,MAAM,GAAG,KAAK;YACnBgB,IAAI,CAACC,IAAI,GAAG1Z,MAAM,CAACC,MAAM,CAACwZ,IAAI,CAACC,IAAI,IAAI,CAAC,CAAC,EAAE;cACzCC,IAAI,EAAE,IAAInM,IAAI,CAAC;YACjB,CAAC,CAAC;UACJ;UACA;QACF,KAAK,KAAK;UACRiM,IAAI,CAACL,eAAe,CAACZ,IAAI,CAACxiB,GAAG,EAAEwiB,IAAI,CAACjd,GAAG,CAAC;UACxC;QACF,KAAK,KAAK;UAER,IAAI,CAACmW,OAAO,CAAC,IAAI,CAACG,cAAc,CAAC,CAAC,CAACvF,eAAe,CAACkM,IAAI,CAAC1jB,GAAG,CAAC,CAACgY,KAAK,CAAC,CAAC,CAAC;UACrE;QACF,KAAK,KAAK;UAIR,IAAI,CAAC0L,IAAI,CAAC6B,GAAG,EAAE;YACb,IAAIZ,IAAI,CAACz3B,GAAG,EAAE;cACZy3B,IAAI,CAACz3B,GAAG,CAAC2D,SAAS,CAAC6yB,IAAI,CAACG,IAAI,CAAC;YAC/B,CAAC,MAAM;cACLc,IAAI,CAACz3B,GAAG,GAAG,IAAIF,uDAAU,CAAC,CAAC,CAAC6D,SAAS,CAAC6yB,IAAI,CAACG,IAAI,CAAC;YAClD;YACAc,IAAI,CAAClM,OAAO,GAAG,IAAIC,IAAI,CAAC,CAAC;UAC3B;UACA;QACF,KAAK,IAAI;UAEPiM,IAAI,CAACC,IAAI,GAAG;YACVC,IAAI,EAAE,IAAInM,IAAI,CAAC,CAAC;YAChB8M,EAAE,EAAE9B,IAAI,CAAC8B;UACX,CAAC;UACD;QACF,KAAK,MAAM;UAET9B,IAAI,CAACxiB,GAAG,GAAGwiB,IAAI,CAACxiB,GAAG,GAAG,CAAC;UACvByjB,IAAI,CAAC/E,IAAI,GAAG+E,IAAI,CAAC/E,IAAI,GAAG/lB,IAAI,CAAC4I,GAAG,CAACkiB,IAAI,CAAC/E,IAAI,EAAE8D,IAAI,CAACxiB,GAAG,CAAC,GAAGwiB,IAAI,CAACxiB,GAAG;UAChE;QACF,KAAK,MAAM;UAETwiB,IAAI,CAACxiB,GAAG,GAAGwiB,IAAI,CAACxiB,GAAG,GAAG,CAAC;UACvByjB,IAAI,CAACpiB,IAAI,GAAGoiB,IAAI,CAACpiB,IAAI,GAAG1I,IAAI,CAAC4I,GAAG,CAACkiB,IAAI,CAACpiB,IAAI,EAAEmhB,IAAI,CAACxiB,GAAG,CAAC,GAAGwiB,IAAI,CAACxiB,GAAG;UAChEyjB,IAAI,CAAC/E,IAAI,GAAG+E,IAAI,CAAC/E,IAAI,GAAG/lB,IAAI,CAAC4I,GAAG,CAACkiB,IAAI,CAACpiB,IAAI,EAAEoiB,IAAI,CAAC/E,IAAI,CAAC,GAAG+E,IAAI,CAAC/E,IAAI;UAClE+E,IAAI,CAACniB,MAAM,GAAGmiB,IAAI,CAACzjB,GAAG,GAAGyjB,IAAI,CAACpiB,IAAI;UAClC;QACF,KAAK,MAAM;UAET,IAAI,CAACsQ,OAAO,CAACkS,aAAa,CAACrB,IAAI,CAAC1jB,GAAG,CAAC;UACpC,IAAI,CAAC2kB,IAAI,CAACrlB,QAAQ,EAAE;YAClBqlB,IAAI,CAACrlB,QAAQ,GAAG,IAAI;YACpBqlB,IAAI,CAACvL,SAAS,GAAG,KAAK;YACtB,IAAI,CAACvG,OAAO,CAACsJ,GAAG,CAAC/c,kBAAkB,CAACskB,IAAI,CAAC1jB,GAAG,EAAE,IAAI,CAAC;UACrD,CAAC,MAAM;YACL,IAAI,CAAC6S,OAAO,CAACsJ,GAAG,CAAC5c,QAAQ,CAACmkB,IAAI,CAAC1jB,GAAG,CAAC;UACrC;UACA;QACF,KAAK,KAAK;UAER;QACF;UACE,IAAI,CAAC6S,OAAO,CAACna,MAAM,CAAC,2CAA2C,EAAEgrB,IAAI,CAACvN,IAAI,CAAC;MAC/E;MAEA,IAAI,CAACiJ,eAAe,CAACsE,IAAI,CAACvN,IAAI,EAAEwO,IAAI,CAAC;IACvC,CAAC,MAAM;MACL,IAAIjB,IAAI,CAACvN,IAAI,IAAI,KAAK,EAAE;QAItB,MAAMjpB,GAAG,GAAG,IAAIF,uDAAU,CAAC02B,IAAI,CAACG,IAAI,CAAC;QACrC,IAAI,CAAC32B,GAAG,IAAIA,GAAG,CAACI,IAAI,IAAIN,uDAAU,CAACiC,QAAQ,EAAE;UAC3C,IAAI,CAAC4jB,OAAO,CAACna,MAAM,CAAC,mCAAmC,EAAEgrB,IAAI,CAAC1jB,GAAG,EAAE0jB,IAAI,CAACG,IAAI,CAAC;UAC7E;QACF,CAAC,MAAM,IAAI32B,GAAG,CAACI,IAAI,IAAIN,uDAAU,CAACgB,KAAK,EAAE;UACvC,IAAI,CAAC6kB,OAAO,CAACna,MAAM,CAAC,6CAA6C,EAAEgrB,IAAI,CAAC1jB,GAAG,EAAE0jB,IAAI,CAACG,IAAI,CAAC;UACvF;QACF,CAAC,MAAM;UAGL,IAAI,CAACjH,OAAO,CAAC,IAAI,CAACG,cAAc,CAAC,CAAC,CAACxF,UAAU,CAAC7lB,SAAS,EAAEgyB,IAAI,CAAC1jB,GAAG,CAAC,CAACgY,KAAK,CAAC,CAAC,CAAC;UAE3E,MAAMyN,KAAK,GAAG,IAAI,CAAC5S,OAAO,CAACmS,QAAQ,CAACtB,IAAI,CAAC1jB,GAAG,CAAC;UAC7CylB,KAAK,CAAC9B,MAAM,GAAG,KAAK;UACpB8B,KAAK,CAACv4B,GAAG,GAAGA,GAAG;UACf,IAAI,CAAC2lB,OAAO,CAACsJ,GAAG,CAACpd,QAAQ,CAAC0mB,KAAK,CAAC;QAClC;MACF,CAAC,MAAM,IAAI/B,IAAI,CAACvN,IAAI,IAAI,MAAM,EAAE;QAC9B,IAAI,CAACyG,OAAO,CAAC,IAAI,CAACG,cAAc,CAAC,CAAC,CAACrF,QAAQ,CAAC,CAAC,CAACM,KAAK,CAAC,CAAC,CAAC;MACxD,CAAC,MAAM,IAAI0L,IAAI,CAACvN,IAAI,IAAI,KAAK,EAAE;QAE7B,IAAI,CAACyG,OAAO,CAAC,IAAI,CAACG,cAAc,CAAC,CAAC,CAACxF,UAAU,CAAC7lB,SAAS,EAAEgyB,IAAI,CAAC1jB,GAAG,CAAC,CAACgY,KAAK,CAAC,CAAC,CAAC;QAE3E,MAAMyN,KAAK,GAAG,IAAI,CAAC5S,OAAO,CAACmS,QAAQ,CAACtB,IAAI,CAAC1jB,GAAG,CAAC;QAC7CylB,KAAK,CAACnmB,QAAQ,GAAG,KAAK;QACtB,IAAI,CAACuT,OAAO,CAACsJ,GAAG,CAACpd,QAAQ,CAAC0mB,KAAK,CAAC;MAClC;MAEA,IAAI,CAACrG,eAAe,CAACsE,IAAI,CAACvN,IAAI,EAAEwO,IAAI,CAAC;IACvC;IAEA,IAAI,IAAI,CAAClL,MAAM,EAAE;MACf,IAAI,CAACA,MAAM,CAACiK,IAAI,CAAC;IACnB;EACF;EAGAtE,eAAeA,CAACjJ,IAAI,EAAEwO,IAAI,EAAE;IAC1B,IAAI,IAAI,CAACH,eAAe,EAAE;MACxB,IAAI,CAACA,eAAe,CAACrO,IAAI,EAAEwO,IAAI,CAAC;IAClC;EACF;EAOAnJ,OAAOA,CAAA,EAAG;IACR,OAAOziB,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,qCAAqC,CAAC,CAAC;EACzE;EAUA63B,aAAaA,CAACC,MAAM,EAAE9jB,KAAK,EAAE;IAC3B,IAAI,CAAC,IAAI,CAACuX,SAAS,EAAE;MACnB,OAAOrgB,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,iDAAiD,CAAC,CAAC;IACrF;IAEA,OAAO,IAAI,CAACglB,OAAO,CAAC6S,aAAa,CAACC,MAAM,EAAE9jB,KAAK,CAAC,CAAC2I,IAAI,CAACxO,IAAI,IAAI;MAE5D,MAAMsN,KAAK,GAAG,IAAI,CAAC2P,YAAY,CAACiM,SAAS,CAAErT,EAAE,IAAK;QAChD,OAAOA,EAAE,CAACsT,IAAI,IAAIQ,MAAM,IAAI9T,EAAE,CAACpkB,GAAG,IAAIoU,KAAK;MAC7C,CAAC,CAAC;MACF,IAAIyH,KAAK,GAAG,CAAC,CAAC,EAAE;QACd,IAAI,CAAC2P,YAAY,CAACnmB,MAAM,CAACwW,KAAK,EAAE,CAAC,CAAC;MACpC;MAEA,IAAI,IAAI,CAACyQ,cAAc,EAAE;QACvB,IAAI,CAACA,cAAc,CAAC,IAAI,CAACd,YAAY,CAAC;MACxC;MACA,OAAOjd,IAAI;IACb,CAAC,CAAC;EACJ;EAiBA4pB,QAAQA,CAAC9xB,QAAQ,EAAEO,MAAM,EAAEJ,OAAO,EAAE;IAClC,IAAI,CAAC4e,OAAO,CAAC/S,SAAS,CAAC,CAACiR,CAAC,EAAEre,GAAG,KAAK;MACjC,IAAIqe,CAAC,CAAC0R,UAAU,CAAC,CAAC,KAAK,CAACpuB,MAAM,IAAIA,MAAM,CAAC0c,CAAC,CAAC,CAAC,EAAE;QAC5Cjd,QAAQ,CAACI,IAAI,CAACD,OAAO,EAAE8c,CAAC,EAAEre,GAAG,CAAC;MAChC;IACF,CAAC,CAAC;EACJ;EASAmzB,UAAUA,CAACnxB,IAAI,EAAE;IACf,OAAO,IAAI,CAACme,OAAO,CAACyS,aAAa,CAAC5wB,IAAI,CAAC;EACzC;EAUAkO,aAAaA,CAAClO,IAAI,EAAE;IAClB,IAAIA,IAAI,EAAE;MACR,MAAMiwB,IAAI,GAAG,IAAI,CAAC9R,OAAO,CAACyS,aAAa,CAAC5wB,IAAI,CAAC;MAC7C,OAAOiwB,IAAI,GAAGA,IAAI,CAACz3B,GAAG,GAAG,IAAI;IAC/B;IACA,OAAO,IAAI,CAACA,GAAG;EACjB;EASAm1B,UAAUA,CAAC3tB,IAAI,EAAE;IACf,MAAMiwB,IAAI,GAAG,IAAI,CAAC9R,OAAO,CAACyS,aAAa,CAAC5wB,IAAI,CAAC;IAC7C,OAAOiwB,IAAI,IAAIA,IAAI,CAAChM,OAAO,IAAI,CAAC,CAACgM,IAAI,CAAChM,OAAO,CAACgF,IAAI;EACpD;EAgBAmI,cAAcA,CAAA,EAAG;IACf,OAAO,IAAI,CAAC7M,YAAY;EAC1B;AACF;AAOO,MAAM8M,QAAQ,SAASzN,KAAK,CAAC;EAElC0N,SAAS,GAAG,CAAC,CAAC;EAOd/4B,WAAWA,CAACsrB,SAAS,EAAE;IACrB,KAAK,CAACL,iDAAe,EAAEK,SAAS,CAAC;EACnC;EAGA4E,gBAAgBA,CAAC4G,IAAI,EAAE;IACrB,IAAIe,WAAW,GAAG5Z,MAAM,CAAC+M,mBAAmB,CAAC,IAAI,CAAC+N,SAAS,CAAC,CAACp3B,MAAM;IAEnE,IAAI,CAACo3B,SAAS,GAAG,CAAC,CAAC;IACnB,KAAK,IAAItzB,GAAG,IAAIqxB,IAAI,EAAE;MACpB,IAAIpjB,GAAG,GAAGojB,IAAI,CAACrxB,GAAG,CAAC;MACnB,MAAMuzB,OAAO,GAAGtlB,GAAG,CAACrC,KAAK,GAAGqC,GAAG,CAACrC,KAAK,GAAGqC,GAAG,CAACH,IAAI;MAEhDG,GAAG,GAAGyX,uDAAY,CAAC,IAAI,CAAC4N,SAAS,EAAEC,OAAO,EAAEtlB,GAAG,CAAC;MAChDmkB,WAAW,EAAE;MAEb,IAAI,IAAI,CAAClL,SAAS,EAAE;QAClB,IAAI,CAACA,SAAS,CAACjZ,GAAG,CAAC;MACrB;IACF;IAEA,IAAImkB,WAAW,GAAG,CAAC,IAAI,IAAI,CAACjL,aAAa,EAAE;MACzC,IAAI,CAACA,aAAa,CAAC3O,MAAM,CAAC8F,IAAI,CAAC,IAAI,CAACgV,SAAS,CAAC,CAAC;IACjD;EACF;EAOAxK,OAAOA,CAAA,EAAG;IACR,OAAOziB,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,sCAAsC,CAAC,CAAC;EAC1E;EAQAqvB,OAAOA,CAACjhB,MAAM,EAAE;IACd,OAAOiP,MAAM,CAACgb,cAAc,CAACH,QAAQ,CAACI,SAAS,CAAC,CAACjJ,OAAO,CAAChpB,IAAI,CAAC,IAAI,EAAE+H,MAAM,CAAC,CAACuO,IAAI,CAAC/S,CAAC,IAAI;MACpF,IAAIyT,MAAM,CAAC8F,IAAI,CAAC,IAAI,CAACgV,SAAS,CAAC,CAACp3B,MAAM,GAAG,CAAC,EAAE;QAC1C,IAAI,CAACo3B,SAAS,GAAG,CAAC,CAAC;QACnB,IAAI,IAAI,CAACnM,aAAa,EAAE;UACtB,IAAI,CAACA,aAAa,CAAC,EAAE,CAAC;QACxB;MACF;IACF,CAAC,CAAC;EACJ;EASA+L,QAAQA,CAAC9xB,QAAQ,EAAEG,OAAO,EAAE;IAC1B,MAAMgsB,EAAE,GAAInsB,QAAQ,IAAI,IAAI,CAAC8lB,SAAU;IACvC,IAAIqG,EAAE,EAAE;MACN,KAAK,IAAIvtB,GAAG,IAAI,IAAI,CAACszB,SAAS,EAAE;QAC9B/F,EAAE,CAAC/rB,IAAI,CAACD,OAAO,EAAE,IAAI,CAAC+xB,SAAS,CAACtzB,GAAG,CAAC,EAAEA,GAAG,EAAE,IAAI,CAACszB,SAAS,CAAC;MAC5D;IACF;EACF;AACF;;;;;;;;;;;;;;;;;;;;;;AC72Ea;;AAE6B;AAGrB;AAGd,SAASzvB,eAAeA,CAACmT,GAAG,EAAEjc,GAAG,EAAE;EAGxC,IAAI,OAAOA,GAAG,IAAI,QAAQ,IAAIA,GAAG,CAACmB,MAAM,IAAI,EAAE,IAAInB,GAAG,CAACmB,MAAM,IAAI,EAAE,IAAI,CAAC,IAAI,EAAE,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,CAAC,CAAChB,QAAQ,CAAC8b,GAAG,CAAC,EAAE;IACzJ,MAAM0c,IAAI,GAAG,IAAI1N,IAAI,CAACjrB,GAAG,CAAC;IAC1B,IAAI,CAAC44B,KAAK,CAACD,IAAI,CAAC,EAAE;MAChB,OAAOA,IAAI;IACb;EACF,CAAC,MAAM,IAAI1c,GAAG,KAAK,KAAK,IAAI,OAAOjc,GAAG,KAAK,QAAQ,EAAE;IACnD,OAAO,IAAIT,uDAAU,CAACS,GAAG,CAAC;EAC5B;EACA,OAAOA,GAAG;AACZ;AAQO,SAAS2kB,aAAaA,CAAC9a,GAAG,EAAE;EACjC,OAAOA,GAAG,IAAI,CAAC,iCAAiC,CAACuM,IAAI,CAACvM,GAAG,CAAC;AAC5D;AAEA,SAASgvB,WAAWA,CAACC,CAAC,EAAE;EACtB,OAAQA,CAAC,YAAY7N,IAAI,IAAK,CAAC2N,KAAK,CAACE,CAAC,CAAC,IAAKA,CAAC,CAACC,OAAO,CAAC,CAAC,IAAI,CAAE;AAC/D;AAGO,SAASC,iBAAiBA,CAACF,CAAC,EAAE;EACnC,IAAI,CAACD,WAAW,CAACC,CAAC,CAAC,EAAE;IACnB,OAAO70B,SAAS;EAClB;EAEA,MAAMg1B,GAAG,GAAG,SAAAA,CAASj5B,GAAG,EAAEk5B,EAAE,EAAE;IAC5BA,EAAE,GAAGA,EAAE,IAAI,CAAC;IACZ,OAAO,GAAG,CAACC,MAAM,CAACD,EAAE,GAAG,CAAC,EAAE,GAAGl5B,GAAG,EAAEmB,MAAM,CAAC,GAAGnB,GAAG;EACjD,CAAC;EAED,MAAMo5B,MAAM,GAAGN,CAAC,CAACO,kBAAkB,CAAC,CAAC;EACrC,OAAOP,CAAC,CAACQ,cAAc,CAAC,CAAC,GAAG,GAAG,GAAGL,GAAG,CAACH,CAAC,CAACS,WAAW,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,GAAG,GAAGN,GAAG,CAACH,CAAC,CAACU,UAAU,CAAC,CAAC,CAAC,GACpF,GAAG,GAAGP,GAAG,CAACH,CAAC,CAACW,WAAW,CAAC,CAAC,CAAC,GAAG,GAAG,GAAGR,GAAG,CAACH,CAAC,CAACY,aAAa,CAAC,CAAC,CAAC,GAAG,GAAG,GAAGT,GAAG,CAACH,CAAC,CAACa,aAAa,CAAC,CAAC,CAAC,IACvFP,MAAM,GAAG,GAAG,GAAGH,GAAG,CAACG,MAAM,EAAE,CAAC,CAAC,GAAG,EAAE,CAAC,GAAG,GAAG;AAC9C;AAKO,SAAS1O,QAAQA,CAACxV,GAAG,EAAE3C,GAAG,EAAEqnB,MAAM,EAAE;EACzC,IAAI,OAAOrnB,GAAG,IAAI,QAAQ,EAAE;IAC1B,IAAIA,GAAG,KAAKtO,SAAS,EAAE;MACrB,OAAOiR,GAAG;IACZ;IACA,IAAI3C,GAAG,KAAK1J,gDAAQ,EAAE;MACpB,OAAO5E,SAAS;IAClB;IACA,OAAOsO,GAAG;EACZ;EAEA,IAAIA,GAAG,KAAK,IAAI,EAAE;IAChB,OAAOA,GAAG;EACZ;EAGA,IAAIA,GAAG,YAAY0Y,IAAI,IAAI,CAAC2N,KAAK,CAACrmB,GAAG,CAAC,EAAE;IACtC,OAAQ,CAAC2C,GAAG,IAAI,EAAEA,GAAG,YAAY+V,IAAI,CAAC,IAAI2N,KAAK,CAAC1jB,GAAG,CAAC,IAAIA,GAAG,GAAG3C,GAAG,GAAIA,GAAG,GAAG2C,GAAG;EAChF;EAGA,IAAI3C,GAAG,YAAYhT,uDAAU,EAAE;IAC7B,OAAO,IAAIA,uDAAU,CAACgT,GAAG,CAAC;EAC5B;EAGA,IAAIA,GAAG,YAAY3M,KAAK,EAAE;IACxB,OAAO2M,GAAG;EACZ;EAEA,IAAI,CAAC2C,GAAG,IAAIA,GAAG,KAAKrM,gDAAQ,EAAE;IAC5BqM,GAAG,GAAG3C,GAAG,CAAC/S,WAAW,CAAC,CAAC;EACzB;EAEA,KAAK,IAAIq6B,IAAI,IAAItnB,GAAG,EAAE;IACpB,IAAIA,GAAG,CAACmC,cAAc,CAACmlB,IAAI,CAAC,KAAK,CAACD,MAAM,IAAI,CAACA,MAAM,CAACC,IAAI,CAAC,CAAC,IAAKA,IAAI,IAAI,eAAgB,EAAE;MACvF,IAAI;QACF3kB,GAAG,CAAC2kB,IAAI,CAAC,GAAGnP,QAAQ,CAACxV,GAAG,CAAC2kB,IAAI,CAAC,EAAEtnB,GAAG,CAACsnB,IAAI,CAAC,CAAC;MAC5C,CAAC,CAAC,OAAO7qB,GAAG,EAAE,CAEd;IACF;EACF;EACA,OAAOkG,GAAG;AACZ;AAGO,SAASyV,YAAYA,CAACmP,KAAK,EAAE7d,GAAG,EAAE8d,MAAM,EAAEH,MAAM,EAAE;EACvDE,KAAK,CAAC7d,GAAG,CAAC,GAAGyO,QAAQ,CAACoP,KAAK,CAAC7d,GAAG,CAAC,EAAE8d,MAAM,EAAEH,MAAM,CAAC;EACjD,OAAOE,KAAK,CAAC7d,GAAG,CAAC;AACnB;AAIO,SAAS+d,QAAQA,CAACtD,GAAG,EAAE;EAC5BjZ,MAAM,CAAC8F,IAAI,CAACmT,GAAG,CAAC,CAACtwB,OAAO,CAAE6V,GAAG,IAAK;IAChC,IAAIA,GAAG,CAAC,CAAC,CAAC,IAAI,GAAG,EAAE;MAEjB,OAAOya,GAAG,CAACza,GAAG,CAAC;IACjB,CAAC,MAAM,IAAI,CAACya,GAAG,CAACza,GAAG,CAAC,EAAE;MAEpB,OAAOya,GAAG,CAACza,GAAG,CAAC;IACjB,CAAC,MAAM,IAAIrW,KAAK,CAACC,OAAO,CAAC6wB,GAAG,CAACza,GAAG,CAAC,CAAC,IAAIya,GAAG,CAACza,GAAG,CAAC,CAAC9a,MAAM,IAAI,CAAC,EAAE;MAE1D,OAAOu1B,GAAG,CAACza,GAAG,CAAC;IACjB,CAAC,MAAM,IAAI,CAACya,GAAG,CAACza,GAAG,CAAC,EAAE;MAEpB,OAAOya,GAAG,CAACza,GAAG,CAAC;IACjB,CAAC,MAAM,IAAIya,GAAG,CAACza,GAAG,CAAC,YAAYgP,IAAI,EAAE;MAEnC,IAAI,CAAC4N,WAAW,CAACnC,GAAG,CAACza,GAAG,CAAC,CAAC,EAAE;QAC1B,OAAOya,GAAG,CAACza,GAAG,CAAC;MACjB;IACF,CAAC,MAAM,IAAI,OAAOya,GAAG,CAACza,GAAG,CAAC,IAAI,QAAQ,EAAE;MACtC+d,QAAQ,CAACtD,GAAG,CAACza,GAAG,CAAC,CAAC;MAElB,IAAIwB,MAAM,CAAC+M,mBAAmB,CAACkM,GAAG,CAACza,GAAG,CAAC,CAAC,CAAC9a,MAAM,IAAI,CAAC,EAAE;QACpD,OAAOu1B,GAAG,CAACza,GAAG,CAAC;MACjB;IACF;EACF,CAAC,CAAC;EACF,OAAOya,GAAG;AACZ;AAAC;AAKM,SAAS9L,cAAcA,CAACjmB,GAAG,EAAE;EAClC,IAAIgsB,GAAG,GAAG,EAAE;EACZ,IAAI/qB,KAAK,CAACC,OAAO,CAAClB,GAAG,CAAC,EAAE;IAEtB,KAAK,IAAIzD,CAAC,GAAG,CAAC,EAAEgK,CAAC,GAAGvG,GAAG,CAACxD,MAAM,EAAED,CAAC,GAAGgK,CAAC,EAAEhK,CAAC,EAAE,EAAE;MAC1C,IAAI+4B,CAAC,GAAGt1B,GAAG,CAACzD,CAAC,CAAC;MACd,IAAI+4B,CAAC,EAAE;QACLA,CAAC,GAAGA,CAAC,CAACC,IAAI,CAAC,CAAC,CAACC,WAAW,CAAC,CAAC;QAC1B,IAAIF,CAAC,CAAC94B,MAAM,GAAG,CAAC,EAAE;UAChBwvB,GAAG,CAACtc,IAAI,CAAC4lB,CAAC,CAAC;QACb;MACF;IACF;IACAtJ,GAAG,CAACvV,IAAI,CAAC,CAAC,CAACxU,MAAM,CAAC,UAASwzB,IAAI,EAAEC,GAAG,EAAEC,GAAG,EAAE;MACzC,OAAO,CAACD,GAAG,IAAID,IAAI,IAAIE,GAAG,CAACD,GAAG,GAAG,CAAC,CAAC;IACrC,CAAC,CAAC;EACJ;EACA,IAAI1J,GAAG,CAACxvB,MAAM,IAAI,CAAC,EAAE;IAGnBwvB,GAAG,CAACtc,IAAI,CAACxL,gDAAQ,CAAC;EACpB;EACA,OAAO8nB,GAAG;AACZ;;;;;;;;;;;;;;AC3KO,MAAMzpB,eAAe,GAAG,SAAS;;;;;;UCAxC;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;;;;WCtBA;WACA;WACA;WACA;WACA;WACA,iCAAiC,WAAW;WAC5C;WACA;;;;;WCPA;WACA;WACA;WACA;WACA,yCAAyC,wCAAwC;WACjF;WACA;WACA;;;;;WCPA;WACA;WACA;WACA;WACA,GAAG;WACH;WACA;WACA,CAAC;;;;;WCPD;;;;;WCAA;WACA;WACA;WACA,uDAAuD,iBAAiB;WACxE;WACA,gDAAgD,aAAa;WAC7D;;;;;;;;;;;;;;;;;;;;;;;;;;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;AAM6B;AACL;AACG;AACC;AACX;AACG;AACa;AACC;AAK3B;AAQA;AAIR;AAGZ,IAAI6B,iBAAiB;AACrB,IAAI,OAAOyxB,SAAS,IAAI,WAAW,EAAE;EACnCzxB,iBAAiB,GAAGyxB,SAAS;AAC/B;AAEA,IAAIxxB,WAAW;AACf,IAAI,OAAOyxB,cAAc,IAAI,WAAW,EAAE;EACxCzxB,WAAW,GAAGyxB,cAAc;AAC9B;AAEA,IAAIC,iBAAiB;AACrB,IAAI,OAAOC,SAAS,IAAI,WAAW,EAAE;EACnCD,iBAAiB,GAAGC,SAAS;AAC/B;AAIQ;AAGRC,oBAAoB,CAAC,CAAC;AAKtB,SAASA,oBAAoBA,CAAA,EAAG;EAE9B,MAAMC,KAAK,GAAG,mEAAmE;EAEjF,IAAI,OAAOC,IAAI,IAAI,WAAW,EAAE;IAC9BC,qBAAM,CAACD,IAAI,GAAG,YAAqB;MAAA,IAAZE,KAAK,GAAAr1B,SAAA,CAAAxE,MAAA,QAAAwE,SAAA,QAAA1B,SAAA,GAAA0B,SAAA,MAAG,EAAE;MAC/B,IAAItF,GAAG,GAAG26B,KAAK;MACf,IAAIC,MAAM,GAAG,EAAE;MAEf,KAAK,IAAI9f,KAAK,GAAG,CAAC,EAAE+f,QAAQ,EAAEh6B,CAAC,GAAG,CAAC,EAAEgb,GAAG,GAAG2e,KAAK,EAAEx6B,GAAG,CAACgB,MAAM,CAACH,CAAC,GAAG,CAAC,CAAC,KAAKgb,GAAG,GAAG,GAAG,EAAEhb,CAAC,GAAG,CAAC,CAAC,EAAE+5B,MAAM,IAAI/e,GAAG,CAAC7a,MAAM,CAAC,EAAE,GAAG8Z,KAAK,IAAI,CAAC,GAAGja,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,EAAE;QAE5Ig6B,QAAQ,GAAG76B,GAAG,CAACiY,UAAU,CAACpX,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAErC,IAAIg6B,QAAQ,GAAG,IAAI,EAAE;UACnB,MAAM,IAAI96B,KAAK,CAAC,0FAA0F,CAAC;QAC7G;QACA+a,KAAK,GAAGA,KAAK,IAAI,CAAC,GAAG+f,QAAQ;MAC/B;MAEA,OAAOD,MAAM;IACf,CAAC;EACH;EAEA,IAAI,OAAO/iB,IAAI,IAAI,WAAW,EAAE;IAC9B6iB,qBAAM,CAAC7iB,IAAI,GAAG,YAAqB;MAAA,IAAZ8iB,KAAK,GAAAr1B,SAAA,CAAAxE,MAAA,QAAAwE,SAAA,QAAA1B,SAAA,GAAA0B,SAAA,MAAG,EAAE;MAC/B,IAAItF,GAAG,GAAG26B,KAAK,CAAC7F,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC;MAClC,IAAI8F,MAAM,GAAG,EAAE;MAEf,IAAI56B,GAAG,CAACc,MAAM,GAAG,CAAC,IAAI,CAAC,EAAE;QACvB,MAAM,IAAIf,KAAK,CAAC,mEAAmE,CAAC;MACtF;MACA,KAAK,IAAI+6B,EAAE,GAAG,CAAC,EAAEC,EAAE,GAAG,CAAC,EAAEj3B,MAAM,EAAEjD,CAAC,GAAG,CAAC,EAAEiD,MAAM,GAAG9D,GAAG,CAACgB,MAAM,CAACH,CAAC,EAAE,CAAC,EAE9D,CAACiD,MAAM,KAAKi3B,EAAE,GAAGD,EAAE,GAAG,CAAC,GAAGC,EAAE,GAAG,EAAE,GAAGj3B,MAAM,GAAGA,MAAM,EACjDg3B,EAAE,EAAE,GAAG,CAAC,CAAC,GAAGF,MAAM,IAAII,MAAM,CAACC,YAAY,CAAC,GAAG,GAAGF,EAAE,KAAK,CAAC,CAAC,GAAGD,EAAE,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,EAC1E;QACAh3B,MAAM,GAAG02B,KAAK,CAAChc,OAAO,CAAC1a,MAAM,CAAC;MAChC;MAEA,OAAO82B,MAAM;IACf,CAAC;EACH;EAEA,IAAI,OAAOnW,MAAM,IAAI,WAAW,EAAE;IAChCiW,qBAAM,CAACjW,MAAM,GAAG;MACd0V,SAAS,EAAEzxB,iBAAiB;MAC5B0xB,cAAc,EAAEzxB,WAAW;MAC3B2xB,SAAS,EAAED,iBAAiB;MAC5BniB,GAAG,EAAE;QACHC,eAAe,EAAE,SAAAA,CAAA,EAAW;UAC1B,MAAM,IAAIpY,KAAK,CAAC,gEAAgE,CAAC;QACnF;MACF;IACF,CAAC;EACH;EAEA0J,sDAAU,CAACgB,mBAAmB,CAAC/B,iBAAiB,EAAEC,WAAW,CAAC;EAC9Dkc,sDAAe,CAACsD,kBAAkB,CAACxf,WAAW,CAAC;EAC/CuxB,8CAAO,CAAChlB,mBAAmB,CAACmlB,iBAAiB,CAAC;AAChD;AAGA,SAASa,eAAeA,CAAA,EAAG;EACzB,IAAI,OAAOzW,MAAM,IAAI,QAAQ,EAAE;IAC7B,IAAIA,MAAM,CAAC,WAAW,CAAC,EAAE;MACvB,OAAO,IAAI;IACb,CAAC,MAAM,IAAIA,MAAM,CAAC,gBAAgB,CAAC,EAAE;MAEnC,OAAO,IAAI;IACb;EACF;EACA,OAAO,IAAI;AACb;AAGA,SAAS0W,gBAAgBA,CAACn7B,GAAG,EAAE;EAI7B,OAAOy6B,IAAI,CAACW,kBAAkB,CAACp7B,GAAG,CAAC,CAAC80B,OAAO,CAAC,iBAAiB,EAC3D,SAASuG,YAAYA,CAACxX,KAAK,EAAEyX,EAAE,EAAE;IAC/B,OAAON,MAAM,CAACC,YAAY,CAAC,IAAI,GAAGK,EAAE,CAAC;EACvC,CAAC,CAAC,CAAC;AACP;AAGA,SAASC,eAAeA,CAAC3f,GAAG,EAAEjc,GAAG,EAAE;EACjC,IAAIA,GAAG,YAAYirB,IAAI,EAAE;IAEvBjrB,GAAG,GAAGg5B,4DAAiB,CAACh5B,GAAG,CAAC;EAC9B,CAAC,MAAM,IAAIA,GAAG,YAAYT,uDAAU,EAAE;IACpCS,GAAG,GAAGA,GAAG,CAACoC,UAAU,CAAC,CAAC;EACxB,CAAC,MAAM,IAAIpC,GAAG,KAAKiE,SAAS,IAAIjE,GAAG,KAAK,IAAI,IAAIA,GAAG,KAAK,KAAK,IAC1D4F,KAAK,CAACC,OAAO,CAAC7F,GAAG,CAAC,IAAIA,GAAG,CAACmB,MAAM,IAAI,CAAE,IACrC,OAAOnB,GAAG,IAAI,QAAQ,IAAMyd,MAAM,CAAC8F,IAAI,CAACvjB,GAAG,CAAC,CAACmB,MAAM,IAAI,CAAG,EAAE;IAE9D,OAAO8C,SAAS;EAClB;EAEA,OAAOjE,GAAG;AACZ;AAAC;AAGD,SAAS67B,gBAAgBA,CAAC5f,GAAG,EAAEjc,GAAG,EAAE;EAClC,IAAI,OAAOA,GAAG,IAAI,QAAQ,IAAIA,GAAG,CAACmB,MAAM,GAAG,GAAG,EAAE;IAC9C,OAAO,GAAG,GAAGnB,GAAG,CAACmB,MAAM,GAAG,WAAW,GAAGnB,GAAG,CAACijB,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,KAAK,GAAGjjB,GAAG,CAACijB,SAAS,CAACjjB,GAAG,CAACmB,MAAM,GAAG,EAAE,CAAC,GAAG,GAAG;EAC7G;EACA,OAAOy6B,eAAe,CAAC3f,GAAG,EAAEjc,GAAG,CAAC;AAClC;AAAC;AAGD,SAAS87B,cAAcA,CAAC/D,EAAE,EAAEgE,OAAO,EAAE;EACnChE,EAAE,GAAGA,EAAE,IAAI,EAAE;EACb,IAAIiE,WAAW,GAAG,EAAE;EAEpB,IAAI,cAAc,CAAC5lB,IAAI,CAAC2lB,OAAO,CAAC,EAAE;IAChCC,WAAW,GAAG,eAAe;EAC/B;EACA,IAAIprB,MAAM;EAEVmnB,EAAE,GAAGA,EAAE,CAAC5C,OAAO,CAAC,sBAAsB,EAAE,EAAE,CAAC;EAE3C,IAAI7yB,CAAC,GAAGy1B,EAAE,CAAC7T,KAAK,CAAC,wBAAwB,CAAC;EAC1C,IAAI5hB,CAAC,EAAE;IAGL,MAAM25B,QAAQ,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,SAAS,CAAC;IACjE,IAAIC,GAAG,GAAGnE,EAAE,CAACoE,MAAM,CAAC75B,CAAC,CAACuZ,KAAK,GAAGvZ,CAAC,CAAC,CAAC,CAAC,CAACnB,MAAM,CAAC,CAACY,KAAK,CAAC,GAAG,CAAC;IACrD,IAAIq6B,MAAM,GAAG,EAAE;IACf,IAAIzyB,OAAO;IAEX,KAAK,IAAIzI,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGg7B,GAAG,CAAC/6B,MAAM,EAAED,CAAC,EAAE,EAAE;MACnC,IAAIm7B,EAAE,GAAG,uBAAuB,CAAC9Z,IAAI,CAAC2Z,GAAG,CAACh7B,CAAC,CAAC,CAAC;MAC7C,IAAIm7B,EAAE,EAAE;QAEND,MAAM,CAAC/nB,IAAI,CAAC,CAACgoB,EAAE,CAAC,CAAC,CAAC,EAAEA,EAAE,CAAC,CAAC,CAAC,EAAEJ,QAAQ,CAACxE,SAAS,CAAEhR,CAAC,IAAK;UACnD,OAAO4V,EAAE,CAAC,CAAC,CAAC,CAAClC,WAAW,CAAC,CAAC,CAAC3Z,UAAU,CAACiG,CAAC,CAAC;QAC1C,CAAC,CAAC,CAAC,CAAC;QACJ,IAAI4V,EAAE,CAAC,CAAC,CAAC,IAAI,SAAS,EAAE;UACtB1yB,OAAO,GAAG0yB,EAAE,CAAC,CAAC,CAAC;QACjB;MACF;IACF;IAEAD,MAAM,CAAChhB,IAAI,CAAC,CAAC9W,CAAC,EAAEC,CAAC,KAAK;MACpB,OAAOD,CAAC,CAAC,CAAC,CAAC,GAAGC,CAAC,CAAC,CAAC,CAAC;IACpB,CAAC,CAAC;IACF,IAAI63B,MAAM,CAACj7B,MAAM,GAAG,CAAC,EAAE;MAErB,IAAIi7B,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAACjC,WAAW,CAAC,CAAC,CAAC3Z,UAAU,CAAC,KAAK,CAAC,EAAE;QAChD4b,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,MAAM;MACvB,CAAC,MAAM,IAAIA,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,EAAE;QAChCA,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,OAAO;MACxB,CAAC,MAAM,IAAIA,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,QAAQ,IAAIzyB,OAAO,EAAE;QAC9CyyB,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAGzyB,OAAO;MACxB;MACAiH,MAAM,GAAGwrB,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,GAAGA,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC5C,CAAC,MAAM;MAELxrB,MAAM,GAAGtO,CAAC,CAAC,CAAC,CAAC;IACf;EACF,CAAC,MAAM,IAAI,UAAU,CAAC8T,IAAI,CAAC2hB,EAAE,CAAC,EAAE;IAC9Bz1B,CAAC,GAAG,oBAAoB,CAACigB,IAAI,CAACwV,EAAE,CAAC;IACjC,IAAIz1B,CAAC,EAAE;MACLsO,MAAM,GAAG,UAAU,GAAGtO,CAAC,CAAC,CAAC,CAAC;IAC5B,CAAC,MAAM;MACLsO,MAAM,GAAG,WAAW;IACtB;EACF,CAAC,MAAM;IAELtO,CAAC,GAAG,oBAAoB,CAACigB,IAAI,CAACwV,EAAE,CAAC;IACjC,IAAIz1B,CAAC,EAAE;MACLsO,MAAM,GAAGtO,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,GAAGA,CAAC,CAAC,CAAC,CAAC;IAC5B,CAAC,MAAM;MACLA,CAAC,GAAGy1B,EAAE,CAACh2B,KAAK,CAAC,GAAG,CAAC;MACjB6O,MAAM,GAAGtO,CAAC,CAAC,CAAC,CAAC;IACf;EACF;EAGAA,CAAC,GAAGsO,MAAM,CAAC7O,KAAK,CAAC,GAAG,CAAC;EACrB,IAAIO,CAAC,CAACnB,MAAM,GAAG,CAAC,EAAE;IAChB,MAAMm7B,CAAC,GAAGh6B,CAAC,CAAC,CAAC,CAAC,CAACP,KAAK,CAAC,GAAG,CAAC;IACzB,MAAMw6B,KAAK,GAAGD,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,GAAGA,CAAC,CAAC,CAAC,CAAC,CAACH,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,GAAG,EAAE;IACjDvrB,MAAM,GAAI,GAAEtO,CAAC,CAAC,CAAC,CAAE,IAAGg6B,CAAC,CAAC,CAAC,CAAE,GAAEC,KAAM,EAAC;EACpC;EACA,OAAOP,WAAW,GAAGprB,MAAM;AAC7B;AAKO,MAAM4rB,MAAM,CAAC;EAClBlV,KAAK;EACLD,OAAO;EAEPoV,QAAQ;EAGRnX,OAAO;EAGPoX,QAAQ,GAAG,EAAE;EACbC,SAAS;EAETC,KAAK,GAAG,WAAW;EACnBC,cAAc,GAAG,IAAI;EAGrBC,eAAe,GAAG,KAAK;EAEvBC,gBAAgB,GAAG,KAAK;EAExB9F,MAAM,GAAG,IAAI;EAEb+F,cAAc,GAAG,KAAK;EAEtBC,MAAM,GAAG,IAAI;EAEb1X,UAAU,GAAG,IAAI;EAEjB2X,cAAc,GAAG,CAAC;EAElBC,UAAU,GAAG/wB,IAAI,CAACgxB,KAAK,CAAEhxB,IAAI,CAACE,MAAM,CAAC,CAAC,GAAG,MAAM,GAAI,MAAM,CAAC;EAE1D+wB,WAAW,GAAG,IAAI;EAElBC,YAAY,GAAG,IAAI;EAGnBC,gBAAgB,GAAG,CAAC,CAAC;EAErBC,eAAe,GAAG,IAAI;EAGtBC,WAAW,GAAG,IAAI;EAGlBC,QAAQ,GAAG,KAAK;EAEhBhP,GAAG,GAAG,IAAI;EAGViP,MAAM,GAAG,CAAC,CAAC;EAeXn+B,WAAWA,CAACgL,MAAM,EAAEozB,UAAU,EAAE;IAC9B,IAAI,CAACtW,KAAK,GAAG9c,MAAM,CAACf,IAAI;IACxB,IAAI,CAAC4d,OAAO,GAAG7c,MAAM,CAACH,MAAM;IAG5B,IAAI,CAACoyB,QAAQ,GAAGjyB,MAAM,CAACqzB,OAAO,IAAI,WAAW;IAG7C,IAAI,CAACvY,OAAO,GAAG9a,MAAM,CAACZ,MAAM;IAG5B,IAAI,CAAC+yB,SAAS,GAAGnyB,MAAM,CAACszB,QAAQ,IAAI,KAAK;IAEzC,IAAI,OAAOC,SAAS,IAAI,WAAW,EAAE;MACnC,IAAI,CAACrB,QAAQ,GAAGZ,cAAc,CAACiC,SAAS,CAACC,SAAS,EAAED,SAAS,CAAChC,OAAO,CAAC;MACtE,IAAI,CAACa,KAAK,GAAGmB,SAAS,CAACD,QAAQ;MAE/B,IAAI,CAACjB,cAAc,GAAGkB,SAAS,CAACE,QAAQ,IAAI,OAAO;IACrD;IAEAn0B,sDAAU,CAACmB,MAAM,GAAG,IAAI,CAACA,MAAM;IAC/BkO,0DAAa,GAAG,IAAI,CAAClO,MAAM;IAG3B,IAAIT,MAAM,CAACG,SAAS,IAAI,IAAI,IAAIH,MAAM,CAACG,SAAS,IAAI,IAAI,EAAE;MACxDH,MAAM,CAACG,SAAS,GAAG4wB,eAAe,CAAC,CAAC;IACtC;IACA,IAAI,CAACkC,WAAW,GAAG,IAAI3zB,sDAAU,CAACU,MAAM,EAAEigB,wDAAsB,EAAsB,IAAI,CAAC;IAC3F,IAAI,CAACgT,WAAW,CAAC7uB,SAAS,GAAIa,IAAI,IAAK;MAErC,IAAI,CAAC,CAACyuB,eAAe,CAACzuB,IAAI,CAAC;IAC7B,CAAC;IAGD,IAAI,CAACguB,WAAW,CAAC9uB,MAAM,GAAG3E,CAAC,IAAI,IAAI,CAAC,CAACm0B,cAAc,CAAC,CAAC;IACrD,IAAI,CAACV,WAAW,CAAC5uB,YAAY,GAAG,CAACG,GAAG,EAAEhI,IAAI,KAAK,IAAI,CAAC,CAACo3B,YAAY,CAACpvB,GAAG,EAAEhI,IAAI,CAAC;IAG5E,IAAI,CAACy2B,WAAW,CAAClxB,wBAAwB,GAAG,CAACJ,OAAO,EAAEqjB,OAAO,KAAK;MAChE,IAAI,IAAI,CAACjjB,wBAAwB,EAAE;QACjC,IAAI,CAACA,wBAAwB,CAACJ,OAAO,EAAEqjB,OAAO,CAAC;MACjD;IACF,CAAC;IAED,IAAI,CAACkO,QAAQ,GAAGlzB,MAAM,CAAC6zB,OAAO;IAE9B,IAAI,CAAC3P,GAAG,GAAG,IAAI6L,8CAAO,CAACvrB,GAAG,IAAI;MAC5B,IAAI,CAAC/D,MAAM,CAAC,IAAI,EAAE+D,GAAG,CAAC;IACxB,CAAC,EAAE,IAAI,CAAC/D,MAAM,CAAC;IAEf,IAAI,IAAI,CAACyyB,QAAQ,EAAE;MAGjB,MAAMjxB,IAAI,GAAG,EAAE;MACf,IAAI,CAACiiB,GAAG,CAAC5d,YAAY,CAAC,CAAC,CAACiM,IAAI,CAAC/S,CAAC,IAAI;QAEhC,OAAO,IAAI,CAAC0kB,GAAG,CAACrc,SAAS,CAAE5C,IAAI,IAAK;UAClC,IAAIoB,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAE7uB,IAAI,CAACxI,IAAI,CAAC;UAC9C,IAAI4J,KAAK,EAAE;YACT;UACF;UACA,IAAIpB,IAAI,CAACxI,IAAI,IAAIwjB,gDAAc,EAAE;YAC/B5Z,KAAK,GAAG,IAAIimB,8CAAO,CAAC,CAAC;UACvB,CAAC,MAAM,IAAIrnB,IAAI,CAACxI,IAAI,IAAIwjB,iDAAe,EAAE;YACvC5Z,KAAK,GAAG,IAAIynB,+CAAQ,CAAC,CAAC;UACxB,CAAC,MAAM;YACLznB,KAAK,GAAG,IAAIga,4CAAK,CAACpb,IAAI,CAACxI,IAAI,CAAC;UAC9B;UACA,IAAI,CAACynB,GAAG,CAACpc,gBAAgB,CAACzB,KAAK,EAAEpB,IAAI,CAAC;UACtC,IAAI,CAAC,CAAC8uB,kBAAkB,CAAC1tB,KAAK,CAAC;UAC/BA,KAAK,CAACyc,aAAa,CAAC,CAAC;UAErB,OAAOzc,KAAK,CAAC+a,IAAI;UAEjBnf,IAAI,CAAC4H,IAAI,CAACxD,KAAK,CAAC0e,aAAa,CAAC,IAAI,CAACb,GAAG,CAAC,CAAC;QAC1C,CAAC,CAAC;MACJ,CAAC,CAAC,CAAC3R,IAAI,CAAC/S,CAAC,IAAI;QAEX,OAAO,IAAI,CAAC0kB,GAAG,CAAC7b,QAAQ,CAAEpD,IAAI,IAAK;UACjC,IAAI,CAAC,CAAC+uB,QAAQ,CAAC,MAAM,EAAE/uB,IAAI,CAACgD,GAAG,EAAEiY,mDAAQ,CAAC,CAAC,CAAC,EAAEjb,IAAI,CAACkD,MAAM,CAAC,CAAC;QAC7D,CAAC,CAAC;MACJ,CAAC,CAAC,CAACoK,IAAI,CAAC/S,CAAC,IAAI;QAEX,OAAOsB,OAAO,CAACmzB,GAAG,CAAChyB,IAAI,CAAC;MAC1B,CAAC,CAAC,CAACsQ,IAAI,CAAC/S,CAAC,IAAI;QACX,IAAI4zB,UAAU,EAAE;UACdA,UAAU,CAAC,CAAC;QACd;QACA,IAAI,CAAC3yB,MAAM,CAAC,+BAA+B,CAAC;MAC9C,CAAC,CAAC,CAACyB,KAAK,CAACsC,GAAG,IAAI;QACd,IAAI4uB,UAAU,EAAE;UACdA,UAAU,CAAC5uB,GAAG,CAAC;QACjB;QACA,IAAI,CAAC/D,MAAM,CAAC,wCAAwC,EAAE+D,GAAG,CAAC;MAC5D,CAAC,CAAC;IACJ,CAAC,MAAM;MACL,IAAI,CAAC0f,GAAG,CAACvd,cAAc,CAAC,CAAC,CAAC4L,IAAI,CAAC/S,CAAC,IAAI;QAClC,IAAI4zB,UAAU,EAAE;UACdA,UAAU,CAAC,CAAC;QACd;MACF,CAAC,CAAC;IACJ;EACF;EAKA3yB,MAAMA,CAAC5K,GAAG,EAAW;IACnB,IAAI,IAAI,CAACy8B,eAAe,EAAE;MACxB,MAAMhE,CAAC,GAAG,IAAI7N,IAAI,CAAC,CAAC;MACpB,MAAMyT,UAAU,GAAG,CAAC,GAAG,GAAG5F,CAAC,CAACW,WAAW,CAAC,CAAC,EAAEnjB,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,GACxD,CAAC,GAAG,GAAGwiB,CAAC,CAACY,aAAa,CAAC,CAAC,EAAEpjB,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,GACzC,CAAC,GAAG,GAAGwiB,CAAC,CAACa,aAAa,CAAC,CAAC,EAAErjB,KAAK,CAAC,CAAC,CAAC,CAAC,GAAG,GAAG,GACzC,CAAC,IAAI,GAAGwiB,CAAC,CAACO,kBAAkB,CAAC,CAAC,EAAE/iB,KAAK,CAAC,CAAC,CAAC,CAAC;MAAC,SAAAqoB,IAAA,GAAAh5B,SAAA,CAAAxE,MAAA,EANjCy9B,IAAI,OAAAh5B,KAAA,CAAA+4B,IAAA,OAAAA,IAAA,WAAAE,IAAA,MAAAA,IAAA,GAAAF,IAAA,EAAAE,IAAA;QAAJD,IAAI,CAAAC,IAAA,QAAAl5B,SAAA,CAAAk5B,IAAA;MAAA;MAQfC,OAAO,CAAC/0B,GAAG,CAAC,GAAG,GAAG20B,UAAU,GAAG,GAAG,EAAEr+B,GAAG,EAAEu+B,IAAI,CAACtf,IAAI,CAAC,GAAG,CAAC,CAAC;IAC1D;EACF;EAGA,CAACyf,WAAWC,CAACjmB,EAAE,EAAE;IACf,IAAIyW,OAAO,GAAG,IAAI;IAClB,IAAIzW,EAAE,EAAE;MACNyW,OAAO,GAAG,IAAIlkB,OAAO,CAAC,CAAC0C,OAAO,EAAEzC,MAAM,KAAK;QAEzC,IAAI,CAACgyB,gBAAgB,CAACxkB,EAAE,CAAC,GAAG;UAC1B,SAAS,EAAE/K,OAAO;UAClB,QAAQ,EAAEzC,MAAM;UAChB,IAAI,EAAE,IAAI0f,IAAI,CAAC;QACjB,CAAC;MACH,CAAC,CAAC;IACJ;IACA,OAAOuE,OAAO;EAChB;EAIA,CAACyP,WAAWC,CAACnmB,EAAE,EAAE/R,IAAI,EAAEm4B,IAAI,EAAEC,SAAS,EAAE;IACtC,MAAMtU,SAAS,GAAG,IAAI,CAACyS,gBAAgB,CAACxkB,EAAE,CAAC;IAC3C,IAAI+R,SAAS,EAAE;MACb,OAAO,IAAI,CAACyS,gBAAgB,CAACxkB,EAAE,CAAC;MAChC,IAAI/R,IAAI,IAAI,GAAG,IAAIA,IAAI,GAAG,GAAG,EAAE;QAC7B,IAAI8jB,SAAS,CAAC9c,OAAO,EAAE;UACrB8c,SAAS,CAAC9c,OAAO,CAACmxB,IAAI,CAAC;QACzB;MACF,CAAC,MAAM,IAAIrU,SAAS,CAACvf,MAAM,EAAE;QAC3Buf,SAAS,CAACvf,MAAM,CAAC,IAAIzE,sDAAS,CAACs4B,SAAS,EAAEp4B,IAAI,CAAC,CAAC;MAClD;IACF;EACF;EAGA,CAAC0H,IAAI2wB,CAAClxB,GAAG,EAAE4K,EAAE,EAAE;IACb,IAAIyW,OAAO;IACX,IAAIzW,EAAE,EAAE;MACNyW,OAAO,GAAG,IAAI,CAAC,CAACuP,WAAW,CAAChmB,EAAE,CAAC;IACjC;IACA5K,GAAG,GAAG6rB,mDAAQ,CAAC7rB,GAAG,CAAC;IACnB,IAAIxC,GAAG,GAAGyC,IAAI,CAACkxB,SAAS,CAACnxB,GAAG,CAAC;IAC7B,IAAI,CAAClD,MAAM,CAAC,OAAO,IAAI,IAAI,CAAC8xB,gBAAgB,GAAG3uB,IAAI,CAACkxB,SAAS,CAACnxB,GAAG,EAAE0tB,gBAAgB,CAAC,GAAGlwB,GAAG,CAAC,CAAC;IAC5F,IAAI;MACF,IAAI,CAAC8xB,WAAW,CAAC/xB,QAAQ,CAACC,GAAG,CAAC;IAChC,CAAC,CAAC,OAAOqD,GAAG,EAAE;MAEZ,IAAI+J,EAAE,EAAE;QACN,IAAI,CAAC,CAACkmB,WAAW,CAAClmB,EAAE,EAAEjP,sDAAU,CAACb,aAAa,EAAE,IAAI,EAAE+F,GAAG,CAACjI,OAAO,CAAC;MACpE,CAAC,MAAM;QACL,MAAMiI,GAAG;MACX;IACF;IACA,OAAOwgB,OAAO;EAChB;EAGA,CAAC0O,eAAeqB,CAAC9vB,IAAI,EAAE;IAErB,IAAI,CAACA,IAAI,EACP;IAEF,IAAI,CAACytB,cAAc,EAAE;IAGrB,IAAI,IAAI,CAACsC,YAAY,EAAE;MACrB,IAAI,CAACA,YAAY,CAAC/vB,IAAI,CAAC;IACzB;IAEA,IAAIA,IAAI,KAAK,GAAG,EAAE;MAEhB,IAAI,IAAI,CAACgwB,cAAc,EAAE;QACvB,IAAI,CAACA,cAAc,CAAC,CAAC;MACvB;MAEA;IACF;IAEA,IAAItxB,GAAG,GAAGC,IAAI,CAACC,KAAK,CAACoB,IAAI,EAAE3G,sDAAe,CAAC;IAC3C,IAAI,CAACqF,GAAG,EAAE;MACR,IAAI,CAAClD,MAAM,CAAC,MAAM,GAAGwE,IAAI,CAAC;MAC1B,IAAI,CAACxE,MAAM,CAAC,6BAA6B,CAAC;IAC5C,CAAC,MAAM;MACL,IAAI,CAACA,MAAM,CAAC,MAAM,IAAI,IAAI,CAAC8xB,gBAAgB,GAAG3uB,IAAI,CAACkxB,SAAS,CAACnxB,GAAG,EAAE0tB,gBAAgB,CAAC,GAAGpsB,IAAI,CAAC,CAAC;MAG5F,IAAI,IAAI,CAACb,SAAS,EAAE;QAClB,IAAI,CAACA,SAAS,CAACT,GAAG,CAAC;MACrB;MAEA,IAAIA,GAAG,CAACI,IAAI,EAAE;QAEZ,IAAI,IAAI,CAACmxB,aAAa,EAAE;UACtB,IAAI,CAACA,aAAa,CAACvxB,GAAG,CAACI,IAAI,CAAC;QAC9B;QAGA,IAAIJ,GAAG,CAACI,IAAI,CAACwK,EAAE,EAAE;UACf,IAAI,CAAC,CAACkmB,WAAW,CAAC9wB,GAAG,CAACI,IAAI,CAACwK,EAAE,EAAE5K,GAAG,CAACI,IAAI,CAACvH,IAAI,EAAEmH,GAAG,CAACI,IAAI,EAAEJ,GAAG,CAACI,IAAI,CAACO,IAAI,CAAC;QACxE;QACAtC,UAAU,CAACxC,CAAC,IAAI;UACd,IAAImE,GAAG,CAACI,IAAI,CAACvH,IAAI,IAAI,GAAG,IAAImH,GAAG,CAACI,IAAI,CAACO,IAAI,IAAI,SAAS,EAAE;YAEtD,MAAM+B,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAEnwB,GAAG,CAACI,IAAI,CAACsC,KAAK,CAAC;YACrD,IAAIA,KAAK,EAAE;cACTA,KAAK,CAACke,SAAS,CAAC,CAAC;cACjB,IAAI5gB,GAAG,CAACI,IAAI,CAACC,MAAM,IAAIL,GAAG,CAACI,IAAI,CAACC,MAAM,CAACsgB,KAAK,EAAE;gBAC5Cje,KAAK,CAACme,KAAK,CAAC,CAAC;cACf;YACF;UACF,CAAC,MAAM,IAAI7gB,GAAG,CAACI,IAAI,CAACvH,IAAI,GAAG,GAAG,IAAImH,GAAG,CAACI,IAAI,CAACC,MAAM,EAAE;YACjD,IAAIL,GAAG,CAACI,IAAI,CAACC,MAAM,CAACka,IAAI,IAAI,MAAM,EAAE;cAElC,MAAM7X,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAEnwB,GAAG,CAACI,IAAI,CAACsC,KAAK,CAAC;cACrD,IAAIA,KAAK,EAAE;gBACTA,KAAK,CAAC4lB,oBAAoB,CAACtoB,GAAG,CAACI,IAAI,CAACC,MAAM,CAACpJ,KAAK,CAAC;cACnD;YACF,CAAC,MAAM,IAAI+I,GAAG,CAACI,IAAI,CAACC,MAAM,CAACka,IAAI,IAAI,KAAK,EAAE;cAExC,MAAM7X,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAEnwB,GAAG,CAACI,IAAI,CAACsC,KAAK,CAAC;cACrD,IAAIA,KAAK,EAAE;gBAETA,KAAK,CAAC6e,gBAAgB,CAAC,EAAE,CAAC;cAC5B;YACF;UACF;QACF,CAAC,EAAE,CAAC,CAAC;MACP,CAAC,MAAM;QACLljB,UAAU,CAACxC,CAAC,IAAI;UACd,IAAImE,GAAG,CAACynB,IAAI,EAAE;YAGZ,MAAM/kB,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAEnwB,GAAG,CAACynB,IAAI,CAAC/kB,KAAK,CAAC;YACrD,IAAIA,KAAK,EAAE;cACTA,KAAK,CAAC8kB,UAAU,CAACxnB,GAAG,CAACynB,IAAI,CAAC;YAC5B;YAEA,IAAIznB,GAAG,CAACynB,IAAI,CAAC7c,EAAE,EAAE;cACf,IAAI,CAAC,CAACkmB,WAAW,CAAC9wB,GAAG,CAACynB,IAAI,CAAC7c,EAAE,EAAE,GAAG,EAAE5K,GAAG,CAACynB,IAAI,EAAE,MAAM,CAAC;YACvD;YAGA,IAAI,IAAI,CAAC+J,aAAa,EAAE;cACtB,IAAI,CAACA,aAAa,CAACxxB,GAAG,CAACynB,IAAI,CAAC;YAC9B;UACF,CAAC,MAAM,IAAIznB,GAAG,CAACsB,IAAI,EAAE;YAGnB,MAAMoB,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAEnwB,GAAG,CAACsB,IAAI,CAACoB,KAAK,CAAC;YACrD,IAAIA,KAAK,EAAE;cACTA,KAAK,CAACwd,UAAU,CAAClgB,GAAG,CAACsB,IAAI,CAAC;YAC5B;YAGA,IAAI,IAAI,CAACmwB,aAAa,EAAE;cACtB,IAAI,CAACA,aAAa,CAACzxB,GAAG,CAACsB,IAAI,CAAC;YAC9B;UACF,CAAC,MAAM,IAAItB,GAAG,CAAC8nB,IAAI,EAAE;YAGnB,MAAMplB,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAEnwB,GAAG,CAAC8nB,IAAI,CAACplB,KAAK,CAAC;YACrD,IAAIA,KAAK,EAAE;cACTA,KAAK,CAACmlB,UAAU,CAAC7nB,GAAG,CAAC8nB,IAAI,CAAC;YAC5B;YAGA,IAAI,IAAI,CAAC4J,aAAa,EAAE;cACtB,IAAI,CAACA,aAAa,CAAC1xB,GAAG,CAAC8nB,IAAI,CAAC;YAC9B;UACF,CAAC,MAAM,IAAI9nB,GAAG,CAACkoB,IAAI,EAAE;YAGnB,MAAMxlB,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAEnwB,GAAG,CAACkoB,IAAI,CAACxlB,KAAK,CAAC;YACrD,IAAIA,KAAK,EAAE;cACTA,KAAK,CAAC6kB,UAAU,CAACvnB,GAAG,CAACkoB,IAAI,CAAC;YAC5B;YAGA,IAAI,IAAI,CAACyJ,aAAa,EAAE;cACtB,IAAI,CAACA,aAAa,CAAC3xB,GAAG,CAACkoB,IAAI,CAAC;YAC9B;UACF,CAAC,MAAM;YACL,IAAI,CAACprB,MAAM,CAAC,iCAAiC,CAAC;UAChD;QACF,CAAC,EAAE,CAAC,CAAC;MACP;IACF;EACF;EAGA,CAACkzB,cAAc4B,CAAA,EAAG;IAChB,IAAI,CAAC,IAAI,CAACvC,eAAe,EAAE;MAEzB,IAAI,CAACA,eAAe,GAAGwC,WAAW,CAACh2B,CAAC,IAAI;QACtC,MAAMgF,GAAG,GAAG,IAAIlI,sDAAS,CAAC,SAAS,EAAE,GAAG,CAAC;QACzC,MAAMm5B,OAAO,GAAG,IAAIhV,IAAI,CAAC,IAAIA,IAAI,CAAC,CAAC,CAAC8N,OAAO,CAAC,CAAC,GAAGtO,+DAA6B,CAAC;QAC9E,KAAK,IAAI1R,EAAE,IAAI,IAAI,CAACwkB,gBAAgB,EAAE;UACpC,IAAIzS,SAAS,GAAG,IAAI,CAACyS,gBAAgB,CAACxkB,EAAE,CAAC;UACzC,IAAI+R,SAAS,IAAIA,SAAS,CAACyC,EAAE,GAAG0S,OAAO,EAAE;YACvC,IAAI,CAACh1B,MAAM,CAAC,iBAAiB,EAAE8N,EAAE,CAAC;YAClC,OAAO,IAAI,CAACwkB,gBAAgB,CAACxkB,EAAE,CAAC;YAChC,IAAI+R,SAAS,CAACvf,MAAM,EAAE;cACpBuf,SAAS,CAACvf,MAAM,CAACyD,GAAG,CAAC;YACvB;UACF;QACF;MACF,CAAC,EAAEyb,8DAA4B,CAAC;IAClC;IACA,IAAI,CAACyV,KAAK,CAAC,CAAC;EACd;EAEA,CAAC9B,YAAY+B,CAACnxB,GAAG,EAAEhI,IAAI,EAAE;IACvB,IAAI,CAACk2B,cAAc,GAAG,CAAC;IACvB,IAAI,CAACG,WAAW,GAAG,IAAI;IACvB,IAAI,CAACL,cAAc,GAAG,KAAK;IAE3B,IAAI,IAAI,CAACQ,eAAe,EAAE;MACxB4C,aAAa,CAAC,IAAI,CAAC5C,eAAe,CAAC;MACnC,IAAI,CAACA,eAAe,GAAG,IAAI;IAC7B;IAGA,IAAI,CAAC,CAAC6C,QAAQ,CAAC,OAAO,EAAE,CAACxvB,KAAK,EAAEoL,GAAG,KAAK;MACtCpL,KAAK,CAACke,SAAS,CAAC,CAAC;IACnB,CAAC,CAAC;IAGF,KAAK,IAAI9S,GAAG,IAAI,IAAI,CAACshB,gBAAgB,EAAE;MACrC,MAAMzS,SAAS,GAAG,IAAI,CAACyS,gBAAgB,CAACthB,GAAG,CAAC;MAC5C,IAAI6O,SAAS,IAAIA,SAAS,CAACvf,MAAM,EAAE;QACjCuf,SAAS,CAACvf,MAAM,CAACyD,GAAG,CAAC;MACvB;IACF;IACA,IAAI,CAACuuB,gBAAgB,GAAG,CAAC,CAAC;IAE1B,IAAI,IAAI,CAAC1uB,YAAY,EAAE;MACrB,IAAI,CAACA,YAAY,CAACG,GAAG,CAAC;IACxB;EACF;EAGA,CAACsxB,YAAYC,CAAA,EAAG;IACd,OAAO,IAAI,CAAC9D,QAAQ,GAAG,IAAI,IAAI,IAAI,CAACC,QAAQ,GAAG,IAAI,CAACA,QAAQ,GAAG,IAAI,GAAG,EAAE,CAAC,GAAG,IAAI,CAACE,KAAK,GAAG,KAAK,GAAGnS,+CAAa;EAChH;EAGA,CAAC+V,UAAUC,CAAC/nB,IAAI,EAAE7H,KAAK,EAAE;IACvB,QAAQ6H,IAAI;MACV,KAAK,IAAI;QACP,OAAO;UACL,IAAI,EAAE;YACJ,IAAI,EAAE,IAAI,CAAC0O,eAAe,CAAC,CAAC;YAC5B,KAAK,EAAEqD,+CAAa;YACpB,IAAI,EAAE,IAAI,CAAC,CAAC6V,YAAY,CAAC,CAAC;YAC1B,KAAK,EAAE,IAAI,CAAChD,YAAY;YACxB,MAAM,EAAE,IAAI,CAACT,cAAc;YAC3B,OAAO,EAAE,IAAI,CAACF;UAChB;QACF,CAAC;MAEH,KAAK,KAAK;QACR,OAAO;UACL,KAAK,EAAE;YACL,IAAI,EAAE,IAAI,CAACvV,eAAe,CAAC,CAAC;YAC5B,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;YACd,QAAQ,EAAE,IAAI;YACd,WAAW,EAAE,IAAI;YACjB,WAAW,EAAE,IAAI;YACjB,OAAO,EAAE,KAAK;YACd,MAAM,EAAE,IAAI;YACZ,MAAM,EAAE,CAAC,CAAC;YACV,MAAM,EAAE,CAAC;UACX;QACF,CAAC;MAEH,KAAK,OAAO;QACV,OAAO;UACL,OAAO,EAAE;YACP,IAAI,EAAE,IAAI,CAACA,eAAe,CAAC,CAAC;YAC5B,QAAQ,EAAE,IAAI;YACd,QAAQ,EAAE;UACZ;QACF,CAAC;MAEH,KAAK,KAAK;QACR,OAAO;UACL,KAAK,EAAE;YACL,IAAI,EAAE,IAAI,CAACA,eAAe,CAAC,CAAC;YAC5B,OAAO,EAAEvW,KAAK;YACd,KAAK,EAAE,CAAC,CAAC;YACT,KAAK,EAAE,CAAC;UACV;QACF,CAAC;MAEH,KAAK,OAAO;QACV,OAAO;UACL,OAAO,EAAE;YACP,IAAI,EAAE,IAAI,CAACuW,eAAe,CAAC,CAAC;YAC5B,OAAO,EAAEvW,KAAK;YACd,OAAO,EAAE;UACX;QACF,CAAC;MAEH,KAAK,KAAK;QACR,OAAO;UACL,KAAK,EAAE;YACL,IAAI,EAAE,IAAI,CAACuW,eAAe,CAAC,CAAC;YAC5B,OAAO,EAAEvW,KAAK;YACd,QAAQ,EAAE,KAAK;YACf,MAAM,EAAE,IAAI;YACZ,SAAS,EAAE,CAAC;UACd;QACF,CAAC;MAEH,KAAK,KAAK;QACR,OAAO;UACL,KAAK,EAAE;YACL,IAAI,EAAE,IAAI,CAACuW,eAAe,CAAC,CAAC;YAC5B,OAAO,EAAEvW,KAAK;YACd,MAAM,EAAE,IAAI;YACZ,MAAM,EAAE,CAAC,CAAC;YACV,KAAK,EAAE,CAAC,CAAC;YACT,MAAM,EAAE,CAAC;UACX;QACF,CAAC;MAEH,KAAK,KAAK;QACR,OAAO;UACL,KAAK,EAAE;YACL,IAAI,EAAE,IAAI,CAACuW,eAAe,CAAC,CAAC;YAC5B,OAAO,EAAEvW,KAAK;YACd,MAAM,EAAE,CAAC,CAAC;YACV,KAAK,EAAE,CAAC,CAAC;YACT,MAAM,EAAE,EAAE;YACV,WAAW,EAAE,CAAC;UAChB;QACF,CAAC;MAEH,KAAK,KAAK;QACR,OAAO;UACL,KAAK,EAAE;YACL,IAAI,EAAE,IAAI,CAACuW,eAAe,CAAC,CAAC;YAC5B,OAAO,EAAEvW,KAAK;YACd,MAAM,EAAE,IAAI;YACZ,QAAQ,EAAE,IAAI;YACd,MAAM,EAAE,IAAI;YACZ,MAAM,EAAE;UACV;QACF,CAAC;MAEH,KAAK,MAAM;QACT,OAAO;UACL,MAAM,EAAE;YAEN,OAAO,EAAEA,KAAK;YACd,MAAM,EAAE,IAAI;YACZ,KAAK,EAAE5M;UACT;QACF,CAAC;MAEH;QACE,MAAM,IAAI7D,KAAK,CAAE,kCAAiCsY,IAAK,EAAC,CAAC;IAC7D;EACF;EAGA,CAAC8lB,QAAQkC,CAAChoB,IAAI,EAAEzR,IAAI,EAAEyvB,GAAG,EAAE;IACzB,IAAI,CAACiH,MAAM,CAACjlB,IAAI,GAAG,GAAG,GAAGzR,IAAI,CAAC,GAAGyvB,GAAG;EACtC;EACA,CAAC4H,QAAQqC,CAACjoB,IAAI,EAAEzR,IAAI,EAAE;IACpB,OAAO,IAAI,CAAC02B,MAAM,CAACjlB,IAAI,GAAG,GAAG,GAAGzR,IAAI,CAAC;EACvC;EACA,CAAC25B,QAAQC,CAACnoB,IAAI,EAAEzR,IAAI,EAAE;IACpB,OAAO,IAAI,CAAC02B,MAAM,CAACjlB,IAAI,GAAG,GAAG,GAAGzR,IAAI,CAAC;EACvC;EAIA,CAACo5B,QAAQS,CAACpoB,IAAI,EAAEqoB,IAAI,EAAEv6B,OAAO,EAAE;IAC7B,MAAMyV,GAAG,GAAGvD,IAAI,GAAGA,IAAI,GAAG,GAAG,GAAGzU,SAAS;IACzC,KAAK,IAAIgB,GAAG,IAAI,IAAI,CAAC04B,MAAM,EAAE;MAC3B,IAAI,CAAC1hB,GAAG,IAAIhX,GAAG,CAAC4Z,OAAO,CAAC5C,GAAG,CAAC,IAAI,CAAC,EAAE;QACjC,IAAI8kB,IAAI,CAACt6B,IAAI,CAACD,OAAO,EAAE,IAAI,CAACm3B,MAAM,CAAC14B,GAAG,CAAC,EAAEA,GAAG,CAAC,EAAE;UAC7C;QACF;MACF;IACF;EACF;EAIA,CAACs5B,kBAAkByC,CAACnwB,KAAK,EAAE;IACzBA,KAAK,CAACuU,OAAO,GAAG,IAAI;IAEpBvU,KAAK,CAACwhB,aAAa,GAAI5f,GAAG,IAAK;MAC7B,MAAMC,GAAG,GAAG,IAAI,CAAC,CAAC4rB,QAAQ,CAAC,MAAM,EAAE7rB,GAAG,CAAC;MACvC,IAAIC,GAAG,EAAE;QACP,OAAO;UACLK,IAAI,EAAEN,GAAG;UACTE,MAAM,EAAE+X,mDAAQ,CAAC,CAAC,CAAC,EAAEhY,GAAG;QAC1B,CAAC;MACH;MACA,OAAOzO,SAAS;IAClB,CAAC;IACD4M,KAAK,CAAC+lB,aAAa,GAAG,CAACnkB,GAAG,EAAEM,IAAI,KAAK;MACnC,IAAI,CAAC,CAACyrB,QAAQ,CAAC,MAAM,EAAE/rB,GAAG,EAAEiY,mDAAQ,CAAC,CAAC,CAAC,EAAE3X,IAAI,CAACJ,MAAM,CAAC,CAAC;IACxD,CAAC;IACD9B,KAAK,CAACowB,aAAa,GAAIxuB,GAAG,IAAK;MAC7B,IAAI,CAAC,CAACmuB,QAAQ,CAAC,MAAM,EAAEnuB,GAAG,CAAC;IAC7B,CAAC;IACD5B,KAAK,CAACyc,aAAa,GAAGtjB,CAAC,IAAI;MACzB,IAAI,CAAC,CAACw0B,QAAQ,CAAC,OAAO,EAAE3tB,KAAK,CAAC5J,IAAI,EAAE4J,KAAK,CAAC;IAC5C,CAAC;IACDA,KAAK,CAACwc,aAAa,GAAGrjB,CAAC,IAAI;MACzB,IAAI,CAAC,CAAC42B,QAAQ,CAAC,OAAO,EAAE/vB,KAAK,CAAC5J,IAAI,CAAC;IACrC,CAAC;EACH;EAGA,CAACi6B,eAAeC,CAAC5yB,IAAI,EAAE;IACrB,IAAI,CAACA,IAAI,CAACC,MAAM,IAAI,CAACD,IAAI,CAACC,MAAM,CAACuE,IAAI,EAAE;MACrC,OAAOxE,IAAI;IACb;IAGA,IAAI,CAAC0oB,MAAM,GAAG1oB,IAAI,CAACC,MAAM,CAACuE,IAAI;IAC9B,IAAI,CAACiqB,cAAc,GAAIzuB,IAAI,IAAIA,IAAI,CAACvH,IAAI,IAAI,GAAG,IAAIuH,IAAI,CAACvH,IAAI,GAAG,GAAI;IACnE,IAAIuH,IAAI,CAACC,MAAM,IAAID,IAAI,CAACC,MAAM,CAAC4X,KAAK,IAAI7X,IAAI,CAACC,MAAM,CAACyxB,OAAO,EAAE;MAC3D,IAAI,CAAC1a,UAAU,GAAG;QAChBa,KAAK,EAAE7X,IAAI,CAACC,MAAM,CAAC4X,KAAK;QACxB6Z,OAAO,EAAE1xB,IAAI,CAACC,MAAM,CAACyxB;MACvB,CAAC;IACH,CAAC,MAAM;MACL,IAAI,CAAC1a,UAAU,GAAG,IAAI;IACxB;IAEA,IAAI,IAAI,CAAC6b,OAAO,EAAE;MAChB,IAAI,CAACA,OAAO,CAAC7yB,IAAI,CAACvH,IAAI,EAAEuH,IAAI,CAACO,IAAI,CAAC;IACpC;IAEA,OAAOP,IAAI;EACb;EAaA,OAAO8yB,UAAUA,CAAC3J,IAAI,EAAE13B,GAAG,EAAEwO,MAAM,EAAEopB,IAAI,EAAE;IACzC,IAAI,OAAOF,IAAI,IAAI,QAAQ,EAAE;MAC3B,CAAC;QACC13B,GAAG;QACHwO,MAAM;QACNopB,IAAI;QACJF;MACF,CAAC,GAAGA,IAAI;IACV;IACA,IAAIA,IAAI,KAAK13B,GAAG,IAAI43B,IAAI,CAAC,EAAE;MACzB,OAAO,CAAC;QACN,MAAM,EAAEF,IAAI;QACZ,KAAK,EAAE13B,GAAG;QACV,MAAM,EAAE43B,IAAI;QACZ,QAAQ,EAAEppB;MACZ,CAAC,CAAC;IACJ;IACA,OAAO,IAAI;EACb;EAQA,OAAOie,SAASA,CAACxlB,IAAI,EAAE;IACrB,OAAO4jB,4CAAK,CAAC4B,SAAS,CAACxlB,IAAI,CAAC;EAC9B;EAOA,OAAO0lB,aAAaA,CAAC1lB,IAAI,EAAE;IACzB,OAAO4jB,4CAAK,CAAC8B,aAAa,CAAC1lB,IAAI,CAAC;EAClC;EAMA,OAAO2lB,gBAAgBA,CAAC3lB,IAAI,EAAE;IAC5B,OAAO4jB,4CAAK,CAAC+B,gBAAgB,CAAC3lB,IAAI,CAAC;EACrC;EAMA,OAAO4lB,cAAcA,CAAC5lB,IAAI,EAAE;IAC1B,OAAO4jB,4CAAK,CAACgC,cAAc,CAAC5lB,IAAI,CAAC;EACnC;EAMA,OAAO6lB,eAAeA,CAAC7lB,IAAI,EAAE;IAC3B,OAAO4jB,4CAAK,CAACiC,eAAe,CAAC7lB,IAAI,CAAC;EACpC;EAMA,OAAO8lB,mBAAmBA,CAAC9lB,IAAI,EAAE;IAC/B,OAAO4jB,4CAAK,CAACkC,mBAAmB,CAAC9lB,IAAI,CAAC;EACxC;EAMA,OAAO+lB,kBAAkBA,CAAC/lB,IAAI,EAAE;IAC9B,OAAO4jB,4CAAK,CAACmC,kBAAkB,CAAC/lB,IAAI,CAAC;EACvC;EAKA,OAAOq6B,UAAUA,CAAA,EAAG;IAClB,OAAO7W,+CAAa;EACtB;EAQA,OAAO3f,mBAAmBA,CAACC,UAAU,EAAEC,WAAW,EAAE;IAClDjC,iBAAiB,GAAGgC,UAAU;IAC9B/B,WAAW,GAAGgC,WAAW;IAEzBlB,sDAAU,CAACgB,mBAAmB,CAAC/B,iBAAiB,EAAEC,WAAW,CAAC;IAC9Dkc,sDAAe,CAACsD,kBAAkB,CAACxf,WAAW,CAAC;EACjD;EAOA,OAAOuM,mBAAmBA,CAACC,WAAW,EAAE;IACtCklB,iBAAiB,GAAGllB,WAAW;IAE/B+kB,8CAAO,CAAChlB,mBAAmB,CAACmlB,iBAAiB,CAAC;EAChD;EAOA,OAAO6G,UAAUA,CAAA,EAAG;IAClB,OAAO9W,+CAAa;EACtB;EAMA,OAAO+W,WAAWA,CAACnhC,GAAG,EAAE;IACtB,OAAOA,GAAG,KAAKoqB,gDAAc;EAC/B;EAKArD,eAAeA,CAAA,EAAG;IAChB,OAAQ,IAAI,CAAC+V,UAAU,IAAI,CAAC,GAAI,EAAE,GAAG,IAAI,CAACA,UAAU,EAAE,GAAGl5B,SAAS;EACpE;EAUAkH,OAAOA,CAACC,KAAK,EAAE;IACb,OAAO,IAAI,CAACqyB,WAAW,CAACtyB,OAAO,CAACC,KAAK,CAAC;EACxC;EAOAI,SAASA,CAACH,KAAK,EAAE;IACf,IAAI,CAACoyB,WAAW,CAACjyB,SAAS,CAACH,KAAK,CAAC;EACnC;EAKAI,UAAUA,CAAA,EAAG;IACX,IAAI,CAACgyB,WAAW,CAAChyB,UAAU,CAAC,CAAC;EAC/B;EAOAg2B,YAAYA,CAAA,EAAG;IACb,IAAI,IAAI,CAAC/S,GAAG,CAACrd,OAAO,CAAC,CAAC,EAAE;MACtB,OAAO,IAAI,CAACqd,GAAG,CAACvd,cAAc,CAAC,CAAC;IAClC;IACA,OAAO7F,OAAO,CAAC0C,OAAO,CAAC,CAAC;EAC1B;EAOA0zB,WAAWA,CAAA,EAAG;IACZ,IAAI,CAAC,IAAI,CAAChT,GAAG,CAACrd,OAAO,CAAC,CAAC,EAAE;MACvB,OAAO,IAAI,CAACqd,GAAG,CAAC5d,YAAY,CAAC,CAAC;IAChC;IACA,OAAOxF,OAAO,CAAC0C,OAAO,CAAC,CAAC;EAC1B;EAKA2zB,YAAYA,CAAA,EAAG;IACb,IAAI,CAAClE,WAAW,CAAC5xB,KAAK,CAAC,CAAC;EAC1B;EAOAD,WAAWA,CAAA,EAAG;IACZ,OAAO,IAAI,CAAC6xB,WAAW,CAAC7xB,WAAW,CAAC,CAAC;EACvC;EAOAg2B,eAAeA,CAAA,EAAG;IAChB,OAAO,IAAI,CAAC5E,cAAc;EAC5B;EASA6E,YAAYA,CAACh4B,GAAG,EAAE;IAChB,IAAI,OAAOA,GAAG,IAAI,QAAQ,EAAE;MAC1B,OAAOA,GAAG;IACZ;IAEA,IAAI8a,wDAAa,CAAC9a,GAAG,CAAC,EAAE;MAEtB,MAAMmc,IAAI,GAAG,gBAAgB;MAC7B,MAAM8b,MAAM,GAAG,IAAIvpB,GAAG,CAAC1O,GAAG,EAAEmc,IAAI,CAAC;MACjC,IAAI,IAAI,CAACV,OAAO,EAAE;QAChBwc,MAAM,CAAC7c,YAAY,CAAC7I,MAAM,CAAC,QAAQ,EAAE,IAAI,CAACkJ,OAAO,CAAC;MACpD;MACA,IAAI,IAAI,CAACC,UAAU,IAAI,IAAI,CAACA,UAAU,CAACa,KAAK,EAAE;QAC5C0b,MAAM,CAAC7c,YAAY,CAAC7I,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC;QAC3C0lB,MAAM,CAAC7c,YAAY,CAAC7I,MAAM,CAAC,QAAQ,EAAE,IAAI,CAACmJ,UAAU,CAACa,KAAK,CAAC;MAC7D;MAEAvc,GAAG,GAAGi4B,MAAM,CAAC3/B,QAAQ,CAAC,CAAC,CAAC8gB,SAAS,CAAC+C,IAAI,CAAC7kB,MAAM,GAAG,CAAC,CAAC;IACpD;IACA,OAAO0I,GAAG;EACZ;EAgCAk4B,OAAOA,CAACtvB,GAAG,EAAEuvB,MAAM,EAAEC,MAAM,EAAEC,KAAK,EAAE1zB,MAAM,EAAE;IAC1C,MAAML,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,CAAC;IACnCryB,GAAG,CAACg0B,GAAG,CAACpvB,IAAI,GAAGN,GAAG;IAClBtE,GAAG,CAACg0B,GAAG,CAACH,MAAM,GAAGA,MAAM;IACvB7zB,GAAG,CAACg0B,GAAG,CAACF,MAAM,GAAGA,MAAM;IAEvB9zB,GAAG,CAACg0B,GAAG,CAACD,KAAK,GAAGA,KAAK;IAErB,IAAI1zB,MAAM,EAAE;MACVL,GAAG,CAACg0B,GAAG,CAACzU,IAAI,CAACiH,MAAM,GAAGnmB,MAAM,CAACmmB,MAAM;MACnCxmB,GAAG,CAACg0B,GAAG,CAACzU,IAAI,CAAC/a,MAAM,GAAGnE,MAAM,CAACmE,MAAM;MACnCxE,GAAG,CAACg0B,GAAG,CAACzU,IAAI,CAACxC,OAAO,GAAG1c,MAAM,CAAC0c,OAAO;MACrC/c,GAAG,CAACg0B,GAAG,CAACzU,IAAI,CAACvC,OAAO,GAAG3c,MAAM,CAAC2c,OAAO;MAErChd,GAAG,CAACg0B,GAAG,CAACxtB,IAAI,GAAGnG,MAAM,CAACmG,IAAI;MAC1BxG,GAAG,CAACg0B,GAAG,CAACvS,IAAI,GAAGphB,MAAM,CAACohB,IAAI;MAE1BzhB,GAAG,CAACg0B,GAAG,CAACC,SAAS,GAAG5zB,MAAM,CAACwzB,MAAM;MACjC7zB,GAAG,CAACg0B,GAAG,CAACE,SAAS,GAAG7zB,MAAM,CAACyzB,MAAM;MAEjC,IAAIr8B,KAAK,CAACC,OAAO,CAAC2I,MAAM,CAAC4S,WAAW,CAAC,IAAI5S,MAAM,CAAC4S,WAAW,CAACjgB,MAAM,GAAG,CAAC,EAAE;QACtEgN,GAAG,CAACm0B,KAAK,GAAG;UACVlhB,WAAW,EAAE5S,MAAM,CAAC4S,WAAW,CAACxa,MAAM,CAACqS,GAAG,IAAI0L,wDAAa,CAAC1L,GAAG,CAAC;QAClE,CAAC;MACH;IACF;IAEA,OAAO,IAAI,CAAC,CAACvK,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACg0B,GAAG,CAACppB,EAAE,CAAC;EACpC;EAYAwpB,aAAaA,CAACP,MAAM,EAAEC,MAAM,EAAEC,KAAK,EAAE1zB,MAAM,EAAE;IAC3C,IAAIghB,OAAO,GAAG,IAAI,CAACuS,OAAO,CAACtX,gDAAc,EAAEuX,MAAM,EAAEC,MAAM,EAAEC,KAAK,EAAE1zB,MAAM,CAAC;IACzE,IAAI0zB,KAAK,EAAE;MACT1S,OAAO,GAAGA,OAAO,CAACzS,IAAI,CAACxO,IAAI,IAAI,IAAI,CAAC,CAAC2yB,eAAe,CAAC3yB,IAAI,CAAC,CAAC;IAC7D;IACA,OAAOihB,OAAO;EAChB;EAYAgT,kBAAkBA,CAACC,QAAQ,EAAEC,QAAQ,EAAEl0B,MAAM,EAAE;IAE7Ci0B,QAAQ,GAAGA,QAAQ,IAAI,EAAE;IACzBC,QAAQ,GAAGA,QAAQ,IAAI,EAAE;IACzB,OAAO,IAAI,CAACH,aAAa,CAAC,OAAO,EAC/B/G,gBAAgB,CAACiH,QAAQ,GAAG,GAAG,GAAGC,QAAQ,CAAC,EAAE,IAAI,EAAEl0B,MAAM,CAAC;EAC9D;EAYAm0B,kBAAkBA,CAAClwB,GAAG,EAAEgwB,QAAQ,EAAEC,QAAQ,EAAEl0B,MAAM,EAAE;IAElDi0B,QAAQ,GAAGA,QAAQ,IAAI,EAAE;IACzBC,QAAQ,GAAGA,QAAQ,IAAI,EAAE;IACzB,OAAO,IAAI,CAACX,OAAO,CAACtvB,GAAG,EAAE,OAAO,EAC9B+oB,gBAAgB,CAACiH,QAAQ,GAAG,GAAG,GAAGC,QAAQ,CAAC,EAAE,KAAK,EAAEl0B,MAAM,CAAC;EAC/D;EAOA0xB,KAAKA,CAAA,EAAG;IACN,MAAM/xB,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,IAAI,CAAC;IAElC,OAAO,IAAI,CAAC,CAAC9xB,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACqiB,EAAE,CAACzX,EAAE,CAAC,CAC9BgE,IAAI,CAACxO,IAAI,IAAI;MAEZ,IAAI,CAACkvB,WAAW,CAAC3xB,YAAY,CAAC,CAAC;MAI/B,IAAIyC,IAAI,CAACC,MAAM,EAAE;QACf,IAAI,CAAC6uB,WAAW,GAAG9uB,IAAI,CAACC,MAAM;MAChC;MAEA,IAAI,IAAI,CAACo0B,SAAS,EAAE;QAClB,IAAI,CAACA,SAAS,CAAC,CAAC;MAClB;MAEA,OAAOr0B,IAAI;IACb,CAAC,CAAC,CAAC7B,KAAK,CAACsC,GAAG,IAAI;MACd,IAAI,CAACyuB,WAAW,CAACjyB,SAAS,CAAC,IAAI,CAAC;MAEhC,IAAI,IAAI,CAACqD,YAAY,EAAE;QACrB,IAAI,CAACA,YAAY,CAACG,GAAG,CAAC;MACxB;IACF,CAAC,CAAC;EACN;EAWA6zB,cAAcA,CAACC,EAAE,EAAE;IACjB,IAAIC,IAAI,GAAG,KAAK;IAEhBD,EAAE,GAAGA,EAAE,IAAI,IAAI;IACf,IAAIA,EAAE,IAAI,IAAI,CAACxF,YAAY,EAAE;MAC3B,IAAI,CAACA,YAAY,GAAGwF,EAAE;MACtB,IAAI,IAAI,CAACl3B,WAAW,CAAC,CAAC,IAAI,IAAI,CAACg2B,eAAe,CAAC,CAAC,EAAE;QAChD,IAAI,CAAC,CAAClzB,IAAI,CAAC;UACT,IAAI,EAAE;YACJ,KAAK,EAAEo0B,EAAE,IAAItG,MAAM,CAAC3zB;UACtB;QACF,CAAC,CAAC;QACFk6B,IAAI,GAAG,IAAI;MACb;IACF;IACA,OAAOA,IAAI;EACb;EAmBAb,KAAKA,CAACF,MAAM,EAAEC,MAAM,EAAErS,IAAI,EAAE;IAC1B,MAAMzhB,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,OAAO,CAAC;IACrCryB,GAAG,CAAC+zB,KAAK,CAACF,MAAM,GAAGA,MAAM;IACzB7zB,GAAG,CAAC+zB,KAAK,CAACD,MAAM,GAAGA,MAAM;IACzB9zB,GAAG,CAAC+zB,KAAK,CAACtS,IAAI,GAAGA,IAAI;IAErB,OAAO,IAAI,CAAC,CAAClhB,IAAI,CAACP,GAAG,EAAEA,GAAG,CAAC+zB,KAAK,CAACnpB,EAAE,CAAC,CACjCgE,IAAI,CAACxO,IAAI,IAAI,IAAI,CAAC,CAAC2yB,eAAe,CAAC3yB,IAAI,CAAC,CAAC;EAC9C;EAWAy0B,UAAUA,CAACC,KAAK,EAAEP,QAAQ,EAAE9S,IAAI,EAAE;IAChC,OAAO,IAAI,CAACsS,KAAK,CAAC,OAAO,EAAE1G,gBAAgB,CAACyH,KAAK,GAAG,GAAG,GAAGP,QAAQ,CAAC,EAAE9S,IAAI,CAAC,CACvE7S,IAAI,CAACxO,IAAI,IAAI;MACZ,IAAI,CAAC0uB,MAAM,GAAGgG,KAAK;MACnB,OAAO10B,IAAI;IACb,CAAC,CAAC;EACN;EAUA20B,UAAUA,CAAC9c,KAAK,EAAEwJ,IAAI,EAAE;IACtB,OAAO,IAAI,CAACsS,KAAK,CAAC,OAAO,EAAE9b,KAAK,EAAEwJ,IAAI,CAAC;EACzC;EAWAuT,sBAAsBA,CAACnB,MAAM,EAAE9J,MAAM,EAAE9jB,KAAK,EAAE;IAC5C,OAAO,IAAI,CAAC8tB,KAAK,CAAC,OAAO,EAAE1G,gBAAgB,CAACwG,MAAM,GAAG,GAAG,GAAG9J,MAAM,GAAG,GAAG,GAAG9jB,KAAK,CAAC,CAAC;EACnF;EAaAoR,YAAYA,CAAA,EAAG;IACb,IAAI,IAAI,CAACD,UAAU,IAAK,IAAI,CAACA,UAAU,CAAC0a,OAAO,CAAClH,OAAO,CAAC,CAAC,GAAG9N,IAAI,CAACmY,GAAG,CAAC,CAAE,EAAE;MACvE,OAAO,IAAI,CAAC7d,UAAU;IACxB,CAAC,MAAM;MACL,IAAI,CAACA,UAAU,GAAG,IAAI;IACxB;IACA,OAAO,IAAI;EACb;EAOA8d,YAAYA,CAACjd,KAAK,EAAE;IAClB,IAAI,CAACb,UAAU,GAAGa,KAAK;EACzB;EAgCA8G,SAASA,CAACja,SAAS,EAAEka,SAAS,EAAEC,SAAS,EAAE;IACzC,MAAMjf,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,EAAEvtB,SAAS,CAAC;IAC9C,IAAI,CAACA,SAAS,EAAE;MACdA,SAAS,GAAGwX,iDAAe;IAC7B;IAEAtc,GAAG,CAAC+E,GAAG,CAAC1B,GAAG,GAAG2b,SAAS;IAEvB,IAAIC,SAAS,EAAE;MACb,IAAIA,SAAS,CAACla,GAAG,EAAE;QACjB/E,GAAG,CAAC+E,GAAG,CAACiU,GAAG,CAACjU,GAAG,GAAGka,SAAS,CAACla,GAAG;MACjC;MAEA,IAAIka,SAAS,CAACM,IAAI,EAAE;QAClB,MAAMA,IAAI,GAAGN,SAAS,CAACM,IAAI;QAC3B,IAAI8O,MAAM,CAACzP,mBAAmB,CAAC9Z,SAAS,CAAC,EAAE;UAEzC9E,GAAG,CAAC+E,GAAG,CAACiU,GAAG,CAACuG,IAAI,GAAGA,IAAI;QACzB,CAAC,MAAM,IAAI8O,MAAM,CAAC3P,cAAc,CAAC5Z,SAAS,CAAC,IAAIya,IAAI,CAACiH,MAAM,EAAE;UAE1DxmB,GAAG,CAAC+E,GAAG,CAACiU,GAAG,CAACuG,IAAI,GAAG;YACjBiH,MAAM,EAAEjH,IAAI,CAACiH;UACf,CAAC;QACH;MACF;MAGA,IAAI/uB,KAAK,CAACC,OAAO,CAACunB,SAAS,CAAChM,WAAW,CAAC,IAAIgM,SAAS,CAAChM,WAAW,CAACjgB,MAAM,GAAG,CAAC,EAAE;QAC5EgN,GAAG,CAACm0B,KAAK,GAAG;UACVlhB,WAAW,EAAEgM,SAAS,CAAChM,WAAW,CAACxa,MAAM,CAACqS,GAAG,IAAI0L,wDAAa,CAAC1L,GAAG,CAAC;QACrE,CAAC;MACH;MAEA,IAAImU,SAAS,CAACzY,IAAI,EAAE;QAClBxG,GAAG,CAAC+E,GAAG,CAACiU,GAAG,CAACxS,IAAI,GAAGyY,SAAS,CAACzY,IAAI;MACnC;IACF;IACA,OAAO,IAAI,CAAC,CAACjG,IAAI,CAACP,GAAG,EAAEA,GAAG,CAAC+E,GAAG,CAAC6F,EAAE,CAAC;EACpC;EAUA8V,KAAKA,CAAChe,KAAK,EAAEie,KAAK,EAAE;IAClB,MAAM3gB,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,OAAO,EAAE3vB,KAAK,CAAC;IAC5C1C,GAAG,CAAC0gB,KAAK,CAACC,KAAK,GAAGA,KAAK;IAEvB,OAAO,IAAI,CAAC,CAACpgB,IAAI,CAACP,GAAG,EAAEA,GAAG,CAAC0gB,KAAK,CAAC9V,EAAE,CAAC;EACtC;EAWA8U,aAAaA,CAAChd,KAAK,EAAE2J,OAAO,EAAEsT,MAAM,EAAE;IACpC,MAAM3f,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,EAAE3vB,KAAK,CAAC;IAE1C,IAAIyyB,GAAG,GAAG,OAAO9oB,OAAO,IAAI,QAAQ,GAAGrB,uDAAY,CAACqB,OAAO,CAAC,GAAGA,OAAO;IACtE,IAAI8oB,GAAG,IAAI,CAACnqB,6DAAkB,CAACmqB,GAAG,CAAC,EAAE;MACnCn1B,GAAG,CAACuE,GAAG,CAACwiB,IAAI,GAAG;QACbhc,IAAI,EAAEC,gEAAqB,CAAC;MAC9B,CAAC;MACDqB,OAAO,GAAG8oB,GAAG;IACf;IACAn1B,GAAG,CAACuE,GAAG,CAAC+b,MAAM,GAAGX,MAAM;IACvB3f,GAAG,CAACuE,GAAG,CAAC8H,OAAO,GAAGA,OAAO;IAEzB,OAAOrM,GAAG,CAACuE,GAAG;EAChB;EAWAqb,OAAOA,CAAC9a,SAAS,EAAEuH,OAAO,EAAEsT,MAAM,EAAE;IAClC,OAAO,IAAI,CAACE,cAAc,CACxB,IAAI,CAACH,aAAa,CAAC5a,SAAS,EAAEuH,OAAO,EAAEsT,MAAM,CAC/C,CAAC;EACH;EAUAE,cAAcA,CAACtb,GAAG,EAAE0O,WAAW,EAAE;IAE/B1O,GAAG,GAAG+K,MAAM,CAACC,MAAM,CAAC,CAAC,CAAC,EAAEhL,GAAG,CAAC;IAC5BA,GAAG,CAACe,GAAG,GAAGxP,SAAS;IACnByO,GAAG,CAACkB,IAAI,GAAG3P,SAAS;IACpByO,GAAG,CAAC6a,EAAE,GAAGtpB,SAAS;IAClB,MAAM0H,GAAG,GAAG;MACV+G,GAAG,EAAEA;IACP,CAAC;IACD,IAAI0O,WAAW,EAAE;MACfzV,GAAG,CAAC22B,KAAK,GAAG;QACVlhB,WAAW,EAAEA,WAAW,CAACxa,MAAM,CAACqS,GAAG,IAAI0L,wDAAa,CAAC1L,GAAG,CAAC;MAC3D,CAAC;IACH;IACA,OAAO,IAAI,CAAC,CAACvK,IAAI,CAAC/C,GAAG,EAAE+G,GAAG,CAACqG,EAAE,CAAC;EAChC;EAaAwqB,eAAeA,CAAC9zB,IAAI,EAAE;IACpB,IAAI,CAACxE,MAAM,CAAC,OAAO,IAAI,IAAI,CAAC8xB,gBAAgB,GAAG3uB,IAAI,CAACkxB,SAAS,CAAC7vB,IAAI,EAAEosB,gBAAgB,CAAC,GAAGpsB,IAAI,CAAC,CAAC;IAE9F,QAAQA,IAAI,CAACiZ,IAAI;MACf,KAAK,KAAK;QACR,IAAI,CAACjZ,IAAI,CAACgE,GAAG,IAAIhE,IAAI,CAACgE,GAAG,GAAG,CAAC,IAAI,CAAChE,IAAI,CAACoB,KAAK,EAAE;UAE5C;QACF;QAEA,IAAI,CAAC,IAAI,CAACjF,WAAW,CAAC,CAAC,EAAE;UAGvB;QACF;QAEA,MAAMiF,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAE7uB,IAAI,CAACoB,KAAK,CAAC;QACjD,IAAI,CAACA,KAAK,EAAE;UAEV;QACF;QAEA,IAAIA,KAAK,CAACoc,YAAY,CAAC,CAAC,EAAE;UAExB;QACF;QAEA,IAAIpc,KAAK,CAAC2iB,SAAS,CAAC,CAAC,GAAG/jB,IAAI,CAACgE,GAAG,EAAE;UAChC,IAAI5C,KAAK,CAACikB,aAAa,CAAC,CAAC,EAAE;YACzBjkB,KAAK,CAACgmB,eAAe,CAACpnB,IAAI,CAACgE,GAAG,EAAE,UAAU,CAAC;UAC7C;UAGA,IAAIhE,IAAI,CAAC+zB,KAAK,IAAI,CAAC,IAAI,CAAC,CAAClF,QAAQ,CAAC,MAAM,EAAE7uB,IAAI,CAAC+zB,KAAK,CAAC,EAAE;YAGrD,IAAI,CAACrU,OAAO,CAAC1f,IAAI,CAAC+zB,KAAK,EAAE,IAAI/a,wDAAc,CAAC,CAAC,CAACc,QAAQ,CAAC,CAAC,CAACgB,KAAK,CAAC,CAAC,CAAC,CAAC7d,KAAK,CAACsC,GAAG,IAAI;cAC7E,IAAI,CAAC/D,MAAM,CAAC,wCAAwC,EAAE+D,GAAG,CAAC;YAC5D,CAAC,CAAC;UACJ;UAEA6B,KAAK,CAACqc,SAAS,CAAC,IAAI,CAAC,CAACnQ,IAAI,CAAC/S,CAAC,IAAI;YAC9B,OAAO6G,KAAK,CAACse,OAAO,CAAC,IAAI1G,wDAAc,CAAC5X,KAAK,CAAC,CAACsY,aAAa,CAAC,EAAE,CAAC,CAACiB,YAAY,CAAC,EAAE,CAAC,CAACG,KAAK,CAAC,CAAC,CAAC;UAC5F,CAAC,CAAC,CAACxN,IAAI,CAAC/S,CAAC,IAAI;YAEX6G,KAAK,CAACoe,YAAY,CAAC,KAAK,EAAE,IAAI,CAAC;UACjC,CAAC,CAAC,CAACviB,KAAK,CAACsC,GAAG,IAAI;YACd,IAAI,CAAC/D,MAAM,CAAC,2BAA2B,EAAE+D,GAAG,CAAC;UAC/C,CAAC,CAAC,CAACy0B,OAAO,CAACz5B,CAAC,IAAI;YACd,IAAI,CAACyjB,UAAU,CAAC,CAAC,CAACkE,eAAe,CAAC,KAAK,EAAE9gB,KAAK,CAAC;UACjD,CAAC,CAAC;QACJ;QACA;MAEF,KAAK,MAAM;QACT,IAAI,CAAC4c,UAAU,CAAC,CAAC,CAACuI,UAAU,CAAC;UAC3BtN,IAAI,EAAE,MAAM;UACZjV,GAAG,EAAEhE,IAAI,CAACgE;QACZ,CAAC,CAAC;QACF;MAEF,KAAK,KAAK;QACR,IAAI,CAAC,IAAI,CAACwhB,IAAI,CAACxlB,IAAI,CAAC+zB,KAAK,CAAC,EAAE;UAE1B;QACF;QAEA,MAAM3jC,IAAI,GAAG;UACXH,KAAK,EAAE+P,IAAI,CAACi0B,SAAS;UACrB9jC,IAAI,EAAE6P,IAAI,CAACk0B;QACb,CAAC;QACD,MAAMlkC,GAAG,GAAG,IAAIF,uDAAU,CAACM,IAAI,CAAC;QAChC,MAAMo2B,IAAI,GAAI,CAACx2B,GAAG,CAACI,IAAI,IAAIJ,GAAG,CAACI,IAAI,IAAIN,uDAAU,CAACgB,KAAK,GAErD;UACEmoB,IAAI,EAAE,MAAM;UACZnW,GAAG,EAAE9C,IAAI,CAACoB;QACZ,CAAC,GAED;UACE6X,IAAI,EAAE,KAAK;UACXnW,GAAG,EAAE9C,IAAI,CAACoB,KAAK;UACfulB,IAAI,EAAEv2B;QACR,CAAC;QACH,IAAI,CAAC4tB,UAAU,CAAC,CAAC,CAACuI,UAAU,CAACC,IAAI,CAAC;QAClC;MAEF;QACE,IAAI,CAAChrB,MAAM,CAAC,2BAA2B,EAAEwE,IAAI,CAACiZ,IAAI,CAAC;IACvD;EACF;EAiCAyG,OAAOA,CAACte,KAAK,EAAErC,MAAM,EAAE;IACrB,MAAML,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,EAAE3vB,KAAK,CAAC;IAE1C1C,GAAG,CAACqD,GAAG,GAAGkZ,mDAAQ,CAACvc,GAAG,CAACqD,GAAG,EAAEhD,MAAM,CAAC;IAEnC,OAAO,IAAI,CAAC,CAACE,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACqD,GAAG,CAACuH,EAAE,CAAC;EACpC;EASA0W,OAAOA,CAAC5e,KAAK,EAAErC,MAAM,EAAE;IACrB,MAAML,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,EAAE3vB,KAAK,CAAC;IAC1C,MAAM6X,IAAI,GAAG,EAAE;IAEf,IAAIla,MAAM,EAAE;MACV,CAAC,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,WAAW,CAAC,CAACpI,OAAO,CAAC,UAAS6V,GAAG,EAAE;QACjE,IAAIzN,MAAM,CAACkG,cAAc,CAACuH,GAAG,CAAC,EAAE;UAC9ByM,IAAI,CAACrU,IAAI,CAAC4H,GAAG,CAAC;UACd9N,GAAG,CAACgZ,GAAG,CAAClL,GAAG,CAAC,GAAGzN,MAAM,CAACyN,GAAG,CAAC;QAC5B;MACF,CAAC,CAAC;MAEF,IAAIrW,KAAK,CAACC,OAAO,CAAC2I,MAAM,CAAC4S,WAAW,CAAC,IAAI5S,MAAM,CAAC4S,WAAW,CAACjgB,MAAM,GAAG,CAAC,EAAE;QACtEgN,GAAG,CAACm0B,KAAK,GAAG;UACVlhB,WAAW,EAAE5S,MAAM,CAAC4S,WAAW,CAACxa,MAAM,CAACqS,GAAG,IAAI0L,wDAAa,CAAC1L,GAAG,CAAC;QAClE,CAAC;MACH;IACF;IAEA,IAAIyP,IAAI,CAACvnB,MAAM,IAAI,CAAC,EAAE;MACpB,OAAOmK,OAAO,CAACC,MAAM,CAAC,IAAInL,KAAK,CAAC,0BAA0B,CAAC,CAAC;IAC9D;IAEA,OAAO,IAAI,CAAC,CAACsO,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACgZ,GAAG,CAACpO,EAAE,CAAC;EACpC;EAmBAoX,WAAWA,CAACtf,KAAK,EAAE8K,MAAM,EAAEyU,IAAI,EAAE;IAC/B,MAAMjiB,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,EAAE3vB,KAAK,CAAC;IAE1C1C,GAAG,CAACyiB,GAAG,CAAClI,IAAI,GAAG,KAAK;IACpBva,GAAG,CAACyiB,GAAG,CAACmF,MAAM,GAAGpa,MAAM;IACvBxN,GAAG,CAACyiB,GAAG,CAACR,IAAI,GAAGA,IAAI;IAEnB,OAAO,IAAI,CAAC,CAAC1hB,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACyiB,GAAG,CAAC7X,EAAE,CAAC;EACpC;EASAwY,QAAQA,CAACte,SAAS,EAAEmd,IAAI,EAAE;IACxB,MAAMjiB,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,EAAEvtB,SAAS,CAAC;IAC9C9E,GAAG,CAACyiB,GAAG,CAAClI,IAAI,GAAG,OAAO;IACtBva,GAAG,CAACyiB,GAAG,CAACR,IAAI,GAAGA,IAAI;IAEnB,OAAO,IAAI,CAAC,CAAC1hB,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACyiB,GAAG,CAAC7X,EAAE,CAAC;EACpC;EASAyY,eAAeA,CAACve,SAAS,EAAEF,IAAI,EAAE;IAC/B,MAAM5E,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,EAAEvtB,SAAS,CAAC;IAC9C9E,GAAG,CAACyiB,GAAG,CAAClI,IAAI,GAAG,KAAK;IACpBva,GAAG,CAACyiB,GAAG,CAAC7d,IAAI,GAAGA,IAAI;IAEnB,OAAO,IAAI,CAAC,CAACrE,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACyiB,GAAG,CAAC7X,EAAE,CAAC;EACpC;EASAkf,aAAaA,CAACC,MAAM,EAAE9jB,KAAK,EAAE;IAC3B,MAAMjG,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,EAAE/V,gDAAc,CAAC;IACnDtc,GAAG,CAACyiB,GAAG,CAAClI,IAAI,GAAG,MAAM;IACrBva,GAAG,CAACyiB,GAAG,CAAChB,IAAI,GAAG;MACb8H,IAAI,EAAEQ,MAAM;MACZl4B,GAAG,EAAEoU;IACP,CAAC;IAED,OAAO,IAAI,CAAC,CAAC1F,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACyiB,GAAG,CAAC7X,EAAE,CAAC;EACpC;EAQA6qB,cAAcA,CAACxT,IAAI,EAAE;IACnB,MAAMjiB,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,KAAK,EAAE,IAAI,CAAC;IACzCryB,GAAG,CAACyiB,GAAG,CAAClI,IAAI,GAAG,MAAM;IACrBva,GAAG,CAACyiB,GAAG,CAACR,IAAI,GAAGA,IAAI;IAEnB,OAAO,IAAI,CAAC,CAAC1hB,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACyiB,GAAG,CAAC7X,EAAE,CAAC,CAACgE,IAAI,CAAC/S,CAAC,IAAI;MAC3C,IAAI,CAACitB,MAAM,GAAG,IAAI;IACpB,CAAC,CAAC;EACJ;EASAxF,IAAIA,CAACxe,SAAS,EAAEyV,IAAI,EAAEjV,GAAG,EAAE;IACzB,IAAIA,GAAG,IAAI,CAAC,IAAIA,GAAG,IAAIgX,mDAAiB,EAAE;MACxC,MAAM,IAAIrqB,KAAK,CAAE,sBAAqBqT,GAAI,EAAC,CAAC;IAC9C;IAEA,MAAMtF,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,MAAM,EAAEvtB,SAAS,CAAC;IAC/C9E,GAAG,CAACsjB,IAAI,CAAC/I,IAAI,GAAGA,IAAI;IACpBva,GAAG,CAACsjB,IAAI,CAAChe,GAAG,GAAGA,GAAG;IAClB,IAAI,CAAC,CAAC/E,IAAI,CAACP,GAAG,CAAC;EACjB;EASA2jB,YAAYA,CAAC7e,SAAS,EAAEyF,IAAI,EAAE;IAC5B,MAAMvK,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,MAAM,EAAEvtB,SAAS,CAAC;IAC/C9E,GAAG,CAACsjB,IAAI,CAAC/I,IAAI,GAAGhQ,IAAI,IAAI,IAAI;IAC5B,IAAI,CAAC,CAAChK,IAAI,CAACP,GAAG,CAAC;EACjB;EAcAkP,SAASA,CAACpK,SAAS,EAAEQ,GAAG,EAAE9F,GAAG,EAAEqkB,OAAO,EAAE;IACtC,MAAM7jB,GAAG,GAAG,IAAI,CAAC,CAACqyB,UAAU,CAAC,MAAM,EAAEvtB,SAAS,CAAC;IAC/C9E,GAAG,CAACsjB,IAAI,CAAChe,GAAG,GAAGA,GAAG;IAClBtF,GAAG,CAACsjB,IAAI,CAAC/I,IAAI,GAAG,MAAM;IACtBva,GAAG,CAACsjB,IAAI,CAACnhB,KAAK,GAAG3C,GAAG;IACpBQ,GAAG,CAACsjB,IAAI,CAACO,OAAO,GAAGA,OAAO;IAC1B,IAAI,CAAC,CAACtjB,IAAI,CAACP,GAAG,EAAEA,GAAG,CAACsjB,IAAI,CAAC1Y,EAAE,CAAC;EAC9B;EAUAwe,QAAQA,CAACtkB,SAAS,EAAE;IAClB,IAAIpC,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAErrB,SAAS,CAAC;IAC9C,IAAI,CAACpC,KAAK,IAAIoC,SAAS,EAAE;MACvB,IAAIA,SAAS,IAAIwX,gDAAc,EAAE;QAC/B5Z,KAAK,GAAG,IAAIimB,8CAAO,CAAC,CAAC;MACvB,CAAC,MAAM,IAAI7jB,SAAS,IAAIwX,iDAAe,EAAE;QACvC5Z,KAAK,GAAG,IAAIynB,+CAAQ,CAAC,CAAC;MACxB,CAAC,MAAM;QACLznB,KAAK,GAAG,IAAIga,4CAAK,CAAC5X,SAAS,CAAC;MAC9B;MAEA,IAAI,CAAC,CAACsrB,kBAAkB,CAAC1tB,KAAK,CAAC;MAC/BA,KAAK,CAACyc,aAAa,CAAC,CAAC;IAEvB;IACA,OAAOzc,KAAK;EACd;EASAgnB,aAAaA,CAAC5kB,SAAS,EAAE;IACvB,OAAO,IAAI,CAAC,CAACqrB,QAAQ,CAAC,OAAO,EAAErrB,SAAS,CAAC;EAC3C;EAOAqkB,aAAaA,CAACrkB,SAAS,EAAE;IACvB,IAAI,CAAC,CAAC2tB,QAAQ,CAAC,OAAO,EAAE3tB,SAAS,CAAC;EACpC;EAQAZ,SAASA,CAAC0uB,IAAI,EAAEv6B,OAAO,EAAE;IACvB,IAAI,CAAC,CAAC65B,QAAQ,CAAC,OAAO,EAAEU,IAAI,EAAEv6B,OAAO,CAAC;EACxC;EAQA2vB,aAAaA,CAACljB,SAAS,EAAE;IACvB,OAAO,CAAC,CAAC,IAAI,CAAC,CAACqrB,QAAQ,CAAC,OAAO,EAAErrB,SAAS,CAAC;EAC7C;EAQA4wB,iBAAiBA,CAACC,MAAM,EAAE;IACxB,OAAO,CAACA,MAAM,GAAGrZ,sDAAoB,GAAGA,iDAAe,IAAI,IAAI,CAACrD,eAAe,CAAC,CAAC;EACnF;EAOAqG,UAAUA,CAAA,EAAG;IACX,OAAO,IAAI,CAAC8J,QAAQ,CAAC9M,gDAAc,CAAC;EACtC;EAOAsZ,WAAWA,CAAA,EAAG;IACZ,OAAO,IAAI,CAACxM,QAAQ,CAAC9M,iDAAe,CAAC;EACvC;EAOAuZ,kBAAkBA,CAAA,EAAG;IACnB,OAAO,IAAI9e,sDAAe,CAAC,IAAI,EAAEuF,wDAAsB,CAAC;EAC1D;EAQA+D,gBAAgBA,CAAA,EAAG;IACjB,OAAO,IAAI,CAACyI,MAAM;EACpB;EASAhC,IAAIA,CAACxiB,GAAG,EAAE;IACR,OAAO,IAAI,CAACwkB,MAAM,KAAKxkB,GAAG;EAC5B;EAOAwxB,eAAeA,CAAA,EAAG;IAChB,OAAO,IAAI,CAAChH,MAAM;EACpB;EAQAiH,aAAaA,CAAA,EAAG;IACd,OAAO,IAAI,CAAC7G,WAAW;EACzB;EAUA8G,MAAMA,CAACviC,MAAM,EAAE2O,MAAM,EAAE;IACrB,OAAO,IAAI,CAACwd,OAAO,CAACtD,iDAAe,EAAEtR,4DAAiB,CAAC,IAAI,EAAE;MAC3D,QAAQ,EAAEvX,MAAM;MAChB,QAAQ,EAAE2O;IACZ,CAAC,CAAC,CAAC;EACL;EAUA6zB,cAAcA,CAACn9B,IAAI,EAAEo9B,YAAY,EAAE;IACjC,OAAO,IAAI,CAAChH,WAAW,IAAI,IAAI,CAACA,WAAW,CAACp2B,IAAI,CAAC,IAAIo9B,YAAY;EACnE;EAQAC,aAAaA,CAACC,OAAO,EAAEC,eAAe,EAAE;IACtC,IAAI,CAAC1H,eAAe,GAAGyH,OAAO;IAC9B,IAAI,CAACxH,gBAAgB,GAAGwH,OAAO,IAAIC,eAAe;EACpD;EAOAC,gBAAgBA,CAACC,EAAE,EAAE;IACnB,IAAIA,EAAE,EAAE;MACN,IAAI,CAAC7H,cAAc,GAAG6H,EAAE;IAC1B;EACF;EAQAC,aAAaA,CAAC19B,IAAI,EAAE;IAClB,MAAM4J,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAEr3B,IAAI,CAAC;IAC3C,OAAO4J,KAAK,IAAIA,KAAK,CAACqlB,MAAM;EAC9B;EAQA0O,kBAAkBA,CAAC39B,IAAI,EAAE;IACvB,MAAM4J,KAAK,GAAG,IAAI,CAAC,CAACytB,QAAQ,CAAC,OAAO,EAAEr3B,IAAI,CAAC;IAC3C,OAAO4J,KAAK,GAAGA,KAAK,CAACpR,GAAG,GAAG,IAAI;EACjC;EASAolC,OAAOA,CAACh3B,MAAM,EAAE;IACd,IAAIA,MAAM,EAAE;MACV,IAAI,CAACsvB,UAAU,GAAG/wB,IAAI,CAACgxB,KAAK,CAAEhxB,IAAI,CAACE,MAAM,CAAC,CAAC,GAAG,QAAQ,GAAI,QAAQ,CAAC;IACrE,CAAC,MAAM;MACL,IAAI,CAAC6wB,UAAU,GAAG,CAAC;IACrB;EACF;EAQA2H,eAAe,GAAG7gC,SAAS;EAqB3B2+B,SAAS,GAAG3+B,SAAS;EAMrB4K,YAAY,GAAG5K,SAAS;EAWxBm9B,OAAO,GAAGn9B,SAAS;EAMnBy7B,aAAa,GAAGz7B,SAAS;EAMzB27B,aAAa,GAAG37B,SAAS;EAMzB47B,aAAa,GAAG57B,SAAS;EAMzB2K,SAAS,GAAG3K,SAAS;EAMrBu7B,YAAY,GAAGv7B,SAAS;EAMxBw7B,cAAc,GAAGx7B,SAAS;EAM1BsI,wBAAwB,GAAGtI,SAAS;AACtC;AAAC;AAGDu4B,MAAM,CAACx0B,mBAAmB,GAAGyiB,2DAAyB;AACtD+R,MAAM,CAACv0B,qBAAqB,GAAGwiB,6DAA2B;AAC1D+R,MAAM,CAACt0B,sBAAsB,GAAGuiB,8DAA4B;AAC5D+R,MAAM,CAACr0B,qBAAqB,GAAGsiB,6DAA2B;AAC1D+R,MAAM,CAACp0B,oBAAoB,GAAGqiB,4DAA0B;AACxD+R,MAAM,CAACn0B,mBAAmB,GAAGoiB,2DAAyB;AACtD+R,MAAM,CAACl0B,uBAAuB,GAAGmiB,+DAA6B;AAC9D+R,MAAM,CAACj0B,mBAAmB,GAAGkiB,2DAAyB;AACtD+R,MAAM,CAACh0B,oBAAoB,GAAGiiB,4DAA0B;AAGxD+R,MAAM,CAAC3zB,QAAQ,GAAG4hB,gDAAc;AAGhC+R,MAAM,CAACuI,gBAAgB,GAAG,gBAAgB;AAC1CvI,MAAM,CAACwI,oBAAoB,GAAG,oBAAoB;AAClDxI,MAAM,CAACyI,aAAa,GAAG,aAAa;AACpCzI,MAAM,CAAC0I,oBAAoB,GAAG,mBAAmB;AACjD1I,MAAM,CAAC2I,mBAAmB,GAAG,SAAS;AAGtC3I,MAAM,CAAC4I,mBAAmB,GAAG,eAAe,C","sources":["webpack://tinode/webpack/universalModuleDefinition","webpack://tinode/./src/access-mode.js","webpack://tinode/./src/cbuffer.js","webpack://tinode/./src/comm-error.js","webpack://tinode/./src/config.js","webpack://tinode/./src/connection.js","webpack://tinode/./src/db.js","webpack://tinode/./src/drafty.js","webpack://tinode/./src/large-file.js","webpack://tinode/./src/meta-builder.js","webpack://tinode/./src/topic.js","webpack://tinode/./src/utils.js","webpack://tinode/./version.js","webpack://tinode/webpack/bootstrap","webpack://tinode/webpack/runtime/compat get default export","webpack://tinode/webpack/runtime/define property getters","webpack://tinode/webpack/runtime/global","webpack://tinode/webpack/runtime/hasOwnProperty shorthand","webpack://tinode/webpack/runtime/make namespace object","webpack://tinode/./src/tinode.js"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"tinode\"] = factory();\n\telse\n\t\troot[\"tinode\"] = factory();\n})(this, () => {\nreturn ","/**\n * @file Access control model.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\n/**\n * Helper class for handling access mode.\n *\n * @class AccessMode\n * @memberof Tinode\n *\n * @param {AccessMode|Object=} acs - AccessMode to copy or access mode object received from the server.\n */\nexport default class AccessMode {\n constructor(acs) {\n if (acs) {\n this.given = typeof acs.given == 'number' ? acs.given : AccessMode.decode(acs.given);\n this.want = typeof acs.want == 'number' ? acs.want : AccessMode.decode(acs.want);\n this.mode = acs.mode ? (typeof acs.mode == 'number' ? acs.mode : AccessMode.decode(acs.mode)) :\n (this.given & this.want);\n }\n }\n\n static #checkFlag(val, side, flag) {\n side = side || 'mode';\n if (['given', 'want', 'mode'].includes(side)) {\n return ((val[side] & flag) != 0);\n }\n throw new Error(`Invalid AccessMode component '${side}'`);\n }\n /**\n * Parse string into an access mode value.\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {string | Number} mode - either a String representation of the access mode to parse or a set of bits to assign.\n * @returns {number} - Access mode as a numeric value.\n */\n static decode(str) {\n if (!str) {\n return null;\n } else if (typeof str == 'number') {\n return str & AccessMode._BITMASK;\n } else if (str === 'N' || str === 'n') {\n return AccessMode._NONE;\n }\n\n const bitmask = {\n 'J': AccessMode._JOIN,\n 'R': AccessMode._READ,\n 'W': AccessMode._WRITE,\n 'P': AccessMode._PRES,\n 'A': AccessMode._APPROVE,\n 'S': AccessMode._SHARE,\n 'D': AccessMode._DELETE,\n 'O': AccessMode._OWNER\n };\n\n let m0 = AccessMode._NONE;\n\n for (let i = 0; i < str.length; i++) {\n const bit = bitmask[str.charAt(i).toUpperCase()];\n if (!bit) {\n // Unrecognized bit, skip.\n continue;\n }\n m0 |= bit;\n }\n return m0;\n }\n /**\n * Convert numeric representation of the access mode into a string.\n *\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {number} val - access mode value to convert to a string.\n * @returns {string} - Access mode as a string.\n */\n static encode(val) {\n if (val === null || val === AccessMode._INVALID) {\n return null;\n } else if (val === AccessMode._NONE) {\n return 'N';\n }\n\n const bitmask = ['J', 'R', 'W', 'P', 'A', 'S', 'D', 'O'];\n let res = '';\n for (let i = 0; i < bitmask.length; i++) {\n if ((val & (1 << i)) != 0) {\n res = res + bitmask[i];\n }\n }\n return res;\n }\n /**\n * Update numeric representation of access mode with the new value. The value\n * is one of the following:\n * - a string starting with '+'
or '-'
then the bits to add or remove, e.g. '+R-W'
or '-PS'
.\n * - a new value of access mode\n *\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {number} val - access mode value to update.\n * @param {string} upd - update to apply to val.\n * @returns {number} - updated access mode.\n */\n static update(val, upd) {\n if (!upd || typeof upd != 'string') {\n return val;\n }\n\n let action = upd.charAt(0);\n if (action == '+' || action == '-') {\n let val0 = val;\n // Split delta-string like '+ABC-DEF+Z' into an array of parts including + and -.\n const parts = upd.split(/([-+])/);\n // Starting iteration from 1 because String.split() creates an array with the first empty element.\n // Iterating by 2 because we parse pairs +/- then data.\n for (let i = 1; i < parts.length - 1; i += 2) {\n action = parts[i];\n const m0 = AccessMode.decode(parts[i + 1]);\n if (m0 == AccessMode._INVALID) {\n return val;\n }\n if (m0 == null) {\n continue;\n }\n if (action === '+') {\n val0 |= m0;\n } else if (action === '-') {\n val0 &= ~m0;\n }\n }\n val = val0;\n } else {\n // The string is an explicit new value 'ABC' rather than delta.\n const val0 = AccessMode.decode(upd);\n if (val0 != AccessMode._INVALID) {\n val = val0;\n }\n }\n\n return val;\n }\n /**\n * Bits present in a1 but missing in a2.\n *\n * @static\n * @memberof Tinode\n *\n * @param {number | string} a1 - access mode to subtract from.\n * @param {number | string} a2 - access mode to subtract.\n * @returns {number} access mode with bits present in a1
but missing in a2
.\n */\n static diff(a1, a2) {\n a1 = AccessMode.decode(a1);\n a2 = AccessMode.decode(a2);\n\n if (a1 == AccessMode._INVALID || a2 == AccessMode._INVALID) {\n return AccessMode._INVALID;\n }\n return a1 & ~a2;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Custom formatter\n */\n toString() {\n return '{\"mode\": \"' + AccessMode.encode(this.mode) +\n '\", \"given\": \"' + AccessMode.encode(this.given) +\n '\", \"want\": \"' + AccessMode.encode(this.want) + '\"}';\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Converts numeric values to strings.\n */\n jsonHelper() {\n return {\n mode: AccessMode.encode(this.mode),\n given: AccessMode.encode(this.given),\n want: AccessMode.encode(this.want)\n };\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign value to 'mode'.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} m - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setMode(m) {\n this.mode = AccessMode.decode(m);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update mode
value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateMode(u) {\n this.mode = AccessMode.update(this.mode, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get mode
value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - mode
value.\n */\n getMode() {\n return AccessMode.encode(this.mode);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign given
value.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} g - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setGiven(g) {\n this.given = AccessMode.decode(g);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'given' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateGiven(u) {\n this.given = AccessMode.update(this.given, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get 'given' value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - given value.\n */\n getGiven() {\n return AccessMode.encode(this.given);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign 'want' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} w - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setWant(w) {\n this.want = AccessMode.decode(w);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'want' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateWant(u) {\n this.want = AccessMode.update(this.want, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get 'want' value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - want value.\n */\n getWant() {\n return AccessMode.encode(this.want);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get permissions present in 'want' but missing in 'given'.\n * Inverse of {@link Tinode.AccessMode#getExcessive}\n *\n * @memberof Tinode.AccessMode\n *\n * @returns {string} permissions present in want but missing in given.\n */\n getMissing() {\n return AccessMode.encode(this.want & ~this.given);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get permissions present in 'given' but missing in 'want'.\n * Inverse of {@link Tinode.AccessMode#getMissing}\n * @memberof Tinode.AccessMode\n *\n * @returns {string} permissions present in given but missing in want.\n */\n getExcessive() {\n return AccessMode.encode(this.given & ~this.want);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'want', 'give', and 'mode' values.\n * @memberof Tinode.AccessMode\n *\n * @param {AccessMode} val - new access mode value.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateAll(val) {\n if (val) {\n this.updateGiven(val.given);\n this.updateWant(val.want);\n this.mode = this.given & this.want;\n }\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Owner (O) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isOwner(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._OWNER);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Presence (P) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isPresencer(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._PRES);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Presence (P) flag is NOT set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isMuted(side) {\n return !this.isPresencer(side);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Join (J) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isJoiner(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._JOIN);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Reader (R) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isReader(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._READ);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Writer (W) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isWriter(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._WRITE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Approver (A) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isApprover(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._APPROVE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if either one of Owner (O) or Approver (A) flags is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isAdmin(side) {\n return this.isOwner(side) || this.isApprover(side);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if either one of Owner (O), Approver (A), or Sharer (S) flags is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isSharer(side) {\n return this.isAdmin(side) || AccessMode.#checkFlag(this, side, AccessMode._SHARE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Deleter (D) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isDeleter(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._DELETE);\n }\n}\n\nAccessMode._NONE = 0x00;\nAccessMode._JOIN = 0x01;\nAccessMode._READ = 0x02;\nAccessMode._WRITE = 0x04;\nAccessMode._PRES = 0x08;\nAccessMode._APPROVE = 0x10;\nAccessMode._SHARE = 0x20;\nAccessMode._DELETE = 0x40;\nAccessMode._OWNER = 0x80;\n\nAccessMode._BITMASK = AccessMode._JOIN | AccessMode._READ | AccessMode._WRITE | AccessMode._PRES |\n AccessMode._APPROVE | AccessMode._SHARE | AccessMode._DELETE | AccessMode._OWNER;\nAccessMode._INVALID = 0x100000;\n","/**\n * @file In-memory sorted cache of objects.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n/**\n * In-memory sorted cache of objects.\n *\n * @class CBuffer\n * @memberof Tinode\n * @protected\n *\n * @param {function} compare custom comparator of objects. Takes two parameters a
and b
;\n * returns -1
if a < b
, 0
if a == b
, 1
otherwise.\n * @param {boolean} unique enforce element uniqueness: when true
replace existing element with a new\n * one on conflict; when false
keep both elements.\n */\nexport default class CBuffer {\n #comparator = undefined;\n #unique = false;\n buffer = [];\n\n constructor(compare_, unique_) {\n this.#comparator = compare_ || ((a, b) => {\n return a === b ? 0 : a < b ? -1 : 1;\n });\n this.#unique = unique_;\n }\n\n #findNearest(elem, arr, exact) {\n let start = 0;\n let end = arr.length - 1;\n let pivot = 0;\n let diff = 0;\n let found = false;\n\n while (start <= end) {\n pivot = (start + end) / 2 | 0;\n diff = this.#comparator(arr[pivot], elem);\n if (diff < 0) {\n start = pivot + 1;\n } else if (diff > 0) {\n end = pivot - 1;\n } else {\n found = true;\n break;\n }\n }\n if (found) {\n return {\n idx: pivot,\n exact: true\n };\n }\n if (exact) {\n return {\n idx: -1\n };\n }\n // Not exact - insertion point\n return {\n idx: diff < 0 ? pivot + 1 : pivot\n };\n }\n\n // Insert element into a sorted array.\n #insertSorted(elem, arr) {\n const found = this.#findNearest(elem, arr, false);\n const count = (found.exact && this.#unique) ? 1 : 0;\n arr.splice(found.idx, count, elem);\n return arr;\n }\n\n /**\n * Get an element at the given position.\n * @memberof Tinode.CBuffer#\n * @param {number} at - Position to fetch from.\n * @returns {Object} Element at the given position or undefined
.\n */\n getAt(at) {\n return this.buffer[at];\n }\n\n /**\n * Convenience method for getting the element from the end of the buffer.\n * @memberof Tinode.CBuffer#\n * @param {number} at - position to fetch from, counting from the end;\n * undefined
or null
mean \"last\".\n * @returns {Object} The last element in the buffer or undefined
if buffer is empty.\n */\n getLast(at) {\n at |= 0;\n return this.buffer.length > at ? this.buffer[this.buffer.length - 1 - at] : undefined;\n }\n\n /**\n * Add new element(s) to the buffer. Variadic: takes one or more arguments. If an array is passed as a single\n * argument, its elements are inserted individually.\n * @memberof Tinode.CBuffer#\n *\n * @param {...Object|Array} - One or more objects to insert.\n */\n put() {\n let insert;\n // inspect arguments: if array, insert its elements, if one or more non-array arguments, insert them one by one\n if (arguments.length == 1 && Array.isArray(arguments[0])) {\n insert = arguments[0];\n } else {\n insert = arguments;\n }\n for (let idx in insert) {\n this.#insertSorted(insert[idx], this.buffer);\n }\n }\n\n /**\n * Remove element at the given position.\n * @memberof Tinode.CBuffer#\n * @param {number} at - Position to delete at.\n * @returns {Object} Element at the given position or undefined
.\n */\n delAt(at) {\n at |= 0;\n let r = this.buffer.splice(at, 1);\n if (r && r.length > 0) {\n return r[0];\n }\n return undefined;\n }\n\n /**\n * Remove elements between two positions.\n * @memberof Tinode.CBuffer#\n * @param {number} since - Position to delete from (inclusive).\n * @param {number} before - Position to delete to (exclusive).\n *\n * @returns {Array} array of removed elements (could be zero length).\n */\n delRange(since, before) {\n return this.buffer.splice(since, before - since);\n }\n\n /**\n * Return the number of elements the buffer holds.\n * @memberof Tinode.CBuffer#\n * @return {number} Number of elements in the buffer.\n */\n length() {\n return this.buffer.length;\n }\n\n /**\n * Reset the buffer discarding all elements\n * @memberof Tinode.CBuffer#\n */\n reset() {\n this.buffer = [];\n }\n\n /**\n * Callback for iterating contents of buffer. See {@link Tinode.CBuffer#forEach}.\n * @callback ForEachCallbackType\n * @memberof Tinode.CBuffer#\n * @param {Object} elem - Current element of the buffer.\n * @param {Object} prev - Previous element of the buffer.\n * @param {Object} next - Next element of the buffer.\n * @param {number} index - Index of the current element.\n */\n\n /**\n * Apply given callback
to all elements of the buffer.\n * @memberof Tinode.CBuffer#\n *\n * @param {Tinode.ForEachCallbackType} callback - Function to call for each element.\n * @param {number} startIdx - Optional index to start iterating from (inclusive).\n * @param {number} beforeIdx - Optional index to stop iterating before (exclusive).\n * @param {Object} context - calling context (i.e. value of this
in callback)\n */\n forEach(callback, startIdx, beforeIdx, context) {\n startIdx = startIdx | 0;\n beforeIdx = beforeIdx || this.buffer.length;\n\n for (let i = startIdx; i < beforeIdx; i++) {\n callback.call(context, this.buffer[i],\n (i > startIdx ? this.buffer[i - 1] : undefined),\n (i < beforeIdx - 1 ? this.buffer[i + 1] : undefined), i);\n }\n }\n\n /**\n * Find element in buffer using buffer's comparison function.\n * @memberof Tinode.CBuffer#\n *\n * @param {Object} elem - element to find.\n * @param {boolean=} nearest - when true and exact match is not found, return the nearest element (insertion point).\n * @returns {number} index of the element in the buffer or -1.\n */\n find(elem, nearest) {\n const {\n idx\n } = this.#findNearest(elem, this.buffer, !nearest);\n return idx;\n }\n\n /**\n * Callback for filtering the buffer. See {@link Tinode.CBuffer#filter}.\n * @callback FilterCallbackType\n * @memberof Tinode.CBuffer#\n * @param {Object} elem - Current element of the buffer.\n * @param {number} index - Index of the current element.\n * @returns {boolen} true
to keep the element, false
to remove.\n */\n\n /**\n * Remove all elements that do not pass the test implemented by the provided callback function.\n * @memberof Tinode.CBuffer#\n *\n * @param {Tinode.FilterCallbackType} callback - Function to call for each element.\n * @param {Object} context - calling context (i.e. value of this
in the callback)\n */\n filter(callback, context) {\n let count = 0;\n for (let i = 0; i < this.buffer.length; i++) {\n if (callback.call(context, this.buffer[i], i)) {\n this.buffer[count] = this.buffer[i];\n count++;\n }\n }\n\n this.buffer.splice(count);\n }\n\n /**\n * Check if buffer is empty.\n * @returns {boolean} true
if the buffer is empty, false
otherwise.\n */\n isEmpty() {\n return this.buffer.length == 0;\n }\n}\n","/**\n * @file Throwable error with numeric error code.\n *\n * @copyright 2015-2023 Tinode LLC.\n */\n'use strict';\n\nexport default class CommError extends Error {\n constructor(message, code) {\n super(`${message} (${code})`);\n this.name = 'CommError';\n this.code = code;\n }\n}\n","/**\n * @file Global constants and configuration parameters.\n *\n * @copyright 2015-2023 Tinode LLC\n */\n'use strict';\n\nimport {\n PACKAGE_VERSION\n} from '../version.js';\n\n// Global constants\nexport const PROTOCOL_VERSION = '0'; // Major component of the version, e.g. '0' in '0.17.1'.\nexport const VERSION = PACKAGE_VERSION || '0.21';\nexport const LIBRARY = 'tinodejs/' + VERSION;\n\n// Topic name prefixes.\nexport const TOPIC_NEW = 'new';\nexport const TOPIC_NEW_CHAN = 'nch';\nexport const TOPIC_ME = 'me';\nexport const TOPIC_FND = 'fnd';\nexport const TOPIC_SYS = 'sys';\nexport const TOPIC_CHAN = 'chn';\nexport const TOPIC_GRP = 'grp';\nexport const TOPIC_P2P = 'p2p';\nexport const USER_NEW = 'new';\n\n// Starting value of a locally-generated seqId used for pending messages.\nexport const LOCAL_SEQID = 0xFFFFFFF;\n\n// Status codes.\nexport const MESSAGE_STATUS_NONE = 0; // Status not assigned.\nexport const MESSAGE_STATUS_QUEUED = 10; // Local ID assigned, in progress to be sent.\nexport const MESSAGE_STATUS_SENDING = 20; // Transmission started.\nexport const MESSAGE_STATUS_FAILED = 30; // At least one attempt was made to send the message.\nexport const MESSAGE_STATUS_FATAL = 40; // Message sending failed and it should not be retried.\nexport const MESSAGE_STATUS_SENT = 50; // Delivered to the server.\nexport const MESSAGE_STATUS_RECEIVED = 60; // Received by the client.\nexport const MESSAGE_STATUS_READ = 70; // Read by the user.\nexport const MESSAGE_STATUS_TO_ME = 80; // The message is received from another user.\n\n// Reject unresolved futures after this many milliseconds.\nexport const EXPIRE_PROMISES_TIMEOUT = 5_000;\n// Periodicity of garbage collection of unresolved futures.\nexport const EXPIRE_PROMISES_PERIOD = 1_000;\n\n// Delay before acknowledging that a message was recived.\nexport const RECV_TIMEOUT = 100;\n\n// Default number of messages to pull into memory from persistent cache.\nexport const DEFAULT_MESSAGES_PAGE = 24;\n\n// Unicode DEL character indicating data was deleted.\nexport const DEL_CHAR = '\\u2421';\n","/**\n * @file Abstraction layer for websocket and long polling connections.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport CommError from './comm-error.js';\nimport {\n jsonParseHelper\n} from './utils.js';\n\nlet WebSocketProvider;\nlet XHRProvider;\n\n// Error code to return in case of a network problem.\nconst NETWORK_ERROR = 503;\nconst NETWORK_ERROR_TEXT = \"Connection failed\";\n\n// Error code to return when user disconnected from server.\nconst NETWORK_USER = 418;\nconst NETWORK_USER_TEXT = \"Disconnected by client\";\n\n// Settings for exponential backoff\nconst _BOFF_BASE = 2000; // 2000 milliseconds, minimum delay between reconnects\nconst _BOFF_MAX_ITER = 10; // Maximum delay between reconnects 2^10 * 2000 ~ 34 minutes\nconst _BOFF_JITTER = 0.3; // Add random delay\n\n// Helper function for creating an endpoint URL.\nfunction makeBaseUrl(host, protocol, version, apiKey) {\n let url = null;\n\n if (['http', 'https', 'ws', 'wss'].includes(protocol)) {\n url = `${protocol}://${host}`;\n if (url.charAt(url.length - 1) !== '/') {\n url += '/';\n }\n url += 'v' + version + '/channels';\n if (['http', 'https'].includes(protocol)) {\n // Long polling endpoint ends with \"lp\", i.e.\n // '/v0/channels/lp' vs just '/v0/channels' for ws\n url += '/lp';\n }\n url += '?apikey=' + apiKey;\n }\n return url;\n}\n\n/**\n * An abstraction for a websocket or a long polling connection.\n *\n * @class Connection\n * @memberof Tinode\n\n * @param {Object} config - configuration parameters.\n * @param {string} config.host - Host name and optional port number to connect to.\n * @param {string} config.apiKey - API key generated by keygen
.\n * @param {string} config.transport - Network transport to use, either \"ws\"/\"wss\"
for websocket or\n * lp
for long polling.\n * @param {boolean} config.secure - Use Secure WebSocket if true
.\n * @param {string} version_ - Major value of the protocol version, e.g. '0' in '0.17.1'.\n * @param {boolean} autoreconnect_ - If connection is lost, try to reconnect automatically.\n */\nexport default class Connection {\n // Logger, does nothing by default.\n static #log = _ => {};\n\n #boffTimer = null;\n #boffIteration = 0;\n #boffClosed = false; // Indicator if the socket was manually closed - don't autoreconnect if true.\n\n // Websocket.\n #socket = null;\n\n host;\n secure;\n apiKey;\n\n version;\n autoreconnect;\n\n initialized;\n\n // (config.host, config.apiKey, config.transport, config.secure), PROTOCOL_VERSION, true\n constructor(config, version_, autoreconnect_) {\n this.host = config.host;\n this.secure = config.secure;\n this.apiKey = config.apiKey;\n\n this.version = version_;\n this.autoreconnect = autoreconnect_;\n\n if (config.transport === 'lp') {\n // explicit request to use long polling\n this.#init_lp();\n this.initialized = 'lp';\n } else if (config.transport === 'ws') {\n // explicit request to use web socket\n // if websockets are not available, horrible things will happen\n this.#init_ws();\n this.initialized = 'ws';\n }\n\n if (!this.initialized) {\n // Invalid or undefined network transport.\n Connection.#log(\"Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.\");\n throw new Error(\"Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.\");\n }\n }\n\n /**\n * To use Connection in a non browser context, supply WebSocket and XMLHttpRequest providers.\n * @static\n * @memberof Connection\n * @param wsProvider WebSocket provider, e.g. for nodeJS , require('ws')
.\n * @param xhrProvider XMLHttpRequest provider, e.g. for node require('xhr')
.\n */\n static setNetworkProviders(wsProvider, xhrProvider) {\n WebSocketProvider = wsProvider;\n XHRProvider = xhrProvider;\n }\n\n /**\n * Assign a non-default logger.\n * @static\n * @memberof Connection\n * @param {function} l variadic logging function.\n */\n static set logger(l) {\n Connection.#log = l;\n }\n\n /**\n * Initiate a new connection\n * @memberof Tinode.Connection#\n * @param {string} host_ Host name to connect to; if null
the old host name will be used.\n * @param {boolean} force Force new connection even if one already exists.\n * @return {Promise} Promise resolved/rejected when the connection call completes, resolution is called without\n * parameters, rejection passes the {Error} as parameter.\n */\n connect(host_, force) {\n return Promise.reject(null);\n }\n\n /**\n * Try to restore a network connection, also reset backoff.\n * @memberof Tinode.Connection#\n *\n * @param {boolean} force - reconnect even if there is a live connection already.\n */\n reconnect(force) {}\n\n /**\n * Terminate the network connection\n * @memberof Tinode.Connection#\n */\n disconnect() {}\n\n /**\n * Send a string to the server.\n * @memberof Tinode.Connection#\n *\n * @param {string} msg - String to send.\n * @throws Throws an exception if the underlying connection is not live.\n */\n sendText(msg) {}\n\n /**\n * Check if connection is alive.\n * @memberof Tinode.Connection#\n * @returns {boolean} true
if connection is live, false
otherwise.\n */\n isConnected() {\n return false;\n }\n\n /**\n * Get the name of the current network transport.\n * @memberof Tinode.Connection#\n * @returns {string} name of the transport such as \"ws\"
or \"lp\"
.\n */\n transport() {\n return this.initialized;\n }\n\n /**\n * Send network probe to check if connection is indeed live.\n * @memberof Tinode.Connection#\n */\n probe() {\n this.sendText('1');\n }\n\n /**\n * Reset autoreconnect counter to zero.\n * @memberof Tinode.Connection#\n */\n backoffReset() {\n this.#boffReset();\n }\n\n // Backoff implementation - reconnect after a timeout.\n #boffReconnect() {\n // Clear timer\n clearTimeout(this.#boffTimer);\n // Calculate when to fire the reconnect attempt\n const timeout = _BOFF_BASE * (Math.pow(2, this.#boffIteration) * (1.0 + _BOFF_JITTER * Math.random()));\n // Update iteration counter for future use\n this.#boffIteration = (this.#boffIteration >= _BOFF_MAX_ITER ? this.#boffIteration : this.#boffIteration + 1);\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(timeout);\n }\n\n this.#boffTimer = setTimeout(_ => {\n Connection.#log(`Reconnecting, iter=${this.#boffIteration}, timeout=${timeout}`);\n // Maybe the socket was closed while we waited for the timer?\n if (!this.#boffClosed) {\n const prom = this.connect();\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(0, prom);\n } else {\n // Suppress error if it's not used.\n prom.catch(_ => {\n /* do nothing */\n });\n }\n } else if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(-1);\n }\n }, timeout);\n }\n\n // Terminate auto-reconnect process.\n #boffStop() {\n clearTimeout(this.#boffTimer);\n this.#boffTimer = null;\n }\n\n // Reset auto-reconnect iteration counter.\n #boffReset() {\n this.#boffIteration = 0;\n }\n\n // Initialization for long polling.\n #init_lp() {\n const XDR_UNSENT = 0; // Client has been created. open() not called yet.\n const XDR_OPENED = 1; // open() has been called.\n const XDR_HEADERS_RECEIVED = 2; // send() has been called, and headers and status are available.\n const XDR_LOADING = 3; // Downloading; responseText holds partial data.\n const XDR_DONE = 4; // The operation is complete.\n\n // Fully composed endpoint URL, with API key & SID\n let _lpURL = null;\n\n let _poller = null;\n let _sender = null;\n\n let lp_sender = (url_) => {\n const sender = new XHRProvider();\n sender.onreadystatechange = (evt) => {\n if (sender.readyState == XDR_DONE && sender.status >= 400) {\n // Some sort of error response\n throw new CommError(\"LP sender failed\", sender.status);\n }\n };\n\n sender.open('POST', url_, true);\n return sender;\n }\n\n let lp_poller = (url_, resolve, reject) => {\n let poller = new XHRProvider();\n let promiseCompleted = false;\n\n poller.onreadystatechange = evt => {\n if (poller.readyState == XDR_DONE) {\n if (poller.status == 201) { // 201 == HTTP.Created, get SID\n let pkt = JSON.parse(poller.responseText, jsonParseHelper);\n _lpURL = url_ + '&sid=' + pkt.ctrl.params.sid;\n poller = lp_poller(_lpURL);\n poller.send(null);\n if (this.onOpen) {\n this.onOpen();\n }\n\n if (resolve) {\n promiseCompleted = true;\n resolve();\n }\n\n if (this.autoreconnect) {\n this.#boffStop();\n }\n } else if (poller.status > 0 && poller.status < 400) { // 0 = network error; 400 = HTTP.BadRequest\n if (this.onMessage) {\n this.onMessage(poller.responseText);\n }\n poller = lp_poller(_lpURL);\n poller.send(null);\n } else {\n // Don't throw an error here, gracefully handle server errors\n if (reject && !promiseCompleted) {\n promiseCompleted = true;\n reject(poller.responseText);\n }\n if (this.onMessage && poller.responseText) {\n this.onMessage(poller.responseText);\n }\n if (this.onDisconnect) {\n const code = poller.status || (this.#boffClosed ? NETWORK_USER : NETWORK_ERROR);\n const text = poller.responseText || (this.#boffClosed ? NETWORK_USER_TEXT : NETWORK_ERROR_TEXT);\n this.onDisconnect(new CommError(text, code), code);\n }\n\n // Polling has stopped. Indicate it by setting poller to null.\n poller = null;\n if (!this.#boffClosed && this.autoreconnect) {\n this.#boffReconnect();\n }\n }\n }\n };\n // Using POST to avoid caching response by service worker.\n poller.open('POST', url_, true);\n return poller;\n }\n\n this.connect = (host_, force) => {\n this.#boffClosed = false;\n\n if (_poller) {\n if (!force) {\n return Promise.resolve();\n }\n _poller.onreadystatechange = undefined;\n _poller.abort();\n _poller = null;\n }\n\n if (host_) {\n this.host = host_;\n }\n\n return new Promise((resolve, reject) => {\n const url = makeBaseUrl(this.host, this.secure ? 'https' : 'http', this.version, this.apiKey);\n Connection.#log(\"LP connecting to:\", url);\n _poller = lp_poller(url, resolve, reject);\n _poller.send(null);\n }).catch(err => {\n Connection.#log(\"LP connection failed:\", err);\n });\n };\n\n this.reconnect = force => {\n this.#boffStop();\n this.connect(null, force);\n };\n\n this.disconnect = _ => {\n this.#boffClosed = true;\n this.#boffStop();\n\n if (_sender) {\n _sender.onreadystatechange = undefined;\n _sender.abort();\n _sender = null;\n }\n if (_poller) {\n _poller.onreadystatechange = undefined;\n _poller.abort();\n _poller = null;\n }\n\n if (this.onDisconnect) {\n this.onDisconnect(new CommError(NETWORK_USER_TEXT, NETWORK_USER), NETWORK_USER);\n }\n // Ensure it's reconstructed\n _lpURL = null;\n };\n\n this.sendText = (msg) => {\n _sender = lp_sender(_lpURL);\n if (_sender && (_sender.readyState == XDR_OPENED)) { // 1 == OPENED\n _sender.send(msg);\n } else {\n throw new Error(\"Long poller failed to connect\");\n }\n };\n\n this.isConnected = _ => {\n return (_poller && true);\n };\n }\n\n // Initialization for Websocket\n #init_ws() {\n this.connect = (host_, force) => {\n this.#boffClosed = false;\n\n if (this.#socket) {\n if (!force && this.#socket.readyState == this.#socket.OPEN) {\n return Promise.resolve();\n }\n this.#socket.close();\n this.#socket = null;\n }\n\n if (host_) {\n this.host = host_;\n }\n\n return new Promise((resolve, reject) => {\n const url = makeBaseUrl(this.host, this.secure ? 'wss' : 'ws', this.version, this.apiKey);\n\n Connection.#log(\"WS connecting to: \", url);\n\n // It throws when the server is not accessible but the exception cannot be caught:\n // https://stackoverflow.com/questions/31002592/javascript-doesnt-catch-error-in-websocket-instantiation/31003057\n const conn = new WebSocketProvider(url);\n\n conn.onerror = err => {\n reject(err);\n };\n\n conn.onopen = _ => {\n if (this.autoreconnect) {\n this.#boffStop();\n }\n\n if (this.onOpen) {\n this.onOpen();\n }\n\n resolve();\n };\n\n conn.onclose = _ => {\n this.#socket = null;\n\n if (this.onDisconnect) {\n const code = this.#boffClosed ? NETWORK_USER : NETWORK_ERROR;\n this.onDisconnect(new CommError(this.#boffClosed ? NETWORK_USER_TEXT : NETWORK_ERROR_TEXT, code), code);\n }\n\n if (!this.#boffClosed && this.autoreconnect) {\n this.#boffReconnect();\n }\n };\n\n conn.onmessage = evt => {\n if (this.onMessage) {\n this.onMessage(evt.data);\n }\n };\n\n this.#socket = conn;\n });\n }\n\n this.reconnect = force => {\n this.#boffStop();\n this.connect(null, force);\n };\n\n this.disconnect = _ => {\n this.#boffClosed = true;\n this.#boffStop();\n\n if (!this.#socket) {\n return;\n }\n this.#socket.close();\n this.#socket = null;\n };\n\n this.sendText = msg => {\n if (this.#socket && (this.#socket.readyState == this.#socket.OPEN)) {\n this.#socket.send(msg);\n } else {\n throw new Error(\"Websocket is not connected\");\n }\n };\n\n this.isConnected = _ => {\n return (this.#socket && (this.#socket.readyState == this.#socket.OPEN));\n };\n }\n\n // Callbacks:\n\n /**\n * A callback to pass incoming messages to. See {@link Tinode.Connection#onMessage}.\n * @callback Tinode.Connection.OnMessage\n * @memberof Tinode.Connection\n * @param {string} message - Message to process.\n */\n onMessage = undefined;\n\n /**\n * A callback for reporting a dropped connection.\n * @type {function}\n * @memberof Tinode.Connection#\n */\n onDisconnect = undefined;\n\n /**\n * A callback called when the connection is ready to be used for sending. For websockets it's socket open,\n * for long polling it's readyState=1
(OPENED)\n * @type {function}\n * @memberof Tinode.Connection#\n */\n onOpen = undefined;\n\n /**\n * A callback to notify of reconnection attempts. See {@link Tinode.Connection#onAutoreconnectIteration}.\n * @memberof Tinode.Connection\n * @callback AutoreconnectIterationType\n * @param {string} timeout - time till the next reconnect attempt in milliseconds. -1
means reconnect was skipped.\n * @param {Promise} promise resolved or rejected when the reconnect attemp completes.\n *\n */\n /**\n * A callback to inform when the next attampt to reconnect will happen and to receive connection promise.\n * @memberof Tinode.Connection#\n * @type {Tinode.Connection.AutoreconnectIterationType}\n */\n onAutoreconnectIteration = undefined;\n}\n\nConnection.NETWORK_ERROR = NETWORK_ERROR;\nConnection.NETWORK_ERROR_TEXT = NETWORK_ERROR_TEXT;\nConnection.NETWORK_USER = NETWORK_USER;\nConnection.NETWORK_USER_TEXT = NETWORK_USER_TEXT;\n","/**\n * @file Helper methods for dealing with IndexedDB cache of messages, users, and topics.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nconst DB_VERSION = 1;\nconst DB_NAME = 'tinode-web';\n\nlet IDBProvider;\n\nexport default class DB {\n #onError = _ => {};\n #logger = _ => {};\n\n // Instance of IndexDB.\n db = null;\n // Indicator that the cache is disabled.\n disabled = true;\n\n constructor(onError, logger) {\n this.#onError = onError || this.#onError;\n this.#logger = logger || this.#logger;\n }\n\n #mapObjects(source, callback, context) {\n if (!this.db) {\n return disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction([source]);\n trx.onerror = event => {\n this.#logger('PCache', 'mapObjects', source, event.target.error);\n reject(event.target.error);\n };\n trx.objectStore(source).getAll().onsuccess = event => {\n if (callback) {\n event.target.result.forEach(topic => {\n callback.call(context, topic);\n });\n }\n resolve(event.target.result);\n };\n });\n }\n\n /**\n * Initialize persistent cache: open or create/upgrade if needed.\n * @returns {Promise} promise to be resolved/rejected when the DB is initialized.\n */\n initDatabase() {\n return new Promise((resolve, reject) => {\n // Open the database and initialize callbacks.\n const req = IDBProvider.open(DB_NAME, DB_VERSION);\n req.onsuccess = event => {\n this.db = event.target.result;\n this.disabled = false;\n resolve(this.db);\n };\n req.onerror = event => {\n this.#logger('PCache', \"failed to initialize\", event);\n reject(event.target.error);\n this.#onError(event.target.error);\n };\n req.onupgradeneeded = event => {\n this.db = event.target.result;\n\n this.db.onerror = event => {\n this.#logger('PCache', \"failed to create storage\", event);\n this.#onError(event.target.error);\n };\n\n // Individual object stores.\n // Object store (table) for topics. The primary key is topic name.\n this.db.createObjectStore('topic', {\n keyPath: 'name'\n });\n\n // Users object store. UID is the primary key.\n this.db.createObjectStore('user', {\n keyPath: 'uid'\n });\n\n // Subscriptions object store topic <-> user. Topic name + UID is the primary key.\n this.db.createObjectStore('subscription', {\n keyPath: ['topic', 'uid']\n });\n\n // Messages object store. The primary key is topic name + seq.\n this.db.createObjectStore('message', {\n keyPath: ['topic', 'seq']\n });\n };\n });\n }\n\n /**\n * Delete persistent cache.\n */\n deleteDatabase() {\n // Close connection, otherwise operations will fail with 'onblocked'.\n if (this.db) {\n this.db.close();\n this.db = null;\n }\n return new Promise((resolve, reject) => {\n const req = IDBProvider.deleteDatabase(DB_NAME);\n req.onblocked = _ => {\n if (this.db) {\n this.db.close();\n }\n const err = new Error(\"blocked\");\n this.#logger('PCache', 'deleteDatabase', err);\n reject(err);\n };\n req.onsuccess = _ => {\n this.db = null;\n this.disabled = true;\n resolve(true);\n };\n req.onerror = event => {\n this.#logger('PCache', 'deleteDatabase', event.target.error);\n reject(event.target.error);\n };\n });\n }\n\n /**\n * Check if persistent cache is ready for use.\n * @memberOf DB\n * @returns {boolean} true
if cache is ready, false
otherwise.\n */\n isReady() {\n return !!this.db;\n }\n\n // Topics.\n\n /**\n * Save to cache or update topic in persistent cache.\n * @memberOf DB\n * @param {Topic} topic - topic to be added or updated.\n * @returns {Promise} promise resolved/rejected on operation completion.\n */\n updTopic(topic) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updTopic', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('topic').get(topic.name);\n req.onsuccess = _ => {\n trx.objectStore('topic').put(DB.#serializeTopic(req.result, topic));\n trx.commit();\n };\n });\n }\n\n /**\n * Mark or unmark topic as deleted.\n * @memberOf DB\n * @param {string} name - name of the topic to mark or unmark.\n * @param {boolean} deleted - status\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n markTopicAsDeleted(name, deleted) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'markTopicAsDeleted', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('topic').get(name);\n req.onsuccess = event => {\n const topic = event.target.result;\n if (topic && topic._deleted != deleted) {\n topic._deleted = deleted;\n trx.objectStore('topic').put(topic);\n }\n trx.commit();\n };\n });\n }\n\n /**\n * Remove topic from persistent cache.\n * @memberOf DB\n * @param {string} name - name of the topic to remove from database.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remTopic(name) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic', 'subscription', 'message'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'remTopic', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('topic').delete(IDBKeyRange.only(name));\n trx.objectStore('subscription').delete(IDBKeyRange.bound([name, '-'], [name, '~']));\n trx.objectStore('message').delete(IDBKeyRange.bound([name, 0], [name, Number.MAX_SAFE_INTEGER]));\n trx.commit();\n });\n }\n\n /**\n * Execute a callback for each stored topic.\n * @memberOf DB\n * @param {function} callback - function to call for each topic.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapTopics(callback, context) {\n return this.#mapObjects('topic', callback, context);\n }\n\n /**\n * Copy data from serialized object to topic.\n * @memberOf DB\n * @param {Topic} topic - target to deserialize to.\n * @param {Object} src - serialized data to copy from.\n */\n deserializeTopic(topic, src) {\n DB.#deserializeTopic(topic, src);\n }\n\n // Users.\n /**\n * Add or update user object in the persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to save or update.\n * @param {Object} pub - user's public
information.\n * @returns {Promise} promise resolved/rejected on operation completion.\n */\n updUser(uid, pub) {\n if (arguments.length < 2 || pub === undefined) {\n // No point inupdating user with invalid data.\n return;\n }\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').put({\n uid: uid,\n public: pub\n });\n trx.commit();\n });\n }\n\n /**\n * Remove user from persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to remove from the cache.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remUser(uid) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'remUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').delete(IDBKeyRange.only(uid));\n trx.commit();\n });\n }\n\n /**\n * Execute a callback for each stored user.\n * @memberOf DB\n * @param {function} callback - function to call for each topic.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapUsers(callback, context) {\n return this.#mapObjects('user', callback, context);\n }\n\n /**\n * Read a single user from persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to fetch from cache.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n getUser(uid) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user']);\n trx.oncomplete = event => {\n const user = event.target.result;\n resolve({\n user: user.uid,\n public: user.public\n });\n };\n trx.onerror = event => {\n this.#logger('PCache', 'getUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').get(uid);\n });\n }\n\n // Subscriptions.\n /**\n * Add or update subscription in persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {string} uid - ID of the subscribed user.\n * @param {Object} sub - subscription to save.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n updSubscription(topicName, uid, sub) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['subscription'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updSubscription', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('subscription').get([topicName, uid]).onsuccess = (event) => {\n trx.objectStore('subscription').put(DB.#serializeSubscription(event.target.result, topicName, uid, sub));\n trx.commit();\n };\n });\n }\n\n /**\n * Execute a callback for each cached subscription in a given topic.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the subscriptions.\n * @param {function} callback - function to call for each subscription.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapSubscriptions(topicName, callback, context) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['subscription']);\n trx.onerror = (event) => {\n this.#logger('PCache', 'mapSubscriptions', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('subscription').getAll(IDBKeyRange.bound([topicName, '-'], [topicName, '~'])).onsuccess = (event) => {\n if (callback) {\n event.target.result.forEach((topic) => {\n callback.call(context, topic);\n });\n }\n resolve(event.target.result);\n };\n });\n }\n\n // Messages.\n\n /**\n * Save message to persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {Object} msg - message to save.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n addMessage(msg) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'addMessage', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('message').add(DB.#serializeMessage(null, msg));\n trx.commit();\n });\n }\n\n /**\n * Update delivery status of a message stored in persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {number} seq - ID of the message to update\n * @param {number} status - new delivery status of the message.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n updMessageStatus(topicName, seq, status) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updMessageStatus', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('message').get(IDBKeyRange.only([topicName, seq]));\n req.onsuccess = event => {\n const src = req.result || event.target.result;\n if (!src || src._status == status) {\n trx.commit();\n return;\n }\n trx.objectStore('message').put(DB.#serializeMessage(src, {\n topic: topicName,\n seq: seq,\n _status: status\n }));\n trx.commit();\n };\n });\n }\n\n /**\n * Remove one or more messages from persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {number} from - id of the message to remove or lower boundary when removing range (inclusive).\n * @param {number=} to - upper boundary (exclusive) when removing a range of messages.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remMessages(topicName, from, to) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n if (!from && !to) {\n from = 0;\n to = Number.MAX_SAFE_INTEGER;\n }\n const range = to > 0 ? IDBKeyRange.bound([topicName, from], [topicName, to], false, true) :\n IDBKeyRange.only([topicName, from]);\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = (event) => {\n resolve(event.target.result);\n };\n trx.onerror = (event) => {\n this.#logger('PCache', 'remMessages', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('message').delete(range);\n trx.commit();\n });\n }\n\n /**\n * Retrieve messages from persistent store.\n * @memberOf DB\n * @param {string} topicName - name of the topic to retrieve messages from.\n * @param {function} callback to call for each retrieved message.\n * @param {Object} query - parameters of the message range to retrieve.\n * @param {number=} query.since - the least message ID to retrieve (inclusive).\n * @param {number=} query.before - the greatest message ID to retrieve (exclusive).\n * @param {number=} query.limit - the maximum number of messages to retrieve.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n readMessages(topicName, query, callback, context) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n query = query || {};\n const since = query.since > 0 ? query.since : 0;\n const before = query.before > 0 ? query.before : Number.MAX_SAFE_INTEGER;\n const limit = query.limit | 0;\n\n const result = [];\n const range = IDBKeyRange.bound([topicName, since], [topicName, before], false, true);\n const trx = this.db.transaction(['message']);\n trx.onerror = (event) => {\n this.#logger('PCache', 'readMessages', event.target.error);\n reject(event.target.error);\n };\n // Iterate in descending order.\n trx.objectStore('message').openCursor(range, 'prev').onsuccess = (event) => {\n const cursor = event.target.result;\n if (cursor) {\n if (callback) {\n callback.call(context, cursor.value);\n }\n result.push(cursor.value);\n if (limit <= 0 || result.length < limit) {\n cursor.continue();\n } else {\n resolve(result);\n }\n } else {\n resolve(result);\n }\n };\n });\n }\n\n // Private methods.\n\n // Serializable topic fields.\n static #topic_fields = ['created', 'updated', 'deleted', 'read', 'recv', 'seq', 'clear', 'defacs',\n 'creds', 'public', 'trusted', 'private', 'touched', '_deleted'\n ];\n\n // Copy data from src to Topic object.\n static #deserializeTopic(topic, src) {\n DB.#topic_fields.forEach((f) => {\n if (src.hasOwnProperty(f)) {\n topic[f] = src[f];\n }\n });\n if (Array.isArray(src.tags)) {\n topic._tags = src.tags;\n }\n if (src.acs) {\n topic.setAccessMode(src.acs);\n }\n topic.seq |= 0;\n topic.read |= 0;\n topic.unread = Math.max(0, topic.seq - topic.read);\n }\n\n // Copy values from 'src' to 'dst'. Allocate dst if it's null or undefined.\n static #serializeTopic(dst, src) {\n const res = dst || {\n name: src.name\n };\n DB.#topic_fields.forEach((f) => {\n if (src.hasOwnProperty(f)) {\n res[f] = src[f];\n }\n });\n if (Array.isArray(src._tags)) {\n res.tags = src._tags;\n }\n if (src.acs) {\n res.acs = src.getAccessMode().jsonHelper();\n }\n return res;\n }\n\n static #serializeSubscription(dst, topicName, uid, sub) {\n const fields = ['updated', 'mode', 'read', 'recv', 'clear', 'lastSeen', 'userAgent'];\n const res = dst || {\n topic: topicName,\n uid: uid\n };\n\n fields.forEach((f) => {\n if (sub.hasOwnProperty(f)) {\n res[f] = sub[f];\n }\n });\n\n return res;\n }\n\n static #serializeMessage(dst, msg) {\n // Serializable fields.\n const fields = ['topic', 'seq', 'ts', '_status', 'from', 'head', 'content'];\n const res = dst || {};\n fields.forEach((f) => {\n if (msg.hasOwnProperty(f)) {\n res[f] = msg[f];\n }\n });\n return res;\n }\n\n /**\n * To use DB in a non browser context, supply indexedDB provider.\n * @static\n * @memberof DB\n * @param idbProvider indexedDB provider, e.g. for node require('fake-indexeddb')
.\n */\n static setDatabaseProvider(idbProvider) {\n IDBProvider = idbProvider;\n }\n}\n","/**\n * @copyright 2015-2022 Tinode LLC.\n * @summary Minimally rich text representation and formatting for Tinode.\n * @license Apache 2.0\n *\n * @file Basic parser and formatter for very simple text markup. Mostly targeted at\n * mobile use cases similar to Telegram, WhatsApp, and FB Messenger.\n *\n * Supports conversion of user keyboard input to formatted text:
\n * \n * - *abc* → abc
\n * - _abc_ → abc
\n * - ~abc~ →
abc \n * - `abc` → abc
\n *
\n * Also supports forms and buttons.\n *\n * Nested formatting is supported, e.g. *abc _def_* -> abc def\n * URLs, @mentions, and #hashtags are extracted and converted into links.\n * Forms and buttons can be added procedurally.\n * JSON data representation is inspired by Draft.js raw formatting.\n *\n *\n * @example\n * Text:\n * \n * this is *bold*, `code` and _italic_, ~strike~\n * combined *bold and _italic_*\n * an url: https://www.example.com/abc#fragment and another _www.tinode.co_\n * this is a @mention and a #hashtag in a string\n * second #hashtag\n *
\n *\n * Sample JSON representation of the text above:\n * {\n * \"txt\": \"this is bold, code and italic, strike combined bold and italic an url: https://www.example.com/abc#fragment \" +\n * \"and another www.tinode.co this is a @mention and a #hashtag in a string second #hashtag\",\n * \"fmt\": [\n * { \"at\":8, \"len\":4,\"tp\":\"ST\" },{ \"at\":14, \"len\":4, \"tp\":\"CO\" },{ \"at\":23, \"len\":6, \"tp\":\"EM\"},\n * { \"at\":31, \"len\":6, \"tp\":\"DL\" },{ \"tp\":\"BR\", \"len\":1, \"at\":37 },{ \"at\":56, \"len\":6, \"tp\":\"EM\" },\n * { \"at\":47, \"len\":15, \"tp\":\"ST\" },{ \"tp\":\"BR\", \"len\":1, \"at\":62 },{ \"at\":120, \"len\":13, \"tp\":\"EM\" },\n * { \"at\":71, \"len\":36, \"key\":0 },{ \"at\":120, \"len\":13, \"key\":1 },{ \"tp\":\"BR\", \"len\":1, \"at\":133 },\n * { \"at\":144, \"len\":8, \"key\":2 },{ \"at\":159, \"len\":8, \"key\":3 },{ \"tp\":\"BR\", \"len\":1, \"at\":179 },\n * { \"at\":187, \"len\":8, \"key\":3 },{ \"tp\":\"BR\", \"len\":1, \"at\":195 }\n * ],\n * \"ent\": [\n * { \"tp\":\"LN\", \"data\":{ \"url\":\"https://www.example.com/abc#fragment\" } },\n * { \"tp\":\"LN\", \"data\":{ \"url\":\"http://www.tinode.co\" } },\n * { \"tp\":\"MN\", \"data\":{ \"val\":\"mention\" } },\n * { \"tp\":\"HT\", \"data\":{ \"val\":\"hashtag\" } }\n * ]\n * }\n */\n\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nconst MAX_FORM_ELEMENTS = 8;\nconst MAX_PREVIEW_ATTACHMENTS = 3;\nconst MAX_PREVIEW_DATA_SIZE = 64;\nconst JSON_MIME_TYPE = 'application/json';\nconst DRAFTY_MIME_TYPE = 'text/x-drafty';\nconst ALLOWED_ENT_FIELDS = ['act', 'height', 'duration', 'incoming', 'mime', 'name', 'premime', 'preref', 'preview',\n 'ref', 'size', 'state', 'url', 'val', 'width'\n];\n\n// Regular expressions for parsing inline formats. Javascript does not support lookbehind,\n// so it's a bit messy.\nconst INLINE_STYLES = [\n // Strong = bold, *bold text*\n {\n name: 'ST',\n start: /(?:^|[\\W_])(\\*)[^\\s*]/,\n end: /[^\\s*](\\*)(?=$|[\\W_])/\n },\n // Emphesized = italic, _italic text_\n {\n name: 'EM',\n start: /(?:^|\\W)(_)[^\\s_]/,\n end: /[^\\s_](_)(?=$|\\W)/\n },\n // Deleted, ~strike this though~\n {\n name: 'DL',\n start: /(?:^|[\\W_])(~)[^\\s~]/,\n end: /[^\\s~](~)(?=$|[\\W_])/\n },\n // Code block `this is monospace`\n {\n name: 'CO',\n start: /(?:^|\\W)(`)[^`]/,\n end: /[^`](`)(?=$|\\W)/\n }\n];\n\n// Relative weights of formatting spans. Greater index in array means greater weight.\nconst FMT_WEIGHT = ['QQ'];\n\n// RegExps for entity extraction (RF = reference)\nconst ENTITY_TYPES = [\n // URLs\n {\n name: 'LN',\n dataName: 'url',\n pack: function(val) {\n // Check if the protocol is specified, if not use http\n if (!/^[a-z]+:\\/\\//i.test(val)) {\n val = 'http://' + val;\n }\n return {\n url: val\n };\n },\n re: /(?:(?:https?|ftp):\\/\\/|www\\.|ftp\\.)[-A-Z0-9+&@#\\/%=~_|$?!:,.]*[A-Z0-9+&@#\\/%=~_|$]/ig\n },\n // Mentions @user (must be 2 or more characters)\n {\n name: 'MN',\n dataName: 'val',\n pack: function(val) {\n return {\n val: val.slice(1)\n };\n },\n re: /\\B@([\\p{L}\\p{N}][._\\p{L}\\p{N}]*[\\p{L}\\p{N}])/ug\n },\n // Hashtags #hashtag, like metion 2 or more characters.\n {\n name: 'HT',\n dataName: 'val',\n pack: function(val) {\n return {\n val: val.slice(1)\n };\n },\n re: /\\B#([\\p{L}\\p{N}][._\\p{L}\\p{N}]*[\\p{L}\\p{N}])/ug\n }\n];\n\n// HTML tag name suggestions\nconst FORMAT_TAGS = {\n AU: {\n html_tag: 'audio',\n md_tag: undefined,\n isVoid: false\n },\n BN: {\n html_tag: 'button',\n md_tag: undefined,\n isVoid: false\n },\n BR: {\n html_tag: 'br',\n md_tag: '\\n',\n isVoid: true\n },\n CO: {\n html_tag: 'tt',\n md_tag: '`',\n isVoid: false\n },\n DL: {\n html_tag: 'del',\n md_tag: '~',\n isVoid: false\n },\n EM: {\n html_tag: 'i',\n md_tag: '_',\n isVoid: false\n },\n EX: {\n html_tag: '',\n md_tag: undefined,\n isVoid: true\n },\n FM: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n HD: {\n html_tag: '',\n md_tag: undefined,\n isVoid: false\n },\n HL: {\n html_tag: 'span',\n md_tag: undefined,\n isVoid: false\n },\n HT: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n IM: {\n html_tag: 'img',\n md_tag: undefined,\n isVoid: false\n },\n LN: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n MN: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n RW: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false,\n },\n QQ: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n ST: {\n html_tag: 'b',\n md_tag: '*',\n isVoid: false\n },\n VC: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n VD: {\n html_tag: 'video',\n md_tag: undefined,\n isVoid: false\n }\n};\n\n// Convert base64-encoded string into Blob.\nfunction base64toObjectUrl(b64, contentType, logger) {\n if (!b64) {\n return null;\n }\n\n try {\n const bin = atob(b64);\n const length = bin.length;\n const buf = new ArrayBuffer(length);\n const arr = new Uint8Array(buf);\n for (let i = 0; i < length; i++) {\n arr[i] = bin.charCodeAt(i);\n }\n\n return URL.createObjectURL(new Blob([buf], {\n type: contentType\n }));\n } catch (err) {\n if (logger) {\n logger(\"Drafty: failed to convert object.\", err.message);\n }\n }\n\n return null;\n}\n\nfunction base64toDataUrl(b64, contentType) {\n if (!b64) {\n return null;\n }\n contentType = contentType || 'image/jpeg';\n return 'data:' + contentType + ';base64,' + b64;\n}\n\n// Helpers for converting Drafty to HTML.\nconst DECORATORS = {\n // Visial styles\n ST: {\n open: _ => '',\n close: _ => ''\n },\n EM: {\n open: _ => '',\n close: _ => ''\n },\n DL: {\n open: _ => '',\n close: _ => ''\n },\n CO: {\n open: _ => '',\n close: _ => ''\n },\n // Line break\n BR: {\n open: _ => '
',\n close: _ => ''\n },\n // Hidden element\n HD: {\n open: _ => '',\n close: _ => ''\n },\n // Highlighted element.\n HL: {\n open: _ => '',\n close: _ => ''\n },\n // Link (URL)\n LN: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n href: data.url,\n target: '_blank'\n } : null;\n },\n },\n // Mention\n MN: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n id: data.val\n } : null;\n },\n },\n // Hashtag\n HT: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n id: data.val\n } : null;\n },\n },\n // Button\n BN: {\n open: _ => '',\n props: (data) => {\n return data ? {\n 'data-act': data.act,\n 'data-val': data.val,\n 'data-name': data.name,\n 'data-ref': data.ref\n } : null;\n },\n },\n // Audio recording\n AU: {\n open: (data) => {\n const url = data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger);\n return '',\n props: (data) => {\n if (!data) return null;\n return {\n // Embedded data or external link.\n src: data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n 'data-preload': data.ref ? 'metadata' : 'auto',\n 'data-duration': data.duration,\n 'data-name': data.name,\n 'data-size': data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0),\n 'data-mime': data.mime,\n };\n }\n },\n // Image\n IM: {\n open: data => {\n // Don't use data.ref for preview: it's a security risk.\n const tmpPreviewUrl = base64toDataUrl(data._tempPreview, data.mime);\n const previewUrl = base64toObjectUrl(data.val, data.mime, Drafty.logger);\n const downloadUrl = data.ref || previewUrl;\n return (data.name ? '' : '') +\n '';\n },\n close: data => {\n return (data.name ? '' : '');\n },\n props: data => {\n if (!data) return null;\n return {\n // Temporary preview, or permanent preview, or external link.\n src: base64toDataUrl(data._tempPreview, data.mime) ||\n data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n title: data.name,\n alt: data.name,\n 'data-width': data.width,\n 'data-height': data.height,\n 'data-name': data.name,\n 'data-size': data.ref ? (data.size | 0) : (data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0)),\n 'data-mime': data.mime,\n };\n },\n },\n // Form - structured layout of elements.\n FM: {\n open: _ => '',\n close: _ => ''\n },\n // Row: logic grouping of elements\n RW: {\n open: _ => '',\n close: _ => ''\n },\n // Quoted block.\n QQ: {\n open: _ => '',\n close: _ => '',\n props: (data) => {\n return data ? {} : null;\n },\n },\n // Video call\n VC: {\n open: _ => '',\n close: _ => '',\n props: data => {\n if (!data) return {};\n return {\n 'data-duration': data.duration,\n 'data-state': data.state,\n };\n }\n },\n // Video.\n VD: {\n open: data => {\n const tmpPreviewUrl = base64toDataUrl(data._tempPreview, data.mime);\n const previewUrl = data.ref || base64toObjectUrl(data.preview, data.premime || 'image/jpeg', Drafty.logger);\n return '';\n },\n close: _ => '',\n props: data => {\n if (!data) return null;\n const poster = data.preref || base64toObjectUrl(data.preview, data.premime || 'image/jpeg', Drafty.logger);\n return {\n // Embedded data or external link.\n src: poster,\n 'data-src': data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n 'data-width': data.width,\n 'data-height': data.height,\n 'data-preload': data.ref ? 'metadata' : 'auto',\n 'data-preview': poster,\n 'data-duration': data.duration | 0,\n 'data-name': data.name,\n 'data-size': data.ref ? (data.size | 0) : (data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0)),\n 'data-mime': data.mime,\n };\n }\n },\n};\n\n/**\n * The main object which performs all the formatting actions.\n * @class Drafty\n * @constructor\n */\nconst Drafty = function() {\n this.txt = '';\n this.fmt = [];\n this.ent = [];\n}\n\n/**\n * Initialize Drafty document to a plain text string.\n *\n * @param {string} plainText - string to use as Drafty content.\n *\n * @returns new Drafty document or null is plainText is not a string or undefined.\n */\nDrafty.init = function(plainText) {\n if (typeof plainText == 'undefined') {\n plainText = '';\n } else if (typeof plainText != 'string') {\n return null;\n }\n\n return {\n txt: plainText\n };\n}\n\n/**\n * Parse plain text into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {string} content - plain-text content to parse.\n * @return {Drafty} parsed document or null if the source is not plain text.\n */\nDrafty.parse = function(content) {\n // Make sure we are parsing strings only.\n if (typeof content != 'string') {\n return null;\n }\n\n // Split text into lines. It makes further processing easier.\n const lines = content.split(/\\r?\\n/);\n\n // Holds entities referenced from text\n const entityMap = [];\n const entityIndex = {};\n\n // Processing lines one by one, hold intermediate result in blx.\n const blx = [];\n lines.forEach((line) => {\n let spans = [];\n let entities;\n\n // Find formatted spans in the string.\n // Try to match each style.\n INLINE_STYLES.forEach((tag) => {\n // Each style could be matched multiple times.\n spans = spans.concat(spannify(line, tag.start, tag.end, tag.name));\n });\n\n let block;\n if (spans.length == 0) {\n block = {\n txt: line\n };\n } else {\n // Sort spans by style occurence early -> late, then by length: first long then short.\n spans.sort((a, b) => {\n const diff = a.at - b.at;\n return diff != 0 ? diff : b.end - a.end;\n });\n\n // Convert an array of possibly overlapping spans into a tree.\n spans = toSpanTree(spans);\n\n // Build a tree representation of the entire string, not\n // just the formatted parts.\n const chunks = chunkify(line, 0, line.length, spans);\n\n const drafty = draftify(chunks, 0);\n\n block = {\n txt: drafty.txt,\n fmt: drafty.fmt\n };\n }\n\n // Extract entities from the cleaned up string.\n entities = extractEntities(block.txt);\n if (entities.length > 0) {\n const ranges = [];\n for (let i in entities) {\n // {offset: match['index'], unique: match[0], len: match[0].length, data: ent.packer(), type: ent.name}\n const entity = entities[i];\n let index = entityIndex[entity.unique];\n if (!index) {\n index = entityMap.length;\n entityIndex[entity.unique] = index;\n entityMap.push({\n tp: entity.type,\n data: entity.data\n });\n }\n ranges.push({\n at: entity.offset,\n len: entity.len,\n key: index\n });\n }\n block.ent = ranges;\n }\n\n blx.push(block);\n });\n\n const result = {\n txt: ''\n };\n\n // Merge lines and save line breaks as BR inline formatting.\n if (blx.length > 0) {\n result.txt = blx[0].txt;\n result.fmt = (blx[0].fmt || []).concat(blx[0].ent || []);\n\n for (let i = 1; i < blx.length; i++) {\n const block = blx[i];\n const offset = result.txt.length + 1;\n\n result.fmt.push({\n tp: 'BR',\n len: 1,\n at: offset - 1\n });\n\n result.txt += ' ' + block.txt;\n if (block.fmt) {\n result.fmt = result.fmt.concat(block.fmt.map((s) => {\n s.at += offset;\n return s;\n }));\n }\n if (block.ent) {\n result.fmt = result.fmt.concat(block.ent.map((s) => {\n s.at += offset;\n return s;\n }));\n }\n }\n\n if (result.fmt.length == 0) {\n delete result.fmt;\n }\n\n if (entityMap.length > 0) {\n result.ent = entityMap;\n }\n }\n return result;\n}\n\n/**\n * Append one Drafty document to another.\n *\n * @param {Drafty} first - Drafty document to append to.\n * @param {Drafty|string} second - Drafty document or string being appended.\n *\n * @return {Drafty} first document with the second appended to it.\n */\nDrafty.append = function(first, second) {\n if (!first) {\n return second;\n }\n if (!second) {\n return first;\n }\n\n first.txt = first.txt || '';\n const len = first.txt.length;\n\n if (typeof second == 'string') {\n first.txt += second;\n } else if (second.txt) {\n first.txt += second.txt;\n }\n\n if (Array.isArray(second.fmt)) {\n first.fmt = first.fmt || [];\n if (Array.isArray(second.ent)) {\n first.ent = first.ent || [];\n }\n second.fmt.forEach(src => {\n const fmt = {\n at: (src.at | 0) + len,\n len: src.len | 0\n };\n // Special case for the outside of the normal rendering flow styles.\n if (src.at == -1) {\n fmt.at = -1;\n fmt.len = 0;\n }\n if (src.tp) {\n fmt.tp = src.tp;\n } else {\n fmt.key = first.ent.length;\n first.ent.push(second.ent[src.key || 0]);\n }\n first.fmt.push(fmt);\n });\n }\n\n return first;\n}\n\n/**\n * Description of an image to attach.\n * @typedef {Object} ImageDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the image, e.g. \"image/png\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - base64-encoded image content. Could be null/undefined.\n * @property {string} preview - base64-encoded thumbnail of the image.\n * @property {integer} width - width of the image.\n * @property {integer} height - height of the image.\n * @property {string} filename - file name suggestion for downloading the image.\n * @property {integer} size - size of the image in bytes. Treat is as an untrusted hint.\n * @property {string} _tempPreview - base64-encoded image preview used during upload process; not serializable.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Insert inline image into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add image to.\n * @param {integer} at - index where the object is inserted. The length of the image is always 1.\n * @param {ImageDesc} imageDesc - object with image paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertImage = function(content, at, imageDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'IM',\n data: {\n mime: imageDesc.mime,\n ref: imageDesc.refurl,\n val: imageDesc.bits || imageDesc.preview,\n width: imageDesc.width,\n height: imageDesc.height,\n name: imageDesc.filename,\n size: imageDesc.size | 0,\n }\n };\n\n if (imageDesc.urlPromise) {\n ex.data._tempPreview = imageDesc._tempPreview;\n ex.data._processing = true;\n imageDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._tempPreview = undefined;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Description of a video to attach.\n * @typedef {Object} VideoDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the video, e.g. \"video/mpeg\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - in-band base64-encoded image data. Could be null/undefined.\n * @property {string} preview - base64-encoded screencapture from the video. Could be null/undefined.\n * @property {string} preref - reference to screencapture from the video. Could be null/undefined.\n * @property {integer} width - width of the video.\n * @property {integer} height - height of the video.\n * @property {integer} duration - duration of the video.\n * @property {string} filename - file name suggestion for downloading the video.\n * @property {integer} size - size of the video in bytes. Treat is as an untrusted hint.\n * @property {string} _tempPreview - base64-encoded screencapture used during upload process; not serializable.\n * @property {Promise} urlPromise - array of two promises, which return URLs of video and preview uploads correspondingly\n * (either could be null).\n */\n\n/**\n * Insert inline image into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add video to.\n * @param {integer} at - index where the object is inserted. The length of the video is always 1.\n * @param {VideoDesc} videoDesc - object with video paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertVideo = function(content, at, videoDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'VD',\n data: {\n mime: videoDesc.mime,\n ref: videoDesc.refurl,\n val: videoDesc.bits,\n preref: videoDesc.preref,\n preview: videoDesc.preview,\n width: videoDesc.width,\n height: videoDesc.height,\n duration: videoDesc.duration | 0,\n name: videoDesc.filename,\n size: videoDesc.size | 0,\n }\n };\n\n if (videoDesc.urlPromise) {\n ex.data._tempPreview = videoDesc._tempPreview;\n ex.data._processing = true;\n videoDesc.urlPromise.then(\n urls => {\n ex.data.ref = urls[0];\n ex.data.preref = urls[1];\n ex.data._tempPreview = undefined;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Description of an audio recording to attach.\n * @typedef {Object} AudioDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the audio, e.g. \"audio/ogg\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - base64-encoded audio content. Could be null/undefined.\n * @property {integer} duration - duration of the record in milliseconds.\n * @property {string} preview - base64 encoded short array of amplitude values 0..100.\n * @property {string} filename - file name suggestion for downloading the audio.\n * @property {integer} size - size of the recording in bytes. Treat is as an untrusted hint.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Insert audio recording into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add audio record to.\n * @param {integer} at - index where the object is inserted. The length of the record is always 1.\n * @param {AudioDesc} audioDesc - object with the audio paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertAudio = function(content, at, audioDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'AU',\n data: {\n mime: audioDesc.mime,\n val: audioDesc.bits,\n duration: audioDesc.duration | 0,\n preview: audioDesc.preview,\n name: audioDesc.filename,\n size: audioDesc.size | 0,\n ref: audioDesc.refurl\n }\n };\n\n if (audioDesc.urlPromise) {\n ex.data._processing = true;\n audioDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Create a (self-contained) video call Drafty document.\n * @memberof Drafty\n * @static\n * @param {boolean} audioOnly true
if the call is initially audio-only.\n * @returns Video Call drafty document.\n */\nDrafty.videoCall = function(audioOnly) {\n const content = {\n txt: ' ',\n fmt: [{\n at: 0,\n len: 1,\n key: 0\n }],\n ent: [{\n tp: 'VC',\n data: {\n aonly: audioOnly\n },\n }]\n };\n return content;\n}\n\n/**\n * Update video call (VC) entity with the new status and duration.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - VC document to update.\n * @param {object} params - new video call parameters.\n * @param {string} params.state - state of video call.\n * @param {number} params.duration - duration of the video call in milliseconds.\n *\n * @returns the same document with update applied.\n */\nDrafty.updateVideoCall = function(content, params) {\n // The video element could be just a format or a format + entity.\n // Must ensure it's the latter first.\n const fmt = ((content || {}).fmt || [])[0];\n if (!fmt) {\n // Unrecognized content.\n return content;\n }\n\n let ent;\n if (fmt.tp == 'VC') {\n // Just a format, convert to format + entity.\n delete fmt.tp;\n fmt.key = 0;\n ent = {\n tp: 'VC'\n };\n content.ent = [ent];\n } else {\n ent = (content.ent || [])[fmt.key | 0];\n if (!ent || ent.tp != 'VC') {\n // Not a VC entity.\n return content;\n }\n }\n ent.data = ent.data || {};\n Object.assign(ent.data, params);\n return content;\n}\n\n/**\n * Create a quote to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {string} header - Quote header (title, etc.).\n * @param {string} uid - UID of the author to mention.\n * @param {Drafty} body - Body of the quoted message.\n *\n * @returns Reply quote Drafty doc with the quote formatting.\n */\nDrafty.quote = function(header, uid, body) {\n const quote = Drafty.append(Drafty.appendLineBreak(Drafty.mention(header, uid)), body);\n\n // Wrap into a quote.\n quote.fmt.push({\n at: 0,\n len: quote.txt.length,\n tp: 'QQ'\n });\n\n return quote;\n}\n\n/**\n * Create a Drafty document with a mention.\n *\n * @param {string} name - mentioned name.\n * @param {string} uid - mentioned user ID.\n *\n * @returns {Drafty} document with the mention.\n */\nDrafty.mention = function(name, uid) {\n return {\n txt: name || '',\n fmt: [{\n at: 0,\n len: (name || '').length,\n key: 0\n }],\n ent: [{\n tp: 'MN',\n data: {\n val: uid\n }\n }]\n };\n}\n\n/**\n * Append a link to a Drafty document.\n *\n * @param {Drafty} content - Drafty document to append link to.\n * @param {Object} linkData - Link info in format {txt: 'ankor text', url: 'http://...'}
.\n *\n * @returns {Drafty} the same document as content
.\n */\nDrafty.appendLink = function(content, linkData) {\n content = content || {\n txt: ''\n };\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: content.txt.length,\n len: linkData.txt.length,\n key: content.ent.length\n });\n content.txt += linkData.txt;\n\n const ex = {\n tp: 'LN',\n data: {\n url: linkData.url\n }\n }\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Append image to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add image to.\n * @param {ImageDesc} imageDesc - object with image paramenets.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendImage = function(content, imageDesc) {\n content = content || {\n txt: ''\n };\n content.txt += ' ';\n return Drafty.insertImage(content, content.txt.length - 1, imageDesc);\n}\n\n/**\n * Append audio recodring to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add recording to.\n * @param {AudioDesc} audioDesc - object with audio data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendAudio = function(content, audioDesc) {\n content = content || {\n txt: ''\n };\n content.txt += ' ';\n return Drafty.insertAudio(content, content.txt.length - 1, audioDesc);\n}\n\n/**\n * Description of a file to attach.\n * @typedef {Object} AttachmentDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the attachment, e.g. \"application/octet-stream\"\n * @property {string} data - base64-encoded in-band content of small attachments. Could be null/undefined.\n * @property {string} filename - file name suggestion for downloading the attachment.\n * @property {integer} size - size of the file in bytes. Treat is as an untrusted hint.\n * @property {string} refurl - reference to the out-of-band content. Could be null/undefined.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Attach file to Drafty content. Either as a blob or as a reference.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to attach file to.\n * @param {AttachmentDesc} object - containing attachment description and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.attachFile = function(content, attachmentDesc) {\n content = content || {\n txt: ''\n };\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: -1,\n len: 0,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'EX',\n data: {\n mime: attachmentDesc.mime,\n val: attachmentDesc.data,\n name: attachmentDesc.filename,\n ref: attachmentDesc.refurl,\n size: attachmentDesc.size | 0\n }\n }\n if (attachmentDesc.urlPromise) {\n ex.data._processing = true;\n attachmentDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._processing = undefined;\n },\n _ => {\n /* catch the error, otherwise it will appear in the console. */\n ex.data._processing = undefined;\n }\n );\n }\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Wraps drafty document into a simple formatting style.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - document or string to wrap into a style.\n * @param {string} style - two-letter style to wrap into.\n * @param {number} at - index where the style starts, default 0.\n * @param {number} len - length of the form content, default all of it.\n *\n * @return {Drafty} updated document.\n */\nDrafty.wrapInto = function(content, style, at, len) {\n if (typeof content == 'string') {\n content = {\n txt: content\n };\n }\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at || 0,\n len: len || content.txt.length,\n tp: style,\n });\n\n return content;\n}\n\n/**\n * Wraps content into an interactive form.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - to wrap into a form.\n * @param {number} at - index where the forms starts.\n * @param {number} len - length of the form content.\n *\n * @return {Drafty} updated document.\n */\nDrafty.wrapAsForm = function(content, at, len) {\n return Drafty.wrapInto(content, 'FM', at, len);\n}\n\n/**\n * Insert clickable button into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - Drafty document to insert button to or a string to be used as button text.\n * @param {number} at - location where the button is inserted.\n * @param {number} len - the length of the text to be used as button title.\n * @param {string} name - the button. Client should return it to the server when the button is clicked.\n * @param {string} actionType - the type of the button, one of 'url' or 'pub'.\n * @param {string} actionValue - the value to return on click:\n * @param {string} refUrl - the URL to go to when the 'url' button is clicked.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertButton = function(content, at, len, name, actionType, actionValue, refUrl) {\n if (typeof content == 'string') {\n content = {\n txt: content\n };\n }\n\n if (!content || !content.txt || content.txt.length < at + len) {\n return null;\n }\n\n if (len <= 0 || ['url', 'pub'].indexOf(actionType) == -1) {\n return null;\n }\n // Ensure refUrl is a string.\n if (actionType == 'url' && !refUrl) {\n return null;\n }\n refUrl = '' + refUrl;\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: len,\n key: content.ent.length\n });\n content.ent.push({\n tp: 'BN',\n data: {\n act: actionType,\n val: actionValue,\n ref: refUrl,\n name: name\n }\n });\n\n return content;\n}\n\n/**\n * Append clickable button to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - Drafty document to insert button to or a string to be used as button text.\n * @param {string} title - the text to be used as button title.\n * @param {string} name - the button. Client should return it to the server when the button is clicked.\n * @param {string} actionType - the type of the button, one of 'url' or 'pub'.\n * @param {string} actionValue - the value to return on click:\n * @param {string} refUrl - the URL to go to when the 'url' button is clicked.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendButton = function(content, title, name, actionType, actionValue, refUrl) {\n content = content || {\n txt: ''\n };\n const at = content.txt.length;\n content.txt += title;\n return Drafty.insertButton(content, at, title.length, name, actionType, actionValue, refUrl);\n}\n\n/**\n * Attach a generic JS object. The object is attached as a json string.\n * Intended for representing a form response.\n *\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - Drafty document to attach file to.\n * @param {Object} data - data to convert to json string and attach.\n * @returns {Drafty} the same document as content
.\n */\nDrafty.attachJSON = function(content, data) {\n content = content || {\n txt: ''\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: -1,\n len: 0,\n key: content.ent.length\n });\n\n content.ent.push({\n tp: 'EX',\n data: {\n mime: JSON_MIME_TYPE,\n val: data\n }\n });\n\n return content;\n}\n/**\n * Append line break to a Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - Drafty document to append linebreak to.\n * @returns {Drafty} the same document as content
.\n */\nDrafty.appendLineBreak = function(content) {\n content = content || {\n txt: ''\n };\n content.fmt = content.fmt || [];\n content.fmt.push({\n at: content.txt.length,\n len: 1,\n tp: 'BR'\n });\n content.txt += ' ';\n\n return content;\n}\n/**\n * Given Drafty document, convert it to HTML.\n * No attempt is made to strip pre-existing html markup.\n * This is potentially unsafe because content.txt
may contain malicious HTML\n * markup.\n * @memberof Tinode.Drafty\n * @static\n *\n * @param {Drafty} doc - document to convert.\n *\n * @returns {string} HTML-representation of content.\n */\nDrafty.UNSAFE_toHTML = function(doc) {\n const tree = draftyToTree(doc);\n const htmlFormatter = function(type, data, values) {\n const tag = DECORATORS[type];\n let result = values ? values.join('') : '';\n if (tag) {\n result = tag.open(data) + result + tag.close(data);\n }\n return result;\n };\n return treeBottomUp(tree, htmlFormatter, 0);\n}\n\n/**\n * Callback for applying custom formatting to a Drafty document.\n * Called once for each style span.\n * @memberof Drafty\n * @static\n *\n * @callback Formatter\n * @param {string} style - style code such as \"ST\" or \"IM\".\n * @param {Object} data - entity's data.\n * @param {Object} values - possibly styled subspans contained in this style span.\n * @param {number} index - index of the element guaranteed to be unique.\n */\n\n/**\n * Convert Drafty document to a representation suitable for display.\n * The context
may expose a function getFormatter(style)
. If it's available\n * it will call it to obtain a formatter
for a subtree of styles under the style
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|Object} content - Drafty document to transform.\n * @param {Formatter} formatter - callback which formats individual elements.\n * @param {Object} context - context provided to formatter as this
.\n *\n * @return {Object} transformed object\n */\nDrafty.format = function(original, formatter, context) {\n return treeBottomUp(draftyToTree(original), formatter, 0, [], context);\n}\n\n/**\n * Shorten Drafty document making the drafty text no longer than the limit.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characrets to shorten to.\n * @param {boolean} light - remove heavy data from entities.\n * @returns new shortened Drafty object leaving the original intact.\n */\nDrafty.shorten = function(original, limit, light) {\n let tree = draftyToTree(original);\n tree = shortenTree(tree, limit, '…');\n if (tree && light) {\n tree = lightEntity(tree);\n }\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Transform Drafty doc for forwarding: strip leading @mention and any leading line breaks or whitespace.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @returns converted Drafty object leaving the original intact.\n */\nDrafty.forwardedContent = function(original) {\n let tree = draftyToTree(original);\n const rmMention = function(node) {\n if (node.type == 'MN') {\n if (!node.parent || !node.parent.type) {\n return null;\n }\n }\n return node;\n }\n // Strip leading mention.\n tree = treeTopDown(tree, rmMention);\n // Remove leading whitespace.\n tree = lTrim(tree);\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Prepare Drafty doc for wrapping into QQ as a reply:\n * - Replace forwarding mention with symbol '➦' and remove data (UID).\n * - Remove quoted text completely.\n * - Replace line breaks with spaces.\n * - Strip entities of heavy content.\n * - Move attachments to the end of the document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characters to shorten to.\n * @returns converted Drafty object leaving the original intact.\n */\nDrafty.replyContent = function(original, limit) {\n const convMNnQQnBR = function(node) {\n if (node.type == 'QQ') {\n return null;\n } else if (node.type == 'MN') {\n if ((!node.parent || !node.parent.type) && (node.text || '').startsWith('➦')) {\n node.text = '➦';\n delete node.children;\n delete node.data;\n }\n } else if (node.type == 'BR') {\n node.text = ' ';\n delete node.type;\n delete node.children;\n }\n return node;\n }\n\n let tree = draftyToTree(original);\n if (!tree) {\n return original;\n }\n\n // Strip leading mention.\n tree = treeTopDown(tree, convMNnQQnBR);\n // Move attachments to the end of the doc.\n tree = attachmentsToEnd(tree, MAX_PREVIEW_ATTACHMENTS);\n // Shorten the doc.\n tree = shortenTree(tree, limit, '…');\n // Strip heavy elements except IM.data['val'] and VD.data['preview'] (have to keep them to generate previews later).\n const filter = node => {\n switch (node.type) {\n case 'IM':\n return ['val'];\n case 'VD':\n return ['preview'];\n }\n return null;\n };\n tree = lightEntity(tree, filter);\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n\n/**\n * Generate drafty preview:\n * - Shorten the document.\n * - Strip all heavy entity data leaving just inline styles and entity references.\n * - Replace line breaks with spaces.\n * - Replace content of QQ with a space.\n * - Replace forwarding mention with symbol '➦'.\n * move all attachments to the end of the document and make them visible.\n * The context
may expose a function getFormatter(style)
. If it's available\n * it will call it to obtain a formatter
for a subtree of styles under the style
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characters to shorten to.\n * @param {boolean} forwarding - this a forwarding message preview.\n * @returns new shortened Drafty object leaving the original intact.\n */\nDrafty.preview = function(original, limit, forwarding) {\n let tree = draftyToTree(original);\n\n // Move attachments to the end.\n tree = attachmentsToEnd(tree, MAX_PREVIEW_ATTACHMENTS);\n\n // Convert leading mention to '➦' and replace QQ and BR with a space ' '.\n const convMNnQQnBR = function(node) {\n if (node.type == 'MN') {\n if ((!node.parent || !node.parent.type) && (node.text || '').startsWith('➦')) {\n node.text = '➦';\n delete node.children;\n }\n } else if (node.type == 'QQ') {\n node.text = ' ';\n delete node.children;\n } else if (node.type == 'BR') {\n node.text = ' ';\n delete node.children;\n delete node.type;\n }\n return node;\n }\n tree = treeTopDown(tree, convMNnQQnBR);\n\n tree = shortenTree(tree, limit, '…');\n if (forwarding) {\n // Keep some IM and VD data for preview.\n const filter = {\n IM: ['val'],\n VD: ['preview']\n };\n tree = lightEntity(tree, node => {\n return filter[node.type];\n });\n } else {\n tree = lightEntity(tree);\n }\n\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Given Drafty document, convert it to plain text.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to convert to plain text.\n * @returns {string} plain-text representation of the drafty document.\n */\nDrafty.toPlainText = function(content) {\n return typeof content == 'string' ? content : content.txt;\n}\n\n/**\n * Check if the document has no markup and no entities.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - content to check for presence of markup.\n * @returns true
is content is plain text, false
otherwise.\n */\nDrafty.isPlainText = function(content) {\n return typeof content == 'string' || !(content.fmt || content.ent);\n}\n\n/**\n * Convert document to plain text with markdown. All elements which cannot\n * be represented in markdown are stripped.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to convert to plain text with markdown.\n */\nDrafty.toMarkdown = function(content) {\n let tree = draftyToTree(content);\n const mdFormatter = function(type, _, values) {\n const def = FORMAT_TAGS[type];\n let result = (values ? values.join('') : '');\n if (def) {\n if (def.isVoid) {\n result = def.md_tag || '';\n } else if (def.md_tag) {\n result = def.md_tag + result + def.md_tag;\n }\n }\n return result;\n };\n return treeBottomUp(tree, mdFormatter, 0);\n}\n\n/**\n * Checks if the object represets is a valid Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - content to check for validity.\n * @returns true
is content is valid, false
otherwise.\n */\nDrafty.isValid = function(content) {\n if (!content) {\n return false;\n }\n\n const {\n txt,\n fmt,\n ent\n } = content;\n\n if (!txt && txt !== '' && !fmt && !ent) {\n return false;\n }\n\n const txt_type = typeof txt;\n if (txt_type != 'string' && txt_type != 'undefined' && txt !== null) {\n return false;\n }\n\n if (typeof fmt != 'undefined' && !Array.isArray(fmt) && fmt !== null) {\n return false;\n }\n\n if (typeof ent != 'undefined' && !Array.isArray(ent) && ent !== null) {\n return false;\n }\n return true;\n}\n\n/**\n * Check if the drafty document has attachments: style EX and outside of normal rendering flow,\n * i.e. at = -1
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to check for attachments.\n * @returns true
if there are attachments.\n */\nDrafty.hasAttachments = function(content) {\n if (!Array.isArray(content.fmt)) {\n return false;\n }\n for (let i in content.fmt) {\n const fmt = content.fmt[i];\n if (fmt && fmt.at < 0) {\n const ent = content.ent[fmt.key | 0];\n return ent && ent.tp == 'EX' && ent.data;\n }\n }\n return false;\n}\n\n/**\n * Callback for enumerating entities in a Drafty document.\n * Called once for each entity.\n * @memberof Drafty\n * @static\n *\n * @callback EntityCallback\n * @param {Object} data entity data.\n * @param {string} entity type.\n * @param {number} index entity's index in `content.ent`.\n *\n * @return 'true-ish' to stop processing, 'false-ish' otherwise.\n */\n\n/**\n * Enumerate attachments: style EX and outside of normal rendering flow, i.e. at = -1
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to process for attachments.\n * @param {EntityCallback} callback - callback to call for each attachment.\n * @param {Object} context - value of \"this\" for callback.\n */\nDrafty.attachments = function(content, callback, context) {\n if (!Array.isArray(content.fmt)) {\n return;\n }\n let count = 0;\n for (let i in content.fmt) {\n let fmt = content.fmt[i];\n if (fmt && fmt.at < 0) {\n const ent = content.ent[fmt.key | 0];\n if (ent && ent.tp == 'EX' && ent.data) {\n if (callback.call(context, ent.data, count++, 'EX')) {\n break;\n }\n }\n }\n };\n}\n\n/**\n * Check if the drafty document has entities.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to check for entities.\n * @returns true
if there are entities.\n */\nDrafty.hasEntities = function(content) {\n return content.ent && content.ent.length > 0;\n}\n\n/**\n * Enumerate entities. Enumeration stops if callback returns 'true'.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with entities to enumerate.\n * @param {EntityCallback} callback - callback to call for each entity.\n * @param {Object} context - value of \"this\" for callback.\n *\n */\nDrafty.entities = function(content, callback, context) {\n if (content.ent && content.ent.length > 0) {\n for (let i in content.ent) {\n if (content.ent[i]) {\n if (callback.call(context, content.ent[i].data, i, content.ent[i].tp)) {\n break;\n }\n }\n }\n }\n}\n\n/**\n * Callback for enumerating styles (inline formats) in a Drafty document.\n * Called once for each style.\n * @memberof Drafty\n * @static\n *\n * @callback StyleCallback\n * @param {string} tp - format type.\n * @param {number} at - starting position of the format in text.\n * @param {number} len - extent of the format in characters.\n * @param {number} key - index of the entity if format is a reference.\n * @param {number} index - style's index in `content.fmt`.\n *\n * @return 'true-ish' to stop processing, 'false-ish' otherwise.\n */\n\n/**\n * Enumerate styles (inline formats). Enumeration stops if callback returns 'true'.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with styles (formats) to enumerate.\n * @param {StyleCallback} callback - callback to call for each format.\n * @param {Object} context - value of \"this\" for callback.\n */\nDrafty.styles = function(content, callback, context) {\n if (content.fmt && content.fmt.length > 0) {\n for (let i in content.fmt) {\n const fmt = content.fmt[i];\n if (fmt) {\n if (callback.call(context, fmt.tp, fmt.at, fmt.len, fmt.key, i)) {\n break;\n }\n }\n }\n }\n}\n\n/**\n * Remove unrecognized fields from entity data\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with entities to enumerate.\n * @returns content.\n */\nDrafty.sanitizeEntities = function(content) {\n if (content && content.ent && content.ent.length > 0) {\n for (let i in content.ent) {\n const ent = content.ent[i];\n if (ent && ent.data) {\n const data = copyEntData(ent.data);\n if (data) {\n content.ent[i].data = data;\n } else {\n delete content.ent[i].data;\n }\n }\n }\n }\n return content;\n}\n\n/**\n * Given the entity, get URL which can be used for downloading\n * entity data.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the URl from.\n * @returns {string} URL to download entity data or null
.\n */\nDrafty.getDownloadUrl = function(entData) {\n let url = null;\n if (entData.mime != JSON_MIME_TYPE && entData.val) {\n url = base64toObjectUrl(entData.val, entData.mime, Drafty.logger);\n } else if (typeof entData.ref == 'string') {\n url = entData.ref;\n }\n return url;\n}\n\n/**\n * Check if the entity data is not ready for sending, such as being uploaded to the server.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entity.data to get the URl from.\n * @returns {boolean} true if upload is in progress, false otherwise.\n */\nDrafty.isProcessing = function(entData) {\n return !!entData._processing;\n}\n\n/**\n * Given the entity, get URL which can be used for previewing\n * the entity.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entity.data to get the URl from.\n *\n * @returns {string} url for previewing or null if no such url is available.\n */\nDrafty.getPreviewUrl = function(entData) {\n return entData.val ? base64toObjectUrl(entData.val, entData.mime, Drafty.logger) : null;\n}\n\n/**\n * Get approximate size of the entity.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the size for.\n * @returns {number} size of entity data in bytes.\n */\nDrafty.getEntitySize = function(entData) {\n // Either size hint or length of value. The value is base64 encoded,\n // the actual object size is smaller than the encoded length.\n return entData.size ? entData.size : entData.val ? (entData.val.length * 0.75) | 0 : 0;\n}\n\n/**\n * Get entity mime type.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the type for.\n * @returns {string} mime type of entity.\n */\nDrafty.getEntityMimeType = function(entData) {\n return entData.mime || 'text/plain';\n}\n\n/**\n * Get HTML tag for a given two-letter style name.\n * @memberof Drafty\n * @static\n *\n * @param {string} style - two-letter style, like ST or LN.\n *\n * @returns {string} HTML tag name if style is found, {code: undefined} if style is falsish or not found.\n */\nDrafty.tagName = function(style) {\n return FORMAT_TAGS[style] && FORMAT_TAGS[style].html_tag;\n}\n\n/**\n * For a given data bundle generate an object with HTML attributes,\n * for instance, given {url: \"http://www.example.com/\"} return\n * {href: \"http://www.example.com/\"}\n * @memberof Drafty\n * @static\n *\n * @param {string} style - two-letter style to generate attributes for.\n * @param {Object} data - data bundle to convert to attributes\n *\n * @returns {Object} object with HTML attributes.\n */\nDrafty.attrValue = function(style, data) {\n if (data && DECORATORS[style] && DECORATORS[style].props) {\n return DECORATORS[style].props(data);\n }\n\n return undefined;\n}\n\n/**\n * Drafty MIME type.\n * @memberof Drafty\n * @static\n *\n * @returns {string} content-Type \"text/x-drafty\".\n */\nDrafty.getContentType = function() {\n return DRAFTY_MIME_TYPE;\n}\n\n// =================\n// Utility methods.\n// =================\n\n// Take a string and defined earlier style spans, re-compose them into a tree where each leaf is\n// a same-style (including unstyled) string. I.e. 'hello *bold _italic_* and ~more~ world' ->\n// ('hello ', (b: 'bold ', (i: 'italic')), ' and ', (s: 'more'), ' world');\n//\n// This is needed in order to clear markup, i.e. 'hello *world*' -> 'hello world' and convert\n// ranges from markup-ed offsets to plain text offsets.\nfunction chunkify(line, start, end, spans) {\n const chunks = [];\n\n if (spans.length == 0) {\n return [];\n }\n\n for (let i in spans) {\n // Get the next chunk from the queue\n const span = spans[i];\n\n // Grab the initial unstyled chunk\n if (span.at > start) {\n chunks.push({\n txt: line.slice(start, span.at)\n });\n }\n\n // Grab the styled chunk. It may include subchunks.\n const chunk = {\n tp: span.tp\n };\n const chld = chunkify(line, span.at + 1, span.end, span.children);\n if (chld.length > 0) {\n chunk.children = chld;\n } else {\n chunk.txt = span.txt;\n }\n chunks.push(chunk);\n start = span.end + 1; // '+1' is to skip the formatting character\n }\n\n // Grab the remaining unstyled chunk, after the last span\n if (start < end) {\n chunks.push({\n txt: line.slice(start, end)\n });\n }\n\n return chunks;\n}\n\n// Detect starts and ends of formatting spans. Unformatted spans are\n// ignored at this stage.\nfunction spannify(original, re_start, re_end, type) {\n const result = [];\n let index = 0;\n let line = original.slice(0); // make a copy;\n\n while (line.length > 0) {\n // match[0]; // match, like '*abc*'\n // match[1]; // match captured in parenthesis, like 'abc'\n // match['index']; // offset where the match started.\n\n // Find the opening token.\n const start = re_start.exec(line);\n if (start == null) {\n break;\n }\n\n // Because javascript RegExp does not support lookbehind, the actual offset may not point\n // at the markup character. Find it in the matched string.\n let start_offset = start['index'] + start[0].lastIndexOf(start[1]);\n // Clip the processed part of the string.\n line = line.slice(start_offset + 1);\n // start_offset is an offset within the clipped string. Convert to original index.\n start_offset += index;\n // Index now point to the beginning of 'line' within the 'original' string.\n index = start_offset + 1;\n\n // Find the matching closing token.\n const end = re_end ? re_end.exec(line) : null;\n if (end == null) {\n break;\n }\n let end_offset = end['index'] + end[0].indexOf(end[1]);\n // Clip the processed part of the string.\n line = line.slice(end_offset + 1);\n // Update offsets\n end_offset += index;\n // Index now points to the beginning of 'line' within the 'original' string.\n index = end_offset + 1;\n\n result.push({\n txt: original.slice(start_offset + 1, end_offset),\n children: [],\n at: start_offset,\n end: end_offset,\n tp: type\n });\n }\n\n return result;\n}\n\n// Convert linear array or spans into a tree representation.\n// Keep standalone and nested spans, throw away partially overlapping spans.\nfunction toSpanTree(spans) {\n if (spans.length == 0) {\n return [];\n }\n\n const tree = [spans[0]];\n let last = spans[0];\n for (let i = 1; i < spans.length; i++) {\n // Keep spans which start after the end of the previous span or those which\n // are complete within the previous span.\n if (spans[i].at > last.end) {\n // Span is completely outside of the previous span.\n tree.push(spans[i]);\n last = spans[i];\n } else if (spans[i].end <= last.end) {\n // Span is fully inside of the previous span. Push to subnode.\n last.children.push(spans[i]);\n }\n // Span could partially overlap, ignoring it as invalid.\n }\n\n // Recursively rearrange the subnodes.\n for (let i in tree) {\n tree[i].children = toSpanTree(tree[i].children);\n }\n\n return tree;\n}\n\n// Convert drafty document to a tree.\nfunction draftyToTree(doc) {\n if (!doc) {\n return null;\n }\n\n doc = (typeof doc == 'string') ? {\n txt: doc\n } : doc;\n let {\n txt,\n fmt,\n ent\n } = doc;\n\n txt = txt || '';\n if (!Array.isArray(ent)) {\n ent = [];\n }\n\n if (!Array.isArray(fmt) || fmt.length == 0) {\n if (ent.length == 0) {\n return {\n text: txt\n };\n }\n\n // Handle special case when all values in fmt are 0 and fmt therefore is skipped.\n fmt = [{\n at: 0,\n len: 0,\n key: 0\n }];\n }\n\n // Sanitize spans.\n const spans = [];\n const attachments = [];\n fmt.forEach((span) => {\n if (!span || typeof span != 'object') {\n return;\n }\n\n if (!['undefined', 'number'].includes(typeof span.at)) {\n // Present, but non-numeric 'at'.\n return;\n }\n if (!['undefined', 'number'].includes(typeof span.len)) {\n // Present, but non-numeric 'len'.\n return;\n }\n let at = span.at | 0;\n let len = span.len | 0;\n if (len < 0) {\n // Invalid span length.\n return;\n }\n\n let key = span.key || 0;\n if (ent.length > 0 && (typeof key != 'number' || key < 0 || key >= ent.length)) {\n // Invalid key value.\n return;\n }\n\n if (at <= -1) {\n // Attachment. Store attachments separately.\n attachments.push({\n start: -1,\n end: 0,\n key: key\n });\n return;\n } else if (at + len > txt.length) {\n // Span is out of bounds.\n return;\n }\n\n if (!span.tp) {\n if (ent.length > 0 && (typeof ent[key] == 'object')) {\n spans.push({\n start: at,\n end: at + len,\n key: key\n });\n }\n } else {\n spans.push({\n type: span.tp,\n start: at,\n end: at + len\n });\n }\n });\n\n // Sort spans first by start index (asc) then by length (desc), then by weight.\n spans.sort((a, b) => {\n let diff = a.start - b.start;\n if (diff != 0) {\n return diff;\n }\n diff = b.end - a.end;\n if (diff != 0) {\n return diff;\n }\n return FMT_WEIGHT.indexOf(b.type) - FMT_WEIGHT.indexOf(a.type);\n });\n\n // Move attachments to the end of the list.\n if (attachments.length > 0) {\n spans.push(...attachments);\n }\n\n spans.forEach((span) => {\n if (ent.length > 0 && !span.type && ent[span.key] && typeof ent[span.key] == 'object') {\n span.type = ent[span.key].tp;\n span.data = ent[span.key].data;\n }\n\n // Is type still undefined? Hide the invalid element!\n if (!span.type) {\n span.type = 'HD';\n }\n });\n\n let tree = spansToTree({}, txt, 0, txt.length, spans);\n\n // Flatten tree nodes.\n const flatten = function(node) {\n if (Array.isArray(node.children) && node.children.length == 1) {\n // Unwrap.\n const child = node.children[0];\n if (!node.type) {\n const parent = node.parent;\n node = child;\n node.parent = parent;\n } else if (!child.type && !child.children) {\n node.text = child.text;\n delete node.children;\n }\n }\n return node;\n }\n tree = treeTopDown(tree, flatten);\n\n return tree;\n}\n\n// Add tree node to a parent tree.\nfunction addNode(parent, n) {\n if (!n) {\n return parent;\n }\n\n if (!parent.children) {\n parent.children = [];\n }\n\n // If text is present, move it to a subnode.\n if (parent.text) {\n parent.children.push({\n text: parent.text,\n parent: parent\n });\n delete parent.text;\n }\n\n n.parent = parent;\n parent.children.push(n);\n\n return parent;\n}\n\n// Returns a tree of nodes.\nfunction spansToTree(parent, text, start, end, spans) {\n if (!spans || spans.length == 0) {\n if (start < end) {\n addNode(parent, {\n text: text.substring(start, end)\n });\n }\n return parent;\n }\n\n // Process subspans.\n for (let i = 0; i < spans.length; i++) {\n const span = spans[i];\n if (span.start < 0 && span.type == 'EX') {\n addNode(parent, {\n type: span.type,\n data: span.data,\n key: span.key,\n att: true\n });\n continue;\n }\n\n // Add un-styled range before the styled span starts.\n if (start < span.start) {\n addNode(parent, {\n text: text.substring(start, span.start)\n });\n start = span.start;\n }\n\n // Get all spans which are within the current span.\n const subspans = [];\n while (i < spans.length - 1) {\n const inner = spans[i + 1];\n if (inner.start < 0) {\n // Attachments are in the end. Stop.\n break;\n } else if (inner.start < span.end) {\n if (inner.end <= span.end) {\n const tag = FORMAT_TAGS[inner.tp] || {};\n if (inner.start < inner.end || tag.isVoid) {\n // Valid subspan: completely within the current span and\n // either non-zero length or zero length is acceptable.\n subspans.push(inner);\n }\n }\n i++;\n // Overlapping subspans are ignored.\n } else {\n // Past the end of the current span. Stop.\n break;\n }\n }\n\n addNode(parent, spansToTree({\n type: span.type,\n data: span.data,\n key: span.key\n }, text, start, span.end, subspans));\n start = span.end;\n }\n\n // Add the last unformatted range.\n if (start < end) {\n addNode(parent, {\n text: text.substring(start, end)\n });\n }\n\n return parent;\n}\n\n// Append a tree to a Drafty doc.\nfunction treeToDrafty(doc, tree, keymap) {\n if (!tree) {\n return doc;\n }\n\n doc.txt = doc.txt || '';\n\n // Checkpoint to measure length of the current tree node.\n const start = doc.txt.length;\n\n if (tree.text) {\n doc.txt += tree.text;\n } else if (Array.isArray(tree.children)) {\n tree.children.forEach((c) => {\n treeToDrafty(doc, c, keymap);\n });\n }\n\n if (tree.type) {\n const len = doc.txt.length - start;\n doc.fmt = doc.fmt || [];\n if (Object.keys(tree.data || {}).length > 0) {\n doc.ent = doc.ent || [];\n const newKey = (typeof keymap[tree.key] == 'undefined') ? doc.ent.length : keymap[tree.key];\n keymap[tree.key] = newKey;\n doc.ent[newKey] = {\n tp: tree.type,\n data: tree.data\n };\n if (tree.att) {\n // Attachment.\n doc.fmt.push({\n at: -1,\n len: 0,\n key: newKey\n });\n } else {\n doc.fmt.push({\n at: start,\n len: len,\n key: newKey\n });\n }\n } else {\n doc.fmt.push({\n tp: tree.type,\n at: start,\n len: len\n });\n }\n }\n return doc;\n}\n\n// Traverse the tree top down transforming the nodes: apply transformer to every tree node.\nfunction treeTopDown(src, transformer, context) {\n if (!src) {\n return null;\n }\n\n let dst = transformer.call(context, src);\n if (!dst || !dst.children) {\n return dst;\n }\n\n const children = [];\n for (let i in dst.children) {\n let n = dst.children[i];\n if (n) {\n n = treeTopDown(n, transformer, context);\n if (n) {\n children.push(n);\n }\n }\n }\n\n if (children.length == 0) {\n dst.children = null;\n } else {\n dst.children = children;\n }\n\n return dst;\n}\n\n// Traverse the tree bottom-up: apply formatter to every node.\n// The formatter must maintain its state through context.\nfunction treeBottomUp(src, formatter, index, stack, context) {\n if (!src) {\n return null;\n }\n\n if (stack && src.type) {\n stack.push(src.type);\n }\n\n let values = [];\n for (let i in src.children) {\n const n = treeBottomUp(src.children[i], formatter, i, stack, context);\n if (n) {\n values.push(n);\n }\n }\n if (values.length == 0) {\n if (src.text) {\n values = [src.text];\n } else {\n values = null;\n }\n }\n\n if (stack && src.type) {\n stack.pop();\n }\n\n return formatter.call(context, src.type, src.data, values, index, stack);\n}\n\n// Clip tree to the provided limit.\nfunction shortenTree(tree, limit, tail) {\n if (!tree) {\n return null;\n }\n\n if (tail) {\n limit -= tail.length;\n }\n\n const shortener = function(node) {\n if (limit <= -1) {\n // Limit -1 means the doc was already clipped.\n return null;\n }\n\n if (node.att) {\n // Attachments are unchanged.\n return node;\n }\n if (limit == 0) {\n node.text = tail;\n limit = -1;\n } else if (node.text) {\n const len = node.text.length;\n if (len > limit) {\n node.text = node.text.substring(0, limit) + tail;\n limit = -1;\n } else {\n limit -= len;\n }\n }\n return node;\n }\n\n return treeTopDown(tree, shortener);\n}\n\n// Strip heavy entities from a tree.\nfunction lightEntity(tree, allow) {\n const lightCopy = node => {\n const data = copyEntData(node.data, true, allow ? allow(node) : null);\n if (data) {\n node.data = data;\n } else {\n delete node.data;\n }\n return node;\n }\n return treeTopDown(tree, lightCopy);\n}\n\n// Remove spaces and breaks on the left.\nfunction lTrim(tree) {\n if (tree.type == 'BR') {\n tree = null;\n } else if (tree.text) {\n if (!tree.type) {\n tree.text = tree.text.trimStart();\n if (!tree.text) {\n tree = null;\n }\n }\n } else if (!tree.type && tree.children && tree.children.length > 0) {\n const c = lTrim(tree.children[0]);\n if (c) {\n tree.children[0] = c;\n } else {\n tree.children.shift();\n if (!tree.type && tree.children.length == 0) {\n tree = null;\n }\n }\n }\n return tree;\n}\n\n// Move attachments to the end. Attachments must be at the top level, no need to traverse the tree.\nfunction attachmentsToEnd(tree, limit) {\n if (!tree) {\n return null;\n }\n\n if (tree.att) {\n tree.text = ' ';\n delete tree.att;\n delete tree.children;\n } else if (tree.children) {\n const attachments = [];\n const children = [];\n for (let i in tree.children) {\n const c = tree.children[i];\n if (c.att) {\n if (attachments.length == limit) {\n // Too many attachments to preview;\n continue;\n }\n if (c.data['mime'] == JSON_MIME_TYPE) {\n // JSON attachments are not shown in preview.\n continue;\n }\n\n delete c.att;\n delete c.children;\n c.text = ' ';\n attachments.push(c);\n } else {\n children.push(c);\n }\n }\n tree.children = children.concat(attachments);\n }\n return tree;\n}\n\n// Get a list of entities from a text.\nfunction extractEntities(line) {\n let match;\n let extracted = [];\n ENTITY_TYPES.forEach((entity) => {\n while ((match = entity.re.exec(line)) !== null) {\n extracted.push({\n offset: match['index'],\n len: match[0].length,\n unique: match[0],\n data: entity.pack(match[0]),\n type: entity.name\n });\n }\n });\n\n if (extracted.length == 0) {\n return extracted;\n }\n\n // Remove entities detected inside other entities, like #hashtag in a URL.\n extracted.sort((a, b) => {\n return a.offset - b.offset;\n });\n\n let idx = -1;\n extracted = extracted.filter((el) => {\n const result = (el.offset > idx);\n idx = el.offset + el.len;\n return result;\n });\n\n return extracted;\n}\n\n// Convert the chunks into format suitable for serialization.\nfunction draftify(chunks, startAt) {\n let plain = '';\n let ranges = [];\n for (let i in chunks) {\n const chunk = chunks[i];\n if (!chunk.txt) {\n const drafty = draftify(chunk.children, plain.length + startAt);\n chunk.txt = drafty.txt;\n ranges = ranges.concat(drafty.fmt);\n }\n\n if (chunk.tp) {\n ranges.push({\n at: plain.length + startAt,\n len: chunk.txt.length,\n tp: chunk.tp\n });\n }\n\n plain += chunk.txt;\n }\n return {\n txt: plain,\n fmt: ranges\n };\n}\n\n// Create a copy of entity data with (light=false) or without (light=true) the large payload.\n// The array 'allow' contains a list of fields exempt from stripping.\nfunction copyEntData(data, light, allow) {\n if (data && Object.entries(data).length > 0) {\n allow = allow || [];\n const dc = {};\n ALLOWED_ENT_FIELDS.forEach(key => {\n if (data[key]) {\n if (light && !allow.includes(key) &&\n (typeof data[key] == 'string' || Array.isArray(data[key])) &&\n data[key].length > MAX_PREVIEW_DATA_SIZE) {\n return;\n }\n if (typeof data[key] == 'object') {\n return;\n }\n dc[key] = data[key];\n }\n });\n\n if (Object.entries(dc).length != 0) {\n return dc;\n }\n }\n return null;\n}\n\nif (typeof module != 'undefined') {\n module.exports = Drafty;\n}\n","/**\n * @file Utilities for uploading and downloading files.\n *\n * @copyright 2015-2023 Tinode LLC.\n */\n'use strict';\n\nimport CommError from './comm-error.js';\nimport {\n isUrlRelative,\n jsonParseHelper\n} from './utils.js';\n\nlet XHRProvider;\n\nfunction addURLParam(relUrl, key, value) {\n const url = new URL(relUrl, window.location.origin);\n url.searchParams.append(key, value);\n return url.toString().substring(window.location.origin.length);\n}\n\n/**\n * @class LargeFileHelper - utilities for uploading and downloading files out of band.\n * Don't instantiate this class directly. Use {Tinode.getLargeFileHelper} instead.\n * @memberof Tinode\n *\n * @param {Tinode} tinode - the main Tinode object.\n * @param {string} version - protocol version, i.e. '0'.\n */\nexport default class LargeFileHelper {\n constructor(tinode, version) {\n this._tinode = tinode;\n this._version = version;\n\n this._apiKey = tinode._apiKey;\n this._authToken = tinode.getAuthToken();\n\n // Ongoing requests.\n this.xhr = [];\n }\n\n /**\n * Start uploading the file to an endpoint at baseUrl.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {string} baseUrl base URL of upload server.\n * @param {File|Blob} data data to upload.\n * @param {string} avatarFor topic name if the upload represents an avatar.\n * @param {Callback} onProgress callback. Takes one {float} parameter 0..1\n * @param {Callback} onSuccess callback. Called when the file is successfully uploaded.\n * @param {Callback} onFailure callback. Called in case of a failure.\n *\n * @returns {Promise} resolved/rejected when the upload is completed/failed.\n */\n uploadWithBaseUrl(baseUrl, data, avatarFor, onProgress, onSuccess, onFailure) {\n let url = `/v${this._version}/file/u/`;\n if (baseUrl) {\n let base = baseUrl;\n if (base.endsWith('/')) {\n // Removing trailing slash.\n base = base.slice(0, -1);\n }\n if (base.startsWith('http://') || base.startsWith('https://')) {\n url = base + url;\n } else {\n throw new Error(`Invalid base URL '${baseUrl}'`);\n }\n }\n\n const instance = this;\n const xhr = new XHRProvider();\n this.xhr.push(xhr);\n\n xhr.open('POST', url, true);\n xhr.setRequestHeader('X-Tinode-APIKey', this._apiKey);\n if (this._authToken) {\n xhr.setRequestHeader('X-Tinode-Auth', `Token ${this._authToken.token}`);\n }\n\n let toResolve = null;\n let toReject = null;\n\n const result = new Promise((resolve, reject) => {\n toResolve = resolve;\n toReject = reject;\n });\n\n xhr.upload.onprogress = e => {\n if (e.lengthComputable) {\n if (onProgress) {\n onProgress(e.loaded / e.total);\n }\n if (this.onProgress) {\n this.onProgress(e.loaded / e.total);\n }\n }\n };\n\n xhr.onload = function() {\n let pkt;\n try {\n pkt = JSON.parse(this.response, jsonParseHelper);\n } catch (err) {\n instance._tinode.logger(\"ERROR: Invalid server response in LargeFileHelper\", this.response);\n pkt = {\n ctrl: {\n code: this.status,\n text: this.statusText\n }\n };\n }\n\n if (this.status >= 200 && this.status < 300) {\n if (toResolve) {\n toResolve(pkt.ctrl.params.url);\n }\n if (onSuccess) {\n onSuccess(pkt.ctrl);\n }\n } else if (this.status >= 400) {\n if (toReject) {\n toReject(new CommError(pkt.ctrl.text, pkt.ctrl.code));\n }\n if (onFailure) {\n onFailure(pkt.ctrl);\n }\n } else {\n instance._tinode.logger(\"ERROR: Unexpected server response status\", this.status, this.response);\n }\n };\n\n xhr.onerror = function(e) {\n if (toReject) {\n toReject(e || new Error(\"failed\"));\n }\n if (onFailure) {\n onFailure(null);\n }\n };\n\n xhr.onabort = function(e) {\n if (toReject) {\n toReject(new Error(\"upload cancelled by user\"));\n }\n if (onFailure) {\n onFailure(null);\n }\n };\n\n try {\n const form = new FormData();\n form.append('file', data);\n form.set('id', this._tinode.getNextUniqueId());\n if (avatarFor) {\n form.set('topic', avatarFor);\n }\n xhr.send(form);\n } catch (err) {\n if (toReject) {\n toReject(err);\n }\n if (onFailure) {\n onFailure(null);\n }\n }\n\n return result;\n }\n /**\n * Start uploading the file to default endpoint.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {File|Blob} data to upload\n * @param {string} avatarFor topic name if the upload represents an avatar.\n * @param {Callback} onProgress callback. Takes one {float} parameter 0..1\n * @param {Callback} onSuccess callback. Called when the file is successfully uploaded.\n * @param {Callback} onFailure callback. Called in case of a failure.\n *\n * @returns {Promise} resolved/rejected when the upload is completed/failed.\n */\n upload(data, avatarFor, onProgress, onSuccess, onFailure) {\n const baseUrl = (this._tinode._secure ? 'https://' : 'http://') + this._tinode._host;\n return this.uploadWithBaseUrl(baseUrl, data, avatarFor, onProgress, onSuccess, onFailure);\n }\n /**\n * Download the file from a given URL using GET request. This method works with the Tinode server only.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {string} relativeUrl - URL to download the file from. Must be relative url, i.e. must not contain the host.\n * @param {string=} filename - file name to use for the downloaded file.\n *\n * @returns {Promise} resolved/rejected when the download is completed/failed.\n */\n download(relativeUrl, filename, mimetype, onProgress, onError) {\n if (!isUrlRelative(relativeUrl)) {\n // As a security measure refuse to download from an absolute URL.\n if (onError) {\n onError(`The URL '${relativeUrl}' must be relative, not absolute`);\n }\n return;\n }\n if (!this._authToken) {\n if (onError) {\n onError(\"Must authenticate first\");\n }\n return;\n }\n const instance = this;\n\n const xhr = new XHRProvider();\n this.xhr.push(xhr);\n\n // Add '&asatt=1' to URL to request 'Content-Disposition: attachment' response header.\n relativeUrl = addURLParam(relativeUrl, 'asatt', '1');\n\n // Get data as blob (stored by the browser as a temporary file).\n xhr.open('GET', relativeUrl, true);\n xhr.setRequestHeader('X-Tinode-APIKey', this._apiKey);\n xhr.setRequestHeader('X-Tinode-Auth', 'Token ' + this._authToken.token);\n xhr.responseType = 'blob';\n\n xhr.onprogress = function(e) {\n if (onProgress) {\n // Passing e.loaded instead of e.loaded/e.total because e.total\n // is always 0 with gzip compression enabled by the server.\n onProgress(e.loaded);\n }\n };\n\n let toResolve = null;\n let toReject = null;\n\n const result = new Promise((resolve, reject) => {\n toResolve = resolve;\n toReject = reject;\n });\n\n // The blob needs to be saved as file. There is no known way to\n // save the blob as file other than to fake a click on an .\n xhr.onload = function() {\n if (this.status == 200) {\n const link = document.createElement('a');\n // URL.createObjectURL is not available in non-browser environment. This call will fail.\n link.href = window.URL.createObjectURL(new Blob([this.response], {\n type: mimetype\n }));\n link.style.display = 'none';\n link.setAttribute('download', filename);\n document.body.appendChild(link);\n link.click();\n document.body.removeChild(link);\n window.URL.revokeObjectURL(link.href);\n if (toResolve) {\n toResolve();\n }\n } else if (this.status >= 400 && toReject) {\n // The this.responseText is undefined, must use this.response which is a blob.\n // Need to convert this.response to JSON. The blob can only be accessed by the\n // FileReader.\n const reader = new FileReader();\n reader.onload = function() {\n try {\n const pkt = JSON.parse(this.result, jsonParseHelper);\n toReject(new CommError(pkt.ctrl.text, pkt.ctrl.code));\n } catch (err) {\n instance._tinode.logger(\"ERROR: Invalid server response in LargeFileHelper\", this.result);\n toReject(err);\n }\n };\n reader.readAsText(this.response);\n }\n };\n\n xhr.onerror = function(e) {\n if (toReject) {\n toReject(new Error(\"failed\"));\n }\n if (onError) {\n onError(e);\n }\n };\n\n xhr.onabort = function() {\n if (toReject) {\n toReject(null);\n }\n };\n\n try {\n xhr.send();\n } catch (err) {\n if (toReject) {\n toReject(err);\n }\n if (onError) {\n onError(err);\n }\n }\n\n return result;\n }\n /**\n * Try to cancel all ongoing uploads or downloads.\n * @memberof Tinode.LargeFileHelper#\n */\n cancel() {\n this.xhr.forEach(req => {\n if (req.readyState < 4) {\n req.abort();\n }\n });\n }\n /**\n * To use LargeFileHelper in a non browser context, supply XMLHttpRequest provider.\n * @static\n * @memberof LargeFileHelper\n * @param xhrProvider XMLHttpRequest provider, e.g. for node require('xhr')
.\n */\n static setNetworkProvider(xhrProvider) {\n XHRProvider = xhrProvider;\n }\n}\n","/**\n * @file Helper class for constructing {@link Tinode.GetQuery}.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n/**\n * Helper class for constructing {@link Tinode.GetQuery}.\n *\n * @class MetaGetBuilder\n * @memberof Tinode\n *\n * @param {Tinode.Topic} parent topic which instantiated this builder.\n */\nexport default class MetaGetBuilder {\n constructor(parent) {\n this.topic = parent;\n this.what = {};\n }\n\n // Get timestamp of the most recent desc update.\n #get_desc_ims() {\n return this.topic._deleted ? undefined : this.topic.updated;\n }\n\n // Get timestamp of the most recent subs update.\n #get_subs_ims() {\n if (this.topic.isP2PType()) {\n return this.#get_desc_ims();\n }\n return this.topic._deleted ? undefined : this.topic._lastSubsUpdate;\n }\n /**\n * Add query parameters to fetch messages within explicit limits.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} since - messages newer than this (inclusive);\n * @param {number=} before - older than this (exclusive)\n * @param {number=} limit - number of messages to fetch\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withData(since, before, limit) {\n this.what['data'] = {\n since: since,\n before: before,\n limit: limit\n };\n return this;\n }\n /**\n * Add query parameters to fetch messages newer than the latest saved message.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - number of messages to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterData(limit) {\n return this.withData(this.topic._maxSeq > 0 ? this.topic._maxSeq + 1 : undefined, undefined, limit);\n }\n /**\n * Add query parameters to fetch messages older than the earliest saved message.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - maximum number of messages to fetch.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withEarlierData(limit) {\n return this.withData(undefined, this.topic._minSeq > 0 ? this.topic._minSeq : undefined, limit);\n }\n /**\n * Add query parameters to fetch topic description if it's newer than the given timestamp.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch messages newer than this timestamp.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withDesc(ims) {\n this.what['desc'] = {\n ims: ims\n };\n return this;\n }\n /**\n * Add query parameters to fetch topic description if it's newer than the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterDesc() {\n return this.withDesc(this.#get_desc_ims());\n }\n /**\n * Add query parameters to fetch subscriptions.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch subscriptions modified more recently than this timestamp\n * @param {number=} limit - maximum number of subscriptions to fetch.\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withSub(ims, limit, userOrTopic) {\n const opts = {\n ims: ims,\n limit: limit\n };\n if (this.topic.getType() == 'me') {\n opts.topic = userOrTopic;\n } else {\n opts.user = userOrTopic;\n }\n this.what['sub'] = opts;\n return this;\n }\n /**\n * Add query parameters to fetch a single subscription.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch subscriptions modified more recently than this timestamp\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withOneSub(ims, userOrTopic) {\n return this.withSub(ims, undefined, userOrTopic);\n }\n /**\n * Add query parameters to fetch a single subscription if it's been updated since the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterOneSub(userOrTopic) {\n return this.withOneSub(this.topic._lastSubsUpdate, userOrTopic);\n }\n /**\n * Add query parameters to fetch subscriptions updated since the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - maximum number of subscriptions to fetch.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterSub(limit) {\n return this.withSub(this.#get_subs_ims(), limit);\n }\n /**\n * Add query parameters to fetch topic tags.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withTags() {\n this.what['tags'] = true;\n return this;\n }\n /**\n * Add query parameters to fetch user's credentials. 'me'
topic only.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withCred() {\n if (this.topic.getType() == 'me') {\n this.what['cred'] = true;\n } else {\n this.topic._tinode.logger(\"ERROR: Invalid topic type for MetaGetBuilder:withCreds\", this.topic.getType());\n }\n return this;\n }\n /**\n * Add query parameters to fetch deleted messages within explicit limits. Any/all parameters can be null.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} since - ids of messages deleted since this 'del' id (inclusive)\n * @param {number=} limit - number of deleted message ids to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withDel(since, limit) {\n if (since || limit) {\n this.what['del'] = {\n since: since,\n limit: limit\n };\n }\n return this;\n }\n /**\n * Add query parameters to fetch messages deleted after the saved 'del'
id.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - number of deleted message ids to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterDel(limit) {\n // Specify 'since' only if we have already received some messages. If\n // we have no locally cached messages then we don't care if any messages were deleted.\n return this.withDel(this.topic._maxSeq > 0 ? this.topic._maxDel + 1 : undefined, limit);\n }\n\n /**\n * Extract subquery: get an object that contains specified subquery.\n * @memberof Tinode.MetaGetBuilder#\n * @param {string} what - subquery to return: one of 'data', 'sub', 'desc', 'tags', 'cred', 'del'.\n * @returns {Object} requested subquery or undefined
.\n */\n extract(what) {\n return this.what[what];\n }\n\n /**\n * Construct parameters.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.GetQuery} Get query\n */\n build() {\n const what = [];\n let params = {};\n ['data', 'sub', 'desc', 'tags', 'cred', 'del'].forEach((key) => {\n if (this.what.hasOwnProperty(key)) {\n what.push(key);\n if (Object.getOwnPropertyNames(this.what[key]).length > 0) {\n params[key] = this.what[key];\n }\n }\n });\n if (what.length > 0) {\n params.what = what.join(' ');\n } else {\n params = undefined;\n }\n return params;\n }\n}\n","/**\n * @file Topic management.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport AccessMode from './access-mode.js';\nimport CBuffer from './cbuffer.js';\nimport CommError from './comm-error.js';\nimport * as Const from './config.js';\nimport Drafty from './drafty.js';\nimport MetaGetBuilder from './meta-builder.js';\nimport {\n mergeObj,\n mergeToCache,\n normalizeArray\n} from './utils.js';\n\n/**\n * Topic is a class representing a logical communication channel.\n */\nexport class Topic {\n /**\n * @callback onData\n * @param {Data} data - Data packet\n */\n\n /**\n * Create topic.\n * @param {string} name - Name of the topic to create.\n * @param {Object=} callbacks - Object with various event callbacks.\n * @param {onData} callbacks.onData - Callback which receives a {data}
message.\n * @param {callback} callbacks.onMeta - Callback which receives a {meta}
message.\n * @param {callback} callbacks.onPres - Callback which receives a {pres}
message.\n * @param {callback} callbacks.onInfo - Callback which receives an {info}
message.\n * @param {callback} callbacks.onMetaDesc - Callback which receives changes to topic desctioption {@link desc}.\n * @param {callback} callbacks.onMetaSub - Called for a single subscription record change.\n * @param {callback} callbacks.onSubsUpdated - Called after a batch of subscription changes have been recieved and cached.\n * @param {callback} callbacks.onDeleteTopic - Called after the topic is deleted.\n * @param {callback} callbacls.onAllMessagesReceived - Called when all requested {data}
messages have been recived.\n */\n constructor(name, callbacks) {\n // Parent Tinode object.\n this._tinode = null;\n\n // Server-provided data, locally immutable.\n // topic name\n this.name = name;\n // Timestamp when the topic was created.\n this.created = null;\n // Timestamp when the topic was last updated.\n this.updated = null;\n // Timestamp of the last messages\n this.touched = new Date(0);\n // Access mode, see AccessMode\n this.acs = new AccessMode(null);\n // Per-topic private data (accessible by current user only).\n this.private = null;\n // Per-topic public data (accessible by all users).\n this.public = null;\n // Per-topic system-provided data (accessible by all users).\n this.trusted = null;\n\n // Locally cached data\n // Subscribed users, for tracking read/recv/msg notifications.\n this._users = {};\n\n // Current value of locally issued seqId, used for pending messages.\n this._queuedSeqId = Const.LOCAL_SEQID;\n\n // The maximum known {data.seq} value.\n this._maxSeq = 0;\n // The minimum known {data.seq} value.\n this._minSeq = 0;\n // Indicator that the last request for earlier messages returned 0.\n this._noEarlierMsgs = false;\n // The maximum known deletion ID.\n this._maxDel = 0;\n // Timer object used to send 'recv' notifications.\n this._recvNotificationTimer = null;\n\n // User discovery tags\n this._tags = [];\n // Credentials such as email or phone number.\n this._credentials = [];\n // Message versions cache (e.g. for edited messages).\n // Keys: original message seq ID.\n // Values: CBuffers containing newer versions of the original message\n // ordered by seq id.\n this._messageVersions = {};\n // Message cache, sorted by message seq values, from old to new.\n this._messages = new CBuffer((a, b) => {\n return a.seq - b.seq;\n }, true);\n // Boolean, true if the topic is currently live\n this._attached = false;\n // Timestap of the most recently updated subscription.\n this._lastSubsUpdate = new Date(0);\n // Topic created but not yet synced with the server. Used only during initialization.\n this._new = true;\n // The topic is deleted at the server, this is a local copy.\n this._deleted = false;\n\n // Timer used to trgger {leave} request after a delay.\n this._delayedLeaveTimer = null;\n\n // Callbacks\n if (callbacks) {\n this.onData = callbacks.onData;\n this.onMeta = callbacks.onMeta;\n this.onPres = callbacks.onPres;\n this.onInfo = callbacks.onInfo;\n // A single desc update;\n this.onMetaDesc = callbacks.onMetaDesc;\n // A single subscription record;\n this.onMetaSub = callbacks.onMetaSub;\n // All subscription records received;\n this.onSubsUpdated = callbacks.onSubsUpdated;\n this.onTagsUpdated = callbacks.onTagsUpdated;\n this.onCredsUpdated = callbacks.onCredsUpdated;\n this.onDeleteTopic = callbacks.onDeleteTopic;\n this.onAllMessagesReceived = callbacks.onAllMessagesReceived;\n }\n }\n\n // Static methods.\n\n /**\n * Determine topic type from topic's name: grp, p2p, me, fnd, sys.\n *\n * @param {string} name - Name of the topic to test.\n * @returns {string} One of \"me\"
, \"fnd\"
, \"sys\"
, \"grp\"
,\n * \"p2p\"
or undefined
.\n */\n static topicType(name) {\n const types = {\n 'me': Const.TOPIC_ME,\n 'fnd': Const.TOPIC_FND,\n 'grp': Const.TOPIC_GRP,\n 'new': Const.TOPIC_GRP,\n 'nch': Const.TOPIC_GRP,\n 'chn': Const.TOPIC_GRP,\n 'usr': Const.TOPIC_P2P,\n 'sys': Const.TOPIC_SYS\n };\n return types[(typeof name == 'string') ? name.substring(0, 3) : 'xxx'];\n }\n\n /**\n * Check if the given topic name is a name of a 'me' topic.\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a 'me' topic, false
otherwise.\n */\n static isMeTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_ME;\n }\n\n /**\n * Check if the given topic name is a name of a group topic.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a group topic, false
otherwise.\n */\n static isGroupTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_GRP;\n }\n\n /**\n * Check if the given topic name is a name of a p2p topic.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p topic, false
otherwise.\n */\n static isP2PTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_P2P;\n }\n\n /**\n * Check if the given topic name is a name of a communication topic, i.e. P2P or group.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p or group topic, false
otherwise.\n */\n static isCommTopicName(name) {\n return Topic.isP2PTopicName(name) || Topic.isGroupTopicName(name);\n }\n\n /**\n * Check if the topic name is a name of a new topic.\n * @static\n *\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a new topic, false
otherwise.\n */\n static isNewGroupTopicName(name) {\n return (typeof name == 'string') &&\n (name.substring(0, 3) == Const.TOPIC_NEW || name.substring(0, 3) == Const.TOPIC_NEW_CHAN);\n }\n\n /**\n * Check if the topic name is a name of a channel.\n * @static\n *\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a channel, false
otherwise.\n */\n static isChannelTopicName(name) {\n return (typeof name == 'string') &&\n (name.substring(0, 3) == Const.TOPIC_CHAN || name.substring(0, 3) == Const.TOPIC_NEW_CHAN);\n }\n\n /**\n * Check if the topic is subscribed.\n * @returns {boolean} True is topic is attached/subscribed, false otherwise.\n */\n isSubscribed() {\n return this._attached;\n }\n\n /**\n * Request topic to subscribe. Wrapper for {@link Tinode#subscribe}.\n *\n * @param {Tinode.GetQuery=} getParams - get query parameters.\n * @param {Tinode.SetParams=} setParams - set parameters.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n subscribe(getParams, setParams) {\n // Clear request to leave topic.\n clearTimeout(this._delayedLeaveTimer);\n this._delayedLeaveTimer = null;\n\n // If the topic is already subscribed, return resolved promise\n if (this._attached) {\n return Promise.resolve(this);\n }\n\n // Send subscribe message, handle async response.\n // If topic name is explicitly provided, use it. If no name, then it's a new group topic,\n // use \"new\".\n return this._tinode.subscribe(this.name || Const.TOPIC_NEW, getParams, setParams).then(ctrl => {\n if (ctrl.code >= 300) {\n // Do nothing if subscription status has not changed.\n return ctrl;\n }\n\n this._attached = true;\n this._deleted = false;\n this.acs = (ctrl.params && ctrl.params.acs) ? ctrl.params.acs : this.acs;\n\n // Set topic name for new topics and add it to cache.\n if (this._new) {\n delete this._new;\n\n if (this.name != ctrl.topic) {\n // Name may change new123456 -> grpAbCdEf. Remove from cache under the old name.\n this._cacheDelSelf();\n this.name = ctrl.topic;\n }\n this._cachePutSelf();\n\n this.created = ctrl.ts;\n this.updated = ctrl.ts;\n\n if (this.name != Const.TOPIC_ME && this.name != Const.TOPIC_FND) {\n // Add the new topic to the list of contacts maintained by the 'me' topic.\n const me = this._tinode.getMeTopic();\n if (me.onMetaSub) {\n me.onMetaSub(this);\n }\n if (me.onSubsUpdated) {\n me.onSubsUpdated([this.name], 1);\n }\n }\n\n if (setParams && setParams.desc) {\n setParams.desc._noForwarding = true;\n this._processMetaDesc(setParams.desc);\n }\n }\n return ctrl;\n });\n }\n\n /**\n * Create a draft of a message without sending it to the server.\n * @memberof Tinode.Topic#\n *\n * @param {string | Object} data - Content to wrap in a draft.\n * @param {boolean=} noEcho - If true
server will not echo message back to originating\n * session. Otherwise the server will send a copy of the message to sender.\n *\n * @returns {Object} message draft.\n */\n createMessage(data, noEcho) {\n return this._tinode.createMessage(this.name, data, noEcho);\n }\n\n /**\n * Immediately publish data to topic. Wrapper for {@link Tinode#publish}.\n * @memberof Tinode.Topic#\n *\n * @param {string | Object} data - Message to publish, either plain string or a Drafty object.\n * @param {boolean=} noEcho - If true
server will not echo message back to originating\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n publish(data, noEcho) {\n return this.publishMessage(this.createMessage(data, noEcho));\n }\n\n /**\n * Publish message created by {@link Tinode.Topic#createMessage}.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub - {data} object to publish. Must be created by {@link Tinode.Topic#createMessage}\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n publishMessage(pub) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot publish on inactive topic\"));\n }\n if (this._sending) {\n return Promise.reject(new Error(\"The message is already being sent\"));\n }\n\n // Send data.\n pub._sending = true;\n pub._failed = false;\n\n // Extract refereces to attachments and out of band image records.\n let attachments = null;\n if (Drafty.hasEntities(pub.content)) {\n attachments = [];\n Drafty.entities(pub.content, data => {\n if (data) {\n if (data.ref) {\n attachments.push(data.ref);\n }\n if (data.preref) {\n attachments.push(data.preref);\n }\n }\n });\n if (attachments.length == 0) {\n attachments = null;\n }\n }\n\n return this._tinode.publishMessage(pub, attachments).then(ctrl => {\n pub._sending = false;\n pub.ts = ctrl.ts;\n this.swapMessageId(pub, ctrl.params.seq);\n this._maybeUpdateMessageVersionsCache(pub);\n this._routeData(pub);\n return ctrl;\n }).catch(err => {\n this._tinode.logger(\"WARNING: Message rejected by the server\", err);\n pub._sending = false;\n pub._failed = true;\n if (this.onData) {\n this.onData();\n }\n });\n }\n\n /**\n * Add message to local message cache, send to the server when the promise is resolved.\n * If promise is null or undefined, the message will be sent immediately.\n * The message is sent when the\n * The message should be created by {@link Tinode.Topic#createMessage}.\n * This is probably not the final API.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub - Message to use as a draft.\n * @param {Promise} prom - Message will be sent when this promise is resolved, discarded if rejected.\n *\n * @returns {Promise} derived promise.\n */\n publishDraft(pub, prom) {\n const seq = pub.seq || this._getQueuedSeqId();\n if (!pub._noForwarding) {\n // The 'seq', 'ts', and 'from' are added to mimic {data}. They are removed later\n // before the message is sent.\n pub._noForwarding = true;\n pub.seq = seq;\n pub.ts = new Date();\n pub.from = this._tinode.getCurrentUserID();\n\n // Don't need an echo message because the message is added to local cache right away.\n pub.noecho = true;\n // Add to cache.\n this._messages.put(pub);\n this._tinode._db.addMessage(pub);\n\n if (this.onData) {\n this.onData(pub);\n }\n }\n // If promise is provided, send the queued message when it's resolved.\n // If no promise is provided, create a resolved one and send immediately.\n return (prom || Promise.resolve())\n .then(_ => {\n if (pub._cancelled) {\n return {\n code: 300,\n text: \"cancelled\"\n };\n }\n return this.publishMessage(pub);\n }).catch(err => {\n this._tinode.logger(\"WARNING: Message draft rejected\", err);\n pub._sending = false;\n pub._failed = true;\n pub._fatal = err instanceof CommError ? (err.code >= 400 && err.code < 500) : false;\n if (this.onData) {\n this.onData();\n }\n // Rethrow to let caller know that the operation failed.\n throw err;\n });\n }\n\n /**\n * Leave the topic, optionally unsibscribe. Leaving the topic means the topic will stop\n * receiving updates from the server. Unsubscribing will terminate user's relationship with the topic.\n * Wrapper for {@link Tinode#leave}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean=} unsub - If true, unsubscribe, otherwise just leave.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n leave(unsub) {\n // It's possible to unsubscribe (unsub==true) from inactive topic.\n if (!this._attached && !unsub) {\n return Promise.reject(new Error(\"Cannot leave inactive topic\"));\n }\n\n // Send a 'leave' message, handle async response\n return this._tinode.leave(this.name, unsub).then(ctrl => {\n this._resetSub();\n if (unsub) {\n this._gone();\n }\n return ctrl;\n });\n }\n\n /**\n * Leave the topic, optionally unsibscribe after a delay. Leaving the topic means the topic will stop\n * receiving updates from the server. Unsubscribing will terminate user's relationship with the topic.\n * Wrapper for {@link Tinode#leave}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} unsub - If true, unsubscribe, otherwise just leave.\n * @param {number} delay - time in milliseconds to delay leave request.\n */\n leaveDelayed(unsub, delay) {\n clearTimeout(this._delayedLeaveTimer);\n this._delayedLeaveTimer = setTimeout(_ => {\n this._delayedLeaveTimer = null;\n this.leave(unsub)\n }, delay);\n }\n\n /**\n * Request topic metadata from the server.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.GetQuery} request parameters\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n getMeta(params) {\n // Send {get} message, return promise.\n return this._tinode.getMeta(this.name, params);\n }\n\n /**\n * Request more messages from the server\n * @memberof Tinode.Topic#\n *\n * @param {number} limit number of messages to get.\n * @param {boolean} forward if true, request newer messages.\n */\n getMessagesPage(limit, forward) {\n let query = forward ?\n this.startMetaQuery().withLaterData(limit) :\n this.startMetaQuery().withEarlierData(limit);\n\n // First try fetching from DB, then from the server.\n return this._loadMessages(this._tinode._db, query.extract('data'))\n .then((count) => {\n if (count == limit) {\n // Got enough messages from local cache.\n return Promise.resolve({\n topic: this.name,\n code: 200,\n params: {\n count: count\n }\n });\n }\n\n // Reduce the count of requested messages.\n limit -= count;\n // Update query with new values loaded from DB.\n query = forward ? this.startMetaQuery().withLaterData(limit) :\n this.startMetaQuery().withEarlierData(limit);\n let promise = this.getMeta(query.build());\n if (!forward) {\n promise = promise.then(ctrl => {\n if (ctrl && ctrl.params && !ctrl.params.count) {\n this._noEarlierMsgs = true;\n }\n });\n }\n return promise;\n });\n }\n /**\n * Update topic metadata.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.SetParams} params parameters to update.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n setMeta(params) {\n if (params.tags) {\n params.tags = normalizeArray(params.tags);\n }\n // Send Set message, handle async response.\n return this._tinode.setMeta(this.name, params)\n .then(ctrl => {\n if (ctrl && ctrl.code >= 300) {\n // Not modified\n return ctrl;\n }\n\n if (params.sub) {\n params.sub.topic = this.name;\n if (ctrl.params && ctrl.params.acs) {\n params.sub.acs = ctrl.params.acs;\n params.sub.updated = ctrl.ts;\n }\n if (!params.sub.user) {\n // This is a subscription update of the current user.\n // Assign user ID otherwise the update will be ignored by _processMetaSubs.\n params.sub.user = this._tinode.getCurrentUserID();\n if (!params.desc) {\n // Force update to topic's asc.\n params.desc = {};\n }\n }\n params.sub._noForwarding = true;\n this._processMetaSubs([params.sub]);\n }\n\n if (params.desc) {\n if (ctrl.params && ctrl.params.acs) {\n params.desc.acs = ctrl.params.acs;\n params.desc.updated = ctrl.ts;\n }\n this._processMetaDesc(params.desc);\n }\n\n if (params.tags) {\n this._processMetaTags(params.tags);\n }\n if (params.cred) {\n this._processMetaCreds([params.cred], true);\n }\n\n return ctrl;\n });\n }\n /**\n * Update access mode of the current user or of another topic subsriber.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - UID of the user to update or null to update current user.\n * @param {string} update - the update value, full or delta.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n updateMode(uid, update) {\n const user = uid ? this.subscriber(uid) : null;\n const am = user ?\n user.acs.updateGiven(update).getGiven() :\n this.getAccessMode().updateWant(update).getWant();\n\n return this.setMeta({\n sub: {\n user: uid,\n mode: am\n }\n });\n }\n /**\n * Create new topic subscription. Wrapper for {@link Tinode#setMeta}.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - ID of the user to invite\n * @param {string=} mode - Access mode. null
means to use default.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n invite(uid, mode) {\n return this.setMeta({\n sub: {\n user: uid,\n mode: mode\n }\n });\n }\n /**\n * Archive or un-archive the topic. Wrapper for {@link Tinode#setMeta}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} arch - true to archive the topic, false otherwise.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n archive(arch) {\n if (this.private && (!this.private.arch == !arch)) {\n return Promise.resolve(arch);\n }\n return this.setMeta({\n desc: {\n private: {\n arch: arch ? true : Const.DEL_CHAR\n }\n }\n });\n }\n /**\n * Delete messages. Hard-deleting messages requires Owner permission.\n * Wrapper for {@link Tinode#delMessages}.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.DelRange[]} ranges - Ranges of message IDs to delete.\n * @param {boolean=} hard - Hard or soft delete\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessages(ranges, hard) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete messages in inactive topic\"));\n }\n\n // Sort ranges in accending order by low, the descending by hi.\n ranges.sort((r1, r2) => {\n if (r1.low < r2.low) {\n return true;\n }\n if (r1.low == r2.low) {\n return !r2.hi || (r1.hi >= r2.hi);\n }\n return false;\n });\n\n // Remove pending messages from ranges possibly clipping some ranges.\n let tosend = ranges.reduce((out, r) => {\n if (r.low < Const.LOCAL_SEQID) {\n if (!r.hi || r.hi < Const.LOCAL_SEQID) {\n out.push(r);\n } else {\n // Clip hi to max allowed value.\n out.push({\n low: r.low,\n hi: this._maxSeq + 1\n });\n }\n }\n return out;\n }, []);\n\n // Send {del} message, return promise\n let result;\n if (tosend.length > 0) {\n result = this._tinode.delMessages(this.name, tosend, hard);\n } else {\n result = Promise.resolve({\n params: {\n del: 0\n }\n });\n }\n // Update local cache.\n return result.then(ctrl => {\n if (ctrl.params.del > this._maxDel) {\n this._maxDel = ctrl.params.del;\n }\n\n ranges.forEach((r) => {\n if (r.hi) {\n this.flushMessageRange(r.low, r.hi);\n } else {\n this.flushMessage(r.low);\n }\n });\n\n if (this.onData) {\n // Calling with no parameters to indicate the messages were deleted.\n this.onData();\n }\n return ctrl;\n });\n }\n /**\n * Delete all messages. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessagesAll(hardDel) {\n if (!this._maxSeq || this._maxSeq <= 0) {\n // There are no messages to delete.\n return Promise.resolve();\n }\n return this.delMessages([{\n low: 1,\n hi: this._maxSeq + 1,\n _all: true\n }], hardDel);\n }\n\n /**\n * Delete multiple messages defined by their IDs. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {Array.} list - list of seq IDs to delete.\n * @param {boolean=} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessagesList(list, hardDel) {\n // Sort the list in ascending order\n list.sort((a, b) => a - b);\n // Convert the array of IDs to ranges.\n let ranges = list.reduce((out, id) => {\n if (out.length == 0) {\n // First element.\n out.push({\n low: id\n });\n } else {\n let prev = out[out.length - 1];\n if ((!prev.hi && (id != prev.low + 1)) || (id > prev.hi)) {\n // New range.\n out.push({\n low: id\n });\n } else {\n // Expand existing range.\n prev.hi = prev.hi ? Math.max(prev.hi, id + 1) : id + 1;\n }\n }\n return out;\n }, []);\n // Send {del} message, return promise\n return this.delMessages(ranges, hardDel);\n }\n\n /**\n * Delete original message and edited variants. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - original seq ID of the message to delete.\n * @param {boolean=} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n delMessagesEdits(seq, hardDel) {\n const list = [seq];\n this.messageVersions(seq, msg => list.push(msg.seq));\n // Send {del} message, return promise\n return this.delMessagesList(list, hardDel);\n }\n\n /**\n * Delete topic. Requires Owner permission. Wrapper for {@link Tinode#delTopic}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} hard - had-delete topic.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n delTopic(hard) {\n if (this._deleted) {\n // The topic is already deleted at the server, just remove from DB.\n this._gone();\n return Promise.resolve(null);\n }\n\n return this._tinode.delTopic(this.name, hard).then(ctrl => {\n this._deleted = true;\n this._resetSub();\n this._gone();\n return ctrl;\n });\n }\n /**\n * Delete subscription. Requires Share permission. Wrapper for {@link Tinode#delSubscription}.\n * @memberof Tinode.Topic#\n *\n * @param {string} user - ID of the user to remove subscription for.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delSubscription(user) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete subscription in inactive topic\"));\n }\n // Send {del} message, return promise\n return this._tinode.delSubscription(this.name, user).then(ctrl => {\n // Remove the object from the subscription cache;\n delete this._users[user];\n // Notify listeners\n if (this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._users));\n }\n return ctrl;\n });\n }\n /**\n * Send a read/recv notification.\n * @memberof Tinode.Topic#\n *\n * @param {string} what - what notification to send: recv
, read
.\n * @param {number} seq - ID or the message read or received.\n */\n note(what, seq) {\n if (!this._attached) {\n // Cannot sending {note} on an inactive topic\".\n return;\n }\n\n // Update local cache with the new count.\n const user = this._users[this._tinode.getCurrentUserID()];\n let update = false;\n if (user) {\n // Self-subscription is found.\n if (!user[what] || user[what] < seq) {\n user[what] = seq;\n update = true;\n }\n } else {\n // Self-subscription is not found.\n update = (this[what] | 0) < seq;\n }\n\n if (update) {\n // Send notification to the server.\n this._tinode.note(this.name, what, seq);\n // Update locally cached contact with the new count.\n this._updateMyReadRecv(what, seq);\n\n if (this.acs != null && !this.acs.isMuted()) {\n const me = this._tinode.getMeTopic();\n // Sent a notification to 'me' listeners.\n me._refreshContact(what, this);\n }\n }\n }\n\n /**\n * Send a 'recv' receipt. Wrapper for {@link Tinode#noteRecv}.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - ID of the message to aknowledge.\n */\n noteRecv(seq) {\n this.note('recv', seq);\n }\n /**\n * Send a 'read' receipt. Wrapper for {@link Tinode#noteRead}.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - ID of the message to aknowledge or 0/undefined to acknowledge the latest messages.\n */\n noteRead(seq) {\n seq = seq || this._maxSeq;\n if (seq > 0) {\n this.note('read', seq);\n }\n }\n /**\n * Send a key-press notification. Wrapper for {@link Tinode#noteKeyPress}.\n * @memberof Tinode.Topic#\n */\n noteKeyPress() {\n if (this._attached) {\n this._tinode.noteKeyPress(this.name);\n } else {\n this._tinode.logger(\"INFO: Cannot send notification in inactive topic\");\n }\n }\n /**\n * Send a notification than a video or audio message is . Wrapper for {@link Tinode#noteKeyPress}.\n * @memberof Tinode.Topic#\n * @param audioOnly - true if the recording is audio-only, false if it's a video recording.\n */\n noteRecording(audioOnly) {\n if (this._attached) {\n this._tinode.noteKeyPress(this.name, audioOnly ? 'kpa' : 'kpv');\n } else {\n this._tinode.logger(\"INFO: Cannot send notification in inactive topic\");\n }\n }\n\n /**\n * Send a {note what='call'}. Wrapper for {@link Tinode#videoCall}.\n * @memberof Tinode#\n *\n * @param {string} evt - Call event.\n * @param {int} seq - ID of the call message the event pertains to.\n * @param {string} payload - Payload associated with this event (e.g. SDP string).\n *\n * @returns {Promise} Promise (for some call events) which will\n * be resolved/rejected on receiving server reply\n */\n videoCall(evt, seq, payload) {\n if (!this._attached && !['ringing', 'hang-up'].includes(evt)) {\n // Cannot {call} on an inactive topic\".\n return;\n }\n return this._tinode.videoCall(this.name, seq, evt, payload);\n }\n\n // Update cached read/recv/unread counts for the current user.\n _updateMyReadRecv(what, seq, ts) {\n let oldVal, doUpdate = false;\n\n seq = seq | 0;\n this.seq = this.seq | 0;\n this.read = this.read | 0;\n this.recv = this.recv | 0;\n switch (what) {\n case 'recv':\n oldVal = this.recv;\n this.recv = Math.max(this.recv, seq);\n doUpdate = (oldVal != this.recv);\n break;\n case 'read':\n oldVal = this.read;\n this.read = Math.max(this.read, seq);\n doUpdate = (oldVal != this.read);\n break;\n case 'msg':\n oldVal = this.seq;\n this.seq = Math.max(this.seq, seq);\n if (!this.touched || this.touched < ts) {\n this.touched = ts;\n }\n doUpdate = (oldVal != this.seq);\n break;\n }\n\n // Sanity checks.\n if (this.recv < this.read) {\n this.recv = this.read;\n doUpdate = true;\n }\n if (this.seq < this.recv) {\n this.seq = this.recv;\n if (!this.touched || this.touched < ts) {\n this.touched = ts;\n }\n doUpdate = true;\n }\n this.unread = this.seq - this.read;\n return doUpdate;\n }\n /**\n * Get user description from global cache. The user does not need to be a\n * subscriber of this topic.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - ID of the user to fetch.\n * @return {Object} user description or undefined.\n */\n userDesc(uid) {\n // TODO: handle asynchronous requests\n const user = this._cacheGetUser(uid);\n if (user) {\n return user; // Promise.resolve(user)\n }\n }\n /**\n * Get description of the p2p peer from subscription cache.\n * @memberof Tinode.Topic#\n *\n * @return {Object} peer's description or undefined.\n */\n p2pPeerDesc() {\n if (!this.isP2PType()) {\n return undefined;\n }\n return this._users[this.name];\n }\n /**\n * Iterate over cached subscribers. If callback is undefined, use this.onMetaSub.\n * @memberof Tinode.Topic#\n *\n * @param {function} callback - Callback which will receive subscribers one by one.\n * @param {Object=} context - Value of `this` inside the `callback`.\n */\n subscribers(callback, context) {\n const cb = (callback || this.onMetaSub);\n if (cb) {\n for (let idx in this._users) {\n cb.call(context, this._users[idx], idx, this._users);\n }\n }\n }\n /**\n * Get a copy of cached tags.\n * @memberof Tinode.Topic#\n *\n * @return {Array.} a copy of tags\n */\n tags() {\n // Return a copy.\n return this._tags.slice(0);\n }\n /**\n * Get cached subscription for the given user ID.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - id of the user to query for\n * @return user description or undefined.\n */\n subscriber(uid) {\n return this._users[uid];\n }\n /**\n * Iterate over versions of a message: call callback
for each version (excluding original).\n * If callback
is undefined, does nothing.\n * @memberof Tinode.Topic#\n *\n * @param {number} origSeq - seq ID of the original message.\n * @param {Tinode.ForEachCallbackType} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {Object} context - Value of `this` inside the `callback`.\n */\n messageVersions(origSeq, callback, context) {\n if (!callback) {\n // No callback? We are done then.\n return;\n }\n const versions = this._messageVersions[origSeq];\n if (!versions) {\n return;\n }\n versions.forEach(callback, undefined, undefined, context);\n }\n /**\n * Iterate over cached messages: call callback
for each message in the range [sinceIdx, beforeIdx).\n * If callback
is undefined, use this.onData
.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.ForEachCallbackType} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {number} sinceId - Optional seqId to start iterating from (inclusive).\n * @param {number} beforeId - Optional seqId to stop iterating before it is reached (exclusive).\n * @param {Object} context - Value of `this` inside the `callback`.\n */\n messages(callback, sinceId, beforeId, context) {\n const cb = (callback || this.onData);\n if (cb) {\n const startIdx = typeof sinceId == 'number' ? this._messages.find({\n seq: sinceId\n }, true) : undefined;\n const beforeIdx = typeof beforeId == 'number' ? this._messages.find({\n seq: beforeId\n }, true) : undefined;\n if (startIdx != -1 && beforeIdx != -1) {\n // Step 1. Filter out all replacement messages and\n // save displayable messages in a temporary buffer.\n let msgs = [];\n this._messages.forEach((msg, unused1, unused2, i) => {\n if (this._isReplacementMsg(msg)) {\n // Skip replacements.\n return;\n }\n // In case the massage was edited, replace timestamp of the version with the original's timestamp.\n const latest = this.latestMsgVersion(msg.seq) || msg;\n if (!latest._origTs) {\n latest._origTs = latest.ts;\n latest._origSeq = latest.seq;\n latest.ts = msg.ts;\n latest.seq = msg.seq;\n }\n msgs.push({\n data: latest,\n idx: i\n });\n }, startIdx, beforeIdx, {});\n // Step 2. Loop over displayble messages invoking cb on each of them.\n msgs.forEach((val, i) => {\n cb.call(context, val.data,\n (i > 0 ? msgs[i - 1].data : undefined),\n (i < msgs.length - 1 ? msgs[i + 1].data : undefined), val.idx);\n });\n }\n }\n }\n /**\n * Get the message from cache by seq
.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - message seqId to search for.\n * @returns {Object} the message with the given seq
or undefined
, if no such message is found.\n */\n findMessage(seq) {\n const idx = this._messages.find({\n seq: seq\n });\n if (idx >= 0) {\n return this._messages.getAt(idx);\n }\n return undefined;\n }\n /**\n * Get the most recent message from cache. This method counts all messages, including deleted ranges.\n * @memberof Tinode.Topic#\n *\n * @returns {Object} the most recent cached message or undefined
, if no messages are cached.\n */\n latestMessage() {\n return this._messages.getLast();\n }\n /**\n * Get the latest version for message.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - original seq ID of the message.\n * @returns {Object} the latest version of the message or null if message not found.\n */\n latestMsgVersion(seq) {\n const versions = this._messageVersions[seq];\n return versions ? versions.getLast() : null;\n }\n /**\n * Get the maximum cached seq ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the greatest seq ID in cache.\n */\n maxMsgSeq() {\n return this._maxSeq;\n }\n /**\n * Get the minimum cached seq ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the smallest seq ID in cache or 0.\n */\n minMsgSeq() {\n return this._minSeq;\n }\n /**\n * Get the maximum deletion ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the greatest deletion ID.\n */\n maxClearId() {\n return this._maxDel;\n }\n /**\n * Get the number of messages in the cache.\n * @memberof Tinode.Topic#\n *\n * @returns {number} count of cached messages.\n */\n messageCount() {\n return this._messages.length();\n }\n /**\n * Iterate over cached unsent messages. Wraps {@link Tinode.Topic#messages}.\n * @memberof Tinode.Topic#\n *\n * @param {function} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {Object} context - Value of this
inside the callback
.\n */\n queuedMessages(callback, context) {\n if (!callback) {\n throw new Error(\"Callback must be provided\");\n }\n this.messages(callback, Const.LOCAL_SEQID, undefined, context);\n }\n /**\n * Get the number of topic subscribers who marked this message as either recv or read\n * Current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {string} what - what action to consider: received \"recv\"
or read \"read\"
.\n * @param {number} seq - ID or the message read or received.\n *\n * @returns {number} the number of subscribers who marked the message with the given ID as read or received.\n */\n msgReceiptCount(what, seq) {\n let count = 0;\n if (seq > 0) {\n const me = this._tinode.getCurrentUserID();\n for (let idx in this._users) {\n const user = this._users[idx];\n if (user.user !== me && user[what] >= seq) {\n count++;\n }\n }\n }\n return count;\n }\n /**\n * Get the number of topic subscribers who marked this message (and all older messages) as read.\n * The current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - message id to check.\n * @returns {number} number of subscribers who claim to have received the message.\n */\n msgReadCount(seq) {\n return this.msgReceiptCount('read', seq);\n }\n /**\n * Get the number of topic subscribers who marked this message (and all older messages) as received.\n * The current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - Message id to check.\n * @returns {number} Number of subscribers who claim to have received the message.\n */\n msgRecvCount(seq) {\n return this.msgReceiptCount('recv', seq);\n }\n /**\n * Check if cached message IDs indicate that the server may have more messages.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} newer - if true
, check for newer messages only.\n */\n msgHasMoreMessages(newer) {\n return newer ? this.seq > this._maxSeq :\n // _minSeq could be more than 1, but earlier messages could have been deleted.\n (this._minSeq > 1 && !this._noEarlierMsgs);\n }\n /**\n * Check if the given seq Id is id of the most recent message.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to check\n */\n isNewMessage(seqId) {\n return this._maxSeq <= seqId;\n }\n /**\n * Remove one message from local cache.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to remove from cache.\n * @returns {Message} removed message or undefined if such message was not found.\n */\n flushMessage(seqId) {\n const idx = this._messages.find({\n seq: seqId\n });\n delete this._messageVersions[seqId];\n if (idx >= 0) {\n this._tinode._db.remMessages(this.name, seqId);\n return this._messages.delAt(idx);\n }\n return undefined;\n }\n /**\n * Remove a range of messages from the local cache.\n * @memberof Tinode.Topic#\n *\n * @param {number} fromId seq ID of the first message to remove (inclusive).\n * @param {number} untilId seqID of the last message to remove (exclusive).\n *\n * @returns {Message[]} array of removed messages (could be empty).\n */\n flushMessageRange(fromId, untilId) {\n // Remove range from persistent cache.\n this._tinode._db.remMessages(this.name, fromId, untilId);\n\n // Remove all versions keyed by IDs in the range.\n for (let i = fromId; i < untilId; i++) {\n delete this._messageVersions[i];\n }\n\n // start, end: find insertion points (nearest == true).\n const since = this._messages.find({\n seq: fromId\n }, true);\n return since >= 0 ? this._messages.delRange(since, this._messages.find({\n seq: untilId\n }, true)) : [];\n }\n /**\n * Update message's seqId.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub message object.\n * @param {number} newSeqId new seq id for pub.\n */\n swapMessageId(pub, newSeqId) {\n const idx = this._messages.find(pub);\n const numMessages = this._messages.length();\n if (0 <= idx && idx < numMessages) {\n // Remove message with the old seq ID.\n this._messages.delAt(idx);\n this._tinode._db.remMessages(this.name, pub.seq);\n // Add message with the new seq ID.\n pub.seq = newSeqId;\n this._messages.put(pub);\n this._tinode._db.addMessage(pub);\n }\n }\n /**\n * Attempt to stop message from being sent.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to stop sending and remove from cache.\n *\n * @returns {boolean} true
if message was cancelled, false
otherwise.\n */\n cancelSend(seqId) {\n const idx = this._messages.find({\n seq: seqId\n });\n if (idx >= 0) {\n const msg = this._messages.getAt(idx);\n const status = this.msgStatus(msg);\n if (status == Const.MESSAGE_STATUS_QUEUED ||\n status == Const.MESSAGE_STATUS_FAILED ||\n status == Const.MESSAGE_STATUS_FATAL) {\n this._tinode._db.remMessages(this.name, seqId);\n msg._cancelled = true;\n this._messages.delAt(idx);\n if (this.onData) {\n // Calling with no parameters to indicate the message was deleted.\n this.onData();\n }\n return true;\n }\n }\n return false;\n }\n /**\n * Get type of the topic: me, p2p, grp, fnd...\n * @memberof Tinode.Topic#\n *\n * @returns {string} One of 'me', 'p2p', 'grp', 'fnd', 'sys' or undefined
.\n */\n getType() {\n return Topic.topicType(this.name);\n }\n /**\n * Get current user's access mode of the topic.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.AccessMode} - user's access mode\n */\n getAccessMode() {\n return this.acs;\n }\n /**\n * Set current user's access mode of the topic.\n * @memberof Tinode.Topic#\n *\n * @param {AccessMode | Object} acs - access mode to set.\n */\n setAccessMode(acs) {\n return this.acs = new AccessMode(acs);\n }\n /**\n * Get topic's default access mode.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.DefAcs} - access mode, such as {auth: `RWP`, anon: `N`}.\n */\n getDefaultAccess() {\n return this.defacs;\n }\n /**\n * Initialize new meta {@link Tinode.GetQuery} builder. The query is attched to the current topic.\n * It will not work correctly if used with a different topic.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.MetaGetBuilder} query attached to the current topic.\n */\n startMetaQuery() {\n return new MetaGetBuilder(this);\n }\n /**\n * Check if topic is archived, i.e. private.arch == true.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is archived, false
otherwise.\n */\n isArchived() {\n return this.private && !!this.private.arch;\n }\n /**\n * Check if topic is a 'me' topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a 'me' topic, false
otherwise.\n */\n isMeType() {\n return Topic.isMeTopicName(this.name);\n }\n /**\n * Check if topic is a channel.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a channel, false
otherwise.\n */\n isChannelType() {\n return Topic.isChannelTopicName(this.name);\n }\n /**\n * Check if topic is a group topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a group, false
otherwise.\n */\n isGroupType() {\n return Topic.isGroupTopicName(this.name);\n }\n /**\n * Check if topic is a p2p topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a p2p topic, false
otherwise.\n */\n isP2PType() {\n return Topic.isP2PTopicName(this.name);\n }\n /**\n * Check if topic is a communication topic, i.e. a group or p2p topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a p2p or group topic, false
otherwise.\n */\n isCommType() {\n return Topic.isCommTopicName(this.name);\n }\n /**\n * Get status (queued, sent, received etc) of a given message in the context\n * of this topic.\n * @memberof Tinode.Topic#\n *\n * @param {Message} msg - message to check for status.\n * @param {boolean} upd - update chached message status.\n *\n * @returns message status constant.\n */\n msgStatus(msg, upd) {\n let status = Const.MESSAGE_STATUS_NONE;\n if (this._tinode.isMe(msg.from)) {\n if (msg._sending) {\n status = Const.MESSAGE_STATUS_SENDING;\n } else if (msg._fatal || msg._cancelled) {\n status = Const.MESSAGE_STATUS_FATAL;\n } else if (msg._failed) {\n status = Const.MESSAGE_STATUS_FAILED;\n } else if (msg.seq >= Const.LOCAL_SEQID) {\n status = Const.MESSAGE_STATUS_QUEUED;\n } else if (this.msgReadCount(msg.seq) > 0) {\n status = Const.MESSAGE_STATUS_READ;\n } else if (this.msgRecvCount(msg.seq) > 0) {\n status = Const.MESSAGE_STATUS_RECEIVED;\n } else if (msg.seq > 0) {\n status = Const.MESSAGE_STATUS_SENT;\n }\n } else {\n status = Const.MESSAGE_STATUS_TO_ME;\n }\n\n if (upd && msg._status != status) {\n msg._status = status;\n this._tinode._db.updMessageStatus(this.name, msg.seq, status);\n }\n\n return status;\n }\n\n // Returns true if pub is meant to replace another message (e.g. original message was edited).\n _isReplacementMsg(pub) {\n return pub.head && pub.head.replace;\n }\n\n // If msg is a replacement for another message, save the msg in the message versions cache\n // as a newer version for the message it's supposed to replace.\n _maybeUpdateMessageVersionsCache(msg) {\n if (!this._isReplacementMsg(msg)) {\n // Check if this message is the original in the chain of edits and if so\n // ensure all version have the same sender.\n if (this._messageVersions[msg.seq]) {\n // Remove versions with different 'from'.\n this._messageVersions[msg.seq].filter(version => version.from == msg.from);\n if (this._messageVersions[msg.seq].isEmpty()) {\n delete this._messageVersions[msg.seq];\n }\n }\n return;\n }\n\n const targetSeq = parseInt(msg.head.replace.split(':')[1]);\n if (targetSeq > msg.seq) {\n // Substitutes are supposed to have higher seq ids.\n return;\n }\n const targetMsg = this.findMessage(targetSeq);\n if (targetMsg && targetMsg.from != msg.from) {\n // Substitute cannot change the sender.\n return;\n }\n const versions = this._messageVersions[targetSeq] || new CBuffer((a, b) => {\n return a.seq - b.seq;\n }, true);\n versions.put(msg);\n this._messageVersions[targetSeq] = versions;\n }\n\n // Process data message\n _routeData(data) {\n if (data.content) {\n if (!this.touched || this.touched < data.ts) {\n this.touched = data.ts;\n this._tinode._db.updTopic(this);\n }\n }\n\n if (data.seq > this._maxSeq) {\n this._maxSeq = data.seq;\n this.msgStatus(data, true);\n // Ackn receiving the message.\n clearTimeout(this._recvNotificationTimer);\n this._recvNotificationTimer = setTimeout(_ => {\n this._recvNotificationTimer = null;\n this.noteRecv(this._maxSeq);\n }, Const.RECV_TIMEOUT);\n }\n\n if (data.seq < this._minSeq || this._minSeq == 0) {\n this._minSeq = data.seq;\n }\n\n const outgoing = ((!this.isChannelType() && !data.from) || this._tinode.isMe(data.from));\n\n if (data.head && data.head.webrtc && data.head.mime == Drafty.getContentType() && data.content) {\n // Rewrite VC body with info from the headers.\n data.content = Drafty.updateVideoCall(data.content, {\n state: data.head.webrtc,\n duration: data.head['webrtc-duration'],\n incoming: !outgoing,\n });\n }\n\n if (!data._noForwarding) {\n this._messages.put(data);\n this._tinode._db.addMessage(data);\n this._maybeUpdateMessageVersionsCache(data);\n }\n\n if (this.onData) {\n this.onData(data);\n }\n\n // Update locally cached contact with the new message count.\n const what = outgoing ? 'read' : 'msg';\n this._updateMyReadRecv(what, data.seq, data.ts);\n\n if (!outgoing && data.from) {\n // Mark messages as read by the sender.\n this._routeInfo({\n what: 'read',\n from: data.from,\n seq: data.seq,\n _noForwarding: true\n });\n }\n\n // Notify 'me' listeners of the change.\n this._tinode.getMeTopic()._refreshContact(what, this);\n }\n\n // Process metadata message\n _routeMeta(meta) {\n if (meta.desc) {\n this._processMetaDesc(meta.desc);\n }\n if (meta.sub && meta.sub.length > 0) {\n this._processMetaSubs(meta.sub);\n }\n if (meta.del) {\n this._processDelMessages(meta.del.clear, meta.del.delseq);\n }\n if (meta.tags) {\n this._processMetaTags(meta.tags);\n }\n if (meta.cred) {\n this._processMetaCreds(meta.cred);\n }\n if (this.onMeta) {\n this.onMeta(meta);\n }\n }\n // Process presence change message\n _routePres(pres) {\n let user, uid;\n switch (pres.what) {\n case 'del':\n // Delete cached messages.\n this._processDelMessages(pres.clear, pres.delseq);\n break;\n case 'on':\n case 'off':\n // Update online status of a subscription.\n user = this._users[pres.src];\n if (user) {\n user.online = pres.what == 'on';\n } else {\n this._tinode.logger(\"WARNING: Presence update for an unknown user\", this.name, pres.src);\n }\n break;\n case 'term':\n // Attachment to topic is terminated probably due to cluster rehashing.\n this._resetSub();\n break;\n case 'upd':\n // A topic subscriber has updated his description.\n // Issue {get sub} only if the current user has no p2p topics with the updated user (p2p name is not in cache).\n // Otherwise 'me' will issue a {get desc} request.\n if (pres.src && !this._tinode.isTopicCached(pres.src)) {\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n }\n break;\n case 'acs':\n uid = pres.src || this._tinode.getCurrentUserID();\n user = this._users[uid];\n if (!user) {\n // Update for an unknown user: notification of a new subscription.\n const acs = new AccessMode().updateAll(pres.dacs);\n if (acs && acs.mode != AccessMode._NONE) {\n user = this._cacheGetUser(uid);\n if (!user) {\n user = {\n user: uid,\n acs: acs\n };\n this.getMeta(this.startMetaQuery().withOneSub(undefined, uid).build());\n } else {\n user.acs = acs;\n }\n user.updated = new Date();\n this._processMetaSubs([user]);\n }\n } else {\n // Known user\n user.acs.updateAll(pres.dacs);\n // Update user's access mode.\n this._processMetaSubs([{\n user: uid,\n updated: new Date(),\n acs: user.acs\n }]);\n }\n break;\n default:\n this._tinode.logger(\"INFO: Ignored presence update\", pres.what);\n }\n\n if (this.onPres) {\n this.onPres(pres);\n }\n }\n // Process {info} message\n _routeInfo(info) {\n switch (info.what) {\n case 'recv':\n case 'read':\n const user = this._users[info.from];\n if (user) {\n user[info.what] = info.seq;\n if (user.recv < user.read) {\n user.recv = user.read;\n }\n }\n const msg = this.latestMessage();\n if (msg) {\n this.msgStatus(msg, true);\n }\n\n // If this is an update from the current user, update the cache with the new count.\n if (this._tinode.isMe(info.from) && !info._noForwarding) {\n this._updateMyReadRecv(info.what, info.seq);\n }\n\n // Notify 'me' listener of the status change.\n this._tinode.getMeTopic()._refreshContact(info.what, this);\n break;\n case 'kp':\n case 'kpa':\n case 'kpv':\n // Typing or audio/video recording notification. Do nothing.\n break;\n case 'call':\n // Do nothing here.\n break;\n default:\n this._tinode.logger(\"INFO: Ignored info update\", info.what);\n }\n\n if (this.onInfo) {\n this.onInfo(info);\n }\n }\n // Called by Tinode when meta.desc packet is received.\n // Called by 'me' topic on contact update (desc._noForwarding is true).\n _processMetaDesc(desc) {\n if (this.isP2PType()) {\n // Synthetic desc may include defacs for p2p topics which is useless.\n // Remove it.\n delete desc.defacs;\n\n // Update to p2p desc is the same as user update. Update cached user.\n this._tinode._db.updUser(this.name, desc.public);\n }\n\n // Copy parameters from desc object to this topic.\n mergeObj(this, desc);\n // Update persistent cache.\n this._tinode._db.updTopic(this);\n\n // Notify 'me' listener, if available:\n if (this.name !== Const.TOPIC_ME && !desc._noForwarding) {\n const me = this._tinode.getMeTopic();\n if (me.onMetaSub) {\n me.onMetaSub(this);\n }\n if (me.onSubsUpdated) {\n me.onSubsUpdated([this.name], 1);\n }\n }\n\n if (this.onMetaDesc) {\n this.onMetaDesc(this);\n }\n }\n // Called by Tinode when meta.sub is recived or in response to received\n // {ctrl} after setMeta-sub.\n _processMetaSubs(subs) {\n for (let idx in subs) {\n const sub = subs[idx];\n\n // Fill defaults.\n sub.online = !!sub.online;\n // Update timestamp of the most recent subscription update.\n this._lastSubsUpdate = new Date(Math.max(this._lastSubsUpdate, sub.updated));\n\n let user = null;\n if (!sub.deleted) {\n // If this is a change to user's own permissions, update them in topic too.\n // Desc will update 'me' topic.\n if (this._tinode.isMe(sub.user) && sub.acs) {\n this._processMetaDesc({\n updated: sub.updated,\n touched: sub.touched,\n acs: sub.acs\n });\n }\n user = this._updateCachedUser(sub.user, sub);\n } else {\n // Subscription is deleted, remove it from topic (but leave in Users cache)\n delete this._users[sub.user];\n user = sub;\n }\n\n if (this.onMetaSub) {\n this.onMetaSub(user);\n }\n }\n\n if (this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._users));\n }\n }\n // Called by Tinode when meta.tags is recived.\n _processMetaTags(tags) {\n if (tags.length == 1 && tags[0] == Const.DEL_CHAR) {\n tags = [];\n }\n this._tags = tags;\n if (this.onTagsUpdated) {\n this.onTagsUpdated(tags);\n }\n }\n // Do nothing for topics other than 'me'\n _processMetaCreds(creds) {}\n // Delete cached messages and update cached transaction IDs\n _processDelMessages(clear, delseq) {\n this._maxDel = Math.max(clear, this._maxDel);\n this.clear = Math.max(clear, this.clear);\n const topic = this;\n let count = 0;\n if (Array.isArray(delseq)) {\n delseq.forEach(function(range) {\n if (!range.hi) {\n count++;\n topic.flushMessage(range.low);\n } else {\n for (let i = range.low; i < range.hi; i++) {\n count++;\n topic.flushMessage(i);\n }\n }\n });\n }\n\n if (count > 0) {\n // this._updateDeletedRanges();\n\n if (this.onData) {\n this.onData();\n }\n }\n }\n // Topic is informed that the entire response to {get what=data} has been received.\n _allMessagesReceived(count) {\n\n if (this.onAllMessagesReceived) {\n this.onAllMessagesReceived(count);\n }\n }\n // Reset subscribed state\n _resetSub() {\n this._attached = false;\n }\n // This topic is either deleted or unsubscribed from.\n _gone() {\n this._messages.reset();\n this._tinode._db.remMessages(this.name);\n this._users = {};\n this.acs = new AccessMode(null);\n this.private = null;\n this.public = null;\n this.trusted = null;\n this._maxSeq = 0;\n this._minSeq = 0;\n this._attached = false;\n\n const me = this._tinode.getMeTopic();\n if (me) {\n me._routePres({\n _noForwarding: true,\n what: 'gone',\n topic: Const.TOPIC_ME,\n src: this.name\n });\n }\n if (this.onDeleteTopic) {\n this.onDeleteTopic();\n }\n }\n // Update global user cache and local subscribers cache.\n // Don't call this method for non-subscribers.\n _updateCachedUser(uid, obj) {\n // Fetch user object from the global cache.\n // This is a clone of the stored object\n let cached = this._cacheGetUser(uid);\n cached = mergeObj(cached || {}, obj);\n // Save to global cache\n this._cachePutUser(uid, cached);\n // Save to the list of topic subsribers.\n return mergeToCache(this._users, uid, cached);\n }\n // Get local seqId for a queued message.\n _getQueuedSeqId() {\n return this._queuedSeqId++;\n }\n\n // Load most recent messages from persistent cache.\n _loadMessages(db, params) {\n const {\n since,\n before,\n limit\n } = params || {};\n return db.readMessages(this.name, {\n since: since,\n before: before,\n limit: limit || Const.DEFAULT_MESSAGES_PAGE\n })\n .then(msgs => {\n msgs.forEach((data) => {\n if (data.seq > this._maxSeq) {\n this._maxSeq = data.seq;\n }\n if (data.seq < this._minSeq || this._minSeq == 0) {\n this._minSeq = data.seq;\n }\n this._messages.put(data);\n this._maybeUpdateMessageVersionsCache(data);\n });\n return msgs.length;\n });\n }\n // Push or {pres}: message received.\n _updateReceived(seq, act) {\n this.touched = new Date();\n this.seq = seq | 0;\n // Check if message is sent by the current user. If so it's been read already.\n if (!act || this._tinode.isMe(act)) {\n this.read = this.read ? Math.max(this.read, this.seq) : this.seq;\n this.recv = this.recv ? Math.max(this.read, this.recv) : this.read;\n }\n this.unread = this.seq - (this.read | 0);\n this._tinode._db.updTopic(this);\n }\n}\n\n/**\n * @class TopicMe - special case of {@link Tinode.Topic} for\n * managing data of the current user, including contact list.\n * @extends Tinode.Topic\n * @memberof Tinode\n *\n * @param {TopicMe.Callbacks} callbacks - Callbacks to receive various events.\n */\nexport class TopicMe extends Topic {\n onContactUpdate;\n\n constructor(callbacks) {\n super(Const.TOPIC_ME, callbacks);\n\n // me-specific callbacks\n if (callbacks) {\n this.onContactUpdate = callbacks.onContactUpdate;\n }\n }\n\n // Override the original Topic._processMetaDesc.\n _processMetaDesc(desc) {\n // Check if online contacts need to be turned off because P permission was removed.\n const turnOff = (desc.acs && !desc.acs.isPresencer()) && (this.acs && this.acs.isPresencer());\n\n // Copy parameters from desc object to this topic.\n mergeObj(this, desc);\n this._tinode._db.updTopic(this);\n // Update current user's record in the global cache.\n this._updateCachedUser(this._tinode._myUID, desc);\n\n // 'P' permission was removed. All topics are offline now.\n if (turnOff) {\n this._tinode.mapTopics((cont) => {\n if (cont.online) {\n cont.online = false;\n cont.seen = Object.assign(cont.seen || {}, {\n when: new Date()\n });\n this._refreshContact('off', cont);\n }\n });\n }\n\n if (this.onMetaDesc) {\n this.onMetaDesc(this);\n }\n }\n\n // Override the original Topic._processMetaSubs\n _processMetaSubs(subs) {\n let updateCount = 0;\n subs.forEach((sub) => {\n const topicName = sub.topic;\n // Don't show 'me' and 'fnd' topics in the list of contacts.\n if (topicName == Const.TOPIC_FND || topicName == Const.TOPIC_ME) {\n return;\n }\n sub.online = !!sub.online;\n\n let cont = null;\n if (sub.deleted) {\n cont = sub;\n this._tinode.cacheRemTopic(topicName);\n this._tinode._db.remTopic(topicName);\n } else {\n // Ensure the values are defined and are integers.\n if (typeof sub.seq != 'undefined') {\n sub.seq = sub.seq | 0;\n sub.recv = sub.recv | 0;\n sub.read = sub.read | 0;\n sub.unread = sub.seq - sub.read;\n }\n\n const topic = this._tinode.getTopic(topicName);\n if (topic._new) {\n delete topic._new;\n }\n\n cont = mergeObj(topic, sub);\n this._tinode._db.updTopic(cont);\n\n if (Topic.isP2PTopicName(topicName)) {\n this._cachePutUser(topicName, cont);\n this._tinode._db.updUser(topicName, cont.public);\n }\n // Notify topic of the update if it's an external update.\n if (!sub._noForwarding && topic) {\n sub._noForwarding = true;\n topic._processMetaDesc(sub);\n }\n }\n\n updateCount++;\n\n if (this.onMetaSub) {\n this.onMetaSub(cont);\n }\n });\n\n if (this.onSubsUpdated && updateCount > 0) {\n const keys = [];\n subs.forEach((s) => {\n keys.push(s.topic);\n });\n this.onSubsUpdated(keys, updateCount);\n }\n }\n\n // Called by Tinode when meta.sub is recived.\n _processMetaCreds(creds, upd) {\n if (creds.length == 1 && creds[0] == Const.DEL_CHAR) {\n creds = [];\n }\n if (upd) {\n creds.forEach((cr) => {\n if (cr.val) {\n // Adding a credential.\n let idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && el.val == cr.val;\n });\n if (idx < 0) {\n // Not found.\n if (!cr.done) {\n // Unconfirmed credential replaces previous unconfirmed credential of the same method.\n idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && !el.done;\n });\n if (idx >= 0) {\n // Remove previous unconfirmed credential.\n this._credentials.splice(idx, 1);\n }\n }\n this._credentials.push(cr);\n } else {\n // Found. Maybe change 'done' status.\n this._credentials[idx].done = cr.done;\n }\n } else if (cr.resp) {\n // Handle credential confirmation.\n const idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && !el.done;\n });\n if (idx >= 0) {\n this._credentials[idx].done = true;\n }\n }\n });\n } else {\n this._credentials = creds;\n }\n if (this.onCredsUpdated) {\n this.onCredsUpdated(this._credentials);\n }\n }\n\n // Process presence change message\n _routePres(pres) {\n if (pres.what == 'term') {\n // The 'me' topic itself is detached. Mark as unsubscribed.\n this._resetSub();\n return;\n }\n\n if (pres.what == 'upd' && pres.src == Const.TOPIC_ME) {\n // Update to me's description. Request updated value.\n this.getMeta(this.startMetaQuery().withDesc().build());\n return;\n }\n\n const cont = this._tinode.cacheGetTopic(pres.src);\n if (cont) {\n switch (pres.what) {\n case 'on': // topic came online\n cont.online = true;\n break;\n case 'off': // topic went offline\n if (cont.online) {\n cont.online = false;\n cont.seen = Object.assign(cont.seen || {}, {\n when: new Date()\n });\n }\n break;\n case 'msg': // new message received\n cont._updateReceived(pres.seq, pres.act);\n break;\n case 'upd': // desc updated\n // Request updated subscription.\n this.getMeta(this.startMetaQuery().withLaterOneSub(pres.src).build());\n break;\n case 'acs': // access mode changed\n // If 'tgt' is not set then this is an update to the permissions of the current user.\n // Otherwise it's an update to group topic subscriber permissions while the topic is offline.\n // Just gnore it then.\n if (!pres.tgt) {\n if (cont.acs) {\n cont.acs.updateAll(pres.dacs);\n } else {\n cont.acs = new AccessMode().updateAll(pres.dacs);\n }\n cont.touched = new Date();\n }\n break;\n case 'ua':\n // user agent changed.\n cont.seen = {\n when: new Date(),\n ua: pres.ua\n };\n break;\n case 'recv':\n // user's other session marked some messges as received.\n pres.seq = pres.seq | 0;\n cont.recv = cont.recv ? Math.max(cont.recv, pres.seq) : pres.seq;\n break;\n case 'read':\n // user's other session marked some messages as read.\n pres.seq = pres.seq | 0;\n cont.read = cont.read ? Math.max(cont.read, pres.seq) : pres.seq;\n cont.recv = cont.recv ? Math.max(cont.read, cont.recv) : cont.recv;\n cont.unread = cont.seq - cont.read;\n break;\n case 'gone':\n // topic deleted or unsubscribed from.\n this._tinode.cacheRemTopic(pres.src);\n if (!cont._deleted) {\n cont._deleted = true;\n cont._attached = false;\n this._tinode._db.markTopicAsDeleted(pres.src, true);\n } else {\n this._tinode._db.remTopic(pres.src);\n }\n break;\n case 'del':\n // Update topic.del value.\n break;\n default:\n this._tinode.logger(\"INFO: Unsupported presence update in 'me'\", pres.what);\n }\n\n this._refreshContact(pres.what, cont);\n } else {\n if (pres.what == 'acs') {\n // New subscriptions and deleted/banned subscriptions have full\n // access mode (no + or - in the dacs string). Changes to known subscriptions are sent as\n // deltas, but they should not happen here.\n const acs = new AccessMode(pres.dacs);\n if (!acs || acs.mode == AccessMode._INVALID) {\n this._tinode.logger(\"ERROR: Invalid access mode update\", pres.src, pres.dacs);\n return;\n } else if (acs.mode == AccessMode._NONE) {\n this._tinode.logger(\"WARNING: Removing non-existent subscription\", pres.src, pres.dacs);\n return;\n } else {\n // New subscription. Send request for the full description.\n // Using .withOneSub (not .withLaterOneSub) to make sure IfModifiedSince is not set.\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n // Create a dummy entry to catch online status update.\n const dummy = this._tinode.getTopic(pres.src);\n dummy.online = false;\n dummy.acs = acs;\n this._tinode._db.updTopic(dummy);\n }\n } else if (pres.what == 'tags') {\n this.getMeta(this.startMetaQuery().withTags().build());\n } else if (pres.what == 'msg') {\n // Message received for un unknown (previously deleted) topic.\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n // Create an entry to catch updates and messages.\n const dummy = this._tinode.getTopic(pres.src);\n dummy._deleted = false;\n this._tinode._db.updTopic(dummy);\n }\n\n this._refreshContact(pres.what, cont);\n }\n\n if (this.onPres) {\n this.onPres(pres);\n }\n }\n\n // Contact is updated, execute callbacks.\n _refreshContact(what, cont) {\n if (this.onContactUpdate) {\n this.onContactUpdate(what, cont);\n }\n }\n\n /**\n * Publishing to TopicMe is not supported. {@link Topic#publish} is overriden and thows an {Error} if called.\n * @memberof Tinode.TopicMe#\n * @throws {Error} Always throws an error.\n */\n publish() {\n return Promise.reject(new Error(\"Publishing to 'me' is not supported\"));\n }\n\n /**\n * Delete validation credential.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} topic - Name of the topic to delete\n * @param {string} user - User ID to remove.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCredential(method, value) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete credential in inactive 'me' topic\"));\n }\n // Send {del} message, return promise\n return this._tinode.delCredential(method, value).then(ctrl => {\n // Remove deleted credential from the cache.\n const index = this._credentials.findIndex((el) => {\n return el.meth == method && el.val == value;\n });\n if (index > -1) {\n this._credentials.splice(index, 1);\n }\n // Notify listeners\n if (this.onCredsUpdated) {\n this.onCredsUpdated(this._credentials);\n }\n return ctrl;\n });\n }\n\n /**\n * @callback contactFilter\n * @param {Object} contact to check for inclusion.\n * @returns {boolean} true
if contact should be processed, false
to exclude it.\n */\n /**\n * Iterate over cached contacts.\n *\n * @function\n * @memberof Tinode.TopicMe#\n * @param {TopicMe.ContactCallback} callback - Callback to call for each contact.\n * @param {contactFilter=} filter - Optionally filter contacts; include all if filter is false-ish, otherwise\n * include those for which filter returns true-ish.\n * @param {Object=} context - Context to use for calling the `callback`, i.e. the value of `this` inside the callback.\n */\n contacts(callback, filter, context) {\n this._tinode.mapTopics((c, idx) => {\n if (c.isCommType() && (!filter || filter(c))) {\n callback.call(context, c, idx);\n }\n });\n }\n\n /**\n * Get a contact from cache.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to get, either a UID (for p2p topics) or a topic name.\n * @returns {Tinode.Contact} - Contact or `undefined`.\n */\n getContact(name) {\n return this._tinode.cacheGetTopic(name);\n }\n\n /**\n * Get access mode of a given contact from cache.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to get access mode for, either a UID (for p2p topics)\n * or a topic name; if missing, access mode for the 'me' topic itself.\n * @returns {string} - access mode, such as `RWP`.\n */\n getAccessMode(name) {\n if (name) {\n const cont = this._tinode.cacheGetTopic(name);\n return cont ? cont.acs : null;\n }\n return this.acs;\n }\n\n /**\n * Check if contact is archived, i.e. contact.private.arch == true.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to check archived status, either a UID (for p2p topics) or a topic name.\n * @returns {boolean} - true if contact is archived, false otherwise.\n */\n isArchived(name) {\n const cont = this._tinode.cacheGetTopic(name);\n return cont && cont.private && !!cont.private.arch;\n }\n\n /**\n * @typedef Tinode.Credential\n * @memberof Tinode\n * @type Object\n * @property {string} meth - validation method such as 'email' or 'tel'.\n * @property {string} val - credential value, i.e. 'jdoe@example.com' or '+17025551234'\n * @property {boolean} done - true if credential is validated.\n */\n /**\n * Get the user's credentials: email, phone, etc.\n * @memberof Tinode.TopicMe#\n *\n * @returns {Tinode.Credential[]} - array of credentials.\n */\n getCredentials() {\n return this._credentials;\n }\n}\n\n/**\n * Special case of {@link Tinode.Topic} for searching for contacts and group topics\n * @extends Tinode.Topic\n *\n */\nexport class TopicFnd extends Topic {\n // List of users and topics uid or topic_name -> Contact object)\n _contacts = {};\n\n /**\n * Create TopicFnd.\n *\n * @param {TopicFnd.Callbacks} callbacks - Callbacks to receive various events.\n */\n constructor(callbacks) {\n super(Const.TOPIC_FND, callbacks);\n }\n\n // Override the original Topic._processMetaSubs\n _processMetaSubs(subs) {\n let updateCount = Object.getOwnPropertyNames(this._contacts).length;\n // Reset contact list.\n this._contacts = {};\n for (let idx in subs) {\n let sub = subs[idx];\n const indexBy = sub.topic ? sub.topic : sub.user;\n\n sub = mergeToCache(this._contacts, indexBy, sub);\n updateCount++;\n\n if (this.onMetaSub) {\n this.onMetaSub(sub);\n }\n }\n\n if (updateCount > 0 && this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._contacts));\n }\n }\n\n /**\n * Publishing to TopicFnd is not supported. {@link Topic#publish} is overriden and thows an {Error} if called.\n * @memberof Tinode.TopicFnd#\n * @throws {Error} Always throws an error.\n */\n publish() {\n return Promise.reject(new Error(\"Publishing to 'fnd' is not supported\"));\n }\n\n /**\n * setMeta to TopicFnd resets contact list in addition to sending the message.\n * @memberof Tinode.TopicFnd#\n * @param {Tinode.SetParams} params parameters to update.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n setMeta(params) {\n return Object.getPrototypeOf(TopicFnd.prototype).setMeta.call(this, params).then(_ => {\n if (Object.keys(this._contacts).length > 0) {\n this._contacts = {};\n if (this.onSubsUpdated) {\n this.onSubsUpdated([]);\n }\n }\n });\n }\n\n /**\n * Iterate over found contacts. If callback is undefined, use {@link this.onMetaSub}.\n * @function\n * @memberof Tinode.TopicFnd#\n * @param {TopicFnd.ContactCallback} callback - Callback to call for each contact.\n * @param {Object} context - Context to use for calling the `callback`, i.e. the value of `this` inside the callback.\n */\n contacts(callback, context) {\n const cb = (callback || this.onMetaSub);\n if (cb) {\n for (let idx in this._contacts) {\n cb.call(context, this._contacts[idx], idx, this._contacts);\n }\n }\n }\n}\n","/**\n * @file Utilities used in multiple places.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport AccessMode from './access-mode.js';\nimport {\n DEL_CHAR\n} from './config.js';\n\n// Attempt to convert date and AccessMode strings to objects.\nexport function jsonParseHelper(key, val) {\n // Try to convert string timestamps with optional milliseconds to Date,\n // e.g. 2015-09-02T01:45:43[.123]Z\n if (typeof val == 'string' && val.length >= 20 && val.length <= 24 && ['ts', 'touched', 'updated', 'created', 'when', 'deleted', 'expires'].includes(key)) {\n const date = new Date(val);\n if (!isNaN(date)) {\n return date;\n }\n } else if (key === 'acs' && typeof val === 'object') {\n return new AccessMode(val);\n }\n return val;\n}\n\n// Checks if URL is a relative url, i.e. has no 'scheme://', including the case of missing scheme '//'.\n// The scheme is expected to be RFC-compliant, e.g. [a-z][a-z0-9+.-]*\n// example.html - ok\n// https:example.com - not ok.\n// http:/example.com - not ok.\n// ' ↲ https://example.com' - not ok. (↲ means carriage return)\nexport function isUrlRelative(url) {\n return url && !/^\\s*([a-z][a-z0-9+.-]*:|\\/\\/)/im.test(url);\n}\n\nfunction isValidDate(d) {\n return (d instanceof Date) && !isNaN(d) && (d.getTime() != 0);\n}\n\n// RFC3339 formater of Date\nexport function rfc3339DateString(d) {\n if (!isValidDate(d)) {\n return undefined;\n }\n\n const pad = function(val, sp) {\n sp = sp || 2;\n return '0'.repeat(sp - ('' + val).length) + val;\n };\n\n const millis = d.getUTCMilliseconds();\n return d.getUTCFullYear() + '-' + pad(d.getUTCMonth() + 1) + '-' + pad(d.getUTCDate()) +\n 'T' + pad(d.getUTCHours()) + ':' + pad(d.getUTCMinutes()) + ':' + pad(d.getUTCSeconds()) +\n (millis ? '.' + pad(millis, 3) : '') + 'Z';\n}\n\n// Recursively merge src's own properties to dst.\n// Ignore properties where ignore[property] is true.\n// Array and Date objects are shallow-copied.\nexport function mergeObj(dst, src, ignore) {\n if (typeof src != 'object') {\n if (src === undefined) {\n return dst;\n }\n if (src === DEL_CHAR) {\n return undefined;\n }\n return src;\n }\n // JS is crazy: typeof null is 'object'.\n if (src === null) {\n return src;\n }\n\n // Handle Date\n if (src instanceof Date && !isNaN(src)) {\n return (!dst || !(dst instanceof Date) || isNaN(dst) || dst < src) ? src : dst;\n }\n\n // Access mode\n if (src instanceof AccessMode) {\n return new AccessMode(src);\n }\n\n // Handle Array\n if (src instanceof Array) {\n return src;\n }\n\n if (!dst || dst === DEL_CHAR) {\n dst = src.constructor();\n }\n\n for (let prop in src) {\n if (src.hasOwnProperty(prop) && (!ignore || !ignore[prop]) && (prop != '_noForwarding')) {\n try {\n dst[prop] = mergeObj(dst[prop], src[prop]);\n } catch (err) {\n // FIXME: probably need to log something here.\n }\n }\n }\n return dst;\n}\n\n// Update object stored in a cache. Returns updated value.\nexport function mergeToCache(cache, key, newval, ignore) {\n cache[key] = mergeObj(cache[key], newval, ignore);\n return cache[key];\n}\n\n// Strips all values from an object of they evaluate to false or if their name starts with '_'.\n// Used on all outgoing object before serialization to string.\nexport function simplify(obj) {\n Object.keys(obj).forEach((key) => {\n if (key[0] == '_') {\n // Strip fields like \"obj._key\".\n delete obj[key];\n } else if (!obj[key]) {\n // Strip fields which evaluate to false.\n delete obj[key];\n } else if (Array.isArray(obj[key]) && obj[key].length == 0) {\n // Strip empty arrays.\n delete obj[key];\n } else if (!obj[key]) {\n // Strip fields which evaluate to false.\n delete obj[key];\n } else if (obj[key] instanceof Date) {\n // Strip invalid or zero date.\n if (!isValidDate(obj[key])) {\n delete obj[key];\n }\n } else if (typeof obj[key] == 'object') {\n simplify(obj[key]);\n // Strip empty objects.\n if (Object.getOwnPropertyNames(obj[key]).length == 0) {\n delete obj[key];\n }\n }\n });\n return obj;\n};\n\n\n// Trim whitespace, strip empty and duplicate elements elements.\n// If the result is an empty array, add a single element \"\\u2421\" (Unicode Del character).\nexport function normalizeArray(arr) {\n let out = [];\n if (Array.isArray(arr)) {\n // Trim, throw away very short and empty tags.\n for (let i = 0, l = arr.length; i < l; i++) {\n let t = arr[i];\n if (t) {\n t = t.trim().toLowerCase();\n if (t.length > 1) {\n out.push(t);\n }\n }\n }\n out.sort().filter(function(item, pos, ary) {\n return !pos || item != ary[pos - 1];\n });\n }\n if (out.length == 0) {\n // Add single tag with a Unicode Del character, otherwise an ampty array\n // is ambiguos. The Del tag will be stripped by the server.\n out.push(DEL_CHAR);\n }\n return out;\n}\n","export const PACKAGE_VERSION = \"0.22.12\";\n","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","/**\n * @module tinode-sdk\n *\n * @copyright 2015-2022 Tinode LLC.\n * @summary Javascript bindings for Tinode.\n * @license Apache 2.0\n * @version 0.20\n *\n * See https://github.com/tinode/webapp for real-life usage.\n *\n * @example\n * \n * \n * \n *\n * \n * ...\n * \n * \n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nimport AccessMode from './access-mode.js';\nimport * as Const from './config.js';\nimport CommError from './comm-error.js';\nimport Connection from './connection.js';\nimport DBCache from './db.js';\nimport Drafty from './drafty.js';\nimport LargeFileHelper from './large-file.js';\nimport MetaGetBuilder from './meta-builder.js';\nimport {\n Topic,\n TopicMe,\n TopicFnd\n} from './topic.js';\n\nimport {\n isUrlRelative,\n jsonParseHelper,\n mergeObj,\n rfc3339DateString,\n simplify\n} from './utils.js';\n\n// Re-export AccessMode\nexport {\n AccessMode\n};\n\nlet WebSocketProvider;\nif (typeof WebSocket != 'undefined') {\n WebSocketProvider = WebSocket;\n}\n\nlet XHRProvider;\nif (typeof XMLHttpRequest != 'undefined') {\n XHRProvider = XMLHttpRequest;\n}\n\nlet IndexedDBProvider;\nif (typeof indexedDB != 'undefined') {\n IndexedDBProvider = indexedDB;\n}\n\n// Re-export Drafty.\nexport {\n Drafty\n}\n\ninitForNonBrowserApp();\n\n// Utility functions\n\n// Polyfill for non-browser context, e.g. NodeJs.\nfunction initForNonBrowserApp() {\n // Tinode requirement in native mode because react native doesn't provide Base64 method\n const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';\n\n if (typeof btoa == 'undefined') {\n global.btoa = function(input = '') {\n let str = input;\n let output = '';\n\n for (let block = 0, charCode, i = 0, map = chars; str.charAt(i | 0) || (map = '=', i % 1); output += map.charAt(63 & block >> 8 - i % 1 * 8)) {\n\n charCode = str.charCodeAt(i += 3 / 4);\n\n if (charCode > 0xFF) {\n throw new Error(\"'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.\");\n }\n block = block << 8 | charCode;\n }\n\n return output;\n };\n }\n\n if (typeof atob == 'undefined') {\n global.atob = function(input = '') {\n let str = input.replace(/=+$/, '');\n let output = '';\n\n if (str.length % 4 == 1) {\n throw new Error(\"'atob' failed: The string to be decoded is not correctly encoded.\");\n }\n for (let bc = 0, bs = 0, buffer, i = 0; buffer = str.charAt(i++);\n\n ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,\n bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0\n ) {\n buffer = chars.indexOf(buffer);\n }\n\n return output;\n };\n }\n\n if (typeof window == 'undefined') {\n global.window = {\n WebSocket: WebSocketProvider,\n XMLHttpRequest: XHRProvider,\n indexedDB: IndexedDBProvider,\n URL: {\n createObjectURL: function() {\n throw new Error(\"Unable to use URL.createObjectURL in a non-browser application\");\n }\n }\n }\n }\n\n Connection.setNetworkProviders(WebSocketProvider, XHRProvider);\n LargeFileHelper.setNetworkProvider(XHRProvider);\n DBCache.setDatabaseProvider(IndexedDBProvider);\n}\n\n// Detect find most useful network transport.\nfunction detectTransport() {\n if (typeof window == 'object') {\n if (window['WebSocket']) {\n return 'ws';\n } else if (window['XMLHttpRequest']) {\n // The browser or node has no websockets, using long polling.\n return 'lp';\n }\n }\n return null;\n}\n\n// btoa replacement. Stock btoa fails on on non-Latin1 strings.\nfunction b64EncodeUnicode(str) {\n // The encodeURIComponent percent-encodes UTF-8 string,\n // then the percent encoding is converted into raw bytes which\n // can be fed into btoa.\n return btoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g,\n function toSolidBytes(match, p1) {\n return String.fromCharCode('0x' + p1);\n }));\n}\n\n// JSON stringify helper - pre-processor for JSON.stringify\nfunction jsonBuildHelper(key, val) {\n if (val instanceof Date) {\n // Convert javascript Date objects to rfc3339 strings\n val = rfc3339DateString(val);\n } else if (val instanceof AccessMode) {\n val = val.jsonHelper();\n } else if (val === undefined || val === null || val === false ||\n (Array.isArray(val) && val.length == 0) ||\n ((typeof val == 'object') && (Object.keys(val).length == 0))) {\n // strip out empty elements while serializing objects to JSON\n return undefined;\n }\n\n return val;\n};\n\n// Trims very long strings (encoded images) to make logged packets more readable.\nfunction jsonLoggerHelper(key, val) {\n if (typeof val == 'string' && val.length > 128) {\n return '<' + val.length + ', bytes: ' + val.substring(0, 12) + '...' + val.substring(val.length - 12) + '>';\n }\n return jsonBuildHelper(key, val);\n};\n\n// Parse browser user agent to extract browser name and version.\nfunction getBrowserInfo(ua, product) {\n ua = ua || '';\n let reactnative = '';\n // Check if this is a ReactNative app.\n if (/reactnative/i.test(product)) {\n reactnative = 'ReactNative; ';\n }\n let result;\n // Remove useless string.\n ua = ua.replace(' (KHTML, like Gecko)', '');\n // Test for WebKit-based browser.\n let m = ua.match(/(AppleWebKit\\/[.\\d]+)/i);\n if (m) {\n // List of common strings, from more useful to less useful.\n // All unknown strings get the highest (-1) priority.\n const priority = ['edg', 'chrome', 'safari', 'mobile', 'version'];\n let tmp = ua.substr(m.index + m[0].length).split(' ');\n let tokens = [];\n let version; // 1.0 in Version/1.0 or undefined;\n // Split string like 'Name/0.0.0' into ['Name', '0.0.0', 3] where the last element is the priority.\n for (let i = 0; i < tmp.length; i++) {\n let m2 = /([\\w.]+)[\\/]([\\.\\d]+)/.exec(tmp[i]);\n if (m2) {\n // Unknown values are highest priority (-1).\n tokens.push([m2[1], m2[2], priority.findIndex((e) => {\n return m2[1].toLowerCase().startsWith(e);\n })]);\n if (m2[1] == 'Version') {\n version = m2[2];\n }\n }\n }\n // Sort by priority: more interesting is earlier than less interesting.\n tokens.sort((a, b) => {\n return a[2] - b[2];\n });\n if (tokens.length > 0) {\n // Return the least common browser string and version.\n if (tokens[0][0].toLowerCase().startsWith('edg')) {\n tokens[0][0] = 'Edge';\n } else if (tokens[0][0] == 'OPR') {\n tokens[0][0] = 'Opera';\n } else if (tokens[0][0] == 'Safari' && version) {\n tokens[0][1] = version;\n }\n result = tokens[0][0] + '/' + tokens[0][1];\n } else {\n // Failed to ID the browser. Return the webkit version.\n result = m[1];\n }\n } else if (/firefox/i.test(ua)) {\n m = /Firefox\\/([.\\d]+)/g.exec(ua);\n if (m) {\n result = 'Firefox/' + m[1];\n } else {\n result = 'Firefox/?';\n }\n } else {\n // Neither AppleWebKit nor Firefox. Try the last resort.\n m = /([\\w.]+)\\/([.\\d]+)/.exec(ua);\n if (m) {\n result = m[1] + '/' + m[2];\n } else {\n m = ua.split(' ');\n result = m[0];\n }\n }\n\n // Shorten the version to one dot 'a.bb.ccc.d -> a.bb' at most.\n m = result.split('/');\n if (m.length > 1) {\n const v = m[1].split('.');\n const minor = v[1] ? '.' + v[1].substr(0, 2) : '';\n result = `${m[0]}/${v[0]}${minor}`;\n }\n return reactnative + result;\n}\n\n/**\n * The main class for interacting with Tinode server.\n */\nexport class Tinode {\n _host;\n _secure;\n\n _appName;\n\n // API Key.\n _apiKey;\n\n // Name and version of the browser.\n _browser = '';\n _platform;\n // Hardware\n _hwos = 'undefined';\n _humanLanguage = 'xx';\n\n // Logging to console enabled\n _loggingEnabled = false;\n // When logging, trip long strings (base64-encoded images) for readability\n _trimLongStrings = false;\n // UID of the currently authenticated user.\n _myUID = null;\n // Status of connection: authenticated or not.\n _authenticated = false;\n // Login used in the last successful basic authentication\n _login = null;\n // Token which can be used for login instead of login/password.\n _authToken = null;\n // Counter of received packets\n _inPacketCount = 0;\n // Counter for generating unique message IDs\n _messageId = Math.floor((Math.random() * 0xFFFF) + 0xFFFF);\n // Information about the server, if connected\n _serverInfo = null;\n // Push notification token. Called deviceToken for consistency with the Android SDK.\n _deviceToken = null;\n\n // Cache of pending promises by message id.\n _pendingPromises = {};\n // The Timeout object returned by the reject expired promises setInterval.\n _expirePromises = null;\n\n // Websocket or long polling connection.\n _connection = null;\n\n // Use indexDB for caching topics and messages.\n _persist = false;\n // IndexedDB wrapper object.\n _db = null;\n\n // Tinode's cache of objects\n _cache = {};\n\n /**\n * Create Tinode object.\n *\n * @param {Object} config - configuration parameters.\n * @param {string} config.appName - Name of the calling application to be reported in the User Agent.\n * @param {string} config.host - Host name and optional port number to connect to.\n * @param {string} config.apiKey - API key generated by keygen
.\n * @param {string} config.transport - See {@link Tinode.Connection#transport}.\n * @param {boolean} config.secure - Use Secure WebSocket if true
.\n * @param {string} config.platform - Optional platform identifier, one of \"ios\"
, \"web\"
, \"android\"
.\n * @param {boolen} config.persist - Use IndexedDB persistent storage.\n * @param {function} onComplete - callback to call when initialization is completed.\n */\n constructor(config, onComplete) {\n this._host = config.host;\n this._secure = config.secure;\n\n // Client-provided application name, format /\n this._appName = config.appName || \"Undefined\";\n\n // API Key.\n this._apiKey = config.apiKey;\n\n // Name and version of the browser.\n this._platform = config.platform || 'web';\n // Underlying OS.\n if (typeof navigator != 'undefined') {\n this._browser = getBrowserInfo(navigator.userAgent, navigator.product);\n this._hwos = navigator.platform;\n // This is the default language. It could be changed by client.\n this._humanLanguage = navigator.language || 'en-US';\n }\n\n Connection.logger = this.logger;\n Drafty.logger = this.logger;\n\n // WebSocket or long polling network connection.\n if (config.transport != 'lp' && config.transport != 'ws') {\n config.transport = detectTransport();\n }\n this._connection = new Connection(config, Const.PROTOCOL_VERSION, /* autoreconnect */ true);\n this._connection.onMessage = (data) => {\n // Call the main message dispatcher.\n this.#dispatchMessage(data);\n }\n\n // Ready to start sending.\n this._connection.onOpen = _ => this.#connectionOpen();\n this._connection.onDisconnect = (err, code) => this.#disconnected(err, code);\n\n // Wrapper for the reconnect iterator callback.\n this._connection.onAutoreconnectIteration = (timeout, promise) => {\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(timeout, promise);\n }\n }\n\n this._persist = config.persist;\n // Initialize object regardless. It simplifies the code.\n this._db = new DBCache(err => {\n this.logger('DB', err);\n }, this.logger);\n\n if (this._persist) {\n // Create the persistent cache.\n // Store promises to be resolved when messages load into memory.\n const prom = [];\n this._db.initDatabase().then(_ => {\n // First load topics into memory.\n return this._db.mapTopics((data) => {\n let topic = this.#cacheGet('topic', data.name);\n if (topic) {\n return;\n }\n if (data.name == Const.TOPIC_ME) {\n topic = new TopicMe();\n } else if (data.name == Const.TOPIC_FND) {\n topic = new TopicFnd();\n } else {\n topic = new Topic(data.name);\n }\n this._db.deserializeTopic(topic, data);\n this.#attachCacheToTopic(topic);\n topic._cachePutSelf();\n // Topic loaded from DB is not new.\n delete topic._new;\n // Request to load messages and save the promise.\n prom.push(topic._loadMessages(this._db));\n });\n }).then(_ => {\n // Then load users.\n return this._db.mapUsers((data) => {\n this.#cachePut('user', data.uid, mergeObj({}, data.public));\n });\n }).then(_ => {\n // Now wait for all messages to finish loading.\n return Promise.all(prom);\n }).then(_ => {\n if (onComplete) {\n onComplete();\n }\n this.logger(\"Persistent cache initialized.\");\n }).catch(err => {\n if (onComplete) {\n onComplete(err);\n }\n this.logger(\"Failed to initialize persistent cache:\", err);\n });\n } else {\n this._db.deleteDatabase().then(_ => {\n if (onComplete) {\n onComplete();\n }\n });\n }\n }\n\n // Private methods.\n\n // Console logger. Babel somehow fails to parse '...rest' parameter.\n logger(str, ...args) {\n if (this._loggingEnabled) {\n const d = new Date();\n const dateString = ('0' + d.getUTCHours()).slice(-2) + ':' +\n ('0' + d.getUTCMinutes()).slice(-2) + ':' +\n ('0' + d.getUTCSeconds()).slice(-2) + '.' +\n ('00' + d.getUTCMilliseconds()).slice(-3);\n\n console.log('[' + dateString + ']', str, args.join(' '));\n }\n }\n\n // Generator of default promises for sent packets.\n #makePromise(id) {\n let promise = null;\n if (id) {\n promise = new Promise((resolve, reject) => {\n // Stored callbacks will be called when the response packet with this Id arrives\n this._pendingPromises[id] = {\n 'resolve': resolve,\n 'reject': reject,\n 'ts': new Date()\n };\n });\n }\n return promise;\n };\n\n // Resolve or reject a pending promise.\n // Unresolved promises are stored in _pendingPromises.\n #execPromise(id, code, onOK, errorText) {\n const callbacks = this._pendingPromises[id];\n if (callbacks) {\n delete this._pendingPromises[id];\n if (code >= 200 && code < 400) {\n if (callbacks.resolve) {\n callbacks.resolve(onOK);\n }\n } else if (callbacks.reject) {\n callbacks.reject(new CommError(errorText, code));\n }\n }\n }\n\n // Send a packet. If packet id is provided return a promise.\n #send(pkt, id) {\n let promise;\n if (id) {\n promise = this.#makePromise(id);\n }\n pkt = simplify(pkt);\n let msg = JSON.stringify(pkt);\n this.logger(\"out: \" + (this._trimLongStrings ? JSON.stringify(pkt, jsonLoggerHelper) : msg));\n try {\n this._connection.sendText(msg);\n } catch (err) {\n // If sendText throws, wrap the error in a promise or rethrow.\n if (id) {\n this.#execPromise(id, Connection.NETWORK_ERROR, null, err.message);\n } else {\n throw err;\n }\n }\n return promise;\n }\n\n // The main message dispatcher.\n #dispatchMessage(data) {\n // Skip empty response. This happens when LP times out.\n if (!data)\n return;\n\n this._inPacketCount++;\n\n // Send raw message to listener\n if (this.onRawMessage) {\n this.onRawMessage(data);\n }\n\n if (data === '0') {\n // Server response to a network probe.\n if (this.onNetworkProbe) {\n this.onNetworkProbe();\n }\n // No processing is necessary.\n return;\n }\n\n let pkt = JSON.parse(data, jsonParseHelper);\n if (!pkt) {\n this.logger(\"in: \" + data);\n this.logger(\"ERROR: failed to parse data\");\n } else {\n this.logger(\"in: \" + (this._trimLongStrings ? JSON.stringify(pkt, jsonLoggerHelper) : data));\n\n // Send complete packet to listener\n if (this.onMessage) {\n this.onMessage(pkt);\n }\n\n if (pkt.ctrl) {\n // Handling {ctrl} message\n if (this.onCtrlMessage) {\n this.onCtrlMessage(pkt.ctrl);\n }\n\n // Resolve or reject a pending promise, if any\n if (pkt.ctrl.id) {\n this.#execPromise(pkt.ctrl.id, pkt.ctrl.code, pkt.ctrl, pkt.ctrl.text);\n }\n setTimeout(_ => {\n if (pkt.ctrl.code == 205 && pkt.ctrl.text == 'evicted') {\n // User evicted from topic.\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n topic._resetSub();\n if (pkt.ctrl.params && pkt.ctrl.params.unsub) {\n topic._gone();\n }\n }\n } else if (pkt.ctrl.code < 300 && pkt.ctrl.params) {\n if (pkt.ctrl.params.what == 'data') {\n // code=208, all messages received: \"params\":{\"count\":11,\"what\":\"data\"},\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n topic._allMessagesReceived(pkt.ctrl.params.count);\n }\n } else if (pkt.ctrl.params.what == 'sub') {\n // code=204, the topic has no (refreshed) subscriptions.\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n // Trigger topic.onSubsUpdated.\n topic._processMetaSubs([]);\n }\n }\n }\n }, 0);\n } else {\n setTimeout(_ => {\n if (pkt.meta) {\n // Handling a {meta} message.\n // Preferred API: Route meta to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.meta.topic);\n if (topic) {\n topic._routeMeta(pkt.meta);\n }\n\n if (pkt.meta.id) {\n this.#execPromise(pkt.meta.id, 200, pkt.meta, 'META');\n }\n\n // Secondary API: callback\n if (this.onMetaMessage) {\n this.onMetaMessage(pkt.meta);\n }\n } else if (pkt.data) {\n // Handling {data} message\n // Preferred API: Route data to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.data.topic);\n if (topic) {\n topic._routeData(pkt.data);\n }\n\n // Secondary API: Call callback\n if (this.onDataMessage) {\n this.onDataMessage(pkt.data);\n }\n } else if (pkt.pres) {\n // Handling {pres} message\n // Preferred API: Route presence to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.pres.topic);\n if (topic) {\n topic._routePres(pkt.pres);\n }\n\n // Secondary API - callback\n if (this.onPresMessage) {\n this.onPresMessage(pkt.pres);\n }\n } else if (pkt.info) {\n // {info} message - read/received notifications and key presses\n // Preferred API: Route {info}} to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.info.topic);\n if (topic) {\n topic._routeInfo(pkt.info);\n }\n\n // Secondary API - callback\n if (this.onInfoMessage) {\n this.onInfoMessage(pkt.info);\n }\n } else {\n this.logger(\"ERROR: Unknown packet received.\");\n }\n }, 0);\n }\n }\n }\n\n // Connection open, ready to start sending.\n #connectionOpen() {\n if (!this._expirePromises) {\n // Reject promises which have not been resolved for too long.\n this._expirePromises = setInterval(_ => {\n const err = new CommError(\"timeout\", 504);\n const expires = new Date(new Date().getTime() - Const.EXPIRE_PROMISES_TIMEOUT);\n for (let id in this._pendingPromises) {\n let callbacks = this._pendingPromises[id];\n if (callbacks && callbacks.ts < expires) {\n this.logger(\"Promise expired\", id);\n delete this._pendingPromises[id];\n if (callbacks.reject) {\n callbacks.reject(err);\n }\n }\n }\n }, Const.EXPIRE_PROMISES_PERIOD);\n }\n this.hello();\n }\n\n #disconnected(err, code) {\n this._inPacketCount = 0;\n this._serverInfo = null;\n this._authenticated = false;\n\n if (this._expirePromises) {\n clearInterval(this._expirePromises);\n this._expirePromises = null;\n }\n\n // Mark all topics as unsubscribed\n this.#cacheMap('topic', (topic, key) => {\n topic._resetSub();\n });\n\n // Reject all pending promises\n for (let key in this._pendingPromises) {\n const callbacks = this._pendingPromises[key];\n if (callbacks && callbacks.reject) {\n callbacks.reject(err);\n }\n }\n this._pendingPromises = {};\n\n if (this.onDisconnect) {\n this.onDisconnect(err);\n }\n }\n\n // Get User Agent string\n #getUserAgent() {\n return this._appName + ' (' + (this._browser ? this._browser + '; ' : '') + this._hwos + '); ' + Const.LIBRARY;\n }\n\n // Generator of packets stubs\n #initPacket(type, topic) {\n switch (type) {\n case 'hi':\n return {\n 'hi': {\n 'id': this.getNextUniqueId(),\n 'ver': Const.VERSION,\n 'ua': this.#getUserAgent(),\n 'dev': this._deviceToken,\n 'lang': this._humanLanguage,\n 'platf': this._platform\n }\n };\n\n case 'acc':\n return {\n 'acc': {\n 'id': this.getNextUniqueId(),\n 'user': null,\n 'scheme': null,\n 'secret': null,\n 'tmpscheme': null,\n 'tmpsecret': null,\n 'login': false,\n 'tags': null,\n 'desc': {},\n 'cred': {}\n }\n };\n\n case 'login':\n return {\n 'login': {\n 'id': this.getNextUniqueId(),\n 'scheme': null,\n 'secret': null\n }\n };\n\n case 'sub':\n return {\n 'sub': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'set': {},\n 'get': {}\n }\n };\n\n case 'leave':\n return {\n 'leave': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'unsub': false\n }\n };\n\n case 'pub':\n return {\n 'pub': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'noecho': false,\n 'head': null,\n 'content': {}\n }\n };\n\n case 'get':\n return {\n 'get': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'what': null,\n 'desc': {},\n 'sub': {},\n 'data': {}\n }\n };\n\n case 'set':\n return {\n 'set': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'desc': {},\n 'sub': {},\n 'tags': [],\n 'ephemeral': {}\n }\n };\n\n case 'del':\n return {\n 'del': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'what': null,\n 'delseq': null,\n 'user': null,\n 'hard': false\n }\n };\n\n case 'note':\n return {\n 'note': {\n // no id by design (except calls).\n 'topic': topic,\n 'what': null, // one of \"recv\", \"read\", \"kp\", \"call\"\n 'seq': undefined // the server-side message id acknowledged as received or read.\n }\n };\n\n default:\n throw new Error(`Unknown packet type requested: ${type}`);\n }\n }\n\n // Cache management\n #cachePut(type, name, obj) {\n this._cache[type + ':' + name] = obj;\n }\n #cacheGet(type, name) {\n return this._cache[type + ':' + name];\n }\n #cacheDel(type, name) {\n delete this._cache[type + ':' + name];\n }\n\n // Enumerate all items in cache, call func for each item.\n // Enumeration stops if func returns true.\n #cacheMap(type, func, context) {\n const key = type ? type + ':' : undefined;\n for (let idx in this._cache) {\n if (!key || idx.indexOf(key) == 0) {\n if (func.call(context, this._cache[idx], idx)) {\n break;\n }\n }\n }\n }\n\n // Make limited cache management available to topic.\n // Caching user.public only. Everything else is per-topic.\n #attachCacheToTopic(topic) {\n topic._tinode = this;\n\n topic._cacheGetUser = (uid) => {\n const pub = this.#cacheGet('user', uid);\n if (pub) {\n return {\n user: uid,\n public: mergeObj({}, pub)\n };\n }\n return undefined;\n };\n topic._cachePutUser = (uid, user) => {\n this.#cachePut('user', uid, mergeObj({}, user.public));\n };\n topic._cacheDelUser = (uid) => {\n this.#cacheDel('user', uid);\n };\n topic._cachePutSelf = _ => {\n this.#cachePut('topic', topic.name, topic);\n };\n topic._cacheDelSelf = _ => {\n this.#cacheDel('topic', topic.name);\n };\n }\n\n // On successful login save server-provided data.\n #loginSuccessful(ctrl) {\n if (!ctrl.params || !ctrl.params.user) {\n return ctrl;\n }\n // This is a response to a successful login,\n // extract UID and security token, save it in Tinode module\n this._myUID = ctrl.params.user;\n this._authenticated = (ctrl && ctrl.code >= 200 && ctrl.code < 300);\n if (ctrl.params && ctrl.params.token && ctrl.params.expires) {\n this._authToken = {\n token: ctrl.params.token,\n expires: ctrl.params.expires\n };\n } else {\n this._authToken = null;\n }\n\n if (this.onLogin) {\n this.onLogin(ctrl.code, ctrl.text);\n }\n\n return ctrl;\n }\n\n // Static methods.\n /**\n * Helper method to package account credential.\n *\n * @param {string | Credential} meth - validation method or object with validation data.\n * @param {string=} val - validation value (e.g. email or phone number).\n * @param {Object=} params - validation parameters.\n * @param {string=} resp - validation response.\n *\n * @returns {Array.} array with a single credential or null
if no valid credentials were given.\n */\n static credential(meth, val, params, resp) {\n if (typeof meth == 'object') {\n ({\n val,\n params,\n resp,\n meth\n } = meth);\n }\n if (meth && (val || resp)) {\n return [{\n 'meth': meth,\n 'val': val,\n 'resp': resp,\n 'params': params\n }];\n }\n return null;\n }\n\n /**\n * Determine topic type from topic's name: grp, p2p, me, fnd, sys.\n * @param {string} name - Name of the topic to test.\n * @returns {string} One of \"me\"
, \"fnd\"
, \"sys\"
, \"grp\"
,\n * \"p2p\"
or undefined
.\n */\n static topicType(name) {\n return Topic.topicType(name);\n }\n\n /**\n * Check if the given topic name is a name of a 'me' topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a 'me' topic, false
otherwise.\n */\n static isMeTopicName(name) {\n return Topic.isMeTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a group topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a group topic, false
otherwise.\n */\n static isGroupTopicName(name) {\n return Topic.isGroupTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a p2p topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p topic, false
otherwise.\n */\n static isP2PTopicName(name) {\n return Topic.isP2PTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a communication topic, i.e. P2P or group.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p or group topic, false
otherwise.\n */\n static isCommTopicName(name) {\n return Topic.isCommTopicName(name);\n }\n /**\n * Check if the topic name is a name of a new topic.\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a new topic, false
otherwise.\n */\n static isNewGroupTopicName(name) {\n return Topic.isNewGroupTopicName(name);\n }\n /**\n * Check if the topic name is a name of a channel.\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a channel, false
otherwise.\n */\n static isChannelTopicName(name) {\n return Topic.isChannelTopicName(name);\n }\n /**\n * Get information about the current version of this Tinode client library.\n * @returns {string} semantic version of the library, e.g. \"0.15.5-rc1\"
.\n */\n static getVersion() {\n return Const.VERSION;\n }\n /**\n * To use Tinode in a non browser context, supply WebSocket and XMLHttpRequest providers.\n * @static\n *\n * @param wsProvider WebSocket
provider, e.g. for nodeJS , require('ws')
.\n * @param xhrProvider XMLHttpRequest
provider, e.g. for node require('xhr')
.\n */\n static setNetworkProviders(wsProvider, xhrProvider) {\n WebSocketProvider = wsProvider;\n XHRProvider = xhrProvider;\n\n Connection.setNetworkProviders(WebSocketProvider, XHRProvider);\n LargeFileHelper.setNetworkProvider(XHRProvider);\n }\n /**\n * To use Tinode in a non browser context, supply indexedDB
provider.\n * @static\n *\n * @param idbProvider indexedDB
provider, e.g. for nodeJS , require('fake-indexeddb')
.\n */\n static setDatabaseProvider(idbProvider) {\n IndexedDBProvider = idbProvider;\n\n DBCache.setDatabaseProvider(IndexedDBProvider);\n }\n /**\n * Return information about the current name and version of this Tinode library.\n * @static\n *\n * @returns {string} the name of the library and it's version.\n */\n static getLibrary() {\n return Const.LIBRARY;\n }\n /**\n * Check if the given string represents NULL
value as defined by Tinode ('\\u2421'
).\n * @param {string} str - string to check for NULL
value.\n * @returns {boolean} true
if string represents NULL
value, false
otherwise.\n */\n static isNullValue(str) {\n return str === Const.DEL_CHAR;\n }\n\n // Instance methods.\n\n // Generates unique message IDs\n getNextUniqueId() {\n return (this._messageId != 0) ? '' + this._messageId++ : undefined;\n };\n\n /**\n * Connect to the server.\n *\n * @param {string} host_ - name of the host to connect to.\n * @return {Promise} Promise resolved/rejected when the connection call completes:\n * resolve()
is called without parameters, reject()
receives the\n * Error
as a single parameter.\n */\n connect(host_) {\n return this._connection.connect(host_);\n }\n\n /**\n * Attempt to reconnect to the server immediately.\n *\n * @param {string} force - if true
, reconnect even if there is a connection already.\n */\n reconnect(force) {\n this._connection.reconnect(force);\n }\n\n /**\n * Disconnect from the server.\n */\n disconnect() {\n this._connection.disconnect();\n }\n\n /**\n * Clear persistent cache: remove IndexedDB.\n *\n * @return {Promise} Promise resolved/rejected when the operation is completed.\n */\n clearStorage() {\n if (this._db.isReady()) {\n return this._db.deleteDatabase();\n }\n return Promise.resolve();\n }\n\n /**\n * Initialize persistent cache: create IndexedDB cache.\n *\n * @return {Promise} Promise resolved/rejected when the operation is completed.\n */\n initStorage() {\n if (!this._db.isReady()) {\n return this._db.initDatabase();\n }\n return Promise.resolve();\n }\n\n /**\n * Send a network probe message to make sure the connection is alive.\n */\n networkProbe() {\n this._connection.probe();\n }\n\n /**\n * Check for live connection to server.\n *\n * @returns {boolean} true
if there is a live connection, false
otherwise.\n */\n isConnected() {\n return this._connection.isConnected();\n }\n\n /**\n * Check if connection is authenticated (last login was successful).\n *\n * @returns {boolean} true
if authenticated, false
otherwise.\n */\n isAuthenticated() {\n return this._authenticated;\n }\n\n /**\n * Add API key and auth token to the relative URL making it usable for getting data\n * from the server in a simple HTTP GET
request.\n *\n * @param {string} URL - URL to wrap.\n * @returns {string} URL with appended API key and token, if valid token is present.\n */\n authorizeURL(url) {\n if (typeof url != 'string') {\n return url;\n }\n\n if (isUrlRelative(url)) {\n // Fake base to make the relative URL parseable.\n const base = 'scheme://host/';\n const parsed = new URL(url, base);\n if (this._apiKey) {\n parsed.searchParams.append('apikey', this._apiKey);\n }\n if (this._authToken && this._authToken.token) {\n parsed.searchParams.append('auth', 'token');\n parsed.searchParams.append('secret', this._authToken.token);\n }\n // Convert back to string and strip fake base URL except for the root slash.\n url = parsed.toString().substring(base.length - 1);\n }\n return url;\n }\n\n /**\n * @typedef AccountParams\n * @type {Object}\n * @property {DefAcs=} defacs - Default access parameters for user's me
topic.\n * @property {Object=} public - Public application-defined data exposed on me
topic.\n * @property {Object=} private - Private application-defined data accessible on me
topic.\n * @property {Object=} trusted - Trusted user data which can be set by a root user only.\n * @property {Array.} tags - array of string tags for user discovery.\n * @property {string} scheme - Temporary authentication scheme for password reset.\n * @property {string} secret - Temporary authentication secret for password reset.\n * @property {Array.=} attachments - Array of references to out of band attachments used in account description.\n */\n /**\n * @typedef DefAcs\n * @type {Object}\n * @property {string=} auth - Access mode for me
for authenticated users.\n * @property {string=} anon - Access mode for me
for anonymous users.\n */\n\n /**\n * Create or update an account.\n *\n * @param {string} uid - User id to update\n * @param {string} scheme - Authentication scheme; \"basic\"
and \"anonymous\"
are the currently supported schemes.\n * @param {string} secret - Authentication secret, assumed to be already base64 encoded.\n * @param {boolean=} login - Use new account to authenticate current session\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n account(uid, scheme, secret, login, params) {\n const pkt = this.#initPacket('acc');\n pkt.acc.user = uid;\n pkt.acc.scheme = scheme;\n pkt.acc.secret = secret;\n // Log in to the new account using selected scheme\n pkt.acc.login = login;\n\n if (params) {\n pkt.acc.desc.defacs = params.defacs;\n pkt.acc.desc.public = params.public;\n pkt.acc.desc.private = params.private;\n pkt.acc.desc.trusted = params.trusted;\n\n pkt.acc.tags = params.tags;\n pkt.acc.cred = params.cred;\n\n pkt.acc.tmpscheme = params.scheme;\n pkt.acc.tmpsecret = params.secret;\n\n if (Array.isArray(params.attachments) && params.attachments.length > 0) {\n pkt.extra = {\n attachments: params.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n }\n\n return this.#send(pkt, pkt.acc.id);\n }\n\n /**\n * Create a new user. Wrapper for {@link Tinode#account}.\n *\n * @param {string} scheme - Authentication scheme; \"basic\"
is the only currently supported scheme.\n * @param {string} secret - Authentication.\n * @param {boolean=} login - Use new account to authenticate current session\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n createAccount(scheme, secret, login, params) {\n let promise = this.account(Const.USER_NEW, scheme, secret, login, params);\n if (login) {\n promise = promise.then(ctrl => this.#loginSuccessful(ctrl));\n }\n return promise;\n }\n\n /**\n * Create user with 'basic'
authentication scheme and immediately\n * use it for authentication. Wrapper for {@link Tinode#account}.\n *\n * @param {string} username - Login to use for the new account.\n * @param {string} password - User's password.\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n createAccountBasic(username, password, params) {\n // Make sure we are not using 'null' or 'undefined';\n username = username || '';\n password = password || '';\n return this.createAccount('basic',\n b64EncodeUnicode(username + ':' + password), true, params);\n }\n\n /**\n * Update user's credentials for 'basic'
authentication scheme. Wrapper for {@link Tinode#account}.\n *\n * @param {string} uid - User ID to update.\n * @param {string} username - Login to use for the new account.\n * @param {string} password - User's password.\n * @param {AccountParams=} params - data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n updateAccountBasic(uid, username, password, params) {\n // Make sure we are not using 'null' or 'undefined';\n username = username || '';\n password = password || '';\n return this.account(uid, 'basic',\n b64EncodeUnicode(username + ':' + password), false, params);\n }\n\n /**\n * Send handshake to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n hello() {\n const pkt = this.#initPacket('hi');\n\n return this.#send(pkt, pkt.hi.id)\n .then(ctrl => {\n // Reset backoff counter on successful connection.\n this._connection.backoffReset();\n\n // Server response contains server protocol version, build, constraints,\n // session ID for long polling. Save them.\n if (ctrl.params) {\n this._serverInfo = ctrl.params;\n }\n\n if (this.onConnect) {\n this.onConnect();\n }\n\n return ctrl;\n }).catch(err => {\n this._connection.reconnect(true);\n\n if (this.onDisconnect) {\n this.onDisconnect(err);\n }\n });\n }\n\n /**\n * Set or refresh the push notifications/device token. If the client is connected,\n * the deviceToken can be sent to the server.\n *\n * @param {string} dt - token obtained from the provider or false
,\n * null
or undefined
to clear the token.\n *\n * @returns true
if attempt was made to send the update to the server.\n */\n setDeviceToken(dt) {\n let sent = false;\n // Convert any falsish value to null.\n dt = dt || null;\n if (dt != this._deviceToken) {\n this._deviceToken = dt;\n if (this.isConnected() && this.isAuthenticated()) {\n this.#send({\n 'hi': {\n 'dev': dt || Tinode.DEL_CHAR\n }\n });\n sent = true;\n }\n }\n return sent;\n }\n\n /**\n * @typedef Credential\n * @type {Object}\n * @property {string} meth - validation method.\n * @property {string} val - value to validate (e.g. email or phone number).\n * @property {string} resp - validation response.\n * @property {Object} params - validation parameters.\n */\n /**\n * Authenticate current session.\n *\n * @param {string} scheme - Authentication scheme; \"basic\"
is the only currently supported scheme.\n * @param {string} secret - Authentication secret, assumed to be already base64 encoded.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n login(scheme, secret, cred) {\n const pkt = this.#initPacket('login');\n pkt.login.scheme = scheme;\n pkt.login.secret = secret;\n pkt.login.cred = cred;\n\n return this.#send(pkt, pkt.login.id)\n .then(ctrl => this.#loginSuccessful(ctrl));\n }\n\n /**\n * Wrapper for {@link Tinode#login} with basic authentication\n *\n * @param {string} uname - User name.\n * @param {string} password - Password.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n loginBasic(uname, password, cred) {\n return this.login('basic', b64EncodeUnicode(uname + ':' + password), cred)\n .then(ctrl => {\n this._login = uname;\n return ctrl;\n });\n }\n\n /**\n * Wrapper for {@link Tinode#login} with token authentication\n *\n * @param {string} token - Token received in response to earlier login.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n loginToken(token, cred) {\n return this.login('token', token, cred);\n }\n\n /**\n * Send a request for resetting an authentication secret.\n *\n * @param {string} scheme - authentication scheme to reset.\n * @param {string} method - method to use for resetting the secret, such as \"email\" or \"tel\".\n * @param {string} value - value of the credential to use, a specific email address or a phone number.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving the server reply.\n */\n requestResetAuthSecret(scheme, method, value) {\n return this.login('reset', b64EncodeUnicode(scheme + ':' + method + ':' + value));\n }\n\n /**\n * @typedef AuthToken\n * @type {Object}\n * @property {string} token - Token value.\n * @property {Date} expires - Token expiration time.\n */\n /**\n * Get stored authentication token.\n *\n * @returns {AuthToken} authentication token.\n */\n getAuthToken() {\n if (this._authToken && (this._authToken.expires.getTime() > Date.now())) {\n return this._authToken;\n } else {\n this._authToken = null;\n }\n return null;\n }\n\n /**\n * Application may provide a saved authentication token.\n *\n * @param {AuthToken} token - authentication token.\n */\n setAuthToken(token) {\n this._authToken = token;\n }\n\n /**\n * @typedef SetParams\n * @type {Object}\n * @property {SetDesc=} desc - Topic initialization parameters when creating a new topic or a new subscription.\n * @property {SetSub=} sub - Subscription initialization parameters.\n * @property {Array.=} attachments - URLs of out of band attachments used in parameters.\n */\n /**\n * @typedef SetDesc\n * @type {Object}\n * @property {DefAcs=} defacs - Default access mode.\n * @property {Object=} public - Free-form topic description, publically accessible.\n * @property {Object=} private - Free-form topic description accessible only to the owner.\n * @property {Object=} trusted - Trusted user data which can be set by a root user only.\n */\n /**\n * @typedef SetSub\n * @type {Object}\n * @property {string=} user - UID of the user affected by the request. Default (empty) - current user.\n * @property {string=} mode - User access mode, either requested or assigned dependent on context.\n */\n /**\n * Send a topic subscription request.\n *\n * @param {string} topic - Name of the topic to subscribe to.\n * @param {GetQuery=} getParams - Optional subscription metadata query\n * @param {SetParams=} setParams - Optional initialization parameters\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n subscribe(topicName, getParams, setParams) {\n const pkt = this.#initPacket('sub', topicName)\n if (!topicName) {\n topicName = Const.TOPIC_NEW;\n }\n\n pkt.sub.get = getParams;\n\n if (setParams) {\n if (setParams.sub) {\n pkt.sub.set.sub = setParams.sub;\n }\n\n if (setParams.desc) {\n const desc = setParams.desc;\n if (Tinode.isNewGroupTopicName(topicName)) {\n // Full set.desc params are used for new topics only\n pkt.sub.set.desc = desc;\n } else if (Tinode.isP2PTopicName(topicName) && desc.defacs) {\n // Use optional default permissions only.\n pkt.sub.set.desc = {\n defacs: desc.defacs\n };\n }\n }\n\n // See if external objects were used in topic description.\n if (Array.isArray(setParams.attachments) && setParams.attachments.length > 0) {\n pkt.extra = {\n attachments: setParams.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n\n if (setParams.tags) {\n pkt.sub.set.tags = setParams.tags;\n }\n }\n return this.#send(pkt, pkt.sub.id);\n }\n\n /**\n * Detach and optionally unsubscribe from the topic\n *\n * @param {string} topic - Topic to detach from.\n * @param {boolean} unsub - If true
, detach and unsubscribe, otherwise just detach.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n leave(topic, unsub) {\n const pkt = this.#initPacket('leave', topic);\n pkt.leave.unsub = unsub;\n\n return this.#send(pkt, pkt.leave.id);\n }\n\n /**\n * Create message draft without sending it to the server.\n *\n * @param {string} topic - Name of the topic to publish to.\n * @param {Object} content - Payload to publish.\n * @param {boolean=} noEcho - If true
, tell the server not to echo the message to the original session.\n *\n * @returns {Object} new message which can be sent to the server or otherwise used.\n */\n createMessage(topic, content, noEcho) {\n const pkt = this.#initPacket('pub', topic);\n\n let dft = typeof content == 'string' ? Drafty.parse(content) : content;\n if (dft && !Drafty.isPlainText(dft)) {\n pkt.pub.head = {\n mime: Drafty.getContentType()\n };\n content = dft;\n }\n pkt.pub.noecho = noEcho;\n pkt.pub.content = content;\n\n return pkt.pub;\n }\n\n /**\n * Publish {data} message to topic.\n *\n * @param {string} topicName - Name of the topic to publish to.\n * @param {Object} content - Payload to publish.\n * @param {boolean=} noEcho - If true
, tell the server not to echo the message to the original session.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n publish(topicName, content, noEcho) {\n return this.publishMessage(\n this.createMessage(topicName, content, noEcho)\n );\n }\n\n /**\n * Publish message to topic. The message should be created by {@link Tinode#createMessage}.\n *\n * @param {Object} pub - Message to publish.\n * @param {Array.=} attachments - array of URLs with attachments.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n publishMessage(pub, attachments) {\n // Make a shallow copy. Needed in order to clear locally-assigned temp values;\n pub = Object.assign({}, pub);\n pub.seq = undefined;\n pub.from = undefined;\n pub.ts = undefined;\n const msg = {\n pub: pub,\n };\n if (attachments) {\n msg.extra = {\n attachments: attachments.filter(ref => isUrlRelative(ref))\n };\n }\n return this.#send(msg, pub.id);\n }\n\n /**\n * Out of band notification: notify topic that an external (push) notification was recived by the client.\n *\n * @param {object} data - notification payload.\n * @param {string} data.what - notification type, 'msg', 'read', 'sub'.\n * @param {string} data.topic - name of the updated topic.\n * @param {number=} data.seq - seq ID of the affected message.\n * @param {string=} data.xfrom - UID of the sender.\n * @param {object=} data.given - new subscription 'given', e.g. 'ASWP...'.\n * @param {object=} data.want - new subscription 'want', e.g. 'RWJ...'.\n */\n oobNotification(data) {\n this.logger('oob: ' + (this._trimLongStrings ? JSON.stringify(data, jsonLoggerHelper) : data));\n\n switch (data.what) {\n case 'msg':\n if (!data.seq || data.seq < 1 || !data.topic) {\n // Server sent invalid data.\n break;\n }\n\n if (!this.isConnected()) {\n // Let's ignore the message if there is no connection: no connection means there are no open\n // tabs with Tinode.\n break;\n }\n\n const topic = this.#cacheGet('topic', data.topic);\n if (!topic) {\n // TODO: check if there is a case when a message can arrive from an unknown topic.\n break;\n }\n\n if (topic.isSubscribed()) {\n // No need to fetch: topic is already subscribed and got data through normal channel.\n break;\n }\n\n if (topic.maxMsgSeq() < data.seq) {\n if (topic.isChannelType()) {\n topic._updateReceived(data.seq, 'fake-uid');\n }\n\n // New message.\n if (data.xfrom && !this.#cacheGet('user', data.xfrom)) {\n // Message from unknown sender, fetch description from the server.\n // Sending asynchronously without a subscription.\n this.getMeta(data.xfrom, new MetaGetBuilder().withDesc().build()).catch(err => {\n this.logger(\"Failed to get the name of a new sender\", err);\n });\n }\n\n topic.subscribe(null).then(_ => {\n return topic.getMeta(new MetaGetBuilder(topic).withLaterData(24).withLaterDel(24).build());\n }).then(_ => {\n // Allow data fetch to complete and get processed successfully.\n topic.leaveDelayed(false, 1000);\n }).catch(err => {\n this.logger(\"On push data fetch failed\", err);\n }).finally(_ => {\n this.getMeTopic()._refreshContact('msg', topic);\n });\n }\n break;\n\n case 'read':\n this.getMeTopic()._routePres({\n what: 'read',\n seq: data.seq\n });\n break;\n\n case 'sub':\n if (!this.isMe(data.xfrom)) {\n // TODO: handle updates from other users.\n break;\n }\n\n const mode = {\n given: data.modeGiven,\n want: data.modeWant\n };\n const acs = new AccessMode(mode);\n const pres = (!acs.mode || acs.mode == AccessMode._NONE) ?\n // Subscription deleted.\n {\n what: 'gone',\n src: data.topic\n } :\n // New subscription or subscription updated.\n {\n what: 'acs',\n src: data.topic,\n dacs: mode\n };\n this.getMeTopic()._routePres(pres);\n break;\n\n default:\n this.logger(\"Unknown push type ignored\", data.what);\n }\n }\n\n /**\n * @typedef GetQuery\n * @type {Object}\n * @property {GetOptsType=} desc - If provided (even if empty), fetch topic description.\n * @property {GetOptsType=} sub - If provided (even if empty), fetch topic subscriptions.\n * @property {GetDataType=} data - If provided (even if empty), get messages.\n */\n\n /**\n * @typedef GetOptsType\n * @type {Object}\n * @property {Date=} ims - \"If modified since\", fetch data only it was was modified since stated date.\n * @property {number=} limit - Maximum number of results to return. Ignored when querying topic description.\n */\n\n /**\n * @typedef GetDataType\n * @type {Object}\n * @property {number=} since - Load messages with seq id equal or greater than this value.\n * @property {number=} before - Load messages with seq id lower than this number.\n * @property {number=} limit - Maximum number of results to return.\n */\n\n /**\n * Request topic metadata\n *\n * @param {string} topic - Name of the topic to query.\n * @param {GetQuery} params - Parameters of the query. Use {@link Tinode.MetaGetBuilder} to generate.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n getMeta(topic, params) {\n const pkt = this.#initPacket('get', topic);\n\n pkt.get = mergeObj(pkt.get, params);\n\n return this.#send(pkt, pkt.get.id);\n }\n\n /**\n * Update topic's metadata: description, subscribtions.\n *\n * @param {string} topic - Topic to update.\n * @param {SetParams} params - topic metadata to update.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n setMeta(topic, params) {\n const pkt = this.#initPacket('set', topic);\n const what = [];\n\n if (params) {\n ['desc', 'sub', 'tags', 'cred', 'ephemeral'].forEach(function(key) {\n if (params.hasOwnProperty(key)) {\n what.push(key);\n pkt.set[key] = params[key];\n }\n });\n\n if (Array.isArray(params.attachments) && params.attachments.length > 0) {\n pkt.extra = {\n attachments: params.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n }\n\n if (what.length == 0) {\n return Promise.reject(new Error(\"Invalid {set} parameters\"));\n }\n\n return this.#send(pkt, pkt.set.id);\n }\n\n /**\n * Range of message IDs to delete.\n *\n * @typedef DelRange\n * @type {Object}\n * @property {number} low - low end of the range, inclusive (closed).\n * @property {number=} hi - high end of the range, exclusive (open).\n */\n /**\n * Delete some or all messages in a topic.\n *\n * @param {string} topic - Topic name to delete messages from.\n * @param {DelRange[]} list - Ranges of message IDs to delete.\n * @param {boolean=} hard - Hard or soft delete\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delMessages(topic, ranges, hard) {\n const pkt = this.#initPacket('del', topic);\n\n pkt.del.what = 'msg';\n pkt.del.delseq = ranges;\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete the topic alltogether. Requires Owner permission.\n *\n * @param {string} topicName - Name of the topic to delete\n * @param {boolean} hard - hard-delete topic.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delTopic(topicName, hard) {\n const pkt = this.#initPacket('del', topicName);\n pkt.del.what = 'topic';\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete subscription. Requires Share permission.\n *\n * @param {string} topicName - Name of the topic to delete\n * @param {string} user - User ID to remove.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delSubscription(topicName, user) {\n const pkt = this.#initPacket('del', topicName);\n pkt.del.what = 'sub';\n pkt.del.user = user;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete credential. Always sent on 'me'
topic.\n *\n * @param {string} method - validation method such as 'email'
or 'tel'
.\n * @param {string} value - validation value, i.e. 'alice@example.com'
.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCredential(method, value) {\n const pkt = this.#initPacket('del', Const.TOPIC_ME);\n pkt.del.what = 'cred';\n pkt.del.cred = {\n meth: method,\n val: value\n };\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Request to delete account of the current user.\n *\n * @param {boolean} hard - hard-delete user.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCurrentUser(hard) {\n const pkt = this.#initPacket('del', null);\n pkt.del.what = 'user';\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id).then(_ => {\n this._myUID = null;\n });\n }\n\n /**\n * Notify server that a message or messages were read or received. Does NOT return promise.\n *\n * @param {string} topicName - Name of the topic where the mesage is being aknowledged.\n * @param {string} what - Action being aknowledged, either \"read\"
or \"recv\"
.\n * @param {number} seq - Maximum id of the message being acknowledged.\n */\n note(topicName, what, seq) {\n if (seq <= 0 || seq >= Const.LOCAL_SEQID) {\n throw new Error(`Invalid message id ${seq}`);\n }\n\n const pkt = this.#initPacket('note', topicName);\n pkt.note.what = what;\n pkt.note.seq = seq;\n this.#send(pkt);\n }\n\n /**\n * Broadcast a key-press notification to topic subscribers. Used to show\n * typing notifications \"user X is typing...\".\n *\n * @param {string} topicName - Name of the topic to broadcast to.\n * @param {string=} type - notification to send, default is 'kp'.\n */\n noteKeyPress(topicName, type) {\n const pkt = this.#initPacket('note', topicName);\n pkt.note.what = type || 'kp';\n this.#send(pkt);\n }\n\n /**\n * Send a video call notification to topic subscribers (including dialing,\n * hangup, etc.).\n *\n * @param {string} topicName - Name of the topic to broadcast to.\n * @param {int} seq - ID of the call message the event pertains to.\n * @param {string} evt - Call event.\n * @param {string} payload - Payload associated with this event (e.g. SDP string).\n *\n * @returns {Promise} Promise (for some call events) which will\n * be resolved/rejected on receiving server reply\n */\n videoCall(topicName, seq, evt, payload) {\n const pkt = this.#initPacket('note', topicName);\n pkt.note.seq = seq;\n pkt.note.what = 'call';\n pkt.note.event = evt;\n pkt.note.payload = payload;\n this.#send(pkt, pkt.note.id);\n }\n\n /**\n * Get a named topic, either pull it from cache or create a new instance.\n * There is a single instance of topic for each name.\n *\n * @param {string} topicName - Name of the topic to get.\n *\n * @returns {Topic} Requested or newly created topic or undefined
if topic name is invalid.\n */\n getTopic(topicName) {\n let topic = this.#cacheGet('topic', topicName);\n if (!topic && topicName) {\n if (topicName == Const.TOPIC_ME) {\n topic = new TopicMe();\n } else if (topicName == Const.TOPIC_FND) {\n topic = new TopicFnd();\n } else {\n topic = new Topic(topicName);\n }\n // Cache management.\n this.#attachCacheToTopic(topic);\n topic._cachePutSelf();\n // Don't save to DB here: a record will be added when the topic is subscribed.\n }\n return topic;\n }\n\n /**\n * Get a named topic from cache.\n *\n * @param {string} topicName - Name of the topic to get.\n *\n * @returns {Topic} Requested topic or undefined
if topic is not found in cache.\n */\n cacheGetTopic(topicName) {\n return this.#cacheGet('topic', topicName);\n }\n\n /**\n * Remove named topic from cache.\n *\n * @param {string} topicName - Name of the topic to remove from cache.\n */\n cacheRemTopic(topicName) {\n this.#cacheDel('topic', topicName);\n }\n\n /**\n * Iterate over cached topics.\n *\n * @param {Function} func - callback to call for each topic.\n * @param {Object} context - 'this' inside the 'func'.\n */\n mapTopics(func, context) {\n this.#cacheMap('topic', func, context);\n }\n\n /**\n * Check if named topic is already present in cache.\n *\n * @param {string} topicName - Name of the topic to check.\n * @returns {boolean} true if topic is found in cache, false otherwise.\n */\n isTopicCached(topicName) {\n return !!this.#cacheGet('topic', topicName);\n }\n\n /**\n * Generate unique name like 'new123456'
suitable for creating a new group topic.\n *\n * @param {boolean} isChan - if the topic is channel-enabled.\n * @returns {string} name which can be used for creating a new group topic.\n */\n newGroupTopicName(isChan) {\n return (isChan ? Const.TOPIC_NEW_CHAN : Const.TOPIC_NEW) + this.getNextUniqueId();\n }\n\n /**\n * Instantiate 'me'
topic or get it from cache.\n *\n * @returns {TopicMe} Instance of 'me'
topic.\n */\n getMeTopic() {\n return this.getTopic(Const.TOPIC_ME);\n }\n\n /**\n * Instantiate 'fnd'
(find) topic or get it from cache.\n *\n * @returns {Topic} Instance of 'fnd'
topic.\n */\n getFndTopic() {\n return this.getTopic(Const.TOPIC_FND);\n }\n\n /**\n * Create a new {@link LargeFileHelper} instance\n *\n * @returns {LargeFileHelper} instance of a {@link Tinode.LargeFileHelper}.\n */\n getLargeFileHelper() {\n return new LargeFileHelper(this, Const.PROTOCOL_VERSION);\n }\n\n /**\n * Get the UID of the the current authenticated user.\n *\n * @returns {string} UID of the current user or undefined
if the session is not yet\n * authenticated or if there is no session.\n */\n getCurrentUserID() {\n return this._myUID;\n }\n\n /**\n * Check if the given user ID is equal to the current user's UID.\n *\n * @param {string} uid - UID to check.\n *\n * @returns {boolean} true if the given UID belongs to the current logged in user.\n */\n isMe(uid) {\n return this._myUID === uid;\n }\n\n /**\n * Get login used for last successful authentication.\n *\n * @returns {string} login last used successfully or undefined
.\n */\n getCurrentLogin() {\n return this._login;\n }\n\n /**\n * Return information about the server: protocol version and build timestamp.\n *\n * @returns {Object} build and version of the server or null
if there is no connection or\n * if the first server response has not been received yet.\n */\n getServerInfo() {\n return this._serverInfo;\n }\n\n /**\n * Report a topic for abuse. Wrapper for {@link Tinode#publish}.\n *\n * @param {string} action - the only supported action is 'report'.\n * @param {string} target - name of the topic being reported.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n report(action, target) {\n return this.publish(Const.TOPIC_SYS, Drafty.attachJSON(null, {\n 'action': action,\n 'target': target\n }));\n }\n\n /**\n * Return server-provided configuration value.\n *\n * @param {string} name of the value to return.\n * @param {Object} defaultValue to return in case the parameter is not set or not found.\n *\n * @returns {Object} named value.\n */\n getServerParam(name, defaultValue) {\n return this._serverInfo && this._serverInfo[name] || defaultValue;\n }\n\n /**\n * Toggle console logging. Logging is off by default.\n *\n * @param {boolean} enabled - Set to true
to enable logging to console.\n * @param {boolean} trimLongStrings - Set to true
to trim long strings.\n */\n enableLogging(enabled, trimLongStrings) {\n this._loggingEnabled = enabled;\n this._trimLongStrings = enabled && trimLongStrings;\n }\n\n /**\n * Set UI language to report to the server. Must be called before 'hi'
is sent, otherwise it will not be used.\n *\n * @param {string} hl - human (UI) language, like \"en_US\"
or \"zh-Hans\"
.\n */\n setHumanLanguage(hl) {\n if (hl) {\n this._humanLanguage = hl;\n }\n }\n\n /**\n * Check if given topic is online.\n *\n * @param {string} name of the topic to test.\n * @returns {boolean} true if topic is online, false otherwise.\n */\n isTopicOnline(name) {\n const topic = this.#cacheGet('topic', name);\n return topic && topic.online;\n }\n\n /**\n * Get access mode for the given contact.\n *\n * @param {string} name of the topic to query.\n * @returns {AccessMode} access mode if topic is found, null otherwise.\n */\n getTopicAccessMode(name) {\n const topic = this.#cacheGet('topic', name);\n return topic ? topic.acs : null;\n }\n\n /**\n * Include message ID into all subsequest messages to server instructin it to send aknowledgemens.\n * Required for promises to function. Default is \"on\"
.\n *\n * @param {boolean} status - Turn aknowledgemens on or off.\n * @deprecated\n */\n wantAkn(status) {\n if (status) {\n this._messageId = Math.floor((Math.random() * 0xFFFFFF) + 0xFFFFFF);\n } else {\n this._messageId = 0;\n }\n }\n\n // Callbacks:\n /**\n * Callback to report when the websocket is opened. The callback has no parameters.\n *\n * @type {onWebsocketOpen}\n */\n onWebsocketOpen = undefined;\n\n /**\n * @typedef ServerParams\n *\n * @type {Object}\n * @property {string} ver - Server version\n * @property {string} build - Server build\n * @property {string=} sid - Session ID, long polling connections only.\n */\n\n /**\n * @callback onConnect\n * @param {number} code - Result code\n * @param {string} text - Text epxplaining the completion, i.e \"OK\" or an error message.\n * @param {ServerParams} params - Parameters returned by the server.\n */\n /**\n * Callback to report when connection with Tinode server is established.\n * @type {onConnect}\n */\n onConnect = undefined;\n\n /**\n * Callback to report when connection is lost. The callback has no parameters.\n * @type {onDisconnect}\n */\n onDisconnect = undefined;\n\n /**\n * @callback onLogin\n * @param {number} code - NUmeric completion code, same as HTTP status codes.\n * @param {string} text - Explanation of the completion code.\n */\n /**\n * Callback to report login completion.\n * @type {onLogin}\n */\n onLogin = undefined;\n\n /**\n * Callback to receive {ctrl}
(control) messages.\n * @type {onCtrlMessage}\n */\n onCtrlMessage = undefined;\n\n /**\n * Callback to recieve {data}
(content) messages.\n * @type {onDataMessage}\n */\n onDataMessage = undefined;\n\n /**\n * Callback to receive {pres}
(presence) messages.\n * @type {onPresMessage}\n */\n onPresMessage = undefined;\n\n /**\n * Callback to receive all messages as objects.\n * @type {onMessage}\n */\n onMessage = undefined;\n\n /**\n * Callback to receive all messages as unparsed text.\n * @type {onRawMessage}\n */\n onRawMessage = undefined;\n\n /**\n * Callback to receive server responses to network probes. See {@link Tinode#networkProbe}\n * @type {onNetworkProbe}\n */\n onNetworkProbe = undefined;\n\n /**\n * Callback to be notified when exponential backoff is iterating.\n * @type {onAutoreconnectIteration}\n */\n onAutoreconnectIteration = undefined;\n};\n\n// Exported constants\nTinode.MESSAGE_STATUS_NONE = Const.MESSAGE_STATUS_NONE;\nTinode.MESSAGE_STATUS_QUEUED = Const.MESSAGE_STATUS_QUEUED;\nTinode.MESSAGE_STATUS_SENDING = Const.MESSAGE_STATUS_SENDING;\nTinode.MESSAGE_STATUS_FAILED = Const.MESSAGE_STATUS_FAILED;\nTinode.MESSAGE_STATUS_FATAL = Const.MESSAGE_STATUS_FATAL;\nTinode.MESSAGE_STATUS_SENT = Const.MESSAGE_STATUS_SENT;\nTinode.MESSAGE_STATUS_RECEIVED = Const.MESSAGE_STATUS_RECEIVED;\nTinode.MESSAGE_STATUS_READ = Const.MESSAGE_STATUS_READ;\nTinode.MESSAGE_STATUS_TO_ME = Const.MESSAGE_STATUS_TO_ME;\n\n// Unicode [del] symbol.\nTinode.DEL_CHAR = Const.DEL_CHAR;\n\n// Names of keys to server-provided configuration limits.\nTinode.MAX_MESSAGE_SIZE = 'maxMessageSize';\nTinode.MAX_SUBSCRIBER_COUNT = 'maxSubscriberCount';\nTinode.MAX_TAG_COUNT = 'maxTagCount';\nTinode.MAX_FILE_UPLOAD_SIZE = 'maxFileUploadSize';\nTinode.REQ_CRED_VALIDATORS = 'reqCred';\n\n// Tinode URI topic ID prefix, 'scheme:path/'.\nTinode.URI_TOPIC_ID_PREFIX = 'tinode:topic/';\n"],"names":["AccessMode","constructor","acs","given","decode","want","mode","checkFlag","#checkFlag","val","side","flag","includes","Error","str","_BITMASK","_NONE","bitmask","_JOIN","_READ","_WRITE","_PRES","_APPROVE","_SHARE","_DELETE","_OWNER","m0","i","length","bit","charAt","toUpperCase","encode","_INVALID","res","update","upd","action","val0","parts","split","diff","a1","a2","toString","jsonHelper","setMode","m","updateMode","u","getMode","setGiven","g","updateGiven","getGiven","setWant","w","updateWant","getWant","getMissing","getExcessive","updateAll","isOwner","isPresencer","isMuted","isJoiner","isReader","isWriter","isApprover","isAdmin","isSharer","isDeleter","CBuffer","comparator","undefined","unique","buffer","compare_","unique_","a","b","findNearest","#findNearest","elem","arr","exact","start","end","pivot","found","idx","insertSorted","#insertSorted","count","splice","getAt","at","getLast","put","insert","arguments","Array","isArray","delAt","r","delRange","since","before","reset","forEach","callback","startIdx","beforeIdx","context","call","find","nearest","filter","isEmpty","CommError","message","code","name","PACKAGE_VERSION","PROTOCOL_VERSION","VERSION","LIBRARY","TOPIC_NEW","TOPIC_NEW_CHAN","TOPIC_ME","TOPIC_FND","TOPIC_SYS","TOPIC_CHAN","TOPIC_GRP","TOPIC_P2P","USER_NEW","LOCAL_SEQID","MESSAGE_STATUS_NONE","MESSAGE_STATUS_QUEUED","MESSAGE_STATUS_SENDING","MESSAGE_STATUS_FAILED","MESSAGE_STATUS_FATAL","MESSAGE_STATUS_SENT","MESSAGE_STATUS_RECEIVED","MESSAGE_STATUS_READ","MESSAGE_STATUS_TO_ME","EXPIRE_PROMISES_TIMEOUT","EXPIRE_PROMISES_PERIOD","RECV_TIMEOUT","DEFAULT_MESSAGES_PAGE","DEL_CHAR","jsonParseHelper","WebSocketProvider","XHRProvider","NETWORK_ERROR","NETWORK_ERROR_TEXT","NETWORK_USER","NETWORK_USER_TEXT","_BOFF_BASE","_BOFF_MAX_ITER","_BOFF_JITTER","makeBaseUrl","host","protocol","version","apiKey","url","Connection","log","_","boffTimer","boffIteration","boffClosed","socket","secure","autoreconnect","initialized","config","version_","autoreconnect_","transport","init_lp","init_ws","setNetworkProviders","wsProvider","xhrProvider","logger","l","connect","host_","force","Promise","reject","reconnect","disconnect","sendText","msg","isConnected","probe","backoffReset","boffReset","boffReconnect","#boffReconnect","clearTimeout","timeout","Math","pow","random","onAutoreconnectIteration","setTimeout","prom","catch","boffStop","#boffStop","#boffReset","#init_lp","XDR_UNSENT","XDR_OPENED","XDR_HEADERS_RECEIVED","XDR_LOADING","XDR_DONE","_lpURL","_poller","_sender","lp_sender","url_","sender","onreadystatechange","evt","readyState","status","open","lp_poller","resolve","poller","promiseCompleted","pkt","JSON","parse","responseText","ctrl","params","sid","send","onOpen","onMessage","onDisconnect","text","abort","err","#init_ws","OPEN","close","conn","onerror","onopen","onclose","onmessage","data","DB_VERSION","DB_NAME","IDBProvider","DB","onError","db","disabled","mapObjects","#mapObjects","source","trx","transaction","event","target","error","objectStore","getAll","onsuccess","result","topic","initDatabase","req","onupgradeneeded","createObjectStore","keyPath","deleteDatabase","onblocked","isReady","updTopic","oncomplete","get","serializeTopic","commit","markTopicAsDeleted","deleted","_deleted","remTopic","delete","IDBKeyRange","only","bound","Number","MAX_SAFE_INTEGER","mapTopics","deserializeTopic","src","updUser","uid","pub","public","remUser","mapUsers","getUser","user","updSubscription","topicName","sub","serializeSubscription","mapSubscriptions","addMessage","add","serializeMessage","updMessageStatus","seq","_status","remMessages","from","to","range","readMessages","query","limit","openCursor","cursor","value","push","continue","topic_fields","#deserializeTopic","f","hasOwnProperty","tags","_tags","setAccessMode","read","unread","max","#serializeTopic","dst","getAccessMode","#serializeSubscription","fields","#serializeMessage","setDatabaseProvider","idbProvider","MAX_FORM_ELEMENTS","MAX_PREVIEW_ATTACHMENTS","MAX_PREVIEW_DATA_SIZE","JSON_MIME_TYPE","DRAFTY_MIME_TYPE","ALLOWED_ENT_FIELDS","INLINE_STYLES","FMT_WEIGHT","ENTITY_TYPES","dataName","pack","test","re","slice","FORMAT_TAGS","AU","html_tag","md_tag","isVoid","BN","BR","CO","DL","EM","EX","FM","HD","HL","HT","IM","LN","MN","RW","QQ","ST","VC","VD","base64toObjectUrl","b64","contentType","bin","atob","buf","ArrayBuffer","Uint8Array","charCodeAt","URL","createObjectURL","Blob","type","base64toDataUrl","DECORATORS","props","href","id","act","ref","mime","Drafty","duration","size","tmpPreviewUrl","_tempPreview","previewUrl","downloadUrl","width","height","title","alt","state","preview","premime","poster","preref","txt","fmt","ent","init","plainText","content","lines","entityMap","entityIndex","blx","line","spans","entities","tag","concat","spannify","block","sort","toSpanTree","chunks","chunkify","drafty","draftify","extractEntities","ranges","entity","index","tp","offset","len","key","map","s","append","first","second","insertImage","imageDesc","ex","refurl","bits","filename","urlPromise","_processing","then","insertVideo","videoDesc","urls","insertAudio","audioDesc","videoCall","audioOnly","aonly","updateVideoCall","Object","assign","quote","header","body","appendLineBreak","mention","appendLink","linkData","appendImage","appendAudio","attachFile","attachmentDesc","wrapInto","style","wrapAsForm","insertButton","actionType","actionValue","refUrl","indexOf","appendButton","attachJSON","UNSAFE_toHTML","doc","tree","draftyToTree","htmlFormatter","values","join","treeBottomUp","format","original","formatter","shorten","light","shortenTree","lightEntity","treeToDrafty","forwardedContent","rmMention","node","parent","treeTopDown","lTrim","replyContent","convMNnQQnBR","startsWith","children","attachmentsToEnd","forwarding","toPlainText","isPlainText","toMarkdown","mdFormatter","def","isValid","txt_type","hasAttachments","attachments","hasEntities","styles","sanitizeEntities","copyEntData","getDownloadUrl","entData","isProcessing","getPreviewUrl","getEntitySize","getEntityMimeType","tagName","attrValue","getContentType","span","chunk","chld","re_start","re_end","exec","start_offset","lastIndexOf","end_offset","last","spansToTree","flatten","child","addNode","n","substring","att","subspans","inner","keymap","c","keys","newKey","transformer","stack","pop","tail","shortener","allow","lightCopy","trimStart","shift","match","extracted","el","startAt","plain","entries","dc","module","exports","isUrlRelative","addURLParam","relUrl","window","location","origin","searchParams","LargeFileHelper","tinode","_tinode","_version","_apiKey","_authToken","getAuthToken","xhr","uploadWithBaseUrl","baseUrl","avatarFor","onProgress","onSuccess","onFailure","base","endsWith","instance","setRequestHeader","token","toResolve","toReject","upload","onprogress","e","lengthComputable","loaded","total","onload","response","statusText","onabort","form","FormData","set","getNextUniqueId","_secure","_host","download","relativeUrl","mimetype","responseType","link","document","createElement","display","setAttribute","appendChild","click","removeChild","revokeObjectURL","reader","FileReader","readAsText","cancel","setNetworkProvider","MetaGetBuilder","what","get_desc_ims","#get_desc_ims","updated","get_subs_ims","#get_subs_ims","isP2PType","_lastSubsUpdate","withData","withLaterData","_maxSeq","withEarlierData","_minSeq","withDesc","ims","withLaterDesc","withSub","userOrTopic","opts","getType","withOneSub","withLaterOneSub","withLaterSub","withTags","withCred","withDel","withLaterDel","_maxDel","extract","build","getOwnPropertyNames","Const","mergeObj","mergeToCache","normalizeArray","Topic","callbacks","created","touched","Date","private","trusted","_users","_queuedSeqId","_noEarlierMsgs","_recvNotificationTimer","_credentials","_messageVersions","_messages","_attached","_new","_delayedLeaveTimer","onData","onMeta","onPres","onInfo","onMetaDesc","onMetaSub","onSubsUpdated","onTagsUpdated","onCredsUpdated","onDeleteTopic","onAllMessagesReceived","topicType","types","isMeTopicName","isGroupTopicName","isP2PTopicName","isCommTopicName","isNewGroupTopicName","isChannelTopicName","isSubscribed","subscribe","getParams","setParams","_cacheDelSelf","_cachePutSelf","ts","me","getMeTopic","desc","_noForwarding","_processMetaDesc","createMessage","noEcho","publish","publishMessage","_sending","_failed","swapMessageId","_maybeUpdateMessageVersionsCache","_routeData","publishDraft","_getQueuedSeqId","getCurrentUserID","noecho","_db","_cancelled","_fatal","leave","unsub","_resetSub","_gone","leaveDelayed","delay","getMeta","getMessagesPage","forward","startMetaQuery","_loadMessages","promise","setMeta","_processMetaSubs","_processMetaTags","cred","_processMetaCreds","subscriber","am","invite","archive","arch","delMessages","hard","r1","r2","low","hi","tosend","reduce","out","del","flushMessageRange","flushMessage","delMessagesAll","hardDel","_all","delMessagesList","list","prev","delMessagesEdits","messageVersions","delTopic","delSubscription","note","_updateMyReadRecv","_refreshContact","noteRecv","noteRead","noteKeyPress","noteRecording","payload","oldVal","doUpdate","recv","userDesc","_cacheGetUser","p2pPeerDesc","subscribers","cb","origSeq","versions","messages","sinceId","beforeId","msgs","unused1","unused2","_isReplacementMsg","latest","latestMsgVersion","_origTs","_origSeq","findMessage","latestMessage","maxMsgSeq","minMsgSeq","maxClearId","messageCount","queuedMessages","msgReceiptCount","msgReadCount","msgRecvCount","msgHasMoreMessages","newer","isNewMessage","seqId","fromId","untilId","newSeqId","numMessages","cancelSend","msgStatus","getDefaultAccess","defacs","isArchived","isMeType","isChannelType","isGroupType","isCommType","isMe","head","replace","targetSeq","parseInt","targetMsg","outgoing","webrtc","incoming","_routeInfo","_routeMeta","meta","_processDelMessages","clear","delseq","_routePres","pres","online","isTopicCached","dacs","info","subs","_updateCachedUser","creds","_allMessagesReceived","obj","cached","_cachePutUser","_updateReceived","TopicMe","onContactUpdate","turnOff","_myUID","cont","seen","when","updateCount","cacheRemTopic","getTopic","cr","findIndex","meth","done","resp","cacheGetTopic","tgt","ua","dummy","delCredential","method","contacts","getContact","getCredentials","TopicFnd","_contacts","indexBy","getPrototypeOf","prototype","date","isNaN","isValidDate","d","getTime","rfc3339DateString","pad","sp","repeat","millis","getUTCMilliseconds","getUTCFullYear","getUTCMonth","getUTCDate","getUTCHours","getUTCMinutes","getUTCSeconds","ignore","prop","cache","newval","simplify","t","trim","toLowerCase","item","pos","ary","DBCache","WebSocket","XMLHttpRequest","IndexedDBProvider","indexedDB","initForNonBrowserApp","chars","btoa","global","input","output","charCode","bc","bs","String","fromCharCode","detectTransport","b64EncodeUnicode","encodeURIComponent","toSolidBytes","p1","jsonBuildHelper","jsonLoggerHelper","getBrowserInfo","product","reactnative","priority","tmp","substr","tokens","m2","v","minor","Tinode","_appName","_browser","_platform","_hwos","_humanLanguage","_loggingEnabled","_trimLongStrings","_authenticated","_login","_inPacketCount","_messageId","floor","_serverInfo","_deviceToken","_pendingPromises","_expirePromises","_connection","_persist","_cache","onComplete","appName","platform","navigator","userAgent","language","dispatchMessage","connectionOpen","disconnected","persist","cacheGet","attachCacheToTopic","cachePut","all","dateString","_len","args","_key","console","makePromise","#makePromise","execPromise","#execPromise","onOK","errorText","#send","stringify","#dispatchMessage","onRawMessage","onNetworkProbe","onCtrlMessage","onMetaMessage","onDataMessage","onPresMessage","onInfoMessage","#connectionOpen","setInterval","expires","hello","#disconnected","clearInterval","cacheMap","getUserAgent","#getUserAgent","initPacket","#initPacket","#cachePut","#cacheGet","cacheDel","#cacheDel","#cacheMap","func","#attachCacheToTopic","_cacheDelUser","loginSuccessful","#loginSuccessful","onLogin","credential","getVersion","getLibrary","isNullValue","clearStorage","initStorage","networkProbe","isAuthenticated","authorizeURL","parsed","account","scheme","secret","login","acc","tmpscheme","tmpsecret","extra","createAccount","createAccountBasic","username","password","updateAccountBasic","onConnect","setDeviceToken","dt","sent","loginBasic","uname","loginToken","requestResetAuthSecret","now","setAuthToken","dft","oobNotification","xfrom","finally","modeGiven","modeWant","delCurrentUser","newGroupTopicName","isChan","getFndTopic","getLargeFileHelper","getCurrentLogin","getServerInfo","report","getServerParam","defaultValue","enableLogging","enabled","trimLongStrings","setHumanLanguage","hl","isTopicOnline","getTopicAccessMode","wantAkn","onWebsocketOpen","MAX_MESSAGE_SIZE","MAX_SUBSCRIBER_COUNT","MAX_TAG_COUNT","MAX_FILE_UPLOAD_SIZE","REQ_CRED_VALIDATORS","URI_TOPIC_ID_PREFIX"],"sourceRoot":""}
\ No newline at end of file
diff --git a/umd/tinode.prod.js b/umd/tinode.prod.js
index 2e7ded4..b02efc1 100644
--- a/umd/tinode.prod.js
+++ b/umd/tinode.prod.js
@@ -1,2 +1,2 @@
-!function(e,t){"object"==typeof exports&&"object"==typeof module?module.exports=t():"function"==typeof define&&define.amd?define([],t):"object"==typeof exports?exports.tinode=t():e.tinode=t()}(this,(()=>(()=>{"use strict";var e={577:e=>{const t=64,s="application/json",i=["act","height","duration","incoming","mime","name","premime","preref","preview","ref","size","state","url","val","width"],n=[{name:"ST",start:/(?:^|[\W_])(\*)[^\s*]/,end:/[^\s*](\*)(?=$|[\W_])/},{name:"EM",start:/(?:^|\W)(_)[^\s_]/,end:/[^\s_](_)(?=$|\W)/},{name:"DL",start:/(?:^|[\W_])(~)[^\s~]/,end:/[^\s~](~)(?=$|[\W_])/},{name:"CO",start:/(?:^|\W)(`)[^`]/,end:/[^`](`)(?=$|\W)/}],r=["QQ"],a=[{name:"LN",dataName:"url",pack:function(e){return/^[a-z]+:\/\//i.test(e)||(e="http://"+e),{url:e}},re:/(?:(?:https?|ftp):\/\/|www\.|ftp\.)[-A-Z0-9+&@#\/%=~_|$?!:,.]*[A-Z0-9+&@#\/%=~_|$]/gi},{name:"MN",dataName:"val",pack:function(e){return{val:e.slice(1)}},re:/\B@([\p{L}\p{N}][._\p{L}\p{N}]*[\p{L}\p{N}])/gu},{name:"HT",dataName:"val",pack:function(e){return{val:e.slice(1)}},re:/\B#([\p{L}\p{N}][._\p{L}\p{N}]*[\p{L}\p{N}])/gu}],o={AU:{html_tag:"audio",md_tag:void 0,isVoid:!1},BN:{html_tag:"button",md_tag:void 0,isVoid:!1},BR:{html_tag:"br",md_tag:"\n",isVoid:!0},CO:{html_tag:"tt",md_tag:"`",isVoid:!1},DL:{html_tag:"del",md_tag:"~",isVoid:!1},EM:{html_tag:"i",md_tag:"_",isVoid:!1},EX:{html_tag:"",md_tag:void 0,isVoid:!0},FM:{html_tag:"div",md_tag:void 0,isVoid:!1},HD:{html_tag:"",md_tag:void 0,isVoid:!1},HL:{html_tag:"span",md_tag:void 0,isVoid:!1},HT:{html_tag:"a",md_tag:void 0,isVoid:!1},IM:{html_tag:"img",md_tag:void 0,isVoid:!1},LN:{html_tag:"a",md_tag:void 0,isVoid:!1},MN:{html_tag:"a",md_tag:void 0,isVoid:!1},RW:{html_tag:"div",md_tag:void 0,isVoid:!1},QQ:{html_tag:"div",md_tag:void 0,isVoid:!1},ST:{html_tag:"b",md_tag:"*",isVoid:!1},VC:{html_tag:"div",md_tag:void 0,isVoid:!1},VD:{html_tag:"video",md_tag:void 0,isVoid:!1}};function c(e,t,s){if(!e)return null;try{const s=atob(e),i=s.length,n=new ArrayBuffer(i),r=new Uint8Array(n);for(let e=0;e"",close:e=>""},EM:{open:e=>"",close:e=>""},DL:{open:e=>"",close:e=>""},CO:{open:e=>"",close:e=>""},BR:{open:e=>"
",close:e=>""},HD:{open:e=>"",close:e=>""},HL:{open:e=>'',close:e=>""},LN:{open:e=>'',close:e=>"",props:e=>e?{href:e.url,target:"_blank"}:null},MN:{open:e=>'',close:e=>"",props:e=>e?{id:e.val}:null},HT:{open:e=>'',close:e=>"",props:e=>e?{id:e.val}:null},BN:{open:e=>"",props:e=>e?{"data-act":e.act,"data-val":e.val,"data-name":e.name,"data-ref":e.ref}:null},AU:{open:e=>'",props:e=>e?{src:e.ref||c(e.val,e.mime,l.logger),"data-preload":e.ref?"metadata":"auto","data-duration":e.duration,"data-name":e.name,"data-size":e.val?.75*e.val.length|0:0|e.size,"data-mime":e.mime}:null},IM:{open:e=>{const t=h(e._tempPreview,e.mime),s=c(e.val,e.mime,l.logger),i=e.ref||s;return(e.name?'':"")+''},close:e=>e.name?"":"",props:e=>e?{src:h(e._tempPreview,e.mime)||e.ref||c(e.val,e.mime,l.logger),title:e.name,alt:e.name,"data-width":e.width,"data-height":e.height,"data-name":e.name,"data-size":e.ref?0|e.size:e.val?.75*e.val.length|0:0|e.size,"data-mime":e.mime}:null},FM:{open:e=>"",close:e=>""},RW:{open:e=>"",close:e=>""},QQ:{open:e=>"",close:e=>"",props:e=>e?{}:null},VC:{open:e=>"",close:e=>"",props:e=>e?{"data-duration":e.duration,"data-state":e.state}:{}},VD:{open:e=>{const t=h(e._tempPreview,e.mime),s=e.ref||c(e.preview,e.premime||"image/jpeg",l.logger);return''},close:e=>"",props:e=>{if(!e)return null;const t=e.preref||c(e.preview,e.premime||"image/jpeg",l.logger);return{src:t,"data-src":e.ref||c(e.val,e.mime,l.logger),"data-width":e.width,"data-height":e.height,"data-preload":e.ref?"metadata":"auto","data-preview":t,"data-duration":0|e.duration,"data-name":e.name,"data-size":e.ref?0|e.size:e.val?.75*e.val.length|0:0|e.size,"data-mime":e.mime}}}},l=function(){this.txt="",this.fmt=[],this.ent=[]};function u(e,t,s,i){const n=[];if(0==i.length)return[];for(let s in i){const r=i[s];r.at>t&&n.push({txt:e.slice(t,r.at)});const a={tp:r.tp},o=u(e,r.at+1,r.end,r.children);o.length>0?a.children=o:a.txt=r.txt,n.push(a),t=r.end+1}return ts.end?(t.push(e[i]),s=e[i]):e[i].end<=s.end&&s.children.push(e[i]);for(let e in t)t[e].children=p(t[e].children);return t}function g(e){if(!e)return null;e="string"==typeof e?{txt:e}:e;let{txt:t,fmt:s,ent:i}=e;if(t=t||"",Array.isArray(i)||(i=[]),!Array.isArray(s)||0==s.length){if(0==i.length)return{text:t};s=[{at:0,len:0,key:0}]}const n=[],a=[];s.forEach((e=>{if(!e||"object"!=typeof e)return;if(!["undefined","number"].includes(typeof e.at))return;if(!["undefined","number"].includes(typeof e.len))return;let s=0|e.at,r=0|e.len;if(r<0)return;let o=e.key||0;i.length>0&&("number"!=typeof o||o<0||o>=i.length)||(s<=-1?a.push({start:-1,end:0,key:o}):s+r>t.length||(e.tp?n.push({type:e.tp,start:s,end:s+r}):i.length>0&&"object"==typeof i[o]&&n.push({start:s,end:s+r,key:o})))})),n.sort(((e,t)=>{let s=e.start-t.start;return 0!=s?s:(s=t.end-e.end,0!=s?s:r.indexOf(t.type)-r.indexOf(e.type))})),a.length>0&&n.push(...a),n.forEach((e=>{i.length>0&&!e.type&&i[e.key]&&"object"==typeof i[e.key]&&(e.type=i[e.key].tp,e.data=i[e.key].data),e.type||(e.type="HD")}));let o=m({},t,0,t.length,n);return o=b(o,(function(e){if(Array.isArray(e.children)&&1==e.children.length){const t=e.children[0];if(e.type)t.type||t.children||(e.text=t.text,delete e.children);else{const s=e.parent;(e=t).parent=s}}return e})),o}function f(e,t){return t?(e.children||(e.children=[]),e.text&&(e.children.push({text:e.text,parent:e}),delete e.text),t.parent=e,e.children.push(t),e):e}function m(e,t,s,i,n){if(!n||0==n.length)return s{_(e,t,s)})),t.type){const n=e.txt.length-i;if(e.fmt=e.fmt||[],Object.keys(t.data||{}).length>0){e.ent=e.ent||[];const r=void 0===s[t.key]?e.ent.length:s[t.key];s[t.key]=r,e.ent[r]={tp:t.type,data:t.data},t.att?e.fmt.push({at:-1,len:0,key:r}):e.fmt.push({at:i,len:n,key:r})}else e.fmt.push({tp:t.type,at:i,len:n})}return e}function b(e,t,s){if(!e)return null;let i=t.call(s,e);if(!i||!i.children)return i;const n=[];for(let e in i.children){let r=i.children[e];r&&(r=b(r,t,s),r&&n.push(r))}return 0==n.length?i.children=null:i.children=n,i}function w(e,t,s,i,n){if(!e)return null;i&&e.type&&i.push(e.type);let r=[];for(let s in e.children){const a=w(e.children[s],t,s,i,n);a&&r.push(a)}return 0==r.length&&(r=e.text?[e.text]:null),i&&e.type&&i.pop(),t.call(n,e.type,e.data,r,s,i)}function v(e,t,s){if(!e)return null;s&&(t-=s.length);return b(e,(function(e){if(t<=-1)return null;if(e.att)return e;if(0==t)e.text=s,t=-1;else if(e.text){const i=e.text.length;i>t?(e.text=e.text.substring(0,t)+s,t=-1):t-=i}return e}))}function y(e,t){return b(e,(e=>{const s=S(e.data,!0,t?t(e):null);return s?e.data=s:delete e.data,e}))}function M(e){if("BR"==e.type)e=null;else if(e.text)e.type||(e.text=e.text.trimStart(),e.text||(e=null));else if(!e.type&&e.children&&e.children.length>0){const t=M(e.children[0]);t?e.children[0]=t:(e.children.shift(),e.type||0!=e.children.length||(e=null))}return e}function T(e,t){if(!e)return null;if(e.att)e.text=" ",delete e.att,delete e.children;else if(e.children){const i=[],n=[];for(let r in e.children){const a=e.children[r];if(a.att){if(i.length==t)continue;if(a.data.mime==s)continue;delete a.att,delete a.children,a.text=" ",i.push(a)}else n.push(a)}e.children=n.concat(i)}return e}function P(e,t){let s="",i=[];for(let n in e){const r=e[n];if(!r.txt){const e=P(r.children,s.length+t);r.txt=e.txt,i=i.concat(e.fmt)}r.tp&&i.push({at:s.length+t,len:r.txt.length,tp:r.tp}),s+=r.txt}return{txt:s,fmt:i}}function S(e,s,n){if(e&&Object.entries(e).length>0){n=n||[];const r={};if(i.forEach((i=>{if(e[i]){if(s&&!n.includes(i)&&("string"==typeof e[i]||Array.isArray(e[i]))&&e[i].length>t)return;if("object"==typeof e[i])return;r[i]=e[i]}})),0!=Object.entries(r).length)return r}return null}l.init=function(e){if(void 0===e)e="";else if("string"!=typeof e)return null;return{txt:e}},l.parse=function(e){if("string"!=typeof e)return null;const t=e.split(/\r?\n/),s=[],i={},r=[];t.forEach((e=>{let t,o,c=[];if(n.forEach((t=>{c=c.concat(function(e,t,s,i){const n=[];let r=0,a=e.slice(0);for(;a.length>0;){const o=t.exec(a);if(null==o)break;let c=o.index+o[0].lastIndexOf(o[1]);a=a.slice(c+1),c+=r,r=c+1;const h=s?s.exec(a):null;if(null==h)break;let d=h.index+h[0].indexOf(h[1]);a=a.slice(d+1),d+=r,r=d+1,n.push({txt:e.slice(c+1,d),children:[],at:c,end:d,tp:i})}return n}(e,t.start,t.end,t.name))})),0==c.length)o={txt:e};else{c.sort(((e,t)=>{const s=e.at-t.at;return 0!=s?s:t.end-e.end})),c=p(c);const t=P(u(e,0,e.length,c),0);o={txt:t.txt,fmt:t.fmt}}if(t=function(e){let t,s=[];if(a.forEach((i=>{for(;null!==(t=i.re.exec(e));)s.push({offset:t.index,len:t[0].length,unique:t[0],data:i.pack(t[0]),type:i.name})})),0==s.length)return s;s.sort(((e,t)=>e.offset-t.offset));let i=-1;return s=s.filter((e=>{const t=e.offset>i;return i=e.offset+e.len,t})),s}(o.txt),t.length>0){const e=[];for(let n in t){const r=t[n];let a=i[r.unique];a||(a=s.length,i[r.unique]=a,s.push({tp:r.type,data:r.data})),e.push({at:r.offset,len:r.len,key:a})}o.ent=e}r.push(o)}));const o={txt:""};if(r.length>0){o.txt=r[0].txt,o.fmt=(r[0].fmt||[]).concat(r[0].ent||[]);for(let e=1;e(e.at+=s,e))))),t.ent&&(o.fmt=o.fmt.concat(t.ent.map((e=>(e.at+=s,e)))))}0==o.fmt.length&&delete o.fmt,s.length>0&&(o.ent=s)}return o},l.append=function(e,t){if(!e)return t;if(!t)return e;e.txt=e.txt||"";const s=e.txt.length;return"string"==typeof t?e.txt+=t:t.txt&&(e.txt+=t.txt),Array.isArray(t.fmt)&&(e.fmt=e.fmt||[],Array.isArray(t.ent)&&(e.ent=e.ent||[]),t.fmt.forEach((i=>{const n={at:(0|i.at)+s,len:0|i.len};-1==i.at&&(n.at=-1,n.len=0),i.tp?n.tp=i.tp:(n.key=e.ent.length,e.ent.push(t.ent[i.key||0])),e.fmt.push(n)}))),e},l.insertImage=function(e,t,s){(e=e||{txt:" "}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:0|t,len:1,key:e.ent.length});const i={tp:"IM",data:{mime:s.mime,ref:s.refurl,val:s.bits||s.preview,width:s.width,height:s.height,name:s.filename,size:0|s.size}};return s.urlPromise&&(i.data._tempPreview=s._tempPreview,i.data._processing=!0,s.urlPromise.then((e=>{i.data.ref=e,i.data._tempPreview=void 0,i.data._processing=void 0}),(e=>{i.data._processing=void 0}))),e.ent.push(i),e},l.insertVideo=function(e,t,s){(e=e||{txt:" "}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:0|t,len:1,key:e.ent.length});const i={tp:"VD",data:{mime:s.mime,ref:s.refurl,val:s.bits,preref:s.preref,preview:s.preview,width:s.width,height:s.height,duration:0|s.duration,name:s.filename,size:0|s.size}};return s.urlPromise&&(i.data._tempPreview=s._tempPreview,i.data._processing=!0,s.urlPromise.then((e=>{i.data.ref=e[0],i.data.preref=e[1],i.data._tempPreview=void 0,i.data._processing=void 0}),(e=>{i.data._processing=void 0}))),e.ent.push(i),e},l.insertAudio=function(e,t,s){(e=e||{txt:" "}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:0|t,len:1,key:e.ent.length});const i={tp:"AU",data:{mime:s.mime,val:s.bits,duration:0|s.duration,preview:s.preview,name:s.filename,size:0|s.size,ref:s.refurl}};return s.urlPromise&&(i.data._processing=!0,s.urlPromise.then((e=>{i.data.ref=e,i.data._processing=void 0}),(e=>{i.data._processing=void 0}))),e.ent.push(i),e},l.videoCall=function(e){return{txt:" ",fmt:[{at:0,len:1,key:0}],ent:[{tp:"VC",data:{aonly:e}}]}},l.updateVideoCall=function(e,t){const s=((e||{}).fmt||[])[0];if(!s)return e;let i;if("VC"==s.tp)delete s.tp,s.key=0,i={tp:"VC"},e.ent=[i];else if(i=(e.ent||[])[0|s.key],!i||"VC"!=i.tp)return e;return i.data=i.data||{},Object.assign(i.data,t),e},l.quote=function(e,t,s){const i=l.append(l.appendLineBreak(l.mention(e,t)),s);return i.fmt.push({at:0,len:i.txt.length,tp:"QQ"}),i},l.mention=function(e,t){return{txt:e||"",fmt:[{at:0,len:(e||"").length,key:0}],ent:[{tp:"MN",data:{val:t}}]}},l.appendLink=function(e,t){(e=e||{txt:""}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:e.txt.length,len:t.txt.length,key:e.ent.length}),e.txt+=t.txt;const s={tp:"LN",data:{url:t.url}};return e.ent.push(s),e},l.appendImage=function(e,t){return(e=e||{txt:""}).txt+=" ",l.insertImage(e,e.txt.length-1,t)},l.appendAudio=function(e,t){return(e=e||{txt:""}).txt+=" ",l.insertAudio(e,e.txt.length-1,t)},l.attachFile=function(e,t){(e=e||{txt:""}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:-1,len:0,key:e.ent.length});const s={tp:"EX",data:{mime:t.mime,val:t.data,name:t.filename,ref:t.refurl,size:0|t.size}};return t.urlPromise&&(s.data._processing=!0,t.urlPromise.then((e=>{s.data.ref=e,s.data._processing=void 0}),(e=>{s.data._processing=void 0}))),e.ent.push(s),e},l.wrapInto=function(e,t,s,i){return"string"==typeof e&&(e={txt:e}),e.fmt=e.fmt||[],e.fmt.push({at:s||0,len:i||e.txt.length,tp:t}),e},l.wrapAsForm=function(e,t,s){return l.wrapInto(e,"FM",t,s)},l.insertButton=function(e,t,s,i,n,r,a){return"string"==typeof e&&(e={txt:e}),!e||!e.txt||e.txt.length{switch(e.type){case"IM":return["val"];case"VD":return["preview"]}return null})),_({},s,[])},l.preview=function(e,t,s){let i=g(e);i=T(i,3);if(i=b(i,(function(e){return"MN"==e.type?e.parent&&e.parent.type||!(e.text||"").startsWith("➦")||(e.text="➦",delete e.children):"QQ"==e.type?(e.text=" ",delete e.children):"BR"==e.type&&(e.text=" ",delete e.children,delete e.type),e})),i=v(i,t,"…"),s){const e={IM:["val"],VD:["preview"]};i=y(i,(t=>e[t.type]))}else i=y(i);return _({},i,[])},l.toPlainText=function(e){return"string"==typeof e?e:e.txt},l.isPlainText=function(e){return"string"==typeof e||!(e.fmt||e.ent)},l.toMarkdown=function(e){return w(g(e),(function(e,t,s){const i=o[e];let n=s?s.join(""):"";return i&&(i.isVoid?n=i.md_tag||"":i.md_tag&&(n=i.md_tag+n+i.md_tag)),n}),0)},l.isValid=function(e){if(!e)return!1;const{txt:t,fmt:s,ent:i}=e;if(!t&&""!==t&&!s&&!i)return!1;const n=typeof t;return("string"==n||"undefined"==n||null===t)&&(!(void 0!==s&&!Array.isArray(s)&&null!==s)&&!(void 0!==i&&!Array.isArray(i)&&null!==i))},l.hasAttachments=function(e){if(!Array.isArray(e.fmt))return!1;for(let t in e.fmt){const s=e.fmt[t];if(s&&s.at<0){const t=e.ent[0|s.key];return t&&"EX"==t.tp&&t.data}}return!1},l.attachments=function(e,t,s){if(!Array.isArray(e.fmt))return;let i=0;for(let n in e.fmt){let r=e.fmt[n];if(r&&r.at<0){const n=e.ent[0|r.key];if(n&&"EX"==n.tp&&n.data&&t.call(s,n.data,i++,"EX"))break}}},l.hasEntities=function(e){return e.ent&&e.ent.length>0},l.entities=function(e,t,s){if(e.ent&&e.ent.length>0)for(let i in e.ent)if(e.ent[i]&&t.call(s,e.ent[i].data,i,e.ent[i].tp))break},l.styles=function(e,t,s){if(e.fmt&&e.fmt.length>0)for(let i in e.fmt){const n=e.fmt[i];if(n&&t.call(s,n.tp,n.at,n.len,n.key,i))break}},l.sanitizeEntities=function(e){if(e&&e.ent&&e.ent.length>0)for(let t in e.ent){const s=e.ent[t];if(s&&s.data){const i=S(s.data);i?e.ent[t].data=i:delete e.ent[t].data}}return e},l.getDownloadUrl=function(e){let t=null;return e.mime!=s&&e.val?t=c(e.val,e.mime,l.logger):"string"==typeof e.ref&&(t=e.ref),t},l.isProcessing=function(e){return!!e._processing},l.getPreviewUrl=function(e){return e.val?c(e.val,e.mime,l.logger):null},l.getEntitySize=function(e){return e.size?e.size:e.val?.75*e.val.length|0:0},l.getEntityMimeType=function(e){return e.mime||"text/plain"},l.tagName=function(e){return o[e]&&o[e].html_tag},l.attrValue=function(e,t){if(t&&d[e]&&d[e].props)return d[e].props(t)},l.getContentType=function(){return"text/x-drafty"},e.exports=l}},t={};function s(i){var n=t[i];if(void 0!==n)return n.exports;var r=t[i]={exports:{}};return e[i](r,r.exports,s),r.exports}s.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return s.d(t,{a:t}),t},s.d=(e,t)=>{for(var i in t)s.o(t,i)&&!s.o(e,i)&&Object.defineProperty(e,i,{enumerable:!0,get:t[i]})},s.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),s.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),s.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})};var i={};return(()=>{s.r(i),s.d(i,{AccessMode:()=>e,Drafty:()=>A(),Tinode:()=>W});class e{constructor(t){t&&(this.given="number"==typeof t.given?t.given:e.decode(t.given),this.want="number"==typeof t.want?t.want:e.decode(t.want),this.mode=t.mode?"number"==typeof t.mode?t.mode:e.decode(t.mode):this.given&this.want)}static#e(e,t,s){if(["given","want","mode"].includes(t=t||"mode"))return 0!=(e[t]&s);throw new Error(`Invalid AccessMode component '${t}'`)}static decode(t){if(!t)return null;if("number"==typeof t)return t&e._BITMASK;if("N"===t||"n"===t)return e._NONE;const s={J:e._JOIN,R:e._READ,W:e._WRITE,P:e._PRES,A:e._APPROVE,S:e._SHARE,D:e._DELETE,O:e._OWNER};let i=e._NONE;for(let e=0;e=20&&s.length<=24&&["ts","touched","updated","created","when","deleted","expires"].includes(t)){const e=new Date(s);if(!isNaN(e))return e}else if("acs"===t&&"object"==typeof s)return new e(s);return s}function g(e){return e&&!/^\s*([a-z][a-z0-9+.-]*:|\/\/)/im.test(e)}function f(e){return e instanceof Date&&!isNaN(e)&&0!=e.getTime()}function m(t,s,i){if("object"!=typeof s){if(void 0===s)return t;if(s===l)return;return s}if(null===s)return s;if(s instanceof Date&&!isNaN(s))return!t||!(t instanceof Date)||isNaN(t)||t{"_"==t[0]?delete e[t]:e[t]?Array.isArray(e[t])&&0==e[t].length?delete e[t]:e[t]?e[t]instanceof Date?f(e[t])||delete e[t]:"object"==typeof e[t]&&(b(e[t]),0==Object.getOwnPropertyNames(e[t]).length&&delete e[t]):delete e[t]:delete e[t]})),e}let w,v;const y="Connection failed",M=418,T="Disconnected by client";function P(e,t,s,i){let n=null;return["http","https","ws","wss"].includes(t)&&(n=`${t}://${e}`,"/"!==n.charAt(n.length-1)&&(n+="/"),n+="v"+s+"/channels",["http","https"].includes(t)&&(n+="/lp"),n+="?apikey="+i),n}class S{static#t=e=>{};#s=null;#i=0;#n=!1;#r=null;host;secure;apiKey;version;autoreconnect;initialized;constructor(e,t,s){if(this.host=e.host,this.secure=e.secure,this.apiKey=e.apiKey,this.version=t,this.autoreconnect=s,"lp"===e.transport?(this.#a(),this.initialized="lp"):"ws"===e.transport&&(this.#o(),this.initialized="ws"),!this.initialized)throw S.#t("Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'."),new Error("Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.")}static setNetworkProviders(e,t){w=e,v=t}static set logger(e){S.#t=e}connect(e,t){return Promise.reject(null)}reconnect(e){}disconnect(){}sendText(e){}isConnected(){return!1}transport(){return this.initialized}probe(){this.sendText("1")}backoffReset(){this.#c()}#h(){clearTimeout(this.#s);const e=Math.pow(2,this.#i)*(1+.3*Math.random())*2e3;this.#i=this.#i>=10?this.#i:this.#i+1,this.onAutoreconnectIteration&&this.onAutoreconnectIteration(e),this.#s=setTimeout((t=>{if(S.#t(`Reconnecting, iter=${this.#i}, timeout=${e}`),this.#n)this.onAutoreconnectIteration&&this.onAutoreconnectIteration(-1);else{const e=this.connect();this.onAutoreconnectIteration?this.onAutoreconnectIteration(0,e):e.catch((e=>{}))}}),e)}#d(){clearTimeout(this.#s),this.#s=null}#c(){this.#i=0}#a(){let e=null,t=null,s=null,i=(t,s,n)=>{let r=new v,a=!1;return r.onreadystatechange=o=>{if(4==r.readyState)if(201==r.status){let n=JSON.parse(r.responseText,p);e=t+"&sid="+n.ctrl.params.sid,r=i(e),r.send(null),this.onOpen&&this.onOpen(),s&&(a=!0,s()),this.autoreconnect&&this.#d()}else if(r.status>0&&r.status<400)this.onMessage&&this.onMessage(r.responseText),r=i(e),r.send(null);else{if(n&&!a&&(a=!0,n(r.responseText)),this.onMessage&&r.responseText&&this.onMessage(r.responseText),this.onDisconnect){const e=r.status||(this.#n?M:503),t=r.responseText||(this.#n?T:y);this.onDisconnect(new u(t,e),e)}r=null,!this.#n&&this.autoreconnect&&this.#h()}},r.open("POST",t,!0),r};this.connect=(e,s)=>{if(this.#n=!1,t){if(!s)return Promise.resolve();t.onreadystatechange=void 0,t.abort(),t=null}return e&&(this.host=e),new Promise(((e,s)=>{const n=P(this.host,this.secure?"https":"http",this.version,this.apiKey);S.#t("LP connecting to:",n),t=i(n,e,s),t.send(null)})).catch((e=>{S.#t("LP connection failed:",e)}))},this.reconnect=e=>{this.#d(),this.connect(null,e)},this.disconnect=i=>{this.#n=!0,this.#d(),s&&(s.onreadystatechange=void 0,s.abort(),s=null),t&&(t.onreadystatechange=void 0,t.abort(),t=null),this.onDisconnect&&this.onDisconnect(new u(T,M),M),e=null},this.sendText=t=>{if(s=(e=>{const t=new v;return t.onreadystatechange=e=>{if(4==t.readyState&&t.status>=400)throw new u("LP sender failed",t.status)},t.open("POST",e,!0),t})(e),!s||1!=s.readyState)throw new Error("Long poller failed to connect");s.send(t)},this.isConnected=e=>t&&!0}#o(){this.connect=(e,t)=>{if(this.#n=!1,this.#r){if(!t&&this.#r.readyState==this.#r.OPEN)return Promise.resolve();this.#r.close(),this.#r=null}return e&&(this.host=e),new Promise(((e,t)=>{const s=P(this.host,this.secure?"wss":"ws",this.version,this.apiKey);S.#t("WS connecting to: ",s);const i=new w(s);i.onerror=e=>{t(e)},i.onopen=t=>{this.autoreconnect&&this.#d(),this.onOpen&&this.onOpen(),e()},i.onclose=e=>{if(this.#r=null,this.onDisconnect){const e=this.#n?M:503;this.onDisconnect(new u(this.#n?T:y,e),e)}!this.#n&&this.autoreconnect&&this.#h()},i.onmessage=e=>{this.onMessage&&this.onMessage(e.data)},this.#r=i}))},this.reconnect=e=>{this.#d(),this.connect(null,e)},this.disconnect=e=>{this.#n=!0,this.#d(),this.#r&&(this.#r.close(),this.#r=null)},this.sendText=e=>{if(!this.#r||this.#r.readyState!=this.#r.OPEN)throw new Error("Websocket is not connected");this.#r.send(e)},this.isConnected=e=>this.#r&&this.#r.readyState==this.#r.OPEN}onMessage=void 0;onDisconnect=void 0;onOpen=void 0;onAutoreconnectIteration=void 0}S.NETWORK_ERROR=503,S.NETWORK_ERROR_TEXT=y,S.NETWORK_USER=M,S.NETWORK_USER_TEXT=T;const x="tinode-web";let k;class E{#l=e=>{};#u=e=>{};db=null;disabled=!0;constructor(e,t){this.#l=e||this.#l,this.#u=t||this.#u}#p(e,t,s){return this.db?new Promise(((i,n)=>{const r=this.db.transaction([e]);r.onerror=t=>{this.#u("PCache","mapObjects",e,t.target.error),n(t.target.error)},r.objectStore(e).getAll().onsuccess=e=>{t&&e.target.result.forEach((e=>{t.call(s,e)})),i(e.target.result)}})):disabled?Promise.resolve([]):Promise.reject(new Error("not initialized"))}initDatabase(){return new Promise(((e,t)=>{const s=k.open(x,1);s.onsuccess=t=>{this.db=t.target.result,this.disabled=!1,e(this.db)},s.onerror=e=>{this.#u("PCache","failed to initialize",e),t(e.target.error),this.#l(e.target.error)},s.onupgradeneeded=e=>{this.db=e.target.result,this.db.onerror=e=>{this.#u("PCache","failed to create storage",e),this.#l(e.target.error)},this.db.createObjectStore("topic",{keyPath:"name"}),this.db.createObjectStore("user",{keyPath:"uid"}),this.db.createObjectStore("subscription",{keyPath:["topic","uid"]}),this.db.createObjectStore("message",{keyPath:["topic","seq"]})}}))}deleteDatabase(){return this.db&&(this.db.close(),this.db=null),new Promise(((e,t)=>{const s=k.deleteDatabase(x);s.onblocked=e=>{this.db&&this.db.close();const s=new Error("blocked");this.#u("PCache","deleteDatabase",s),t(s)},s.onsuccess=t=>{this.db=null,this.disabled=!0,e(!0)},s.onerror=e=>{this.#u("PCache","deleteDatabase",e.target.error),t(e.target.error)}}))}isReady(){return!!this.db}updTopic(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["topic"],"readwrite");i.oncomplete=e=>{t(e.target.result)},i.onerror=e=>{this.#u("PCache","updTopic",e.target.error),s(e.target.error)};const n=i.objectStore("topic").get(e.name);n.onsuccess=t=>{i.objectStore("topic").put(E.#g(n.result,e)),i.commit()}})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}markTopicAsDeleted(e,t){return this.isReady()?new Promise(((s,i)=>{const n=this.db.transaction(["topic"],"readwrite");n.oncomplete=e=>{s(e.target.result)},n.onerror=e=>{this.#u("PCache","markTopicAsDeleted",e.target.error),i(e.target.error)};n.objectStore("topic").get(e).onsuccess=e=>{const s=e.target.result;s&&s._deleted!=t&&(s._deleted=t,n.objectStore("topic").put(s)),n.commit()}})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}remTopic(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["topic","subscription","message"],"readwrite");i.oncomplete=e=>{t(e.target.result)},i.onerror=e=>{this.#u("PCache","remTopic",e.target.error),s(e.target.error)},i.objectStore("topic").delete(IDBKeyRange.only(e)),i.objectStore("subscription").delete(IDBKeyRange.bound([e,"-"],[e,"~"])),i.objectStore("message").delete(IDBKeyRange.bound([e,0],[e,Number.MAX_SAFE_INTEGER])),i.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}mapTopics(e,t){return this.#p("topic",e,t)}deserializeTopic(e,t){E.#f(e,t)}updUser(e,t){if(!(arguments.length<2||void 0===t))return this.isReady()?new Promise(((s,i)=>{const n=this.db.transaction(["user"],"readwrite");n.oncomplete=e=>{s(e.target.result)},n.onerror=e=>{this.#u("PCache","updUser",e.target.error),i(e.target.error)},n.objectStore("user").put({uid:e,public:t}),n.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}remUser(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["user"],"readwrite");i.oncomplete=e=>{t(e.target.result)},i.onerror=e=>{this.#u("PCache","remUser",e.target.error),s(e.target.error)},i.objectStore("user").delete(IDBKeyRange.only(e)),i.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}mapUsers(e,t){return this.#p("user",e,t)}getUser(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["user"]);i.oncomplete=e=>{const s=e.target.result;t({user:s.uid,public:s.public})},i.onerror=e=>{this.#u("PCache","getUser",e.target.error),s(e.target.error)},i.objectStore("user").get(e)})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}updSubscription(e,t,s){return this.isReady()?new Promise(((i,n)=>{const r=this.db.transaction(["subscription"],"readwrite");r.oncomplete=e=>{i(e.target.result)},r.onerror=e=>{this.#u("PCache","updSubscription",e.target.error),n(e.target.error)},r.objectStore("subscription").get([e,t]).onsuccess=i=>{r.objectStore("subscription").put(E.#m(i.target.result,e,t,s)),r.commit()}})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}mapSubscriptions(e,t,s){return this.isReady()?new Promise(((i,n)=>{const r=this.db.transaction(["subscription"]);r.onerror=e=>{this.#u("PCache","mapSubscriptions",e.target.error),n(e.target.error)},r.objectStore("subscription").getAll(IDBKeyRange.bound([e,"-"],[e,"~"])).onsuccess=e=>{t&&e.target.result.forEach((e=>{t.call(s,e)})),i(e.target.result)}})):this.disabled?Promise.resolve([]):Promise.reject(new Error("not initialized"))}addMessage(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["message"],"readwrite");i.onsuccess=e=>{t(e.target.result)},i.onerror=e=>{this.#u("PCache","addMessage",e.target.error),s(e.target.error)},i.objectStore("message").add(E.#_(null,e)),i.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}updMessageStatus(e,t,s){return this.isReady()?new Promise(((i,n)=>{const r=this.db.transaction(["message"],"readwrite");r.onsuccess=e=>{i(e.target.result)},r.onerror=e=>{this.#u("PCache","updMessageStatus",e.target.error),n(e.target.error)};const a=r.objectStore("message").get(IDBKeyRange.only([e,t]));a.onsuccess=i=>{const n=a.result||i.target.result;n&&n._status!=s?(r.objectStore("message").put(E.#_(n,{topic:e,seq:t,_status:s})),r.commit()):r.commit()}})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}remMessages(e,t,s){return this.isReady()?new Promise(((i,n)=>{t||s||(t=0,s=Number.MAX_SAFE_INTEGER);const r=s>0?IDBKeyRange.bound([e,t],[e,s],!1,!0):IDBKeyRange.only([e,t]),a=this.db.transaction(["message"],"readwrite");a.onsuccess=e=>{i(e.target.result)},a.onerror=e=>{this.#u("PCache","remMessages",e.target.error),n(e.target.error)},a.objectStore("message").delete(r),a.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}readMessages(e,t,s,i){return this.isReady()?new Promise(((n,r)=>{const a=(t=t||{}).since>0?t.since:0,o=t.before>0?t.before:Number.MAX_SAFE_INTEGER,c=0|t.limit,h=[],d=IDBKeyRange.bound([e,a],[e,o],!1,!0),l=this.db.transaction(["message"]);l.onerror=e=>{this.#u("PCache","readMessages",e.target.error),r(e.target.error)},l.objectStore("message").openCursor(d,"prev").onsuccess=e=>{const t=e.target.result;t?(s&&s.call(i,t.value),h.push(t.value),c<=0||h.length{t.hasOwnProperty(s)&&(e[s]=t[s])})),Array.isArray(t.tags)&&(e._tags=t.tags),t.acs&&e.setAccessMode(t.acs),e.seq|=0,e.read|=0,e.unread=Math.max(0,e.seq-e.read)}static#g(e,t){const s=e||{name:t.name};return E.#b.forEach((e=>{t.hasOwnProperty(e)&&(s[e]=t[e])})),Array.isArray(t._tags)&&(s.tags=t._tags),t.acs&&(s.acs=t.getAccessMode().jsonHelper()),s}static#m(e,t,s,i){const n=e||{topic:t,uid:s};return["updated","mode","read","recv","clear","lastSeen","userAgent"].forEach((e=>{i.hasOwnProperty(e)&&(n[e]=i[e])})),n}static#_(e,t){const s=e||{};return["topic","seq","ts","_status","from","head","content"].forEach((e=>{t.hasOwnProperty(e)&&(s[e]=t[e])})),s}static setDatabaseProvider(e){k=e}}var D=s(577),A=s.n(D);let R,C,N,q;class I{constructor(e,t){this._tinode=e,this._version=t,this._apiKey=e._apiKey,this._authToken=e.getAuthToken(),this.xhr=[]}uploadWithBaseUrl(e,t,s,i,n,r){let a=`/v${this._version}/file/u/`;if(e){let t=e;if(t.endsWith("/")&&(t=t.slice(0,-1)),!t.startsWith("http://")&&!t.startsWith("https://"))throw new Error(`Invalid base URL '${e}'`);a=t+a}const o=this,c=new R;this.xhr.push(c),c.open("POST",a,!0),c.setRequestHeader("X-Tinode-APIKey",this._apiKey),this._authToken&&c.setRequestHeader("X-Tinode-Auth",`Token ${this._authToken.token}`);let h=null,d=null;const l=new Promise(((e,t)=>{h=e,d=t}));c.upload.onprogress=e=>{e.lengthComputable&&(i&&i(e.loaded/e.total),this.onProgress&&this.onProgress(e.loaded/e.total))},c.onload=function(){let e;try{e=JSON.parse(this.response,p)}catch(t){o._tinode.logger("ERROR: Invalid server response in LargeFileHelper",this.response),e={ctrl:{code:this.status,text:this.statusText}}}this.status>=200&&this.status<300?(h&&h(e.ctrl.params.url),n&&n(e.ctrl)):this.status>=400?(d&&d(new u(e.ctrl.text,e.ctrl.code)),r&&r(e.ctrl)):o._tinode.logger("ERROR: Unexpected server response status",this.status,this.response)},c.onerror=function(e){d&&d(e||new Error("failed")),r&&r(null)},c.onabort=function(e){d&&d(new Error("upload cancelled by user")),r&&r(null)};try{const e=new FormData;e.append("file",t),e.set("id",this._tinode.getNextUniqueId()),s&&e.set("topic",s),c.send(e)}catch(e){d&&d(e),r&&r(null)}return l}upload(e,t,s,i,n){const r=(this._tinode._secure?"https://":"http://")+this._tinode._host;return this.uploadWithBaseUrl(r,e,t,s,i,n)}download(e,t,s,i,n){if(!g(e))return void(n&&n(`The URL '${e}' must be relative, not absolute`));if(!this._authToken)return void(n&&n("Must authenticate first"));const r=this,a=new R;this.xhr.push(a),e=function(e,t,s){const i=new URL(e,window.location.origin);return i.searchParams.append(t,s),i.toString().substring(window.location.origin.length)}(e,"asatt","1"),a.open("GET",e,!0),a.setRequestHeader("X-Tinode-APIKey",this._apiKey),a.setRequestHeader("X-Tinode-Auth","Token "+this._authToken.token),a.responseType="blob",a.onprogress=function(e){i&&i(e.loaded)};let o=null,c=null;const h=new Promise(((e,t)=>{o=e,c=t}));a.onload=function(){if(200==this.status){const e=document.createElement("a");e.href=window.URL.createObjectURL(new Blob([this.response],{type:s})),e.style.display="none",e.setAttribute("download",t),document.body.appendChild(e),e.click(),document.body.removeChild(e),window.URL.revokeObjectURL(e.href),o&&o()}else if(this.status>=400&&c){const e=new FileReader;e.onload=function(){try{const e=JSON.parse(this.result,p);c(new u(e.ctrl.text,e.ctrl.code))}catch(e){r._tinode.logger("ERROR: Invalid server response in LargeFileHelper",this.result),c(e)}},e.readAsText(this.response)}},a.onerror=function(e){c&&c(new Error("failed")),n&&n(e)},a.onabort=function(){c&&c(null)};try{a.send()}catch(e){c&&c(e),n&&n(e)}return h}cancel(){this.xhr.forEach((e=>{e.readyState<4&&e.abort()}))}static setNetworkProvider(e){R=e}}class U{constructor(e){this.topic=e,this.what={}}#w(){return this.topic._deleted?void 0:this.topic.updated}#v(){return this.topic.isP2PType()?this.#w():this.topic._deleted?void 0:this.topic._lastSubsUpdate}withData(e,t,s){return this.what.data={since:e,before:t,limit:s},this}withLaterData(e){return this.withData(this.topic._maxSeq>0?this.topic._maxSeq+1:void 0,void 0,e)}withEarlierData(e){return this.withData(void 0,this.topic._minSeq>0?this.topic._minSeq:void 0,e)}withDesc(e){return this.what.desc={ims:e},this}withLaterDesc(){return this.withDesc(this.#w())}withSub(e,t,s){const i={ims:e,limit:t};return"me"==this.topic.getType()?i.topic=s:i.user=s,this.what.sub=i,this}withOneSub(e,t){return this.withSub(e,void 0,t)}withLaterOneSub(e){return this.withOneSub(this.topic._lastSubsUpdate,e)}withLaterSub(e){return this.withSub(this.#v(),e)}withTags(){return this.what.tags=!0,this}withCred(){return"me"==this.topic.getType()?this.what.cred=!0:this.topic._tinode.logger("ERROR: Invalid topic type for MetaGetBuilder:withCreds",this.topic.getType()),this}withDel(e,t){return(e||t)&&(this.what.del={since:e,limit:t}),this}withLaterDel(e){return this.withDel(this.topic._maxSeq>0?this.topic._maxDel+1:void 0,e)}extract(e){return this.what[e]}build(){const e=[];let t={};return["data","sub","desc","tags","cred","del"].forEach((s=>{this.what.hasOwnProperty(s)&&(e.push(s),Object.getOwnPropertyNames(this.what[s]).length>0&&(t[s]=this.what[s]))})),e.length>0?t.what=e.join(" "):t=void 0,t}}class O{#y=void 0;#M=!1;buffer=[];constructor(e,t){this.#y=e||((e,t)=>e===t?0:e0)){o=!0;break}n=r-1}return o?{idx:r,exact:!0}:s?{idx:-1}:{idx:a<0?r+1:r}}#P(e,t){const s=this.#T(e,t,!1),i=s.exact&&this.#M?1:0;return t.splice(s.idx,i,e),t}getAt(e){return this.buffer[e]}getLast(e){return e|=0,this.buffer.length>e?this.buffer[this.buffer.length-1-e]:void 0}put(){let e;e=1==arguments.length&&Array.isArray(arguments[0])?arguments[0]:arguments;for(let t in e)this.#P(e[t],this.buffer)}delAt(e){e|=0;let t=this.buffer.splice(e,1);if(t&&t.length>0)return t[0]}delRange(e,t){return this.buffer.splice(e,t-e)}length(){return this.buffer.length}reset(){this.buffer=[]}forEach(e,t,s,i){t|=0,s=s||this.buffer.length;for(let n=t;nt?this.buffer[n-1]:void 0,ne.seq-t.seq),!0),this._attached=!1,this._lastSubsUpdate=new Date(0),this._new=!0,this._deleted=!1,this._delayedLeaveTimer=null,s&&(this.onData=s.onData,this.onMeta=s.onMeta,this.onPres=s.onPres,this.onInfo=s.onInfo,this.onMetaDesc=s.onMetaDesc,this.onMetaSub=s.onMetaSub,this.onSubsUpdated=s.onSubsUpdated,this.onTagsUpdated=s.onTagsUpdated,this.onCredsUpdated=s.onCredsUpdated,this.onDeleteTopic=s.onDeleteTopic,this.onAllMessagesReceived=s.onAllMessagesReceived)}static topicType(e){return{me:o,fnd:c,grp:h,new:h,nch:h,chn:h,usr:"p2p",sys:"sys"}["string"==typeof e?e.substring(0,3):"xxx"]}static isMeTopicName(e){return j.topicType(e)==o}static isGroupTopicName(e){return j.topicType(e)==h}static isP2PTopicName(e){return"p2p"==j.topicType(e)}static isCommTopicName(e){return j.isP2PTopicName(e)||j.isGroupTopicName(e)}static isNewGroupTopicName(e){return"string"==typeof e&&(e.substring(0,3)==r||e.substring(0,3)==a)}static isChannelTopicName(e){return"string"==typeof e&&("chn"==e.substring(0,3)||e.substring(0,3)==a)}isSubscribed(){return this._attached}subscribe(e,t){return clearTimeout(this._delayedLeaveTimer),this._delayedLeaveTimer=null,this._attached?Promise.resolve(this):this._tinode.subscribe(this.name||r,e,t).then((e=>{if(e.code>=300)return e;if(this._attached=!0,this._deleted=!1,this.acs=e.params&&e.params.acs?e.params.acs:this.acs,this._new){if(delete this._new,this.name!=e.topic&&(this._cacheDelSelf(),this.name=e.topic),this._cachePutSelf(),this.created=e.ts,this.updated=e.ts,this.name!=o&&this.name!=c){const e=this._tinode.getMeTopic();e.onMetaSub&&e.onMetaSub(this),e.onSubsUpdated&&e.onSubsUpdated([this.name],1)}t&&t.desc&&(t.desc._noForwarding=!0,this._processMetaDesc(t.desc))}return e}))}createMessage(e,t){return this._tinode.createMessage(this.name,e,t)}publish(e,t){return this.publishMessage(this.createMessage(e,t))}publishMessage(e){if(!this._attached)return Promise.reject(new Error("Cannot publish on inactive topic"));if(this._sending)return Promise.reject(new Error("The message is already being sent"));e._sending=!0,e._failed=!1;let t=null;return A().hasEntities(e.content)&&(t=[],A().entities(e.content,(e=>{e&&(e.ref&&t.push(e.ref),e.preref&&t.push(e.preref))})),0==t.length&&(t=null)),this._tinode.publishMessage(e,t).then((t=>(e._sending=!1,e.ts=t.ts,this.swapMessageId(e,t.params.seq),this._maybeUpdateMessageVersionsCache(e),this._routeData(e),t))).catch((t=>{this._tinode.logger("WARNING: Message rejected by the server",t),e._sending=!1,e._failed=!0,this.onData&&this.onData()}))}publishDraft(e,t){const s=e.seq||this._getQueuedSeqId();return e._noForwarding||(e._noForwarding=!0,e.seq=s,e.ts=new Date,e.from=this._tinode.getCurrentUserID(),e.noecho=!0,this._messages.put(e),this._tinode._db.addMessage(e),this.onData&&this.onData(e)),(t||Promise.resolve()).then((t=>e._cancelled?{code:300,text:"cancelled"}:this.publishMessage(e))).catch((t=>{throw this._tinode.logger("WARNING: Message draft rejected",t),e._sending=!1,e._failed=!0,e._fatal=t instanceof u&&(t.code>=400&&t.code<500),this.onData&&this.onData(),t}))}leave(e){return this._attached||e?this._tinode.leave(this.name,e).then((t=>(this._resetSub(),e&&this._gone(),t))):Promise.reject(new Error("Cannot leave inactive topic"))}leaveDelayed(e,t){clearTimeout(this._delayedLeaveTimer),this._delayedLeaveTimer=setTimeout((t=>{this._delayedLeaveTimer=null,this.leave(e)}),t)}getMeta(e){return this._tinode.getMeta(this.name,e)}getMessagesPage(e,t){let s=t?this.startMetaQuery().withLaterData(e):this.startMetaQuery().withEarlierData(e);return this._loadMessages(this._tinode._db,s.extract("data")).then((i=>{if(i==e)return Promise.resolve({topic:this.name,code:200,params:{count:i}});e-=i,s=t?this.startMetaQuery().withLaterData(e):this.startMetaQuery().withEarlierData(e);let n=this.getMeta(s.build());return t||(n=n.then((e=>{e&&e.params&&!e.params.count&&(this._noEarlierMsgs=!0)}))),n}))}setMeta(e){return e.tags&&(e.tags=function(e){let t=[];if(Array.isArray(e)){for(let s=0,i=e.length;s1&&t.push(i))}t.sort().filter((function(e,t,s){return!t||e!=s[t-1]}))}return 0==t.length&&t.push(l),t}(e.tags)),this._tinode.setMeta(this.name,e).then((t=>(t&&t.code>=300||(e.sub&&(e.sub.topic=this.name,t.params&&t.params.acs&&(e.sub.acs=t.params.acs,e.sub.updated=t.ts),e.sub.user||(e.sub.user=this._tinode.getCurrentUserID(),e.desc||(e.desc={})),e.sub._noForwarding=!0,this._processMetaSubs([e.sub])),e.desc&&(t.params&&t.params.acs&&(e.desc.acs=t.params.acs,e.desc.updated=t.ts),this._processMetaDesc(e.desc)),e.tags&&this._processMetaTags(e.tags),e.cred&&this._processMetaCreds([e.cred],!0)),t)))}updateMode(e,t){const s=e?this.subscriber(e):null,i=s?s.acs.updateGiven(t).getGiven():this.getAccessMode().updateWant(t).getWant();return this.setMeta({sub:{user:e,mode:i}})}invite(e,t){return this.setMeta({sub:{user:e,mode:t}})}archive(e){return this.private&&!this.private.arch==!e?Promise.resolve(e):this.setMeta({desc:{private:{arch:!!e||l}}})}delMessages(e,t){if(!this._attached)return Promise.reject(new Error("Cannot delete messages in inactive topic"));e.sort(((e,t)=>e.low=t.hi)));let s,i=e.reduce(((e,t)=>(t.low0?this._tinode.delMessages(this.name,i,t):Promise.resolve({params:{del:0}}),s.then((t=>(t.params.del>this._maxDel&&(this._maxDel=t.params.del),e.forEach((e=>{e.hi?this.flushMessageRange(e.low,e.hi):this.flushMessage(e.low)})),this.onData&&this.onData(),t)))}delMessagesAll(e){return!this._maxSeq||this._maxSeq<=0?Promise.resolve():this.delMessages([{low:1,hi:this._maxSeq+1,_all:!0}],e)}delMessagesList(e,t){e.sort(((e,t)=>e-t));let s=e.reduce(((e,t)=>{if(0==e.length)e.push({low:t});else{let s=e[e.length-1];!s.hi&&t!=s.low+1||t>s.hi?e.push({low:t}):s.hi=s.hi?Math.max(s.hi,t+1):t+1}return e}),[]);return this.delMessages(s,t)}delMessagesEdits(e,t){const s=[e];return this.messageVersions(e,(e=>s.push(e.seq))),this.delMessagesList(s,t)}delTopic(e){return this._deleted?(this._gone(),Promise.resolve(null)):this._tinode.delTopic(this.name,e).then((e=>(this._deleted=!0,this._resetSub(),this._gone(),e)))}delSubscription(e){return this._attached?this._tinode.delSubscription(this.name,e).then((t=>(delete this._users[e],this.onSubsUpdated&&this.onSubsUpdated(Object.keys(this._users)),t))):Promise.reject(new Error("Cannot delete subscription in inactive topic"))}note(e,t){if(!this._attached)return;const s=this._users[this._tinode.getCurrentUserID()];let i=!1;if(s?(!s[e]||s[e]0&&this.note("read",e)}noteKeyPress(){this._attached?this._tinode.noteKeyPress(this.name):this._tinode.logger("INFO: Cannot send notification in inactive topic")}noteRecording(e){this._attached?this._tinode.noteKeyPress(this.name,e?"kpa":"kpv"):this._tinode.logger("INFO: Cannot send notification in inactive topic")}videoCall(e,t,s){if(this._attached||["ringing","hang-up"].includes(e))return this._tinode.videoCall(this.name,t,e,s)}_updateMyReadRecv(e,t,s){let i,n=!1;switch(t|=0,this.seq=0|this.seq,this.read=0|this.read,this.recv=0|this.recv,e){case"recv":i=this.recv,this.recv=Math.max(this.recv,t),n=i!=this.recv;break;case"read":i=this.read,this.read=Math.max(this.read,t),n=i!=this.read;break;case"msg":i=this.seq,this.seq=Math.max(this.seq,t),(!this.touched||this.touched{if(this._isReplacementMsg(e))return;const r=this.latestMsgVersion(e.seq)||e;r._origTs||(r._origTs=r.ts,r._origSeq=r.seq,r.ts=e.ts,r.seq=e.seq),t.push({data:r,idx:n})}),e,r,{}),t.forEach(((e,s)=>{n.call(i,e.data,s>0?t[s-1].data:void 0,s=0)return this._messages.getAt(t)}latestMessage(){return this._messages.getLast()}latestMsgVersion(e){const t=this._messageVersions[e];return t?t.getLast():null}maxMsgSeq(){return this._maxSeq}minMsgSeq(){return this._minSeq}maxClearId(){return this._maxDel}messageCount(){return this._messages.length()}queuedMessages(e,t){if(!e)throw new Error("Callback must be provided");this.messages(e,d,void 0,t)}msgReceiptCount(e,t){let s=0;if(t>0){const i=this._tinode.getCurrentUserID();for(let n in this._users){const r=this._users[n];r.user!==i&&r[e]>=t&&s++}}return s}msgReadCount(e){return this.msgReceiptCount("read",e)}msgRecvCount(e){return this.msgReceiptCount("recv",e)}msgHasMoreMessages(e){return e?this.seq>this._maxSeq:this._minSeq>1&&!this._noEarlierMsgs}isNewMessage(e){return this._maxSeq<=e}flushMessage(e){const t=this._messages.find({seq:e});if(delete this._messageVersions[e],t>=0)return this._tinode._db.remMessages(this.name,e),this._messages.delAt(t)}flushMessageRange(e,t){this._tinode._db.remMessages(this.name,e,t);for(let s=e;s=0?this._messages.delRange(s,this._messages.find({seq:t},!0)):[]}swapMessageId(e,t){const s=this._messages.find(e),i=this._messages.length();0<=s&&s=0){const s=this._messages.getAt(t),i=this.msgStatus(s);if(10==i||30==i||40==i)return this._tinode._db.remMessages(this.name,e),s._cancelled=!0,this._messages.delAt(t),this.onData&&this.onData(),!0}return!1}getType(){return j.topicType(this.name)}getAccessMode(){return this.acs}setAccessMode(t){return this.acs=new e(t)}getDefaultAccess(){return this.defacs}startMetaQuery(){return new U(this)}isArchived(){return this.private&&!!this.private.arch}isMeType(){return j.isMeTopicName(this.name)}isChannelType(){return j.isChannelTopicName(this.name)}isGroupType(){return j.isGroupTopicName(this.name)}isP2PType(){return j.isP2PTopicName(this.name)}isCommType(){return j.isCommTopicName(this.name)}msgStatus(e,t){let s=0;return this._tinode.isMe(e.from)?e._sending?s=20:e._fatal||e._cancelled?s=40:e._failed?s=30:e.seq>=d?s=10:this.msgReadCount(e.seq)>0?s=70:this.msgRecvCount(e.seq)>0?s=60:e.seq>0&&(s=50):s=80,t&&e._status!=s&&(e._status=s,this._tinode._db.updMessageStatus(this.name,e.seq,s)),s}_isReplacementMsg(e){return e.head&&e.head.replace}_maybeUpdateMessageVersionsCache(e){if(!this._isReplacementMsg(e))return void(this._messageVersions[e.seq]&&(this._messageVersions[e.seq].filter((t=>t.from==e.from)),this._messageVersions[e.seq].isEmpty()&&delete this._messageVersions[e.seq]));const t=parseInt(e.head.replace.split(":")[1]);if(t>e.seq)return;const s=this.findMessage(t);if(s&&s.from!=e.from)return;const i=this._messageVersions[t]||new O(((e,t)=>e.seq-t.seq),!0);i.put(e),this._messageVersions[t]=i}_routeData(e){e.content&&(!this.touched||this.touchedthis._maxSeq&&(this._maxSeq=e.seq,this.msgStatus(e,!0),clearTimeout(this._recvNotificationTimer),this._recvNotificationTimer=setTimeout((e=>{this._recvNotificationTimer=null,this.noteRecv(this._maxSeq)}),100)),(e.seq0&&this._processMetaSubs(e.sub),e.del&&this._processDelMessages(e.del.clear,e.del.delseq),e.tags&&this._processMetaTags(e.tags),e.cred&&this._processMetaCreds(e.cred),this.onMeta&&this.onMeta(e)}_routePres(t){let s,i;switch(t.what){case"del":this._processDelMessages(t.clear,t.delseq);break;case"on":case"off":s=this._users[t.src],s?s.online="on"==t.what:this._tinode.logger("WARNING: Presence update for an unknown user",this.name,t.src);break;case"term":this._resetSub();break;case"upd":t.src&&!this._tinode.isTopicCached(t.src)&&this.getMeta(this.startMetaQuery().withOneSub(void 0,t.src).build());break;case"acs":if(i=t.src||this._tinode.getCurrentUserID(),s=this._users[i],s)s.acs.updateAll(t.dacs),this._processMetaSubs([{user:i,updated:new Date,acs:s.acs}]);else{const n=(new e).updateAll(t.dacs);n&&n.mode!=e._NONE&&(s=this._cacheGetUser(i),s?s.acs=n:(s={user:i,acs:n},this.getMeta(this.startMetaQuery().withOneSub(void 0,i).build())),s.updated=new Date,this._processMetaSubs([s]))}break;default:this._tinode.logger("INFO: Ignored presence update",t.what)}this.onPres&&this.onPres(t)}_routeInfo(e){switch(e.what){case"recv":case"read":const t=this._users[e.from];t&&(t[e.what]=e.seq,t.recv0&&this.onData&&this.onData()}_allMessagesReceived(e){this.onAllMessagesReceived&&this.onAllMessagesReceived(e)}_resetSub(){this._attached=!1}_gone(){this._messages.reset(),this._tinode._db.remMessages(this.name),this._users={},this.acs=new e(null),this.private=null,this.public=null,this.trusted=null,this._maxSeq=0,this._minSeq=0,this._attached=!1;const t=this._tinode.getMeTopic();t&&t._routePres({_noForwarding:!0,what:"gone",topic:o,src:this.name}),this.onDeleteTopic&&this.onDeleteTopic()}_updateCachedUser(e,t){let s=this._cacheGetUser(e);return s=m(s||{},t),this._cachePutUser(e,s),_(this._users,e,s)}_getQueuedSeqId(){return this._queuedSeqId++}_loadMessages(e,t){const{since:s,before:i,limit:n}=t||{};return e.readMessages(this.name,{since:s,before:i,limit:n||24}).then((e=>(e.forEach((e=>{e.seq>this._maxSeq&&(this._maxSeq=e.seq),(e.seq{e.online&&(e.online=!1,e.seen=Object.assign(e.seen||{},{when:new Date}),this._refreshContact("off",e))})),this.onMetaDesc&&this.onMetaDesc(this)}_processMetaSubs(e){let t=0;if(e.forEach((e=>{const s=e.topic;if(s==c||s==o)return;e.online=!!e.online;let i=null;if(e.deleted)i=e,this._tinode.cacheRemTopic(s),this._tinode._db.remTopic(s);else{void 0!==e.seq&&(e.seq=0|e.seq,e.recv=0|e.recv,e.read=0|e.read,e.unread=e.seq-e.read);const t=this._tinode.getTopic(s);t._new&&delete t._new,i=m(t,e),this._tinode._db.updTopic(i),j.isP2PTopicName(s)&&(this._cachePutUser(s,i),this._tinode._db.updUser(s,i.public)),!e._noForwarding&&t&&(e._noForwarding=!0,t._processMetaDesc(e))}t++,this.onMetaSub&&this.onMetaSub(i)})),this.onSubsUpdated&&t>0){const s=[];e.forEach((e=>{s.push(e.topic)})),this.onSubsUpdated(s,t)}}_processMetaCreds(e,t){1==e.length&&e[0]==l&&(e=[]),t?e.forEach((e=>{if(e.val){let t=this._credentials.findIndex((t=>t.meth==e.meth&&t.val==e.val));t<0?(e.done||(t=this._credentials.findIndex((t=>t.meth==e.meth&&!t.done)),t>=0&&this._credentials.splice(t,1)),this._credentials.push(e)):this._credentials[t].done=e.done}else if(e.resp){const t=this._credentials.findIndex((t=>t.meth==e.meth&&!t.done));t>=0&&(this._credentials[t].done=!0)}})):this._credentials=e,this.onCredsUpdated&&this.onCredsUpdated(this._credentials)}_routePres(t){if("term"==t.what)return void this._resetSub();if("upd"==t.what&&t.src==o)return void this.getMeta(this.startMetaQuery().withDesc().build());const s=this._tinode.cacheGetTopic(t.src);if(s){switch(t.what){case"on":s.online=!0;break;case"off":s.online&&(s.online=!1,s.seen=Object.assign(s.seen||{},{when:new Date}));break;case"msg":s._updateReceived(t.seq,t.act);break;case"upd":this.getMeta(this.startMetaQuery().withLaterOneSub(t.src).build());break;case"acs":t.tgt||(s.acs?s.acs.updateAll(t.dacs):s.acs=(new e).updateAll(t.dacs),s.touched=new Date);break;case"ua":s.seen={when:new Date,ua:t.ua};break;case"recv":t.seq=0|t.seq,s.recv=s.recv?Math.max(s.recv,t.seq):t.seq;break;case"read":t.seq=0|t.seq,s.read=s.read?Math.max(s.read,t.seq):t.seq,s.recv=s.recv?Math.max(s.read,s.recv):s.recv,s.unread=s.seq-s.read;break;case"gone":this._tinode.cacheRemTopic(t.src),s._deleted?this._tinode._db.remTopic(t.src):(s._deleted=!0,s._attached=!1,this._tinode._db.markTopicAsDeleted(t.src,!0));break;case"del":break;default:this._tinode.logger("INFO: Unsupported presence update in 'me'",t.what)}this._refreshContact(t.what,s)}else{if("acs"==t.what){const s=new e(t.dacs);if(!s||s.mode==e._INVALID)return void this._tinode.logger("ERROR: Invalid access mode update",t.src,t.dacs);if(s.mode==e._NONE)return void this._tinode.logger("WARNING: Removing non-existent subscription",t.src,t.dacs);{this.getMeta(this.startMetaQuery().withOneSub(void 0,t.src).build());const e=this._tinode.getTopic(t.src);e.online=!1,e.acs=s,this._tinode._db.updTopic(e)}}else if("tags"==t.what)this.getMeta(this.startMetaQuery().withTags().build());else if("msg"==t.what){this.getMeta(this.startMetaQuery().withOneSub(void 0,t.src).build());const e=this._tinode.getTopic(t.src);e._deleted=!1,this._tinode._db.updTopic(e)}this._refreshContact(t.what,s)}this.onPres&&this.onPres(t)}_refreshContact(e,t){this.onContactUpdate&&this.onContactUpdate(e,t)}publish(){return Promise.reject(new Error("Publishing to 'me' is not supported"))}delCredential(e,t){return this._attached?this._tinode.delCredential(e,t).then((s=>{const i=this._credentials.findIndex((s=>s.meth==e&&s.val==t));return i>-1&&this._credentials.splice(i,1),this.onCredsUpdated&&this.onCredsUpdated(this._credentials),s})):Promise.reject(new Error("Cannot delete credential in inactive 'me' topic"))}contacts(e,t,s){this._tinode.mapTopics(((i,n)=>{!i.isCommType()||t&&!t(i)||e.call(s,i,n)}))}getContact(e){return this._tinode.cacheGetTopic(e)}getAccessMode(e){if(e){const t=this._tinode.cacheGetTopic(e);return t?t.acs:null}return this.acs}isArchived(e){const t=this._tinode.cacheGetTopic(e);return t&&t.private&&!!t.private.arch}getCredentials(){return this._credentials}}class V extends j{_contacts={};constructor(e){super(c,e)}_processMetaSubs(e){let t=Object.getOwnPropertyNames(this._contacts).length;this._contacts={};for(let s in e){let i=e[s];const n=i.topic?i.topic:i.user;i=_(this._contacts,n,i),t++,this.onMetaSub&&this.onMetaSub(i)}t>0&&this.onSubsUpdated&&this.onSubsUpdated(Object.keys(this._contacts))}publish(){return Promise.reject(new Error("Publishing to 'fnd' is not supported"))}setMeta(e){return Object.getPrototypeOf(V.prototype).setMeta.call(this,e).then((e=>{Object.keys(this._contacts).length>0&&(this._contacts={},this.onSubsUpdated&&this.onSubsUpdated([]))}))}contacts(e,t){const s=e||this.onMetaSub;if(s)for(let e in this._contacts)s.call(t,this._contacts[e],e,this._contacts)}}function G(e){return btoa(encodeURIComponent(e).replace(/%([0-9A-F]{2})/g,(function(e,t){return String.fromCharCode("0x"+t)})))}function z(t,s){if(s instanceof Date)s=function(e){if(!f(e))return;const t=function(e,t){return"0".repeat((t=t||2)-(""+e).length)+e},s=e.getUTCMilliseconds();return e.getUTCFullYear()+"-"+t(e.getUTCMonth()+1)+"-"+t(e.getUTCDate())+"T"+t(e.getUTCHours())+":"+t(e.getUTCMinutes())+":"+t(e.getUTCSeconds())+(s?"."+t(s,3):"")+"Z"}(s);else if(s instanceof e)s=s.jsonHelper();else if(null==s||!1===s||Array.isArray(s)&&0==s.length||"object"==typeof s&&0==Object.keys(s).length)return;return s}function F(e,t){return"string"==typeof t&&t.length>128?"<"+t.length+", bytes: "+t.substring(0,12)+"..."+t.substring(t.length-12)+">":z(0,t)}"undefined"!=typeof WebSocket&&(C=WebSocket),"undefined"!=typeof XMLHttpRequest&&(N=XMLHttpRequest),"undefined"!=typeof indexedDB&&(q=indexedDB),function(){const e="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";"undefined"==typeof btoa&&(s.g.btoa=function(){let t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:"",s="";for(let i,n=0,r=0,a=e;t.charAt(0|r)||(a="=",r%1);s+=a.charAt(63&n>>8-r%1*8)){if(i=t.charCodeAt(r+=3/4),i>255)throw new Error("'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.");n=n<<8|i}return s});"undefined"==typeof atob&&(s.g.atob=function(){let t=(arguments.length>0&&void 0!==arguments[0]?arguments[0]:"").replace(/=+$/,""),s="";if(t.length%4==1)throw new Error("'atob' failed: The string to be decoded is not correctly encoded.");for(let i,n=0,r=0,a=0;i=t.charAt(a++);~i&&(r=n%4?64*r+i:i,n++%4)?s+=String.fromCharCode(255&r>>(-2*n&6)):0)i=e.indexOf(i);return s});"undefined"==typeof window&&(s.g.window={WebSocket:C,XMLHttpRequest:N,indexedDB:q,URL:{createObjectURL:function(){throw new Error("Unable to use URL.createObjectURL in a non-browser application")}}});S.setNetworkProviders(C,N),I.setNetworkProvider(N),E.setDatabaseProvider(q)}();class W{_host;_secure;_appName;_apiKey;_browser="";_platform;_hwos="undefined";_humanLanguage="xx";_loggingEnabled=!1;_trimLongStrings=!1;_myUID=null;_authenticated=!1;_login=null;_authToken=null;_inPacketCount=0;_messageId=Math.floor(65535*Math.random()+65535);_serverInfo=null;_deviceToken=null;_pendingPromises={};_expirePromises=null;_connection=null;_persist=!1;_db=null;_cache={};constructor(e,t){if(this._host=e.host,this._secure=e.secure,this._appName=e.appName||"Undefined",this._apiKey=e.apiKey,this._platform=e.platform||"web","undefined"!=typeof navigator&&(this._browser=function(e,t){e=e||"";let s,i="";/reactnative/i.test(t)&&(i="ReactNative; ");let n=(e=e.replace(" (KHTML, like Gecko)","")).match(/(AppleWebKit\/[.\d]+)/i);if(n){const t=["edg","chrome","safari","mobile","version"];let i,r=e.substr(n.index+n[0].length).split(" "),a=[];for(let e=0;es[1].toLowerCase().startsWith(e)))]),"Version"==s[1]&&(i=s[2]))}a.sort(((e,t)=>e[2]-t[2])),a.length>0?(a[0][0].toLowerCase().startsWith("edg")?a[0][0]="Edge":"OPR"==a[0][0]?a[0][0]="Opera":"Safari"==a[0][0]&&i&&(a[0][1]=i),s=a[0][0]+"/"+a[0][1]):s=n[1]}else/firefox/i.test(e)?(n=/Firefox\/([.\d]+)/g.exec(e),s=n?"Firefox/"+n[1]:"Firefox/?"):(n=/([\w.]+)\/([.\d]+)/.exec(e),n?s=n[1]+"/"+n[2]:(n=e.split(" "),s=n[0]));if(n=s.split("/"),n.length>1){const e=n[1].split("."),t=e[1]?"."+e[1].substr(0,2):"";s=`${n[0]}/${e[0]}${t}`}return i+s}(navigator.userAgent,navigator.product),this._hwos=navigator.platform,this._humanLanguage=navigator.language||"en-US"),S.logger=this.logger,A().logger=this.logger,"lp"!=e.transport&&"ws"!=e.transport&&(e.transport=function(){if("object"==typeof window){if(window.WebSocket)return"ws";if(window.XMLHttpRequest)return"lp"}return null}()),this._connection=new S(e,"0",!0),this._connection.onMessage=e=>{this.#S(e)},this._connection.onOpen=e=>this.#x(),this._connection.onDisconnect=(e,t)=>this.#k(e,t),this._connection.onAutoreconnectIteration=(e,t)=>{this.onAutoreconnectIteration&&this.onAutoreconnectIteration(e,t)},this._persist=e.persist,this._db=new E((e=>{this.logger("DB",e)}),this.logger),this._persist){const e=[];this._db.initDatabase().then((t=>this._db.mapTopics((t=>{let s=this.#E("topic",t.name);s||(s=t.name==o?new L:t.name==c?new V:new j(t.name),this._db.deserializeTopic(s,t),this.#D(s),s._cachePutSelf(),delete s._new,e.push(s._loadMessages(this._db)))})))).then((e=>this._db.mapUsers((e=>{this.#A("user",e.uid,m({},e.public))})))).then((t=>Promise.all(e))).then((e=>{t&&t(),this.logger("Persistent cache initialized.")})).catch((e=>{t&&t(e),this.logger("Failed to initialize persistent cache:",e)}))}else this._db.deleteDatabase().then((e=>{t&&t()}))}logger(e){if(this._loggingEnabled){const n=new Date,r=("0"+n.getUTCHours()).slice(-2)+":"+("0"+n.getUTCMinutes()).slice(-2)+":"+("0"+n.getUTCSeconds()).slice(-2)+"."+("00"+n.getUTCMilliseconds()).slice(-3);for(var t=arguments.length,s=new Array(t>1?t-1:0),i=1;i{this._pendingPromises[e]={resolve:t,reject:s,ts:new Date}}))),t}#C(e,t,s,i){const n=this._pendingPromises[e];n&&(delete this._pendingPromises[e],t>=200&&t<400?n.resolve&&n.resolve(s):n.reject&&n.reject(new u(i,t)))}#N(e,t){let s;t&&(s=this.#R(t)),e=b(e);let i=JSON.stringify(e);this.logger("out: "+(this._trimLongStrings?JSON.stringify(e,F):i));try{this._connection.sendText(i)}catch(e){if(!t)throw e;this.#C(t,S.NETWORK_ERROR,null,e.message)}return s}#S(e){if(!e)return;if(this._inPacketCount++,this.onRawMessage&&this.onRawMessage(e),"0"===e)return void(this.onNetworkProbe&&this.onNetworkProbe());let t=JSON.parse(e,p);t?(this.logger("in: "+(this._trimLongStrings?JSON.stringify(t,F):e)),this.onMessage&&this.onMessage(t),t.ctrl?(this.onCtrlMessage&&this.onCtrlMessage(t.ctrl),t.ctrl.id&&this.#C(t.ctrl.id,t.ctrl.code,t.ctrl,t.ctrl.text),setTimeout((e=>{if(205==t.ctrl.code&&"evicted"==t.ctrl.text){const e=this.#E("topic",t.ctrl.topic);e&&(e._resetSub(),t.ctrl.params&&t.ctrl.params.unsub&&e._gone())}else if(t.ctrl.code<300&&t.ctrl.params)if("data"==t.ctrl.params.what){const e=this.#E("topic",t.ctrl.topic);e&&e._allMessagesReceived(t.ctrl.params.count)}else if("sub"==t.ctrl.params.what){const e=this.#E("topic",t.ctrl.topic);e&&e._processMetaSubs([])}}),0)):setTimeout((e=>{if(t.meta){const e=this.#E("topic",t.meta.topic);e&&e._routeMeta(t.meta),t.meta.id&&this.#C(t.meta.id,200,t.meta,"META"),this.onMetaMessage&&this.onMetaMessage(t.meta)}else if(t.data){const e=this.#E("topic",t.data.topic);e&&e._routeData(t.data),this.onDataMessage&&this.onDataMessage(t.data)}else if(t.pres){const e=this.#E("topic",t.pres.topic);e&&e._routePres(t.pres),this.onPresMessage&&this.onPresMessage(t.pres)}else if(t.info){const e=this.#E("topic",t.info.topic);e&&e._routeInfo(t.info),this.onInfoMessage&&this.onInfoMessage(t.info)}else this.logger("ERROR: Unknown packet received.")}),0)):(this.logger("in: "+e),this.logger("ERROR: failed to parse data"))}#x(){this._expirePromises||(this._expirePromises=setInterval((e=>{const t=new u("timeout",504),s=new Date((new Date).getTime()-5e3);for(let e in this._pendingPromises){let i=this._pendingPromises[e];i&&i.ts{e._resetSub()}));for(let t in this._pendingPromises){const s=this._pendingPromises[t];s&&s.reject&&s.reject(e)}this._pendingPromises={},this.onDisconnect&&this.onDisconnect(e)}#I(){return this._appName+" ("+(this._browser?this._browser+"; ":"")+this._hwos+"); "+n}#U(e,s){switch(e){case"hi":return{hi:{id:this.getNextUniqueId(),ver:t,ua:this.#I(),dev:this._deviceToken,lang:this._humanLanguage,platf:this._platform}};case"acc":return{acc:{id:this.getNextUniqueId(),user:null,scheme:null,secret:null,tmpscheme:null,tmpsecret:null,login:!1,tags:null,desc:{},cred:{}}};case"login":return{login:{id:this.getNextUniqueId(),scheme:null,secret:null}};case"sub":return{sub:{id:this.getNextUniqueId(),topic:s,set:{},get:{}}};case"leave":return{leave:{id:this.getNextUniqueId(),topic:s,unsub:!1}};case"pub":return{pub:{id:this.getNextUniqueId(),topic:s,noecho:!1,head:null,content:{}}};case"get":return{get:{id:this.getNextUniqueId(),topic:s,what:null,desc:{},sub:{},data:{}}};case"set":return{set:{id:this.getNextUniqueId(),topic:s,desc:{},sub:{},tags:[],ephemeral:{}}};case"del":return{del:{id:this.getNextUniqueId(),topic:s,what:null,delseq:null,user:null,hard:!1}};case"note":return{note:{topic:s,what:null,seq:void 0}};default:throw new Error(`Unknown packet type requested: ${e}`)}}#A(e,t,s){this._cache[e+":"+t]=s}#E(e,t){return this._cache[e+":"+t]}#O(e,t){delete this._cache[e+":"+t]}#q(e,t,s){const i=e?e+":":void 0;for(let e in this._cache)if((!i||0==e.indexOf(i))&&t.call(s,this._cache[e],e))break}#D(e){e._tinode=this,e._cacheGetUser=e=>{const t=this.#E("user",e);if(t)return{user:e,public:m({},t)}},e._cachePutUser=(e,t)=>{this.#A("user",e,m({},t.public))},e._cacheDelUser=e=>{this.#O("user",e)},e._cachePutSelf=t=>{this.#A("topic",e.name,e)},e._cacheDelSelf=t=>{this.#O("topic",e.name)}}#j(e){return e.params&&e.params.user?(this._myUID=e.params.user,this._authenticated=e&&e.code>=200&&e.code<300,e.params&&e.params.token&&e.params.expires?this._authToken={token:e.params.token,expires:e.params.expires}:this._authToken=null,this.onLogin&&this.onLogin(e.code,e.text),e):e}static credential(e,t,s,i){return"object"==typeof e&&({val:t,params:s,resp:i,meth:e}=e),e&&(t||i)?[{meth:e,val:t,resp:i,params:s}]:null}static topicType(e){return j.topicType(e)}static isMeTopicName(e){return j.isMeTopicName(e)}static isGroupTopicName(e){return j.isGroupTopicName(e)}static isP2PTopicName(e){return j.isP2PTopicName(e)}static isCommTopicName(e){return j.isCommTopicName(e)}static isNewGroupTopicName(e){return j.isNewGroupTopicName(e)}static isChannelTopicName(e){return j.isChannelTopicName(e)}static getVersion(){return t}static setNetworkProviders(e,t){C=e,N=t,S.setNetworkProviders(C,N),I.setNetworkProvider(N)}static setDatabaseProvider(e){q=e,E.setDatabaseProvider(q)}static getLibrary(){return n}static isNullValue(e){return e===l}getNextUniqueId(){return 0!=this._messageId?""+this._messageId++:void 0}connect(e){return this._connection.connect(e)}reconnect(e){this._connection.reconnect(e)}disconnect(){this._connection.disconnect()}clearStorage(){return this._db.isReady()?this._db.deleteDatabase():Promise.resolve()}initStorage(){return this._db.isReady()?Promise.resolve():this._db.initDatabase()}networkProbe(){this._connection.probe()}isConnected(){return this._connection.isConnected()}isAuthenticated(){return this._authenticated}authorizeURL(e){if("string"!=typeof e)return e;if(g(e)){const t="scheme://host/",s=new URL(e,t);this._apiKey&&s.searchParams.append("apikey",this._apiKey),this._authToken&&this._authToken.token&&(s.searchParams.append("auth","token"),s.searchParams.append("secret",this._authToken.token)),e=s.toString().substring(t.length-1)}return e}account(e,t,s,i,n){const r=this.#U("acc");return r.acc.user=e,r.acc.scheme=t,r.acc.secret=s,r.acc.login=i,n&&(r.acc.desc.defacs=n.defacs,r.acc.desc.public=n.public,r.acc.desc.private=n.private,r.acc.desc.trusted=n.trusted,r.acc.tags=n.tags,r.acc.cred=n.cred,r.acc.tmpscheme=n.scheme,r.acc.tmpsecret=n.secret,Array.isArray(n.attachments)&&n.attachments.length>0&&(r.extra={attachments:n.attachments.filter((e=>g(e)))})),this.#N(r,r.acc.id)}createAccount(e,t,s,i){let n=this.account("new",e,t,s,i);return s&&(n=n.then((e=>this.#j(e)))),n}createAccountBasic(e,t,s){return e=e||"",t=t||"",this.createAccount("basic",G(e+":"+t),!0,s)}updateAccountBasic(e,t,s,i){return t=t||"",s=s||"",this.account(e,"basic",G(t+":"+s),!1,i)}hello(){const e=this.#U("hi");return this.#N(e,e.hi.id).then((e=>(this._connection.backoffReset(),e.params&&(this._serverInfo=e.params),this.onConnect&&this.onConnect(),e))).catch((e=>{this._connection.reconnect(!0),this.onDisconnect&&this.onDisconnect(e)}))}setDeviceToken(e){let t=!1;return(e=e||null)!=this._deviceToken&&(this._deviceToken=e,this.isConnected()&&this.isAuthenticated()&&(this.#N({hi:{dev:e||W.DEL_CHAR}}),t=!0)),t}login(e,t,s){const i=this.#U("login");return i.login.scheme=e,i.login.secret=t,i.login.cred=s,this.#N(i,i.login.id).then((e=>this.#j(e)))}loginBasic(e,t,s){return this.login("basic",G(e+":"+t),s).then((t=>(this._login=e,t)))}loginToken(e,t){return this.login("token",e,t)}requestResetAuthSecret(e,t,s){return this.login("reset",G(e+":"+t+":"+s))}getAuthToken(){return this._authToken&&this._authToken.expires.getTime()>Date.now()?this._authToken:(this._authToken=null,null)}setAuthToken(e){this._authToken=e}subscribe(e,t,s){const i=this.#U("sub",e);if(e||(e=r),i.sub.get=t,s){if(s.sub&&(i.sub.set.sub=s.sub),s.desc){const t=s.desc;W.isNewGroupTopicName(e)?i.sub.set.desc=t:W.isP2PTopicName(e)&&t.defacs&&(i.sub.set.desc={defacs:t.defacs})}Array.isArray(s.attachments)&&s.attachments.length>0&&(i.extra={attachments:s.attachments.filter((e=>g(e)))}),s.tags&&(i.sub.set.tags=s.tags)}return this.#N(i,i.sub.id)}leave(e,t){const s=this.#U("leave",e);return s.leave.unsub=t,this.#N(s,s.leave.id)}createMessage(e,t,s){const i=this.#U("pub",e);let n="string"==typeof t?A().parse(t):t;return n&&!A().isPlainText(n)&&(i.pub.head={mime:A().getContentType()},t=n),i.pub.noecho=s,i.pub.content=t,i.pub}publish(e,t,s){return this.publishMessage(this.createMessage(e,t,s))}publishMessage(e,t){(e=Object.assign({},e)).seq=void 0,e.from=void 0,e.ts=void 0;const s={pub:e};return t&&(s.extra={attachments:t.filter((e=>g(e)))}),this.#N(s,e.id)}oobNotification(t){switch(this.logger("oob: "+(this._trimLongStrings?JSON.stringify(t,F):t)),t.what){case"msg":if(!t.seq||t.seq<1||!t.topic)break;if(!this.isConnected())break;const s=this.#E("topic",t.topic);if(!s)break;if(s.isSubscribed())break;s.maxMsgSeq(){this.logger("Failed to get the name of a new sender",e)})),s.subscribe(null).then((e=>s.getMeta(new U(s).withLaterData(24).withLaterDel(24).build()))).then((e=>{s.leaveDelayed(!1,1e3)})).catch((e=>{this.logger("On push data fetch failed",e)})).finally((e=>{this.getMeTopic()._refreshContact("msg",s)})));break;case"read":this.getMeTopic()._routePres({what:"read",seq:t.seq});break;case"sub":if(!this.isMe(t.xfrom))break;const i={given:t.modeGiven,want:t.modeWant},n=new e(i),r=n.mode&&n.mode!=e._NONE?{what:"acs",src:t.topic,dacs:i}:{what:"gone",src:t.topic};this.getMeTopic()._routePres(r);break;default:this.logger("Unknown push type ignored",t.what)}}getMeta(e,t){const s=this.#U("get",e);return s.get=m(s.get,t),this.#N(s,s.get.id)}setMeta(e,t){const s=this.#U("set",e),i=[];return t&&(["desc","sub","tags","cred","ephemeral"].forEach((function(e){t.hasOwnProperty(e)&&(i.push(e),s.set[e]=t[e])})),Array.isArray(t.attachments)&&t.attachments.length>0&&(s.extra={attachments:t.attachments.filter((e=>g(e)))})),0==i.length?Promise.reject(new Error("Invalid {set} parameters")):this.#N(s,s.set.id)}delMessages(e,t,s){const i=this.#U("del",e);return i.del.what="msg",i.del.delseq=t,i.del.hard=s,this.#N(i,i.del.id)}delTopic(e,t){const s=this.#U("del",e);return s.del.what="topic",s.del.hard=t,this.#N(s,s.del.id)}delSubscription(e,t){const s=this.#U("del",e);return s.del.what="sub",s.del.user=t,this.#N(s,s.del.id)}delCredential(e,t){const s=this.#U("del",o);return s.del.what="cred",s.del.cred={meth:e,val:t},this.#N(s,s.del.id)}delCurrentUser(e){const t=this.#U("del",null);return t.del.what="user",t.del.hard=e,this.#N(t,t.del.id).then((e=>{this._myUID=null}))}note(e,t,s){if(s<=0||s>=d)throw new Error(`Invalid message id ${s}`);const i=this.#U("note",e);i.note.what=t,i.note.seq=s,this.#N(i)}noteKeyPress(e,t){const s=this.#U("note",e);s.note.what=t||"kp",this.#N(s)}videoCall(e,t,s,i){const n=this.#U("note",e);n.note.seq=t,n.note.what="call",n.note.event=s,n.note.payload=i,this.#N(n,n.note.id)}getTopic(e){let t=this.#E("topic",e);return!t&&e&&(t=e==o?new L:e==c?new V:new j(e),this.#D(t),t._cachePutSelf()),t}cacheGetTopic(e){return this.#E("topic",e)}cacheRemTopic(e){this.#O("topic",e)}mapTopics(e,t){this.#q("topic",e,t)}isTopicCached(e){return!!this.#E("topic",e)}newGroupTopicName(e){return(e?a:r)+this.getNextUniqueId()}getMeTopic(){return this.getTopic(o)}getFndTopic(){return this.getTopic(c)}getLargeFileHelper(){return new I(this,"0")}getCurrentUserID(){return this._myUID}isMe(e){return this._myUID===e}getCurrentLogin(){return this._login}getServerInfo(){return this._serverInfo}report(e,t){return this.publish("sys",A().attachJSON(null,{action:e,target:t}))}getServerParam(e,t){return this._serverInfo&&this._serverInfo[e]||t}enableLogging(e,t){this._loggingEnabled=e,this._trimLongStrings=e&&t}setHumanLanguage(e){e&&(this._humanLanguage=e)}isTopicOnline(e){const t=this.#E("topic",e);return t&&t.online}getTopicAccessMode(e){const t=this.#E("topic",e);return t?t.acs:null}wantAkn(e){this._messageId=e?Math.floor(16777215*Math.random()+16777215):0}onWebsocketOpen=void 0;onConnect=void 0;onDisconnect=void 0;onLogin=void 0;onCtrlMessage=void 0;onDataMessage=void 0;onPresMessage=void 0;onMessage=void 0;onRawMessage=void 0;onNetworkProbe=void 0;onAutoreconnectIteration=void 0}W.MESSAGE_STATUS_NONE=0,W.MESSAGE_STATUS_QUEUED=10,W.MESSAGE_STATUS_SENDING=20,W.MESSAGE_STATUS_FAILED=30,W.MESSAGE_STATUS_FATAL=40,W.MESSAGE_STATUS_SENT=50,W.MESSAGE_STATUS_RECEIVED=60,W.MESSAGE_STATUS_READ=70,W.MESSAGE_STATUS_TO_ME=80,W.DEL_CHAR=l,W.MAX_MESSAGE_SIZE="maxMessageSize",W.MAX_SUBSCRIBER_COUNT="maxSubscriberCount",W.MAX_TAG_COUNT="maxTagCount",W.MAX_FILE_UPLOAD_SIZE="maxFileUploadSize",W.REQ_CRED_VALIDATORS="reqCred",W.URI_TOPIC_ID_PREFIX="tinode:topic/"})(),i})()));
+!function(e,t){"object"==typeof exports&&"object"==typeof module?module.exports=t():"function"==typeof define&&define.amd?define([],t):"object"==typeof exports?exports.tinode=t():e.tinode=t()}(this,(()=>(()=>{"use strict";var e={672:e=>{const t=64,s="application/json",i=["act","height","duration","incoming","mime","name","premime","preref","preview","ref","size","state","url","val","width"],n=[{name:"ST",start:/(?:^|[\W_])(\*)[^\s*]/,end:/[^\s*](\*)(?=$|[\W_])/},{name:"EM",start:/(?:^|\W)(_)[^\s_]/,end:/[^\s_](_)(?=$|\W)/},{name:"DL",start:/(?:^|[\W_])(~)[^\s~]/,end:/[^\s~](~)(?=$|[\W_])/},{name:"CO",start:/(?:^|\W)(`)[^`]/,end:/[^`](`)(?=$|\W)/}],r=["QQ"],a=[{name:"LN",dataName:"url",pack:function(e){return/^[a-z]+:\/\//i.test(e)||(e="http://"+e),{url:e}},re:/(?:(?:https?|ftp):\/\/|www\.|ftp\.)[-A-Z0-9+&@#\/%=~_|$?!:,.]*[A-Z0-9+&@#\/%=~_|$]/gi},{name:"MN",dataName:"val",pack:function(e){return{val:e.slice(1)}},re:/\B@([\p{L}\p{N}][._\p{L}\p{N}]*[\p{L}\p{N}])/gu},{name:"HT",dataName:"val",pack:function(e){return{val:e.slice(1)}},re:/\B#([\p{L}\p{N}][._\p{L}\p{N}]*[\p{L}\p{N}])/gu}],o={AU:{html_tag:"audio",md_tag:void 0,isVoid:!1},BN:{html_tag:"button",md_tag:void 0,isVoid:!1},BR:{html_tag:"br",md_tag:"\n",isVoid:!0},CO:{html_tag:"tt",md_tag:"`",isVoid:!1},DL:{html_tag:"del",md_tag:"~",isVoid:!1},EM:{html_tag:"i",md_tag:"_",isVoid:!1},EX:{html_tag:"",md_tag:void 0,isVoid:!0},FM:{html_tag:"div",md_tag:void 0,isVoid:!1},HD:{html_tag:"",md_tag:void 0,isVoid:!1},HL:{html_tag:"span",md_tag:void 0,isVoid:!1},HT:{html_tag:"a",md_tag:void 0,isVoid:!1},IM:{html_tag:"img",md_tag:void 0,isVoid:!1},LN:{html_tag:"a",md_tag:void 0,isVoid:!1},MN:{html_tag:"a",md_tag:void 0,isVoid:!1},RW:{html_tag:"div",md_tag:void 0,isVoid:!1},QQ:{html_tag:"div",md_tag:void 0,isVoid:!1},ST:{html_tag:"b",md_tag:"*",isVoid:!1},VC:{html_tag:"div",md_tag:void 0,isVoid:!1},VD:{html_tag:"video",md_tag:void 0,isVoid:!1}};function c(e,t,s){if(!e)return null;try{const s=atob(e),i=s.length,n=new ArrayBuffer(i),r=new Uint8Array(n);for(let e=0;e"",close:e=>""},EM:{open:e=>"",close:e=>""},DL:{open:e=>"",close:e=>""},CO:{open:e=>"",close:e=>""},BR:{open:e=>"
",close:e=>""},HD:{open:e=>"",close:e=>""},HL:{open:e=>'',close:e=>""},LN:{open:e=>'',close:e=>"",props:e=>e?{href:e.url,target:"_blank"}:null},MN:{open:e=>'',close:e=>"",props:e=>e?{id:e.val}:null},HT:{open:e=>'',close:e=>"",props:e=>e?{id:e.val}:null},BN:{open:e=>"",props:e=>e?{"data-act":e.act,"data-val":e.val,"data-name":e.name,"data-ref":e.ref}:null},AU:{open:e=>'",props:e=>e?{src:e.ref||c(e.val,e.mime,l.logger),"data-preload":e.ref?"metadata":"auto","data-duration":e.duration,"data-name":e.name,"data-size":e.val?.75*e.val.length|0:0|e.size,"data-mime":e.mime}:null},IM:{open:e=>{const t=h(e._tempPreview,e.mime),s=c(e.val,e.mime,l.logger),i=e.ref||s;return(e.name?'':"")+''},close:e=>e.name?"":"",props:e=>e?{src:h(e._tempPreview,e.mime)||e.ref||c(e.val,e.mime,l.logger),title:e.name,alt:e.name,"data-width":e.width,"data-height":e.height,"data-name":e.name,"data-size":e.ref?0|e.size:e.val?.75*e.val.length|0:0|e.size,"data-mime":e.mime}:null},FM:{open:e=>"",close:e=>""},RW:{open:e=>"",close:e=>""},QQ:{open:e=>"",close:e=>"",props:e=>e?{}:null},VC:{open:e=>"",close:e=>"",props:e=>e?{"data-duration":e.duration,"data-state":e.state}:{}},VD:{open:e=>{const t=h(e._tempPreview,e.mime),s=e.ref||c(e.preview,e.premime||"image/jpeg",l.logger);return''},close:e=>"",props:e=>{if(!e)return null;const t=e.preref||c(e.preview,e.premime||"image/jpeg",l.logger);return{src:t,"data-src":e.ref||c(e.val,e.mime,l.logger),"data-width":e.width,"data-height":e.height,"data-preload":e.ref?"metadata":"auto","data-preview":t,"data-duration":0|e.duration,"data-name":e.name,"data-size":e.ref?0|e.size:e.val?.75*e.val.length|0:0|e.size,"data-mime":e.mime}}}},l=function(){this.txt="",this.fmt=[],this.ent=[]};function u(e,t,s,i){const n=[];if(0==i.length)return[];for(let s in i){const r=i[s];r.at>t&&n.push({txt:e.slice(t,r.at)});const a={tp:r.tp},o=u(e,r.at+1,r.end,r.children);o.length>0?a.children=o:a.txt=r.txt,n.push(a),t=r.end+1}return ts.end?(t.push(e[i]),s=e[i]):e[i].end<=s.end&&s.children.push(e[i]);for(let e in t)t[e].children=p(t[e].children);return t}function g(e){if(!e)return null;e="string"==typeof e?{txt:e}:e;let{txt:t,fmt:s,ent:i}=e;if(t=t||"",Array.isArray(i)||(i=[]),!Array.isArray(s)||0==s.length){if(0==i.length)return{text:t};s=[{at:0,len:0,key:0}]}const n=[],a=[];s.forEach((e=>{if(!e||"object"!=typeof e)return;if(!["undefined","number"].includes(typeof e.at))return;if(!["undefined","number"].includes(typeof e.len))return;let s=0|e.at,r=0|e.len;if(r<0)return;let o=e.key||0;i.length>0&&("number"!=typeof o||o<0||o>=i.length)||(s<=-1?a.push({start:-1,end:0,key:o}):s+r>t.length||(e.tp?n.push({type:e.tp,start:s,end:s+r}):i.length>0&&"object"==typeof i[o]&&n.push({start:s,end:s+r,key:o})))})),n.sort(((e,t)=>{let s=e.start-t.start;return 0!=s?s:(s=t.end-e.end,0!=s?s:r.indexOf(t.type)-r.indexOf(e.type))})),a.length>0&&n.push(...a),n.forEach((e=>{i.length>0&&!e.type&&i[e.key]&&"object"==typeof i[e.key]&&(e.type=i[e.key].tp,e.data=i[e.key].data),e.type||(e.type="HD")}));let o=m({},t,0,t.length,n);return o=b(o,(function(e){if(Array.isArray(e.children)&&1==e.children.length){const t=e.children[0];if(e.type)t.type||t.children||(e.text=t.text,delete e.children);else{const s=e.parent;(e=t).parent=s}}return e})),o}function f(e,t){return t?(e.children||(e.children=[]),e.text&&(e.children.push({text:e.text,parent:e}),delete e.text),t.parent=e,e.children.push(t),e):e}function m(e,t,s,i,n){if(!n||0==n.length)return s{_(e,t,s)})),t.type){const n=e.txt.length-i;if(e.fmt=e.fmt||[],Object.keys(t.data||{}).length>0){e.ent=e.ent||[];const r=void 0===s[t.key]?e.ent.length:s[t.key];s[t.key]=r,e.ent[r]={tp:t.type,data:t.data},t.att?e.fmt.push({at:-1,len:0,key:r}):e.fmt.push({at:i,len:n,key:r})}else e.fmt.push({tp:t.type,at:i,len:n})}return e}function b(e,t,s){if(!e)return null;let i=t.call(s,e);if(!i||!i.children)return i;const n=[];for(let e in i.children){let r=i.children[e];r&&(r=b(r,t,s),r&&n.push(r))}return 0==n.length?i.children=null:i.children=n,i}function w(e,t,s,i,n){if(!e)return null;i&&e.type&&i.push(e.type);let r=[];for(let s in e.children){const a=w(e.children[s],t,s,i,n);a&&r.push(a)}return 0==r.length&&(r=e.text?[e.text]:null),i&&e.type&&i.pop(),t.call(n,e.type,e.data,r,s,i)}function v(e,t,s){if(!e)return null;s&&(t-=s.length);return b(e,(function(e){if(t<=-1)return null;if(e.att)return e;if(0==t)e.text=s,t=-1;else if(e.text){const i=e.text.length;i>t?(e.text=e.text.substring(0,t)+s,t=-1):t-=i}return e}))}function y(e,t){return b(e,(e=>{const s=S(e.data,!0,t?t(e):null);return s?e.data=s:delete e.data,e}))}function M(e){if("BR"==e.type)e=null;else if(e.text)e.type||(e.text=e.text.trimStart(),e.text||(e=null));else if(!e.type&&e.children&&e.children.length>0){const t=M(e.children[0]);t?e.children[0]=t:(e.children.shift(),e.type||0!=e.children.length||(e=null))}return e}function T(e,t){if(!e)return null;if(e.att)e.text=" ",delete e.att,delete e.children;else if(e.children){const i=[],n=[];for(let r in e.children){const a=e.children[r];if(a.att){if(i.length==t)continue;if(a.data.mime==s)continue;delete a.att,delete a.children,a.text=" ",i.push(a)}else n.push(a)}e.children=n.concat(i)}return e}function P(e,t){let s="",i=[];for(let n in e){const r=e[n];if(!r.txt){const e=P(r.children,s.length+t);r.txt=e.txt,i=i.concat(e.fmt)}r.tp&&i.push({at:s.length+t,len:r.txt.length,tp:r.tp}),s+=r.txt}return{txt:s,fmt:i}}function S(e,s,n){if(e&&Object.entries(e).length>0){n=n||[];const r={};if(i.forEach((i=>{if(e[i]){if(s&&!n.includes(i)&&("string"==typeof e[i]||Array.isArray(e[i]))&&e[i].length>t)return;if("object"==typeof e[i])return;r[i]=e[i]}})),0!=Object.entries(r).length)return r}return null}l.init=function(e){if(void 0===e)e="";else if("string"!=typeof e)return null;return{txt:e}},l.parse=function(e){if("string"!=typeof e)return null;const t=e.split(/\r?\n/),s=[],i={},r=[];t.forEach((e=>{let t,o,c=[];if(n.forEach((t=>{c=c.concat(function(e,t,s,i){const n=[];let r=0,a=e.slice(0);for(;a.length>0;){const o=t.exec(a);if(null==o)break;let c=o.index+o[0].lastIndexOf(o[1]);a=a.slice(c+1),c+=r,r=c+1;const h=s?s.exec(a):null;if(null==h)break;let d=h.index+h[0].indexOf(h[1]);a=a.slice(d+1),d+=r,r=d+1,n.push({txt:e.slice(c+1,d),children:[],at:c,end:d,tp:i})}return n}(e,t.start,t.end,t.name))})),0==c.length)o={txt:e};else{c.sort(((e,t)=>{const s=e.at-t.at;return 0!=s?s:t.end-e.end})),c=p(c);const t=P(u(e,0,e.length,c),0);o={txt:t.txt,fmt:t.fmt}}if(t=function(e){let t,s=[];if(a.forEach((i=>{for(;null!==(t=i.re.exec(e));)s.push({offset:t.index,len:t[0].length,unique:t[0],data:i.pack(t[0]),type:i.name})})),0==s.length)return s;s.sort(((e,t)=>e.offset-t.offset));let i=-1;return s=s.filter((e=>{const t=e.offset>i;return i=e.offset+e.len,t})),s}(o.txt),t.length>0){const e=[];for(let n in t){const r=t[n];let a=i[r.unique];a||(a=s.length,i[r.unique]=a,s.push({tp:r.type,data:r.data})),e.push({at:r.offset,len:r.len,key:a})}o.ent=e}r.push(o)}));const o={txt:""};if(r.length>0){o.txt=r[0].txt,o.fmt=(r[0].fmt||[]).concat(r[0].ent||[]);for(let e=1;e(e.at+=s,e))))),t.ent&&(o.fmt=o.fmt.concat(t.ent.map((e=>(e.at+=s,e)))))}0==o.fmt.length&&delete o.fmt,s.length>0&&(o.ent=s)}return o},l.append=function(e,t){if(!e)return t;if(!t)return e;e.txt=e.txt||"";const s=e.txt.length;return"string"==typeof t?e.txt+=t:t.txt&&(e.txt+=t.txt),Array.isArray(t.fmt)&&(e.fmt=e.fmt||[],Array.isArray(t.ent)&&(e.ent=e.ent||[]),t.fmt.forEach((i=>{const n={at:(0|i.at)+s,len:0|i.len};-1==i.at&&(n.at=-1,n.len=0),i.tp?n.tp=i.tp:(n.key=e.ent.length,e.ent.push(t.ent[i.key||0])),e.fmt.push(n)}))),e},l.insertImage=function(e,t,s){(e=e||{txt:" "}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:0|t,len:1,key:e.ent.length});const i={tp:"IM",data:{mime:s.mime,ref:s.refurl,val:s.bits||s.preview,width:s.width,height:s.height,name:s.filename,size:0|s.size}};return s.urlPromise&&(i.data._tempPreview=s._tempPreview,i.data._processing=!0,s.urlPromise.then((e=>{i.data.ref=e,i.data._tempPreview=void 0,i.data._processing=void 0}),(e=>{i.data._processing=void 0}))),e.ent.push(i),e},l.insertVideo=function(e,t,s){(e=e||{txt:" "}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:0|t,len:1,key:e.ent.length});const i={tp:"VD",data:{mime:s.mime,ref:s.refurl,val:s.bits,preref:s.preref,preview:s.preview,width:s.width,height:s.height,duration:0|s.duration,name:s.filename,size:0|s.size}};return s.urlPromise&&(i.data._tempPreview=s._tempPreview,i.data._processing=!0,s.urlPromise.then((e=>{i.data.ref=e[0],i.data.preref=e[1],i.data._tempPreview=void 0,i.data._processing=void 0}),(e=>{i.data._processing=void 0}))),e.ent.push(i),e},l.insertAudio=function(e,t,s){(e=e||{txt:" "}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:0|t,len:1,key:e.ent.length});const i={tp:"AU",data:{mime:s.mime,val:s.bits,duration:0|s.duration,preview:s.preview,name:s.filename,size:0|s.size,ref:s.refurl}};return s.urlPromise&&(i.data._processing=!0,s.urlPromise.then((e=>{i.data.ref=e,i.data._processing=void 0}),(e=>{i.data._processing=void 0}))),e.ent.push(i),e},l.videoCall=function(e){return{txt:" ",fmt:[{at:0,len:1,key:0}],ent:[{tp:"VC",data:{aonly:e}}]}},l.updateVideoCall=function(e,t){const s=((e||{}).fmt||[])[0];if(!s)return e;let i;if("VC"==s.tp)delete s.tp,s.key=0,i={tp:"VC"},e.ent=[i];else if(i=(e.ent||[])[0|s.key],!i||"VC"!=i.tp)return e;return i.data=i.data||{},Object.assign(i.data,t),e},l.quote=function(e,t,s){const i=l.append(l.appendLineBreak(l.mention(e,t)),s);return i.fmt.push({at:0,len:i.txt.length,tp:"QQ"}),i},l.mention=function(e,t){return{txt:e||"",fmt:[{at:0,len:(e||"").length,key:0}],ent:[{tp:"MN",data:{val:t}}]}},l.appendLink=function(e,t){(e=e||{txt:""}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:e.txt.length,len:t.txt.length,key:e.ent.length}),e.txt+=t.txt;const s={tp:"LN",data:{url:t.url}};return e.ent.push(s),e},l.appendImage=function(e,t){return(e=e||{txt:""}).txt+=" ",l.insertImage(e,e.txt.length-1,t)},l.appendAudio=function(e,t){return(e=e||{txt:""}).txt+=" ",l.insertAudio(e,e.txt.length-1,t)},l.attachFile=function(e,t){(e=e||{txt:""}).ent=e.ent||[],e.fmt=e.fmt||[],e.fmt.push({at:-1,len:0,key:e.ent.length});const s={tp:"EX",data:{mime:t.mime,val:t.data,name:t.filename,ref:t.refurl,size:0|t.size}};return t.urlPromise&&(s.data._processing=!0,t.urlPromise.then((e=>{s.data.ref=e,s.data._processing=void 0}),(e=>{s.data._processing=void 0}))),e.ent.push(s),e},l.wrapInto=function(e,t,s,i){return"string"==typeof e&&(e={txt:e}),e.fmt=e.fmt||[],e.fmt.push({at:s||0,len:i||e.txt.length,tp:t}),e},l.wrapAsForm=function(e,t,s){return l.wrapInto(e,"FM",t,s)},l.insertButton=function(e,t,s,i,n,r,a){return"string"==typeof e&&(e={txt:e}),!e||!e.txt||e.txt.length{switch(e.type){case"IM":return["val"];case"VD":return["preview"]}return null})),_({},s,[])},l.preview=function(e,t,s){let i=g(e);i=T(i,3);if(i=b(i,(function(e){return"MN"==e.type?e.parent&&e.parent.type||!(e.text||"").startsWith("➦")||(e.text="➦",delete e.children):"QQ"==e.type?(e.text=" ",delete e.children):"BR"==e.type&&(e.text=" ",delete e.children,delete e.type),e})),i=v(i,t,"…"),s){const e={IM:["val"],VD:["preview"]};i=y(i,(t=>e[t.type]))}else i=y(i);return _({},i,[])},l.toPlainText=function(e){return"string"==typeof e?e:e.txt},l.isPlainText=function(e){return"string"==typeof e||!(e.fmt||e.ent)},l.toMarkdown=function(e){return w(g(e),(function(e,t,s){const i=o[e];let n=s?s.join(""):"";return i&&(i.isVoid?n=i.md_tag||"":i.md_tag&&(n=i.md_tag+n+i.md_tag)),n}),0)},l.isValid=function(e){if(!e)return!1;const{txt:t,fmt:s,ent:i}=e;if(!t&&""!==t&&!s&&!i)return!1;const n=typeof t;return("string"==n||"undefined"==n||null===t)&&(!(void 0!==s&&!Array.isArray(s)&&null!==s)&&!(void 0!==i&&!Array.isArray(i)&&null!==i))},l.hasAttachments=function(e){if(!Array.isArray(e.fmt))return!1;for(let t in e.fmt){const s=e.fmt[t];if(s&&s.at<0){const t=e.ent[0|s.key];return t&&"EX"==t.tp&&t.data}}return!1},l.attachments=function(e,t,s){if(!Array.isArray(e.fmt))return;let i=0;for(let n in e.fmt){let r=e.fmt[n];if(r&&r.at<0){const n=e.ent[0|r.key];if(n&&"EX"==n.tp&&n.data&&t.call(s,n.data,i++,"EX"))break}}},l.hasEntities=function(e){return e.ent&&e.ent.length>0},l.entities=function(e,t,s){if(e.ent&&e.ent.length>0)for(let i in e.ent)if(e.ent[i]&&t.call(s,e.ent[i].data,i,e.ent[i].tp))break},l.styles=function(e,t,s){if(e.fmt&&e.fmt.length>0)for(let i in e.fmt){const n=e.fmt[i];if(n&&t.call(s,n.tp,n.at,n.len,n.key,i))break}},l.sanitizeEntities=function(e){if(e&&e.ent&&e.ent.length>0)for(let t in e.ent){const s=e.ent[t];if(s&&s.data){const i=S(s.data);i?e.ent[t].data=i:delete e.ent[t].data}}return e},l.getDownloadUrl=function(e){let t=null;return e.mime!=s&&e.val?t=c(e.val,e.mime,l.logger):"string"==typeof e.ref&&(t=e.ref),t},l.isProcessing=function(e){return!!e._processing},l.getPreviewUrl=function(e){return e.val?c(e.val,e.mime,l.logger):null},l.getEntitySize=function(e){return e.size?e.size:e.val?.75*e.val.length|0:0},l.getEntityMimeType=function(e){return e.mime||"text/plain"},l.tagName=function(e){return o[e]&&o[e].html_tag},l.attrValue=function(e,t){if(t&&d[e]&&d[e].props)return d[e].props(t)},l.getContentType=function(){return"text/x-drafty"},e.exports=l}},t={};function s(i){var n=t[i];if(void 0!==n)return n.exports;var r=t[i]={exports:{}};return e[i](r,r.exports,s),r.exports}s.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return s.d(t,{a:t}),t},s.d=(e,t)=>{for(var i in t)s.o(t,i)&&!s.o(e,i)&&Object.defineProperty(e,i,{enumerable:!0,get:t[i]})},s.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(e){if("object"==typeof window)return window}}(),s.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),s.r=e=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})};var i={};return(()=>{s.r(i),s.d(i,{AccessMode:()=>e,Drafty:()=>A(),Tinode:()=>W});class e{constructor(t){t&&(this.given="number"==typeof t.given?t.given:e.decode(t.given),this.want="number"==typeof t.want?t.want:e.decode(t.want),this.mode=t.mode?"number"==typeof t.mode?t.mode:e.decode(t.mode):this.given&this.want)}static#e(e,t,s){if(["given","want","mode"].includes(t=t||"mode"))return 0!=(e[t]&s);throw new Error(`Invalid AccessMode component '${t}'`)}static decode(t){if(!t)return null;if("number"==typeof t)return t&e._BITMASK;if("N"===t||"n"===t)return e._NONE;const s={J:e._JOIN,R:e._READ,W:e._WRITE,P:e._PRES,A:e._APPROVE,S:e._SHARE,D:e._DELETE,O:e._OWNER};let i=e._NONE;for(let e=0;e=20&&s.length<=24&&["ts","touched","updated","created","when","deleted","expires"].includes(t)){const e=new Date(s);if(!isNaN(e))return e}else if("acs"===t&&"object"==typeof s)return new e(s);return s}function g(e){return e&&!/^\s*([a-z][a-z0-9+.-]*:|\/\/)/im.test(e)}function f(e){return e instanceof Date&&!isNaN(e)&&0!=e.getTime()}function m(t,s,i){if("object"!=typeof s){if(void 0===s)return t;if(s===l)return;return s}if(null===s)return s;if(s instanceof Date&&!isNaN(s))return!t||!(t instanceof Date)||isNaN(t)||t{"_"==t[0]?delete e[t]:e[t]?Array.isArray(e[t])&&0==e[t].length?delete e[t]:e[t]?e[t]instanceof Date?f(e[t])||delete e[t]:"object"==typeof e[t]&&(b(e[t]),0==Object.getOwnPropertyNames(e[t]).length&&delete e[t]):delete e[t]:delete e[t]})),e}let w,v;const y="Connection failed",M=418,T="Disconnected by client";function P(e,t,s,i){let n=null;return["http","https","ws","wss"].includes(t)&&(n=`${t}://${e}`,"/"!==n.charAt(n.length-1)&&(n+="/"),n+="v"+s+"/channels",["http","https"].includes(t)&&(n+="/lp"),n+="?apikey="+i),n}class S{static#t=e=>{};#s=null;#i=0;#n=!1;#r=null;host;secure;apiKey;version;autoreconnect;initialized;constructor(e,t,s){if(this.host=e.host,this.secure=e.secure,this.apiKey=e.apiKey,this.version=t,this.autoreconnect=s,"lp"===e.transport?(this.#a(),this.initialized="lp"):"ws"===e.transport&&(this.#o(),this.initialized="ws"),!this.initialized)throw S.#t("Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'."),new Error("Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.")}static setNetworkProviders(e,t){w=e,v=t}static set logger(e){S.#t=e}connect(e,t){return Promise.reject(null)}reconnect(e){}disconnect(){}sendText(e){}isConnected(){return!1}transport(){return this.initialized}probe(){this.sendText("1")}backoffReset(){this.#c()}#h(){clearTimeout(this.#s);const e=Math.pow(2,this.#i)*(1+.3*Math.random())*2e3;this.#i=this.#i>=10?this.#i:this.#i+1,this.onAutoreconnectIteration&&this.onAutoreconnectIteration(e),this.#s=setTimeout((t=>{if(S.#t(`Reconnecting, iter=${this.#i}, timeout=${e}`),this.#n)this.onAutoreconnectIteration&&this.onAutoreconnectIteration(-1);else{const e=this.connect();this.onAutoreconnectIteration?this.onAutoreconnectIteration(0,e):e.catch((e=>{}))}}),e)}#d(){clearTimeout(this.#s),this.#s=null}#c(){this.#i=0}#a(){let e=null,t=null,s=null,i=(t,s,n)=>{let r=new v,a=!1;return r.onreadystatechange=o=>{if(4==r.readyState)if(201==r.status){let n=JSON.parse(r.responseText,p);e=t+"&sid="+n.ctrl.params.sid,r=i(e),r.send(null),this.onOpen&&this.onOpen(),s&&(a=!0,s()),this.autoreconnect&&this.#d()}else if(r.status>0&&r.status<400)this.onMessage&&this.onMessage(r.responseText),r=i(e),r.send(null);else{if(n&&!a&&(a=!0,n(r.responseText)),this.onMessage&&r.responseText&&this.onMessage(r.responseText),this.onDisconnect){const e=r.status||(this.#n?M:503),t=r.responseText||(this.#n?T:y);this.onDisconnect(new u(t,e),e)}r=null,!this.#n&&this.autoreconnect&&this.#h()}},r.open("POST",t,!0),r};this.connect=(e,s)=>{if(this.#n=!1,t){if(!s)return Promise.resolve();t.onreadystatechange=void 0,t.abort(),t=null}return e&&(this.host=e),new Promise(((e,s)=>{const n=P(this.host,this.secure?"https":"http",this.version,this.apiKey);S.#t("LP connecting to:",n),t=i(n,e,s),t.send(null)})).catch((e=>{S.#t("LP connection failed:",e)}))},this.reconnect=e=>{this.#d(),this.connect(null,e)},this.disconnect=i=>{this.#n=!0,this.#d(),s&&(s.onreadystatechange=void 0,s.abort(),s=null),t&&(t.onreadystatechange=void 0,t.abort(),t=null),this.onDisconnect&&this.onDisconnect(new u(T,M),M),e=null},this.sendText=t=>{if(s=(e=>{const t=new v;return t.onreadystatechange=e=>{if(4==t.readyState&&t.status>=400)throw new u("LP sender failed",t.status)},t.open("POST",e,!0),t})(e),!s||1!=s.readyState)throw new Error("Long poller failed to connect");s.send(t)},this.isConnected=e=>t&&!0}#o(){this.connect=(e,t)=>{if(this.#n=!1,this.#r){if(!t&&this.#r.readyState==this.#r.OPEN)return Promise.resolve();this.#r.close(),this.#r=null}return e&&(this.host=e),new Promise(((e,t)=>{const s=P(this.host,this.secure?"wss":"ws",this.version,this.apiKey);S.#t("WS connecting to: ",s);const i=new w(s);i.onerror=e=>{t(e)},i.onopen=t=>{this.autoreconnect&&this.#d(),this.onOpen&&this.onOpen(),e()},i.onclose=e=>{if(this.#r=null,this.onDisconnect){const e=this.#n?M:503;this.onDisconnect(new u(this.#n?T:y,e),e)}!this.#n&&this.autoreconnect&&this.#h()},i.onmessage=e=>{this.onMessage&&this.onMessage(e.data)},this.#r=i}))},this.reconnect=e=>{this.#d(),this.connect(null,e)},this.disconnect=e=>{this.#n=!0,this.#d(),this.#r&&(this.#r.close(),this.#r=null)},this.sendText=e=>{if(!this.#r||this.#r.readyState!=this.#r.OPEN)throw new Error("Websocket is not connected");this.#r.send(e)},this.isConnected=e=>this.#r&&this.#r.readyState==this.#r.OPEN}onMessage=void 0;onDisconnect=void 0;onOpen=void 0;onAutoreconnectIteration=void 0}S.NETWORK_ERROR=503,S.NETWORK_ERROR_TEXT=y,S.NETWORK_USER=M,S.NETWORK_USER_TEXT=T;const x="tinode-web";let k;class E{#l=e=>{};#u=e=>{};db=null;disabled=!0;constructor(e,t){this.#l=e||this.#l,this.#u=t||this.#u}#p(e,t,s){return this.db?new Promise(((i,n)=>{const r=this.db.transaction([e]);r.onerror=t=>{this.#u("PCache","mapObjects",e,t.target.error),n(t.target.error)},r.objectStore(e).getAll().onsuccess=e=>{t&&e.target.result.forEach((e=>{t.call(s,e)})),i(e.target.result)}})):disabled?Promise.resolve([]):Promise.reject(new Error("not initialized"))}initDatabase(){return new Promise(((e,t)=>{const s=k.open(x,1);s.onsuccess=t=>{this.db=t.target.result,this.disabled=!1,e(this.db)},s.onerror=e=>{this.#u("PCache","failed to initialize",e),t(e.target.error),this.#l(e.target.error)},s.onupgradeneeded=e=>{this.db=e.target.result,this.db.onerror=e=>{this.#u("PCache","failed to create storage",e),this.#l(e.target.error)},this.db.createObjectStore("topic",{keyPath:"name"}),this.db.createObjectStore("user",{keyPath:"uid"}),this.db.createObjectStore("subscription",{keyPath:["topic","uid"]}),this.db.createObjectStore("message",{keyPath:["topic","seq"]})}}))}deleteDatabase(){return this.db&&(this.db.close(),this.db=null),new Promise(((e,t)=>{const s=k.deleteDatabase(x);s.onblocked=e=>{this.db&&this.db.close();const s=new Error("blocked");this.#u("PCache","deleteDatabase",s),t(s)},s.onsuccess=t=>{this.db=null,this.disabled=!0,e(!0)},s.onerror=e=>{this.#u("PCache","deleteDatabase",e.target.error),t(e.target.error)}}))}isReady(){return!!this.db}updTopic(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["topic"],"readwrite");i.oncomplete=e=>{t(e.target.result)},i.onerror=e=>{this.#u("PCache","updTopic",e.target.error),s(e.target.error)};const n=i.objectStore("topic").get(e.name);n.onsuccess=t=>{i.objectStore("topic").put(E.#g(n.result,e)),i.commit()}})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}markTopicAsDeleted(e,t){return this.isReady()?new Promise(((s,i)=>{const n=this.db.transaction(["topic"],"readwrite");n.oncomplete=e=>{s(e.target.result)},n.onerror=e=>{this.#u("PCache","markTopicAsDeleted",e.target.error),i(e.target.error)};n.objectStore("topic").get(e).onsuccess=e=>{const s=e.target.result;s&&s._deleted!=t&&(s._deleted=t,n.objectStore("topic").put(s)),n.commit()}})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}remTopic(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["topic","subscription","message"],"readwrite");i.oncomplete=e=>{t(e.target.result)},i.onerror=e=>{this.#u("PCache","remTopic",e.target.error),s(e.target.error)},i.objectStore("topic").delete(IDBKeyRange.only(e)),i.objectStore("subscription").delete(IDBKeyRange.bound([e,"-"],[e,"~"])),i.objectStore("message").delete(IDBKeyRange.bound([e,0],[e,Number.MAX_SAFE_INTEGER])),i.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}mapTopics(e,t){return this.#p("topic",e,t)}deserializeTopic(e,t){E.#f(e,t)}updUser(e,t){if(!(arguments.length<2||void 0===t))return this.isReady()?new Promise(((s,i)=>{const n=this.db.transaction(["user"],"readwrite");n.oncomplete=e=>{s(e.target.result)},n.onerror=e=>{this.#u("PCache","updUser",e.target.error),i(e.target.error)},n.objectStore("user").put({uid:e,public:t}),n.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}remUser(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["user"],"readwrite");i.oncomplete=e=>{t(e.target.result)},i.onerror=e=>{this.#u("PCache","remUser",e.target.error),s(e.target.error)},i.objectStore("user").delete(IDBKeyRange.only(e)),i.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}mapUsers(e,t){return this.#p("user",e,t)}getUser(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["user"]);i.oncomplete=e=>{const s=e.target.result;t({user:s.uid,public:s.public})},i.onerror=e=>{this.#u("PCache","getUser",e.target.error),s(e.target.error)},i.objectStore("user").get(e)})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}updSubscription(e,t,s){return this.isReady()?new Promise(((i,n)=>{const r=this.db.transaction(["subscription"],"readwrite");r.oncomplete=e=>{i(e.target.result)},r.onerror=e=>{this.#u("PCache","updSubscription",e.target.error),n(e.target.error)},r.objectStore("subscription").get([e,t]).onsuccess=i=>{r.objectStore("subscription").put(E.#m(i.target.result,e,t,s)),r.commit()}})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}mapSubscriptions(e,t,s){return this.isReady()?new Promise(((i,n)=>{const r=this.db.transaction(["subscription"]);r.onerror=e=>{this.#u("PCache","mapSubscriptions",e.target.error),n(e.target.error)},r.objectStore("subscription").getAll(IDBKeyRange.bound([e,"-"],[e,"~"])).onsuccess=e=>{t&&e.target.result.forEach((e=>{t.call(s,e)})),i(e.target.result)}})):this.disabled?Promise.resolve([]):Promise.reject(new Error("not initialized"))}addMessage(e){return this.isReady()?new Promise(((t,s)=>{const i=this.db.transaction(["message"],"readwrite");i.onsuccess=e=>{t(e.target.result)},i.onerror=e=>{this.#u("PCache","addMessage",e.target.error),s(e.target.error)},i.objectStore("message").add(E.#_(null,e)),i.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}updMessageStatus(e,t,s){return this.isReady()?new Promise(((i,n)=>{const r=this.db.transaction(["message"],"readwrite");r.onsuccess=e=>{i(e.target.result)},r.onerror=e=>{this.#u("PCache","updMessageStatus",e.target.error),n(e.target.error)};const a=r.objectStore("message").get(IDBKeyRange.only([e,t]));a.onsuccess=i=>{const n=a.result||i.target.result;n&&n._status!=s?(r.objectStore("message").put(E.#_(n,{topic:e,seq:t,_status:s})),r.commit()):r.commit()}})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}remMessages(e,t,s){return this.isReady()?new Promise(((i,n)=>{t||s||(t=0,s=Number.MAX_SAFE_INTEGER);const r=s>0?IDBKeyRange.bound([e,t],[e,s],!1,!0):IDBKeyRange.only([e,t]),a=this.db.transaction(["message"],"readwrite");a.onsuccess=e=>{i(e.target.result)},a.onerror=e=>{this.#u("PCache","remMessages",e.target.error),n(e.target.error)},a.objectStore("message").delete(r),a.commit()})):this.disabled?Promise.resolve():Promise.reject(new Error("not initialized"))}readMessages(e,t,s,i){return this.isReady()?new Promise(((n,r)=>{const a=(t=t||{}).since>0?t.since:0,o=t.before>0?t.before:Number.MAX_SAFE_INTEGER,c=0|t.limit,h=[],d=IDBKeyRange.bound([e,a],[e,o],!1,!0),l=this.db.transaction(["message"]);l.onerror=e=>{this.#u("PCache","readMessages",e.target.error),r(e.target.error)},l.objectStore("message").openCursor(d,"prev").onsuccess=e=>{const t=e.target.result;t?(s&&s.call(i,t.value),h.push(t.value),c<=0||h.length{t.hasOwnProperty(s)&&(e[s]=t[s])})),Array.isArray(t.tags)&&(e._tags=t.tags),t.acs&&e.setAccessMode(t.acs),e.seq|=0,e.read|=0,e.unread=Math.max(0,e.seq-e.read)}static#g(e,t){const s=e||{name:t.name};return E.#b.forEach((e=>{t.hasOwnProperty(e)&&(s[e]=t[e])})),Array.isArray(t._tags)&&(s.tags=t._tags),t.acs&&(s.acs=t.getAccessMode().jsonHelper()),s}static#m(e,t,s,i){const n=e||{topic:t,uid:s};return["updated","mode","read","recv","clear","lastSeen","userAgent"].forEach((e=>{i.hasOwnProperty(e)&&(n[e]=i[e])})),n}static#_(e,t){const s=e||{};return["topic","seq","ts","_status","from","head","content"].forEach((e=>{t.hasOwnProperty(e)&&(s[e]=t[e])})),s}static setDatabaseProvider(e){k=e}}var D=s(672),A=s.n(D);let R,C,N,q;class I{constructor(e,t){this._tinode=e,this._version=t,this._apiKey=e._apiKey,this._authToken=e.getAuthToken(),this.xhr=[]}uploadWithBaseUrl(e,t,s,i,n,r){let a=`/v${this._version}/file/u/`;if(e){let t=e;if(t.endsWith("/")&&(t=t.slice(0,-1)),!t.startsWith("http://")&&!t.startsWith("https://"))throw new Error(`Invalid base URL '${e}'`);a=t+a}const o=this,c=new R;this.xhr.push(c),c.open("POST",a,!0),c.setRequestHeader("X-Tinode-APIKey",this._apiKey),this._authToken&&c.setRequestHeader("X-Tinode-Auth",`Token ${this._authToken.token}`);let h=null,d=null;const l=new Promise(((e,t)=>{h=e,d=t}));c.upload.onprogress=e=>{e.lengthComputable&&(i&&i(e.loaded/e.total),this.onProgress&&this.onProgress(e.loaded/e.total))},c.onload=function(){let e;try{e=JSON.parse(this.response,p)}catch(t){o._tinode.logger("ERROR: Invalid server response in LargeFileHelper",this.response),e={ctrl:{code:this.status,text:this.statusText}}}this.status>=200&&this.status<300?(h&&h(e.ctrl.params.url),n&&n(e.ctrl)):this.status>=400?(d&&d(new u(e.ctrl.text,e.ctrl.code)),r&&r(e.ctrl)):o._tinode.logger("ERROR: Unexpected server response status",this.status,this.response)},c.onerror=function(e){d&&d(e||new Error("failed")),r&&r(null)},c.onabort=function(e){d&&d(new Error("upload cancelled by user")),r&&r(null)};try{const e=new FormData;e.append("file",t),e.set("id",this._tinode.getNextUniqueId()),s&&e.set("topic",s),c.send(e)}catch(e){d&&d(e),r&&r(null)}return l}upload(e,t,s,i,n){const r=(this._tinode._secure?"https://":"http://")+this._tinode._host;return this.uploadWithBaseUrl(r,e,t,s,i,n)}download(e,t,s,i,n){if(!g(e))return void(n&&n(`The URL '${e}' must be relative, not absolute`));if(!this._authToken)return void(n&&n("Must authenticate first"));const r=this,a=new R;this.xhr.push(a),e=function(e,t,s){const i=new URL(e,window.location.origin);return i.searchParams.append(t,s),i.toString().substring(window.location.origin.length)}(e,"asatt","1"),a.open("GET",e,!0),a.setRequestHeader("X-Tinode-APIKey",this._apiKey),a.setRequestHeader("X-Tinode-Auth","Token "+this._authToken.token),a.responseType="blob",a.onprogress=function(e){i&&i(e.loaded)};let o=null,c=null;const h=new Promise(((e,t)=>{o=e,c=t}));a.onload=function(){if(200==this.status){const e=document.createElement("a");e.href=window.URL.createObjectURL(new Blob([this.response],{type:s})),e.style.display="none",e.setAttribute("download",t),document.body.appendChild(e),e.click(),document.body.removeChild(e),window.URL.revokeObjectURL(e.href),o&&o()}else if(this.status>=400&&c){const e=new FileReader;e.onload=function(){try{const e=JSON.parse(this.result,p);c(new u(e.ctrl.text,e.ctrl.code))}catch(e){r._tinode.logger("ERROR: Invalid server response in LargeFileHelper",this.result),c(e)}},e.readAsText(this.response)}},a.onerror=function(e){c&&c(new Error("failed")),n&&n(e)},a.onabort=function(){c&&c(null)};try{a.send()}catch(e){c&&c(e),n&&n(e)}return h}cancel(){this.xhr.forEach((e=>{e.readyState<4&&e.abort()}))}static setNetworkProvider(e){R=e}}class U{constructor(e){this.topic=e,this.what={}}#w(){return this.topic._deleted?void 0:this.topic.updated}#v(){return this.topic.isP2PType()?this.#w():this.topic._deleted?void 0:this.topic._lastSubsUpdate}withData(e,t,s){return this.what.data={since:e,before:t,limit:s},this}withLaterData(e){return this.withData(this.topic._maxSeq>0?this.topic._maxSeq+1:void 0,void 0,e)}withEarlierData(e){return this.withData(void 0,this.topic._minSeq>0?this.topic._minSeq:void 0,e)}withDesc(e){return this.what.desc={ims:e},this}withLaterDesc(){return this.withDesc(this.#w())}withSub(e,t,s){const i={ims:e,limit:t};return"me"==this.topic.getType()?i.topic=s:i.user=s,this.what.sub=i,this}withOneSub(e,t){return this.withSub(e,void 0,t)}withLaterOneSub(e){return this.withOneSub(this.topic._lastSubsUpdate,e)}withLaterSub(e){return this.withSub(this.#v(),e)}withTags(){return this.what.tags=!0,this}withCred(){return"me"==this.topic.getType()?this.what.cred=!0:this.topic._tinode.logger("ERROR: Invalid topic type for MetaGetBuilder:withCreds",this.topic.getType()),this}withDel(e,t){return(e||t)&&(this.what.del={since:e,limit:t}),this}withLaterDel(e){return this.withDel(this.topic._maxSeq>0?this.topic._maxDel+1:void 0,e)}extract(e){return this.what[e]}build(){const e=[];let t={};return["data","sub","desc","tags","cred","del"].forEach((s=>{this.what.hasOwnProperty(s)&&(e.push(s),Object.getOwnPropertyNames(this.what[s]).length>0&&(t[s]=this.what[s]))})),e.length>0?t.what=e.join(" "):t=void 0,t}}class O{#y=void 0;#M=!1;buffer=[];constructor(e,t){this.#y=e||((e,t)=>e===t?0:e0)){o=!0;break}n=r-1}return o?{idx:r,exact:!0}:s?{idx:-1}:{idx:a<0?r+1:r}}#P(e,t){const s=this.#T(e,t,!1),i=s.exact&&this.#M?1:0;return t.splice(s.idx,i,e),t}getAt(e){return this.buffer[e]}getLast(e){return e|=0,this.buffer.length>e?this.buffer[this.buffer.length-1-e]:void 0}put(){let e;e=1==arguments.length&&Array.isArray(arguments[0])?arguments[0]:arguments;for(let t in e)this.#P(e[t],this.buffer)}delAt(e){e|=0;let t=this.buffer.splice(e,1);if(t&&t.length>0)return t[0]}delRange(e,t){return this.buffer.splice(e,t-e)}length(){return this.buffer.length}reset(){this.buffer=[]}forEach(e,t,s,i){t|=0,s=s||this.buffer.length;for(let n=t;nt?this.buffer[n-1]:void 0,ne.seq-t.seq),!0),this._attached=!1,this._lastSubsUpdate=new Date(0),this._new=!0,this._deleted=!1,this._delayedLeaveTimer=null,s&&(this.onData=s.onData,this.onMeta=s.onMeta,this.onPres=s.onPres,this.onInfo=s.onInfo,this.onMetaDesc=s.onMetaDesc,this.onMetaSub=s.onMetaSub,this.onSubsUpdated=s.onSubsUpdated,this.onTagsUpdated=s.onTagsUpdated,this.onCredsUpdated=s.onCredsUpdated,this.onDeleteTopic=s.onDeleteTopic,this.onAllMessagesReceived=s.onAllMessagesReceived)}static topicType(e){return{me:o,fnd:c,grp:h,new:h,nch:h,chn:h,usr:"p2p",sys:"sys"}["string"==typeof e?e.substring(0,3):"xxx"]}static isMeTopicName(e){return j.topicType(e)==o}static isGroupTopicName(e){return j.topicType(e)==h}static isP2PTopicName(e){return"p2p"==j.topicType(e)}static isCommTopicName(e){return j.isP2PTopicName(e)||j.isGroupTopicName(e)}static isNewGroupTopicName(e){return"string"==typeof e&&(e.substring(0,3)==r||e.substring(0,3)==a)}static isChannelTopicName(e){return"string"==typeof e&&("chn"==e.substring(0,3)||e.substring(0,3)==a)}isSubscribed(){return this._attached}subscribe(e,t){return clearTimeout(this._delayedLeaveTimer),this._delayedLeaveTimer=null,this._attached?Promise.resolve(this):this._tinode.subscribe(this.name||r,e,t).then((e=>{if(e.code>=300)return e;if(this._attached=!0,this._deleted=!1,this.acs=e.params&&e.params.acs?e.params.acs:this.acs,this._new){if(delete this._new,this.name!=e.topic&&(this._cacheDelSelf(),this.name=e.topic),this._cachePutSelf(),this.created=e.ts,this.updated=e.ts,this.name!=o&&this.name!=c){const e=this._tinode.getMeTopic();e.onMetaSub&&e.onMetaSub(this),e.onSubsUpdated&&e.onSubsUpdated([this.name],1)}t&&t.desc&&(t.desc._noForwarding=!0,this._processMetaDesc(t.desc))}return e}))}createMessage(e,t){return this._tinode.createMessage(this.name,e,t)}publish(e,t){return this.publishMessage(this.createMessage(e,t))}publishMessage(e){if(!this._attached)return Promise.reject(new Error("Cannot publish on inactive topic"));if(this._sending)return Promise.reject(new Error("The message is already being sent"));e._sending=!0,e._failed=!1;let t=null;return A().hasEntities(e.content)&&(t=[],A().entities(e.content,(e=>{e&&(e.ref&&t.push(e.ref),e.preref&&t.push(e.preref))})),0==t.length&&(t=null)),this._tinode.publishMessage(e,t).then((t=>(e._sending=!1,e.ts=t.ts,this.swapMessageId(e,t.params.seq),this._maybeUpdateMessageVersionsCache(e),this._routeData(e),t))).catch((t=>{this._tinode.logger("WARNING: Message rejected by the server",t),e._sending=!1,e._failed=!0,this.onData&&this.onData()}))}publishDraft(e,t){const s=e.seq||this._getQueuedSeqId();return e._noForwarding||(e._noForwarding=!0,e.seq=s,e.ts=new Date,e.from=this._tinode.getCurrentUserID(),e.noecho=!0,this._messages.put(e),this._tinode._db.addMessage(e),this.onData&&this.onData(e)),(t||Promise.resolve()).then((t=>e._cancelled?{code:300,text:"cancelled"}:this.publishMessage(e))).catch((t=>{throw this._tinode.logger("WARNING: Message draft rejected",t),e._sending=!1,e._failed=!0,e._fatal=t instanceof u&&(t.code>=400&&t.code<500),this.onData&&this.onData(),t}))}leave(e){return this._attached||e?this._tinode.leave(this.name,e).then((t=>(this._resetSub(),e&&this._gone(),t))):Promise.reject(new Error("Cannot leave inactive topic"))}leaveDelayed(e,t){clearTimeout(this._delayedLeaveTimer),this._delayedLeaveTimer=setTimeout((t=>{this._delayedLeaveTimer=null,this.leave(e)}),t)}getMeta(e){return this._tinode.getMeta(this.name,e)}getMessagesPage(e,t){let s=t?this.startMetaQuery().withLaterData(e):this.startMetaQuery().withEarlierData(e);return this._loadMessages(this._tinode._db,s.extract("data")).then((i=>{if(i==e)return Promise.resolve({topic:this.name,code:200,params:{count:i}});e-=i,s=t?this.startMetaQuery().withLaterData(e):this.startMetaQuery().withEarlierData(e);let n=this.getMeta(s.build());return t||(n=n.then((e=>{e&&e.params&&!e.params.count&&(this._noEarlierMsgs=!0)}))),n}))}setMeta(e){return e.tags&&(e.tags=function(e){let t=[];if(Array.isArray(e)){for(let s=0,i=e.length;s1&&t.push(i))}t.sort().filter((function(e,t,s){return!t||e!=s[t-1]}))}return 0==t.length&&t.push(l),t}(e.tags)),this._tinode.setMeta(this.name,e).then((t=>(t&&t.code>=300||(e.sub&&(e.sub.topic=this.name,t.params&&t.params.acs&&(e.sub.acs=t.params.acs,e.sub.updated=t.ts),e.sub.user||(e.sub.user=this._tinode.getCurrentUserID(),e.desc||(e.desc={})),e.sub._noForwarding=!0,this._processMetaSubs([e.sub])),e.desc&&(t.params&&t.params.acs&&(e.desc.acs=t.params.acs,e.desc.updated=t.ts),this._processMetaDesc(e.desc)),e.tags&&this._processMetaTags(e.tags),e.cred&&this._processMetaCreds([e.cred],!0)),t)))}updateMode(e,t){const s=e?this.subscriber(e):null,i=s?s.acs.updateGiven(t).getGiven():this.getAccessMode().updateWant(t).getWant();return this.setMeta({sub:{user:e,mode:i}})}invite(e,t){return this.setMeta({sub:{user:e,mode:t}})}archive(e){return this.private&&!this.private.arch==!e?Promise.resolve(e):this.setMeta({desc:{private:{arch:!!e||l}}})}delMessages(e,t){if(!this._attached)return Promise.reject(new Error("Cannot delete messages in inactive topic"));e.sort(((e,t)=>e.low=t.hi)));let s,i=e.reduce(((e,t)=>(t.low0?this._tinode.delMessages(this.name,i,t):Promise.resolve({params:{del:0}}),s.then((t=>(t.params.del>this._maxDel&&(this._maxDel=t.params.del),e.forEach((e=>{e.hi?this.flushMessageRange(e.low,e.hi):this.flushMessage(e.low)})),this.onData&&this.onData(),t)))}delMessagesAll(e){return!this._maxSeq||this._maxSeq<=0?Promise.resolve():this.delMessages([{low:1,hi:this._maxSeq+1,_all:!0}],e)}delMessagesList(e,t){e.sort(((e,t)=>e-t));let s=e.reduce(((e,t)=>{if(0==e.length)e.push({low:t});else{let s=e[e.length-1];!s.hi&&t!=s.low+1||t>s.hi?e.push({low:t}):s.hi=s.hi?Math.max(s.hi,t+1):t+1}return e}),[]);return this.delMessages(s,t)}delMessagesEdits(e,t){const s=[e];return this.messageVersions(e,(e=>s.push(e.seq))),this.delMessagesList(s,t)}delTopic(e){return this._deleted?(this._gone(),Promise.resolve(null)):this._tinode.delTopic(this.name,e).then((e=>(this._deleted=!0,this._resetSub(),this._gone(),e)))}delSubscription(e){return this._attached?this._tinode.delSubscription(this.name,e).then((t=>(delete this._users[e],this.onSubsUpdated&&this.onSubsUpdated(Object.keys(this._users)),t))):Promise.reject(new Error("Cannot delete subscription in inactive topic"))}note(e,t){if(!this._attached)return;const s=this._users[this._tinode.getCurrentUserID()];let i=!1;if(s?(!s[e]||s[e]0&&this.note("read",e)}noteKeyPress(){this._attached?this._tinode.noteKeyPress(this.name):this._tinode.logger("INFO: Cannot send notification in inactive topic")}noteRecording(e){this._attached?this._tinode.noteKeyPress(this.name,e?"kpa":"kpv"):this._tinode.logger("INFO: Cannot send notification in inactive topic")}videoCall(e,t,s){if(this._attached||["ringing","hang-up"].includes(e))return this._tinode.videoCall(this.name,t,e,s)}_updateMyReadRecv(e,t,s){let i,n=!1;switch(t|=0,this.seq=0|this.seq,this.read=0|this.read,this.recv=0|this.recv,e){case"recv":i=this.recv,this.recv=Math.max(this.recv,t),n=i!=this.recv;break;case"read":i=this.read,this.read=Math.max(this.read,t),n=i!=this.read;break;case"msg":i=this.seq,this.seq=Math.max(this.seq,t),(!this.touched||this.touched{if(this._isReplacementMsg(e))return;const r=this.latestMsgVersion(e.seq)||e;r._origTs||(r._origTs=r.ts,r._origSeq=r.seq,r.ts=e.ts,r.seq=e.seq),t.push({data:r,idx:n})}),e,r,{}),t.forEach(((e,s)=>{n.call(i,e.data,s>0?t[s-1].data:void 0,s=0)return this._messages.getAt(t)}latestMessage(){return this._messages.getLast()}latestMsgVersion(e){const t=this._messageVersions[e];return t?t.getLast():null}maxMsgSeq(){return this._maxSeq}minMsgSeq(){return this._minSeq}maxClearId(){return this._maxDel}messageCount(){return this._messages.length()}queuedMessages(e,t){if(!e)throw new Error("Callback must be provided");this.messages(e,d,void 0,t)}msgReceiptCount(e,t){let s=0;if(t>0){const i=this._tinode.getCurrentUserID();for(let n in this._users){const r=this._users[n];r.user!==i&&r[e]>=t&&s++}}return s}msgReadCount(e){return this.msgReceiptCount("read",e)}msgRecvCount(e){return this.msgReceiptCount("recv",e)}msgHasMoreMessages(e){return e?this.seq>this._maxSeq:this._minSeq>1&&!this._noEarlierMsgs}isNewMessage(e){return this._maxSeq<=e}flushMessage(e){const t=this._messages.find({seq:e});if(delete this._messageVersions[e],t>=0)return this._tinode._db.remMessages(this.name,e),this._messages.delAt(t)}flushMessageRange(e,t){this._tinode._db.remMessages(this.name,e,t);for(let s=e;s=0?this._messages.delRange(s,this._messages.find({seq:t},!0)):[]}swapMessageId(e,t){const s=this._messages.find(e),i=this._messages.length();0<=s&&s=0){const s=this._messages.getAt(t),i=this.msgStatus(s);if(10==i||30==i||40==i)return this._tinode._db.remMessages(this.name,e),s._cancelled=!0,this._messages.delAt(t),this.onData&&this.onData(),!0}return!1}getType(){return j.topicType(this.name)}getAccessMode(){return this.acs}setAccessMode(t){return this.acs=new e(t)}getDefaultAccess(){return this.defacs}startMetaQuery(){return new U(this)}isArchived(){return this.private&&!!this.private.arch}isMeType(){return j.isMeTopicName(this.name)}isChannelType(){return j.isChannelTopicName(this.name)}isGroupType(){return j.isGroupTopicName(this.name)}isP2PType(){return j.isP2PTopicName(this.name)}isCommType(){return j.isCommTopicName(this.name)}msgStatus(e,t){let s=0;return this._tinode.isMe(e.from)?e._sending?s=20:e._fatal||e._cancelled?s=40:e._failed?s=30:e.seq>=d?s=10:this.msgReadCount(e.seq)>0?s=70:this.msgRecvCount(e.seq)>0?s=60:e.seq>0&&(s=50):s=80,t&&e._status!=s&&(e._status=s,this._tinode._db.updMessageStatus(this.name,e.seq,s)),s}_isReplacementMsg(e){return e.head&&e.head.replace}_maybeUpdateMessageVersionsCache(e){if(!this._isReplacementMsg(e))return void(this._messageVersions[e.seq]&&(this._messageVersions[e.seq].filter((t=>t.from==e.from)),this._messageVersions[e.seq].isEmpty()&&delete this._messageVersions[e.seq]));const t=parseInt(e.head.replace.split(":")[1]);if(t>e.seq)return;const s=this.findMessage(t);if(s&&s.from!=e.from)return;const i=this._messageVersions[t]||new O(((e,t)=>e.seq-t.seq),!0);i.put(e),this._messageVersions[t]=i}_routeData(e){e.content&&(!this.touched||this.touchedthis._maxSeq&&(this._maxSeq=e.seq,this.msgStatus(e,!0),clearTimeout(this._recvNotificationTimer),this._recvNotificationTimer=setTimeout((e=>{this._recvNotificationTimer=null,this.noteRecv(this._maxSeq)}),100)),(e.seq0&&this._processMetaSubs(e.sub),e.del&&this._processDelMessages(e.del.clear,e.del.delseq),e.tags&&this._processMetaTags(e.tags),e.cred&&this._processMetaCreds(e.cred),this.onMeta&&this.onMeta(e)}_routePres(t){let s,i;switch(t.what){case"del":this._processDelMessages(t.clear,t.delseq);break;case"on":case"off":s=this._users[t.src],s?s.online="on"==t.what:this._tinode.logger("WARNING: Presence update for an unknown user",this.name,t.src);break;case"term":this._resetSub();break;case"upd":t.src&&!this._tinode.isTopicCached(t.src)&&this.getMeta(this.startMetaQuery().withOneSub(void 0,t.src).build());break;case"acs":if(i=t.src||this._tinode.getCurrentUserID(),s=this._users[i],s)s.acs.updateAll(t.dacs),this._processMetaSubs([{user:i,updated:new Date,acs:s.acs}]);else{const n=(new e).updateAll(t.dacs);n&&n.mode!=e._NONE&&(s=this._cacheGetUser(i),s?s.acs=n:(s={user:i,acs:n},this.getMeta(this.startMetaQuery().withOneSub(void 0,i).build())),s.updated=new Date,this._processMetaSubs([s]))}break;default:this._tinode.logger("INFO: Ignored presence update",t.what)}this.onPres&&this.onPres(t)}_routeInfo(e){switch(e.what){case"recv":case"read":const t=this._users[e.from];t&&(t[e.what]=e.seq,t.recv0&&this.onData&&this.onData()}_allMessagesReceived(e){this.onAllMessagesReceived&&this.onAllMessagesReceived(e)}_resetSub(){this._attached=!1}_gone(){this._messages.reset(),this._tinode._db.remMessages(this.name),this._users={},this.acs=new e(null),this.private=null,this.public=null,this.trusted=null,this._maxSeq=0,this._minSeq=0,this._attached=!1;const t=this._tinode.getMeTopic();t&&t._routePres({_noForwarding:!0,what:"gone",topic:o,src:this.name}),this.onDeleteTopic&&this.onDeleteTopic()}_updateCachedUser(e,t){let s=this._cacheGetUser(e);return s=m(s||{},t),this._cachePutUser(e,s),_(this._users,e,s)}_getQueuedSeqId(){return this._queuedSeqId++}_loadMessages(e,t){const{since:s,before:i,limit:n}=t||{};return e.readMessages(this.name,{since:s,before:i,limit:n||24}).then((e=>(e.forEach((e=>{e.seq>this._maxSeq&&(this._maxSeq=e.seq),(e.seq{e.online&&(e.online=!1,e.seen=Object.assign(e.seen||{},{when:new Date}),this._refreshContact("off",e))})),this.onMetaDesc&&this.onMetaDesc(this)}_processMetaSubs(e){let t=0;if(e.forEach((e=>{const s=e.topic;if(s==c||s==o)return;e.online=!!e.online;let i=null;if(e.deleted)i=e,this._tinode.cacheRemTopic(s),this._tinode._db.remTopic(s);else{void 0!==e.seq&&(e.seq=0|e.seq,e.recv=0|e.recv,e.read=0|e.read,e.unread=e.seq-e.read);const t=this._tinode.getTopic(s);t._new&&delete t._new,i=m(t,e),this._tinode._db.updTopic(i),j.isP2PTopicName(s)&&(this._cachePutUser(s,i),this._tinode._db.updUser(s,i.public)),!e._noForwarding&&t&&(e._noForwarding=!0,t._processMetaDesc(e))}t++,this.onMetaSub&&this.onMetaSub(i)})),this.onSubsUpdated&&t>0){const s=[];e.forEach((e=>{s.push(e.topic)})),this.onSubsUpdated(s,t)}}_processMetaCreds(e,t){1==e.length&&e[0]==l&&(e=[]),t?e.forEach((e=>{if(e.val){let t=this._credentials.findIndex((t=>t.meth==e.meth&&t.val==e.val));t<0?(e.done||(t=this._credentials.findIndex((t=>t.meth==e.meth&&!t.done)),t>=0&&this._credentials.splice(t,1)),this._credentials.push(e)):this._credentials[t].done=e.done}else if(e.resp){const t=this._credentials.findIndex((t=>t.meth==e.meth&&!t.done));t>=0&&(this._credentials[t].done=!0)}})):this._credentials=e,this.onCredsUpdated&&this.onCredsUpdated(this._credentials)}_routePres(t){if("term"==t.what)return void this._resetSub();if("upd"==t.what&&t.src==o)return void this.getMeta(this.startMetaQuery().withDesc().build());const s=this._tinode.cacheGetTopic(t.src);if(s){switch(t.what){case"on":s.online=!0;break;case"off":s.online&&(s.online=!1,s.seen=Object.assign(s.seen||{},{when:new Date}));break;case"msg":s._updateReceived(t.seq,t.act);break;case"upd":this.getMeta(this.startMetaQuery().withLaterOneSub(t.src).build());break;case"acs":t.tgt||(s.acs?s.acs.updateAll(t.dacs):s.acs=(new e).updateAll(t.dacs),s.touched=new Date);break;case"ua":s.seen={when:new Date,ua:t.ua};break;case"recv":t.seq=0|t.seq,s.recv=s.recv?Math.max(s.recv,t.seq):t.seq;break;case"read":t.seq=0|t.seq,s.read=s.read?Math.max(s.read,t.seq):t.seq,s.recv=s.recv?Math.max(s.read,s.recv):s.recv,s.unread=s.seq-s.read;break;case"gone":this._tinode.cacheRemTopic(t.src),s._deleted?this._tinode._db.remTopic(t.src):(s._deleted=!0,s._attached=!1,this._tinode._db.markTopicAsDeleted(t.src,!0));break;case"del":break;default:this._tinode.logger("INFO: Unsupported presence update in 'me'",t.what)}this._refreshContact(t.what,s)}else{if("acs"==t.what){const s=new e(t.dacs);if(!s||s.mode==e._INVALID)return void this._tinode.logger("ERROR: Invalid access mode update",t.src,t.dacs);if(s.mode==e._NONE)return void this._tinode.logger("WARNING: Removing non-existent subscription",t.src,t.dacs);{this.getMeta(this.startMetaQuery().withOneSub(void 0,t.src).build());const e=this._tinode.getTopic(t.src);e.online=!1,e.acs=s,this._tinode._db.updTopic(e)}}else if("tags"==t.what)this.getMeta(this.startMetaQuery().withTags().build());else if("msg"==t.what){this.getMeta(this.startMetaQuery().withOneSub(void 0,t.src).build());const e=this._tinode.getTopic(t.src);e._deleted=!1,this._tinode._db.updTopic(e)}this._refreshContact(t.what,s)}this.onPres&&this.onPres(t)}_refreshContact(e,t){this.onContactUpdate&&this.onContactUpdate(e,t)}publish(){return Promise.reject(new Error("Publishing to 'me' is not supported"))}delCredential(e,t){return this._attached?this._tinode.delCredential(e,t).then((s=>{const i=this._credentials.findIndex((s=>s.meth==e&&s.val==t));return i>-1&&this._credentials.splice(i,1),this.onCredsUpdated&&this.onCredsUpdated(this._credentials),s})):Promise.reject(new Error("Cannot delete credential in inactive 'me' topic"))}contacts(e,t,s){this._tinode.mapTopics(((i,n)=>{!i.isCommType()||t&&!t(i)||e.call(s,i,n)}))}getContact(e){return this._tinode.cacheGetTopic(e)}getAccessMode(e){if(e){const t=this._tinode.cacheGetTopic(e);return t?t.acs:null}return this.acs}isArchived(e){const t=this._tinode.cacheGetTopic(e);return t&&t.private&&!!t.private.arch}getCredentials(){return this._credentials}}class V extends j{_contacts={};constructor(e){super(c,e)}_processMetaSubs(e){let t=Object.getOwnPropertyNames(this._contacts).length;this._contacts={};for(let s in e){let i=e[s];const n=i.topic?i.topic:i.user;i=_(this._contacts,n,i),t++,this.onMetaSub&&this.onMetaSub(i)}t>0&&this.onSubsUpdated&&this.onSubsUpdated(Object.keys(this._contacts))}publish(){return Promise.reject(new Error("Publishing to 'fnd' is not supported"))}setMeta(e){return Object.getPrototypeOf(V.prototype).setMeta.call(this,e).then((e=>{Object.keys(this._contacts).length>0&&(this._contacts={},this.onSubsUpdated&&this.onSubsUpdated([]))}))}contacts(e,t){const s=e||this.onMetaSub;if(s)for(let e in this._contacts)s.call(t,this._contacts[e],e,this._contacts)}}function G(e){return btoa(encodeURIComponent(e).replace(/%([0-9A-F]{2})/g,(function(e,t){return String.fromCharCode("0x"+t)})))}function z(t,s){if(s instanceof Date)s=function(e){if(!f(e))return;const t=function(e,t){return"0".repeat((t=t||2)-(""+e).length)+e},s=e.getUTCMilliseconds();return e.getUTCFullYear()+"-"+t(e.getUTCMonth()+1)+"-"+t(e.getUTCDate())+"T"+t(e.getUTCHours())+":"+t(e.getUTCMinutes())+":"+t(e.getUTCSeconds())+(s?"."+t(s,3):"")+"Z"}(s);else if(s instanceof e)s=s.jsonHelper();else if(null==s||!1===s||Array.isArray(s)&&0==s.length||"object"==typeof s&&0==Object.keys(s).length)return;return s}function F(e,t){return"string"==typeof t&&t.length>128?"<"+t.length+", bytes: "+t.substring(0,12)+"..."+t.substring(t.length-12)+">":z(0,t)}"undefined"!=typeof WebSocket&&(C=WebSocket),"undefined"!=typeof XMLHttpRequest&&(N=XMLHttpRequest),"undefined"!=typeof indexedDB&&(q=indexedDB),function(){const e="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";"undefined"==typeof btoa&&(s.g.btoa=function(){let t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:"",s="";for(let i,n=0,r=0,a=e;t.charAt(0|r)||(a="=",r%1);s+=a.charAt(63&n>>8-r%1*8)){if(i=t.charCodeAt(r+=3/4),i>255)throw new Error("'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.");n=n<<8|i}return s});"undefined"==typeof atob&&(s.g.atob=function(){let t=(arguments.length>0&&void 0!==arguments[0]?arguments[0]:"").replace(/=+$/,""),s="";if(t.length%4==1)throw new Error("'atob' failed: The string to be decoded is not correctly encoded.");for(let i,n=0,r=0,a=0;i=t.charAt(a++);~i&&(r=n%4?64*r+i:i,n++%4)?s+=String.fromCharCode(255&r>>(-2*n&6)):0)i=e.indexOf(i);return s});"undefined"==typeof window&&(s.g.window={WebSocket:C,XMLHttpRequest:N,indexedDB:q,URL:{createObjectURL:function(){throw new Error("Unable to use URL.createObjectURL in a non-browser application")}}});S.setNetworkProviders(C,N),I.setNetworkProvider(N),E.setDatabaseProvider(q)}();class W{_host;_secure;_appName;_apiKey;_browser="";_platform;_hwos="undefined";_humanLanguage="xx";_loggingEnabled=!1;_trimLongStrings=!1;_myUID=null;_authenticated=!1;_login=null;_authToken=null;_inPacketCount=0;_messageId=Math.floor(65535*Math.random()+65535);_serverInfo=null;_deviceToken=null;_pendingPromises={};_expirePromises=null;_connection=null;_persist=!1;_db=null;_cache={};constructor(e,t){if(this._host=e.host,this._secure=e.secure,this._appName=e.appName||"Undefined",this._apiKey=e.apiKey,this._platform=e.platform||"web","undefined"!=typeof navigator&&(this._browser=function(e,t){e=e||"";let s,i="";/reactnative/i.test(t)&&(i="ReactNative; ");let n=(e=e.replace(" (KHTML, like Gecko)","")).match(/(AppleWebKit\/[.\d]+)/i);if(n){const t=["edg","chrome","safari","mobile","version"];let i,r=e.substr(n.index+n[0].length).split(" "),a=[];for(let e=0;es[1].toLowerCase().startsWith(e)))]),"Version"==s[1]&&(i=s[2]))}a.sort(((e,t)=>e[2]-t[2])),a.length>0?(a[0][0].toLowerCase().startsWith("edg")?a[0][0]="Edge":"OPR"==a[0][0]?a[0][0]="Opera":"Safari"==a[0][0]&&i&&(a[0][1]=i),s=a[0][0]+"/"+a[0][1]):s=n[1]}else/firefox/i.test(e)?(n=/Firefox\/([.\d]+)/g.exec(e),s=n?"Firefox/"+n[1]:"Firefox/?"):(n=/([\w.]+)\/([.\d]+)/.exec(e),n?s=n[1]+"/"+n[2]:(n=e.split(" "),s=n[0]));if(n=s.split("/"),n.length>1){const e=n[1].split("."),t=e[1]?"."+e[1].substr(0,2):"";s=`${n[0]}/${e[0]}${t}`}return i+s}(navigator.userAgent,navigator.product),this._hwos=navigator.platform,this._humanLanguage=navigator.language||"en-US"),S.logger=this.logger,A().logger=this.logger,"lp"!=e.transport&&"ws"!=e.transport&&(e.transport=function(){if("object"==typeof window){if(window.WebSocket)return"ws";if(window.XMLHttpRequest)return"lp"}return null}()),this._connection=new S(e,"0",!0),this._connection.onMessage=e=>{this.#S(e)},this._connection.onOpen=e=>this.#x(),this._connection.onDisconnect=(e,t)=>this.#k(e,t),this._connection.onAutoreconnectIteration=(e,t)=>{this.onAutoreconnectIteration&&this.onAutoreconnectIteration(e,t)},this._persist=e.persist,this._db=new E((e=>{this.logger("DB",e)}),this.logger),this._persist){const e=[];this._db.initDatabase().then((t=>this._db.mapTopics((t=>{let s=this.#E("topic",t.name);s||(s=t.name==o?new L:t.name==c?new V:new j(t.name),this._db.deserializeTopic(s,t),this.#D(s),s._cachePutSelf(),delete s._new,e.push(s._loadMessages(this._db)))})))).then((e=>this._db.mapUsers((e=>{this.#A("user",e.uid,m({},e.public))})))).then((t=>Promise.all(e))).then((e=>{t&&t(),this.logger("Persistent cache initialized.")})).catch((e=>{t&&t(e),this.logger("Failed to initialize persistent cache:",e)}))}else this._db.deleteDatabase().then((e=>{t&&t()}))}logger(e){if(this._loggingEnabled){const n=new Date,r=("0"+n.getUTCHours()).slice(-2)+":"+("0"+n.getUTCMinutes()).slice(-2)+":"+("0"+n.getUTCSeconds()).slice(-2)+"."+("00"+n.getUTCMilliseconds()).slice(-3);for(var t=arguments.length,s=new Array(t>1?t-1:0),i=1;i{this._pendingPromises[e]={resolve:t,reject:s,ts:new Date}}))),t}#C(e,t,s,i){const n=this._pendingPromises[e];n&&(delete this._pendingPromises[e],t>=200&&t<400?n.resolve&&n.resolve(s):n.reject&&n.reject(new u(i,t)))}#N(e,t){let s;t&&(s=this.#R(t)),e=b(e);let i=JSON.stringify(e);this.logger("out: "+(this._trimLongStrings?JSON.stringify(e,F):i));try{this._connection.sendText(i)}catch(e){if(!t)throw e;this.#C(t,S.NETWORK_ERROR,null,e.message)}return s}#S(e){if(!e)return;if(this._inPacketCount++,this.onRawMessage&&this.onRawMessage(e),"0"===e)return void(this.onNetworkProbe&&this.onNetworkProbe());let t=JSON.parse(e,p);t?(this.logger("in: "+(this._trimLongStrings?JSON.stringify(t,F):e)),this.onMessage&&this.onMessage(t),t.ctrl?(this.onCtrlMessage&&this.onCtrlMessage(t.ctrl),t.ctrl.id&&this.#C(t.ctrl.id,t.ctrl.code,t.ctrl,t.ctrl.text),setTimeout((e=>{if(205==t.ctrl.code&&"evicted"==t.ctrl.text){const e=this.#E("topic",t.ctrl.topic);e&&(e._resetSub(),t.ctrl.params&&t.ctrl.params.unsub&&e._gone())}else if(t.ctrl.code<300&&t.ctrl.params)if("data"==t.ctrl.params.what){const e=this.#E("topic",t.ctrl.topic);e&&e._allMessagesReceived(t.ctrl.params.count)}else if("sub"==t.ctrl.params.what){const e=this.#E("topic",t.ctrl.topic);e&&e._processMetaSubs([])}}),0)):setTimeout((e=>{if(t.meta){const e=this.#E("topic",t.meta.topic);e&&e._routeMeta(t.meta),t.meta.id&&this.#C(t.meta.id,200,t.meta,"META"),this.onMetaMessage&&this.onMetaMessage(t.meta)}else if(t.data){const e=this.#E("topic",t.data.topic);e&&e._routeData(t.data),this.onDataMessage&&this.onDataMessage(t.data)}else if(t.pres){const e=this.#E("topic",t.pres.topic);e&&e._routePres(t.pres),this.onPresMessage&&this.onPresMessage(t.pres)}else if(t.info){const e=this.#E("topic",t.info.topic);e&&e._routeInfo(t.info),this.onInfoMessage&&this.onInfoMessage(t.info)}else this.logger("ERROR: Unknown packet received.")}),0)):(this.logger("in: "+e),this.logger("ERROR: failed to parse data"))}#x(){this._expirePromises||(this._expirePromises=setInterval((e=>{const t=new u("timeout",504),s=new Date((new Date).getTime()-5e3);for(let e in this._pendingPromises){let i=this._pendingPromises[e];i&&i.ts{e._resetSub()}));for(let t in this._pendingPromises){const s=this._pendingPromises[t];s&&s.reject&&s.reject(e)}this._pendingPromises={},this.onDisconnect&&this.onDisconnect(e)}#I(){return this._appName+" ("+(this._browser?this._browser+"; ":"")+this._hwos+"); "+n}#U(e,s){switch(e){case"hi":return{hi:{id:this.getNextUniqueId(),ver:t,ua:this.#I(),dev:this._deviceToken,lang:this._humanLanguage,platf:this._platform}};case"acc":return{acc:{id:this.getNextUniqueId(),user:null,scheme:null,secret:null,tmpscheme:null,tmpsecret:null,login:!1,tags:null,desc:{},cred:{}}};case"login":return{login:{id:this.getNextUniqueId(),scheme:null,secret:null}};case"sub":return{sub:{id:this.getNextUniqueId(),topic:s,set:{},get:{}}};case"leave":return{leave:{id:this.getNextUniqueId(),topic:s,unsub:!1}};case"pub":return{pub:{id:this.getNextUniqueId(),topic:s,noecho:!1,head:null,content:{}}};case"get":return{get:{id:this.getNextUniqueId(),topic:s,what:null,desc:{},sub:{},data:{}}};case"set":return{set:{id:this.getNextUniqueId(),topic:s,desc:{},sub:{},tags:[],ephemeral:{}}};case"del":return{del:{id:this.getNextUniqueId(),topic:s,what:null,delseq:null,user:null,hard:!1}};case"note":return{note:{topic:s,what:null,seq:void 0}};default:throw new Error(`Unknown packet type requested: ${e}`)}}#A(e,t,s){this._cache[e+":"+t]=s}#E(e,t){return this._cache[e+":"+t]}#O(e,t){delete this._cache[e+":"+t]}#q(e,t,s){const i=e?e+":":void 0;for(let e in this._cache)if((!i||0==e.indexOf(i))&&t.call(s,this._cache[e],e))break}#D(e){e._tinode=this,e._cacheGetUser=e=>{const t=this.#E("user",e);if(t)return{user:e,public:m({},t)}},e._cachePutUser=(e,t)=>{this.#A("user",e,m({},t.public))},e._cacheDelUser=e=>{this.#O("user",e)},e._cachePutSelf=t=>{this.#A("topic",e.name,e)},e._cacheDelSelf=t=>{this.#O("topic",e.name)}}#j(e){return e.params&&e.params.user?(this._myUID=e.params.user,this._authenticated=e&&e.code>=200&&e.code<300,e.params&&e.params.token&&e.params.expires?this._authToken={token:e.params.token,expires:e.params.expires}:this._authToken=null,this.onLogin&&this.onLogin(e.code,e.text),e):e}static credential(e,t,s,i){return"object"==typeof e&&({val:t,params:s,resp:i,meth:e}=e),e&&(t||i)?[{meth:e,val:t,resp:i,params:s}]:null}static topicType(e){return j.topicType(e)}static isMeTopicName(e){return j.isMeTopicName(e)}static isGroupTopicName(e){return j.isGroupTopicName(e)}static isP2PTopicName(e){return j.isP2PTopicName(e)}static isCommTopicName(e){return j.isCommTopicName(e)}static isNewGroupTopicName(e){return j.isNewGroupTopicName(e)}static isChannelTopicName(e){return j.isChannelTopicName(e)}static getVersion(){return t}static setNetworkProviders(e,t){C=e,N=t,S.setNetworkProviders(C,N),I.setNetworkProvider(N)}static setDatabaseProvider(e){q=e,E.setDatabaseProvider(q)}static getLibrary(){return n}static isNullValue(e){return e===l}getNextUniqueId(){return 0!=this._messageId?""+this._messageId++:void 0}connect(e){return this._connection.connect(e)}reconnect(e){this._connection.reconnect(e)}disconnect(){this._connection.disconnect()}clearStorage(){return this._db.isReady()?this._db.deleteDatabase():Promise.resolve()}initStorage(){return this._db.isReady()?Promise.resolve():this._db.initDatabase()}networkProbe(){this._connection.probe()}isConnected(){return this._connection.isConnected()}isAuthenticated(){return this._authenticated}authorizeURL(e){if("string"!=typeof e)return e;if(g(e)){const t="scheme://host/",s=new URL(e,t);this._apiKey&&s.searchParams.append("apikey",this._apiKey),this._authToken&&this._authToken.token&&(s.searchParams.append("auth","token"),s.searchParams.append("secret",this._authToken.token)),e=s.toString().substring(t.length-1)}return e}account(e,t,s,i,n){const r=this.#U("acc");return r.acc.user=e,r.acc.scheme=t,r.acc.secret=s,r.acc.login=i,n&&(r.acc.desc.defacs=n.defacs,r.acc.desc.public=n.public,r.acc.desc.private=n.private,r.acc.desc.trusted=n.trusted,r.acc.tags=n.tags,r.acc.cred=n.cred,r.acc.tmpscheme=n.scheme,r.acc.tmpsecret=n.secret,Array.isArray(n.attachments)&&n.attachments.length>0&&(r.extra={attachments:n.attachments.filter((e=>g(e)))})),this.#N(r,r.acc.id)}createAccount(e,t,s,i){let n=this.account("new",e,t,s,i);return s&&(n=n.then((e=>this.#j(e)))),n}createAccountBasic(e,t,s){return e=e||"",t=t||"",this.createAccount("basic",G(e+":"+t),!0,s)}updateAccountBasic(e,t,s,i){return t=t||"",s=s||"",this.account(e,"basic",G(t+":"+s),!1,i)}hello(){const e=this.#U("hi");return this.#N(e,e.hi.id).then((e=>(this._connection.backoffReset(),e.params&&(this._serverInfo=e.params),this.onConnect&&this.onConnect(),e))).catch((e=>{this._connection.reconnect(!0),this.onDisconnect&&this.onDisconnect(e)}))}setDeviceToken(e){let t=!1;return(e=e||null)!=this._deviceToken&&(this._deviceToken=e,this.isConnected()&&this.isAuthenticated()&&(this.#N({hi:{dev:e||W.DEL_CHAR}}),t=!0)),t}login(e,t,s){const i=this.#U("login");return i.login.scheme=e,i.login.secret=t,i.login.cred=s,this.#N(i,i.login.id).then((e=>this.#j(e)))}loginBasic(e,t,s){return this.login("basic",G(e+":"+t),s).then((t=>(this._login=e,t)))}loginToken(e,t){return this.login("token",e,t)}requestResetAuthSecret(e,t,s){return this.login("reset",G(e+":"+t+":"+s))}getAuthToken(){return this._authToken&&this._authToken.expires.getTime()>Date.now()?this._authToken:(this._authToken=null,null)}setAuthToken(e){this._authToken=e}subscribe(e,t,s){const i=this.#U("sub",e);if(e||(e=r),i.sub.get=t,s){if(s.sub&&(i.sub.set.sub=s.sub),s.desc){const t=s.desc;W.isNewGroupTopicName(e)?i.sub.set.desc=t:W.isP2PTopicName(e)&&t.defacs&&(i.sub.set.desc={defacs:t.defacs})}Array.isArray(s.attachments)&&s.attachments.length>0&&(i.extra={attachments:s.attachments.filter((e=>g(e)))}),s.tags&&(i.sub.set.tags=s.tags)}return this.#N(i,i.sub.id)}leave(e,t){const s=this.#U("leave",e);return s.leave.unsub=t,this.#N(s,s.leave.id)}createMessage(e,t,s){const i=this.#U("pub",e);let n="string"==typeof t?A().parse(t):t;return n&&!A().isPlainText(n)&&(i.pub.head={mime:A().getContentType()},t=n),i.pub.noecho=s,i.pub.content=t,i.pub}publish(e,t,s){return this.publishMessage(this.createMessage(e,t,s))}publishMessage(e,t){(e=Object.assign({},e)).seq=void 0,e.from=void 0,e.ts=void 0;const s={pub:e};return t&&(s.extra={attachments:t.filter((e=>g(e)))}),this.#N(s,e.id)}oobNotification(t){switch(this.logger("oob: "+(this._trimLongStrings?JSON.stringify(t,F):t)),t.what){case"msg":if(!t.seq||t.seq<1||!t.topic)break;if(!this.isConnected())break;const s=this.#E("topic",t.topic);if(!s)break;if(s.isSubscribed())break;s.maxMsgSeq(){this.logger("Failed to get the name of a new sender",e)})),s.subscribe(null).then((e=>s.getMeta(new U(s).withLaterData(24).withLaterDel(24).build()))).then((e=>{s.leaveDelayed(!1,1e3)})).catch((e=>{this.logger("On push data fetch failed",e)})).finally((e=>{this.getMeTopic()._refreshContact("msg",s)})));break;case"read":this.getMeTopic()._routePres({what:"read",seq:t.seq});break;case"sub":if(!this.isMe(t.xfrom))break;const i={given:t.modeGiven,want:t.modeWant},n=new e(i),r=n.mode&&n.mode!=e._NONE?{what:"acs",src:t.topic,dacs:i}:{what:"gone",src:t.topic};this.getMeTopic()._routePres(r);break;default:this.logger("Unknown push type ignored",t.what)}}getMeta(e,t){const s=this.#U("get",e);return s.get=m(s.get,t),this.#N(s,s.get.id)}setMeta(e,t){const s=this.#U("set",e),i=[];return t&&(["desc","sub","tags","cred","ephemeral"].forEach((function(e){t.hasOwnProperty(e)&&(i.push(e),s.set[e]=t[e])})),Array.isArray(t.attachments)&&t.attachments.length>0&&(s.extra={attachments:t.attachments.filter((e=>g(e)))})),0==i.length?Promise.reject(new Error("Invalid {set} parameters")):this.#N(s,s.set.id)}delMessages(e,t,s){const i=this.#U("del",e);return i.del.what="msg",i.del.delseq=t,i.del.hard=s,this.#N(i,i.del.id)}delTopic(e,t){const s=this.#U("del",e);return s.del.what="topic",s.del.hard=t,this.#N(s,s.del.id)}delSubscription(e,t){const s=this.#U("del",e);return s.del.what="sub",s.del.user=t,this.#N(s,s.del.id)}delCredential(e,t){const s=this.#U("del",o);return s.del.what="cred",s.del.cred={meth:e,val:t},this.#N(s,s.del.id)}delCurrentUser(e){const t=this.#U("del",null);return t.del.what="user",t.del.hard=e,this.#N(t,t.del.id).then((e=>{this._myUID=null}))}note(e,t,s){if(s<=0||s>=d)throw new Error(`Invalid message id ${s}`);const i=this.#U("note",e);i.note.what=t,i.note.seq=s,this.#N(i)}noteKeyPress(e,t){const s=this.#U("note",e);s.note.what=t||"kp",this.#N(s)}videoCall(e,t,s,i){const n=this.#U("note",e);n.note.seq=t,n.note.what="call",n.note.event=s,n.note.payload=i,this.#N(n,n.note.id)}getTopic(e){let t=this.#E("topic",e);return!t&&e&&(t=e==o?new L:e==c?new V:new j(e),this.#D(t),t._cachePutSelf()),t}cacheGetTopic(e){return this.#E("topic",e)}cacheRemTopic(e){this.#O("topic",e)}mapTopics(e,t){this.#q("topic",e,t)}isTopicCached(e){return!!this.#E("topic",e)}newGroupTopicName(e){return(e?a:r)+this.getNextUniqueId()}getMeTopic(){return this.getTopic(o)}getFndTopic(){return this.getTopic(c)}getLargeFileHelper(){return new I(this,"0")}getCurrentUserID(){return this._myUID}isMe(e){return this._myUID===e}getCurrentLogin(){return this._login}getServerInfo(){return this._serverInfo}report(e,t){return this.publish("sys",A().attachJSON(null,{action:e,target:t}))}getServerParam(e,t){return this._serverInfo&&this._serverInfo[e]||t}enableLogging(e,t){this._loggingEnabled=e,this._trimLongStrings=e&&t}setHumanLanguage(e){e&&(this._humanLanguage=e)}isTopicOnline(e){const t=this.#E("topic",e);return t&&t.online}getTopicAccessMode(e){const t=this.#E("topic",e);return t?t.acs:null}wantAkn(e){this._messageId=e?Math.floor(16777215*Math.random()+16777215):0}onWebsocketOpen=void 0;onConnect=void 0;onDisconnect=void 0;onLogin=void 0;onCtrlMessage=void 0;onDataMessage=void 0;onPresMessage=void 0;onMessage=void 0;onRawMessage=void 0;onNetworkProbe=void 0;onAutoreconnectIteration=void 0}W.MESSAGE_STATUS_NONE=0,W.MESSAGE_STATUS_QUEUED=10,W.MESSAGE_STATUS_SENDING=20,W.MESSAGE_STATUS_FAILED=30,W.MESSAGE_STATUS_FATAL=40,W.MESSAGE_STATUS_SENT=50,W.MESSAGE_STATUS_RECEIVED=60,W.MESSAGE_STATUS_READ=70,W.MESSAGE_STATUS_TO_ME=80,W.DEL_CHAR=l,W.MAX_MESSAGE_SIZE="maxMessageSize",W.MAX_SUBSCRIBER_COUNT="maxSubscriberCount",W.MAX_TAG_COUNT="maxTagCount",W.MAX_FILE_UPLOAD_SIZE="maxFileUploadSize",W.REQ_CRED_VALIDATORS="reqCred",W.URI_TOPIC_ID_PREFIX="tinode:topic/"})(),i})()));
//# sourceMappingURL=tinode.prod.js.map
\ No newline at end of file
diff --git a/umd/tinode.prod.js.map b/umd/tinode.prod.js.map
index 8f39858..01774eb 100644
--- a/umd/tinode.prod.js.map
+++ b/umd/tinode.prod.js.map
@@ -1 +1 @@
-{"version":3,"file":"tinode.prod.js","mappings":"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","sources":["webpack://tinode/webpack/universalModuleDefinition","webpack://tinode/./src/drafty.js","webpack://tinode/webpack/bootstrap","webpack://tinode/webpack/runtime/compat get default export","webpack://tinode/webpack/runtime/define property getters","webpack://tinode/webpack/runtime/global","webpack://tinode/webpack/runtime/hasOwnProperty shorthand","webpack://tinode/webpack/runtime/make namespace object","webpack://tinode/./src/access-mode.js","webpack://tinode/./version.js","webpack://tinode/./src/config.js","webpack://tinode/./src/comm-error.js","webpack://tinode/./src/utils.js","webpack://tinode/./src/connection.js","webpack://tinode/./src/db.js","webpack://tinode/./src/large-file.js","webpack://tinode/./src/tinode.js","webpack://tinode/./src/meta-builder.js","webpack://tinode/./src/cbuffer.js","webpack://tinode/./src/topic.js"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"tinode\"] = factory();\n\telse\n\t\troot[\"tinode\"] = factory();\n})(this, () => {\nreturn ","/**\n * @copyright 2015-2022 Tinode LLC.\n * @summary Minimally rich text representation and formatting for Tinode.\n * @license Apache 2.0\n *\n * @file Basic parser and formatter for very simple text markup. Mostly targeted at\n * mobile use cases similar to Telegram, WhatsApp, and FB Messenger.\n *\n * Supports conversion of user keyboard input to formatted text:
\n * \n * - *abc* → abc
\n * - _abc_ → abc
\n * - ~abc~ →
abc \n * - `abc` → abc
\n *
\n * Also supports forms and buttons.\n *\n * Nested formatting is supported, e.g. *abc _def_* -> abc def\n * URLs, @mentions, and #hashtags are extracted and converted into links.\n * Forms and buttons can be added procedurally.\n * JSON data representation is inspired by Draft.js raw formatting.\n *\n *\n * @example\n * Text:\n * \n * this is *bold*, `code` and _italic_, ~strike~\n * combined *bold and _italic_*\n * an url: https://www.example.com/abc#fragment and another _www.tinode.co_\n * this is a @mention and a #hashtag in a string\n * second #hashtag\n *
\n *\n * Sample JSON representation of the text above:\n * {\n * \"txt\": \"this is bold, code and italic, strike combined bold and italic an url: https://www.example.com/abc#fragment \" +\n * \"and another www.tinode.co this is a @mention and a #hashtag in a string second #hashtag\",\n * \"fmt\": [\n * { \"at\":8, \"len\":4,\"tp\":\"ST\" },{ \"at\":14, \"len\":4, \"tp\":\"CO\" },{ \"at\":23, \"len\":6, \"tp\":\"EM\"},\n * { \"at\":31, \"len\":6, \"tp\":\"DL\" },{ \"tp\":\"BR\", \"len\":1, \"at\":37 },{ \"at\":56, \"len\":6, \"tp\":\"EM\" },\n * { \"at\":47, \"len\":15, \"tp\":\"ST\" },{ \"tp\":\"BR\", \"len\":1, \"at\":62 },{ \"at\":120, \"len\":13, \"tp\":\"EM\" },\n * { \"at\":71, \"len\":36, \"key\":0 },{ \"at\":120, \"len\":13, \"key\":1 },{ \"tp\":\"BR\", \"len\":1, \"at\":133 },\n * { \"at\":144, \"len\":8, \"key\":2 },{ \"at\":159, \"len\":8, \"key\":3 },{ \"tp\":\"BR\", \"len\":1, \"at\":179 },\n * { \"at\":187, \"len\":8, \"key\":3 },{ \"tp\":\"BR\", \"len\":1, \"at\":195 }\n * ],\n * \"ent\": [\n * { \"tp\":\"LN\", \"data\":{ \"url\":\"https://www.example.com/abc#fragment\" } },\n * { \"tp\":\"LN\", \"data\":{ \"url\":\"http://www.tinode.co\" } },\n * { \"tp\":\"MN\", \"data\":{ \"val\":\"mention\" } },\n * { \"tp\":\"HT\", \"data\":{ \"val\":\"hashtag\" } }\n * ]\n * }\n */\n\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nconst MAX_FORM_ELEMENTS = 8;\nconst MAX_PREVIEW_ATTACHMENTS = 3;\nconst MAX_PREVIEW_DATA_SIZE = 64;\nconst JSON_MIME_TYPE = 'application/json';\nconst DRAFTY_MIME_TYPE = 'text/x-drafty';\nconst ALLOWED_ENT_FIELDS = ['act', 'height', 'duration', 'incoming', 'mime', 'name', 'premime', 'preref', 'preview',\n 'ref', 'size', 'state', 'url', 'val', 'width'\n];\n\n// Regular expressions for parsing inline formats. Javascript does not support lookbehind,\n// so it's a bit messy.\nconst INLINE_STYLES = [\n // Strong = bold, *bold text*\n {\n name: 'ST',\n start: /(?:^|[\\W_])(\\*)[^\\s*]/,\n end: /[^\\s*](\\*)(?=$|[\\W_])/\n },\n // Emphesized = italic, _italic text_\n {\n name: 'EM',\n start: /(?:^|\\W)(_)[^\\s_]/,\n end: /[^\\s_](_)(?=$|\\W)/\n },\n // Deleted, ~strike this though~\n {\n name: 'DL',\n start: /(?:^|[\\W_])(~)[^\\s~]/,\n end: /[^\\s~](~)(?=$|[\\W_])/\n },\n // Code block `this is monospace`\n {\n name: 'CO',\n start: /(?:^|\\W)(`)[^`]/,\n end: /[^`](`)(?=$|\\W)/\n }\n];\n\n// Relative weights of formatting spans. Greater index in array means greater weight.\nconst FMT_WEIGHT = ['QQ'];\n\n// RegExps for entity extraction (RF = reference)\nconst ENTITY_TYPES = [\n // URLs\n {\n name: 'LN',\n dataName: 'url',\n pack: function(val) {\n // Check if the protocol is specified, if not use http\n if (!/^[a-z]+:\\/\\//i.test(val)) {\n val = 'http://' + val;\n }\n return {\n url: val\n };\n },\n re: /(?:(?:https?|ftp):\\/\\/|www\\.|ftp\\.)[-A-Z0-9+&@#\\/%=~_|$?!:,.]*[A-Z0-9+&@#\\/%=~_|$]/ig\n },\n // Mentions @user (must be 2 or more characters)\n {\n name: 'MN',\n dataName: 'val',\n pack: function(val) {\n return {\n val: val.slice(1)\n };\n },\n re: /\\B@([\\p{L}\\p{N}][._\\p{L}\\p{N}]*[\\p{L}\\p{N}])/ug\n },\n // Hashtags #hashtag, like metion 2 or more characters.\n {\n name: 'HT',\n dataName: 'val',\n pack: function(val) {\n return {\n val: val.slice(1)\n };\n },\n re: /\\B#([\\p{L}\\p{N}][._\\p{L}\\p{N}]*[\\p{L}\\p{N}])/ug\n }\n];\n\n// HTML tag name suggestions\nconst FORMAT_TAGS = {\n AU: {\n html_tag: 'audio',\n md_tag: undefined,\n isVoid: false\n },\n BN: {\n html_tag: 'button',\n md_tag: undefined,\n isVoid: false\n },\n BR: {\n html_tag: 'br',\n md_tag: '\\n',\n isVoid: true\n },\n CO: {\n html_tag: 'tt',\n md_tag: '`',\n isVoid: false\n },\n DL: {\n html_tag: 'del',\n md_tag: '~',\n isVoid: false\n },\n EM: {\n html_tag: 'i',\n md_tag: '_',\n isVoid: false\n },\n EX: {\n html_tag: '',\n md_tag: undefined,\n isVoid: true\n },\n FM: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n HD: {\n html_tag: '',\n md_tag: undefined,\n isVoid: false\n },\n HL: {\n html_tag: 'span',\n md_tag: undefined,\n isVoid: false\n },\n HT: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n IM: {\n html_tag: 'img',\n md_tag: undefined,\n isVoid: false\n },\n LN: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n MN: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n RW: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false,\n },\n QQ: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n ST: {\n html_tag: 'b',\n md_tag: '*',\n isVoid: false\n },\n VC: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n VD: {\n html_tag: 'video',\n md_tag: undefined,\n isVoid: false\n }\n};\n\n// Convert base64-encoded string into Blob.\nfunction base64toObjectUrl(b64, contentType, logger) {\n if (!b64) {\n return null;\n }\n\n try {\n const bin = atob(b64);\n const length = bin.length;\n const buf = new ArrayBuffer(length);\n const arr = new Uint8Array(buf);\n for (let i = 0; i < length; i++) {\n arr[i] = bin.charCodeAt(i);\n }\n\n return URL.createObjectURL(new Blob([buf], {\n type: contentType\n }));\n } catch (err) {\n if (logger) {\n logger(\"Drafty: failed to convert object.\", err.message);\n }\n }\n\n return null;\n}\n\nfunction base64toDataUrl(b64, contentType) {\n if (!b64) {\n return null;\n }\n contentType = contentType || 'image/jpeg';\n return 'data:' + contentType + ';base64,' + b64;\n}\n\n// Helpers for converting Drafty to HTML.\nconst DECORATORS = {\n // Visial styles\n ST: {\n open: _ => '',\n close: _ => ''\n },\n EM: {\n open: _ => '',\n close: _ => ''\n },\n DL: {\n open: _ => '',\n close: _ => ''\n },\n CO: {\n open: _ => '',\n close: _ => ''\n },\n // Line break\n BR: {\n open: _ => '
',\n close: _ => ''\n },\n // Hidden element\n HD: {\n open: _ => '',\n close: _ => ''\n },\n // Highlighted element.\n HL: {\n open: _ => '',\n close: _ => ''\n },\n // Link (URL)\n LN: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n href: data.url,\n target: '_blank'\n } : null;\n },\n },\n // Mention\n MN: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n id: data.val\n } : null;\n },\n },\n // Hashtag\n HT: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n id: data.val\n } : null;\n },\n },\n // Button\n BN: {\n open: _ => '',\n props: (data) => {\n return data ? {\n 'data-act': data.act,\n 'data-val': data.val,\n 'data-name': data.name,\n 'data-ref': data.ref\n } : null;\n },\n },\n // Audio recording\n AU: {\n open: (data) => {\n const url = data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger);\n return '',\n props: (data) => {\n if (!data) return null;\n return {\n // Embedded data or external link.\n src: data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n 'data-preload': data.ref ? 'metadata' : 'auto',\n 'data-duration': data.duration,\n 'data-name': data.name,\n 'data-size': data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0),\n 'data-mime': data.mime,\n };\n }\n },\n // Image\n IM: {\n open: data => {\n // Don't use data.ref for preview: it's a security risk.\n const tmpPreviewUrl = base64toDataUrl(data._tempPreview, data.mime);\n const previewUrl = base64toObjectUrl(data.val, data.mime, Drafty.logger);\n const downloadUrl = data.ref || previewUrl;\n return (data.name ? '' : '') +\n '';\n },\n close: data => {\n return (data.name ? '' : '');\n },\n props: data => {\n if (!data) return null;\n return {\n // Temporary preview, or permanent preview, or external link.\n src: base64toDataUrl(data._tempPreview, data.mime) ||\n data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n title: data.name,\n alt: data.name,\n 'data-width': data.width,\n 'data-height': data.height,\n 'data-name': data.name,\n 'data-size': data.ref ? (data.size | 0) : (data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0)),\n 'data-mime': data.mime,\n };\n },\n },\n // Form - structured layout of elements.\n FM: {\n open: _ => '',\n close: _ => ''\n },\n // Row: logic grouping of elements\n RW: {\n open: _ => '',\n close: _ => ''\n },\n // Quoted block.\n QQ: {\n open: _ => '',\n close: _ => '',\n props: (data) => {\n return data ? {} : null;\n },\n },\n // Video call\n VC: {\n open: _ => '',\n close: _ => '',\n props: data => {\n if (!data) return {};\n return {\n 'data-duration': data.duration,\n 'data-state': data.state,\n };\n }\n },\n // Video.\n VD: {\n open: data => {\n const tmpPreviewUrl = base64toDataUrl(data._tempPreview, data.mime);\n const previewUrl = data.ref || base64toObjectUrl(data.preview, data.premime || 'image/jpeg', Drafty.logger);\n return '';\n },\n close: _ => '',\n props: data => {\n if (!data) return null;\n const poster = data.preref || base64toObjectUrl(data.preview, data.premime || 'image/jpeg', Drafty.logger);\n return {\n // Embedded data or external link.\n src: poster,\n 'data-src': data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n 'data-width': data.width,\n 'data-height': data.height,\n 'data-preload': data.ref ? 'metadata' : 'auto',\n 'data-preview': poster,\n 'data-duration': data.duration | 0,\n 'data-name': data.name,\n 'data-size': data.ref ? (data.size | 0) : (data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0)),\n 'data-mime': data.mime,\n };\n }\n },\n};\n\n/**\n * The main object which performs all the formatting actions.\n * @class Drafty\n * @constructor\n */\nconst Drafty = function() {\n this.txt = '';\n this.fmt = [];\n this.ent = [];\n}\n\n/**\n * Initialize Drafty document to a plain text string.\n *\n * @param {string} plainText - string to use as Drafty content.\n *\n * @returns new Drafty document or null is plainText is not a string or undefined.\n */\nDrafty.init = function(plainText) {\n if (typeof plainText == 'undefined') {\n plainText = '';\n } else if (typeof plainText != 'string') {\n return null;\n }\n\n return {\n txt: plainText\n };\n}\n\n/**\n * Parse plain text into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {string} content - plain-text content to parse.\n * @return {Drafty} parsed document or null if the source is not plain text.\n */\nDrafty.parse = function(content) {\n // Make sure we are parsing strings only.\n if (typeof content != 'string') {\n return null;\n }\n\n // Split text into lines. It makes further processing easier.\n const lines = content.split(/\\r?\\n/);\n\n // Holds entities referenced from text\n const entityMap = [];\n const entityIndex = {};\n\n // Processing lines one by one, hold intermediate result in blx.\n const blx = [];\n lines.forEach((line) => {\n let spans = [];\n let entities;\n\n // Find formatted spans in the string.\n // Try to match each style.\n INLINE_STYLES.forEach((tag) => {\n // Each style could be matched multiple times.\n spans = spans.concat(spannify(line, tag.start, tag.end, tag.name));\n });\n\n let block;\n if (spans.length == 0) {\n block = {\n txt: line\n };\n } else {\n // Sort spans by style occurence early -> late, then by length: first long then short.\n spans.sort((a, b) => {\n const diff = a.at - b.at;\n return diff != 0 ? diff : b.end - a.end;\n });\n\n // Convert an array of possibly overlapping spans into a tree.\n spans = toSpanTree(spans);\n\n // Build a tree representation of the entire string, not\n // just the formatted parts.\n const chunks = chunkify(line, 0, line.length, spans);\n\n const drafty = draftify(chunks, 0);\n\n block = {\n txt: drafty.txt,\n fmt: drafty.fmt\n };\n }\n\n // Extract entities from the cleaned up string.\n entities = extractEntities(block.txt);\n if (entities.length > 0) {\n const ranges = [];\n for (let i in entities) {\n // {offset: match['index'], unique: match[0], len: match[0].length, data: ent.packer(), type: ent.name}\n const entity = entities[i];\n let index = entityIndex[entity.unique];\n if (!index) {\n index = entityMap.length;\n entityIndex[entity.unique] = index;\n entityMap.push({\n tp: entity.type,\n data: entity.data\n });\n }\n ranges.push({\n at: entity.offset,\n len: entity.len,\n key: index\n });\n }\n block.ent = ranges;\n }\n\n blx.push(block);\n });\n\n const result = {\n txt: ''\n };\n\n // Merge lines and save line breaks as BR inline formatting.\n if (blx.length > 0) {\n result.txt = blx[0].txt;\n result.fmt = (blx[0].fmt || []).concat(blx[0].ent || []);\n\n for (let i = 1; i < blx.length; i++) {\n const block = blx[i];\n const offset = result.txt.length + 1;\n\n result.fmt.push({\n tp: 'BR',\n len: 1,\n at: offset - 1\n });\n\n result.txt += ' ' + block.txt;\n if (block.fmt) {\n result.fmt = result.fmt.concat(block.fmt.map((s) => {\n s.at += offset;\n return s;\n }));\n }\n if (block.ent) {\n result.fmt = result.fmt.concat(block.ent.map((s) => {\n s.at += offset;\n return s;\n }));\n }\n }\n\n if (result.fmt.length == 0) {\n delete result.fmt;\n }\n\n if (entityMap.length > 0) {\n result.ent = entityMap;\n }\n }\n return result;\n}\n\n/**\n * Append one Drafty document to another.\n *\n * @param {Drafty} first - Drafty document to append to.\n * @param {Drafty|string} second - Drafty document or string being appended.\n *\n * @return {Drafty} first document with the second appended to it.\n */\nDrafty.append = function(first, second) {\n if (!first) {\n return second;\n }\n if (!second) {\n return first;\n }\n\n first.txt = first.txt || '';\n const len = first.txt.length;\n\n if (typeof second == 'string') {\n first.txt += second;\n } else if (second.txt) {\n first.txt += second.txt;\n }\n\n if (Array.isArray(second.fmt)) {\n first.fmt = first.fmt || [];\n if (Array.isArray(second.ent)) {\n first.ent = first.ent || [];\n }\n second.fmt.forEach(src => {\n const fmt = {\n at: (src.at | 0) + len,\n len: src.len | 0\n };\n // Special case for the outside of the normal rendering flow styles.\n if (src.at == -1) {\n fmt.at = -1;\n fmt.len = 0;\n }\n if (src.tp) {\n fmt.tp = src.tp;\n } else {\n fmt.key = first.ent.length;\n first.ent.push(second.ent[src.key || 0]);\n }\n first.fmt.push(fmt);\n });\n }\n\n return first;\n}\n\n/**\n * Description of an image to attach.\n * @typedef {Object} ImageDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the image, e.g. \"image/png\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - base64-encoded image content. Could be null/undefined.\n * @property {string} preview - base64-encoded thumbnail of the image.\n * @property {integer} width - width of the image.\n * @property {integer} height - height of the image.\n * @property {string} filename - file name suggestion for downloading the image.\n * @property {integer} size - size of the image in bytes. Treat is as an untrusted hint.\n * @property {string} _tempPreview - base64-encoded image preview used during upload process; not serializable.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Insert inline image into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add image to.\n * @param {integer} at - index where the object is inserted. The length of the image is always 1.\n * @param {ImageDesc} imageDesc - object with image paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertImage = function(content, at, imageDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'IM',\n data: {\n mime: imageDesc.mime,\n ref: imageDesc.refurl,\n val: imageDesc.bits || imageDesc.preview,\n width: imageDesc.width,\n height: imageDesc.height,\n name: imageDesc.filename,\n size: imageDesc.size | 0,\n }\n };\n\n if (imageDesc.urlPromise) {\n ex.data._tempPreview = imageDesc._tempPreview;\n ex.data._processing = true;\n imageDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._tempPreview = undefined;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Description of a video to attach.\n * @typedef {Object} VideoDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the video, e.g. \"video/mpeg\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - in-band base64-encoded image data. Could be null/undefined.\n * @property {string} preview - base64-encoded screencapture from the video. Could be null/undefined.\n * @property {string} preref - reference to screencapture from the video. Could be null/undefined.\n * @property {integer} width - width of the video.\n * @property {integer} height - height of the video.\n * @property {integer} duration - duration of the video.\n * @property {string} filename - file name suggestion for downloading the video.\n * @property {integer} size - size of the video in bytes. Treat is as an untrusted hint.\n * @property {string} _tempPreview - base64-encoded screencapture used during upload process; not serializable.\n * @property {Promise} urlPromise - array of two promises, which return URLs of video and preview uploads correspondingly\n * (either could be null).\n */\n\n/**\n * Insert inline image into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add video to.\n * @param {integer} at - index where the object is inserted. The length of the video is always 1.\n * @param {VideoDesc} videoDesc - object with video paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertVideo = function(content, at, videoDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'VD',\n data: {\n mime: videoDesc.mime,\n ref: videoDesc.refurl,\n val: videoDesc.bits,\n preref: videoDesc.preref,\n preview: videoDesc.preview,\n width: videoDesc.width,\n height: videoDesc.height,\n duration: videoDesc.duration | 0,\n name: videoDesc.filename,\n size: videoDesc.size | 0,\n }\n };\n\n if (videoDesc.urlPromise) {\n ex.data._tempPreview = videoDesc._tempPreview;\n ex.data._processing = true;\n videoDesc.urlPromise.then(\n urls => {\n ex.data.ref = urls[0];\n ex.data.preref = urls[1];\n ex.data._tempPreview = undefined;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Description of an audio recording to attach.\n * @typedef {Object} AudioDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the audio, e.g. \"audio/ogg\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - base64-encoded audio content. Could be null/undefined.\n * @property {integer} duration - duration of the record in milliseconds.\n * @property {string} preview - base64 encoded short array of amplitude values 0..100.\n * @property {string} filename - file name suggestion for downloading the audio.\n * @property {integer} size - size of the recording in bytes. Treat is as an untrusted hint.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Insert audio recording into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add audio record to.\n * @param {integer} at - index where the object is inserted. The length of the record is always 1.\n * @param {AudioDesc} audioDesc - object with the audio paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertAudio = function(content, at, audioDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'AU',\n data: {\n mime: audioDesc.mime,\n val: audioDesc.bits,\n duration: audioDesc.duration | 0,\n preview: audioDesc.preview,\n name: audioDesc.filename,\n size: audioDesc.size | 0,\n ref: audioDesc.refurl\n }\n };\n\n if (audioDesc.urlPromise) {\n ex.data._processing = true;\n audioDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Create a (self-contained) video call Drafty document.\n * @memberof Drafty\n * @static\n * @param {boolean} audioOnly true
if the call is initially audio-only.\n * @returns Video Call drafty document.\n */\nDrafty.videoCall = function(audioOnly) {\n const content = {\n txt: ' ',\n fmt: [{\n at: 0,\n len: 1,\n key: 0\n }],\n ent: [{\n tp: 'VC',\n data: {\n aonly: audioOnly\n },\n }]\n };\n return content;\n}\n\n/**\n * Update video call (VC) entity with the new status and duration.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - VC document to update.\n * @param {object} params - new video call parameters.\n * @param {string} params.state - state of video call.\n * @param {number} params.duration - duration of the video call in milliseconds.\n *\n * @returns the same document with update applied.\n */\nDrafty.updateVideoCall = function(content, params) {\n // The video element could be just a format or a format + entity.\n // Must ensure it's the latter first.\n const fmt = ((content || {}).fmt || [])[0];\n if (!fmt) {\n // Unrecognized content.\n return content;\n }\n\n let ent;\n if (fmt.tp == 'VC') {\n // Just a format, convert to format + entity.\n delete fmt.tp;\n fmt.key = 0;\n ent = {\n tp: 'VC'\n };\n content.ent = [ent];\n } else {\n ent = (content.ent || [])[fmt.key | 0];\n if (!ent || ent.tp != 'VC') {\n // Not a VC entity.\n return content;\n }\n }\n ent.data = ent.data || {};\n Object.assign(ent.data, params);\n return content;\n}\n\n/**\n * Create a quote to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {string} header - Quote header (title, etc.).\n * @param {string} uid - UID of the author to mention.\n * @param {Drafty} body - Body of the quoted message.\n *\n * @returns Reply quote Drafty doc with the quote formatting.\n */\nDrafty.quote = function(header, uid, body) {\n const quote = Drafty.append(Drafty.appendLineBreak(Drafty.mention(header, uid)), body);\n\n // Wrap into a quote.\n quote.fmt.push({\n at: 0,\n len: quote.txt.length,\n tp: 'QQ'\n });\n\n return quote;\n}\n\n/**\n * Create a Drafty document with a mention.\n *\n * @param {string} name - mentioned name.\n * @param {string} uid - mentioned user ID.\n *\n * @returns {Drafty} document with the mention.\n */\nDrafty.mention = function(name, uid) {\n return {\n txt: name || '',\n fmt: [{\n at: 0,\n len: (name || '').length,\n key: 0\n }],\n ent: [{\n tp: 'MN',\n data: {\n val: uid\n }\n }]\n };\n}\n\n/**\n * Append a link to a Drafty document.\n *\n * @param {Drafty} content - Drafty document to append link to.\n * @param {Object} linkData - Link info in format {txt: 'ankor text', url: 'http://...'}
.\n *\n * @returns {Drafty} the same document as content
.\n */\nDrafty.appendLink = function(content, linkData) {\n content = content || {\n txt: ''\n };\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: content.txt.length,\n len: linkData.txt.length,\n key: content.ent.length\n });\n content.txt += linkData.txt;\n\n const ex = {\n tp: 'LN',\n data: {\n url: linkData.url\n }\n }\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Append image to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add image to.\n * @param {ImageDesc} imageDesc - object with image paramenets.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendImage = function(content, imageDesc) {\n content = content || {\n txt: ''\n };\n content.txt += ' ';\n return Drafty.insertImage(content, content.txt.length - 1, imageDesc);\n}\n\n/**\n * Append audio recodring to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add recording to.\n * @param {AudioDesc} audioDesc - object with audio data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendAudio = function(content, audioDesc) {\n content = content || {\n txt: ''\n };\n content.txt += ' ';\n return Drafty.insertAudio(content, content.txt.length - 1, audioDesc);\n}\n\n/**\n * Description of a file to attach.\n * @typedef {Object} AttachmentDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the attachment, e.g. \"application/octet-stream\"\n * @property {string} data - base64-encoded in-band content of small attachments. Could be null/undefined.\n * @property {string} filename - file name suggestion for downloading the attachment.\n * @property {integer} size - size of the file in bytes. Treat is as an untrusted hint.\n * @property {string} refurl - reference to the out-of-band content. Could be null/undefined.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Attach file to Drafty content. Either as a blob or as a reference.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to attach file to.\n * @param {AttachmentDesc} object - containing attachment description and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.attachFile = function(content, attachmentDesc) {\n content = content || {\n txt: ''\n };\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: -1,\n len: 0,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'EX',\n data: {\n mime: attachmentDesc.mime,\n val: attachmentDesc.data,\n name: attachmentDesc.filename,\n ref: attachmentDesc.refurl,\n size: attachmentDesc.size | 0\n }\n }\n if (attachmentDesc.urlPromise) {\n ex.data._processing = true;\n attachmentDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._processing = undefined;\n },\n _ => {\n /* catch the error, otherwise it will appear in the console. */\n ex.data._processing = undefined;\n }\n );\n }\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Wraps drafty document into a simple formatting style.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - document or string to wrap into a style.\n * @param {string} style - two-letter style to wrap into.\n * @param {number} at - index where the style starts, default 0.\n * @param {number} len - length of the form content, default all of it.\n *\n * @return {Drafty} updated document.\n */\nDrafty.wrapInto = function(content, style, at, len) {\n if (typeof content == 'string') {\n content = {\n txt: content\n };\n }\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at || 0,\n len: len || content.txt.length,\n tp: style,\n });\n\n return content;\n}\n\n/**\n * Wraps content into an interactive form.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - to wrap into a form.\n * @param {number} at - index where the forms starts.\n * @param {number} len - length of the form content.\n *\n * @return {Drafty} updated document.\n */\nDrafty.wrapAsForm = function(content, at, len) {\n return Drafty.wrapInto(content, 'FM', at, len);\n}\n\n/**\n * Insert clickable button into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - Drafty document to insert button to or a string to be used as button text.\n * @param {number} at - location where the button is inserted.\n * @param {number} len - the length of the text to be used as button title.\n * @param {string} name - the button. Client should return it to the server when the button is clicked.\n * @param {string} actionType - the type of the button, one of 'url' or 'pub'.\n * @param {string} actionValue - the value to return on click:\n * @param {string} refUrl - the URL to go to when the 'url' button is clicked.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertButton = function(content, at, len, name, actionType, actionValue, refUrl) {\n if (typeof content == 'string') {\n content = {\n txt: content\n };\n }\n\n if (!content || !content.txt || content.txt.length < at + len) {\n return null;\n }\n\n if (len <= 0 || ['url', 'pub'].indexOf(actionType) == -1) {\n return null;\n }\n // Ensure refUrl is a string.\n if (actionType == 'url' && !refUrl) {\n return null;\n }\n refUrl = '' + refUrl;\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: len,\n key: content.ent.length\n });\n content.ent.push({\n tp: 'BN',\n data: {\n act: actionType,\n val: actionValue,\n ref: refUrl,\n name: name\n }\n });\n\n return content;\n}\n\n/**\n * Append clickable button to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - Drafty document to insert button to or a string to be used as button text.\n * @param {string} title - the text to be used as button title.\n * @param {string} name - the button. Client should return it to the server when the button is clicked.\n * @param {string} actionType - the type of the button, one of 'url' or 'pub'.\n * @param {string} actionValue - the value to return on click:\n * @param {string} refUrl - the URL to go to when the 'url' button is clicked.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendButton = function(content, title, name, actionType, actionValue, refUrl) {\n content = content || {\n txt: ''\n };\n const at = content.txt.length;\n content.txt += title;\n return Drafty.insertButton(content, at, title.length, name, actionType, actionValue, refUrl);\n}\n\n/**\n * Attach a generic JS object. The object is attached as a json string.\n * Intended for representing a form response.\n *\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - Drafty document to attach file to.\n * @param {Object} data - data to convert to json string and attach.\n * @returns {Drafty} the same document as content
.\n */\nDrafty.attachJSON = function(content, data) {\n content = content || {\n txt: ''\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: -1,\n len: 0,\n key: content.ent.length\n });\n\n content.ent.push({\n tp: 'EX',\n data: {\n mime: JSON_MIME_TYPE,\n val: data\n }\n });\n\n return content;\n}\n/**\n * Append line break to a Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - Drafty document to append linebreak to.\n * @returns {Drafty} the same document as content
.\n */\nDrafty.appendLineBreak = function(content) {\n content = content || {\n txt: ''\n };\n content.fmt = content.fmt || [];\n content.fmt.push({\n at: content.txt.length,\n len: 1,\n tp: 'BR'\n });\n content.txt += ' ';\n\n return content;\n}\n/**\n * Given Drafty document, convert it to HTML.\n * No attempt is made to strip pre-existing html markup.\n * This is potentially unsafe because content.txt
may contain malicious HTML\n * markup.\n * @memberof Tinode.Drafty\n * @static\n *\n * @param {Drafty} doc - document to convert.\n *\n * @returns {string} HTML-representation of content.\n */\nDrafty.UNSAFE_toHTML = function(doc) {\n const tree = draftyToTree(doc);\n const htmlFormatter = function(type, data, values) {\n const tag = DECORATORS[type];\n let result = values ? values.join('') : '';\n if (tag) {\n result = tag.open(data) + result + tag.close(data);\n }\n return result;\n };\n return treeBottomUp(tree, htmlFormatter, 0);\n}\n\n/**\n * Callback for applying custom formatting to a Drafty document.\n * Called once for each style span.\n * @memberof Drafty\n * @static\n *\n * @callback Formatter\n * @param {string} style - style code such as \"ST\" or \"IM\".\n * @param {Object} data - entity's data.\n * @param {Object} values - possibly styled subspans contained in this style span.\n * @param {number} index - index of the element guaranteed to be unique.\n */\n\n/**\n * Convert Drafty document to a representation suitable for display.\n * The context
may expose a function getFormatter(style)
. If it's available\n * it will call it to obtain a formatter
for a subtree of styles under the style
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|Object} content - Drafty document to transform.\n * @param {Formatter} formatter - callback which formats individual elements.\n * @param {Object} context - context provided to formatter as this
.\n *\n * @return {Object} transformed object\n */\nDrafty.format = function(original, formatter, context) {\n return treeBottomUp(draftyToTree(original), formatter, 0, [], context);\n}\n\n/**\n * Shorten Drafty document making the drafty text no longer than the limit.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characrets to shorten to.\n * @param {boolean} light - remove heavy data from entities.\n * @returns new shortened Drafty object leaving the original intact.\n */\nDrafty.shorten = function(original, limit, light) {\n let tree = draftyToTree(original);\n tree = shortenTree(tree, limit, '…');\n if (tree && light) {\n tree = lightEntity(tree);\n }\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Transform Drafty doc for forwarding: strip leading @mention and any leading line breaks or whitespace.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @returns converted Drafty object leaving the original intact.\n */\nDrafty.forwardedContent = function(original) {\n let tree = draftyToTree(original);\n const rmMention = function(node) {\n if (node.type == 'MN') {\n if (!node.parent || !node.parent.type) {\n return null;\n }\n }\n return node;\n }\n // Strip leading mention.\n tree = treeTopDown(tree, rmMention);\n // Remove leading whitespace.\n tree = lTrim(tree);\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Prepare Drafty doc for wrapping into QQ as a reply:\n * - Replace forwarding mention with symbol '➦' and remove data (UID).\n * - Remove quoted text completely.\n * - Replace line breaks with spaces.\n * - Strip entities of heavy content.\n * - Move attachments to the end of the document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characters to shorten to.\n * @returns converted Drafty object leaving the original intact.\n */\nDrafty.replyContent = function(original, limit) {\n const convMNnQQnBR = function(node) {\n if (node.type == 'QQ') {\n return null;\n } else if (node.type == 'MN') {\n if ((!node.parent || !node.parent.type) && (node.text || '').startsWith('➦')) {\n node.text = '➦';\n delete node.children;\n delete node.data;\n }\n } else if (node.type == 'BR') {\n node.text = ' ';\n delete node.type;\n delete node.children;\n }\n return node;\n }\n\n let tree = draftyToTree(original);\n if (!tree) {\n return original;\n }\n\n // Strip leading mention.\n tree = treeTopDown(tree, convMNnQQnBR);\n // Move attachments to the end of the doc.\n tree = attachmentsToEnd(tree, MAX_PREVIEW_ATTACHMENTS);\n // Shorten the doc.\n tree = shortenTree(tree, limit, '…');\n // Strip heavy elements except IM.data['val'] and VD.data['preview'] (have to keep them to generate previews later).\n const filter = node => {\n switch (node.type) {\n case 'IM':\n return ['val'];\n case 'VD':\n return ['preview'];\n }\n return null;\n };\n tree = lightEntity(tree, filter);\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n\n/**\n * Generate drafty preview:\n * - Shorten the document.\n * - Strip all heavy entity data leaving just inline styles and entity references.\n * - Replace line breaks with spaces.\n * - Replace content of QQ with a space.\n * - Replace forwarding mention with symbol '➦'.\n * move all attachments to the end of the document and make them visible.\n * The context
may expose a function getFormatter(style)
. If it's available\n * it will call it to obtain a formatter
for a subtree of styles under the style
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characters to shorten to.\n * @param {boolean} forwarding - this a forwarding message preview.\n * @returns new shortened Drafty object leaving the original intact.\n */\nDrafty.preview = function(original, limit, forwarding) {\n let tree = draftyToTree(original);\n\n // Move attachments to the end.\n tree = attachmentsToEnd(tree, MAX_PREVIEW_ATTACHMENTS);\n\n // Convert leading mention to '➦' and replace QQ and BR with a space ' '.\n const convMNnQQnBR = function(node) {\n if (node.type == 'MN') {\n if ((!node.parent || !node.parent.type) && (node.text || '').startsWith('➦')) {\n node.text = '➦';\n delete node.children;\n }\n } else if (node.type == 'QQ') {\n node.text = ' ';\n delete node.children;\n } else if (node.type == 'BR') {\n node.text = ' ';\n delete node.children;\n delete node.type;\n }\n return node;\n }\n tree = treeTopDown(tree, convMNnQQnBR);\n\n tree = shortenTree(tree, limit, '…');\n if (forwarding) {\n // Keep some IM and VD data for preview.\n const filter = {\n IM: ['val'],\n VD: ['preview']\n };\n tree = lightEntity(tree, node => {\n return filter[node.type];\n });\n } else {\n tree = lightEntity(tree);\n }\n\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Given Drafty document, convert it to plain text.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to convert to plain text.\n * @returns {string} plain-text representation of the drafty document.\n */\nDrafty.toPlainText = function(content) {\n return typeof content == 'string' ? content : content.txt;\n}\n\n/**\n * Check if the document has no markup and no entities.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - content to check for presence of markup.\n * @returns true
is content is plain text, false
otherwise.\n */\nDrafty.isPlainText = function(content) {\n return typeof content == 'string' || !(content.fmt || content.ent);\n}\n\n/**\n * Convert document to plain text with markdown. All elements which cannot\n * be represented in markdown are stripped.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to convert to plain text with markdown.\n */\nDrafty.toMarkdown = function(content) {\n let tree = draftyToTree(content);\n const mdFormatter = function(type, _, values) {\n const def = FORMAT_TAGS[type];\n let result = (values ? values.join('') : '');\n if (def) {\n if (def.isVoid) {\n result = def.md_tag || '';\n } else if (def.md_tag) {\n result = def.md_tag + result + def.md_tag;\n }\n }\n return result;\n };\n return treeBottomUp(tree, mdFormatter, 0);\n}\n\n/**\n * Checks if the object represets is a valid Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - content to check for validity.\n * @returns true
is content is valid, false
otherwise.\n */\nDrafty.isValid = function(content) {\n if (!content) {\n return false;\n }\n\n const {\n txt,\n fmt,\n ent\n } = content;\n\n if (!txt && txt !== '' && !fmt && !ent) {\n return false;\n }\n\n const txt_type = typeof txt;\n if (txt_type != 'string' && txt_type != 'undefined' && txt !== null) {\n return false;\n }\n\n if (typeof fmt != 'undefined' && !Array.isArray(fmt) && fmt !== null) {\n return false;\n }\n\n if (typeof ent != 'undefined' && !Array.isArray(ent) && ent !== null) {\n return false;\n }\n return true;\n}\n\n/**\n * Check if the drafty document has attachments: style EX and outside of normal rendering flow,\n * i.e. at = -1
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to check for attachments.\n * @returns true
if there are attachments.\n */\nDrafty.hasAttachments = function(content) {\n if (!Array.isArray(content.fmt)) {\n return false;\n }\n for (let i in content.fmt) {\n const fmt = content.fmt[i];\n if (fmt && fmt.at < 0) {\n const ent = content.ent[fmt.key | 0];\n return ent && ent.tp == 'EX' && ent.data;\n }\n }\n return false;\n}\n\n/**\n * Callback for enumerating entities in a Drafty document.\n * Called once for each entity.\n * @memberof Drafty\n * @static\n *\n * @callback EntityCallback\n * @param {Object} data entity data.\n * @param {string} entity type.\n * @param {number} index entity's index in `content.ent`.\n *\n * @return 'true-ish' to stop processing, 'false-ish' otherwise.\n */\n\n/**\n * Enumerate attachments: style EX and outside of normal rendering flow, i.e. at = -1
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to process for attachments.\n * @param {EntityCallback} callback - callback to call for each attachment.\n * @param {Object} context - value of \"this\" for callback.\n */\nDrafty.attachments = function(content, callback, context) {\n if (!Array.isArray(content.fmt)) {\n return;\n }\n let count = 0;\n for (let i in content.fmt) {\n let fmt = content.fmt[i];\n if (fmt && fmt.at < 0) {\n const ent = content.ent[fmt.key | 0];\n if (ent && ent.tp == 'EX' && ent.data) {\n if (callback.call(context, ent.data, count++, 'EX')) {\n break;\n }\n }\n }\n };\n}\n\n/**\n * Check if the drafty document has entities.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to check for entities.\n * @returns true
if there are entities.\n */\nDrafty.hasEntities = function(content) {\n return content.ent && content.ent.length > 0;\n}\n\n/**\n * Enumerate entities. Enumeration stops if callback returns 'true'.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with entities to enumerate.\n * @param {EntityCallback} callback - callback to call for each entity.\n * @param {Object} context - value of \"this\" for callback.\n *\n */\nDrafty.entities = function(content, callback, context) {\n if (content.ent && content.ent.length > 0) {\n for (let i in content.ent) {\n if (content.ent[i]) {\n if (callback.call(context, content.ent[i].data, i, content.ent[i].tp)) {\n break;\n }\n }\n }\n }\n}\n\n/**\n * Callback for enumerating styles (inline formats) in a Drafty document.\n * Called once for each style.\n * @memberof Drafty\n * @static\n *\n * @callback StyleCallback\n * @param {string} tp - format type.\n * @param {number} at - starting position of the format in text.\n * @param {number} len - extent of the format in characters.\n * @param {number} key - index of the entity if format is a reference.\n * @param {number} index - style's index in `content.fmt`.\n *\n * @return 'true-ish' to stop processing, 'false-ish' otherwise.\n */\n\n/**\n * Enumerate styles (inline formats). Enumeration stops if callback returns 'true'.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with styles (formats) to enumerate.\n * @param {StyleCallback} callback - callback to call for each format.\n * @param {Object} context - value of \"this\" for callback.\n */\nDrafty.styles = function(content, callback, context) {\n if (content.fmt && content.fmt.length > 0) {\n for (let i in content.fmt) {\n const fmt = content.fmt[i];\n if (fmt) {\n if (callback.call(context, fmt.tp, fmt.at, fmt.len, fmt.key, i)) {\n break;\n }\n }\n }\n }\n}\n\n/**\n * Remove unrecognized fields from entity data\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with entities to enumerate.\n * @returns content.\n */\nDrafty.sanitizeEntities = function(content) {\n if (content && content.ent && content.ent.length > 0) {\n for (let i in content.ent) {\n const ent = content.ent[i];\n if (ent && ent.data) {\n const data = copyEntData(ent.data);\n if (data) {\n content.ent[i].data = data;\n } else {\n delete content.ent[i].data;\n }\n }\n }\n }\n return content;\n}\n\n/**\n * Given the entity, get URL which can be used for downloading\n * entity data.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the URl from.\n * @returns {string} URL to download entity data or null
.\n */\nDrafty.getDownloadUrl = function(entData) {\n let url = null;\n if (entData.mime != JSON_MIME_TYPE && entData.val) {\n url = base64toObjectUrl(entData.val, entData.mime, Drafty.logger);\n } else if (typeof entData.ref == 'string') {\n url = entData.ref;\n }\n return url;\n}\n\n/**\n * Check if the entity data is not ready for sending, such as being uploaded to the server.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entity.data to get the URl from.\n * @returns {boolean} true if upload is in progress, false otherwise.\n */\nDrafty.isProcessing = function(entData) {\n return !!entData._processing;\n}\n\n/**\n * Given the entity, get URL which can be used for previewing\n * the entity.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entity.data to get the URl from.\n *\n * @returns {string} url for previewing or null if no such url is available.\n */\nDrafty.getPreviewUrl = function(entData) {\n return entData.val ? base64toObjectUrl(entData.val, entData.mime, Drafty.logger) : null;\n}\n\n/**\n * Get approximate size of the entity.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the size for.\n * @returns {number} size of entity data in bytes.\n */\nDrafty.getEntitySize = function(entData) {\n // Either size hint or length of value. The value is base64 encoded,\n // the actual object size is smaller than the encoded length.\n return entData.size ? entData.size : entData.val ? (entData.val.length * 0.75) | 0 : 0;\n}\n\n/**\n * Get entity mime type.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the type for.\n * @returns {string} mime type of entity.\n */\nDrafty.getEntityMimeType = function(entData) {\n return entData.mime || 'text/plain';\n}\n\n/**\n * Get HTML tag for a given two-letter style name.\n * @memberof Drafty\n * @static\n *\n * @param {string} style - two-letter style, like ST or LN.\n *\n * @returns {string} HTML tag name if style is found, {code: undefined} if style is falsish or not found.\n */\nDrafty.tagName = function(style) {\n return FORMAT_TAGS[style] && FORMAT_TAGS[style].html_tag;\n}\n\n/**\n * For a given data bundle generate an object with HTML attributes,\n * for instance, given {url: \"http://www.example.com/\"} return\n * {href: \"http://www.example.com/\"}\n * @memberof Drafty\n * @static\n *\n * @param {string} style - two-letter style to generate attributes for.\n * @param {Object} data - data bundle to convert to attributes\n *\n * @returns {Object} object with HTML attributes.\n */\nDrafty.attrValue = function(style, data) {\n if (data && DECORATORS[style] && DECORATORS[style].props) {\n return DECORATORS[style].props(data);\n }\n\n return undefined;\n}\n\n/**\n * Drafty MIME type.\n * @memberof Drafty\n * @static\n *\n * @returns {string} content-Type \"text/x-drafty\".\n */\nDrafty.getContentType = function() {\n return DRAFTY_MIME_TYPE;\n}\n\n// =================\n// Utility methods.\n// =================\n\n// Take a string and defined earlier style spans, re-compose them into a tree where each leaf is\n// a same-style (including unstyled) string. I.e. 'hello *bold _italic_* and ~more~ world' ->\n// ('hello ', (b: 'bold ', (i: 'italic')), ' and ', (s: 'more'), ' world');\n//\n// This is needed in order to clear markup, i.e. 'hello *world*' -> 'hello world' and convert\n// ranges from markup-ed offsets to plain text offsets.\nfunction chunkify(line, start, end, spans) {\n const chunks = [];\n\n if (spans.length == 0) {\n return [];\n }\n\n for (let i in spans) {\n // Get the next chunk from the queue\n const span = spans[i];\n\n // Grab the initial unstyled chunk\n if (span.at > start) {\n chunks.push({\n txt: line.slice(start, span.at)\n });\n }\n\n // Grab the styled chunk. It may include subchunks.\n const chunk = {\n tp: span.tp\n };\n const chld = chunkify(line, span.at + 1, span.end, span.children);\n if (chld.length > 0) {\n chunk.children = chld;\n } else {\n chunk.txt = span.txt;\n }\n chunks.push(chunk);\n start = span.end + 1; // '+1' is to skip the formatting character\n }\n\n // Grab the remaining unstyled chunk, after the last span\n if (start < end) {\n chunks.push({\n txt: line.slice(start, end)\n });\n }\n\n return chunks;\n}\n\n// Detect starts and ends of formatting spans. Unformatted spans are\n// ignored at this stage.\nfunction spannify(original, re_start, re_end, type) {\n const result = [];\n let index = 0;\n let line = original.slice(0); // make a copy;\n\n while (line.length > 0) {\n // match[0]; // match, like '*abc*'\n // match[1]; // match captured in parenthesis, like 'abc'\n // match['index']; // offset where the match started.\n\n // Find the opening token.\n const start = re_start.exec(line);\n if (start == null) {\n break;\n }\n\n // Because javascript RegExp does not support lookbehind, the actual offset may not point\n // at the markup character. Find it in the matched string.\n let start_offset = start['index'] + start[0].lastIndexOf(start[1]);\n // Clip the processed part of the string.\n line = line.slice(start_offset + 1);\n // start_offset is an offset within the clipped string. Convert to original index.\n start_offset += index;\n // Index now point to the beginning of 'line' within the 'original' string.\n index = start_offset + 1;\n\n // Find the matching closing token.\n const end = re_end ? re_end.exec(line) : null;\n if (end == null) {\n break;\n }\n let end_offset = end['index'] + end[0].indexOf(end[1]);\n // Clip the processed part of the string.\n line = line.slice(end_offset + 1);\n // Update offsets\n end_offset += index;\n // Index now points to the beginning of 'line' within the 'original' string.\n index = end_offset + 1;\n\n result.push({\n txt: original.slice(start_offset + 1, end_offset),\n children: [],\n at: start_offset,\n end: end_offset,\n tp: type\n });\n }\n\n return result;\n}\n\n// Convert linear array or spans into a tree representation.\n// Keep standalone and nested spans, throw away partially overlapping spans.\nfunction toSpanTree(spans) {\n if (spans.length == 0) {\n return [];\n }\n\n const tree = [spans[0]];\n let last = spans[0];\n for (let i = 1; i < spans.length; i++) {\n // Keep spans which start after the end of the previous span or those which\n // are complete within the previous span.\n if (spans[i].at > last.end) {\n // Span is completely outside of the previous span.\n tree.push(spans[i]);\n last = spans[i];\n } else if (spans[i].end <= last.end) {\n // Span is fully inside of the previous span. Push to subnode.\n last.children.push(spans[i]);\n }\n // Span could partially overlap, ignoring it as invalid.\n }\n\n // Recursively rearrange the subnodes.\n for (let i in tree) {\n tree[i].children = toSpanTree(tree[i].children);\n }\n\n return tree;\n}\n\n// Convert drafty document to a tree.\nfunction draftyToTree(doc) {\n if (!doc) {\n return null;\n }\n\n doc = (typeof doc == 'string') ? {\n txt: doc\n } : doc;\n let {\n txt,\n fmt,\n ent\n } = doc;\n\n txt = txt || '';\n if (!Array.isArray(ent)) {\n ent = [];\n }\n\n if (!Array.isArray(fmt) || fmt.length == 0) {\n if (ent.length == 0) {\n return {\n text: txt\n };\n }\n\n // Handle special case when all values in fmt are 0 and fmt therefore is skipped.\n fmt = [{\n at: 0,\n len: 0,\n key: 0\n }];\n }\n\n // Sanitize spans.\n const spans = [];\n const attachments = [];\n fmt.forEach((span) => {\n if (!span || typeof span != 'object') {\n return;\n }\n\n if (!['undefined', 'number'].includes(typeof span.at)) {\n // Present, but non-numeric 'at'.\n return;\n }\n if (!['undefined', 'number'].includes(typeof span.len)) {\n // Present, but non-numeric 'len'.\n return;\n }\n let at = span.at | 0;\n let len = span.len | 0;\n if (len < 0) {\n // Invalid span length.\n return;\n }\n\n let key = span.key || 0;\n if (ent.length > 0 && (typeof key != 'number' || key < 0 || key >= ent.length)) {\n // Invalid key value.\n return;\n }\n\n if (at <= -1) {\n // Attachment. Store attachments separately.\n attachments.push({\n start: -1,\n end: 0,\n key: key\n });\n return;\n } else if (at + len > txt.length) {\n // Span is out of bounds.\n return;\n }\n\n if (!span.tp) {\n if (ent.length > 0 && (typeof ent[key] == 'object')) {\n spans.push({\n start: at,\n end: at + len,\n key: key\n });\n }\n } else {\n spans.push({\n type: span.tp,\n start: at,\n end: at + len\n });\n }\n });\n\n // Sort spans first by start index (asc) then by length (desc), then by weight.\n spans.sort((a, b) => {\n let diff = a.start - b.start;\n if (diff != 0) {\n return diff;\n }\n diff = b.end - a.end;\n if (diff != 0) {\n return diff;\n }\n return FMT_WEIGHT.indexOf(b.type) - FMT_WEIGHT.indexOf(a.type);\n });\n\n // Move attachments to the end of the list.\n if (attachments.length > 0) {\n spans.push(...attachments);\n }\n\n spans.forEach((span) => {\n if (ent.length > 0 && !span.type && ent[span.key] && typeof ent[span.key] == 'object') {\n span.type = ent[span.key].tp;\n span.data = ent[span.key].data;\n }\n\n // Is type still undefined? Hide the invalid element!\n if (!span.type) {\n span.type = 'HD';\n }\n });\n\n let tree = spansToTree({}, txt, 0, txt.length, spans);\n\n // Flatten tree nodes.\n const flatten = function(node) {\n if (Array.isArray(node.children) && node.children.length == 1) {\n // Unwrap.\n const child = node.children[0];\n if (!node.type) {\n const parent = node.parent;\n node = child;\n node.parent = parent;\n } else if (!child.type && !child.children) {\n node.text = child.text;\n delete node.children;\n }\n }\n return node;\n }\n tree = treeTopDown(tree, flatten);\n\n return tree;\n}\n\n// Add tree node to a parent tree.\nfunction addNode(parent, n) {\n if (!n) {\n return parent;\n }\n\n if (!parent.children) {\n parent.children = [];\n }\n\n // If text is present, move it to a subnode.\n if (parent.text) {\n parent.children.push({\n text: parent.text,\n parent: parent\n });\n delete parent.text;\n }\n\n n.parent = parent;\n parent.children.push(n);\n\n return parent;\n}\n\n// Returns a tree of nodes.\nfunction spansToTree(parent, text, start, end, spans) {\n if (!spans || spans.length == 0) {\n if (start < end) {\n addNode(parent, {\n text: text.substring(start, end)\n });\n }\n return parent;\n }\n\n // Process subspans.\n for (let i = 0; i < spans.length; i++) {\n const span = spans[i];\n if (span.start < 0 && span.type == 'EX') {\n addNode(parent, {\n type: span.type,\n data: span.data,\n key: span.key,\n att: true\n });\n continue;\n }\n\n // Add un-styled range before the styled span starts.\n if (start < span.start) {\n addNode(parent, {\n text: text.substring(start, span.start)\n });\n start = span.start;\n }\n\n // Get all spans which are within the current span.\n const subspans = [];\n while (i < spans.length - 1) {\n const inner = spans[i + 1];\n if (inner.start < 0) {\n // Attachments are in the end. Stop.\n break;\n } else if (inner.start < span.end) {\n if (inner.end <= span.end) {\n const tag = FORMAT_TAGS[inner.tp] || {};\n if (inner.start < inner.end || tag.isVoid) {\n // Valid subspan: completely within the current span and\n // either non-zero length or zero length is acceptable.\n subspans.push(inner);\n }\n }\n i++;\n // Overlapping subspans are ignored.\n } else {\n // Past the end of the current span. Stop.\n break;\n }\n }\n\n addNode(parent, spansToTree({\n type: span.type,\n data: span.data,\n key: span.key\n }, text, start, span.end, subspans));\n start = span.end;\n }\n\n // Add the last unformatted range.\n if (start < end) {\n addNode(parent, {\n text: text.substring(start, end)\n });\n }\n\n return parent;\n}\n\n// Append a tree to a Drafty doc.\nfunction treeToDrafty(doc, tree, keymap) {\n if (!tree) {\n return doc;\n }\n\n doc.txt = doc.txt || '';\n\n // Checkpoint to measure length of the current tree node.\n const start = doc.txt.length;\n\n if (tree.text) {\n doc.txt += tree.text;\n } else if (Array.isArray(tree.children)) {\n tree.children.forEach((c) => {\n treeToDrafty(doc, c, keymap);\n });\n }\n\n if (tree.type) {\n const len = doc.txt.length - start;\n doc.fmt = doc.fmt || [];\n if (Object.keys(tree.data || {}).length > 0) {\n doc.ent = doc.ent || [];\n const newKey = (typeof keymap[tree.key] == 'undefined') ? doc.ent.length : keymap[tree.key];\n keymap[tree.key] = newKey;\n doc.ent[newKey] = {\n tp: tree.type,\n data: tree.data\n };\n if (tree.att) {\n // Attachment.\n doc.fmt.push({\n at: -1,\n len: 0,\n key: newKey\n });\n } else {\n doc.fmt.push({\n at: start,\n len: len,\n key: newKey\n });\n }\n } else {\n doc.fmt.push({\n tp: tree.type,\n at: start,\n len: len\n });\n }\n }\n return doc;\n}\n\n// Traverse the tree top down transforming the nodes: apply transformer to every tree node.\nfunction treeTopDown(src, transformer, context) {\n if (!src) {\n return null;\n }\n\n let dst = transformer.call(context, src);\n if (!dst || !dst.children) {\n return dst;\n }\n\n const children = [];\n for (let i in dst.children) {\n let n = dst.children[i];\n if (n) {\n n = treeTopDown(n, transformer, context);\n if (n) {\n children.push(n);\n }\n }\n }\n\n if (children.length == 0) {\n dst.children = null;\n } else {\n dst.children = children;\n }\n\n return dst;\n}\n\n// Traverse the tree bottom-up: apply formatter to every node.\n// The formatter must maintain its state through context.\nfunction treeBottomUp(src, formatter, index, stack, context) {\n if (!src) {\n return null;\n }\n\n if (stack && src.type) {\n stack.push(src.type);\n }\n\n let values = [];\n for (let i in src.children) {\n const n = treeBottomUp(src.children[i], formatter, i, stack, context);\n if (n) {\n values.push(n);\n }\n }\n if (values.length == 0) {\n if (src.text) {\n values = [src.text];\n } else {\n values = null;\n }\n }\n\n if (stack && src.type) {\n stack.pop();\n }\n\n return formatter.call(context, src.type, src.data, values, index, stack);\n}\n\n// Clip tree to the provided limit.\nfunction shortenTree(tree, limit, tail) {\n if (!tree) {\n return null;\n }\n\n if (tail) {\n limit -= tail.length;\n }\n\n const shortener = function(node) {\n if (limit <= -1) {\n // Limit -1 means the doc was already clipped.\n return null;\n }\n\n if (node.att) {\n // Attachments are unchanged.\n return node;\n }\n if (limit == 0) {\n node.text = tail;\n limit = -1;\n } else if (node.text) {\n const len = node.text.length;\n if (len > limit) {\n node.text = node.text.substring(0, limit) + tail;\n limit = -1;\n } else {\n limit -= len;\n }\n }\n return node;\n }\n\n return treeTopDown(tree, shortener);\n}\n\n// Strip heavy entities from a tree.\nfunction lightEntity(tree, allow) {\n const lightCopy = node => {\n const data = copyEntData(node.data, true, allow ? allow(node) : null);\n if (data) {\n node.data = data;\n } else {\n delete node.data;\n }\n return node;\n }\n return treeTopDown(tree, lightCopy);\n}\n\n// Remove spaces and breaks on the left.\nfunction lTrim(tree) {\n if (tree.type == 'BR') {\n tree = null;\n } else if (tree.text) {\n if (!tree.type) {\n tree.text = tree.text.trimStart();\n if (!tree.text) {\n tree = null;\n }\n }\n } else if (!tree.type && tree.children && tree.children.length > 0) {\n const c = lTrim(tree.children[0]);\n if (c) {\n tree.children[0] = c;\n } else {\n tree.children.shift();\n if (!tree.type && tree.children.length == 0) {\n tree = null;\n }\n }\n }\n return tree;\n}\n\n// Move attachments to the end. Attachments must be at the top level, no need to traverse the tree.\nfunction attachmentsToEnd(tree, limit) {\n if (!tree) {\n return null;\n }\n\n if (tree.att) {\n tree.text = ' ';\n delete tree.att;\n delete tree.children;\n } else if (tree.children) {\n const attachments = [];\n const children = [];\n for (let i in tree.children) {\n const c = tree.children[i];\n if (c.att) {\n if (attachments.length == limit) {\n // Too many attachments to preview;\n continue;\n }\n if (c.data['mime'] == JSON_MIME_TYPE) {\n // JSON attachments are not shown in preview.\n continue;\n }\n\n delete c.att;\n delete c.children;\n c.text = ' ';\n attachments.push(c);\n } else {\n children.push(c);\n }\n }\n tree.children = children.concat(attachments);\n }\n return tree;\n}\n\n// Get a list of entities from a text.\nfunction extractEntities(line) {\n let match;\n let extracted = [];\n ENTITY_TYPES.forEach((entity) => {\n while ((match = entity.re.exec(line)) !== null) {\n extracted.push({\n offset: match['index'],\n len: match[0].length,\n unique: match[0],\n data: entity.pack(match[0]),\n type: entity.name\n });\n }\n });\n\n if (extracted.length == 0) {\n return extracted;\n }\n\n // Remove entities detected inside other entities, like #hashtag in a URL.\n extracted.sort((a, b) => {\n return a.offset - b.offset;\n });\n\n let idx = -1;\n extracted = extracted.filter((el) => {\n const result = (el.offset > idx);\n idx = el.offset + el.len;\n return result;\n });\n\n return extracted;\n}\n\n// Convert the chunks into format suitable for serialization.\nfunction draftify(chunks, startAt) {\n let plain = '';\n let ranges = [];\n for (let i in chunks) {\n const chunk = chunks[i];\n if (!chunk.txt) {\n const drafty = draftify(chunk.children, plain.length + startAt);\n chunk.txt = drafty.txt;\n ranges = ranges.concat(drafty.fmt);\n }\n\n if (chunk.tp) {\n ranges.push({\n at: plain.length + startAt,\n len: chunk.txt.length,\n tp: chunk.tp\n });\n }\n\n plain += chunk.txt;\n }\n return {\n txt: plain,\n fmt: ranges\n };\n}\n\n// Create a copy of entity data with (light=false) or without (light=true) the large payload.\n// The array 'allow' contains a list of fields exempt from stripping.\nfunction copyEntData(data, light, allow) {\n if (data && Object.entries(data).length > 0) {\n allow = allow || [];\n const dc = {};\n ALLOWED_ENT_FIELDS.forEach(key => {\n if (data[key]) {\n if (light && !allow.includes(key) &&\n (typeof data[key] == 'string' || Array.isArray(data[key])) &&\n data[key].length > MAX_PREVIEW_DATA_SIZE) {\n return;\n }\n if (typeof data[key] == 'object') {\n return;\n }\n dc[key] = data[key];\n }\n });\n\n if (Object.entries(dc).length != 0) {\n return dc;\n }\n }\n return null;\n}\n\nif (typeof module != 'undefined') {\n module.exports = Drafty;\n}\n","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","/**\n * @file Access control model.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\n/**\n * Helper class for handling access mode.\n *\n * @class AccessMode\n * @memberof Tinode\n *\n * @param {AccessMode|Object=} acs - AccessMode to copy or access mode object received from the server.\n */\nexport default class AccessMode {\n constructor(acs) {\n if (acs) {\n this.given = typeof acs.given == 'number' ? acs.given : AccessMode.decode(acs.given);\n this.want = typeof acs.want == 'number' ? acs.want : AccessMode.decode(acs.want);\n this.mode = acs.mode ? (typeof acs.mode == 'number' ? acs.mode : AccessMode.decode(acs.mode)) :\n (this.given & this.want);\n }\n }\n\n static #checkFlag(val, side, flag) {\n side = side || 'mode';\n if (['given', 'want', 'mode'].includes(side)) {\n return ((val[side] & flag) != 0);\n }\n throw new Error(`Invalid AccessMode component '${side}'`);\n }\n /**\n * Parse string into an access mode value.\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {string | Number} mode - either a String representation of the access mode to parse or a set of bits to assign.\n * @returns {number} - Access mode as a numeric value.\n */\n static decode(str) {\n if (!str) {\n return null;\n } else if (typeof str == 'number') {\n return str & AccessMode._BITMASK;\n } else if (str === 'N' || str === 'n') {\n return AccessMode._NONE;\n }\n\n const bitmask = {\n 'J': AccessMode._JOIN,\n 'R': AccessMode._READ,\n 'W': AccessMode._WRITE,\n 'P': AccessMode._PRES,\n 'A': AccessMode._APPROVE,\n 'S': AccessMode._SHARE,\n 'D': AccessMode._DELETE,\n 'O': AccessMode._OWNER\n };\n\n let m0 = AccessMode._NONE;\n\n for (let i = 0; i < str.length; i++) {\n const bit = bitmask[str.charAt(i).toUpperCase()];\n if (!bit) {\n // Unrecognized bit, skip.\n continue;\n }\n m0 |= bit;\n }\n return m0;\n }\n /**\n * Convert numeric representation of the access mode into a string.\n *\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {number} val - access mode value to convert to a string.\n * @returns {string} - Access mode as a string.\n */\n static encode(val) {\n if (val === null || val === AccessMode._INVALID) {\n return null;\n } else if (val === AccessMode._NONE) {\n return 'N';\n }\n\n const bitmask = ['J', 'R', 'W', 'P', 'A', 'S', 'D', 'O'];\n let res = '';\n for (let i = 0; i < bitmask.length; i++) {\n if ((val & (1 << i)) != 0) {\n res = res + bitmask[i];\n }\n }\n return res;\n }\n /**\n * Update numeric representation of access mode with the new value. The value\n * is one of the following:\n * - a string starting with '+'
or '-'
then the bits to add or remove, e.g. '+R-W'
or '-PS'
.\n * - a new value of access mode\n *\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {number} val - access mode value to update.\n * @param {string} upd - update to apply to val.\n * @returns {number} - updated access mode.\n */\n static update(val, upd) {\n if (!upd || typeof upd != 'string') {\n return val;\n }\n\n let action = upd.charAt(0);\n if (action == '+' || action == '-') {\n let val0 = val;\n // Split delta-string like '+ABC-DEF+Z' into an array of parts including + and -.\n const parts = upd.split(/([-+])/);\n // Starting iteration from 1 because String.split() creates an array with the first empty element.\n // Iterating by 2 because we parse pairs +/- then data.\n for (let i = 1; i < parts.length - 1; i += 2) {\n action = parts[i];\n const m0 = AccessMode.decode(parts[i + 1]);\n if (m0 == AccessMode._INVALID) {\n return val;\n }\n if (m0 == null) {\n continue;\n }\n if (action === '+') {\n val0 |= m0;\n } else if (action === '-') {\n val0 &= ~m0;\n }\n }\n val = val0;\n } else {\n // The string is an explicit new value 'ABC' rather than delta.\n const val0 = AccessMode.decode(upd);\n if (val0 != AccessMode._INVALID) {\n val = val0;\n }\n }\n\n return val;\n }\n /**\n * Bits present in a1 but missing in a2.\n *\n * @static\n * @memberof Tinode\n *\n * @param {number | string} a1 - access mode to subtract from.\n * @param {number | string} a2 - access mode to subtract.\n * @returns {number} access mode with bits present in a1
but missing in a2
.\n */\n static diff(a1, a2) {\n a1 = AccessMode.decode(a1);\n a2 = AccessMode.decode(a2);\n\n if (a1 == AccessMode._INVALID || a2 == AccessMode._INVALID) {\n return AccessMode._INVALID;\n }\n return a1 & ~a2;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Custom formatter\n */\n toString() {\n return '{\"mode\": \"' + AccessMode.encode(this.mode) +\n '\", \"given\": \"' + AccessMode.encode(this.given) +\n '\", \"want\": \"' + AccessMode.encode(this.want) + '\"}';\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Converts numeric values to strings.\n */\n jsonHelper() {\n return {\n mode: AccessMode.encode(this.mode),\n given: AccessMode.encode(this.given),\n want: AccessMode.encode(this.want)\n };\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign value to 'mode'.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} m - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setMode(m) {\n this.mode = AccessMode.decode(m);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update mode
value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateMode(u) {\n this.mode = AccessMode.update(this.mode, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get mode
value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - mode
value.\n */\n getMode() {\n return AccessMode.encode(this.mode);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign given
value.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} g - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setGiven(g) {\n this.given = AccessMode.decode(g);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'given' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateGiven(u) {\n this.given = AccessMode.update(this.given, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get 'given' value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - given value.\n */\n getGiven() {\n return AccessMode.encode(this.given);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign 'want' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} w - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setWant(w) {\n this.want = AccessMode.decode(w);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'want' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateWant(u) {\n this.want = AccessMode.update(this.want, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get 'want' value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - want value.\n */\n getWant() {\n return AccessMode.encode(this.want);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get permissions present in 'want' but missing in 'given'.\n * Inverse of {@link Tinode.AccessMode#getExcessive}\n *\n * @memberof Tinode.AccessMode\n *\n * @returns {string} permissions present in want but missing in given.\n */\n getMissing() {\n return AccessMode.encode(this.want & ~this.given);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get permissions present in 'given' but missing in 'want'.\n * Inverse of {@link Tinode.AccessMode#getMissing}\n * @memberof Tinode.AccessMode\n *\n * @returns {string} permissions present in given but missing in want.\n */\n getExcessive() {\n return AccessMode.encode(this.given & ~this.want);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'want', 'give', and 'mode' values.\n * @memberof Tinode.AccessMode\n *\n * @param {AccessMode} val - new access mode value.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateAll(val) {\n if (val) {\n this.updateGiven(val.given);\n this.updateWant(val.want);\n this.mode = this.given & this.want;\n }\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Owner (O) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isOwner(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._OWNER);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Presence (P) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isPresencer(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._PRES);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Presence (P) flag is NOT set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isMuted(side) {\n return !this.isPresencer(side);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Join (J) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isJoiner(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._JOIN);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Reader (R) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isReader(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._READ);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Writer (W) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isWriter(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._WRITE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Approver (A) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isApprover(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._APPROVE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if either one of Owner (O) or Approver (A) flags is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isAdmin(side) {\n return this.isOwner(side) || this.isApprover(side);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if either one of Owner (O), Approver (A), or Sharer (S) flags is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isSharer(side) {\n return this.isAdmin(side) || AccessMode.#checkFlag(this, side, AccessMode._SHARE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Deleter (D) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isDeleter(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._DELETE);\n }\n}\n\nAccessMode._NONE = 0x00;\nAccessMode._JOIN = 0x01;\nAccessMode._READ = 0x02;\nAccessMode._WRITE = 0x04;\nAccessMode._PRES = 0x08;\nAccessMode._APPROVE = 0x10;\nAccessMode._SHARE = 0x20;\nAccessMode._DELETE = 0x40;\nAccessMode._OWNER = 0x80;\n\nAccessMode._BITMASK = AccessMode._JOIN | AccessMode._READ | AccessMode._WRITE | AccessMode._PRES |\n AccessMode._APPROVE | AccessMode._SHARE | AccessMode._DELETE | AccessMode._OWNER;\nAccessMode._INVALID = 0x100000;\n","export const PACKAGE_VERSION = \"0.22.11\";\n","/**\n * @file Global constants and configuration parameters.\n *\n * @copyright 2015-2023 Tinode LLC\n */\n'use strict';\n\nimport {\n PACKAGE_VERSION\n} from '../version.js';\n\n// Global constants\nexport const PROTOCOL_VERSION = '0'; // Major component of the version, e.g. '0' in '0.17.1'.\nexport const VERSION = PACKAGE_VERSION || '0.21';\nexport const LIBRARY = 'tinodejs/' + VERSION;\n\n// Topic name prefixes.\nexport const TOPIC_NEW = 'new';\nexport const TOPIC_NEW_CHAN = 'nch';\nexport const TOPIC_ME = 'me';\nexport const TOPIC_FND = 'fnd';\nexport const TOPIC_SYS = 'sys';\nexport const TOPIC_CHAN = 'chn';\nexport const TOPIC_GRP = 'grp';\nexport const TOPIC_P2P = 'p2p';\nexport const USER_NEW = 'new';\n\n// Starting value of a locally-generated seqId used for pending messages.\nexport const LOCAL_SEQID = 0xFFFFFFF;\n\n// Status codes.\nexport const MESSAGE_STATUS_NONE = 0; // Status not assigned.\nexport const MESSAGE_STATUS_QUEUED = 10; // Local ID assigned, in progress to be sent.\nexport const MESSAGE_STATUS_SENDING = 20; // Transmission started.\nexport const MESSAGE_STATUS_FAILED = 30; // At least one attempt was made to send the message.\nexport const MESSAGE_STATUS_FATAL = 40; // Message sending failed and it should not be retried.\nexport const MESSAGE_STATUS_SENT = 50; // Delivered to the server.\nexport const MESSAGE_STATUS_RECEIVED = 60; // Received by the client.\nexport const MESSAGE_STATUS_READ = 70; // Read by the user.\nexport const MESSAGE_STATUS_TO_ME = 80; // The message is received from another user.\n\n// Reject unresolved futures after this many milliseconds.\nexport const EXPIRE_PROMISES_TIMEOUT = 5_000;\n// Periodicity of garbage collection of unresolved futures.\nexport const EXPIRE_PROMISES_PERIOD = 1_000;\n\n// Delay before acknowledging that a message was recived.\nexport const RECV_TIMEOUT = 100;\n\n// Default number of messages to pull into memory from persistent cache.\nexport const DEFAULT_MESSAGES_PAGE = 24;\n\n// Unicode DEL character indicating data was deleted.\nexport const DEL_CHAR = '\\u2421';\n","/**\n * @file Throwable error with numeric error code.\n *\n * @copyright 2015-2023 Tinode LLC.\n */\n'use strict';\n\nexport default class CommError extends Error {\n constructor(message, code) {\n super(`${message} (${code})`);\n this.name = 'CommError';\n this.code = code;\n }\n}\n","/**\n * @file Utilities used in multiple places.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport AccessMode from './access-mode.js';\nimport {\n DEL_CHAR\n} from './config.js';\n\n// Attempt to convert date and AccessMode strings to objects.\nexport function jsonParseHelper(key, val) {\n // Try to convert string timestamps with optional milliseconds to Date,\n // e.g. 2015-09-02T01:45:43[.123]Z\n if (typeof val == 'string' && val.length >= 20 && val.length <= 24 && ['ts', 'touched', 'updated', 'created', 'when', 'deleted', 'expires'].includes(key)) {\n const date = new Date(val);\n if (!isNaN(date)) {\n return date;\n }\n } else if (key === 'acs' && typeof val === 'object') {\n return new AccessMode(val);\n }\n return val;\n}\n\n// Checks if URL is a relative url, i.e. has no 'scheme://', including the case of missing scheme '//'.\n// The scheme is expected to be RFC-compliant, e.g. [a-z][a-z0-9+.-]*\n// example.html - ok\n// https:example.com - not ok.\n// http:/example.com - not ok.\n// ' ↲ https://example.com' - not ok. (↲ means carriage return)\nexport function isUrlRelative(url) {\n return url && !/^\\s*([a-z][a-z0-9+.-]*:|\\/\\/)/im.test(url);\n}\n\nfunction isValidDate(d) {\n return (d instanceof Date) && !isNaN(d) && (d.getTime() != 0);\n}\n\n// RFC3339 formater of Date\nexport function rfc3339DateString(d) {\n if (!isValidDate(d)) {\n return undefined;\n }\n\n const pad = function(val, sp) {\n sp = sp || 2;\n return '0'.repeat(sp - ('' + val).length) + val;\n };\n\n const millis = d.getUTCMilliseconds();\n return d.getUTCFullYear() + '-' + pad(d.getUTCMonth() + 1) + '-' + pad(d.getUTCDate()) +\n 'T' + pad(d.getUTCHours()) + ':' + pad(d.getUTCMinutes()) + ':' + pad(d.getUTCSeconds()) +\n (millis ? '.' + pad(millis, 3) : '') + 'Z';\n}\n\n// Recursively merge src's own properties to dst.\n// Ignore properties where ignore[property] is true.\n// Array and Date objects are shallow-copied.\nexport function mergeObj(dst, src, ignore) {\n if (typeof src != 'object') {\n if (src === undefined) {\n return dst;\n }\n if (src === DEL_CHAR) {\n return undefined;\n }\n return src;\n }\n // JS is crazy: typeof null is 'object'.\n if (src === null) {\n return src;\n }\n\n // Handle Date\n if (src instanceof Date && !isNaN(src)) {\n return (!dst || !(dst instanceof Date) || isNaN(dst) || dst < src) ? src : dst;\n }\n\n // Access mode\n if (src instanceof AccessMode) {\n return new AccessMode(src);\n }\n\n // Handle Array\n if (src instanceof Array) {\n return src;\n }\n\n if (!dst || dst === DEL_CHAR) {\n dst = src.constructor();\n }\n\n for (let prop in src) {\n if (src.hasOwnProperty(prop) && (!ignore || !ignore[prop]) && (prop != '_noForwarding')) {\n try {\n dst[prop] = mergeObj(dst[prop], src[prop]);\n } catch (err) {\n // FIXME: probably need to log something here.\n }\n }\n }\n return dst;\n}\n\n// Update object stored in a cache. Returns updated value.\nexport function mergeToCache(cache, key, newval, ignore) {\n cache[key] = mergeObj(cache[key], newval, ignore);\n return cache[key];\n}\n\n// Strips all values from an object of they evaluate to false or if their name starts with '_'.\n// Used on all outgoing object before serialization to string.\nexport function simplify(obj) {\n Object.keys(obj).forEach((key) => {\n if (key[0] == '_') {\n // Strip fields like \"obj._key\".\n delete obj[key];\n } else if (!obj[key]) {\n // Strip fields which evaluate to false.\n delete obj[key];\n } else if (Array.isArray(obj[key]) && obj[key].length == 0) {\n // Strip empty arrays.\n delete obj[key];\n } else if (!obj[key]) {\n // Strip fields which evaluate to false.\n delete obj[key];\n } else if (obj[key] instanceof Date) {\n // Strip invalid or zero date.\n if (!isValidDate(obj[key])) {\n delete obj[key];\n }\n } else if (typeof obj[key] == 'object') {\n simplify(obj[key]);\n // Strip empty objects.\n if (Object.getOwnPropertyNames(obj[key]).length == 0) {\n delete obj[key];\n }\n }\n });\n return obj;\n};\n\n\n// Trim whitespace, strip empty and duplicate elements elements.\n// If the result is an empty array, add a single element \"\\u2421\" (Unicode Del character).\nexport function normalizeArray(arr) {\n let out = [];\n if (Array.isArray(arr)) {\n // Trim, throw away very short and empty tags.\n for (let i = 0, l = arr.length; i < l; i++) {\n let t = arr[i];\n if (t) {\n t = t.trim().toLowerCase();\n if (t.length > 1) {\n out.push(t);\n }\n }\n }\n out.sort().filter(function(item, pos, ary) {\n return !pos || item != ary[pos - 1];\n });\n }\n if (out.length == 0) {\n // Add single tag with a Unicode Del character, otherwise an ampty array\n // is ambiguos. The Del tag will be stripped by the server.\n out.push(DEL_CHAR);\n }\n return out;\n}\n","/**\n * @file Abstraction layer for websocket and long polling connections.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport CommError from './comm-error.js';\nimport {\n jsonParseHelper\n} from './utils.js';\n\nlet WebSocketProvider;\nlet XHRProvider;\n\n// Error code to return in case of a network problem.\nconst NETWORK_ERROR = 503;\nconst NETWORK_ERROR_TEXT = \"Connection failed\";\n\n// Error code to return when user disconnected from server.\nconst NETWORK_USER = 418;\nconst NETWORK_USER_TEXT = \"Disconnected by client\";\n\n// Settings for exponential backoff\nconst _BOFF_BASE = 2000; // 2000 milliseconds, minimum delay between reconnects\nconst _BOFF_MAX_ITER = 10; // Maximum delay between reconnects 2^10 * 2000 ~ 34 minutes\nconst _BOFF_JITTER = 0.3; // Add random delay\n\n// Helper function for creating an endpoint URL.\nfunction makeBaseUrl(host, protocol, version, apiKey) {\n let url = null;\n\n if (['http', 'https', 'ws', 'wss'].includes(protocol)) {\n url = `${protocol}://${host}`;\n if (url.charAt(url.length - 1) !== '/') {\n url += '/';\n }\n url += 'v' + version + '/channels';\n if (['http', 'https'].includes(protocol)) {\n // Long polling endpoint ends with \"lp\", i.e.\n // '/v0/channels/lp' vs just '/v0/channels' for ws\n url += '/lp';\n }\n url += '?apikey=' + apiKey;\n }\n return url;\n}\n\n/**\n * An abstraction for a websocket or a long polling connection.\n *\n * @class Connection\n * @memberof Tinode\n\n * @param {Object} config - configuration parameters.\n * @param {string} config.host - Host name and optional port number to connect to.\n * @param {string} config.apiKey - API key generated by keygen
.\n * @param {string} config.transport - Network transport to use, either \"ws\"/\"wss\"
for websocket or\n * lp
for long polling.\n * @param {boolean} config.secure - Use Secure WebSocket if true
.\n * @param {string} version_ - Major value of the protocol version, e.g. '0' in '0.17.1'.\n * @param {boolean} autoreconnect_ - If connection is lost, try to reconnect automatically.\n */\nexport default class Connection {\n // Logger, does nothing by default.\n static #log = _ => {};\n\n #boffTimer = null;\n #boffIteration = 0;\n #boffClosed = false; // Indicator if the socket was manually closed - don't autoreconnect if true.\n\n // Websocket.\n #socket = null;\n\n host;\n secure;\n apiKey;\n\n version;\n autoreconnect;\n\n initialized;\n\n // (config.host, config.apiKey, config.transport, config.secure), PROTOCOL_VERSION, true\n constructor(config, version_, autoreconnect_) {\n this.host = config.host;\n this.secure = config.secure;\n this.apiKey = config.apiKey;\n\n this.version = version_;\n this.autoreconnect = autoreconnect_;\n\n if (config.transport === 'lp') {\n // explicit request to use long polling\n this.#init_lp();\n this.initialized = 'lp';\n } else if (config.transport === 'ws') {\n // explicit request to use web socket\n // if websockets are not available, horrible things will happen\n this.#init_ws();\n this.initialized = 'ws';\n }\n\n if (!this.initialized) {\n // Invalid or undefined network transport.\n Connection.#log(\"Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.\");\n throw new Error(\"Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.\");\n }\n }\n\n /**\n * To use Connection in a non browser context, supply WebSocket and XMLHttpRequest providers.\n * @static\n * @memberof Connection\n * @param wsProvider WebSocket provider, e.g. for nodeJS , require('ws')
.\n * @param xhrProvider XMLHttpRequest provider, e.g. for node require('xhr')
.\n */\n static setNetworkProviders(wsProvider, xhrProvider) {\n WebSocketProvider = wsProvider;\n XHRProvider = xhrProvider;\n }\n\n /**\n * Assign a non-default logger.\n * @static\n * @memberof Connection\n * @param {function} l variadic logging function.\n */\n static set logger(l) {\n Connection.#log = l;\n }\n\n /**\n * Initiate a new connection\n * @memberof Tinode.Connection#\n * @param {string} host_ Host name to connect to; if null
the old host name will be used.\n * @param {boolean} force Force new connection even if one already exists.\n * @return {Promise} Promise resolved/rejected when the connection call completes, resolution is called without\n * parameters, rejection passes the {Error} as parameter.\n */\n connect(host_, force) {\n return Promise.reject(null);\n }\n\n /**\n * Try to restore a network connection, also reset backoff.\n * @memberof Tinode.Connection#\n *\n * @param {boolean} force - reconnect even if there is a live connection already.\n */\n reconnect(force) {}\n\n /**\n * Terminate the network connection\n * @memberof Tinode.Connection#\n */\n disconnect() {}\n\n /**\n * Send a string to the server.\n * @memberof Tinode.Connection#\n *\n * @param {string} msg - String to send.\n * @throws Throws an exception if the underlying connection is not live.\n */\n sendText(msg) {}\n\n /**\n * Check if connection is alive.\n * @memberof Tinode.Connection#\n * @returns {boolean} true
if connection is live, false
otherwise.\n */\n isConnected() {\n return false;\n }\n\n /**\n * Get the name of the current network transport.\n * @memberof Tinode.Connection#\n * @returns {string} name of the transport such as \"ws\"
or \"lp\"
.\n */\n transport() {\n return this.initialized;\n }\n\n /**\n * Send network probe to check if connection is indeed live.\n * @memberof Tinode.Connection#\n */\n probe() {\n this.sendText('1');\n }\n\n /**\n * Reset autoreconnect counter to zero.\n * @memberof Tinode.Connection#\n */\n backoffReset() {\n this.#boffReset();\n }\n\n // Backoff implementation - reconnect after a timeout.\n #boffReconnect() {\n // Clear timer\n clearTimeout(this.#boffTimer);\n // Calculate when to fire the reconnect attempt\n const timeout = _BOFF_BASE * (Math.pow(2, this.#boffIteration) * (1.0 + _BOFF_JITTER * Math.random()));\n // Update iteration counter for future use\n this.#boffIteration = (this.#boffIteration >= _BOFF_MAX_ITER ? this.#boffIteration : this.#boffIteration + 1);\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(timeout);\n }\n\n this.#boffTimer = setTimeout(_ => {\n Connection.#log(`Reconnecting, iter=${this.#boffIteration}, timeout=${timeout}`);\n // Maybe the socket was closed while we waited for the timer?\n if (!this.#boffClosed) {\n const prom = this.connect();\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(0, prom);\n } else {\n // Suppress error if it's not used.\n prom.catch(_ => {\n /* do nothing */\n });\n }\n } else if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(-1);\n }\n }, timeout);\n }\n\n // Terminate auto-reconnect process.\n #boffStop() {\n clearTimeout(this.#boffTimer);\n this.#boffTimer = null;\n }\n\n // Reset auto-reconnect iteration counter.\n #boffReset() {\n this.#boffIteration = 0;\n }\n\n // Initialization for long polling.\n #init_lp() {\n const XDR_UNSENT = 0; // Client has been created. open() not called yet.\n const XDR_OPENED = 1; // open() has been called.\n const XDR_HEADERS_RECEIVED = 2; // send() has been called, and headers and status are available.\n const XDR_LOADING = 3; // Downloading; responseText holds partial data.\n const XDR_DONE = 4; // The operation is complete.\n\n // Fully composed endpoint URL, with API key & SID\n let _lpURL = null;\n\n let _poller = null;\n let _sender = null;\n\n let lp_sender = (url_) => {\n const sender = new XHRProvider();\n sender.onreadystatechange = (evt) => {\n if (sender.readyState == XDR_DONE && sender.status >= 400) {\n // Some sort of error response\n throw new CommError(\"LP sender failed\", sender.status);\n }\n };\n\n sender.open('POST', url_, true);\n return sender;\n }\n\n let lp_poller = (url_, resolve, reject) => {\n let poller = new XHRProvider();\n let promiseCompleted = false;\n\n poller.onreadystatechange = evt => {\n if (poller.readyState == XDR_DONE) {\n if (poller.status == 201) { // 201 == HTTP.Created, get SID\n let pkt = JSON.parse(poller.responseText, jsonParseHelper);\n _lpURL = url_ + '&sid=' + pkt.ctrl.params.sid;\n poller = lp_poller(_lpURL);\n poller.send(null);\n if (this.onOpen) {\n this.onOpen();\n }\n\n if (resolve) {\n promiseCompleted = true;\n resolve();\n }\n\n if (this.autoreconnect) {\n this.#boffStop();\n }\n } else if (poller.status > 0 && poller.status < 400) { // 0 = network error; 400 = HTTP.BadRequest\n if (this.onMessage) {\n this.onMessage(poller.responseText);\n }\n poller = lp_poller(_lpURL);\n poller.send(null);\n } else {\n // Don't throw an error here, gracefully handle server errors\n if (reject && !promiseCompleted) {\n promiseCompleted = true;\n reject(poller.responseText);\n }\n if (this.onMessage && poller.responseText) {\n this.onMessage(poller.responseText);\n }\n if (this.onDisconnect) {\n const code = poller.status || (this.#boffClosed ? NETWORK_USER : NETWORK_ERROR);\n const text = poller.responseText || (this.#boffClosed ? NETWORK_USER_TEXT : NETWORK_ERROR_TEXT);\n this.onDisconnect(new CommError(text, code), code);\n }\n\n // Polling has stopped. Indicate it by setting poller to null.\n poller = null;\n if (!this.#boffClosed && this.autoreconnect) {\n this.#boffReconnect();\n }\n }\n }\n };\n // Using POST to avoid caching response by service worker.\n poller.open('POST', url_, true);\n return poller;\n }\n\n this.connect = (host_, force) => {\n this.#boffClosed = false;\n\n if (_poller) {\n if (!force) {\n return Promise.resolve();\n }\n _poller.onreadystatechange = undefined;\n _poller.abort();\n _poller = null;\n }\n\n if (host_) {\n this.host = host_;\n }\n\n return new Promise((resolve, reject) => {\n const url = makeBaseUrl(this.host, this.secure ? 'https' : 'http', this.version, this.apiKey);\n Connection.#log(\"LP connecting to:\", url);\n _poller = lp_poller(url, resolve, reject);\n _poller.send(null);\n }).catch(err => {\n Connection.#log(\"LP connection failed:\", err);\n });\n };\n\n this.reconnect = force => {\n this.#boffStop();\n this.connect(null, force);\n };\n\n this.disconnect = _ => {\n this.#boffClosed = true;\n this.#boffStop();\n\n if (_sender) {\n _sender.onreadystatechange = undefined;\n _sender.abort();\n _sender = null;\n }\n if (_poller) {\n _poller.onreadystatechange = undefined;\n _poller.abort();\n _poller = null;\n }\n\n if (this.onDisconnect) {\n this.onDisconnect(new CommError(NETWORK_USER_TEXT, NETWORK_USER), NETWORK_USER);\n }\n // Ensure it's reconstructed\n _lpURL = null;\n };\n\n this.sendText = (msg) => {\n _sender = lp_sender(_lpURL);\n if (_sender && (_sender.readyState == XDR_OPENED)) { // 1 == OPENED\n _sender.send(msg);\n } else {\n throw new Error(\"Long poller failed to connect\");\n }\n };\n\n this.isConnected = _ => {\n return (_poller && true);\n };\n }\n\n // Initialization for Websocket\n #init_ws() {\n this.connect = (host_, force) => {\n this.#boffClosed = false;\n\n if (this.#socket) {\n if (!force && this.#socket.readyState == this.#socket.OPEN) {\n return Promise.resolve();\n }\n this.#socket.close();\n this.#socket = null;\n }\n\n if (host_) {\n this.host = host_;\n }\n\n return new Promise((resolve, reject) => {\n const url = makeBaseUrl(this.host, this.secure ? 'wss' : 'ws', this.version, this.apiKey);\n\n Connection.#log(\"WS connecting to: \", url);\n\n // It throws when the server is not accessible but the exception cannot be caught:\n // https://stackoverflow.com/questions/31002592/javascript-doesnt-catch-error-in-websocket-instantiation/31003057\n const conn = new WebSocketProvider(url);\n\n conn.onerror = err => {\n reject(err);\n };\n\n conn.onopen = _ => {\n if (this.autoreconnect) {\n this.#boffStop();\n }\n\n if (this.onOpen) {\n this.onOpen();\n }\n\n resolve();\n };\n\n conn.onclose = _ => {\n this.#socket = null;\n\n if (this.onDisconnect) {\n const code = this.#boffClosed ? NETWORK_USER : NETWORK_ERROR;\n this.onDisconnect(new CommError(this.#boffClosed ? NETWORK_USER_TEXT : NETWORK_ERROR_TEXT, code), code);\n }\n\n if (!this.#boffClosed && this.autoreconnect) {\n this.#boffReconnect();\n }\n };\n\n conn.onmessage = evt => {\n if (this.onMessage) {\n this.onMessage(evt.data);\n }\n };\n\n this.#socket = conn;\n });\n }\n\n this.reconnect = force => {\n this.#boffStop();\n this.connect(null, force);\n };\n\n this.disconnect = _ => {\n this.#boffClosed = true;\n this.#boffStop();\n\n if (!this.#socket) {\n return;\n }\n this.#socket.close();\n this.#socket = null;\n };\n\n this.sendText = msg => {\n if (this.#socket && (this.#socket.readyState == this.#socket.OPEN)) {\n this.#socket.send(msg);\n } else {\n throw new Error(\"Websocket is not connected\");\n }\n };\n\n this.isConnected = _ => {\n return (this.#socket && (this.#socket.readyState == this.#socket.OPEN));\n };\n }\n\n // Callbacks:\n\n /**\n * A callback to pass incoming messages to. See {@link Tinode.Connection#onMessage}.\n * @callback Tinode.Connection.OnMessage\n * @memberof Tinode.Connection\n * @param {string} message - Message to process.\n */\n onMessage = undefined;\n\n /**\n * A callback for reporting a dropped connection.\n * @type {function}\n * @memberof Tinode.Connection#\n */\n onDisconnect = undefined;\n\n /**\n * A callback called when the connection is ready to be used for sending. For websockets it's socket open,\n * for long polling it's readyState=1
(OPENED)\n * @type {function}\n * @memberof Tinode.Connection#\n */\n onOpen = undefined;\n\n /**\n * A callback to notify of reconnection attempts. See {@link Tinode.Connection#onAutoreconnectIteration}.\n * @memberof Tinode.Connection\n * @callback AutoreconnectIterationType\n * @param {string} timeout - time till the next reconnect attempt in milliseconds. -1
means reconnect was skipped.\n * @param {Promise} promise resolved or rejected when the reconnect attemp completes.\n *\n */\n /**\n * A callback to inform when the next attampt to reconnect will happen and to receive connection promise.\n * @memberof Tinode.Connection#\n * @type {Tinode.Connection.AutoreconnectIterationType}\n */\n onAutoreconnectIteration = undefined;\n}\n\nConnection.NETWORK_ERROR = NETWORK_ERROR;\nConnection.NETWORK_ERROR_TEXT = NETWORK_ERROR_TEXT;\nConnection.NETWORK_USER = NETWORK_USER;\nConnection.NETWORK_USER_TEXT = NETWORK_USER_TEXT;\n","/**\n * @file Helper methods for dealing with IndexedDB cache of messages, users, and topics.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nconst DB_VERSION = 1;\nconst DB_NAME = 'tinode-web';\n\nlet IDBProvider;\n\nexport default class DB {\n #onError = _ => {};\n #logger = _ => {};\n\n // Instance of IndexDB.\n db = null;\n // Indicator that the cache is disabled.\n disabled = true;\n\n constructor(onError, logger) {\n this.#onError = onError || this.#onError;\n this.#logger = logger || this.#logger;\n }\n\n #mapObjects(source, callback, context) {\n if (!this.db) {\n return disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction([source]);\n trx.onerror = event => {\n this.#logger('PCache', 'mapObjects', source, event.target.error);\n reject(event.target.error);\n };\n trx.objectStore(source).getAll().onsuccess = event => {\n if (callback) {\n event.target.result.forEach(topic => {\n callback.call(context, topic);\n });\n }\n resolve(event.target.result);\n };\n });\n }\n\n /**\n * Initialize persistent cache: open or create/upgrade if needed.\n * @returns {Promise} promise to be resolved/rejected when the DB is initialized.\n */\n initDatabase() {\n return new Promise((resolve, reject) => {\n // Open the database and initialize callbacks.\n const req = IDBProvider.open(DB_NAME, DB_VERSION);\n req.onsuccess = event => {\n this.db = event.target.result;\n this.disabled = false;\n resolve(this.db);\n };\n req.onerror = event => {\n this.#logger('PCache', \"failed to initialize\", event);\n reject(event.target.error);\n this.#onError(event.target.error);\n };\n req.onupgradeneeded = event => {\n this.db = event.target.result;\n\n this.db.onerror = event => {\n this.#logger('PCache', \"failed to create storage\", event);\n this.#onError(event.target.error);\n };\n\n // Individual object stores.\n // Object store (table) for topics. The primary key is topic name.\n this.db.createObjectStore('topic', {\n keyPath: 'name'\n });\n\n // Users object store. UID is the primary key.\n this.db.createObjectStore('user', {\n keyPath: 'uid'\n });\n\n // Subscriptions object store topic <-> user. Topic name + UID is the primary key.\n this.db.createObjectStore('subscription', {\n keyPath: ['topic', 'uid']\n });\n\n // Messages object store. The primary key is topic name + seq.\n this.db.createObjectStore('message', {\n keyPath: ['topic', 'seq']\n });\n };\n });\n }\n\n /**\n * Delete persistent cache.\n */\n deleteDatabase() {\n // Close connection, otherwise operations will fail with 'onblocked'.\n if (this.db) {\n this.db.close();\n this.db = null;\n }\n return new Promise((resolve, reject) => {\n const req = IDBProvider.deleteDatabase(DB_NAME);\n req.onblocked = _ => {\n if (this.db) {\n this.db.close();\n }\n const err = new Error(\"blocked\");\n this.#logger('PCache', 'deleteDatabase', err);\n reject(err);\n };\n req.onsuccess = _ => {\n this.db = null;\n this.disabled = true;\n resolve(true);\n };\n req.onerror = event => {\n this.#logger('PCache', 'deleteDatabase', event.target.error);\n reject(event.target.error);\n };\n });\n }\n\n /**\n * Check if persistent cache is ready for use.\n * @memberOf DB\n * @returns {boolean} true
if cache is ready, false
otherwise.\n */\n isReady() {\n return !!this.db;\n }\n\n // Topics.\n\n /**\n * Save to cache or update topic in persistent cache.\n * @memberOf DB\n * @param {Topic} topic - topic to be added or updated.\n * @returns {Promise} promise resolved/rejected on operation completion.\n */\n updTopic(topic) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updTopic', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('topic').get(topic.name);\n req.onsuccess = _ => {\n trx.objectStore('topic').put(DB.#serializeTopic(req.result, topic));\n trx.commit();\n };\n });\n }\n\n /**\n * Mark or unmark topic as deleted.\n * @memberOf DB\n * @param {string} name - name of the topic to mark or unmark.\n * @param {boolean} deleted - status\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n markTopicAsDeleted(name, deleted) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'markTopicAsDeleted', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('topic').get(name);\n req.onsuccess = event => {\n const topic = event.target.result;\n if (topic && topic._deleted != deleted) {\n topic._deleted = deleted;\n trx.objectStore('topic').put(topic);\n }\n trx.commit();\n };\n });\n }\n\n /**\n * Remove topic from persistent cache.\n * @memberOf DB\n * @param {string} name - name of the topic to remove from database.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remTopic(name) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic', 'subscription', 'message'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'remTopic', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('topic').delete(IDBKeyRange.only(name));\n trx.objectStore('subscription').delete(IDBKeyRange.bound([name, '-'], [name, '~']));\n trx.objectStore('message').delete(IDBKeyRange.bound([name, 0], [name, Number.MAX_SAFE_INTEGER]));\n trx.commit();\n });\n }\n\n /**\n * Execute a callback for each stored topic.\n * @memberOf DB\n * @param {function} callback - function to call for each topic.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapTopics(callback, context) {\n return this.#mapObjects('topic', callback, context);\n }\n\n /**\n * Copy data from serialized object to topic.\n * @memberOf DB\n * @param {Topic} topic - target to deserialize to.\n * @param {Object} src - serialized data to copy from.\n */\n deserializeTopic(topic, src) {\n DB.#deserializeTopic(topic, src);\n }\n\n // Users.\n /**\n * Add or update user object in the persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to save or update.\n * @param {Object} pub - user's public
information.\n * @returns {Promise} promise resolved/rejected on operation completion.\n */\n updUser(uid, pub) {\n if (arguments.length < 2 || pub === undefined) {\n // No point inupdating user with invalid data.\n return;\n }\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').put({\n uid: uid,\n public: pub\n });\n trx.commit();\n });\n }\n\n /**\n * Remove user from persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to remove from the cache.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remUser(uid) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'remUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').delete(IDBKeyRange.only(uid));\n trx.commit();\n });\n }\n\n /**\n * Execute a callback for each stored user.\n * @memberOf DB\n * @param {function} callback - function to call for each topic.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapUsers(callback, context) {\n return this.#mapObjects('user', callback, context);\n }\n\n /**\n * Read a single user from persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to fetch from cache.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n getUser(uid) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user']);\n trx.oncomplete = event => {\n const user = event.target.result;\n resolve({\n user: user.uid,\n public: user.public\n });\n };\n trx.onerror = event => {\n this.#logger('PCache', 'getUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').get(uid);\n });\n }\n\n // Subscriptions.\n /**\n * Add or update subscription in persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {string} uid - ID of the subscribed user.\n * @param {Object} sub - subscription to save.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n updSubscription(topicName, uid, sub) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['subscription'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updSubscription', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('subscription').get([topicName, uid]).onsuccess = (event) => {\n trx.objectStore('subscription').put(DB.#serializeSubscription(event.target.result, topicName, uid, sub));\n trx.commit();\n };\n });\n }\n\n /**\n * Execute a callback for each cached subscription in a given topic.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the subscriptions.\n * @param {function} callback - function to call for each subscription.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapSubscriptions(topicName, callback, context) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['subscription']);\n trx.onerror = (event) => {\n this.#logger('PCache', 'mapSubscriptions', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('subscription').getAll(IDBKeyRange.bound([topicName, '-'], [topicName, '~'])).onsuccess = (event) => {\n if (callback) {\n event.target.result.forEach((topic) => {\n callback.call(context, topic);\n });\n }\n resolve(event.target.result);\n };\n });\n }\n\n // Messages.\n\n /**\n * Save message to persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {Object} msg - message to save.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n addMessage(msg) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'addMessage', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('message').add(DB.#serializeMessage(null, msg));\n trx.commit();\n });\n }\n\n /**\n * Update delivery status of a message stored in persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {number} seq - ID of the message to update\n * @param {number} status - new delivery status of the message.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n updMessageStatus(topicName, seq, status) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updMessageStatus', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('message').get(IDBKeyRange.only([topicName, seq]));\n req.onsuccess = event => {\n const src = req.result || event.target.result;\n if (!src || src._status == status) {\n trx.commit();\n return;\n }\n trx.objectStore('message').put(DB.#serializeMessage(src, {\n topic: topicName,\n seq: seq,\n _status: status\n }));\n trx.commit();\n };\n });\n }\n\n /**\n * Remove one or more messages from persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {number} from - id of the message to remove or lower boundary when removing range (inclusive).\n * @param {number=} to - upper boundary (exclusive) when removing a range of messages.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remMessages(topicName, from, to) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n if (!from && !to) {\n from = 0;\n to = Number.MAX_SAFE_INTEGER;\n }\n const range = to > 0 ? IDBKeyRange.bound([topicName, from], [topicName, to], false, true) :\n IDBKeyRange.only([topicName, from]);\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = (event) => {\n resolve(event.target.result);\n };\n trx.onerror = (event) => {\n this.#logger('PCache', 'remMessages', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('message').delete(range);\n trx.commit();\n });\n }\n\n /**\n * Retrieve messages from persistent store.\n * @memberOf DB\n * @param {string} topicName - name of the topic to retrieve messages from.\n * @param {function} callback to call for each retrieved message.\n * @param {Object} query - parameters of the message range to retrieve.\n * @param {number=} query.since - the least message ID to retrieve (inclusive).\n * @param {number=} query.before - the greatest message ID to retrieve (exclusive).\n * @param {number=} query.limit - the maximum number of messages to retrieve.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n readMessages(topicName, query, callback, context) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n query = query || {};\n const since = query.since > 0 ? query.since : 0;\n const before = query.before > 0 ? query.before : Number.MAX_SAFE_INTEGER;\n const limit = query.limit | 0;\n\n const result = [];\n const range = IDBKeyRange.bound([topicName, since], [topicName, before], false, true);\n const trx = this.db.transaction(['message']);\n trx.onerror = (event) => {\n this.#logger('PCache', 'readMessages', event.target.error);\n reject(event.target.error);\n };\n // Iterate in descending order.\n trx.objectStore('message').openCursor(range, 'prev').onsuccess = (event) => {\n const cursor = event.target.result;\n if (cursor) {\n if (callback) {\n callback.call(context, cursor.value);\n }\n result.push(cursor.value);\n if (limit <= 0 || result.length < limit) {\n cursor.continue();\n } else {\n resolve(result);\n }\n } else {\n resolve(result);\n }\n };\n });\n }\n\n // Private methods.\n\n // Serializable topic fields.\n static #topic_fields = ['created', 'updated', 'deleted', 'read', 'recv', 'seq', 'clear', 'defacs',\n 'creds', 'public', 'trusted', 'private', 'touched', '_deleted'\n ];\n\n // Copy data from src to Topic object.\n static #deserializeTopic(topic, src) {\n DB.#topic_fields.forEach((f) => {\n if (src.hasOwnProperty(f)) {\n topic[f] = src[f];\n }\n });\n if (Array.isArray(src.tags)) {\n topic._tags = src.tags;\n }\n if (src.acs) {\n topic.setAccessMode(src.acs);\n }\n topic.seq |= 0;\n topic.read |= 0;\n topic.unread = Math.max(0, topic.seq - topic.read);\n }\n\n // Copy values from 'src' to 'dst'. Allocate dst if it's null or undefined.\n static #serializeTopic(dst, src) {\n const res = dst || {\n name: src.name\n };\n DB.#topic_fields.forEach((f) => {\n if (src.hasOwnProperty(f)) {\n res[f] = src[f];\n }\n });\n if (Array.isArray(src._tags)) {\n res.tags = src._tags;\n }\n if (src.acs) {\n res.acs = src.getAccessMode().jsonHelper();\n }\n return res;\n }\n\n static #serializeSubscription(dst, topicName, uid, sub) {\n const fields = ['updated', 'mode', 'read', 'recv', 'clear', 'lastSeen', 'userAgent'];\n const res = dst || {\n topic: topicName,\n uid: uid\n };\n\n fields.forEach((f) => {\n if (sub.hasOwnProperty(f)) {\n res[f] = sub[f];\n }\n });\n\n return res;\n }\n\n static #serializeMessage(dst, msg) {\n // Serializable fields.\n const fields = ['topic', 'seq', 'ts', '_status', 'from', 'head', 'content'];\n const res = dst || {};\n fields.forEach((f) => {\n if (msg.hasOwnProperty(f)) {\n res[f] = msg[f];\n }\n });\n return res;\n }\n\n /**\n * To use DB in a non browser context, supply indexedDB provider.\n * @static\n * @memberof DB\n * @param idbProvider indexedDB provider, e.g. for node require('fake-indexeddb')
.\n */\n static setDatabaseProvider(idbProvider) {\n IDBProvider = idbProvider;\n }\n}\n","/**\n * @file Utilities for uploading and downloading files.\n *\n * @copyright 2015-2023 Tinode LLC.\n */\n'use strict';\n\nimport CommError from './comm-error.js';\nimport {\n isUrlRelative,\n jsonParseHelper\n} from './utils.js';\n\nlet XHRProvider;\n\nfunction addURLParam(relUrl, key, value) {\n const url = new URL(relUrl, window.location.origin);\n url.searchParams.append(key, value);\n return url.toString().substring(window.location.origin.length);\n}\n\n/**\n * @class LargeFileHelper - utilities for uploading and downloading files out of band.\n * Don't instantiate this class directly. Use {Tinode.getLargeFileHelper} instead.\n * @memberof Tinode\n *\n * @param {Tinode} tinode - the main Tinode object.\n * @param {string} version - protocol version, i.e. '0'.\n */\nexport default class LargeFileHelper {\n constructor(tinode, version) {\n this._tinode = tinode;\n this._version = version;\n\n this._apiKey = tinode._apiKey;\n this._authToken = tinode.getAuthToken();\n\n // Ongoing requests.\n this.xhr = [];\n }\n\n /**\n * Start uploading the file to an endpoint at baseUrl.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {string} baseUrl base URL of upload server.\n * @param {File|Blob} data data to upload.\n * @param {string} avatarFor topic name if the upload represents an avatar.\n * @param {Callback} onProgress callback. Takes one {float} parameter 0..1\n * @param {Callback} onSuccess callback. Called when the file is successfully uploaded.\n * @param {Callback} onFailure callback. Called in case of a failure.\n *\n * @returns {Promise} resolved/rejected when the upload is completed/failed.\n */\n uploadWithBaseUrl(baseUrl, data, avatarFor, onProgress, onSuccess, onFailure) {\n let url = `/v${this._version}/file/u/`;\n if (baseUrl) {\n let base = baseUrl;\n if (base.endsWith('/')) {\n // Removing trailing slash.\n base = base.slice(0, -1);\n }\n if (base.startsWith('http://') || base.startsWith('https://')) {\n url = base + url;\n } else {\n throw new Error(`Invalid base URL '${baseUrl}'`);\n }\n }\n\n const instance = this;\n const xhr = new XHRProvider();\n this.xhr.push(xhr);\n\n xhr.open('POST', url, true);\n xhr.setRequestHeader('X-Tinode-APIKey', this._apiKey);\n if (this._authToken) {\n xhr.setRequestHeader('X-Tinode-Auth', `Token ${this._authToken.token}`);\n }\n\n let toResolve = null;\n let toReject = null;\n\n const result = new Promise((resolve, reject) => {\n toResolve = resolve;\n toReject = reject;\n });\n\n xhr.upload.onprogress = e => {\n if (e.lengthComputable) {\n if (onProgress) {\n onProgress(e.loaded / e.total);\n }\n if (this.onProgress) {\n this.onProgress(e.loaded / e.total);\n }\n }\n };\n\n xhr.onload = function() {\n let pkt;\n try {\n pkt = JSON.parse(this.response, jsonParseHelper);\n } catch (err) {\n instance._tinode.logger(\"ERROR: Invalid server response in LargeFileHelper\", this.response);\n pkt = {\n ctrl: {\n code: this.status,\n text: this.statusText\n }\n };\n }\n\n if (this.status >= 200 && this.status < 300) {\n if (toResolve) {\n toResolve(pkt.ctrl.params.url);\n }\n if (onSuccess) {\n onSuccess(pkt.ctrl);\n }\n } else if (this.status >= 400) {\n if (toReject) {\n toReject(new CommError(pkt.ctrl.text, pkt.ctrl.code));\n }\n if (onFailure) {\n onFailure(pkt.ctrl);\n }\n } else {\n instance._tinode.logger(\"ERROR: Unexpected server response status\", this.status, this.response);\n }\n };\n\n xhr.onerror = function(e) {\n if (toReject) {\n toReject(e || new Error(\"failed\"));\n }\n if (onFailure) {\n onFailure(null);\n }\n };\n\n xhr.onabort = function(e) {\n if (toReject) {\n toReject(new Error(\"upload cancelled by user\"));\n }\n if (onFailure) {\n onFailure(null);\n }\n };\n\n try {\n const form = new FormData();\n form.append('file', data);\n form.set('id', this._tinode.getNextUniqueId());\n if (avatarFor) {\n form.set('topic', avatarFor);\n }\n xhr.send(form);\n } catch (err) {\n if (toReject) {\n toReject(err);\n }\n if (onFailure) {\n onFailure(null);\n }\n }\n\n return result;\n }\n /**\n * Start uploading the file to default endpoint.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {File|Blob} data to upload\n * @param {string} avatarFor topic name if the upload represents an avatar.\n * @param {Callback} onProgress callback. Takes one {float} parameter 0..1\n * @param {Callback} onSuccess callback. Called when the file is successfully uploaded.\n * @param {Callback} onFailure callback. Called in case of a failure.\n *\n * @returns {Promise} resolved/rejected when the upload is completed/failed.\n */\n upload(data, avatarFor, onProgress, onSuccess, onFailure) {\n const baseUrl = (this._tinode._secure ? 'https://' : 'http://') + this._tinode._host;\n return this.uploadWithBaseUrl(baseUrl, data, avatarFor, onProgress, onSuccess, onFailure);\n }\n /**\n * Download the file from a given URL using GET request. This method works with the Tinode server only.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {string} relativeUrl - URL to download the file from. Must be relative url, i.e. must not contain the host.\n * @param {string=} filename - file name to use for the downloaded file.\n *\n * @returns {Promise} resolved/rejected when the download is completed/failed.\n */\n download(relativeUrl, filename, mimetype, onProgress, onError) {\n if (!isUrlRelative(relativeUrl)) {\n // As a security measure refuse to download from an absolute URL.\n if (onError) {\n onError(`The URL '${relativeUrl}' must be relative, not absolute`);\n }\n return;\n }\n if (!this._authToken) {\n if (onError) {\n onError(\"Must authenticate first\");\n }\n return;\n }\n const instance = this;\n\n const xhr = new XHRProvider();\n this.xhr.push(xhr);\n\n // Add '&asatt=1' to URL to request 'Content-Disposition: attachment' response header.\n relativeUrl = addURLParam(relativeUrl, 'asatt', '1');\n\n // Get data as blob (stored by the browser as a temporary file).\n xhr.open('GET', relativeUrl, true);\n xhr.setRequestHeader('X-Tinode-APIKey', this._apiKey);\n xhr.setRequestHeader('X-Tinode-Auth', 'Token ' + this._authToken.token);\n xhr.responseType = 'blob';\n\n xhr.onprogress = function(e) {\n if (onProgress) {\n // Passing e.loaded instead of e.loaded/e.total because e.total\n // is always 0 with gzip compression enabled by the server.\n onProgress(e.loaded);\n }\n };\n\n let toResolve = null;\n let toReject = null;\n\n const result = new Promise((resolve, reject) => {\n toResolve = resolve;\n toReject = reject;\n });\n\n // The blob needs to be saved as file. There is no known way to\n // save the blob as file other than to fake a click on an .\n xhr.onload = function() {\n if (this.status == 200) {\n const link = document.createElement('a');\n // URL.createObjectURL is not available in non-browser environment. This call will fail.\n link.href = window.URL.createObjectURL(new Blob([this.response], {\n type: mimetype\n }));\n link.style.display = 'none';\n link.setAttribute('download', filename);\n document.body.appendChild(link);\n link.click();\n document.body.removeChild(link);\n window.URL.revokeObjectURL(link.href);\n if (toResolve) {\n toResolve();\n }\n } else if (this.status >= 400 && toReject) {\n // The this.responseText is undefined, must use this.response which is a blob.\n // Need to convert this.response to JSON. The blob can only be accessed by the\n // FileReader.\n const reader = new FileReader();\n reader.onload = function() {\n try {\n const pkt = JSON.parse(this.result, jsonParseHelper);\n toReject(new CommError(pkt.ctrl.text, pkt.ctrl.code));\n } catch (err) {\n instance._tinode.logger(\"ERROR: Invalid server response in LargeFileHelper\", this.result);\n toReject(err);\n }\n };\n reader.readAsText(this.response);\n }\n };\n\n xhr.onerror = function(e) {\n if (toReject) {\n toReject(new Error(\"failed\"));\n }\n if (onError) {\n onError(e);\n }\n };\n\n xhr.onabort = function() {\n if (toReject) {\n toReject(null);\n }\n };\n\n try {\n xhr.send();\n } catch (err) {\n if (toReject) {\n toReject(err);\n }\n if (onError) {\n onError(err);\n }\n }\n\n return result;\n }\n /**\n * Try to cancel all ongoing uploads or downloads.\n * @memberof Tinode.LargeFileHelper#\n */\n cancel() {\n this.xhr.forEach(req => {\n if (req.readyState < 4) {\n req.abort();\n }\n });\n }\n /**\n * To use LargeFileHelper in a non browser context, supply XMLHttpRequest provider.\n * @static\n * @memberof LargeFileHelper\n * @param xhrProvider XMLHttpRequest provider, e.g. for node require('xhr')
.\n */\n static setNetworkProvider(xhrProvider) {\n XHRProvider = xhrProvider;\n }\n}\n","/**\n * @module tinode-sdk\n *\n * @copyright 2015-2022 Tinode LLC.\n * @summary Javascript bindings for Tinode.\n * @license Apache 2.0\n * @version 0.20\n *\n * See https://github.com/tinode/webapp for real-life usage.\n *\n * @example\n * \n * \n * \n *\n * \n * ...\n * \n * \n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nimport AccessMode from './access-mode.js';\nimport * as Const from './config.js';\nimport CommError from './comm-error.js';\nimport Connection from './connection.js';\nimport DBCache from './db.js';\nimport Drafty from './drafty.js';\nimport LargeFileHelper from './large-file.js';\nimport MetaGetBuilder from './meta-builder.js';\nimport {\n Topic,\n TopicMe,\n TopicFnd\n} from './topic.js';\n\nimport {\n isUrlRelative,\n jsonParseHelper,\n mergeObj,\n rfc3339DateString,\n simplify\n} from './utils.js';\n\n// Re-export AccessMode\nexport {\n AccessMode\n};\n\nlet WebSocketProvider;\nif (typeof WebSocket != 'undefined') {\n WebSocketProvider = WebSocket;\n}\n\nlet XHRProvider;\nif (typeof XMLHttpRequest != 'undefined') {\n XHRProvider = XMLHttpRequest;\n}\n\nlet IndexedDBProvider;\nif (typeof indexedDB != 'undefined') {\n IndexedDBProvider = indexedDB;\n}\n\n// Re-export Drafty.\nexport {\n Drafty\n}\n\ninitForNonBrowserApp();\n\n// Utility functions\n\n// Polyfill for non-browser context, e.g. NodeJs.\nfunction initForNonBrowserApp() {\n // Tinode requirement in native mode because react native doesn't provide Base64 method\n const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';\n\n if (typeof btoa == 'undefined') {\n global.btoa = function(input = '') {\n let str = input;\n let output = '';\n\n for (let block = 0, charCode, i = 0, map = chars; str.charAt(i | 0) || (map = '=', i % 1); output += map.charAt(63 & block >> 8 - i % 1 * 8)) {\n\n charCode = str.charCodeAt(i += 3 / 4);\n\n if (charCode > 0xFF) {\n throw new Error(\"'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.\");\n }\n block = block << 8 | charCode;\n }\n\n return output;\n };\n }\n\n if (typeof atob == 'undefined') {\n global.atob = function(input = '') {\n let str = input.replace(/=+$/, '');\n let output = '';\n\n if (str.length % 4 == 1) {\n throw new Error(\"'atob' failed: The string to be decoded is not correctly encoded.\");\n }\n for (let bc = 0, bs = 0, buffer, i = 0; buffer = str.charAt(i++);\n\n ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,\n bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0\n ) {\n buffer = chars.indexOf(buffer);\n }\n\n return output;\n };\n }\n\n if (typeof window == 'undefined') {\n global.window = {\n WebSocket: WebSocketProvider,\n XMLHttpRequest: XHRProvider,\n indexedDB: IndexedDBProvider,\n URL: {\n createObjectURL: function() {\n throw new Error(\"Unable to use URL.createObjectURL in a non-browser application\");\n }\n }\n }\n }\n\n Connection.setNetworkProviders(WebSocketProvider, XHRProvider);\n LargeFileHelper.setNetworkProvider(XHRProvider);\n DBCache.setDatabaseProvider(IndexedDBProvider);\n}\n\n// Detect find most useful network transport.\nfunction detectTransport() {\n if (typeof window == 'object') {\n if (window['WebSocket']) {\n return 'ws';\n } else if (window['XMLHttpRequest']) {\n // The browser or node has no websockets, using long polling.\n return 'lp';\n }\n }\n return null;\n}\n\n// btoa replacement. Stock btoa fails on on non-Latin1 strings.\nfunction b64EncodeUnicode(str) {\n // The encodeURIComponent percent-encodes UTF-8 string,\n // then the percent encoding is converted into raw bytes which\n // can be fed into btoa.\n return btoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g,\n function toSolidBytes(match, p1) {\n return String.fromCharCode('0x' + p1);\n }));\n}\n\n// JSON stringify helper - pre-processor for JSON.stringify\nfunction jsonBuildHelper(key, val) {\n if (val instanceof Date) {\n // Convert javascript Date objects to rfc3339 strings\n val = rfc3339DateString(val);\n } else if (val instanceof AccessMode) {\n val = val.jsonHelper();\n } else if (val === undefined || val === null || val === false ||\n (Array.isArray(val) && val.length == 0) ||\n ((typeof val == 'object') && (Object.keys(val).length == 0))) {\n // strip out empty elements while serializing objects to JSON\n return undefined;\n }\n\n return val;\n};\n\n// Trims very long strings (encoded images) to make logged packets more readable.\nfunction jsonLoggerHelper(key, val) {\n if (typeof val == 'string' && val.length > 128) {\n return '<' + val.length + ', bytes: ' + val.substring(0, 12) + '...' + val.substring(val.length - 12) + '>';\n }\n return jsonBuildHelper(key, val);\n};\n\n// Parse browser user agent to extract browser name and version.\nfunction getBrowserInfo(ua, product) {\n ua = ua || '';\n let reactnative = '';\n // Check if this is a ReactNative app.\n if (/reactnative/i.test(product)) {\n reactnative = 'ReactNative; ';\n }\n let result;\n // Remove useless string.\n ua = ua.replace(' (KHTML, like Gecko)', '');\n // Test for WebKit-based browser.\n let m = ua.match(/(AppleWebKit\\/[.\\d]+)/i);\n if (m) {\n // List of common strings, from more useful to less useful.\n // All unknown strings get the highest (-1) priority.\n const priority = ['edg', 'chrome', 'safari', 'mobile', 'version'];\n let tmp = ua.substr(m.index + m[0].length).split(' ');\n let tokens = [];\n let version; // 1.0 in Version/1.0 or undefined;\n // Split string like 'Name/0.0.0' into ['Name', '0.0.0', 3] where the last element is the priority.\n for (let i = 0; i < tmp.length; i++) {\n let m2 = /([\\w.]+)[\\/]([\\.\\d]+)/.exec(tmp[i]);\n if (m2) {\n // Unknown values are highest priority (-1).\n tokens.push([m2[1], m2[2], priority.findIndex((e) => {\n return m2[1].toLowerCase().startsWith(e);\n })]);\n if (m2[1] == 'Version') {\n version = m2[2];\n }\n }\n }\n // Sort by priority: more interesting is earlier than less interesting.\n tokens.sort((a, b) => {\n return a[2] - b[2];\n });\n if (tokens.length > 0) {\n // Return the least common browser string and version.\n if (tokens[0][0].toLowerCase().startsWith('edg')) {\n tokens[0][0] = 'Edge';\n } else if (tokens[0][0] == 'OPR') {\n tokens[0][0] = 'Opera';\n } else if (tokens[0][0] == 'Safari' && version) {\n tokens[0][1] = version;\n }\n result = tokens[0][0] + '/' + tokens[0][1];\n } else {\n // Failed to ID the browser. Return the webkit version.\n result = m[1];\n }\n } else if (/firefox/i.test(ua)) {\n m = /Firefox\\/([.\\d]+)/g.exec(ua);\n if (m) {\n result = 'Firefox/' + m[1];\n } else {\n result = 'Firefox/?';\n }\n } else {\n // Neither AppleWebKit nor Firefox. Try the last resort.\n m = /([\\w.]+)\\/([.\\d]+)/.exec(ua);\n if (m) {\n result = m[1] + '/' + m[2];\n } else {\n m = ua.split(' ');\n result = m[0];\n }\n }\n\n // Shorten the version to one dot 'a.bb.ccc.d -> a.bb' at most.\n m = result.split('/');\n if (m.length > 1) {\n const v = m[1].split('.');\n const minor = v[1] ? '.' + v[1].substr(0, 2) : '';\n result = `${m[0]}/${v[0]}${minor}`;\n }\n return reactnative + result;\n}\n\n/**\n * The main class for interacting with Tinode server.\n */\nexport class Tinode {\n _host;\n _secure;\n\n _appName;\n\n // API Key.\n _apiKey;\n\n // Name and version of the browser.\n _browser = '';\n _platform;\n // Hardware\n _hwos = 'undefined';\n _humanLanguage = 'xx';\n\n // Logging to console enabled\n _loggingEnabled = false;\n // When logging, trip long strings (base64-encoded images) for readability\n _trimLongStrings = false;\n // UID of the currently authenticated user.\n _myUID = null;\n // Status of connection: authenticated or not.\n _authenticated = false;\n // Login used in the last successful basic authentication\n _login = null;\n // Token which can be used for login instead of login/password.\n _authToken = null;\n // Counter of received packets\n _inPacketCount = 0;\n // Counter for generating unique message IDs\n _messageId = Math.floor((Math.random() * 0xFFFF) + 0xFFFF);\n // Information about the server, if connected\n _serverInfo = null;\n // Push notification token. Called deviceToken for consistency with the Android SDK.\n _deviceToken = null;\n\n // Cache of pending promises by message id.\n _pendingPromises = {};\n // The Timeout object returned by the reject expired promises setInterval.\n _expirePromises = null;\n\n // Websocket or long polling connection.\n _connection = null;\n\n // Use indexDB for caching topics and messages.\n _persist = false;\n // IndexedDB wrapper object.\n _db = null;\n\n // Tinode's cache of objects\n _cache = {};\n\n /**\n * Create Tinode object.\n *\n * @param {Object} config - configuration parameters.\n * @param {string} config.appName - Name of the calling application to be reported in the User Agent.\n * @param {string} config.host - Host name and optional port number to connect to.\n * @param {string} config.apiKey - API key generated by keygen
.\n * @param {string} config.transport - See {@link Tinode.Connection#transport}.\n * @param {boolean} config.secure - Use Secure WebSocket if true
.\n * @param {string} config.platform - Optional platform identifier, one of \"ios\"
, \"web\"
, \"android\"
.\n * @param {boolen} config.persist - Use IndexedDB persistent storage.\n * @param {function} onComplete - callback to call when initialization is completed.\n */\n constructor(config, onComplete) {\n this._host = config.host;\n this._secure = config.secure;\n\n // Client-provided application name, format /\n this._appName = config.appName || \"Undefined\";\n\n // API Key.\n this._apiKey = config.apiKey;\n\n // Name and version of the browser.\n this._platform = config.platform || 'web';\n // Underlying OS.\n if (typeof navigator != 'undefined') {\n this._browser = getBrowserInfo(navigator.userAgent, navigator.product);\n this._hwos = navigator.platform;\n // This is the default language. It could be changed by client.\n this._humanLanguage = navigator.language || 'en-US';\n }\n\n Connection.logger = this.logger;\n Drafty.logger = this.logger;\n\n // WebSocket or long polling network connection.\n if (config.transport != 'lp' && config.transport != 'ws') {\n config.transport = detectTransport();\n }\n this._connection = new Connection(config, Const.PROTOCOL_VERSION, /* autoreconnect */ true);\n this._connection.onMessage = (data) => {\n // Call the main message dispatcher.\n this.#dispatchMessage(data);\n }\n\n // Ready to start sending.\n this._connection.onOpen = _ => this.#connectionOpen();\n this._connection.onDisconnect = (err, code) => this.#disconnected(err, code);\n\n // Wrapper for the reconnect iterator callback.\n this._connection.onAutoreconnectIteration = (timeout, promise) => {\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(timeout, promise);\n }\n }\n\n this._persist = config.persist;\n // Initialize object regardless. It simplifies the code.\n this._db = new DBCache(err => {\n this.logger('DB', err);\n }, this.logger);\n\n if (this._persist) {\n // Create the persistent cache.\n // Store promises to be resolved when messages load into memory.\n const prom = [];\n this._db.initDatabase().then(_ => {\n // First load topics into memory.\n return this._db.mapTopics((data) => {\n let topic = this.#cacheGet('topic', data.name);\n if (topic) {\n return;\n }\n if (data.name == Const.TOPIC_ME) {\n topic = new TopicMe();\n } else if (data.name == Const.TOPIC_FND) {\n topic = new TopicFnd();\n } else {\n topic = new Topic(data.name);\n }\n this._db.deserializeTopic(topic, data);\n this.#attachCacheToTopic(topic);\n topic._cachePutSelf();\n // Topic loaded from DB is not new.\n delete topic._new;\n // Request to load messages and save the promise.\n prom.push(topic._loadMessages(this._db));\n });\n }).then(_ => {\n // Then load users.\n return this._db.mapUsers((data) => {\n this.#cachePut('user', data.uid, mergeObj({}, data.public));\n });\n }).then(_ => {\n // Now wait for all messages to finish loading.\n return Promise.all(prom);\n }).then(_ => {\n if (onComplete) {\n onComplete();\n }\n this.logger(\"Persistent cache initialized.\");\n }).catch(err => {\n if (onComplete) {\n onComplete(err);\n }\n this.logger(\"Failed to initialize persistent cache:\", err);\n });\n } else {\n this._db.deleteDatabase().then(_ => {\n if (onComplete) {\n onComplete();\n }\n });\n }\n }\n\n // Private methods.\n\n // Console logger. Babel somehow fails to parse '...rest' parameter.\n logger(str, ...args) {\n if (this._loggingEnabled) {\n const d = new Date();\n const dateString = ('0' + d.getUTCHours()).slice(-2) + ':' +\n ('0' + d.getUTCMinutes()).slice(-2) + ':' +\n ('0' + d.getUTCSeconds()).slice(-2) + '.' +\n ('00' + d.getUTCMilliseconds()).slice(-3);\n\n console.log('[' + dateString + ']', str, args.join(' '));\n }\n }\n\n // Generator of default promises for sent packets.\n #makePromise(id) {\n let promise = null;\n if (id) {\n promise = new Promise((resolve, reject) => {\n // Stored callbacks will be called when the response packet with this Id arrives\n this._pendingPromises[id] = {\n 'resolve': resolve,\n 'reject': reject,\n 'ts': new Date()\n };\n });\n }\n return promise;\n };\n\n // Resolve or reject a pending promise.\n // Unresolved promises are stored in _pendingPromises.\n #execPromise(id, code, onOK, errorText) {\n const callbacks = this._pendingPromises[id];\n if (callbacks) {\n delete this._pendingPromises[id];\n if (code >= 200 && code < 400) {\n if (callbacks.resolve) {\n callbacks.resolve(onOK);\n }\n } else if (callbacks.reject) {\n callbacks.reject(new CommError(errorText, code));\n }\n }\n }\n\n // Send a packet. If packet id is provided return a promise.\n #send(pkt, id) {\n let promise;\n if (id) {\n promise = this.#makePromise(id);\n }\n pkt = simplify(pkt);\n let msg = JSON.stringify(pkt);\n this.logger(\"out: \" + (this._trimLongStrings ? JSON.stringify(pkt, jsonLoggerHelper) : msg));\n try {\n this._connection.sendText(msg);\n } catch (err) {\n // If sendText throws, wrap the error in a promise or rethrow.\n if (id) {\n this.#execPromise(id, Connection.NETWORK_ERROR, null, err.message);\n } else {\n throw err;\n }\n }\n return promise;\n }\n\n // The main message dispatcher.\n #dispatchMessage(data) {\n // Skip empty response. This happens when LP times out.\n if (!data)\n return;\n\n this._inPacketCount++;\n\n // Send raw message to listener\n if (this.onRawMessage) {\n this.onRawMessage(data);\n }\n\n if (data === '0') {\n // Server response to a network probe.\n if (this.onNetworkProbe) {\n this.onNetworkProbe();\n }\n // No processing is necessary.\n return;\n }\n\n let pkt = JSON.parse(data, jsonParseHelper);\n if (!pkt) {\n this.logger(\"in: \" + data);\n this.logger(\"ERROR: failed to parse data\");\n } else {\n this.logger(\"in: \" + (this._trimLongStrings ? JSON.stringify(pkt, jsonLoggerHelper) : data));\n\n // Send complete packet to listener\n if (this.onMessage) {\n this.onMessage(pkt);\n }\n\n if (pkt.ctrl) {\n // Handling {ctrl} message\n if (this.onCtrlMessage) {\n this.onCtrlMessage(pkt.ctrl);\n }\n\n // Resolve or reject a pending promise, if any\n if (pkt.ctrl.id) {\n this.#execPromise(pkt.ctrl.id, pkt.ctrl.code, pkt.ctrl, pkt.ctrl.text);\n }\n setTimeout(_ => {\n if (pkt.ctrl.code == 205 && pkt.ctrl.text == 'evicted') {\n // User evicted from topic.\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n topic._resetSub();\n if (pkt.ctrl.params && pkt.ctrl.params.unsub) {\n topic._gone();\n }\n }\n } else if (pkt.ctrl.code < 300 && pkt.ctrl.params) {\n if (pkt.ctrl.params.what == 'data') {\n // code=208, all messages received: \"params\":{\"count\":11,\"what\":\"data\"},\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n topic._allMessagesReceived(pkt.ctrl.params.count);\n }\n } else if (pkt.ctrl.params.what == 'sub') {\n // code=204, the topic has no (refreshed) subscriptions.\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n // Trigger topic.onSubsUpdated.\n topic._processMetaSubs([]);\n }\n }\n }\n }, 0);\n } else {\n setTimeout(_ => {\n if (pkt.meta) {\n // Handling a {meta} message.\n // Preferred API: Route meta to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.meta.topic);\n if (topic) {\n topic._routeMeta(pkt.meta);\n }\n\n if (pkt.meta.id) {\n this.#execPromise(pkt.meta.id, 200, pkt.meta, 'META');\n }\n\n // Secondary API: callback\n if (this.onMetaMessage) {\n this.onMetaMessage(pkt.meta);\n }\n } else if (pkt.data) {\n // Handling {data} message\n // Preferred API: Route data to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.data.topic);\n if (topic) {\n topic._routeData(pkt.data);\n }\n\n // Secondary API: Call callback\n if (this.onDataMessage) {\n this.onDataMessage(pkt.data);\n }\n } else if (pkt.pres) {\n // Handling {pres} message\n // Preferred API: Route presence to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.pres.topic);\n if (topic) {\n topic._routePres(pkt.pres);\n }\n\n // Secondary API - callback\n if (this.onPresMessage) {\n this.onPresMessage(pkt.pres);\n }\n } else if (pkt.info) {\n // {info} message - read/received notifications and key presses\n // Preferred API: Route {info}} to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.info.topic);\n if (topic) {\n topic._routeInfo(pkt.info);\n }\n\n // Secondary API - callback\n if (this.onInfoMessage) {\n this.onInfoMessage(pkt.info);\n }\n } else {\n this.logger(\"ERROR: Unknown packet received.\");\n }\n }, 0);\n }\n }\n }\n\n // Connection open, ready to start sending.\n #connectionOpen() {\n if (!this._expirePromises) {\n // Reject promises which have not been resolved for too long.\n this._expirePromises = setInterval(_ => {\n const err = new CommError(\"timeout\", 504);\n const expires = new Date(new Date().getTime() - Const.EXPIRE_PROMISES_TIMEOUT);\n for (let id in this._pendingPromises) {\n let callbacks = this._pendingPromises[id];\n if (callbacks && callbacks.ts < expires) {\n this.logger(\"Promise expired\", id);\n delete this._pendingPromises[id];\n if (callbacks.reject) {\n callbacks.reject(err);\n }\n }\n }\n }, Const.EXPIRE_PROMISES_PERIOD);\n }\n this.hello();\n }\n\n #disconnected(err, code) {\n this._inPacketCount = 0;\n this._serverInfo = null;\n this._authenticated = false;\n\n if (this._expirePromises) {\n clearInterval(this._expirePromises);\n this._expirePromises = null;\n }\n\n // Mark all topics as unsubscribed\n this.#cacheMap('topic', (topic, key) => {\n topic._resetSub();\n });\n\n // Reject all pending promises\n for (let key in this._pendingPromises) {\n const callbacks = this._pendingPromises[key];\n if (callbacks && callbacks.reject) {\n callbacks.reject(err);\n }\n }\n this._pendingPromises = {};\n\n if (this.onDisconnect) {\n this.onDisconnect(err);\n }\n }\n\n // Get User Agent string\n #getUserAgent() {\n return this._appName + ' (' + (this._browser ? this._browser + '; ' : '') + this._hwos + '); ' + Const.LIBRARY;\n }\n\n // Generator of packets stubs\n #initPacket(type, topic) {\n switch (type) {\n case 'hi':\n return {\n 'hi': {\n 'id': this.getNextUniqueId(),\n 'ver': Const.VERSION,\n 'ua': this.#getUserAgent(),\n 'dev': this._deviceToken,\n 'lang': this._humanLanguage,\n 'platf': this._platform\n }\n };\n\n case 'acc':\n return {\n 'acc': {\n 'id': this.getNextUniqueId(),\n 'user': null,\n 'scheme': null,\n 'secret': null,\n 'tmpscheme': null,\n 'tmpsecret': null,\n 'login': false,\n 'tags': null,\n 'desc': {},\n 'cred': {}\n }\n };\n\n case 'login':\n return {\n 'login': {\n 'id': this.getNextUniqueId(),\n 'scheme': null,\n 'secret': null\n }\n };\n\n case 'sub':\n return {\n 'sub': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'set': {},\n 'get': {}\n }\n };\n\n case 'leave':\n return {\n 'leave': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'unsub': false\n }\n };\n\n case 'pub':\n return {\n 'pub': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'noecho': false,\n 'head': null,\n 'content': {}\n }\n };\n\n case 'get':\n return {\n 'get': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'what': null,\n 'desc': {},\n 'sub': {},\n 'data': {}\n }\n };\n\n case 'set':\n return {\n 'set': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'desc': {},\n 'sub': {},\n 'tags': [],\n 'ephemeral': {}\n }\n };\n\n case 'del':\n return {\n 'del': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'what': null,\n 'delseq': null,\n 'user': null,\n 'hard': false\n }\n };\n\n case 'note':\n return {\n 'note': {\n // no id by design (except calls).\n 'topic': topic,\n 'what': null, // one of \"recv\", \"read\", \"kp\", \"call\"\n 'seq': undefined // the server-side message id acknowledged as received or read.\n }\n };\n\n default:\n throw new Error(`Unknown packet type requested: ${type}`);\n }\n }\n\n // Cache management\n #cachePut(type, name, obj) {\n this._cache[type + ':' + name] = obj;\n }\n #cacheGet(type, name) {\n return this._cache[type + ':' + name];\n }\n #cacheDel(type, name) {\n delete this._cache[type + ':' + name];\n }\n\n // Enumerate all items in cache, call func for each item.\n // Enumeration stops if func returns true.\n #cacheMap(type, func, context) {\n const key = type ? type + ':' : undefined;\n for (let idx in this._cache) {\n if (!key || idx.indexOf(key) == 0) {\n if (func.call(context, this._cache[idx], idx)) {\n break;\n }\n }\n }\n }\n\n // Make limited cache management available to topic.\n // Caching user.public only. Everything else is per-topic.\n #attachCacheToTopic(topic) {\n topic._tinode = this;\n\n topic._cacheGetUser = (uid) => {\n const pub = this.#cacheGet('user', uid);\n if (pub) {\n return {\n user: uid,\n public: mergeObj({}, pub)\n };\n }\n return undefined;\n };\n topic._cachePutUser = (uid, user) => {\n this.#cachePut('user', uid, mergeObj({}, user.public));\n };\n topic._cacheDelUser = (uid) => {\n this.#cacheDel('user', uid);\n };\n topic._cachePutSelf = _ => {\n this.#cachePut('topic', topic.name, topic);\n };\n topic._cacheDelSelf = _ => {\n this.#cacheDel('topic', topic.name);\n };\n }\n\n // On successful login save server-provided data.\n #loginSuccessful(ctrl) {\n if (!ctrl.params || !ctrl.params.user) {\n return ctrl;\n }\n // This is a response to a successful login,\n // extract UID and security token, save it in Tinode module\n this._myUID = ctrl.params.user;\n this._authenticated = (ctrl && ctrl.code >= 200 && ctrl.code < 300);\n if (ctrl.params && ctrl.params.token && ctrl.params.expires) {\n this._authToken = {\n token: ctrl.params.token,\n expires: ctrl.params.expires\n };\n } else {\n this._authToken = null;\n }\n\n if (this.onLogin) {\n this.onLogin(ctrl.code, ctrl.text);\n }\n\n return ctrl;\n }\n\n // Static methods.\n /**\n * Helper method to package account credential.\n *\n * @param {string | Credential} meth - validation method or object with validation data.\n * @param {string=} val - validation value (e.g. email or phone number).\n * @param {Object=} params - validation parameters.\n * @param {string=} resp - validation response.\n *\n * @returns {Array.} array with a single credential or null
if no valid credentials were given.\n */\n static credential(meth, val, params, resp) {\n if (typeof meth == 'object') {\n ({\n val,\n params,\n resp,\n meth\n } = meth);\n }\n if (meth && (val || resp)) {\n return [{\n 'meth': meth,\n 'val': val,\n 'resp': resp,\n 'params': params\n }];\n }\n return null;\n }\n\n /**\n * Determine topic type from topic's name: grp, p2p, me, fnd, sys.\n * @param {string} name - Name of the topic to test.\n * @returns {string} One of \"me\"
, \"fnd\"
, \"sys\"
, \"grp\"
,\n * \"p2p\"
or undefined
.\n */\n static topicType(name) {\n return Topic.topicType(name);\n }\n\n /**\n * Check if the given topic name is a name of a 'me' topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a 'me' topic, false
otherwise.\n */\n static isMeTopicName(name) {\n return Topic.isMeTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a group topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a group topic, false
otherwise.\n */\n static isGroupTopicName(name) {\n return Topic.isGroupTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a p2p topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p topic, false
otherwise.\n */\n static isP2PTopicName(name) {\n return Topic.isP2PTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a communication topic, i.e. P2P or group.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p or group topic, false
otherwise.\n */\n static isCommTopicName(name) {\n return Topic.isCommTopicName(name);\n }\n /**\n * Check if the topic name is a name of a new topic.\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a new topic, false
otherwise.\n */\n static isNewGroupTopicName(name) {\n return Topic.isNewGroupTopicName(name);\n }\n /**\n * Check if the topic name is a name of a channel.\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a channel, false
otherwise.\n */\n static isChannelTopicName(name) {\n return Topic.isChannelTopicName(name);\n }\n /**\n * Get information about the current version of this Tinode client library.\n * @returns {string} semantic version of the library, e.g. \"0.15.5-rc1\"
.\n */\n static getVersion() {\n return Const.VERSION;\n }\n /**\n * To use Tinode in a non browser context, supply WebSocket and XMLHttpRequest providers.\n * @static\n *\n * @param wsProvider WebSocket
provider, e.g. for nodeJS , require('ws')
.\n * @param xhrProvider XMLHttpRequest
provider, e.g. for node require('xhr')
.\n */\n static setNetworkProviders(wsProvider, xhrProvider) {\n WebSocketProvider = wsProvider;\n XHRProvider = xhrProvider;\n\n Connection.setNetworkProviders(WebSocketProvider, XHRProvider);\n LargeFileHelper.setNetworkProvider(XHRProvider);\n }\n /**\n * To use Tinode in a non browser context, supply indexedDB
provider.\n * @static\n *\n * @param idbProvider indexedDB
provider, e.g. for nodeJS , require('fake-indexeddb')
.\n */\n static setDatabaseProvider(idbProvider) {\n IndexedDBProvider = idbProvider;\n\n DBCache.setDatabaseProvider(IndexedDBProvider);\n }\n /**\n * Return information about the current name and version of this Tinode library.\n * @static\n *\n * @returns {string} the name of the library and it's version.\n */\n static getLibrary() {\n return Const.LIBRARY;\n }\n /**\n * Check if the given string represents NULL
value as defined by Tinode ('\\u2421'
).\n * @param {string} str - string to check for NULL
value.\n * @returns {boolean} true
if string represents NULL
value, false
otherwise.\n */\n static isNullValue(str) {\n return str === Const.DEL_CHAR;\n }\n\n // Instance methods.\n\n // Generates unique message IDs\n getNextUniqueId() {\n return (this._messageId != 0) ? '' + this._messageId++ : undefined;\n };\n\n /**\n * Connect to the server.\n *\n * @param {string} host_ - name of the host to connect to.\n * @return {Promise} Promise resolved/rejected when the connection call completes:\n * resolve()
is called without parameters, reject()
receives the\n * Error
as a single parameter.\n */\n connect(host_) {\n return this._connection.connect(host_);\n }\n\n /**\n * Attempt to reconnect to the server immediately.\n *\n * @param {string} force - if true
, reconnect even if there is a connection already.\n */\n reconnect(force) {\n this._connection.reconnect(force);\n }\n\n /**\n * Disconnect from the server.\n */\n disconnect() {\n this._connection.disconnect();\n }\n\n /**\n * Clear persistent cache: remove IndexedDB.\n *\n * @return {Promise} Promise resolved/rejected when the operation is completed.\n */\n clearStorage() {\n if (this._db.isReady()) {\n return this._db.deleteDatabase();\n }\n return Promise.resolve();\n }\n\n /**\n * Initialize persistent cache: create IndexedDB cache.\n *\n * @return {Promise} Promise resolved/rejected when the operation is completed.\n */\n initStorage() {\n if (!this._db.isReady()) {\n return this._db.initDatabase();\n }\n return Promise.resolve();\n }\n\n /**\n * Send a network probe message to make sure the connection is alive.\n */\n networkProbe() {\n this._connection.probe();\n }\n\n /**\n * Check for live connection to server.\n *\n * @returns {boolean} true
if there is a live connection, false
otherwise.\n */\n isConnected() {\n return this._connection.isConnected();\n }\n\n /**\n * Check if connection is authenticated (last login was successful).\n *\n * @returns {boolean} true
if authenticated, false
otherwise.\n */\n isAuthenticated() {\n return this._authenticated;\n }\n\n /**\n * Add API key and auth token to the relative URL making it usable for getting data\n * from the server in a simple HTTP GET
request.\n *\n * @param {string} URL - URL to wrap.\n * @returns {string} URL with appended API key and token, if valid token is present.\n */\n authorizeURL(url) {\n if (typeof url != 'string') {\n return url;\n }\n\n if (isUrlRelative(url)) {\n // Fake base to make the relative URL parseable.\n const base = 'scheme://host/';\n const parsed = new URL(url, base);\n if (this._apiKey) {\n parsed.searchParams.append('apikey', this._apiKey);\n }\n if (this._authToken && this._authToken.token) {\n parsed.searchParams.append('auth', 'token');\n parsed.searchParams.append('secret', this._authToken.token);\n }\n // Convert back to string and strip fake base URL except for the root slash.\n url = parsed.toString().substring(base.length - 1);\n }\n return url;\n }\n\n /**\n * @typedef AccountParams\n * @type {Object}\n * @property {DefAcs=} defacs - Default access parameters for user's me
topic.\n * @property {Object=} public - Public application-defined data exposed on me
topic.\n * @property {Object=} private - Private application-defined data accessible on me
topic.\n * @property {Object=} trusted - Trusted user data which can be set by a root user only.\n * @property {Array.} tags - array of string tags for user discovery.\n * @property {string} scheme - Temporary authentication scheme for password reset.\n * @property {string} secret - Temporary authentication secret for password reset.\n * @property {Array.=} attachments - Array of references to out of band attachments used in account description.\n */\n /**\n * @typedef DefAcs\n * @type {Object}\n * @property {string=} auth - Access mode for me
for authenticated users.\n * @property {string=} anon - Access mode for me
for anonymous users.\n */\n\n /**\n * Create or update an account.\n *\n * @param {string} uid - User id to update\n * @param {string} scheme - Authentication scheme; \"basic\"
and \"anonymous\"
are the currently supported schemes.\n * @param {string} secret - Authentication secret, assumed to be already base64 encoded.\n * @param {boolean=} login - Use new account to authenticate current session\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n account(uid, scheme, secret, login, params) {\n const pkt = this.#initPacket('acc');\n pkt.acc.user = uid;\n pkt.acc.scheme = scheme;\n pkt.acc.secret = secret;\n // Log in to the new account using selected scheme\n pkt.acc.login = login;\n\n if (params) {\n pkt.acc.desc.defacs = params.defacs;\n pkt.acc.desc.public = params.public;\n pkt.acc.desc.private = params.private;\n pkt.acc.desc.trusted = params.trusted;\n\n pkt.acc.tags = params.tags;\n pkt.acc.cred = params.cred;\n\n pkt.acc.tmpscheme = params.scheme;\n pkt.acc.tmpsecret = params.secret;\n\n if (Array.isArray(params.attachments) && params.attachments.length > 0) {\n pkt.extra = {\n attachments: params.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n }\n\n return this.#send(pkt, pkt.acc.id);\n }\n\n /**\n * Create a new user. Wrapper for {@link Tinode#account}.\n *\n * @param {string} scheme - Authentication scheme; \"basic\"
is the only currently supported scheme.\n * @param {string} secret - Authentication.\n * @param {boolean=} login - Use new account to authenticate current session\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n createAccount(scheme, secret, login, params) {\n let promise = this.account(Const.USER_NEW, scheme, secret, login, params);\n if (login) {\n promise = promise.then(ctrl => this.#loginSuccessful(ctrl));\n }\n return promise;\n }\n\n /**\n * Create user with 'basic'
authentication scheme and immediately\n * use it for authentication. Wrapper for {@link Tinode#account}.\n *\n * @param {string} username - Login to use for the new account.\n * @param {string} password - User's password.\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n createAccountBasic(username, password, params) {\n // Make sure we are not using 'null' or 'undefined';\n username = username || '';\n password = password || '';\n return this.createAccount('basic',\n b64EncodeUnicode(username + ':' + password), true, params);\n }\n\n /**\n * Update user's credentials for 'basic'
authentication scheme. Wrapper for {@link Tinode#account}.\n *\n * @param {string} uid - User ID to update.\n * @param {string} username - Login to use for the new account.\n * @param {string} password - User's password.\n * @param {AccountParams=} params - data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n updateAccountBasic(uid, username, password, params) {\n // Make sure we are not using 'null' or 'undefined';\n username = username || '';\n password = password || '';\n return this.account(uid, 'basic',\n b64EncodeUnicode(username + ':' + password), false, params);\n }\n\n /**\n * Send handshake to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n hello() {\n const pkt = this.#initPacket('hi');\n\n return this.#send(pkt, pkt.hi.id)\n .then(ctrl => {\n // Reset backoff counter on successful connection.\n this._connection.backoffReset();\n\n // Server response contains server protocol version, build, constraints,\n // session ID for long polling. Save them.\n if (ctrl.params) {\n this._serverInfo = ctrl.params;\n }\n\n if (this.onConnect) {\n this.onConnect();\n }\n\n return ctrl;\n }).catch(err => {\n this._connection.reconnect(true);\n\n if (this.onDisconnect) {\n this.onDisconnect(err);\n }\n });\n }\n\n /**\n * Set or refresh the push notifications/device token. If the client is connected,\n * the deviceToken can be sent to the server.\n *\n * @param {string} dt - token obtained from the provider or false
,\n * null
or undefined
to clear the token.\n *\n * @returns true
if attempt was made to send the update to the server.\n */\n setDeviceToken(dt) {\n let sent = false;\n // Convert any falsish value to null.\n dt = dt || null;\n if (dt != this._deviceToken) {\n this._deviceToken = dt;\n if (this.isConnected() && this.isAuthenticated()) {\n this.#send({\n 'hi': {\n 'dev': dt || Tinode.DEL_CHAR\n }\n });\n sent = true;\n }\n }\n return sent;\n }\n\n /**\n * @typedef Credential\n * @type {Object}\n * @property {string} meth - validation method.\n * @property {string} val - value to validate (e.g. email or phone number).\n * @property {string} resp - validation response.\n * @property {Object} params - validation parameters.\n */\n /**\n * Authenticate current session.\n *\n * @param {string} scheme - Authentication scheme; \"basic\"
is the only currently supported scheme.\n * @param {string} secret - Authentication secret, assumed to be already base64 encoded.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n login(scheme, secret, cred) {\n const pkt = this.#initPacket('login');\n pkt.login.scheme = scheme;\n pkt.login.secret = secret;\n pkt.login.cred = cred;\n\n return this.#send(pkt, pkt.login.id)\n .then(ctrl => this.#loginSuccessful(ctrl));\n }\n\n /**\n * Wrapper for {@link Tinode#login} with basic authentication\n *\n * @param {string} uname - User name.\n * @param {string} password - Password.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n loginBasic(uname, password, cred) {\n return this.login('basic', b64EncodeUnicode(uname + ':' + password), cred)\n .then(ctrl => {\n this._login = uname;\n return ctrl;\n });\n }\n\n /**\n * Wrapper for {@link Tinode#login} with token authentication\n *\n * @param {string} token - Token received in response to earlier login.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n loginToken(token, cred) {\n return this.login('token', token, cred);\n }\n\n /**\n * Send a request for resetting an authentication secret.\n *\n * @param {string} scheme - authentication scheme to reset.\n * @param {string} method - method to use for resetting the secret, such as \"email\" or \"tel\".\n * @param {string} value - value of the credential to use, a specific email address or a phone number.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving the server reply.\n */\n requestResetAuthSecret(scheme, method, value) {\n return this.login('reset', b64EncodeUnicode(scheme + ':' + method + ':' + value));\n }\n\n /**\n * @typedef AuthToken\n * @type {Object}\n * @property {string} token - Token value.\n * @property {Date} expires - Token expiration time.\n */\n /**\n * Get stored authentication token.\n *\n * @returns {AuthToken} authentication token.\n */\n getAuthToken() {\n if (this._authToken && (this._authToken.expires.getTime() > Date.now())) {\n return this._authToken;\n } else {\n this._authToken = null;\n }\n return null;\n }\n\n /**\n * Application may provide a saved authentication token.\n *\n * @param {AuthToken} token - authentication token.\n */\n setAuthToken(token) {\n this._authToken = token;\n }\n\n /**\n * @typedef SetParams\n * @type {Object}\n * @property {SetDesc=} desc - Topic initialization parameters when creating a new topic or a new subscription.\n * @property {SetSub=} sub - Subscription initialization parameters.\n * @property {Array.=} attachments - URLs of out of band attachments used in parameters.\n */\n /**\n * @typedef SetDesc\n * @type {Object}\n * @property {DefAcs=} defacs - Default access mode.\n * @property {Object=} public - Free-form topic description, publically accessible.\n * @property {Object=} private - Free-form topic description accessible only to the owner.\n * @property {Object=} trusted - Trusted user data which can be set by a root user only.\n */\n /**\n * @typedef SetSub\n * @type {Object}\n * @property {string=} user - UID of the user affected by the request. Default (empty) - current user.\n * @property {string=} mode - User access mode, either requested or assigned dependent on context.\n */\n /**\n * Send a topic subscription request.\n *\n * @param {string} topic - Name of the topic to subscribe to.\n * @param {GetQuery=} getParams - Optional subscription metadata query\n * @param {SetParams=} setParams - Optional initialization parameters\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n subscribe(topicName, getParams, setParams) {\n const pkt = this.#initPacket('sub', topicName)\n if (!topicName) {\n topicName = Const.TOPIC_NEW;\n }\n\n pkt.sub.get = getParams;\n\n if (setParams) {\n if (setParams.sub) {\n pkt.sub.set.sub = setParams.sub;\n }\n\n if (setParams.desc) {\n const desc = setParams.desc;\n if (Tinode.isNewGroupTopicName(topicName)) {\n // Full set.desc params are used for new topics only\n pkt.sub.set.desc = desc;\n } else if (Tinode.isP2PTopicName(topicName) && desc.defacs) {\n // Use optional default permissions only.\n pkt.sub.set.desc = {\n defacs: desc.defacs\n };\n }\n }\n\n // See if external objects were used in topic description.\n if (Array.isArray(setParams.attachments) && setParams.attachments.length > 0) {\n pkt.extra = {\n attachments: setParams.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n\n if (setParams.tags) {\n pkt.sub.set.tags = setParams.tags;\n }\n }\n return this.#send(pkt, pkt.sub.id);\n }\n\n /**\n * Detach and optionally unsubscribe from the topic\n *\n * @param {string} topic - Topic to detach from.\n * @param {boolean} unsub - If true
, detach and unsubscribe, otherwise just detach.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n leave(topic, unsub) {\n const pkt = this.#initPacket('leave', topic);\n pkt.leave.unsub = unsub;\n\n return this.#send(pkt, pkt.leave.id);\n }\n\n /**\n * Create message draft without sending it to the server.\n *\n * @param {string} topic - Name of the topic to publish to.\n * @param {Object} content - Payload to publish.\n * @param {boolean=} noEcho - If true
, tell the server not to echo the message to the original session.\n *\n * @returns {Object} new message which can be sent to the server or otherwise used.\n */\n createMessage(topic, content, noEcho) {\n const pkt = this.#initPacket('pub', topic);\n\n let dft = typeof content == 'string' ? Drafty.parse(content) : content;\n if (dft && !Drafty.isPlainText(dft)) {\n pkt.pub.head = {\n mime: Drafty.getContentType()\n };\n content = dft;\n }\n pkt.pub.noecho = noEcho;\n pkt.pub.content = content;\n\n return pkt.pub;\n }\n\n /**\n * Publish {data} message to topic.\n *\n * @param {string} topicName - Name of the topic to publish to.\n * @param {Object} content - Payload to publish.\n * @param {boolean=} noEcho - If true
, tell the server not to echo the message to the original session.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n publish(topicName, content, noEcho) {\n return this.publishMessage(\n this.createMessage(topicName, content, noEcho)\n );\n }\n\n /**\n * Publish message to topic. The message should be created by {@link Tinode#createMessage}.\n *\n * @param {Object} pub - Message to publish.\n * @param {Array.=} attachments - array of URLs with attachments.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n publishMessage(pub, attachments) {\n // Make a shallow copy. Needed in order to clear locally-assigned temp values;\n pub = Object.assign({}, pub);\n pub.seq = undefined;\n pub.from = undefined;\n pub.ts = undefined;\n const msg = {\n pub: pub,\n };\n if (attachments) {\n msg.extra = {\n attachments: attachments.filter(ref => isUrlRelative(ref))\n };\n }\n return this.#send(msg, pub.id);\n }\n\n /**\n * Out of band notification: notify topic that an external (push) notification was recived by the client.\n *\n * @param {object} data - notification payload.\n * @param {string} data.what - notification type, 'msg', 'read', 'sub'.\n * @param {string} data.topic - name of the updated topic.\n * @param {number=} data.seq - seq ID of the affected message.\n * @param {string=} data.xfrom - UID of the sender.\n * @param {object=} data.given - new subscription 'given', e.g. 'ASWP...'.\n * @param {object=} data.want - new subscription 'want', e.g. 'RWJ...'.\n */\n oobNotification(data) {\n this.logger('oob: ' + (this._trimLongStrings ? JSON.stringify(data, jsonLoggerHelper) : data));\n\n switch (data.what) {\n case 'msg':\n if (!data.seq || data.seq < 1 || !data.topic) {\n // Server sent invalid data.\n break;\n }\n\n if (!this.isConnected()) {\n // Let's ignore the message if there is no connection: no connection means there are no open\n // tabs with Tinode.\n break;\n }\n\n const topic = this.#cacheGet('topic', data.topic);\n if (!topic) {\n // TODO: check if there is a case when a message can arrive from an unknown topic.\n break;\n }\n\n if (topic.isSubscribed()) {\n // No need to fetch: topic is already subscribed and got data through normal channel.\n break;\n }\n\n if (topic.maxMsgSeq() < data.seq) {\n if (topic.isChannelType()) {\n topic._updateReceived(data.seq, 'fake-uid');\n }\n\n // New message.\n if (data.xfrom && !this.#cacheGet('user', data.xfrom)) {\n // Message from unknown sender, fetch description from the server.\n // Sending asynchronously without a subscription.\n this.getMeta(data.xfrom, new MetaGetBuilder().withDesc().build()).catch(err => {\n this.logger(\"Failed to get the name of a new sender\", err);\n });\n }\n\n topic.subscribe(null).then(_ => {\n return topic.getMeta(new MetaGetBuilder(topic).withLaterData(24).withLaterDel(24).build());\n }).then(_ => {\n // Allow data fetch to complete and get processed successfully.\n topic.leaveDelayed(false, 1000);\n }).catch(err => {\n this.logger(\"On push data fetch failed\", err);\n }).finally(_ => {\n this.getMeTopic()._refreshContact('msg', topic);\n });\n }\n break;\n\n case 'read':\n this.getMeTopic()._routePres({\n what: 'read',\n seq: data.seq\n });\n break;\n\n case 'sub':\n if (!this.isMe(data.xfrom)) {\n // TODO: handle updates from other users.\n break;\n }\n\n const mode = {\n given: data.modeGiven,\n want: data.modeWant\n };\n const acs = new AccessMode(mode);\n const pres = (!acs.mode || acs.mode == AccessMode._NONE) ?\n // Subscription deleted.\n {\n what: 'gone',\n src: data.topic\n } :\n // New subscription or subscription updated.\n {\n what: 'acs',\n src: data.topic,\n dacs: mode\n };\n this.getMeTopic()._routePres(pres);\n break;\n\n default:\n this.logger(\"Unknown push type ignored\", data.what);\n }\n }\n\n /**\n * @typedef GetQuery\n * @type {Object}\n * @property {GetOptsType=} desc - If provided (even if empty), fetch topic description.\n * @property {GetOptsType=} sub - If provided (even if empty), fetch topic subscriptions.\n * @property {GetDataType=} data - If provided (even if empty), get messages.\n */\n\n /**\n * @typedef GetOptsType\n * @type {Object}\n * @property {Date=} ims - \"If modified since\", fetch data only it was was modified since stated date.\n * @property {number=} limit - Maximum number of results to return. Ignored when querying topic description.\n */\n\n /**\n * @typedef GetDataType\n * @type {Object}\n * @property {number=} since - Load messages with seq id equal or greater than this value.\n * @property {number=} before - Load messages with seq id lower than this number.\n * @property {number=} limit - Maximum number of results to return.\n */\n\n /**\n * Request topic metadata\n *\n * @param {string} topic - Name of the topic to query.\n * @param {GetQuery} params - Parameters of the query. Use {@link Tinode.MetaGetBuilder} to generate.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n getMeta(topic, params) {\n const pkt = this.#initPacket('get', topic);\n\n pkt.get = mergeObj(pkt.get, params);\n\n return this.#send(pkt, pkt.get.id);\n }\n\n /**\n * Update topic's metadata: description, subscribtions.\n *\n * @param {string} topic - Topic to update.\n * @param {SetParams} params - topic metadata to update.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n setMeta(topic, params) {\n const pkt = this.#initPacket('set', topic);\n const what = [];\n\n if (params) {\n ['desc', 'sub', 'tags', 'cred', 'ephemeral'].forEach(function(key) {\n if (params.hasOwnProperty(key)) {\n what.push(key);\n pkt.set[key] = params[key];\n }\n });\n\n if (Array.isArray(params.attachments) && params.attachments.length > 0) {\n pkt.extra = {\n attachments: params.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n }\n\n if (what.length == 0) {\n return Promise.reject(new Error(\"Invalid {set} parameters\"));\n }\n\n return this.#send(pkt, pkt.set.id);\n }\n\n /**\n * Range of message IDs to delete.\n *\n * @typedef DelRange\n * @type {Object}\n * @property {number} low - low end of the range, inclusive (closed).\n * @property {number=} hi - high end of the range, exclusive (open).\n */\n /**\n * Delete some or all messages in a topic.\n *\n * @param {string} topic - Topic name to delete messages from.\n * @param {DelRange[]} list - Ranges of message IDs to delete.\n * @param {boolean=} hard - Hard or soft delete\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delMessages(topic, ranges, hard) {\n const pkt = this.#initPacket('del', topic);\n\n pkt.del.what = 'msg';\n pkt.del.delseq = ranges;\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete the topic alltogether. Requires Owner permission.\n *\n * @param {string} topicName - Name of the topic to delete\n * @param {boolean} hard - hard-delete topic.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delTopic(topicName, hard) {\n const pkt = this.#initPacket('del', topicName);\n pkt.del.what = 'topic';\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete subscription. Requires Share permission.\n *\n * @param {string} topicName - Name of the topic to delete\n * @param {string} user - User ID to remove.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delSubscription(topicName, user) {\n const pkt = this.#initPacket('del', topicName);\n pkt.del.what = 'sub';\n pkt.del.user = user;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete credential. Always sent on 'me'
topic.\n *\n * @param {string} method - validation method such as 'email'
or 'tel'
.\n * @param {string} value - validation value, i.e. 'alice@example.com'
.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCredential(method, value) {\n const pkt = this.#initPacket('del', Const.TOPIC_ME);\n pkt.del.what = 'cred';\n pkt.del.cred = {\n meth: method,\n val: value\n };\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Request to delete account of the current user.\n *\n * @param {boolean} hard - hard-delete user.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCurrentUser(hard) {\n const pkt = this.#initPacket('del', null);\n pkt.del.what = 'user';\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id).then(_ => {\n this._myUID = null;\n });\n }\n\n /**\n * Notify server that a message or messages were read or received. Does NOT return promise.\n *\n * @param {string} topicName - Name of the topic where the mesage is being aknowledged.\n * @param {string} what - Action being aknowledged, either \"read\"
or \"recv\"
.\n * @param {number} seq - Maximum id of the message being acknowledged.\n */\n note(topicName, what, seq) {\n if (seq <= 0 || seq >= Const.LOCAL_SEQID) {\n throw new Error(`Invalid message id ${seq}`);\n }\n\n const pkt = this.#initPacket('note', topicName);\n pkt.note.what = what;\n pkt.note.seq = seq;\n this.#send(pkt);\n }\n\n /**\n * Broadcast a key-press notification to topic subscribers. Used to show\n * typing notifications \"user X is typing...\".\n *\n * @param {string} topicName - Name of the topic to broadcast to.\n * @param {string=} type - notification to send, default is 'kp'.\n */\n noteKeyPress(topicName, type) {\n const pkt = this.#initPacket('note', topicName);\n pkt.note.what = type || 'kp';\n this.#send(pkt);\n }\n\n /**\n * Send a video call notification to topic subscribers (including dialing,\n * hangup, etc.).\n *\n * @param {string} topicName - Name of the topic to broadcast to.\n * @param {int} seq - ID of the call message the event pertains to.\n * @param {string} evt - Call event.\n * @param {string} payload - Payload associated with this event (e.g. SDP string).\n *\n * @returns {Promise} Promise (for some call events) which will\n * be resolved/rejected on receiving server reply\n */\n videoCall(topicName, seq, evt, payload) {\n const pkt = this.#initPacket('note', topicName);\n pkt.note.seq = seq;\n pkt.note.what = 'call';\n pkt.note.event = evt;\n pkt.note.payload = payload;\n this.#send(pkt, pkt.note.id);\n }\n\n /**\n * Get a named topic, either pull it from cache or create a new instance.\n * There is a single instance of topic for each name.\n *\n * @param {string} topicName - Name of the topic to get.\n *\n * @returns {Topic} Requested or newly created topic or undefined
if topic name is invalid.\n */\n getTopic(topicName) {\n let topic = this.#cacheGet('topic', topicName);\n if (!topic && topicName) {\n if (topicName == Const.TOPIC_ME) {\n topic = new TopicMe();\n } else if (topicName == Const.TOPIC_FND) {\n topic = new TopicFnd();\n } else {\n topic = new Topic(topicName);\n }\n // Cache management.\n this.#attachCacheToTopic(topic);\n topic._cachePutSelf();\n // Don't save to DB here: a record will be added when the topic is subscribed.\n }\n return topic;\n }\n\n /**\n * Get a named topic from cache.\n *\n * @param {string} topicName - Name of the topic to get.\n *\n * @returns {Topic} Requested topic or undefined
if topic is not found in cache.\n */\n cacheGetTopic(topicName) {\n return this.#cacheGet('topic', topicName);\n }\n\n /**\n * Remove named topic from cache.\n *\n * @param {string} topicName - Name of the topic to remove from cache.\n */\n cacheRemTopic(topicName) {\n this.#cacheDel('topic', topicName);\n }\n\n /**\n * Iterate over cached topics.\n *\n * @param {Function} func - callback to call for each topic.\n * @param {Object} context - 'this' inside the 'func'.\n */\n mapTopics(func, context) {\n this.#cacheMap('topic', func, context);\n }\n\n /**\n * Check if named topic is already present in cache.\n *\n * @param {string} topicName - Name of the topic to check.\n * @returns {boolean} true if topic is found in cache, false otherwise.\n */\n isTopicCached(topicName) {\n return !!this.#cacheGet('topic', topicName);\n }\n\n /**\n * Generate unique name like 'new123456'
suitable for creating a new group topic.\n *\n * @param {boolean} isChan - if the topic is channel-enabled.\n * @returns {string} name which can be used for creating a new group topic.\n */\n newGroupTopicName(isChan) {\n return (isChan ? Const.TOPIC_NEW_CHAN : Const.TOPIC_NEW) + this.getNextUniqueId();\n }\n\n /**\n * Instantiate 'me'
topic or get it from cache.\n *\n * @returns {TopicMe} Instance of 'me'
topic.\n */\n getMeTopic() {\n return this.getTopic(Const.TOPIC_ME);\n }\n\n /**\n * Instantiate 'fnd'
(find) topic or get it from cache.\n *\n * @returns {Topic} Instance of 'fnd'
topic.\n */\n getFndTopic() {\n return this.getTopic(Const.TOPIC_FND);\n }\n\n /**\n * Create a new {@link LargeFileHelper} instance\n *\n * @returns {LargeFileHelper} instance of a {@link Tinode.LargeFileHelper}.\n */\n getLargeFileHelper() {\n return new LargeFileHelper(this, Const.PROTOCOL_VERSION);\n }\n\n /**\n * Get the UID of the the current authenticated user.\n *\n * @returns {string} UID of the current user or undefined
if the session is not yet\n * authenticated or if there is no session.\n */\n getCurrentUserID() {\n return this._myUID;\n }\n\n /**\n * Check if the given user ID is equal to the current user's UID.\n *\n * @param {string} uid - UID to check.\n *\n * @returns {boolean} true if the given UID belongs to the current logged in user.\n */\n isMe(uid) {\n return this._myUID === uid;\n }\n\n /**\n * Get login used for last successful authentication.\n *\n * @returns {string} login last used successfully or undefined
.\n */\n getCurrentLogin() {\n return this._login;\n }\n\n /**\n * Return information about the server: protocol version and build timestamp.\n *\n * @returns {Object} build and version of the server or null
if there is no connection or\n * if the first server response has not been received yet.\n */\n getServerInfo() {\n return this._serverInfo;\n }\n\n /**\n * Report a topic for abuse. Wrapper for {@link Tinode#publish}.\n *\n * @param {string} action - the only supported action is 'report'.\n * @param {string} target - name of the topic being reported.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n report(action, target) {\n return this.publish(Const.TOPIC_SYS, Drafty.attachJSON(null, {\n 'action': action,\n 'target': target\n }));\n }\n\n /**\n * Return server-provided configuration value.\n *\n * @param {string} name of the value to return.\n * @param {Object} defaultValue to return in case the parameter is not set or not found.\n *\n * @returns {Object} named value.\n */\n getServerParam(name, defaultValue) {\n return this._serverInfo && this._serverInfo[name] || defaultValue;\n }\n\n /**\n * Toggle console logging. Logging is off by default.\n *\n * @param {boolean} enabled - Set to true
to enable logging to console.\n * @param {boolean} trimLongStrings - Set to true
to trim long strings.\n */\n enableLogging(enabled, trimLongStrings) {\n this._loggingEnabled = enabled;\n this._trimLongStrings = enabled && trimLongStrings;\n }\n\n /**\n * Set UI language to report to the server. Must be called before 'hi'
is sent, otherwise it will not be used.\n *\n * @param {string} hl - human (UI) language, like \"en_US\"
or \"zh-Hans\"
.\n */\n setHumanLanguage(hl) {\n if (hl) {\n this._humanLanguage = hl;\n }\n }\n\n /**\n * Check if given topic is online.\n *\n * @param {string} name of the topic to test.\n * @returns {boolean} true if topic is online, false otherwise.\n */\n isTopicOnline(name) {\n const topic = this.#cacheGet('topic', name);\n return topic && topic.online;\n }\n\n /**\n * Get access mode for the given contact.\n *\n * @param {string} name of the topic to query.\n * @returns {AccessMode} access mode if topic is found, null otherwise.\n */\n getTopicAccessMode(name) {\n const topic = this.#cacheGet('topic', name);\n return topic ? topic.acs : null;\n }\n\n /**\n * Include message ID into all subsequest messages to server instructin it to send aknowledgemens.\n * Required for promises to function. Default is \"on\"
.\n *\n * @param {boolean} status - Turn aknowledgemens on or off.\n * @deprecated\n */\n wantAkn(status) {\n if (status) {\n this._messageId = Math.floor((Math.random() * 0xFFFFFF) + 0xFFFFFF);\n } else {\n this._messageId = 0;\n }\n }\n\n // Callbacks:\n /**\n * Callback to report when the websocket is opened. The callback has no parameters.\n *\n * @type {onWebsocketOpen}\n */\n onWebsocketOpen = undefined;\n\n /**\n * @typedef ServerParams\n *\n * @type {Object}\n * @property {string} ver - Server version\n * @property {string} build - Server build\n * @property {string=} sid - Session ID, long polling connections only.\n */\n\n /**\n * @callback onConnect\n * @param {number} code - Result code\n * @param {string} text - Text epxplaining the completion, i.e \"OK\" or an error message.\n * @param {ServerParams} params - Parameters returned by the server.\n */\n /**\n * Callback to report when connection with Tinode server is established.\n * @type {onConnect}\n */\n onConnect = undefined;\n\n /**\n * Callback to report when connection is lost. The callback has no parameters.\n * @type {onDisconnect}\n */\n onDisconnect = undefined;\n\n /**\n * @callback onLogin\n * @param {number} code - NUmeric completion code, same as HTTP status codes.\n * @param {string} text - Explanation of the completion code.\n */\n /**\n * Callback to report login completion.\n * @type {onLogin}\n */\n onLogin = undefined;\n\n /**\n * Callback to receive {ctrl}
(control) messages.\n * @type {onCtrlMessage}\n */\n onCtrlMessage = undefined;\n\n /**\n * Callback to recieve {data}
(content) messages.\n * @type {onDataMessage}\n */\n onDataMessage = undefined;\n\n /**\n * Callback to receive {pres}
(presence) messages.\n * @type {onPresMessage}\n */\n onPresMessage = undefined;\n\n /**\n * Callback to receive all messages as objects.\n * @type {onMessage}\n */\n onMessage = undefined;\n\n /**\n * Callback to receive all messages as unparsed text.\n * @type {onRawMessage}\n */\n onRawMessage = undefined;\n\n /**\n * Callback to receive server responses to network probes. See {@link Tinode#networkProbe}\n * @type {onNetworkProbe}\n */\n onNetworkProbe = undefined;\n\n /**\n * Callback to be notified when exponential backoff is iterating.\n * @type {onAutoreconnectIteration}\n */\n onAutoreconnectIteration = undefined;\n};\n\n// Exported constants\nTinode.MESSAGE_STATUS_NONE = Const.MESSAGE_STATUS_NONE;\nTinode.MESSAGE_STATUS_QUEUED = Const.MESSAGE_STATUS_QUEUED;\nTinode.MESSAGE_STATUS_SENDING = Const.MESSAGE_STATUS_SENDING;\nTinode.MESSAGE_STATUS_FAILED = Const.MESSAGE_STATUS_FAILED;\nTinode.MESSAGE_STATUS_FATAL = Const.MESSAGE_STATUS_FATAL;\nTinode.MESSAGE_STATUS_SENT = Const.MESSAGE_STATUS_SENT;\nTinode.MESSAGE_STATUS_RECEIVED = Const.MESSAGE_STATUS_RECEIVED;\nTinode.MESSAGE_STATUS_READ = Const.MESSAGE_STATUS_READ;\nTinode.MESSAGE_STATUS_TO_ME = Const.MESSAGE_STATUS_TO_ME;\n\n// Unicode [del] symbol.\nTinode.DEL_CHAR = Const.DEL_CHAR;\n\n// Names of keys to server-provided configuration limits.\nTinode.MAX_MESSAGE_SIZE = 'maxMessageSize';\nTinode.MAX_SUBSCRIBER_COUNT = 'maxSubscriberCount';\nTinode.MAX_TAG_COUNT = 'maxTagCount';\nTinode.MAX_FILE_UPLOAD_SIZE = 'maxFileUploadSize';\nTinode.REQ_CRED_VALIDATORS = 'reqCred';\n\n// Tinode URI topic ID prefix, 'scheme:path/'.\nTinode.URI_TOPIC_ID_PREFIX = 'tinode:topic/';\n","/**\n * @file Helper class for constructing {@link Tinode.GetQuery}.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n/**\n * Helper class for constructing {@link Tinode.GetQuery}.\n *\n * @class MetaGetBuilder\n * @memberof Tinode\n *\n * @param {Tinode.Topic} parent topic which instantiated this builder.\n */\nexport default class MetaGetBuilder {\n constructor(parent) {\n this.topic = parent;\n this.what = {};\n }\n\n // Get timestamp of the most recent desc update.\n #get_desc_ims() {\n return this.topic._deleted ? undefined : this.topic.updated;\n }\n\n // Get timestamp of the most recent subs update.\n #get_subs_ims() {\n if (this.topic.isP2PType()) {\n return this.#get_desc_ims();\n }\n return this.topic._deleted ? undefined : this.topic._lastSubsUpdate;\n }\n /**\n * Add query parameters to fetch messages within explicit limits.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} since - messages newer than this (inclusive);\n * @param {number=} before - older than this (exclusive)\n * @param {number=} limit - number of messages to fetch\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withData(since, before, limit) {\n this.what['data'] = {\n since: since,\n before: before,\n limit: limit\n };\n return this;\n }\n /**\n * Add query parameters to fetch messages newer than the latest saved message.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - number of messages to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterData(limit) {\n return this.withData(this.topic._maxSeq > 0 ? this.topic._maxSeq + 1 : undefined, undefined, limit);\n }\n /**\n * Add query parameters to fetch messages older than the earliest saved message.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - maximum number of messages to fetch.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withEarlierData(limit) {\n return this.withData(undefined, this.topic._minSeq > 0 ? this.topic._minSeq : undefined, limit);\n }\n /**\n * Add query parameters to fetch topic description if it's newer than the given timestamp.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch messages newer than this timestamp.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withDesc(ims) {\n this.what['desc'] = {\n ims: ims\n };\n return this;\n }\n /**\n * Add query parameters to fetch topic description if it's newer than the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterDesc() {\n return this.withDesc(this.#get_desc_ims());\n }\n /**\n * Add query parameters to fetch subscriptions.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch subscriptions modified more recently than this timestamp\n * @param {number=} limit - maximum number of subscriptions to fetch.\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withSub(ims, limit, userOrTopic) {\n const opts = {\n ims: ims,\n limit: limit\n };\n if (this.topic.getType() == 'me') {\n opts.topic = userOrTopic;\n } else {\n opts.user = userOrTopic;\n }\n this.what['sub'] = opts;\n return this;\n }\n /**\n * Add query parameters to fetch a single subscription.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch subscriptions modified more recently than this timestamp\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withOneSub(ims, userOrTopic) {\n return this.withSub(ims, undefined, userOrTopic);\n }\n /**\n * Add query parameters to fetch a single subscription if it's been updated since the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterOneSub(userOrTopic) {\n return this.withOneSub(this.topic._lastSubsUpdate, userOrTopic);\n }\n /**\n * Add query parameters to fetch subscriptions updated since the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - maximum number of subscriptions to fetch.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterSub(limit) {\n return this.withSub(this.#get_subs_ims(), limit);\n }\n /**\n * Add query parameters to fetch topic tags.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withTags() {\n this.what['tags'] = true;\n return this;\n }\n /**\n * Add query parameters to fetch user's credentials. 'me'
topic only.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withCred() {\n if (this.topic.getType() == 'me') {\n this.what['cred'] = true;\n } else {\n this.topic._tinode.logger(\"ERROR: Invalid topic type for MetaGetBuilder:withCreds\", this.topic.getType());\n }\n return this;\n }\n /**\n * Add query parameters to fetch deleted messages within explicit limits. Any/all parameters can be null.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} since - ids of messages deleted since this 'del' id (inclusive)\n * @param {number=} limit - number of deleted message ids to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withDel(since, limit) {\n if (since || limit) {\n this.what['del'] = {\n since: since,\n limit: limit\n };\n }\n return this;\n }\n /**\n * Add query parameters to fetch messages deleted after the saved 'del'
id.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - number of deleted message ids to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterDel(limit) {\n // Specify 'since' only if we have already received some messages. If\n // we have no locally cached messages then we don't care if any messages were deleted.\n return this.withDel(this.topic._maxSeq > 0 ? this.topic._maxDel + 1 : undefined, limit);\n }\n\n /**\n * Extract subquery: get an object that contains specified subquery.\n * @memberof Tinode.MetaGetBuilder#\n * @param {string} what - subquery to return: one of 'data', 'sub', 'desc', 'tags', 'cred', 'del'.\n * @returns {Object} requested subquery or undefined
.\n */\n extract(what) {\n return this.what[what];\n }\n\n /**\n * Construct parameters.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.GetQuery} Get query\n */\n build() {\n const what = [];\n let params = {};\n ['data', 'sub', 'desc', 'tags', 'cred', 'del'].forEach((key) => {\n if (this.what.hasOwnProperty(key)) {\n what.push(key);\n if (Object.getOwnPropertyNames(this.what[key]).length > 0) {\n params[key] = this.what[key];\n }\n }\n });\n if (what.length > 0) {\n params.what = what.join(' ');\n } else {\n params = undefined;\n }\n return params;\n }\n}\n","/**\n * @file In-memory sorted cache of objects.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n/**\n * In-memory sorted cache of objects.\n *\n * @class CBuffer\n * @memberof Tinode\n * @protected\n *\n * @param {function} compare custom comparator of objects. Takes two parameters a
and b
;\n * returns -1
if a < b
, 0
if a == b
, 1
otherwise.\n * @param {boolean} unique enforce element uniqueness: when true
replace existing element with a new\n * one on conflict; when false
keep both elements.\n */\nexport default class CBuffer {\n #comparator = undefined;\n #unique = false;\n buffer = [];\n\n constructor(compare_, unique_) {\n this.#comparator = compare_ || ((a, b) => {\n return a === b ? 0 : a < b ? -1 : 1;\n });\n this.#unique = unique_;\n }\n\n #findNearest(elem, arr, exact) {\n let start = 0;\n let end = arr.length - 1;\n let pivot = 0;\n let diff = 0;\n let found = false;\n\n while (start <= end) {\n pivot = (start + end) / 2 | 0;\n diff = this.#comparator(arr[pivot], elem);\n if (diff < 0) {\n start = pivot + 1;\n } else if (diff > 0) {\n end = pivot - 1;\n } else {\n found = true;\n break;\n }\n }\n if (found) {\n return {\n idx: pivot,\n exact: true\n };\n }\n if (exact) {\n return {\n idx: -1\n };\n }\n // Not exact - insertion point\n return {\n idx: diff < 0 ? pivot + 1 : pivot\n };\n }\n\n // Insert element into a sorted array.\n #insertSorted(elem, arr) {\n const found = this.#findNearest(elem, arr, false);\n const count = (found.exact && this.#unique) ? 1 : 0;\n arr.splice(found.idx, count, elem);\n return arr;\n }\n\n /**\n * Get an element at the given position.\n * @memberof Tinode.CBuffer#\n * @param {number} at - Position to fetch from.\n * @returns {Object} Element at the given position or undefined
.\n */\n getAt(at) {\n return this.buffer[at];\n }\n\n /**\n * Convenience method for getting the element from the end of the buffer.\n * @memberof Tinode.CBuffer#\n * @param {number} at - position to fetch from, counting from the end;\n * undefined
or null
mean \"last\".\n * @returns {Object} The last element in the buffer or undefined
if buffer is empty.\n */\n getLast(at) {\n at |= 0;\n return this.buffer.length > at ? this.buffer[this.buffer.length - 1 - at] : undefined;\n }\n\n /**\n * Add new element(s) to the buffer. Variadic: takes one or more arguments. If an array is passed as a single\n * argument, its elements are inserted individually.\n * @memberof Tinode.CBuffer#\n *\n * @param {...Object|Array} - One or more objects to insert.\n */\n put() {\n let insert;\n // inspect arguments: if array, insert its elements, if one or more non-array arguments, insert them one by one\n if (arguments.length == 1 && Array.isArray(arguments[0])) {\n insert = arguments[0];\n } else {\n insert = arguments;\n }\n for (let idx in insert) {\n this.#insertSorted(insert[idx], this.buffer);\n }\n }\n\n /**\n * Remove element at the given position.\n * @memberof Tinode.CBuffer#\n * @param {number} at - Position to delete at.\n * @returns {Object} Element at the given position or undefined
.\n */\n delAt(at) {\n at |= 0;\n let r = this.buffer.splice(at, 1);\n if (r && r.length > 0) {\n return r[0];\n }\n return undefined;\n }\n\n /**\n * Remove elements between two positions.\n * @memberof Tinode.CBuffer#\n * @param {number} since - Position to delete from (inclusive).\n * @param {number} before - Position to delete to (exclusive).\n *\n * @returns {Array} array of removed elements (could be zero length).\n */\n delRange(since, before) {\n return this.buffer.splice(since, before - since);\n }\n\n /**\n * Return the number of elements the buffer holds.\n * @memberof Tinode.CBuffer#\n * @return {number} Number of elements in the buffer.\n */\n length() {\n return this.buffer.length;\n }\n\n /**\n * Reset the buffer discarding all elements\n * @memberof Tinode.CBuffer#\n */\n reset() {\n this.buffer = [];\n }\n\n /**\n * Callback for iterating contents of buffer. See {@link Tinode.CBuffer#forEach}.\n * @callback ForEachCallbackType\n * @memberof Tinode.CBuffer#\n * @param {Object} elem - Current element of the buffer.\n * @param {Object} prev - Previous element of the buffer.\n * @param {Object} next - Next element of the buffer.\n * @param {number} index - Index of the current element.\n */\n\n /**\n * Apply given callback
to all elements of the buffer.\n * @memberof Tinode.CBuffer#\n *\n * @param {Tinode.ForEachCallbackType} callback - Function to call for each element.\n * @param {number} startIdx - Optional index to start iterating from (inclusive).\n * @param {number} beforeIdx - Optional index to stop iterating before (exclusive).\n * @param {Object} context - calling context (i.e. value of this
in callback)\n */\n forEach(callback, startIdx, beforeIdx, context) {\n startIdx = startIdx | 0;\n beforeIdx = beforeIdx || this.buffer.length;\n\n for (let i = startIdx; i < beforeIdx; i++) {\n callback.call(context, this.buffer[i],\n (i > startIdx ? this.buffer[i - 1] : undefined),\n (i < beforeIdx - 1 ? this.buffer[i + 1] : undefined), i);\n }\n }\n\n /**\n * Find element in buffer using buffer's comparison function.\n * @memberof Tinode.CBuffer#\n *\n * @param {Object} elem - element to find.\n * @param {boolean=} nearest - when true and exact match is not found, return the nearest element (insertion point).\n * @returns {number} index of the element in the buffer or -1.\n */\n find(elem, nearest) {\n const {\n idx\n } = this.#findNearest(elem, this.buffer, !nearest);\n return idx;\n }\n\n /**\n * Callback for filtering the buffer. See {@link Tinode.CBuffer#filter}.\n * @callback FilterCallbackType\n * @memberof Tinode.CBuffer#\n * @param {Object} elem - Current element of the buffer.\n * @param {number} index - Index of the current element.\n * @returns {boolen} true
to keep the element, false
to remove.\n */\n\n /**\n * Remove all elements that do not pass the test implemented by the provided callback function.\n * @memberof Tinode.CBuffer#\n *\n * @param {Tinode.FilterCallbackType} callback - Function to call for each element.\n * @param {Object} context - calling context (i.e. value of this
in the callback)\n */\n filter(callback, context) {\n let count = 0;\n for (let i = 0; i < this.buffer.length; i++) {\n if (callback.call(context, this.buffer[i], i)) {\n this.buffer[count] = this.buffer[i];\n count++;\n }\n }\n\n this.buffer.splice(count);\n }\n\n /**\n * Check if buffer is empty.\n * @returns {boolean} true
if the buffer is empty, false
otherwise.\n */\n isEmpty() {\n return this.buffer.length == 0;\n }\n}\n","/**\n * @file Topic management.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport AccessMode from './access-mode.js';\nimport CBuffer from './cbuffer.js';\nimport CommError from './comm-error.js';\nimport * as Const from './config.js';\nimport Drafty from './drafty.js';\nimport MetaGetBuilder from './meta-builder.js';\nimport {\n mergeObj,\n mergeToCache,\n normalizeArray\n} from './utils.js';\n\n/**\n * Topic is a class representing a logical communication channel.\n */\nexport class Topic {\n /**\n * @callback onData\n * @param {Data} data - Data packet\n */\n\n /**\n * Create topic.\n * @param {string} name - Name of the topic to create.\n * @param {Object=} callbacks - Object with various event callbacks.\n * @param {onData} callbacks.onData - Callback which receives a {data}
message.\n * @param {callback} callbacks.onMeta - Callback which receives a {meta}
message.\n * @param {callback} callbacks.onPres - Callback which receives a {pres}
message.\n * @param {callback} callbacks.onInfo - Callback which receives an {info}
message.\n * @param {callback} callbacks.onMetaDesc - Callback which receives changes to topic desctioption {@link desc}.\n * @param {callback} callbacks.onMetaSub - Called for a single subscription record change.\n * @param {callback} callbacks.onSubsUpdated - Called after a batch of subscription changes have been recieved and cached.\n * @param {callback} callbacks.onDeleteTopic - Called after the topic is deleted.\n * @param {callback} callbacls.onAllMessagesReceived - Called when all requested {data}
messages have been recived.\n */\n constructor(name, callbacks) {\n // Parent Tinode object.\n this._tinode = null;\n\n // Server-provided data, locally immutable.\n // topic name\n this.name = name;\n // Timestamp when the topic was created.\n this.created = null;\n // Timestamp when the topic was last updated.\n this.updated = null;\n // Timestamp of the last messages\n this.touched = new Date(0);\n // Access mode, see AccessMode\n this.acs = new AccessMode(null);\n // Per-topic private data (accessible by current user only).\n this.private = null;\n // Per-topic public data (accessible by all users).\n this.public = null;\n // Per-topic system-provided data (accessible by all users).\n this.trusted = null;\n\n // Locally cached data\n // Subscribed users, for tracking read/recv/msg notifications.\n this._users = {};\n\n // Current value of locally issued seqId, used for pending messages.\n this._queuedSeqId = Const.LOCAL_SEQID;\n\n // The maximum known {data.seq} value.\n this._maxSeq = 0;\n // The minimum known {data.seq} value.\n this._minSeq = 0;\n // Indicator that the last request for earlier messages returned 0.\n this._noEarlierMsgs = false;\n // The maximum known deletion ID.\n this._maxDel = 0;\n // Timer object used to send 'recv' notifications.\n this._recvNotificationTimer = null;\n\n // User discovery tags\n this._tags = [];\n // Credentials such as email or phone number.\n this._credentials = [];\n // Message versions cache (e.g. for edited messages).\n // Keys: original message seq ID.\n // Values: CBuffers containing newer versions of the original message\n // ordered by seq id.\n this._messageVersions = {};\n // Message cache, sorted by message seq values, from old to new.\n this._messages = new CBuffer((a, b) => {\n return a.seq - b.seq;\n }, true);\n // Boolean, true if the topic is currently live\n this._attached = false;\n // Timestap of the most recently updated subscription.\n this._lastSubsUpdate = new Date(0);\n // Topic created but not yet synced with the server. Used only during initialization.\n this._new = true;\n // The topic is deleted at the server, this is a local copy.\n this._deleted = false;\n\n // Timer used to trgger {leave} request after a delay.\n this._delayedLeaveTimer = null;\n\n // Callbacks\n if (callbacks) {\n this.onData = callbacks.onData;\n this.onMeta = callbacks.onMeta;\n this.onPres = callbacks.onPres;\n this.onInfo = callbacks.onInfo;\n // A single desc update;\n this.onMetaDesc = callbacks.onMetaDesc;\n // A single subscription record;\n this.onMetaSub = callbacks.onMetaSub;\n // All subscription records received;\n this.onSubsUpdated = callbacks.onSubsUpdated;\n this.onTagsUpdated = callbacks.onTagsUpdated;\n this.onCredsUpdated = callbacks.onCredsUpdated;\n this.onDeleteTopic = callbacks.onDeleteTopic;\n this.onAllMessagesReceived = callbacks.onAllMessagesReceived;\n }\n }\n\n // Static methods.\n\n /**\n * Determine topic type from topic's name: grp, p2p, me, fnd, sys.\n *\n * @param {string} name - Name of the topic to test.\n * @returns {string} One of \"me\"
, \"fnd\"
, \"sys\"
, \"grp\"
,\n * \"p2p\"
or undefined
.\n */\n static topicType(name) {\n const types = {\n 'me': Const.TOPIC_ME,\n 'fnd': Const.TOPIC_FND,\n 'grp': Const.TOPIC_GRP,\n 'new': Const.TOPIC_GRP,\n 'nch': Const.TOPIC_GRP,\n 'chn': Const.TOPIC_GRP,\n 'usr': Const.TOPIC_P2P,\n 'sys': Const.TOPIC_SYS\n };\n return types[(typeof name == 'string') ? name.substring(0, 3) : 'xxx'];\n }\n\n /**\n * Check if the given topic name is a name of a 'me' topic.\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a 'me' topic, false
otherwise.\n */\n static isMeTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_ME;\n }\n\n /**\n * Check if the given topic name is a name of a group topic.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a group topic, false
otherwise.\n */\n static isGroupTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_GRP;\n }\n\n /**\n * Check if the given topic name is a name of a p2p topic.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p topic, false
otherwise.\n */\n static isP2PTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_P2P;\n }\n\n /**\n * Check if the given topic name is a name of a communication topic, i.e. P2P or group.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p or group topic, false
otherwise.\n */\n static isCommTopicName(name) {\n return Topic.isP2PTopicName(name) || Topic.isGroupTopicName(name);\n }\n\n /**\n * Check if the topic name is a name of a new topic.\n * @static\n *\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a new topic, false
otherwise.\n */\n static isNewGroupTopicName(name) {\n return (typeof name == 'string') &&\n (name.substring(0, 3) == Const.TOPIC_NEW || name.substring(0, 3) == Const.TOPIC_NEW_CHAN);\n }\n\n /**\n * Check if the topic name is a name of a channel.\n * @static\n *\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a channel, false
otherwise.\n */\n static isChannelTopicName(name) {\n return (typeof name == 'string') &&\n (name.substring(0, 3) == Const.TOPIC_CHAN || name.substring(0, 3) == Const.TOPIC_NEW_CHAN);\n }\n\n /**\n * Check if the topic is subscribed.\n * @returns {boolean} True is topic is attached/subscribed, false otherwise.\n */\n isSubscribed() {\n return this._attached;\n }\n\n /**\n * Request topic to subscribe. Wrapper for {@link Tinode#subscribe}.\n *\n * @param {Tinode.GetQuery=} getParams - get query parameters.\n * @param {Tinode.SetParams=} setParams - set parameters.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n subscribe(getParams, setParams) {\n // Clear request to leave topic.\n clearTimeout(this._delayedLeaveTimer);\n this._delayedLeaveTimer = null;\n\n // If the topic is already subscribed, return resolved promise\n if (this._attached) {\n return Promise.resolve(this);\n }\n\n // Send subscribe message, handle async response.\n // If topic name is explicitly provided, use it. If no name, then it's a new group topic,\n // use \"new\".\n return this._tinode.subscribe(this.name || Const.TOPIC_NEW, getParams, setParams).then(ctrl => {\n if (ctrl.code >= 300) {\n // Do nothing if subscription status has not changed.\n return ctrl;\n }\n\n this._attached = true;\n this._deleted = false;\n this.acs = (ctrl.params && ctrl.params.acs) ? ctrl.params.acs : this.acs;\n\n // Set topic name for new topics and add it to cache.\n if (this._new) {\n delete this._new;\n\n if (this.name != ctrl.topic) {\n // Name may change new123456 -> grpAbCdEf. Remove from cache under the old name.\n this._cacheDelSelf();\n this.name = ctrl.topic;\n }\n this._cachePutSelf();\n\n this.created = ctrl.ts;\n this.updated = ctrl.ts;\n\n if (this.name != Const.TOPIC_ME && this.name != Const.TOPIC_FND) {\n // Add the new topic to the list of contacts maintained by the 'me' topic.\n const me = this._tinode.getMeTopic();\n if (me.onMetaSub) {\n me.onMetaSub(this);\n }\n if (me.onSubsUpdated) {\n me.onSubsUpdated([this.name], 1);\n }\n }\n\n if (setParams && setParams.desc) {\n setParams.desc._noForwarding = true;\n this._processMetaDesc(setParams.desc);\n }\n }\n return ctrl;\n });\n }\n\n /**\n * Create a draft of a message without sending it to the server.\n * @memberof Tinode.Topic#\n *\n * @param {string | Object} data - Content to wrap in a draft.\n * @param {boolean=} noEcho - If true
server will not echo message back to originating\n * session. Otherwise the server will send a copy of the message to sender.\n *\n * @returns {Object} message draft.\n */\n createMessage(data, noEcho) {\n return this._tinode.createMessage(this.name, data, noEcho);\n }\n\n /**\n * Immediately publish data to topic. Wrapper for {@link Tinode#publish}.\n * @memberof Tinode.Topic#\n *\n * @param {string | Object} data - Message to publish, either plain string or a Drafty object.\n * @param {boolean=} noEcho - If true
server will not echo message back to originating\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n publish(data, noEcho) {\n return this.publishMessage(this.createMessage(data, noEcho));\n }\n\n /**\n * Publish message created by {@link Tinode.Topic#createMessage}.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub - {data} object to publish. Must be created by {@link Tinode.Topic#createMessage}\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n publishMessage(pub) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot publish on inactive topic\"));\n }\n if (this._sending) {\n return Promise.reject(new Error(\"The message is already being sent\"));\n }\n\n // Send data.\n pub._sending = true;\n pub._failed = false;\n\n // Extract refereces to attachments and out of band image records.\n let attachments = null;\n if (Drafty.hasEntities(pub.content)) {\n attachments = [];\n Drafty.entities(pub.content, data => {\n if (data) {\n if (data.ref) {\n attachments.push(data.ref);\n }\n if (data.preref) {\n attachments.push(data.preref);\n }\n }\n });\n if (attachments.length == 0) {\n attachments = null;\n }\n }\n\n return this._tinode.publishMessage(pub, attachments).then(ctrl => {\n pub._sending = false;\n pub.ts = ctrl.ts;\n this.swapMessageId(pub, ctrl.params.seq);\n this._maybeUpdateMessageVersionsCache(pub);\n this._routeData(pub);\n return ctrl;\n }).catch(err => {\n this._tinode.logger(\"WARNING: Message rejected by the server\", err);\n pub._sending = false;\n pub._failed = true;\n if (this.onData) {\n this.onData();\n }\n });\n }\n\n /**\n * Add message to local message cache, send to the server when the promise is resolved.\n * If promise is null or undefined, the message will be sent immediately.\n * The message is sent when the\n * The message should be created by {@link Tinode.Topic#createMessage}.\n * This is probably not the final API.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub - Message to use as a draft.\n * @param {Promise} prom - Message will be sent when this promise is resolved, discarded if rejected.\n *\n * @returns {Promise} derived promise.\n */\n publishDraft(pub, prom) {\n const seq = pub.seq || this._getQueuedSeqId();\n if (!pub._noForwarding) {\n // The 'seq', 'ts', and 'from' are added to mimic {data}. They are removed later\n // before the message is sent.\n pub._noForwarding = true;\n pub.seq = seq;\n pub.ts = new Date();\n pub.from = this._tinode.getCurrentUserID();\n\n // Don't need an echo message because the message is added to local cache right away.\n pub.noecho = true;\n // Add to cache.\n this._messages.put(pub);\n this._tinode._db.addMessage(pub);\n\n if (this.onData) {\n this.onData(pub);\n }\n }\n // If promise is provided, send the queued message when it's resolved.\n // If no promise is provided, create a resolved one and send immediately.\n return (prom || Promise.resolve())\n .then(_ => {\n if (pub._cancelled) {\n return {\n code: 300,\n text: \"cancelled\"\n };\n }\n return this.publishMessage(pub);\n }).catch(err => {\n this._tinode.logger(\"WARNING: Message draft rejected\", err);\n pub._sending = false;\n pub._failed = true;\n pub._fatal = err instanceof CommError ? (err.code >= 400 && err.code < 500) : false;\n if (this.onData) {\n this.onData();\n }\n // Rethrow to let caller know that the operation failed.\n throw err;\n });\n }\n\n /**\n * Leave the topic, optionally unsibscribe. Leaving the topic means the topic will stop\n * receiving updates from the server. Unsubscribing will terminate user's relationship with the topic.\n * Wrapper for {@link Tinode#leave}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean=} unsub - If true, unsubscribe, otherwise just leave.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n leave(unsub) {\n // It's possible to unsubscribe (unsub==true) from inactive topic.\n if (!this._attached && !unsub) {\n return Promise.reject(new Error(\"Cannot leave inactive topic\"));\n }\n\n // Send a 'leave' message, handle async response\n return this._tinode.leave(this.name, unsub).then(ctrl => {\n this._resetSub();\n if (unsub) {\n this._gone();\n }\n return ctrl;\n });\n }\n\n /**\n * Leave the topic, optionally unsibscribe after a delay. Leaving the topic means the topic will stop\n * receiving updates from the server. Unsubscribing will terminate user's relationship with the topic.\n * Wrapper for {@link Tinode#leave}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} unsub - If true, unsubscribe, otherwise just leave.\n * @param {number} delay - time in milliseconds to delay leave request.\n */\n leaveDelayed(unsub, delay) {\n clearTimeout(this._delayedLeaveTimer);\n this._delayedLeaveTimer = setTimeout(_ => {\n this._delayedLeaveTimer = null;\n this.leave(unsub)\n }, delay);\n }\n\n /**\n * Request topic metadata from the server.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.GetQuery} request parameters\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n getMeta(params) {\n // Send {get} message, return promise.\n return this._tinode.getMeta(this.name, params);\n }\n\n /**\n * Request more messages from the server\n * @memberof Tinode.Topic#\n *\n * @param {number} limit number of messages to get.\n * @param {boolean} forward if true, request newer messages.\n */\n getMessagesPage(limit, forward) {\n let query = forward ?\n this.startMetaQuery().withLaterData(limit) :\n this.startMetaQuery().withEarlierData(limit);\n\n // First try fetching from DB, then from the server.\n return this._loadMessages(this._tinode._db, query.extract('data'))\n .then((count) => {\n if (count == limit) {\n // Got enough messages from local cache.\n return Promise.resolve({\n topic: this.name,\n code: 200,\n params: {\n count: count\n }\n });\n }\n\n // Reduce the count of requested messages.\n limit -= count;\n // Update query with new values loaded from DB.\n query = forward ? this.startMetaQuery().withLaterData(limit) :\n this.startMetaQuery().withEarlierData(limit);\n let promise = this.getMeta(query.build());\n if (!forward) {\n promise = promise.then(ctrl => {\n if (ctrl && ctrl.params && !ctrl.params.count) {\n this._noEarlierMsgs = true;\n }\n });\n }\n return promise;\n });\n }\n /**\n * Update topic metadata.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.SetParams} params parameters to update.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n setMeta(params) {\n if (params.tags) {\n params.tags = normalizeArray(params.tags);\n }\n // Send Set message, handle async response.\n return this._tinode.setMeta(this.name, params)\n .then(ctrl => {\n if (ctrl && ctrl.code >= 300) {\n // Not modified\n return ctrl;\n }\n\n if (params.sub) {\n params.sub.topic = this.name;\n if (ctrl.params && ctrl.params.acs) {\n params.sub.acs = ctrl.params.acs;\n params.sub.updated = ctrl.ts;\n }\n if (!params.sub.user) {\n // This is a subscription update of the current user.\n // Assign user ID otherwise the update will be ignored by _processMetaSubs.\n params.sub.user = this._tinode.getCurrentUserID();\n if (!params.desc) {\n // Force update to topic's asc.\n params.desc = {};\n }\n }\n params.sub._noForwarding = true;\n this._processMetaSubs([params.sub]);\n }\n\n if (params.desc) {\n if (ctrl.params && ctrl.params.acs) {\n params.desc.acs = ctrl.params.acs;\n params.desc.updated = ctrl.ts;\n }\n this._processMetaDesc(params.desc);\n }\n\n if (params.tags) {\n this._processMetaTags(params.tags);\n }\n if (params.cred) {\n this._processMetaCreds([params.cred], true);\n }\n\n return ctrl;\n });\n }\n /**\n * Update access mode of the current user or of another topic subsriber.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - UID of the user to update or null to update current user.\n * @param {string} update - the update value, full or delta.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n updateMode(uid, update) {\n const user = uid ? this.subscriber(uid) : null;\n const am = user ?\n user.acs.updateGiven(update).getGiven() :\n this.getAccessMode().updateWant(update).getWant();\n\n return this.setMeta({\n sub: {\n user: uid,\n mode: am\n }\n });\n }\n /**\n * Create new topic subscription. Wrapper for {@link Tinode#setMeta}.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - ID of the user to invite\n * @param {string=} mode - Access mode. null
means to use default.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n invite(uid, mode) {\n return this.setMeta({\n sub: {\n user: uid,\n mode: mode\n }\n });\n }\n /**\n * Archive or un-archive the topic. Wrapper for {@link Tinode#setMeta}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} arch - true to archive the topic, false otherwise.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n archive(arch) {\n if (this.private && (!this.private.arch == !arch)) {\n return Promise.resolve(arch);\n }\n return this.setMeta({\n desc: {\n private: {\n arch: arch ? true : Const.DEL_CHAR\n }\n }\n });\n }\n /**\n * Delete messages. Hard-deleting messages requires Owner permission.\n * Wrapper for {@link Tinode#delMessages}.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.DelRange[]} ranges - Ranges of message IDs to delete.\n * @param {boolean=} hard - Hard or soft delete\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessages(ranges, hard) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete messages in inactive topic\"));\n }\n\n // Sort ranges in accending order by low, the descending by hi.\n ranges.sort((r1, r2) => {\n if (r1.low < r2.low) {\n return true;\n }\n if (r1.low == r2.low) {\n return !r2.hi || (r1.hi >= r2.hi);\n }\n return false;\n });\n\n // Remove pending messages from ranges possibly clipping some ranges.\n let tosend = ranges.reduce((out, r) => {\n if (r.low < Const.LOCAL_SEQID) {\n if (!r.hi || r.hi < Const.LOCAL_SEQID) {\n out.push(r);\n } else {\n // Clip hi to max allowed value.\n out.push({\n low: r.low,\n hi: this._maxSeq + 1\n });\n }\n }\n return out;\n }, []);\n\n // Send {del} message, return promise\n let result;\n if (tosend.length > 0) {\n result = this._tinode.delMessages(this.name, tosend, hard);\n } else {\n result = Promise.resolve({\n params: {\n del: 0\n }\n });\n }\n // Update local cache.\n return result.then(ctrl => {\n if (ctrl.params.del > this._maxDel) {\n this._maxDel = ctrl.params.del;\n }\n\n ranges.forEach((r) => {\n if (r.hi) {\n this.flushMessageRange(r.low, r.hi);\n } else {\n this.flushMessage(r.low);\n }\n });\n\n if (this.onData) {\n // Calling with no parameters to indicate the messages were deleted.\n this.onData();\n }\n return ctrl;\n });\n }\n /**\n * Delete all messages. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessagesAll(hardDel) {\n if (!this._maxSeq || this._maxSeq <= 0) {\n // There are no messages to delete.\n return Promise.resolve();\n }\n return this.delMessages([{\n low: 1,\n hi: this._maxSeq + 1,\n _all: true\n }], hardDel);\n }\n\n /**\n * Delete multiple messages defined by their IDs. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {Array.} list - list of seq IDs to delete.\n * @param {boolean=} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessagesList(list, hardDel) {\n // Sort the list in ascending order\n list.sort((a, b) => a - b);\n // Convert the array of IDs to ranges.\n let ranges = list.reduce((out, id) => {\n if (out.length == 0) {\n // First element.\n out.push({\n low: id\n });\n } else {\n let prev = out[out.length - 1];\n if ((!prev.hi && (id != prev.low + 1)) || (id > prev.hi)) {\n // New range.\n out.push({\n low: id\n });\n } else {\n // Expand existing range.\n prev.hi = prev.hi ? Math.max(prev.hi, id + 1) : id + 1;\n }\n }\n return out;\n }, []);\n // Send {del} message, return promise\n return this.delMessages(ranges, hardDel);\n }\n\n /**\n * Delete original message and edited variants. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - original seq ID of the message to delete.\n * @param {boolean=} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n delMessagesEdits(seq, hardDel) {\n const list = [seq];\n this.messageVersions(seq, msg => list.push(msg.seq));\n // Send {del} message, return promise\n return this.delMessagesList(list, hardDel);\n }\n\n /**\n * Delete topic. Requires Owner permission. Wrapper for {@link Tinode#delTopic}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} hard - had-delete topic.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n delTopic(hard) {\n if (this._deleted) {\n // The topic is already deleted at the server, just remove from DB.\n this._gone();\n return Promise.resolve(null);\n }\n\n return this._tinode.delTopic(this.name, hard).then(ctrl => {\n this._deleted = true;\n this._resetSub();\n this._gone();\n return ctrl;\n });\n }\n /**\n * Delete subscription. Requires Share permission. Wrapper for {@link Tinode#delSubscription}.\n * @memberof Tinode.Topic#\n *\n * @param {string} user - ID of the user to remove subscription for.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delSubscription(user) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete subscription in inactive topic\"));\n }\n // Send {del} message, return promise\n return this._tinode.delSubscription(this.name, user).then(ctrl => {\n // Remove the object from the subscription cache;\n delete this._users[user];\n // Notify listeners\n if (this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._users));\n }\n return ctrl;\n });\n }\n /**\n * Send a read/recv notification.\n * @memberof Tinode.Topic#\n *\n * @param {string} what - what notification to send: recv
, read
.\n * @param {number} seq - ID or the message read or received.\n */\n note(what, seq) {\n if (!this._attached) {\n // Cannot sending {note} on an inactive topic\".\n return;\n }\n\n // Update local cache with the new count.\n const user = this._users[this._tinode.getCurrentUserID()];\n let update = false;\n if (user) {\n // Self-subscription is found.\n if (!user[what] || user[what] < seq) {\n user[what] = seq;\n update = true;\n }\n } else {\n // Self-subscription is not found.\n update = (this[what] | 0) < seq;\n }\n\n if (update) {\n // Send notification to the server.\n this._tinode.note(this.name, what, seq);\n // Update locally cached contact with the new count.\n this._updateMyReadRecv(what, seq);\n\n if (this.acs != null && !this.acs.isMuted()) {\n const me = this._tinode.getMeTopic();\n // Sent a notification to 'me' listeners.\n me._refreshContact(what, this);\n }\n }\n }\n\n /**\n * Send a 'recv' receipt. Wrapper for {@link Tinode#noteRecv}.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - ID of the message to aknowledge.\n */\n noteRecv(seq) {\n this.note('recv', seq);\n }\n /**\n * Send a 'read' receipt. Wrapper for {@link Tinode#noteRead}.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - ID of the message to aknowledge or 0/undefined to acknowledge the latest messages.\n */\n noteRead(seq) {\n seq = seq || this._maxSeq;\n if (seq > 0) {\n this.note('read', seq);\n }\n }\n /**\n * Send a key-press notification. Wrapper for {@link Tinode#noteKeyPress}.\n * @memberof Tinode.Topic#\n */\n noteKeyPress() {\n if (this._attached) {\n this._tinode.noteKeyPress(this.name);\n } else {\n this._tinode.logger(\"INFO: Cannot send notification in inactive topic\");\n }\n }\n /**\n * Send a notification than a video or audio message is . Wrapper for {@link Tinode#noteKeyPress}.\n * @memberof Tinode.Topic#\n * @param audioOnly - true if the recording is audio-only, false if it's a video recording.\n */\n noteRecording(audioOnly) {\n if (this._attached) {\n this._tinode.noteKeyPress(this.name, audioOnly ? 'kpa' : 'kpv');\n } else {\n this._tinode.logger(\"INFO: Cannot send notification in inactive topic\");\n }\n }\n\n /**\n * Send a {note what='call'}. Wrapper for {@link Tinode#videoCall}.\n * @memberof Tinode#\n *\n * @param {string} evt - Call event.\n * @param {int} seq - ID of the call message the event pertains to.\n * @param {string} payload - Payload associated with this event (e.g. SDP string).\n *\n * @returns {Promise} Promise (for some call events) which will\n * be resolved/rejected on receiving server reply\n */\n videoCall(evt, seq, payload) {\n if (!this._attached && !['ringing', 'hang-up'].includes(evt)) {\n // Cannot {call} on an inactive topic\".\n return;\n }\n return this._tinode.videoCall(this.name, seq, evt, payload);\n }\n\n // Update cached read/recv/unread counts for the current user.\n _updateMyReadRecv(what, seq, ts) {\n let oldVal, doUpdate = false;\n\n seq = seq | 0;\n this.seq = this.seq | 0;\n this.read = this.read | 0;\n this.recv = this.recv | 0;\n switch (what) {\n case 'recv':\n oldVal = this.recv;\n this.recv = Math.max(this.recv, seq);\n doUpdate = (oldVal != this.recv);\n break;\n case 'read':\n oldVal = this.read;\n this.read = Math.max(this.read, seq);\n doUpdate = (oldVal != this.read);\n break;\n case 'msg':\n oldVal = this.seq;\n this.seq = Math.max(this.seq, seq);\n if (!this.touched || this.touched < ts) {\n this.touched = ts;\n }\n doUpdate = (oldVal != this.seq);\n break;\n }\n\n // Sanity checks.\n if (this.recv < this.read) {\n this.recv = this.read;\n doUpdate = true;\n }\n if (this.seq < this.recv) {\n this.seq = this.recv;\n if (!this.touched || this.touched < ts) {\n this.touched = ts;\n }\n doUpdate = true;\n }\n this.unread = this.seq - this.read;\n return doUpdate;\n }\n /**\n * Get user description from global cache. The user does not need to be a\n * subscriber of this topic.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - ID of the user to fetch.\n * @return {Object} user description or undefined.\n */\n userDesc(uid) {\n // TODO: handle asynchronous requests\n const user = this._cacheGetUser(uid);\n if (user) {\n return user; // Promise.resolve(user)\n }\n }\n /**\n * Get description of the p2p peer from subscription cache.\n * @memberof Tinode.Topic#\n *\n * @return {Object} peer's description or undefined.\n */\n p2pPeerDesc() {\n if (!this.isP2PType()) {\n return undefined;\n }\n return this._users[this.name];\n }\n /**\n * Iterate over cached subscribers. If callback is undefined, use this.onMetaSub.\n * @memberof Tinode.Topic#\n *\n * @param {function} callback - Callback which will receive subscribers one by one.\n * @param {Object=} context - Value of `this` inside the `callback`.\n */\n subscribers(callback, context) {\n const cb = (callback || this.onMetaSub);\n if (cb) {\n for (let idx in this._users) {\n cb.call(context, this._users[idx], idx, this._users);\n }\n }\n }\n /**\n * Get a copy of cached tags.\n * @memberof Tinode.Topic#\n *\n * @return {Array.} a copy of tags\n */\n tags() {\n // Return a copy.\n return this._tags.slice(0);\n }\n /**\n * Get cached subscription for the given user ID.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - id of the user to query for\n * @return user description or undefined.\n */\n subscriber(uid) {\n return this._users[uid];\n }\n /**\n * Iterate over versions of a message: call callback
for each version (excluding original).\n * If callback
is undefined, does nothing.\n * @memberof Tinode.Topic#\n *\n * @param {number} origSeq - seq ID of the original message.\n * @param {Tinode.ForEachCallbackType} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {Object} context - Value of `this` inside the `callback`.\n */\n messageVersions(origSeq, callback, context) {\n if (!callback) {\n // No callback? We are done then.\n return;\n }\n const versions = this._messageVersions[origSeq];\n if (!versions) {\n return;\n }\n versions.forEach(callback, undefined, undefined, context);\n }\n /**\n * Iterate over cached messages: call callback
for each message in the range [sinceIdx, beforeIdx).\n * If callback
is undefined, use this.onData
.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.ForEachCallbackType} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {number} sinceId - Optional seqId to start iterating from (inclusive).\n * @param {number} beforeId - Optional seqId to stop iterating before it is reached (exclusive).\n * @param {Object} context - Value of `this` inside the `callback`.\n */\n messages(callback, sinceId, beforeId, context) {\n const cb = (callback || this.onData);\n if (cb) {\n const startIdx = typeof sinceId == 'number' ? this._messages.find({\n seq: sinceId\n }, true) : undefined;\n const beforeIdx = typeof beforeId == 'number' ? this._messages.find({\n seq: beforeId\n }, true) : undefined;\n if (startIdx != -1 && beforeIdx != -1) {\n // Step 1. Filter out all replacement messages and\n // save displayable messages in a temporary buffer.\n let msgs = [];\n this._messages.forEach((msg, unused1, unused2, i) => {\n if (this._isReplacementMsg(msg)) {\n // Skip replacements.\n return;\n }\n // In case the massage was edited, replace timestamp of the version with the original's timestamp.\n const latest = this.latestMsgVersion(msg.seq) || msg;\n if (!latest._origTs) {\n latest._origTs = latest.ts;\n latest._origSeq = latest.seq;\n latest.ts = msg.ts;\n latest.seq = msg.seq;\n }\n msgs.push({\n data: latest,\n idx: i\n });\n }, startIdx, beforeIdx, {});\n // Step 2. Loop over displayble messages invoking cb on each of them.\n msgs.forEach((val, i) => {\n cb.call(context, val.data,\n (i > 0 ? msgs[i - 1].data : undefined),\n (i < msgs.length - 1 ? msgs[i + 1].data : undefined), val.idx);\n });\n }\n }\n }\n /**\n * Get the message from cache by seq
.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - message seqId to search for.\n * @returns {Object} the message with the given seq
or undefined
, if no such message is found.\n */\n findMessage(seq) {\n const idx = this._messages.find({\n seq: seq\n });\n if (idx >= 0) {\n return this._messages.getAt(idx);\n }\n return undefined;\n }\n /**\n * Get the most recent message from cache. This method counts all messages, including deleted ranges.\n * @memberof Tinode.Topic#\n *\n * @returns {Object} the most recent cached message or undefined
, if no messages are cached.\n */\n latestMessage() {\n return this._messages.getLast();\n }\n /**\n * Get the latest version for message.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - original seq ID of the message.\n * @returns {Object} the latest version of the message or null if message not found.\n */\n latestMsgVersion(seq) {\n const versions = this._messageVersions[seq];\n return versions ? versions.getLast() : null;\n }\n /**\n * Get the maximum cached seq ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the greatest seq ID in cache.\n */\n maxMsgSeq() {\n return this._maxSeq;\n }\n /**\n * Get the minimum cached seq ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the smallest seq ID in cache or 0.\n */\n minMsgSeq() {\n return this._minSeq;\n }\n /**\n * Get the maximum deletion ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the greatest deletion ID.\n */\n maxClearId() {\n return this._maxDel;\n }\n /**\n * Get the number of messages in the cache.\n * @memberof Tinode.Topic#\n *\n * @returns {number} count of cached messages.\n */\n messageCount() {\n return this._messages.length();\n }\n /**\n * Iterate over cached unsent messages. Wraps {@link Tinode.Topic#messages}.\n * @memberof Tinode.Topic#\n *\n * @param {function} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {Object} context - Value of this
inside the callback
.\n */\n queuedMessages(callback, context) {\n if (!callback) {\n throw new Error(\"Callback must be provided\");\n }\n this.messages(callback, Const.LOCAL_SEQID, undefined, context);\n }\n /**\n * Get the number of topic subscribers who marked this message as either recv or read\n * Current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {string} what - what action to consider: received \"recv\"
or read \"read\"
.\n * @param {number} seq - ID or the message read or received.\n *\n * @returns {number} the number of subscribers who marked the message with the given ID as read or received.\n */\n msgReceiptCount(what, seq) {\n let count = 0;\n if (seq > 0) {\n const me = this._tinode.getCurrentUserID();\n for (let idx in this._users) {\n const user = this._users[idx];\n if (user.user !== me && user[what] >= seq) {\n count++;\n }\n }\n }\n return count;\n }\n /**\n * Get the number of topic subscribers who marked this message (and all older messages) as read.\n * The current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - message id to check.\n * @returns {number} number of subscribers who claim to have received the message.\n */\n msgReadCount(seq) {\n return this.msgReceiptCount('read', seq);\n }\n /**\n * Get the number of topic subscribers who marked this message (and all older messages) as received.\n * The current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - Message id to check.\n * @returns {number} Number of subscribers who claim to have received the message.\n */\n msgRecvCount(seq) {\n return this.msgReceiptCount('recv', seq);\n }\n /**\n * Check if cached message IDs indicate that the server may have more messages.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} newer - if true
, check for newer messages only.\n */\n msgHasMoreMessages(newer) {\n return newer ? this.seq > this._maxSeq :\n // _minSeq could be more than 1, but earlier messages could have been deleted.\n (this._minSeq > 1 && !this._noEarlierMsgs);\n }\n /**\n * Check if the given seq Id is id of the most recent message.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to check\n */\n isNewMessage(seqId) {\n return this._maxSeq <= seqId;\n }\n /**\n * Remove one message from local cache.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to remove from cache.\n * @returns {Message} removed message or undefined if such message was not found.\n */\n flushMessage(seqId) {\n const idx = this._messages.find({\n seq: seqId\n });\n delete this._messageVersions[seqId];\n if (idx >= 0) {\n this._tinode._db.remMessages(this.name, seqId);\n return this._messages.delAt(idx);\n }\n return undefined;\n }\n /**\n * Remove a range of messages from the local cache.\n * @memberof Tinode.Topic#\n *\n * @param {number} fromId seq ID of the first message to remove (inclusive).\n * @param {number} untilId seqID of the last message to remove (exclusive).\n *\n * @returns {Message[]} array of removed messages (could be empty).\n */\n flushMessageRange(fromId, untilId) {\n // Remove range from persistent cache.\n this._tinode._db.remMessages(this.name, fromId, untilId);\n\n // Remove all versions keyed by IDs in the range.\n for (let i = fromId; i < untilId; i++) {\n delete this._messageVersions[i];\n }\n\n // start, end: find insertion points (nearest == true).\n const since = this._messages.find({\n seq: fromId\n }, true);\n return since >= 0 ? this._messages.delRange(since, this._messages.find({\n seq: untilId\n }, true)) : [];\n }\n /**\n * Update message's seqId.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub message object.\n * @param {number} newSeqId new seq id for pub.\n */\n swapMessageId(pub, newSeqId) {\n const idx = this._messages.find(pub);\n const numMessages = this._messages.length();\n if (0 <= idx && idx < numMessages) {\n // Remove message with the old seq ID.\n this._messages.delAt(idx);\n this._tinode._db.remMessages(this.name, pub.seq);\n // Add message with the new seq ID.\n pub.seq = newSeqId;\n this._messages.put(pub);\n this._tinode._db.addMessage(pub);\n }\n }\n /**\n * Attempt to stop message from being sent.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to stop sending and remove from cache.\n *\n * @returns {boolean} true
if message was cancelled, false
otherwise.\n */\n cancelSend(seqId) {\n const idx = this._messages.find({\n seq: seqId\n });\n if (idx >= 0) {\n const msg = this._messages.getAt(idx);\n const status = this.msgStatus(msg);\n if (status == Const.MESSAGE_STATUS_QUEUED ||\n status == Const.MESSAGE_STATUS_FAILED ||\n status == Const.MESSAGE_STATUS_FATAL) {\n this._tinode._db.remMessages(this.name, seqId);\n msg._cancelled = true;\n this._messages.delAt(idx);\n if (this.onData) {\n // Calling with no parameters to indicate the message was deleted.\n this.onData();\n }\n return true;\n }\n }\n return false;\n }\n /**\n * Get type of the topic: me, p2p, grp, fnd...\n * @memberof Tinode.Topic#\n *\n * @returns {string} One of 'me', 'p2p', 'grp', 'fnd', 'sys' or undefined
.\n */\n getType() {\n return Topic.topicType(this.name);\n }\n /**\n * Get current user's access mode of the topic.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.AccessMode} - user's access mode\n */\n getAccessMode() {\n return this.acs;\n }\n /**\n * Set current user's access mode of the topic.\n * @memberof Tinode.Topic#\n *\n * @param {AccessMode | Object} acs - access mode to set.\n */\n setAccessMode(acs) {\n return this.acs = new AccessMode(acs);\n }\n /**\n * Get topic's default access mode.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.DefAcs} - access mode, such as {auth: `RWP`, anon: `N`}.\n */\n getDefaultAccess() {\n return this.defacs;\n }\n /**\n * Initialize new meta {@link Tinode.GetQuery} builder. The query is attched to the current topic.\n * It will not work correctly if used with a different topic.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.MetaGetBuilder} query attached to the current topic.\n */\n startMetaQuery() {\n return new MetaGetBuilder(this);\n }\n /**\n * Check if topic is archived, i.e. private.arch == true.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is archived, false
otherwise.\n */\n isArchived() {\n return this.private && !!this.private.arch;\n }\n /**\n * Check if topic is a 'me' topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a 'me' topic, false
otherwise.\n */\n isMeType() {\n return Topic.isMeTopicName(this.name);\n }\n /**\n * Check if topic is a channel.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a channel, false
otherwise.\n */\n isChannelType() {\n return Topic.isChannelTopicName(this.name);\n }\n /**\n * Check if topic is a group topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a group, false
otherwise.\n */\n isGroupType() {\n return Topic.isGroupTopicName(this.name);\n }\n /**\n * Check if topic is a p2p topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a p2p topic, false
otherwise.\n */\n isP2PType() {\n return Topic.isP2PTopicName(this.name);\n }\n /**\n * Check if topic is a communication topic, i.e. a group or p2p topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a p2p or group topic, false
otherwise.\n */\n isCommType() {\n return Topic.isCommTopicName(this.name);\n }\n /**\n * Get status (queued, sent, received etc) of a given message in the context\n * of this topic.\n * @memberof Tinode.Topic#\n *\n * @param {Message} msg - message to check for status.\n * @param {boolean} upd - update chached message status.\n *\n * @returns message status constant.\n */\n msgStatus(msg, upd) {\n let status = Const.MESSAGE_STATUS_NONE;\n if (this._tinode.isMe(msg.from)) {\n if (msg._sending) {\n status = Const.MESSAGE_STATUS_SENDING;\n } else if (msg._fatal || msg._cancelled) {\n status = Const.MESSAGE_STATUS_FATAL;\n } else if (msg._failed) {\n status = Const.MESSAGE_STATUS_FAILED;\n } else if (msg.seq >= Const.LOCAL_SEQID) {\n status = Const.MESSAGE_STATUS_QUEUED;\n } else if (this.msgReadCount(msg.seq) > 0) {\n status = Const.MESSAGE_STATUS_READ;\n } else if (this.msgRecvCount(msg.seq) > 0) {\n status = Const.MESSAGE_STATUS_RECEIVED;\n } else if (msg.seq > 0) {\n status = Const.MESSAGE_STATUS_SENT;\n }\n } else {\n status = Const.MESSAGE_STATUS_TO_ME;\n }\n\n if (upd && msg._status != status) {\n msg._status = status;\n this._tinode._db.updMessageStatus(this.name, msg.seq, status);\n }\n\n return status;\n }\n\n // Returns true if pub is meant to replace another message (e.g. original message was edited).\n _isReplacementMsg(pub) {\n return pub.head && pub.head.replace;\n }\n\n // If msg is a replacement for another message, save the msg in the message versions cache\n // as a newer version for the message it's supposed to replace.\n _maybeUpdateMessageVersionsCache(msg) {\n if (!this._isReplacementMsg(msg)) {\n // Check if this message is the original in the chain of edits and if so\n // ensure all version have the same sender.\n if (this._messageVersions[msg.seq]) {\n // Remove versions with different 'from'.\n this._messageVersions[msg.seq].filter(version => version.from == msg.from);\n if (this._messageVersions[msg.seq].isEmpty()) {\n delete this._messageVersions[msg.seq];\n }\n }\n return;\n }\n\n const targetSeq = parseInt(msg.head.replace.split(':')[1]);\n if (targetSeq > msg.seq) {\n // Substitutes are supposed to have higher seq ids.\n return;\n }\n const targetMsg = this.findMessage(targetSeq);\n if (targetMsg && targetMsg.from != msg.from) {\n // Substitute cannot change the sender.\n return;\n }\n const versions = this._messageVersions[targetSeq] || new CBuffer((a, b) => {\n return a.seq - b.seq;\n }, true);\n versions.put(msg);\n this._messageVersions[targetSeq] = versions;\n }\n\n // Process data message\n _routeData(data) {\n if (data.content) {\n if (!this.touched || this.touched < data.ts) {\n this.touched = data.ts;\n this._tinode._db.updTopic(this);\n }\n }\n\n if (data.seq > this._maxSeq) {\n this._maxSeq = data.seq;\n this.msgStatus(data, true);\n // Ackn receiving the message.\n clearTimeout(this._recvNotificationTimer);\n this._recvNotificationTimer = setTimeout(_ => {\n this._recvNotificationTimer = null;\n this.noteRecv(this._maxSeq);\n }, Const.RECV_TIMEOUT);\n }\n\n if (data.seq < this._minSeq || this._minSeq == 0) {\n this._minSeq = data.seq;\n }\n\n const outgoing = ((!this.isChannelType() && !data.from) || this._tinode.isMe(data.from));\n\n if (data.head && data.head.webrtc && data.head.mime == Drafty.getContentType() && data.content) {\n // Rewrite VC body with info from the headers.\n data.content = Drafty.updateVideoCall(data.content, {\n state: data.head.webrtc,\n duration: data.head['webrtc-duration'],\n incoming: !outgoing,\n });\n }\n\n if (!data._noForwarding) {\n this._messages.put(data);\n this._tinode._db.addMessage(data);\n this._maybeUpdateMessageVersionsCache(data);\n }\n\n if (this.onData) {\n this.onData(data);\n }\n\n // Update locally cached contact with the new message count.\n const what = outgoing ? 'read' : 'msg';\n this._updateMyReadRecv(what, data.seq, data.ts);\n\n if (!outgoing && data.from) {\n // Mark messages as read by the sender.\n this._routeInfo({\n what: 'read',\n from: data.from,\n seq: data.seq,\n _noForwarding: true\n });\n }\n\n // Notify 'me' listeners of the change.\n this._tinode.getMeTopic()._refreshContact(what, this);\n }\n\n // Process metadata message\n _routeMeta(meta) {\n if (meta.desc) {\n this._processMetaDesc(meta.desc);\n }\n if (meta.sub && meta.sub.length > 0) {\n this._processMetaSubs(meta.sub);\n }\n if (meta.del) {\n this._processDelMessages(meta.del.clear, meta.del.delseq);\n }\n if (meta.tags) {\n this._processMetaTags(meta.tags);\n }\n if (meta.cred) {\n this._processMetaCreds(meta.cred);\n }\n if (this.onMeta) {\n this.onMeta(meta);\n }\n }\n // Process presence change message\n _routePres(pres) {\n let user, uid;\n switch (pres.what) {\n case 'del':\n // Delete cached messages.\n this._processDelMessages(pres.clear, pres.delseq);\n break;\n case 'on':\n case 'off':\n // Update online status of a subscription.\n user = this._users[pres.src];\n if (user) {\n user.online = pres.what == 'on';\n } else {\n this._tinode.logger(\"WARNING: Presence update for an unknown user\", this.name, pres.src);\n }\n break;\n case 'term':\n // Attachment to topic is terminated probably due to cluster rehashing.\n this._resetSub();\n break;\n case 'upd':\n // A topic subscriber has updated his description.\n // Issue {get sub} only if the current user has no p2p topics with the updated user (p2p name is not in cache).\n // Otherwise 'me' will issue a {get desc} request.\n if (pres.src && !this._tinode.isTopicCached(pres.src)) {\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n }\n break;\n case 'acs':\n uid = pres.src || this._tinode.getCurrentUserID();\n user = this._users[uid];\n if (!user) {\n // Update for an unknown user: notification of a new subscription.\n const acs = new AccessMode().updateAll(pres.dacs);\n if (acs && acs.mode != AccessMode._NONE) {\n user = this._cacheGetUser(uid);\n if (!user) {\n user = {\n user: uid,\n acs: acs\n };\n this.getMeta(this.startMetaQuery().withOneSub(undefined, uid).build());\n } else {\n user.acs = acs;\n }\n user.updated = new Date();\n this._processMetaSubs([user]);\n }\n } else {\n // Known user\n user.acs.updateAll(pres.dacs);\n // Update user's access mode.\n this._processMetaSubs([{\n user: uid,\n updated: new Date(),\n acs: user.acs\n }]);\n }\n break;\n default:\n this._tinode.logger(\"INFO: Ignored presence update\", pres.what);\n }\n\n if (this.onPres) {\n this.onPres(pres);\n }\n }\n // Process {info} message\n _routeInfo(info) {\n switch (info.what) {\n case 'recv':\n case 'read':\n const user = this._users[info.from];\n if (user) {\n user[info.what] = info.seq;\n if (user.recv < user.read) {\n user.recv = user.read;\n }\n }\n const msg = this.latestMessage();\n if (msg) {\n this.msgStatus(msg, true);\n }\n\n // If this is an update from the current user, update the cache with the new count.\n if (this._tinode.isMe(info.from) && !info._noForwarding) {\n this._updateMyReadRecv(info.what, info.seq);\n }\n\n // Notify 'me' listener of the status change.\n this._tinode.getMeTopic()._refreshContact(info.what, this);\n break;\n case 'kp':\n case 'kpa':\n case 'kpv':\n // Typing or audio/video recording notification. Do nothing.\n break;\n case 'call':\n // Do nothing here.\n break;\n default:\n this._tinode.logger(\"INFO: Ignored info update\", info.what);\n }\n\n if (this.onInfo) {\n this.onInfo(info);\n }\n }\n // Called by Tinode when meta.desc packet is received.\n // Called by 'me' topic on contact update (desc._noForwarding is true).\n _processMetaDesc(desc) {\n if (this.isP2PType()) {\n // Synthetic desc may include defacs for p2p topics which is useless.\n // Remove it.\n delete desc.defacs;\n\n // Update to p2p desc is the same as user update. Update cached user.\n this._tinode._db.updUser(this.name, desc.public);\n }\n\n // Copy parameters from desc object to this topic.\n mergeObj(this, desc);\n // Update persistent cache.\n this._tinode._db.updTopic(this);\n\n // Notify 'me' listener, if available:\n if (this.name !== Const.TOPIC_ME && !desc._noForwarding) {\n const me = this._tinode.getMeTopic();\n if (me.onMetaSub) {\n me.onMetaSub(this);\n }\n if (me.onSubsUpdated) {\n me.onSubsUpdated([this.name], 1);\n }\n }\n\n if (this.onMetaDesc) {\n this.onMetaDesc(this);\n }\n }\n // Called by Tinode when meta.sub is recived or in response to received\n // {ctrl} after setMeta-sub.\n _processMetaSubs(subs) {\n for (let idx in subs) {\n const sub = subs[idx];\n\n // Fill defaults.\n sub.online = !!sub.online;\n // Update timestamp of the most recent subscription update.\n this._lastSubsUpdate = new Date(Math.max(this._lastSubsUpdate, sub.updated));\n\n let user = null;\n if (!sub.deleted) {\n // If this is a change to user's own permissions, update them in topic too.\n // Desc will update 'me' topic.\n if (this._tinode.isMe(sub.user) && sub.acs) {\n this._processMetaDesc({\n updated: sub.updated,\n touched: sub.touched,\n acs: sub.acs\n });\n }\n user = this._updateCachedUser(sub.user, sub);\n } else {\n // Subscription is deleted, remove it from topic (but leave in Users cache)\n delete this._users[sub.user];\n user = sub;\n }\n\n if (this.onMetaSub) {\n this.onMetaSub(user);\n }\n }\n\n if (this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._users));\n }\n }\n // Called by Tinode when meta.tags is recived.\n _processMetaTags(tags) {\n if (tags.length == 1 && tags[0] == Const.DEL_CHAR) {\n tags = [];\n }\n this._tags = tags;\n if (this.onTagsUpdated) {\n this.onTagsUpdated(tags);\n }\n }\n // Do nothing for topics other than 'me'\n _processMetaCreds(creds) {}\n // Delete cached messages and update cached transaction IDs\n _processDelMessages(clear, delseq) {\n this._maxDel = Math.max(clear, this._maxDel);\n this.clear = Math.max(clear, this.clear);\n const topic = this;\n let count = 0;\n if (Array.isArray(delseq)) {\n delseq.forEach(function(range) {\n if (!range.hi) {\n count++;\n topic.flushMessage(range.low);\n } else {\n for (let i = range.low; i < range.hi; i++) {\n count++;\n topic.flushMessage(i);\n }\n }\n });\n }\n\n if (count > 0) {\n // this._updateDeletedRanges();\n\n if (this.onData) {\n this.onData();\n }\n }\n }\n // Topic is informed that the entire response to {get what=data} has been received.\n _allMessagesReceived(count) {\n\n if (this.onAllMessagesReceived) {\n this.onAllMessagesReceived(count);\n }\n }\n // Reset subscribed state\n _resetSub() {\n this._attached = false;\n }\n // This topic is either deleted or unsubscribed from.\n _gone() {\n this._messages.reset();\n this._tinode._db.remMessages(this.name);\n this._users = {};\n this.acs = new AccessMode(null);\n this.private = null;\n this.public = null;\n this.trusted = null;\n this._maxSeq = 0;\n this._minSeq = 0;\n this._attached = false;\n\n const me = this._tinode.getMeTopic();\n if (me) {\n me._routePres({\n _noForwarding: true,\n what: 'gone',\n topic: Const.TOPIC_ME,\n src: this.name\n });\n }\n if (this.onDeleteTopic) {\n this.onDeleteTopic();\n }\n }\n // Update global user cache and local subscribers cache.\n // Don't call this method for non-subscribers.\n _updateCachedUser(uid, obj) {\n // Fetch user object from the global cache.\n // This is a clone of the stored object\n let cached = this._cacheGetUser(uid);\n cached = mergeObj(cached || {}, obj);\n // Save to global cache\n this._cachePutUser(uid, cached);\n // Save to the list of topic subsribers.\n return mergeToCache(this._users, uid, cached);\n }\n // Get local seqId for a queued message.\n _getQueuedSeqId() {\n return this._queuedSeqId++;\n }\n\n // Load most recent messages from persistent cache.\n _loadMessages(db, params) {\n const {\n since,\n before,\n limit\n } = params || {};\n return db.readMessages(this.name, {\n since: since,\n before: before,\n limit: limit || Const.DEFAULT_MESSAGES_PAGE\n })\n .then(msgs => {\n msgs.forEach((data) => {\n if (data.seq > this._maxSeq) {\n this._maxSeq = data.seq;\n }\n if (data.seq < this._minSeq || this._minSeq == 0) {\n this._minSeq = data.seq;\n }\n this._messages.put(data);\n this._maybeUpdateMessageVersionsCache(data);\n });\n return msgs.length;\n });\n }\n // Push or {pres}: message received.\n _updateReceived(seq, act) {\n this.touched = new Date();\n this.seq = seq | 0;\n // Check if message is sent by the current user. If so it's been read already.\n if (!act || this._tinode.isMe(act)) {\n this.read = this.read ? Math.max(this.read, this.seq) : this.seq;\n this.recv = this.recv ? Math.max(this.read, this.recv) : this.read;\n }\n this.unread = this.seq - (this.read | 0);\n this._tinode._db.updTopic(this);\n }\n}\n\n/**\n * @class TopicMe - special case of {@link Tinode.Topic} for\n * managing data of the current user, including contact list.\n * @extends Tinode.Topic\n * @memberof Tinode\n *\n * @param {TopicMe.Callbacks} callbacks - Callbacks to receive various events.\n */\nexport class TopicMe extends Topic {\n onContactUpdate;\n\n constructor(callbacks) {\n super(Const.TOPIC_ME, callbacks);\n\n // me-specific callbacks\n if (callbacks) {\n this.onContactUpdate = callbacks.onContactUpdate;\n }\n }\n\n // Override the original Topic._processMetaDesc.\n _processMetaDesc(desc) {\n // Check if online contacts need to be turned off because P permission was removed.\n const turnOff = (desc.acs && !desc.acs.isPresencer()) && (this.acs && this.acs.isPresencer());\n\n // Copy parameters from desc object to this topic.\n mergeObj(this, desc);\n this._tinode._db.updTopic(this);\n // Update current user's record in the global cache.\n this._updateCachedUser(this._tinode._myUID, desc);\n\n // 'P' permission was removed. All topics are offline now.\n if (turnOff) {\n this._tinode.mapTopics((cont) => {\n if (cont.online) {\n cont.online = false;\n cont.seen = Object.assign(cont.seen || {}, {\n when: new Date()\n });\n this._refreshContact('off', cont);\n }\n });\n }\n\n if (this.onMetaDesc) {\n this.onMetaDesc(this);\n }\n }\n\n // Override the original Topic._processMetaSubs\n _processMetaSubs(subs) {\n let updateCount = 0;\n subs.forEach((sub) => {\n const topicName = sub.topic;\n // Don't show 'me' and 'fnd' topics in the list of contacts.\n if (topicName == Const.TOPIC_FND || topicName == Const.TOPIC_ME) {\n return;\n }\n sub.online = !!sub.online;\n\n let cont = null;\n if (sub.deleted) {\n cont = sub;\n this._tinode.cacheRemTopic(topicName);\n this._tinode._db.remTopic(topicName);\n } else {\n // Ensure the values are defined and are integers.\n if (typeof sub.seq != 'undefined') {\n sub.seq = sub.seq | 0;\n sub.recv = sub.recv | 0;\n sub.read = sub.read | 0;\n sub.unread = sub.seq - sub.read;\n }\n\n const topic = this._tinode.getTopic(topicName);\n if (topic._new) {\n delete topic._new;\n }\n\n cont = mergeObj(topic, sub);\n this._tinode._db.updTopic(cont);\n\n if (Topic.isP2PTopicName(topicName)) {\n this._cachePutUser(topicName, cont);\n this._tinode._db.updUser(topicName, cont.public);\n }\n // Notify topic of the update if it's an external update.\n if (!sub._noForwarding && topic) {\n sub._noForwarding = true;\n topic._processMetaDesc(sub);\n }\n }\n\n updateCount++;\n\n if (this.onMetaSub) {\n this.onMetaSub(cont);\n }\n });\n\n if (this.onSubsUpdated && updateCount > 0) {\n const keys = [];\n subs.forEach((s) => {\n keys.push(s.topic);\n });\n this.onSubsUpdated(keys, updateCount);\n }\n }\n\n // Called by Tinode when meta.sub is recived.\n _processMetaCreds(creds, upd) {\n if (creds.length == 1 && creds[0] == Const.DEL_CHAR) {\n creds = [];\n }\n if (upd) {\n creds.forEach((cr) => {\n if (cr.val) {\n // Adding a credential.\n let idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && el.val == cr.val;\n });\n if (idx < 0) {\n // Not found.\n if (!cr.done) {\n // Unconfirmed credential replaces previous unconfirmed credential of the same method.\n idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && !el.done;\n });\n if (idx >= 0) {\n // Remove previous unconfirmed credential.\n this._credentials.splice(idx, 1);\n }\n }\n this._credentials.push(cr);\n } else {\n // Found. Maybe change 'done' status.\n this._credentials[idx].done = cr.done;\n }\n } else if (cr.resp) {\n // Handle credential confirmation.\n const idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && !el.done;\n });\n if (idx >= 0) {\n this._credentials[idx].done = true;\n }\n }\n });\n } else {\n this._credentials = creds;\n }\n if (this.onCredsUpdated) {\n this.onCredsUpdated(this._credentials);\n }\n }\n\n // Process presence change message\n _routePres(pres) {\n if (pres.what == 'term') {\n // The 'me' topic itself is detached. Mark as unsubscribed.\n this._resetSub();\n return;\n }\n\n if (pres.what == 'upd' && pres.src == Const.TOPIC_ME) {\n // Update to me's description. Request updated value.\n this.getMeta(this.startMetaQuery().withDesc().build());\n return;\n }\n\n const cont = this._tinode.cacheGetTopic(pres.src);\n if (cont) {\n switch (pres.what) {\n case 'on': // topic came online\n cont.online = true;\n break;\n case 'off': // topic went offline\n if (cont.online) {\n cont.online = false;\n cont.seen = Object.assign(cont.seen || {}, {\n when: new Date()\n });\n }\n break;\n case 'msg': // new message received\n cont._updateReceived(pres.seq, pres.act);\n break;\n case 'upd': // desc updated\n // Request updated subscription.\n this.getMeta(this.startMetaQuery().withLaterOneSub(pres.src).build());\n break;\n case 'acs': // access mode changed\n // If 'tgt' is not set then this is an update to the permissions of the current user.\n // Otherwise it's an update to group topic subscriber permissions while the topic is offline.\n // Just gnore it then.\n if (!pres.tgt) {\n if (cont.acs) {\n cont.acs.updateAll(pres.dacs);\n } else {\n cont.acs = new AccessMode().updateAll(pres.dacs);\n }\n cont.touched = new Date();\n }\n break;\n case 'ua':\n // user agent changed.\n cont.seen = {\n when: new Date(),\n ua: pres.ua\n };\n break;\n case 'recv':\n // user's other session marked some messges as received.\n pres.seq = pres.seq | 0;\n cont.recv = cont.recv ? Math.max(cont.recv, pres.seq) : pres.seq;\n break;\n case 'read':\n // user's other session marked some messages as read.\n pres.seq = pres.seq | 0;\n cont.read = cont.read ? Math.max(cont.read, pres.seq) : pres.seq;\n cont.recv = cont.recv ? Math.max(cont.read, cont.recv) : cont.recv;\n cont.unread = cont.seq - cont.read;\n break;\n case 'gone':\n // topic deleted or unsubscribed from.\n this._tinode.cacheRemTopic(pres.src);\n if (!cont._deleted) {\n cont._deleted = true;\n cont._attached = false;\n this._tinode._db.markTopicAsDeleted(pres.src, true);\n } else {\n this._tinode._db.remTopic(pres.src);\n }\n break;\n case 'del':\n // Update topic.del value.\n break;\n default:\n this._tinode.logger(\"INFO: Unsupported presence update in 'me'\", pres.what);\n }\n\n this._refreshContact(pres.what, cont);\n } else {\n if (pres.what == 'acs') {\n // New subscriptions and deleted/banned subscriptions have full\n // access mode (no + or - in the dacs string). Changes to known subscriptions are sent as\n // deltas, but they should not happen here.\n const acs = new AccessMode(pres.dacs);\n if (!acs || acs.mode == AccessMode._INVALID) {\n this._tinode.logger(\"ERROR: Invalid access mode update\", pres.src, pres.dacs);\n return;\n } else if (acs.mode == AccessMode._NONE) {\n this._tinode.logger(\"WARNING: Removing non-existent subscription\", pres.src, pres.dacs);\n return;\n } else {\n // New subscription. Send request for the full description.\n // Using .withOneSub (not .withLaterOneSub) to make sure IfModifiedSince is not set.\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n // Create a dummy entry to catch online status update.\n const dummy = this._tinode.getTopic(pres.src);\n dummy.online = false;\n dummy.acs = acs;\n this._tinode._db.updTopic(dummy);\n }\n } else if (pres.what == 'tags') {\n this.getMeta(this.startMetaQuery().withTags().build());\n } else if (pres.what == 'msg') {\n // Message received for un unknown (previously deleted) topic.\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n // Create an entry to catch updates and messages.\n const dummy = this._tinode.getTopic(pres.src);\n dummy._deleted = false;\n this._tinode._db.updTopic(dummy);\n }\n\n this._refreshContact(pres.what, cont);\n }\n\n if (this.onPres) {\n this.onPres(pres);\n }\n }\n\n // Contact is updated, execute callbacks.\n _refreshContact(what, cont) {\n if (this.onContactUpdate) {\n this.onContactUpdate(what, cont);\n }\n }\n\n /**\n * Publishing to TopicMe is not supported. {@link Topic#publish} is overriden and thows an {Error} if called.\n * @memberof Tinode.TopicMe#\n * @throws {Error} Always throws an error.\n */\n publish() {\n return Promise.reject(new Error(\"Publishing to 'me' is not supported\"));\n }\n\n /**\n * Delete validation credential.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} topic - Name of the topic to delete\n * @param {string} user - User ID to remove.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCredential(method, value) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete credential in inactive 'me' topic\"));\n }\n // Send {del} message, return promise\n return this._tinode.delCredential(method, value).then(ctrl => {\n // Remove deleted credential from the cache.\n const index = this._credentials.findIndex((el) => {\n return el.meth == method && el.val == value;\n });\n if (index > -1) {\n this._credentials.splice(index, 1);\n }\n // Notify listeners\n if (this.onCredsUpdated) {\n this.onCredsUpdated(this._credentials);\n }\n return ctrl;\n });\n }\n\n /**\n * @callback contactFilter\n * @param {Object} contact to check for inclusion.\n * @returns {boolean} true
if contact should be processed, false
to exclude it.\n */\n /**\n * Iterate over cached contacts.\n *\n * @function\n * @memberof Tinode.TopicMe#\n * @param {TopicMe.ContactCallback} callback - Callback to call for each contact.\n * @param {contactFilter=} filter - Optionally filter contacts; include all if filter is false-ish, otherwise\n * include those for which filter returns true-ish.\n * @param {Object=} context - Context to use for calling the `callback`, i.e. the value of `this` inside the callback.\n */\n contacts(callback, filter, context) {\n this._tinode.mapTopics((c, idx) => {\n if (c.isCommType() && (!filter || filter(c))) {\n callback.call(context, c, idx);\n }\n });\n }\n\n /**\n * Get a contact from cache.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to get, either a UID (for p2p topics) or a topic name.\n * @returns {Tinode.Contact} - Contact or `undefined`.\n */\n getContact(name) {\n return this._tinode.cacheGetTopic(name);\n }\n\n /**\n * Get access mode of a given contact from cache.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to get access mode for, either a UID (for p2p topics)\n * or a topic name; if missing, access mode for the 'me' topic itself.\n * @returns {string} - access mode, such as `RWP`.\n */\n getAccessMode(name) {\n if (name) {\n const cont = this._tinode.cacheGetTopic(name);\n return cont ? cont.acs : null;\n }\n return this.acs;\n }\n\n /**\n * Check if contact is archived, i.e. contact.private.arch == true.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to check archived status, either a UID (for p2p topics) or a topic name.\n * @returns {boolean} - true if contact is archived, false otherwise.\n */\n isArchived(name) {\n const cont = this._tinode.cacheGetTopic(name);\n return cont && cont.private && !!cont.private.arch;\n }\n\n /**\n * @typedef Tinode.Credential\n * @memberof Tinode\n * @type Object\n * @property {string} meth - validation method such as 'email' or 'tel'.\n * @property {string} val - credential value, i.e. 'jdoe@example.com' or '+17025551234'\n * @property {boolean} done - true if credential is validated.\n */\n /**\n * Get the user's credentials: email, phone, etc.\n * @memberof Tinode.TopicMe#\n *\n * @returns {Tinode.Credential[]} - array of credentials.\n */\n getCredentials() {\n return this._credentials;\n }\n}\n\n/**\n * Special case of {@link Tinode.Topic} for searching for contacts and group topics\n * @extends Tinode.Topic\n *\n */\nexport class TopicFnd extends Topic {\n // List of users and topics uid or topic_name -> Contact object)\n _contacts = {};\n\n /**\n * Create TopicFnd.\n *\n * @param {TopicFnd.Callbacks} callbacks - Callbacks to receive various events.\n */\n constructor(callbacks) {\n super(Const.TOPIC_FND, callbacks);\n }\n\n // Override the original Topic._processMetaSubs\n _processMetaSubs(subs) {\n let updateCount = Object.getOwnPropertyNames(this._contacts).length;\n // Reset contact list.\n this._contacts = {};\n for (let idx in subs) {\n let sub = subs[idx];\n const indexBy = sub.topic ? sub.topic : sub.user;\n\n sub = mergeToCache(this._contacts, indexBy, sub);\n updateCount++;\n\n if (this.onMetaSub) {\n this.onMetaSub(sub);\n }\n }\n\n if (updateCount > 0 && this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._contacts));\n }\n }\n\n /**\n * Publishing to TopicFnd is not supported. {@link Topic#publish} is overriden and thows an {Error} if called.\n * @memberof Tinode.TopicFnd#\n * @throws {Error} Always throws an error.\n */\n publish() {\n return Promise.reject(new Error(\"Publishing to 'fnd' is not supported\"));\n }\n\n /**\n * setMeta to TopicFnd resets contact list in addition to sending the message.\n * @memberof Tinode.TopicFnd#\n * @param {Tinode.SetParams} params parameters to update.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n setMeta(params) {\n return Object.getPrototypeOf(TopicFnd.prototype).setMeta.call(this, params).then(_ => {\n if (Object.keys(this._contacts).length > 0) {\n this._contacts = {};\n if (this.onSubsUpdated) {\n this.onSubsUpdated([]);\n }\n }\n });\n }\n\n /**\n * Iterate over found contacts. If callback is undefined, use {@link this.onMetaSub}.\n * @function\n * @memberof Tinode.TopicFnd#\n * @param {TopicFnd.ContactCallback} callback - Callback to call for each contact.\n * @param {Object} context - Context to use for calling the `callback`, i.e. the value of `this` inside the callback.\n */\n contacts(callback, context) {\n const cb = (callback || this.onMetaSub);\n if (cb) {\n for (let idx in this._contacts) {\n cb.call(context, this._contacts[idx], idx, this._contacts);\n }\n }\n }\n}\n"],"names":["root","factory","exports","module","define","amd","this","MAX_PREVIEW_DATA_SIZE","JSON_MIME_TYPE","ALLOWED_ENT_FIELDS","INLINE_STYLES","name","start","end","FMT_WEIGHT","ENTITY_TYPES","dataName","pack","val","test","url","re","slice","FORMAT_TAGS","AU","html_tag","md_tag","undefined","isVoid","BN","BR","CO","DL","EM","EX","FM","HD","HL","HT","IM","LN","MN","RW","QQ","ST","VC","VD","base64toObjectUrl","b64","contentType","logger","bin","atob","length","buf","ArrayBuffer","arr","Uint8Array","i","charCodeAt","URL","createObjectURL","Blob","type","err","message","base64toDataUrl","DECORATORS","open","_","close","data","props","href","target","id","act","ref","mime","Drafty","src","duration","size","tmpPreviewUrl","_tempPreview","previewUrl","downloadUrl","width","height","title","alt","state","preview","premime","poster","preref","txt","fmt","ent","chunkify","line","spans","chunks","span","at","push","chunk","tp","chld","children","toSpanTree","tree","last","draftyToTree","doc","Array","isArray","text","len","key","attachments","forEach","includes","sort","a","b","diff","indexOf","spansToTree","treeTopDown","node","child","parent","addNode","n","substring","att","subspans","inner","tag","treeToDrafty","keymap","c","Object","keys","newKey","transformer","context","dst","call","treeBottomUp","formatter","index","stack","values","pop","shortenTree","limit","tail","lightEntity","allow","copyEntData","lTrim","trimStart","shift","attachmentsToEnd","concat","draftify","startAt","plain","ranges","drafty","light","entries","dc","init","plainText","parse","content","lines","split","entityMap","entityIndex","blx","entities","block","original","re_start","re_end","result","exec","start_offset","lastIndexOf","end_offset","spannify","match","extracted","entity","offset","unique","idx","filter","el","extractEntities","map","s","append","first","second","insertImage","imageDesc","ex","refurl","bits","filename","urlPromise","_processing","then","insertVideo","videoDesc","urls","insertAudio","audioDesc","videoCall","audioOnly","aonly","updateVideoCall","params","assign","quote","header","uid","body","appendLineBreak","mention","appendLink","linkData","appendImage","appendAudio","attachFile","attachmentDesc","wrapInto","style","wrapAsForm","insertButton","actionType","actionValue","refUrl","appendButton","attachJSON","UNSAFE_toHTML","join","format","shorten","forwardedContent","replyContent","startsWith","forwarding","toPlainText","isPlainText","toMarkdown","def","isValid","txt_type","hasAttachments","callback","count","hasEntities","styles","sanitizeEntities","getDownloadUrl","entData","isProcessing","getPreviewUrl","getEntitySize","getEntityMimeType","tagName","attrValue","getContentType","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","__webpack_modules__","getter","__esModule","d","definition","o","defineProperty","enumerable","get","g","globalThis","Function","e","window","obj","prop","prototype","hasOwnProperty","r","Symbol","toStringTag","value","AccessMode","constructor","acs","given","decode","want","mode","side","flag","Error","str","_BITMASK","_NONE","bitmask","_JOIN","_READ","_WRITE","_PRES","_APPROVE","_SHARE","_DELETE","_OWNER","m0","bit","charAt","toUpperCase","encode","_INVALID","res","update","upd","action","val0","parts","a1","a2","toString","jsonHelper","setMode","m","updateMode","u","getMode","setGiven","updateGiven","getGiven","setWant","w","updateWant","getWant","getMissing","getExcessive","updateAll","isOwner","isPresencer","isMuted","isJoiner","isReader","isWriter","isApprover","isAdmin","isSharer","isDeleter","VERSION","LIBRARY","TOPIC_NEW","TOPIC_NEW_CHAN","TOPIC_ME","TOPIC_FND","TOPIC_GRP","LOCAL_SEQID","DEL_CHAR","CommError","code","super","jsonParseHelper","date","Date","isNaN","isUrlRelative","isValidDate","getTime","mergeObj","ignore","mergeToCache","cache","newval","simplify","getOwnPropertyNames","WebSocketProvider","XHRProvider","NETWORK_ERROR_TEXT","NETWORK_USER","NETWORK_USER_TEXT","makeBaseUrl","host","protocol","version","apiKey","Connection","static","secure","autoreconnect","initialized","config","version_","autoreconnect_","transport","setNetworkProviders","wsProvider","xhrProvider","l","connect","host_","force","Promise","reject","reconnect","disconnect","sendText","msg","isConnected","probe","backoffReset","clearTimeout","timeout","Math","pow","random","onAutoreconnectIteration","setTimeout","prom","catch","_lpURL","_poller","_sender","lp_poller","url_","resolve","poller","promiseCompleted","onreadystatechange","evt","readyState","status","pkt","JSON","responseText","ctrl","sid","send","onOpen","onMessage","onDisconnect","abort","sender","lp_sender","OPEN","conn","onerror","onopen","onclose","onmessage","NETWORK_ERROR","DB_NAME","IDBProvider","DB","db","disabled","onError","source","trx","transaction","event","error","objectStore","getAll","onsuccess","topic","initDatabase","req","onupgradeneeded","createObjectStore","keyPath","deleteDatabase","onblocked","isReady","updTopic","oncomplete","put","commit","markTopicAsDeleted","deleted","_deleted","remTopic","delete","IDBKeyRange","only","bound","Number","MAX_SAFE_INTEGER","mapTopics","deserializeTopic","updUser","pub","arguments","public","remUser","mapUsers","getUser","user","updSubscription","topicName","sub","mapSubscriptions","addMessage","add","updMessageStatus","seq","_status","remMessages","from","to","range","readMessages","query","since","before","openCursor","cursor","continue","f","tags","_tags","setAccessMode","read","unread","max","getAccessMode","setDatabaseProvider","idbProvider","IndexedDBProvider","LargeFileHelper","tinode","_tinode","_version","_apiKey","_authToken","getAuthToken","xhr","uploadWithBaseUrl","baseUrl","avatarFor","onProgress","onSuccess","onFailure","base","endsWith","instance","setRequestHeader","token","toResolve","toReject","upload","onprogress","lengthComputable","loaded","total","onload","response","statusText","onabort","form","FormData","set","getNextUniqueId","_secure","_host","download","relativeUrl","mimetype","relUrl","location","origin","searchParams","addURLParam","responseType","link","document","createElement","display","setAttribute","appendChild","click","removeChild","revokeObjectURL","reader","FileReader","readAsText","cancel","setNetworkProvider","MetaGetBuilder","what","updated","isP2PType","_lastSubsUpdate","withData","withLaterData","_maxSeq","withEarlierData","_minSeq","withDesc","ims","withLaterDesc","withSub","userOrTopic","opts","getType","withOneSub","withLaterOneSub","withLaterSub","withTags","withCred","withDel","withLaterDel","_maxDel","extract","build","CBuffer","buffer","compare_","unique_","elem","exact","pivot","found","splice","getAt","getLast","insert","delAt","delRange","reset","startIdx","beforeIdx","find","nearest","isEmpty","Topic","callbacks","created","touched","private","trusted","_users","_queuedSeqId","Const","_noEarlierMsgs","_recvNotificationTimer","_credentials","_messageVersions","_messages","_attached","_new","_delayedLeaveTimer","onData","onMeta","onPres","onInfo","onMetaDesc","onMetaSub","onSubsUpdated","onTagsUpdated","onCredsUpdated","onDeleteTopic","onAllMessagesReceived","topicType","isMeTopicName","isGroupTopicName","isP2PTopicName","isCommTopicName","isNewGroupTopicName","isChannelTopicName","isSubscribed","subscribe","getParams","setParams","_cacheDelSelf","_cachePutSelf","ts","me","getMeTopic","desc","_noForwarding","_processMetaDesc","createMessage","noEcho","publish","publishMessage","_sending","_failed","swapMessageId","_maybeUpdateMessageVersionsCache","_routeData","publishDraft","_getQueuedSeqId","getCurrentUserID","noecho","_db","_cancelled","_fatal","leave","unsub","_resetSub","_gone","leaveDelayed","delay","getMeta","getMessagesPage","forward","startMetaQuery","_loadMessages","promise","setMeta","out","t","trim","toLowerCase","item","pos","ary","normalizeArray","_processMetaSubs","_processMetaTags","cred","_processMetaCreds","subscriber","am","invite","archive","arch","delMessages","hard","r1","r2","low","hi","tosend","reduce","del","flushMessageRange","flushMessage","delMessagesAll","hardDel","_all","delMessagesList","list","prev","delMessagesEdits","messageVersions","delTopic","delSubscription","note","_updateMyReadRecv","_refreshContact","noteRecv","noteRead","noteKeyPress","noteRecording","payload","oldVal","doUpdate","recv","userDesc","_cacheGetUser","p2pPeerDesc","subscribers","cb","origSeq","versions","messages","sinceId","beforeId","msgs","unused1","unused2","_isReplacementMsg","latest","latestMsgVersion","_origTs","_origSeq","findMessage","latestMessage","maxMsgSeq","minMsgSeq","maxClearId","messageCount","queuedMessages","msgReceiptCount","msgReadCount","msgRecvCount","msgHasMoreMessages","newer","isNewMessage","seqId","fromId","untilId","newSeqId","numMessages","cancelSend","msgStatus","getDefaultAccess","defacs","isArchived","isMeType","isChannelType","isGroupType","isCommType","isMe","head","replace","targetSeq","parseInt","targetMsg","outgoing","webrtc","incoming","_routeInfo","_routeMeta","meta","_processDelMessages","clear","delseq","_routePres","pres","online","isTopicCached","dacs","info","subs","_updateCachedUser","creds","_allMessagesReceived","cached","_cachePutUser","_updateReceived","TopicMe","onContactUpdate","turnOff","_myUID","cont","seen","when","updateCount","cacheRemTopic","getTopic","cr","findIndex","meth","done","resp","cacheGetTopic","tgt","ua","dummy","delCredential","method","contacts","getContact","getCredentials","TopicFnd","_contacts","indexBy","getPrototypeOf","b64EncodeUnicode","btoa","encodeURIComponent","p1","String","fromCharCode","jsonBuildHelper","pad","sp","repeat","millis","getUTCMilliseconds","getUTCFullYear","getUTCMonth","getUTCDate","getUTCHours","getUTCMinutes","getUTCSeconds","rfc3339DateString","jsonLoggerHelper","WebSocket","XMLHttpRequest","indexedDB","chars","global","output","charCode","bc","bs","DBCache","initForNonBrowserApp","Tinode","_appName","_browser","_platform","_hwos","_humanLanguage","_loggingEnabled","_trimLongStrings","_authenticated","_login","_inPacketCount","_messageId","floor","_serverInfo","_deviceToken","_pendingPromises","_expirePromises","_connection","_persist","_cache","onComplete","appName","platform","navigator","product","reactnative","priority","tmp","substr","tokens","m2","v","minor","getBrowserInfo","userAgent","language","detectTransport","persist","all","dateString","_len","args","_key","console","log","onOK","errorText","stringify","onRawMessage","onNetworkProbe","onCtrlMessage","onMetaMessage","onDataMessage","onPresMessage","onInfoMessage","setInterval","expires","hello","clearInterval","func","_cacheDelUser","onLogin","credential","getVersion","getLibrary","isNullValue","clearStorage","initStorage","networkProbe","isAuthenticated","authorizeURL","parsed","account","scheme","secret","login","acc","tmpscheme","tmpsecret","extra","createAccount","createAccountBasic","username","password","updateAccountBasic","onConnect","setDeviceToken","dt","sent","loginBasic","uname","loginToken","requestResetAuthSecret","now","setAuthToken","dft","oobNotification","xfrom","finally","modeGiven","modeWant","delCurrentUser","newGroupTopicName","isChan","getFndTopic","getLargeFileHelper","getCurrentLogin","getServerInfo","report","getServerParam","defaultValue","enableLogging","enabled","trimLongStrings","setHumanLanguage","hl","isTopicOnline","getTopicAccessMode","wantAkn","onWebsocketOpen","MESSAGE_STATUS_NONE","MESSAGE_STATUS_QUEUED","MESSAGE_STATUS_SENDING","MESSAGE_STATUS_FAILED","MESSAGE_STATUS_FATAL","MESSAGE_STATUS_SENT","MESSAGE_STATUS_RECEIVED","MESSAGE_STATUS_READ","MESSAGE_STATUS_TO_ME","MAX_MESSAGE_SIZE","MAX_SUBSCRIBER_COUNT","MAX_TAG_COUNT","MAX_FILE_UPLOAD_SIZE","REQ_CRED_VALIDATORS","URI_TOPIC_ID_PREFIX"],"sourceRoot":""}
\ No newline at end of file
+{"version":3,"file":"tinode.prod.js","mappings":"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","sources":["webpack://tinode/webpack/universalModuleDefinition","webpack://tinode/./src/drafty.js","webpack://tinode/webpack/bootstrap","webpack://tinode/webpack/runtime/compat get default export","webpack://tinode/webpack/runtime/define property getters","webpack://tinode/webpack/runtime/global","webpack://tinode/webpack/runtime/hasOwnProperty shorthand","webpack://tinode/webpack/runtime/make namespace object","webpack://tinode/./src/access-mode.js","webpack://tinode/./version.js","webpack://tinode/./src/config.js","webpack://tinode/./src/comm-error.js","webpack://tinode/./src/utils.js","webpack://tinode/./src/connection.js","webpack://tinode/./src/db.js","webpack://tinode/./src/large-file.js","webpack://tinode/./src/tinode.js","webpack://tinode/./src/meta-builder.js","webpack://tinode/./src/cbuffer.js","webpack://tinode/./src/topic.js"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"tinode\"] = factory();\n\telse\n\t\troot[\"tinode\"] = factory();\n})(this, () => {\nreturn ","/**\n * @copyright 2015-2022 Tinode LLC.\n * @summary Minimally rich text representation and formatting for Tinode.\n * @license Apache 2.0\n *\n * @file Basic parser and formatter for very simple text markup. Mostly targeted at\n * mobile use cases similar to Telegram, WhatsApp, and FB Messenger.\n *\n * Supports conversion of user keyboard input to formatted text:
\n * \n * - *abc* → abc
\n * - _abc_ → abc
\n * - ~abc~ →
abc \n * - `abc` → abc
\n *
\n * Also supports forms and buttons.\n *\n * Nested formatting is supported, e.g. *abc _def_* -> abc def\n * URLs, @mentions, and #hashtags are extracted and converted into links.\n * Forms and buttons can be added procedurally.\n * JSON data representation is inspired by Draft.js raw formatting.\n *\n *\n * @example\n * Text:\n * \n * this is *bold*, `code` and _italic_, ~strike~\n * combined *bold and _italic_*\n * an url: https://www.example.com/abc#fragment and another _www.tinode.co_\n * this is a @mention and a #hashtag in a string\n * second #hashtag\n *
\n *\n * Sample JSON representation of the text above:\n * {\n * \"txt\": \"this is bold, code and italic, strike combined bold and italic an url: https://www.example.com/abc#fragment \" +\n * \"and another www.tinode.co this is a @mention and a #hashtag in a string second #hashtag\",\n * \"fmt\": [\n * { \"at\":8, \"len\":4,\"tp\":\"ST\" },{ \"at\":14, \"len\":4, \"tp\":\"CO\" },{ \"at\":23, \"len\":6, \"tp\":\"EM\"},\n * { \"at\":31, \"len\":6, \"tp\":\"DL\" },{ \"tp\":\"BR\", \"len\":1, \"at\":37 },{ \"at\":56, \"len\":6, \"tp\":\"EM\" },\n * { \"at\":47, \"len\":15, \"tp\":\"ST\" },{ \"tp\":\"BR\", \"len\":1, \"at\":62 },{ \"at\":120, \"len\":13, \"tp\":\"EM\" },\n * { \"at\":71, \"len\":36, \"key\":0 },{ \"at\":120, \"len\":13, \"key\":1 },{ \"tp\":\"BR\", \"len\":1, \"at\":133 },\n * { \"at\":144, \"len\":8, \"key\":2 },{ \"at\":159, \"len\":8, \"key\":3 },{ \"tp\":\"BR\", \"len\":1, \"at\":179 },\n * { \"at\":187, \"len\":8, \"key\":3 },{ \"tp\":\"BR\", \"len\":1, \"at\":195 }\n * ],\n * \"ent\": [\n * { \"tp\":\"LN\", \"data\":{ \"url\":\"https://www.example.com/abc#fragment\" } },\n * { \"tp\":\"LN\", \"data\":{ \"url\":\"http://www.tinode.co\" } },\n * { \"tp\":\"MN\", \"data\":{ \"val\":\"mention\" } },\n * { \"tp\":\"HT\", \"data\":{ \"val\":\"hashtag\" } }\n * ]\n * }\n */\n\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nconst MAX_FORM_ELEMENTS = 8;\nconst MAX_PREVIEW_ATTACHMENTS = 3;\nconst MAX_PREVIEW_DATA_SIZE = 64;\nconst JSON_MIME_TYPE = 'application/json';\nconst DRAFTY_MIME_TYPE = 'text/x-drafty';\nconst ALLOWED_ENT_FIELDS = ['act', 'height', 'duration', 'incoming', 'mime', 'name', 'premime', 'preref', 'preview',\n 'ref', 'size', 'state', 'url', 'val', 'width'\n];\n\n// Regular expressions for parsing inline formats. Javascript does not support lookbehind,\n// so it's a bit messy.\nconst INLINE_STYLES = [\n // Strong = bold, *bold text*\n {\n name: 'ST',\n start: /(?:^|[\\W_])(\\*)[^\\s*]/,\n end: /[^\\s*](\\*)(?=$|[\\W_])/\n },\n // Emphesized = italic, _italic text_\n {\n name: 'EM',\n start: /(?:^|\\W)(_)[^\\s_]/,\n end: /[^\\s_](_)(?=$|\\W)/\n },\n // Deleted, ~strike this though~\n {\n name: 'DL',\n start: /(?:^|[\\W_])(~)[^\\s~]/,\n end: /[^\\s~](~)(?=$|[\\W_])/\n },\n // Code block `this is monospace`\n {\n name: 'CO',\n start: /(?:^|\\W)(`)[^`]/,\n end: /[^`](`)(?=$|\\W)/\n }\n];\n\n// Relative weights of formatting spans. Greater index in array means greater weight.\nconst FMT_WEIGHT = ['QQ'];\n\n// RegExps for entity extraction (RF = reference)\nconst ENTITY_TYPES = [\n // URLs\n {\n name: 'LN',\n dataName: 'url',\n pack: function(val) {\n // Check if the protocol is specified, if not use http\n if (!/^[a-z]+:\\/\\//i.test(val)) {\n val = 'http://' + val;\n }\n return {\n url: val\n };\n },\n re: /(?:(?:https?|ftp):\\/\\/|www\\.|ftp\\.)[-A-Z0-9+&@#\\/%=~_|$?!:,.]*[A-Z0-9+&@#\\/%=~_|$]/ig\n },\n // Mentions @user (must be 2 or more characters)\n {\n name: 'MN',\n dataName: 'val',\n pack: function(val) {\n return {\n val: val.slice(1)\n };\n },\n re: /\\B@([\\p{L}\\p{N}][._\\p{L}\\p{N}]*[\\p{L}\\p{N}])/ug\n },\n // Hashtags #hashtag, like metion 2 or more characters.\n {\n name: 'HT',\n dataName: 'val',\n pack: function(val) {\n return {\n val: val.slice(1)\n };\n },\n re: /\\B#([\\p{L}\\p{N}][._\\p{L}\\p{N}]*[\\p{L}\\p{N}])/ug\n }\n];\n\n// HTML tag name suggestions\nconst FORMAT_TAGS = {\n AU: {\n html_tag: 'audio',\n md_tag: undefined,\n isVoid: false\n },\n BN: {\n html_tag: 'button',\n md_tag: undefined,\n isVoid: false\n },\n BR: {\n html_tag: 'br',\n md_tag: '\\n',\n isVoid: true\n },\n CO: {\n html_tag: 'tt',\n md_tag: '`',\n isVoid: false\n },\n DL: {\n html_tag: 'del',\n md_tag: '~',\n isVoid: false\n },\n EM: {\n html_tag: 'i',\n md_tag: '_',\n isVoid: false\n },\n EX: {\n html_tag: '',\n md_tag: undefined,\n isVoid: true\n },\n FM: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n HD: {\n html_tag: '',\n md_tag: undefined,\n isVoid: false\n },\n HL: {\n html_tag: 'span',\n md_tag: undefined,\n isVoid: false\n },\n HT: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n IM: {\n html_tag: 'img',\n md_tag: undefined,\n isVoid: false\n },\n LN: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n MN: {\n html_tag: 'a',\n md_tag: undefined,\n isVoid: false\n },\n RW: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false,\n },\n QQ: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n ST: {\n html_tag: 'b',\n md_tag: '*',\n isVoid: false\n },\n VC: {\n html_tag: 'div',\n md_tag: undefined,\n isVoid: false\n },\n VD: {\n html_tag: 'video',\n md_tag: undefined,\n isVoid: false\n }\n};\n\n// Convert base64-encoded string into Blob.\nfunction base64toObjectUrl(b64, contentType, logger) {\n if (!b64) {\n return null;\n }\n\n try {\n const bin = atob(b64);\n const length = bin.length;\n const buf = new ArrayBuffer(length);\n const arr = new Uint8Array(buf);\n for (let i = 0; i < length; i++) {\n arr[i] = bin.charCodeAt(i);\n }\n\n return URL.createObjectURL(new Blob([buf], {\n type: contentType\n }));\n } catch (err) {\n if (logger) {\n logger(\"Drafty: failed to convert object.\", err.message);\n }\n }\n\n return null;\n}\n\nfunction base64toDataUrl(b64, contentType) {\n if (!b64) {\n return null;\n }\n contentType = contentType || 'image/jpeg';\n return 'data:' + contentType + ';base64,' + b64;\n}\n\n// Helpers for converting Drafty to HTML.\nconst DECORATORS = {\n // Visial styles\n ST: {\n open: _ => '',\n close: _ => ''\n },\n EM: {\n open: _ => '',\n close: _ => ''\n },\n DL: {\n open: _ => '',\n close: _ => ''\n },\n CO: {\n open: _ => '',\n close: _ => ''\n },\n // Line break\n BR: {\n open: _ => '
',\n close: _ => ''\n },\n // Hidden element\n HD: {\n open: _ => '',\n close: _ => ''\n },\n // Highlighted element.\n HL: {\n open: _ => '',\n close: _ => ''\n },\n // Link (URL)\n LN: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n href: data.url,\n target: '_blank'\n } : null;\n },\n },\n // Mention\n MN: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n id: data.val\n } : null;\n },\n },\n // Hashtag\n HT: {\n open: (data) => {\n return '';\n },\n close: _ => '',\n props: (data) => {\n return data ? {\n id: data.val\n } : null;\n },\n },\n // Button\n BN: {\n open: _ => '',\n props: (data) => {\n return data ? {\n 'data-act': data.act,\n 'data-val': data.val,\n 'data-name': data.name,\n 'data-ref': data.ref\n } : null;\n },\n },\n // Audio recording\n AU: {\n open: (data) => {\n const url = data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger);\n return '',\n props: (data) => {\n if (!data) return null;\n return {\n // Embedded data or external link.\n src: data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n 'data-preload': data.ref ? 'metadata' : 'auto',\n 'data-duration': data.duration,\n 'data-name': data.name,\n 'data-size': data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0),\n 'data-mime': data.mime,\n };\n }\n },\n // Image\n IM: {\n open: data => {\n // Don't use data.ref for preview: it's a security risk.\n const tmpPreviewUrl = base64toDataUrl(data._tempPreview, data.mime);\n const previewUrl = base64toObjectUrl(data.val, data.mime, Drafty.logger);\n const downloadUrl = data.ref || previewUrl;\n return (data.name ? '' : '') +\n '';\n },\n close: data => {\n return (data.name ? '' : '');\n },\n props: data => {\n if (!data) return null;\n return {\n // Temporary preview, or permanent preview, or external link.\n src: base64toDataUrl(data._tempPreview, data.mime) ||\n data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n title: data.name,\n alt: data.name,\n 'data-width': data.width,\n 'data-height': data.height,\n 'data-name': data.name,\n 'data-size': data.ref ? (data.size | 0) : (data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0)),\n 'data-mime': data.mime,\n };\n },\n },\n // Form - structured layout of elements.\n FM: {\n open: _ => '',\n close: _ => ''\n },\n // Row: logic grouping of elements\n RW: {\n open: _ => '',\n close: _ => ''\n },\n // Quoted block.\n QQ: {\n open: _ => '',\n close: _ => '',\n props: (data) => {\n return data ? {} : null;\n },\n },\n // Video call\n VC: {\n open: _ => '',\n close: _ => '',\n props: data => {\n if (!data) return {};\n return {\n 'data-duration': data.duration,\n 'data-state': data.state,\n };\n }\n },\n // Video.\n VD: {\n open: data => {\n const tmpPreviewUrl = base64toDataUrl(data._tempPreview, data.mime);\n const previewUrl = data.ref || base64toObjectUrl(data.preview, data.premime || 'image/jpeg', Drafty.logger);\n return '';\n },\n close: _ => '',\n props: data => {\n if (!data) return null;\n const poster = data.preref || base64toObjectUrl(data.preview, data.premime || 'image/jpeg', Drafty.logger);\n return {\n // Embedded data or external link.\n src: poster,\n 'data-src': data.ref || base64toObjectUrl(data.val, data.mime, Drafty.logger),\n 'data-width': data.width,\n 'data-height': data.height,\n 'data-preload': data.ref ? 'metadata' : 'auto',\n 'data-preview': poster,\n 'data-duration': data.duration | 0,\n 'data-name': data.name,\n 'data-size': data.ref ? (data.size | 0) : (data.val ? ((data.val.length * 0.75) | 0) : (data.size | 0)),\n 'data-mime': data.mime,\n };\n }\n },\n};\n\n/**\n * The main object which performs all the formatting actions.\n * @class Drafty\n * @constructor\n */\nconst Drafty = function() {\n this.txt = '';\n this.fmt = [];\n this.ent = [];\n}\n\n/**\n * Initialize Drafty document to a plain text string.\n *\n * @param {string} plainText - string to use as Drafty content.\n *\n * @returns new Drafty document or null is plainText is not a string or undefined.\n */\nDrafty.init = function(plainText) {\n if (typeof plainText == 'undefined') {\n plainText = '';\n } else if (typeof plainText != 'string') {\n return null;\n }\n\n return {\n txt: plainText\n };\n}\n\n/**\n * Parse plain text into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {string} content - plain-text content to parse.\n * @return {Drafty} parsed document or null if the source is not plain text.\n */\nDrafty.parse = function(content) {\n // Make sure we are parsing strings only.\n if (typeof content != 'string') {\n return null;\n }\n\n // Split text into lines. It makes further processing easier.\n const lines = content.split(/\\r?\\n/);\n\n // Holds entities referenced from text\n const entityMap = [];\n const entityIndex = {};\n\n // Processing lines one by one, hold intermediate result in blx.\n const blx = [];\n lines.forEach((line) => {\n let spans = [];\n let entities;\n\n // Find formatted spans in the string.\n // Try to match each style.\n INLINE_STYLES.forEach((tag) => {\n // Each style could be matched multiple times.\n spans = spans.concat(spannify(line, tag.start, tag.end, tag.name));\n });\n\n let block;\n if (spans.length == 0) {\n block = {\n txt: line\n };\n } else {\n // Sort spans by style occurence early -> late, then by length: first long then short.\n spans.sort((a, b) => {\n const diff = a.at - b.at;\n return diff != 0 ? diff : b.end - a.end;\n });\n\n // Convert an array of possibly overlapping spans into a tree.\n spans = toSpanTree(spans);\n\n // Build a tree representation of the entire string, not\n // just the formatted parts.\n const chunks = chunkify(line, 0, line.length, spans);\n\n const drafty = draftify(chunks, 0);\n\n block = {\n txt: drafty.txt,\n fmt: drafty.fmt\n };\n }\n\n // Extract entities from the cleaned up string.\n entities = extractEntities(block.txt);\n if (entities.length > 0) {\n const ranges = [];\n for (let i in entities) {\n // {offset: match['index'], unique: match[0], len: match[0].length, data: ent.packer(), type: ent.name}\n const entity = entities[i];\n let index = entityIndex[entity.unique];\n if (!index) {\n index = entityMap.length;\n entityIndex[entity.unique] = index;\n entityMap.push({\n tp: entity.type,\n data: entity.data\n });\n }\n ranges.push({\n at: entity.offset,\n len: entity.len,\n key: index\n });\n }\n block.ent = ranges;\n }\n\n blx.push(block);\n });\n\n const result = {\n txt: ''\n };\n\n // Merge lines and save line breaks as BR inline formatting.\n if (blx.length > 0) {\n result.txt = blx[0].txt;\n result.fmt = (blx[0].fmt || []).concat(blx[0].ent || []);\n\n for (let i = 1; i < blx.length; i++) {\n const block = blx[i];\n const offset = result.txt.length + 1;\n\n result.fmt.push({\n tp: 'BR',\n len: 1,\n at: offset - 1\n });\n\n result.txt += ' ' + block.txt;\n if (block.fmt) {\n result.fmt = result.fmt.concat(block.fmt.map((s) => {\n s.at += offset;\n return s;\n }));\n }\n if (block.ent) {\n result.fmt = result.fmt.concat(block.ent.map((s) => {\n s.at += offset;\n return s;\n }));\n }\n }\n\n if (result.fmt.length == 0) {\n delete result.fmt;\n }\n\n if (entityMap.length > 0) {\n result.ent = entityMap;\n }\n }\n return result;\n}\n\n/**\n * Append one Drafty document to another.\n *\n * @param {Drafty} first - Drafty document to append to.\n * @param {Drafty|string} second - Drafty document or string being appended.\n *\n * @return {Drafty} first document with the second appended to it.\n */\nDrafty.append = function(first, second) {\n if (!first) {\n return second;\n }\n if (!second) {\n return first;\n }\n\n first.txt = first.txt || '';\n const len = first.txt.length;\n\n if (typeof second == 'string') {\n first.txt += second;\n } else if (second.txt) {\n first.txt += second.txt;\n }\n\n if (Array.isArray(second.fmt)) {\n first.fmt = first.fmt || [];\n if (Array.isArray(second.ent)) {\n first.ent = first.ent || [];\n }\n second.fmt.forEach(src => {\n const fmt = {\n at: (src.at | 0) + len,\n len: src.len | 0\n };\n // Special case for the outside of the normal rendering flow styles.\n if (src.at == -1) {\n fmt.at = -1;\n fmt.len = 0;\n }\n if (src.tp) {\n fmt.tp = src.tp;\n } else {\n fmt.key = first.ent.length;\n first.ent.push(second.ent[src.key || 0]);\n }\n first.fmt.push(fmt);\n });\n }\n\n return first;\n}\n\n/**\n * Description of an image to attach.\n * @typedef {Object} ImageDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the image, e.g. \"image/png\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - base64-encoded image content. Could be null/undefined.\n * @property {string} preview - base64-encoded thumbnail of the image.\n * @property {integer} width - width of the image.\n * @property {integer} height - height of the image.\n * @property {string} filename - file name suggestion for downloading the image.\n * @property {integer} size - size of the image in bytes. Treat is as an untrusted hint.\n * @property {string} _tempPreview - base64-encoded image preview used during upload process; not serializable.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Insert inline image into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add image to.\n * @param {integer} at - index where the object is inserted. The length of the image is always 1.\n * @param {ImageDesc} imageDesc - object with image paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertImage = function(content, at, imageDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'IM',\n data: {\n mime: imageDesc.mime,\n ref: imageDesc.refurl,\n val: imageDesc.bits || imageDesc.preview,\n width: imageDesc.width,\n height: imageDesc.height,\n name: imageDesc.filename,\n size: imageDesc.size | 0,\n }\n };\n\n if (imageDesc.urlPromise) {\n ex.data._tempPreview = imageDesc._tempPreview;\n ex.data._processing = true;\n imageDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._tempPreview = undefined;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Description of a video to attach.\n * @typedef {Object} VideoDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the video, e.g. \"video/mpeg\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - in-band base64-encoded image data. Could be null/undefined.\n * @property {string} preview - base64-encoded screencapture from the video. Could be null/undefined.\n * @property {string} preref - reference to screencapture from the video. Could be null/undefined.\n * @property {integer} width - width of the video.\n * @property {integer} height - height of the video.\n * @property {integer} duration - duration of the video.\n * @property {string} filename - file name suggestion for downloading the video.\n * @property {integer} size - size of the video in bytes. Treat is as an untrusted hint.\n * @property {string} _tempPreview - base64-encoded screencapture used during upload process; not serializable.\n * @property {Promise} urlPromise - array of two promises, which return URLs of video and preview uploads correspondingly\n * (either could be null).\n */\n\n/**\n * Insert inline image into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add video to.\n * @param {integer} at - index where the object is inserted. The length of the video is always 1.\n * @param {VideoDesc} videoDesc - object with video paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertVideo = function(content, at, videoDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'VD',\n data: {\n mime: videoDesc.mime,\n ref: videoDesc.refurl,\n val: videoDesc.bits,\n preref: videoDesc.preref,\n preview: videoDesc.preview,\n width: videoDesc.width,\n height: videoDesc.height,\n duration: videoDesc.duration | 0,\n name: videoDesc.filename,\n size: videoDesc.size | 0,\n }\n };\n\n if (videoDesc.urlPromise) {\n ex.data._tempPreview = videoDesc._tempPreview;\n ex.data._processing = true;\n videoDesc.urlPromise.then(\n urls => {\n ex.data.ref = urls[0];\n ex.data.preref = urls[1];\n ex.data._tempPreview = undefined;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Description of an audio recording to attach.\n * @typedef {Object} AudioDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the audio, e.g. \"audio/ogg\".\n * @property {string} refurl - reference to the content. Could be null/undefined.\n * @property {string} bits - base64-encoded audio content. Could be null/undefined.\n * @property {integer} duration - duration of the record in milliseconds.\n * @property {string} preview - base64 encoded short array of amplitude values 0..100.\n * @property {string} filename - file name suggestion for downloading the audio.\n * @property {integer} size - size of the recording in bytes. Treat is as an untrusted hint.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Insert audio recording into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add audio record to.\n * @param {integer} at - index where the object is inserted. The length of the record is always 1.\n * @param {AudioDesc} audioDesc - object with the audio paramenets and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertAudio = function(content, at, audioDesc) {\n content = content || {\n txt: ' '\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: 1,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'AU',\n data: {\n mime: audioDesc.mime,\n val: audioDesc.bits,\n duration: audioDesc.duration | 0,\n preview: audioDesc.preview,\n name: audioDesc.filename,\n size: audioDesc.size | 0,\n ref: audioDesc.refurl\n }\n };\n\n if (audioDesc.urlPromise) {\n ex.data._processing = true;\n audioDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._processing = undefined;\n },\n _ => {\n // Catch the error, otherwise it will appear in the console.\n ex.data._processing = undefined;\n }\n );\n }\n\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Create a (self-contained) video call Drafty document.\n * @memberof Drafty\n * @static\n * @param {boolean} audioOnly true
if the call is initially audio-only.\n * @returns Video Call drafty document.\n */\nDrafty.videoCall = function(audioOnly) {\n const content = {\n txt: ' ',\n fmt: [{\n at: 0,\n len: 1,\n key: 0\n }],\n ent: [{\n tp: 'VC',\n data: {\n aonly: audioOnly\n },\n }]\n };\n return content;\n}\n\n/**\n * Update video call (VC) entity with the new status and duration.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - VC document to update.\n * @param {object} params - new video call parameters.\n * @param {string} params.state - state of video call.\n * @param {number} params.duration - duration of the video call in milliseconds.\n *\n * @returns the same document with update applied.\n */\nDrafty.updateVideoCall = function(content, params) {\n // The video element could be just a format or a format + entity.\n // Must ensure it's the latter first.\n const fmt = ((content || {}).fmt || [])[0];\n if (!fmt) {\n // Unrecognized content.\n return content;\n }\n\n let ent;\n if (fmt.tp == 'VC') {\n // Just a format, convert to format + entity.\n delete fmt.tp;\n fmt.key = 0;\n ent = {\n tp: 'VC'\n };\n content.ent = [ent];\n } else {\n ent = (content.ent || [])[fmt.key | 0];\n if (!ent || ent.tp != 'VC') {\n // Not a VC entity.\n return content;\n }\n }\n ent.data = ent.data || {};\n Object.assign(ent.data, params);\n return content;\n}\n\n/**\n * Create a quote to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {string} header - Quote header (title, etc.).\n * @param {string} uid - UID of the author to mention.\n * @param {Drafty} body - Body of the quoted message.\n *\n * @returns Reply quote Drafty doc with the quote formatting.\n */\nDrafty.quote = function(header, uid, body) {\n const quote = Drafty.append(Drafty.appendLineBreak(Drafty.mention(header, uid)), body);\n\n // Wrap into a quote.\n quote.fmt.push({\n at: 0,\n len: quote.txt.length,\n tp: 'QQ'\n });\n\n return quote;\n}\n\n/**\n * Create a Drafty document with a mention.\n *\n * @param {string} name - mentioned name.\n * @param {string} uid - mentioned user ID.\n *\n * @returns {Drafty} document with the mention.\n */\nDrafty.mention = function(name, uid) {\n return {\n txt: name || '',\n fmt: [{\n at: 0,\n len: (name || '').length,\n key: 0\n }],\n ent: [{\n tp: 'MN',\n data: {\n val: uid\n }\n }]\n };\n}\n\n/**\n * Append a link to a Drafty document.\n *\n * @param {Drafty} content - Drafty document to append link to.\n * @param {Object} linkData - Link info in format {txt: 'ankor text', url: 'http://...'}
.\n *\n * @returns {Drafty} the same document as content
.\n */\nDrafty.appendLink = function(content, linkData) {\n content = content || {\n txt: ''\n };\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: content.txt.length,\n len: linkData.txt.length,\n key: content.ent.length\n });\n content.txt += linkData.txt;\n\n const ex = {\n tp: 'LN',\n data: {\n url: linkData.url\n }\n }\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Append image to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add image to.\n * @param {ImageDesc} imageDesc - object with image paramenets.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendImage = function(content, imageDesc) {\n content = content || {\n txt: ''\n };\n content.txt += ' ';\n return Drafty.insertImage(content, content.txt.length - 1, imageDesc);\n}\n\n/**\n * Append audio recodring to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to add recording to.\n * @param {AudioDesc} audioDesc - object with audio data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendAudio = function(content, audioDesc) {\n content = content || {\n txt: ''\n };\n content.txt += ' ';\n return Drafty.insertAudio(content, content.txt.length - 1, audioDesc);\n}\n\n/**\n * Description of a file to attach.\n * @typedef {Object} AttachmentDesc\n * @memberof Drafty\n *\n * @property {string} mime - mime-type of the attachment, e.g. \"application/octet-stream\"\n * @property {string} data - base64-encoded in-band content of small attachments. Could be null/undefined.\n * @property {string} filename - file name suggestion for downloading the attachment.\n * @property {integer} size - size of the file in bytes. Treat is as an untrusted hint.\n * @property {string} refurl - reference to the out-of-band content. Could be null/undefined.\n * @property {Promise} urlPromise - Promise which returns content URL when resolved.\n */\n\n/**\n * Attach file to Drafty content. Either as a blob or as a reference.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to attach file to.\n * @param {AttachmentDesc} object - containing attachment description and data.\n *\n * @return {Drafty} updated document.\n */\nDrafty.attachFile = function(content, attachmentDesc) {\n content = content || {\n txt: ''\n };\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: -1,\n len: 0,\n key: content.ent.length\n });\n\n const ex = {\n tp: 'EX',\n data: {\n mime: attachmentDesc.mime,\n val: attachmentDesc.data,\n name: attachmentDesc.filename,\n ref: attachmentDesc.refurl,\n size: attachmentDesc.size | 0\n }\n }\n if (attachmentDesc.urlPromise) {\n ex.data._processing = true;\n attachmentDesc.urlPromise.then(\n url => {\n ex.data.ref = url;\n ex.data._processing = undefined;\n },\n _ => {\n /* catch the error, otherwise it will appear in the console. */\n ex.data._processing = undefined;\n }\n );\n }\n content.ent.push(ex);\n\n return content;\n}\n\n/**\n * Wraps drafty document into a simple formatting style.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - document or string to wrap into a style.\n * @param {string} style - two-letter style to wrap into.\n * @param {number} at - index where the style starts, default 0.\n * @param {number} len - length of the form content, default all of it.\n *\n * @return {Drafty} updated document.\n */\nDrafty.wrapInto = function(content, style, at, len) {\n if (typeof content == 'string') {\n content = {\n txt: content\n };\n }\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at || 0,\n len: len || content.txt.length,\n tp: style,\n });\n\n return content;\n}\n\n/**\n * Wraps content into an interactive form.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - to wrap into a form.\n * @param {number} at - index where the forms starts.\n * @param {number} len - length of the form content.\n *\n * @return {Drafty} updated document.\n */\nDrafty.wrapAsForm = function(content, at, len) {\n return Drafty.wrapInto(content, 'FM', at, len);\n}\n\n/**\n * Insert clickable button into Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - Drafty document to insert button to or a string to be used as button text.\n * @param {number} at - location where the button is inserted.\n * @param {number} len - the length of the text to be used as button title.\n * @param {string} name - the button. Client should return it to the server when the button is clicked.\n * @param {string} actionType - the type of the button, one of 'url' or 'pub'.\n * @param {string} actionValue - the value to return on click:\n * @param {string} refUrl - the URL to go to when the 'url' button is clicked.\n *\n * @return {Drafty} updated document.\n */\nDrafty.insertButton = function(content, at, len, name, actionType, actionValue, refUrl) {\n if (typeof content == 'string') {\n content = {\n txt: content\n };\n }\n\n if (!content || !content.txt || content.txt.length < at + len) {\n return null;\n }\n\n if (len <= 0 || ['url', 'pub'].indexOf(actionType) == -1) {\n return null;\n }\n // Ensure refUrl is a string.\n if (actionType == 'url' && !refUrl) {\n return null;\n }\n refUrl = '' + refUrl;\n\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: at | 0,\n len: len,\n key: content.ent.length\n });\n content.ent.push({\n tp: 'BN',\n data: {\n act: actionType,\n val: actionValue,\n ref: refUrl,\n name: name\n }\n });\n\n return content;\n}\n\n/**\n * Append clickable button to Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} content - Drafty document to insert button to or a string to be used as button text.\n * @param {string} title - the text to be used as button title.\n * @param {string} name - the button. Client should return it to the server when the button is clicked.\n * @param {string} actionType - the type of the button, one of 'url' or 'pub'.\n * @param {string} actionValue - the value to return on click:\n * @param {string} refUrl - the URL to go to when the 'url' button is clicked.\n *\n * @return {Drafty} updated document.\n */\nDrafty.appendButton = function(content, title, name, actionType, actionValue, refUrl) {\n content = content || {\n txt: ''\n };\n const at = content.txt.length;\n content.txt += title;\n return Drafty.insertButton(content, at, title.length, name, actionType, actionValue, refUrl);\n}\n\n/**\n * Attach a generic JS object. The object is attached as a json string.\n * Intended for representing a form response.\n *\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - Drafty document to attach file to.\n * @param {Object} data - data to convert to json string and attach.\n * @returns {Drafty} the same document as content
.\n */\nDrafty.attachJSON = function(content, data) {\n content = content || {\n txt: ''\n };\n content.ent = content.ent || [];\n content.fmt = content.fmt || [];\n\n content.fmt.push({\n at: -1,\n len: 0,\n key: content.ent.length\n });\n\n content.ent.push({\n tp: 'EX',\n data: {\n mime: JSON_MIME_TYPE,\n val: data\n }\n });\n\n return content;\n}\n/**\n * Append line break to a Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - Drafty document to append linebreak to.\n * @returns {Drafty} the same document as content
.\n */\nDrafty.appendLineBreak = function(content) {\n content = content || {\n txt: ''\n };\n content.fmt = content.fmt || [];\n content.fmt.push({\n at: content.txt.length,\n len: 1,\n tp: 'BR'\n });\n content.txt += ' ';\n\n return content;\n}\n/**\n * Given Drafty document, convert it to HTML.\n * No attempt is made to strip pre-existing html markup.\n * This is potentially unsafe because content.txt
may contain malicious HTML\n * markup.\n * @memberof Tinode.Drafty\n * @static\n *\n * @param {Drafty} doc - document to convert.\n *\n * @returns {string} HTML-representation of content.\n */\nDrafty.UNSAFE_toHTML = function(doc) {\n const tree = draftyToTree(doc);\n const htmlFormatter = function(type, data, values) {\n const tag = DECORATORS[type];\n let result = values ? values.join('') : '';\n if (tag) {\n result = tag.open(data) + result + tag.close(data);\n }\n return result;\n };\n return treeBottomUp(tree, htmlFormatter, 0);\n}\n\n/**\n * Callback for applying custom formatting to a Drafty document.\n * Called once for each style span.\n * @memberof Drafty\n * @static\n *\n * @callback Formatter\n * @param {string} style - style code such as \"ST\" or \"IM\".\n * @param {Object} data - entity's data.\n * @param {Object} values - possibly styled subspans contained in this style span.\n * @param {number} index - index of the element guaranteed to be unique.\n */\n\n/**\n * Convert Drafty document to a representation suitable for display.\n * The context
may expose a function getFormatter(style)
. If it's available\n * it will call it to obtain a formatter
for a subtree of styles under the style
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|Object} content - Drafty document to transform.\n * @param {Formatter} formatter - callback which formats individual elements.\n * @param {Object} context - context provided to formatter as this
.\n *\n * @return {Object} transformed object\n */\nDrafty.format = function(original, formatter, context) {\n return treeBottomUp(draftyToTree(original), formatter, 0, [], context);\n}\n\n/**\n * Shorten Drafty document making the drafty text no longer than the limit.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characrets to shorten to.\n * @param {boolean} light - remove heavy data from entities.\n * @returns new shortened Drafty object leaving the original intact.\n */\nDrafty.shorten = function(original, limit, light) {\n let tree = draftyToTree(original);\n tree = shortenTree(tree, limit, '…');\n if (tree && light) {\n tree = lightEntity(tree);\n }\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Transform Drafty doc for forwarding: strip leading @mention and any leading line breaks or whitespace.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @returns converted Drafty object leaving the original intact.\n */\nDrafty.forwardedContent = function(original) {\n let tree = draftyToTree(original);\n const rmMention = function(node) {\n if (node.type == 'MN') {\n if (!node.parent || !node.parent.type) {\n return null;\n }\n }\n return node;\n }\n // Strip leading mention.\n tree = treeTopDown(tree, rmMention);\n // Remove leading whitespace.\n tree = lTrim(tree);\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Prepare Drafty doc for wrapping into QQ as a reply:\n * - Replace forwarding mention with symbol '➦' and remove data (UID).\n * - Remove quoted text completely.\n * - Replace line breaks with spaces.\n * - Strip entities of heavy content.\n * - Move attachments to the end of the document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characters to shorten to.\n * @returns converted Drafty object leaving the original intact.\n */\nDrafty.replyContent = function(original, limit) {\n const convMNnQQnBR = function(node) {\n if (node.type == 'QQ') {\n return null;\n } else if (node.type == 'MN') {\n if ((!node.parent || !node.parent.type) && (node.text || '').startsWith('➦')) {\n node.text = '➦';\n delete node.children;\n delete node.data;\n }\n } else if (node.type == 'BR') {\n node.text = ' ';\n delete node.type;\n delete node.children;\n }\n return node;\n }\n\n let tree = draftyToTree(original);\n if (!tree) {\n return original;\n }\n\n // Strip leading mention.\n tree = treeTopDown(tree, convMNnQQnBR);\n // Move attachments to the end of the doc.\n tree = attachmentsToEnd(tree, MAX_PREVIEW_ATTACHMENTS);\n // Shorten the doc.\n tree = shortenTree(tree, limit, '…');\n // Strip heavy elements except IM.data['val'] and VD.data['preview'] (have to keep them to generate previews later).\n const filter = node => {\n switch (node.type) {\n case 'IM':\n return ['val'];\n case 'VD':\n return ['preview'];\n }\n return null;\n };\n tree = lightEntity(tree, filter);\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n\n/**\n * Generate drafty preview:\n * - Shorten the document.\n * - Strip all heavy entity data leaving just inline styles and entity references.\n * - Replace line breaks with spaces.\n * - Replace content of QQ with a space.\n * - Replace forwarding mention with symbol '➦'.\n * move all attachments to the end of the document and make them visible.\n * The context
may expose a function getFormatter(style)
. If it's available\n * it will call it to obtain a formatter
for a subtree of styles under the style
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty|string} original - Drafty object to shorten.\n * @param {number} limit - length in characters to shorten to.\n * @param {boolean} forwarding - this a forwarding message preview.\n * @returns new shortened Drafty object leaving the original intact.\n */\nDrafty.preview = function(original, limit, forwarding) {\n let tree = draftyToTree(original);\n\n // Move attachments to the end.\n tree = attachmentsToEnd(tree, MAX_PREVIEW_ATTACHMENTS);\n\n // Convert leading mention to '➦' and replace QQ and BR with a space ' '.\n const convMNnQQnBR = function(node) {\n if (node.type == 'MN') {\n if ((!node.parent || !node.parent.type) && (node.text || '').startsWith('➦')) {\n node.text = '➦';\n delete node.children;\n }\n } else if (node.type == 'QQ') {\n node.text = ' ';\n delete node.children;\n } else if (node.type == 'BR') {\n node.text = ' ';\n delete node.children;\n delete node.type;\n }\n return node;\n }\n tree = treeTopDown(tree, convMNnQQnBR);\n\n tree = shortenTree(tree, limit, '…');\n if (forwarding) {\n // Keep some IM and VD data for preview.\n const filter = {\n IM: ['val'],\n VD: ['preview']\n };\n tree = lightEntity(tree, node => {\n return filter[node.type];\n });\n } else {\n tree = lightEntity(tree);\n }\n\n // Convert back to Drafty.\n return treeToDrafty({}, tree, []);\n}\n\n/**\n * Given Drafty document, convert it to plain text.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to convert to plain text.\n * @returns {string} plain-text representation of the drafty document.\n */\nDrafty.toPlainText = function(content) {\n return typeof content == 'string' ? content : content.txt;\n}\n\n/**\n * Check if the document has no markup and no entities.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - content to check for presence of markup.\n * @returns true
is content is plain text, false
otherwise.\n */\nDrafty.isPlainText = function(content) {\n return typeof content == 'string' || !(content.fmt || content.ent);\n}\n\n/**\n * Convert document to plain text with markdown. All elements which cannot\n * be represented in markdown are stripped.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to convert to plain text with markdown.\n */\nDrafty.toMarkdown = function(content) {\n let tree = draftyToTree(content);\n const mdFormatter = function(type, _, values) {\n const def = FORMAT_TAGS[type];\n let result = (values ? values.join('') : '');\n if (def) {\n if (def.isVoid) {\n result = def.md_tag || '';\n } else if (def.md_tag) {\n result = def.md_tag + result + def.md_tag;\n }\n }\n return result;\n };\n return treeBottomUp(tree, mdFormatter, 0);\n}\n\n/**\n * Checks if the object represets is a valid Drafty document.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - content to check for validity.\n * @returns true
is content is valid, false
otherwise.\n */\nDrafty.isValid = function(content) {\n if (!content) {\n return false;\n }\n\n const {\n txt,\n fmt,\n ent\n } = content;\n\n if (!txt && txt !== '' && !fmt && !ent) {\n return false;\n }\n\n const txt_type = typeof txt;\n if (txt_type != 'string' && txt_type != 'undefined' && txt !== null) {\n return false;\n }\n\n if (typeof fmt != 'undefined' && !Array.isArray(fmt) && fmt !== null) {\n return false;\n }\n\n if (typeof ent != 'undefined' && !Array.isArray(ent) && ent !== null) {\n return false;\n }\n return true;\n}\n\n/**\n * Check if the drafty document has attachments: style EX and outside of normal rendering flow,\n * i.e. at = -1
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to check for attachments.\n * @returns true
if there are attachments.\n */\nDrafty.hasAttachments = function(content) {\n if (!Array.isArray(content.fmt)) {\n return false;\n }\n for (let i in content.fmt) {\n const fmt = content.fmt[i];\n if (fmt && fmt.at < 0) {\n const ent = content.ent[fmt.key | 0];\n return ent && ent.tp == 'EX' && ent.data;\n }\n }\n return false;\n}\n\n/**\n * Callback for enumerating entities in a Drafty document.\n * Called once for each entity.\n * @memberof Drafty\n * @static\n *\n * @callback EntityCallback\n * @param {Object} data entity data.\n * @param {string} entity type.\n * @param {number} index entity's index in `content.ent`.\n *\n * @return 'true-ish' to stop processing, 'false-ish' otherwise.\n */\n\n/**\n * Enumerate attachments: style EX and outside of normal rendering flow, i.e. at = -1
.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to process for attachments.\n * @param {EntityCallback} callback - callback to call for each attachment.\n * @param {Object} context - value of \"this\" for callback.\n */\nDrafty.attachments = function(content, callback, context) {\n if (!Array.isArray(content.fmt)) {\n return;\n }\n let count = 0;\n for (let i in content.fmt) {\n let fmt = content.fmt[i];\n if (fmt && fmt.at < 0) {\n const ent = content.ent[fmt.key | 0];\n if (ent && ent.tp == 'EX' && ent.data) {\n if (callback.call(context, ent.data, count++, 'EX')) {\n break;\n }\n }\n }\n };\n}\n\n/**\n * Check if the drafty document has entities.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document to check for entities.\n * @returns true
if there are entities.\n */\nDrafty.hasEntities = function(content) {\n return content.ent && content.ent.length > 0;\n}\n\n/**\n * Enumerate entities. Enumeration stops if callback returns 'true'.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with entities to enumerate.\n * @param {EntityCallback} callback - callback to call for each entity.\n * @param {Object} context - value of \"this\" for callback.\n *\n */\nDrafty.entities = function(content, callback, context) {\n if (content.ent && content.ent.length > 0) {\n for (let i in content.ent) {\n if (content.ent[i]) {\n if (callback.call(context, content.ent[i].data, i, content.ent[i].tp)) {\n break;\n }\n }\n }\n }\n}\n\n/**\n * Callback for enumerating styles (inline formats) in a Drafty document.\n * Called once for each style.\n * @memberof Drafty\n * @static\n *\n * @callback StyleCallback\n * @param {string} tp - format type.\n * @param {number} at - starting position of the format in text.\n * @param {number} len - extent of the format in characters.\n * @param {number} key - index of the entity if format is a reference.\n * @param {number} index - style's index in `content.fmt`.\n *\n * @return 'true-ish' to stop processing, 'false-ish' otherwise.\n */\n\n/**\n * Enumerate styles (inline formats). Enumeration stops if callback returns 'true'.\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with styles (formats) to enumerate.\n * @param {StyleCallback} callback - callback to call for each format.\n * @param {Object} context - value of \"this\" for callback.\n */\nDrafty.styles = function(content, callback, context) {\n if (content.fmt && content.fmt.length > 0) {\n for (let i in content.fmt) {\n const fmt = content.fmt[i];\n if (fmt) {\n if (callback.call(context, fmt.tp, fmt.at, fmt.len, fmt.key, i)) {\n break;\n }\n }\n }\n }\n}\n\n/**\n * Remove unrecognized fields from entity data\n * @memberof Drafty\n * @static\n *\n * @param {Drafty} content - document with entities to enumerate.\n * @returns content.\n */\nDrafty.sanitizeEntities = function(content) {\n if (content && content.ent && content.ent.length > 0) {\n for (let i in content.ent) {\n const ent = content.ent[i];\n if (ent && ent.data) {\n const data = copyEntData(ent.data);\n if (data) {\n content.ent[i].data = data;\n } else {\n delete content.ent[i].data;\n }\n }\n }\n }\n return content;\n}\n\n/**\n * Given the entity, get URL which can be used for downloading\n * entity data.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the URl from.\n * @returns {string} URL to download entity data or null
.\n */\nDrafty.getDownloadUrl = function(entData) {\n let url = null;\n if (entData.mime != JSON_MIME_TYPE && entData.val) {\n url = base64toObjectUrl(entData.val, entData.mime, Drafty.logger);\n } else if (typeof entData.ref == 'string') {\n url = entData.ref;\n }\n return url;\n}\n\n/**\n * Check if the entity data is not ready for sending, such as being uploaded to the server.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entity.data to get the URl from.\n * @returns {boolean} true if upload is in progress, false otherwise.\n */\nDrafty.isProcessing = function(entData) {\n return !!entData._processing;\n}\n\n/**\n * Given the entity, get URL which can be used for previewing\n * the entity.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entity.data to get the URl from.\n *\n * @returns {string} url for previewing or null if no such url is available.\n */\nDrafty.getPreviewUrl = function(entData) {\n return entData.val ? base64toObjectUrl(entData.val, entData.mime, Drafty.logger) : null;\n}\n\n/**\n * Get approximate size of the entity.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the size for.\n * @returns {number} size of entity data in bytes.\n */\nDrafty.getEntitySize = function(entData) {\n // Either size hint or length of value. The value is base64 encoded,\n // the actual object size is smaller than the encoded length.\n return entData.size ? entData.size : entData.val ? (entData.val.length * 0.75) | 0 : 0;\n}\n\n/**\n * Get entity mime type.\n * @memberof Drafty\n * @static\n *\n * @param {Object} entData - entity.data to get the type for.\n * @returns {string} mime type of entity.\n */\nDrafty.getEntityMimeType = function(entData) {\n return entData.mime || 'text/plain';\n}\n\n/**\n * Get HTML tag for a given two-letter style name.\n * @memberof Drafty\n * @static\n *\n * @param {string} style - two-letter style, like ST or LN.\n *\n * @returns {string} HTML tag name if style is found, {code: undefined} if style is falsish or not found.\n */\nDrafty.tagName = function(style) {\n return FORMAT_TAGS[style] && FORMAT_TAGS[style].html_tag;\n}\n\n/**\n * For a given data bundle generate an object with HTML attributes,\n * for instance, given {url: \"http://www.example.com/\"} return\n * {href: \"http://www.example.com/\"}\n * @memberof Drafty\n * @static\n *\n * @param {string} style - two-letter style to generate attributes for.\n * @param {Object} data - data bundle to convert to attributes\n *\n * @returns {Object} object with HTML attributes.\n */\nDrafty.attrValue = function(style, data) {\n if (data && DECORATORS[style] && DECORATORS[style].props) {\n return DECORATORS[style].props(data);\n }\n\n return undefined;\n}\n\n/**\n * Drafty MIME type.\n * @memberof Drafty\n * @static\n *\n * @returns {string} content-Type \"text/x-drafty\".\n */\nDrafty.getContentType = function() {\n return DRAFTY_MIME_TYPE;\n}\n\n// =================\n// Utility methods.\n// =================\n\n// Take a string and defined earlier style spans, re-compose them into a tree where each leaf is\n// a same-style (including unstyled) string. I.e. 'hello *bold _italic_* and ~more~ world' ->\n// ('hello ', (b: 'bold ', (i: 'italic')), ' and ', (s: 'more'), ' world');\n//\n// This is needed in order to clear markup, i.e. 'hello *world*' -> 'hello world' and convert\n// ranges from markup-ed offsets to plain text offsets.\nfunction chunkify(line, start, end, spans) {\n const chunks = [];\n\n if (spans.length == 0) {\n return [];\n }\n\n for (let i in spans) {\n // Get the next chunk from the queue\n const span = spans[i];\n\n // Grab the initial unstyled chunk\n if (span.at > start) {\n chunks.push({\n txt: line.slice(start, span.at)\n });\n }\n\n // Grab the styled chunk. It may include subchunks.\n const chunk = {\n tp: span.tp\n };\n const chld = chunkify(line, span.at + 1, span.end, span.children);\n if (chld.length > 0) {\n chunk.children = chld;\n } else {\n chunk.txt = span.txt;\n }\n chunks.push(chunk);\n start = span.end + 1; // '+1' is to skip the formatting character\n }\n\n // Grab the remaining unstyled chunk, after the last span\n if (start < end) {\n chunks.push({\n txt: line.slice(start, end)\n });\n }\n\n return chunks;\n}\n\n// Detect starts and ends of formatting spans. Unformatted spans are\n// ignored at this stage.\nfunction spannify(original, re_start, re_end, type) {\n const result = [];\n let index = 0;\n let line = original.slice(0); // make a copy;\n\n while (line.length > 0) {\n // match[0]; // match, like '*abc*'\n // match[1]; // match captured in parenthesis, like 'abc'\n // match['index']; // offset where the match started.\n\n // Find the opening token.\n const start = re_start.exec(line);\n if (start == null) {\n break;\n }\n\n // Because javascript RegExp does not support lookbehind, the actual offset may not point\n // at the markup character. Find it in the matched string.\n let start_offset = start['index'] + start[0].lastIndexOf(start[1]);\n // Clip the processed part of the string.\n line = line.slice(start_offset + 1);\n // start_offset is an offset within the clipped string. Convert to original index.\n start_offset += index;\n // Index now point to the beginning of 'line' within the 'original' string.\n index = start_offset + 1;\n\n // Find the matching closing token.\n const end = re_end ? re_end.exec(line) : null;\n if (end == null) {\n break;\n }\n let end_offset = end['index'] + end[0].indexOf(end[1]);\n // Clip the processed part of the string.\n line = line.slice(end_offset + 1);\n // Update offsets\n end_offset += index;\n // Index now points to the beginning of 'line' within the 'original' string.\n index = end_offset + 1;\n\n result.push({\n txt: original.slice(start_offset + 1, end_offset),\n children: [],\n at: start_offset,\n end: end_offset,\n tp: type\n });\n }\n\n return result;\n}\n\n// Convert linear array or spans into a tree representation.\n// Keep standalone and nested spans, throw away partially overlapping spans.\nfunction toSpanTree(spans) {\n if (spans.length == 0) {\n return [];\n }\n\n const tree = [spans[0]];\n let last = spans[0];\n for (let i = 1; i < spans.length; i++) {\n // Keep spans which start after the end of the previous span or those which\n // are complete within the previous span.\n if (spans[i].at > last.end) {\n // Span is completely outside of the previous span.\n tree.push(spans[i]);\n last = spans[i];\n } else if (spans[i].end <= last.end) {\n // Span is fully inside of the previous span. Push to subnode.\n last.children.push(spans[i]);\n }\n // Span could partially overlap, ignoring it as invalid.\n }\n\n // Recursively rearrange the subnodes.\n for (let i in tree) {\n tree[i].children = toSpanTree(tree[i].children);\n }\n\n return tree;\n}\n\n// Convert drafty document to a tree.\nfunction draftyToTree(doc) {\n if (!doc) {\n return null;\n }\n\n doc = (typeof doc == 'string') ? {\n txt: doc\n } : doc;\n let {\n txt,\n fmt,\n ent\n } = doc;\n\n txt = txt || '';\n if (!Array.isArray(ent)) {\n ent = [];\n }\n\n if (!Array.isArray(fmt) || fmt.length == 0) {\n if (ent.length == 0) {\n return {\n text: txt\n };\n }\n\n // Handle special case when all values in fmt are 0 and fmt therefore is skipped.\n fmt = [{\n at: 0,\n len: 0,\n key: 0\n }];\n }\n\n // Sanitize spans.\n const spans = [];\n const attachments = [];\n fmt.forEach((span) => {\n if (!span || typeof span != 'object') {\n return;\n }\n\n if (!['undefined', 'number'].includes(typeof span.at)) {\n // Present, but non-numeric 'at'.\n return;\n }\n if (!['undefined', 'number'].includes(typeof span.len)) {\n // Present, but non-numeric 'len'.\n return;\n }\n let at = span.at | 0;\n let len = span.len | 0;\n if (len < 0) {\n // Invalid span length.\n return;\n }\n\n let key = span.key || 0;\n if (ent.length > 0 && (typeof key != 'number' || key < 0 || key >= ent.length)) {\n // Invalid key value.\n return;\n }\n\n if (at <= -1) {\n // Attachment. Store attachments separately.\n attachments.push({\n start: -1,\n end: 0,\n key: key\n });\n return;\n } else if (at + len > txt.length) {\n // Span is out of bounds.\n return;\n }\n\n if (!span.tp) {\n if (ent.length > 0 && (typeof ent[key] == 'object')) {\n spans.push({\n start: at,\n end: at + len,\n key: key\n });\n }\n } else {\n spans.push({\n type: span.tp,\n start: at,\n end: at + len\n });\n }\n });\n\n // Sort spans first by start index (asc) then by length (desc), then by weight.\n spans.sort((a, b) => {\n let diff = a.start - b.start;\n if (diff != 0) {\n return diff;\n }\n diff = b.end - a.end;\n if (diff != 0) {\n return diff;\n }\n return FMT_WEIGHT.indexOf(b.type) - FMT_WEIGHT.indexOf(a.type);\n });\n\n // Move attachments to the end of the list.\n if (attachments.length > 0) {\n spans.push(...attachments);\n }\n\n spans.forEach((span) => {\n if (ent.length > 0 && !span.type && ent[span.key] && typeof ent[span.key] == 'object') {\n span.type = ent[span.key].tp;\n span.data = ent[span.key].data;\n }\n\n // Is type still undefined? Hide the invalid element!\n if (!span.type) {\n span.type = 'HD';\n }\n });\n\n let tree = spansToTree({}, txt, 0, txt.length, spans);\n\n // Flatten tree nodes.\n const flatten = function(node) {\n if (Array.isArray(node.children) && node.children.length == 1) {\n // Unwrap.\n const child = node.children[0];\n if (!node.type) {\n const parent = node.parent;\n node = child;\n node.parent = parent;\n } else if (!child.type && !child.children) {\n node.text = child.text;\n delete node.children;\n }\n }\n return node;\n }\n tree = treeTopDown(tree, flatten);\n\n return tree;\n}\n\n// Add tree node to a parent tree.\nfunction addNode(parent, n) {\n if (!n) {\n return parent;\n }\n\n if (!parent.children) {\n parent.children = [];\n }\n\n // If text is present, move it to a subnode.\n if (parent.text) {\n parent.children.push({\n text: parent.text,\n parent: parent\n });\n delete parent.text;\n }\n\n n.parent = parent;\n parent.children.push(n);\n\n return parent;\n}\n\n// Returns a tree of nodes.\nfunction spansToTree(parent, text, start, end, spans) {\n if (!spans || spans.length == 0) {\n if (start < end) {\n addNode(parent, {\n text: text.substring(start, end)\n });\n }\n return parent;\n }\n\n // Process subspans.\n for (let i = 0; i < spans.length; i++) {\n const span = spans[i];\n if (span.start < 0 && span.type == 'EX') {\n addNode(parent, {\n type: span.type,\n data: span.data,\n key: span.key,\n att: true\n });\n continue;\n }\n\n // Add un-styled range before the styled span starts.\n if (start < span.start) {\n addNode(parent, {\n text: text.substring(start, span.start)\n });\n start = span.start;\n }\n\n // Get all spans which are within the current span.\n const subspans = [];\n while (i < spans.length - 1) {\n const inner = spans[i + 1];\n if (inner.start < 0) {\n // Attachments are in the end. Stop.\n break;\n } else if (inner.start < span.end) {\n if (inner.end <= span.end) {\n const tag = FORMAT_TAGS[inner.tp] || {};\n if (inner.start < inner.end || tag.isVoid) {\n // Valid subspan: completely within the current span and\n // either non-zero length or zero length is acceptable.\n subspans.push(inner);\n }\n }\n i++;\n // Overlapping subspans are ignored.\n } else {\n // Past the end of the current span. Stop.\n break;\n }\n }\n\n addNode(parent, spansToTree({\n type: span.type,\n data: span.data,\n key: span.key\n }, text, start, span.end, subspans));\n start = span.end;\n }\n\n // Add the last unformatted range.\n if (start < end) {\n addNode(parent, {\n text: text.substring(start, end)\n });\n }\n\n return parent;\n}\n\n// Append a tree to a Drafty doc.\nfunction treeToDrafty(doc, tree, keymap) {\n if (!tree) {\n return doc;\n }\n\n doc.txt = doc.txt || '';\n\n // Checkpoint to measure length of the current tree node.\n const start = doc.txt.length;\n\n if (tree.text) {\n doc.txt += tree.text;\n } else if (Array.isArray(tree.children)) {\n tree.children.forEach((c) => {\n treeToDrafty(doc, c, keymap);\n });\n }\n\n if (tree.type) {\n const len = doc.txt.length - start;\n doc.fmt = doc.fmt || [];\n if (Object.keys(tree.data || {}).length > 0) {\n doc.ent = doc.ent || [];\n const newKey = (typeof keymap[tree.key] == 'undefined') ? doc.ent.length : keymap[tree.key];\n keymap[tree.key] = newKey;\n doc.ent[newKey] = {\n tp: tree.type,\n data: tree.data\n };\n if (tree.att) {\n // Attachment.\n doc.fmt.push({\n at: -1,\n len: 0,\n key: newKey\n });\n } else {\n doc.fmt.push({\n at: start,\n len: len,\n key: newKey\n });\n }\n } else {\n doc.fmt.push({\n tp: tree.type,\n at: start,\n len: len\n });\n }\n }\n return doc;\n}\n\n// Traverse the tree top down transforming the nodes: apply transformer to every tree node.\nfunction treeTopDown(src, transformer, context) {\n if (!src) {\n return null;\n }\n\n let dst = transformer.call(context, src);\n if (!dst || !dst.children) {\n return dst;\n }\n\n const children = [];\n for (let i in dst.children) {\n let n = dst.children[i];\n if (n) {\n n = treeTopDown(n, transformer, context);\n if (n) {\n children.push(n);\n }\n }\n }\n\n if (children.length == 0) {\n dst.children = null;\n } else {\n dst.children = children;\n }\n\n return dst;\n}\n\n// Traverse the tree bottom-up: apply formatter to every node.\n// The formatter must maintain its state through context.\nfunction treeBottomUp(src, formatter, index, stack, context) {\n if (!src) {\n return null;\n }\n\n if (stack && src.type) {\n stack.push(src.type);\n }\n\n let values = [];\n for (let i in src.children) {\n const n = treeBottomUp(src.children[i], formatter, i, stack, context);\n if (n) {\n values.push(n);\n }\n }\n if (values.length == 0) {\n if (src.text) {\n values = [src.text];\n } else {\n values = null;\n }\n }\n\n if (stack && src.type) {\n stack.pop();\n }\n\n return formatter.call(context, src.type, src.data, values, index, stack);\n}\n\n// Clip tree to the provided limit.\nfunction shortenTree(tree, limit, tail) {\n if (!tree) {\n return null;\n }\n\n if (tail) {\n limit -= tail.length;\n }\n\n const shortener = function(node) {\n if (limit <= -1) {\n // Limit -1 means the doc was already clipped.\n return null;\n }\n\n if (node.att) {\n // Attachments are unchanged.\n return node;\n }\n if (limit == 0) {\n node.text = tail;\n limit = -1;\n } else if (node.text) {\n const len = node.text.length;\n if (len > limit) {\n node.text = node.text.substring(0, limit) + tail;\n limit = -1;\n } else {\n limit -= len;\n }\n }\n return node;\n }\n\n return treeTopDown(tree, shortener);\n}\n\n// Strip heavy entities from a tree.\nfunction lightEntity(tree, allow) {\n const lightCopy = node => {\n const data = copyEntData(node.data, true, allow ? allow(node) : null);\n if (data) {\n node.data = data;\n } else {\n delete node.data;\n }\n return node;\n }\n return treeTopDown(tree, lightCopy);\n}\n\n// Remove spaces and breaks on the left.\nfunction lTrim(tree) {\n if (tree.type == 'BR') {\n tree = null;\n } else if (tree.text) {\n if (!tree.type) {\n tree.text = tree.text.trimStart();\n if (!tree.text) {\n tree = null;\n }\n }\n } else if (!tree.type && tree.children && tree.children.length > 0) {\n const c = lTrim(tree.children[0]);\n if (c) {\n tree.children[0] = c;\n } else {\n tree.children.shift();\n if (!tree.type && tree.children.length == 0) {\n tree = null;\n }\n }\n }\n return tree;\n}\n\n// Move attachments to the end. Attachments must be at the top level, no need to traverse the tree.\nfunction attachmentsToEnd(tree, limit) {\n if (!tree) {\n return null;\n }\n\n if (tree.att) {\n tree.text = ' ';\n delete tree.att;\n delete tree.children;\n } else if (tree.children) {\n const attachments = [];\n const children = [];\n for (let i in tree.children) {\n const c = tree.children[i];\n if (c.att) {\n if (attachments.length == limit) {\n // Too many attachments to preview;\n continue;\n }\n if (c.data['mime'] == JSON_MIME_TYPE) {\n // JSON attachments are not shown in preview.\n continue;\n }\n\n delete c.att;\n delete c.children;\n c.text = ' ';\n attachments.push(c);\n } else {\n children.push(c);\n }\n }\n tree.children = children.concat(attachments);\n }\n return tree;\n}\n\n// Get a list of entities from a text.\nfunction extractEntities(line) {\n let match;\n let extracted = [];\n ENTITY_TYPES.forEach((entity) => {\n while ((match = entity.re.exec(line)) !== null) {\n extracted.push({\n offset: match['index'],\n len: match[0].length,\n unique: match[0],\n data: entity.pack(match[0]),\n type: entity.name\n });\n }\n });\n\n if (extracted.length == 0) {\n return extracted;\n }\n\n // Remove entities detected inside other entities, like #hashtag in a URL.\n extracted.sort((a, b) => {\n return a.offset - b.offset;\n });\n\n let idx = -1;\n extracted = extracted.filter((el) => {\n const result = (el.offset > idx);\n idx = el.offset + el.len;\n return result;\n });\n\n return extracted;\n}\n\n// Convert the chunks into format suitable for serialization.\nfunction draftify(chunks, startAt) {\n let plain = '';\n let ranges = [];\n for (let i in chunks) {\n const chunk = chunks[i];\n if (!chunk.txt) {\n const drafty = draftify(chunk.children, plain.length + startAt);\n chunk.txt = drafty.txt;\n ranges = ranges.concat(drafty.fmt);\n }\n\n if (chunk.tp) {\n ranges.push({\n at: plain.length + startAt,\n len: chunk.txt.length,\n tp: chunk.tp\n });\n }\n\n plain += chunk.txt;\n }\n return {\n txt: plain,\n fmt: ranges\n };\n}\n\n// Create a copy of entity data with (light=false) or without (light=true) the large payload.\n// The array 'allow' contains a list of fields exempt from stripping.\nfunction copyEntData(data, light, allow) {\n if (data && Object.entries(data).length > 0) {\n allow = allow || [];\n const dc = {};\n ALLOWED_ENT_FIELDS.forEach(key => {\n if (data[key]) {\n if (light && !allow.includes(key) &&\n (typeof data[key] == 'string' || Array.isArray(data[key])) &&\n data[key].length > MAX_PREVIEW_DATA_SIZE) {\n return;\n }\n if (typeof data[key] == 'object') {\n return;\n }\n dc[key] = data[key];\n }\n });\n\n if (Object.entries(dc).length != 0) {\n return dc;\n }\n }\n return null;\n}\n\nif (typeof module != 'undefined') {\n module.exports = Drafty;\n}\n","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","/**\n * @file Access control model.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\n/**\n * Helper class for handling access mode.\n *\n * @class AccessMode\n * @memberof Tinode\n *\n * @param {AccessMode|Object=} acs - AccessMode to copy or access mode object received from the server.\n */\nexport default class AccessMode {\n constructor(acs) {\n if (acs) {\n this.given = typeof acs.given == 'number' ? acs.given : AccessMode.decode(acs.given);\n this.want = typeof acs.want == 'number' ? acs.want : AccessMode.decode(acs.want);\n this.mode = acs.mode ? (typeof acs.mode == 'number' ? acs.mode : AccessMode.decode(acs.mode)) :\n (this.given & this.want);\n }\n }\n\n static #checkFlag(val, side, flag) {\n side = side || 'mode';\n if (['given', 'want', 'mode'].includes(side)) {\n return ((val[side] & flag) != 0);\n }\n throw new Error(`Invalid AccessMode component '${side}'`);\n }\n /**\n * Parse string into an access mode value.\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {string | Number} mode - either a String representation of the access mode to parse or a set of bits to assign.\n * @returns {number} - Access mode as a numeric value.\n */\n static decode(str) {\n if (!str) {\n return null;\n } else if (typeof str == 'number') {\n return str & AccessMode._BITMASK;\n } else if (str === 'N' || str === 'n') {\n return AccessMode._NONE;\n }\n\n const bitmask = {\n 'J': AccessMode._JOIN,\n 'R': AccessMode._READ,\n 'W': AccessMode._WRITE,\n 'P': AccessMode._PRES,\n 'A': AccessMode._APPROVE,\n 'S': AccessMode._SHARE,\n 'D': AccessMode._DELETE,\n 'O': AccessMode._OWNER\n };\n\n let m0 = AccessMode._NONE;\n\n for (let i = 0; i < str.length; i++) {\n const bit = bitmask[str.charAt(i).toUpperCase()];\n if (!bit) {\n // Unrecognized bit, skip.\n continue;\n }\n m0 |= bit;\n }\n return m0;\n }\n /**\n * Convert numeric representation of the access mode into a string.\n *\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {number} val - access mode value to convert to a string.\n * @returns {string} - Access mode as a string.\n */\n static encode(val) {\n if (val === null || val === AccessMode._INVALID) {\n return null;\n } else if (val === AccessMode._NONE) {\n return 'N';\n }\n\n const bitmask = ['J', 'R', 'W', 'P', 'A', 'S', 'D', 'O'];\n let res = '';\n for (let i = 0; i < bitmask.length; i++) {\n if ((val & (1 << i)) != 0) {\n res = res + bitmask[i];\n }\n }\n return res;\n }\n /**\n * Update numeric representation of access mode with the new value. The value\n * is one of the following:\n * - a string starting with '+'
or '-'
then the bits to add or remove, e.g. '+R-W'
or '-PS'
.\n * - a new value of access mode\n *\n * @memberof Tinode.AccessMode\n * @static\n *\n * @param {number} val - access mode value to update.\n * @param {string} upd - update to apply to val.\n * @returns {number} - updated access mode.\n */\n static update(val, upd) {\n if (!upd || typeof upd != 'string') {\n return val;\n }\n\n let action = upd.charAt(0);\n if (action == '+' || action == '-') {\n let val0 = val;\n // Split delta-string like '+ABC-DEF+Z' into an array of parts including + and -.\n const parts = upd.split(/([-+])/);\n // Starting iteration from 1 because String.split() creates an array with the first empty element.\n // Iterating by 2 because we parse pairs +/- then data.\n for (let i = 1; i < parts.length - 1; i += 2) {\n action = parts[i];\n const m0 = AccessMode.decode(parts[i + 1]);\n if (m0 == AccessMode._INVALID) {\n return val;\n }\n if (m0 == null) {\n continue;\n }\n if (action === '+') {\n val0 |= m0;\n } else if (action === '-') {\n val0 &= ~m0;\n }\n }\n val = val0;\n } else {\n // The string is an explicit new value 'ABC' rather than delta.\n const val0 = AccessMode.decode(upd);\n if (val0 != AccessMode._INVALID) {\n val = val0;\n }\n }\n\n return val;\n }\n /**\n * Bits present in a1 but missing in a2.\n *\n * @static\n * @memberof Tinode\n *\n * @param {number | string} a1 - access mode to subtract from.\n * @param {number | string} a2 - access mode to subtract.\n * @returns {number} access mode with bits present in a1
but missing in a2
.\n */\n static diff(a1, a2) {\n a1 = AccessMode.decode(a1);\n a2 = AccessMode.decode(a2);\n\n if (a1 == AccessMode._INVALID || a2 == AccessMode._INVALID) {\n return AccessMode._INVALID;\n }\n return a1 & ~a2;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Custom formatter\n */\n toString() {\n return '{\"mode\": \"' + AccessMode.encode(this.mode) +\n '\", \"given\": \"' + AccessMode.encode(this.given) +\n '\", \"want\": \"' + AccessMode.encode(this.want) + '\"}';\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Converts numeric values to strings.\n */\n jsonHelper() {\n return {\n mode: AccessMode.encode(this.mode),\n given: AccessMode.encode(this.given),\n want: AccessMode.encode(this.want)\n };\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign value to 'mode'.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} m - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setMode(m) {\n this.mode = AccessMode.decode(m);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update mode
value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateMode(u) {\n this.mode = AccessMode.update(this.mode, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get mode
value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - mode
value.\n */\n getMode() {\n return AccessMode.encode(this.mode);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign given
value.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} g - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setGiven(g) {\n this.given = AccessMode.decode(g);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'given' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateGiven(u) {\n this.given = AccessMode.update(this.given, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get 'given' value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - given value.\n */\n getGiven() {\n return AccessMode.encode(this.given);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Assign 'want' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string | Number} w - either a string representation of the access mode or a set of bits.\n * @returns {AccessMode} - this
AccessMode.\n */\n setWant(w) {\n this.want = AccessMode.decode(w);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'want' value.\n * @memberof Tinode.AccessMode\n *\n * @param {string} u - string representation of the changes to apply to access mode.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateWant(u) {\n this.want = AccessMode.update(this.want, u);\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get 'want' value as a string.\n * @memberof Tinode.AccessMode\n *\n * @returns {string} - want value.\n */\n getWant() {\n return AccessMode.encode(this.want);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get permissions present in 'want' but missing in 'given'.\n * Inverse of {@link Tinode.AccessMode#getExcessive}\n *\n * @memberof Tinode.AccessMode\n *\n * @returns {string} permissions present in want but missing in given.\n */\n getMissing() {\n return AccessMode.encode(this.want & ~this.given);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Get permissions present in 'given' but missing in 'want'.\n * Inverse of {@link Tinode.AccessMode#getMissing}\n * @memberof Tinode.AccessMode\n *\n * @returns {string} permissions present in given but missing in want.\n */\n getExcessive() {\n return AccessMode.encode(this.given & ~this.want);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Update 'want', 'give', and 'mode' values.\n * @memberof Tinode.AccessMode\n *\n * @param {AccessMode} val - new access mode value.\n * @returns {AccessMode} - this
AccessMode.\n */\n updateAll(val) {\n if (val) {\n this.updateGiven(val.given);\n this.updateWant(val.want);\n this.mode = this.given & this.want;\n }\n return this;\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Owner (O) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isOwner(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._OWNER);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Presence (P) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isPresencer(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._PRES);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Presence (P) flag is NOT set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isMuted(side) {\n return !this.isPresencer(side);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Join (J) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isJoiner(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._JOIN);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Reader (R) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isReader(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._READ);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Writer (W) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isWriter(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._WRITE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Approver (A) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isApprover(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._APPROVE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if either one of Owner (O) or Approver (A) flags is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isAdmin(side) {\n return this.isOwner(side) || this.isApprover(side);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if either one of Owner (O), Approver (A), or Sharer (S) flags is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isSharer(side) {\n return this.isAdmin(side) || AccessMode.#checkFlag(this, side, AccessMode._SHARE);\n }\n /**\n * AccessMode is a class representing topic access mode.\n *\n * @memberof Tinode\n * @class AccessMode\n */\n /**\n * Check if Deleter (D) flag is set.\n * @memberof Tinode.AccessMode\n * @param {string=} side - which permission to check: given, want, mode; default: mode.\n * @returns {boolean} - true
if flag is set.\n */\n isDeleter(side) {\n return AccessMode.#checkFlag(this, side, AccessMode._DELETE);\n }\n}\n\nAccessMode._NONE = 0x00;\nAccessMode._JOIN = 0x01;\nAccessMode._READ = 0x02;\nAccessMode._WRITE = 0x04;\nAccessMode._PRES = 0x08;\nAccessMode._APPROVE = 0x10;\nAccessMode._SHARE = 0x20;\nAccessMode._DELETE = 0x40;\nAccessMode._OWNER = 0x80;\n\nAccessMode._BITMASK = AccessMode._JOIN | AccessMode._READ | AccessMode._WRITE | AccessMode._PRES |\n AccessMode._APPROVE | AccessMode._SHARE | AccessMode._DELETE | AccessMode._OWNER;\nAccessMode._INVALID = 0x100000;\n","export const PACKAGE_VERSION = \"0.22.12\";\n","/**\n * @file Global constants and configuration parameters.\n *\n * @copyright 2015-2023 Tinode LLC\n */\n'use strict';\n\nimport {\n PACKAGE_VERSION\n} from '../version.js';\n\n// Global constants\nexport const PROTOCOL_VERSION = '0'; // Major component of the version, e.g. '0' in '0.17.1'.\nexport const VERSION = PACKAGE_VERSION || '0.21';\nexport const LIBRARY = 'tinodejs/' + VERSION;\n\n// Topic name prefixes.\nexport const TOPIC_NEW = 'new';\nexport const TOPIC_NEW_CHAN = 'nch';\nexport const TOPIC_ME = 'me';\nexport const TOPIC_FND = 'fnd';\nexport const TOPIC_SYS = 'sys';\nexport const TOPIC_CHAN = 'chn';\nexport const TOPIC_GRP = 'grp';\nexport const TOPIC_P2P = 'p2p';\nexport const USER_NEW = 'new';\n\n// Starting value of a locally-generated seqId used for pending messages.\nexport const LOCAL_SEQID = 0xFFFFFFF;\n\n// Status codes.\nexport const MESSAGE_STATUS_NONE = 0; // Status not assigned.\nexport const MESSAGE_STATUS_QUEUED = 10; // Local ID assigned, in progress to be sent.\nexport const MESSAGE_STATUS_SENDING = 20; // Transmission started.\nexport const MESSAGE_STATUS_FAILED = 30; // At least one attempt was made to send the message.\nexport const MESSAGE_STATUS_FATAL = 40; // Message sending failed and it should not be retried.\nexport const MESSAGE_STATUS_SENT = 50; // Delivered to the server.\nexport const MESSAGE_STATUS_RECEIVED = 60; // Received by the client.\nexport const MESSAGE_STATUS_READ = 70; // Read by the user.\nexport const MESSAGE_STATUS_TO_ME = 80; // The message is received from another user.\n\n// Reject unresolved futures after this many milliseconds.\nexport const EXPIRE_PROMISES_TIMEOUT = 5_000;\n// Periodicity of garbage collection of unresolved futures.\nexport const EXPIRE_PROMISES_PERIOD = 1_000;\n\n// Delay before acknowledging that a message was recived.\nexport const RECV_TIMEOUT = 100;\n\n// Default number of messages to pull into memory from persistent cache.\nexport const DEFAULT_MESSAGES_PAGE = 24;\n\n// Unicode DEL character indicating data was deleted.\nexport const DEL_CHAR = '\\u2421';\n","/**\n * @file Throwable error with numeric error code.\n *\n * @copyright 2015-2023 Tinode LLC.\n */\n'use strict';\n\nexport default class CommError extends Error {\n constructor(message, code) {\n super(`${message} (${code})`);\n this.name = 'CommError';\n this.code = code;\n }\n}\n","/**\n * @file Utilities used in multiple places.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport AccessMode from './access-mode.js';\nimport {\n DEL_CHAR\n} from './config.js';\n\n// Attempt to convert date and AccessMode strings to objects.\nexport function jsonParseHelper(key, val) {\n // Try to convert string timestamps with optional milliseconds to Date,\n // e.g. 2015-09-02T01:45:43[.123]Z\n if (typeof val == 'string' && val.length >= 20 && val.length <= 24 && ['ts', 'touched', 'updated', 'created', 'when', 'deleted', 'expires'].includes(key)) {\n const date = new Date(val);\n if (!isNaN(date)) {\n return date;\n }\n } else if (key === 'acs' && typeof val === 'object') {\n return new AccessMode(val);\n }\n return val;\n}\n\n// Checks if URL is a relative url, i.e. has no 'scheme://', including the case of missing scheme '//'.\n// The scheme is expected to be RFC-compliant, e.g. [a-z][a-z0-9+.-]*\n// example.html - ok\n// https:example.com - not ok.\n// http:/example.com - not ok.\n// ' ↲ https://example.com' - not ok. (↲ means carriage return)\nexport function isUrlRelative(url) {\n return url && !/^\\s*([a-z][a-z0-9+.-]*:|\\/\\/)/im.test(url);\n}\n\nfunction isValidDate(d) {\n return (d instanceof Date) && !isNaN(d) && (d.getTime() != 0);\n}\n\n// RFC3339 formater of Date\nexport function rfc3339DateString(d) {\n if (!isValidDate(d)) {\n return undefined;\n }\n\n const pad = function(val, sp) {\n sp = sp || 2;\n return '0'.repeat(sp - ('' + val).length) + val;\n };\n\n const millis = d.getUTCMilliseconds();\n return d.getUTCFullYear() + '-' + pad(d.getUTCMonth() + 1) + '-' + pad(d.getUTCDate()) +\n 'T' + pad(d.getUTCHours()) + ':' + pad(d.getUTCMinutes()) + ':' + pad(d.getUTCSeconds()) +\n (millis ? '.' + pad(millis, 3) : '') + 'Z';\n}\n\n// Recursively merge src's own properties to dst.\n// Ignore properties where ignore[property] is true.\n// Array and Date objects are shallow-copied.\nexport function mergeObj(dst, src, ignore) {\n if (typeof src != 'object') {\n if (src === undefined) {\n return dst;\n }\n if (src === DEL_CHAR) {\n return undefined;\n }\n return src;\n }\n // JS is crazy: typeof null is 'object'.\n if (src === null) {\n return src;\n }\n\n // Handle Date\n if (src instanceof Date && !isNaN(src)) {\n return (!dst || !(dst instanceof Date) || isNaN(dst) || dst < src) ? src : dst;\n }\n\n // Access mode\n if (src instanceof AccessMode) {\n return new AccessMode(src);\n }\n\n // Handle Array\n if (src instanceof Array) {\n return src;\n }\n\n if (!dst || dst === DEL_CHAR) {\n dst = src.constructor();\n }\n\n for (let prop in src) {\n if (src.hasOwnProperty(prop) && (!ignore || !ignore[prop]) && (prop != '_noForwarding')) {\n try {\n dst[prop] = mergeObj(dst[prop], src[prop]);\n } catch (err) {\n // FIXME: probably need to log something here.\n }\n }\n }\n return dst;\n}\n\n// Update object stored in a cache. Returns updated value.\nexport function mergeToCache(cache, key, newval, ignore) {\n cache[key] = mergeObj(cache[key], newval, ignore);\n return cache[key];\n}\n\n// Strips all values from an object of they evaluate to false or if their name starts with '_'.\n// Used on all outgoing object before serialization to string.\nexport function simplify(obj) {\n Object.keys(obj).forEach((key) => {\n if (key[0] == '_') {\n // Strip fields like \"obj._key\".\n delete obj[key];\n } else if (!obj[key]) {\n // Strip fields which evaluate to false.\n delete obj[key];\n } else if (Array.isArray(obj[key]) && obj[key].length == 0) {\n // Strip empty arrays.\n delete obj[key];\n } else if (!obj[key]) {\n // Strip fields which evaluate to false.\n delete obj[key];\n } else if (obj[key] instanceof Date) {\n // Strip invalid or zero date.\n if (!isValidDate(obj[key])) {\n delete obj[key];\n }\n } else if (typeof obj[key] == 'object') {\n simplify(obj[key]);\n // Strip empty objects.\n if (Object.getOwnPropertyNames(obj[key]).length == 0) {\n delete obj[key];\n }\n }\n });\n return obj;\n};\n\n\n// Trim whitespace, strip empty and duplicate elements elements.\n// If the result is an empty array, add a single element \"\\u2421\" (Unicode Del character).\nexport function normalizeArray(arr) {\n let out = [];\n if (Array.isArray(arr)) {\n // Trim, throw away very short and empty tags.\n for (let i = 0, l = arr.length; i < l; i++) {\n let t = arr[i];\n if (t) {\n t = t.trim().toLowerCase();\n if (t.length > 1) {\n out.push(t);\n }\n }\n }\n out.sort().filter(function(item, pos, ary) {\n return !pos || item != ary[pos - 1];\n });\n }\n if (out.length == 0) {\n // Add single tag with a Unicode Del character, otherwise an ampty array\n // is ambiguos. The Del tag will be stripped by the server.\n out.push(DEL_CHAR);\n }\n return out;\n}\n","/**\n * @file Abstraction layer for websocket and long polling connections.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport CommError from './comm-error.js';\nimport {\n jsonParseHelper\n} from './utils.js';\n\nlet WebSocketProvider;\nlet XHRProvider;\n\n// Error code to return in case of a network problem.\nconst NETWORK_ERROR = 503;\nconst NETWORK_ERROR_TEXT = \"Connection failed\";\n\n// Error code to return when user disconnected from server.\nconst NETWORK_USER = 418;\nconst NETWORK_USER_TEXT = \"Disconnected by client\";\n\n// Settings for exponential backoff\nconst _BOFF_BASE = 2000; // 2000 milliseconds, minimum delay between reconnects\nconst _BOFF_MAX_ITER = 10; // Maximum delay between reconnects 2^10 * 2000 ~ 34 minutes\nconst _BOFF_JITTER = 0.3; // Add random delay\n\n// Helper function for creating an endpoint URL.\nfunction makeBaseUrl(host, protocol, version, apiKey) {\n let url = null;\n\n if (['http', 'https', 'ws', 'wss'].includes(protocol)) {\n url = `${protocol}://${host}`;\n if (url.charAt(url.length - 1) !== '/') {\n url += '/';\n }\n url += 'v' + version + '/channels';\n if (['http', 'https'].includes(protocol)) {\n // Long polling endpoint ends with \"lp\", i.e.\n // '/v0/channels/lp' vs just '/v0/channels' for ws\n url += '/lp';\n }\n url += '?apikey=' + apiKey;\n }\n return url;\n}\n\n/**\n * An abstraction for a websocket or a long polling connection.\n *\n * @class Connection\n * @memberof Tinode\n\n * @param {Object} config - configuration parameters.\n * @param {string} config.host - Host name and optional port number to connect to.\n * @param {string} config.apiKey - API key generated by keygen
.\n * @param {string} config.transport - Network transport to use, either \"ws\"/\"wss\"
for websocket or\n * lp
for long polling.\n * @param {boolean} config.secure - Use Secure WebSocket if true
.\n * @param {string} version_ - Major value of the protocol version, e.g. '0' in '0.17.1'.\n * @param {boolean} autoreconnect_ - If connection is lost, try to reconnect automatically.\n */\nexport default class Connection {\n // Logger, does nothing by default.\n static #log = _ => {};\n\n #boffTimer = null;\n #boffIteration = 0;\n #boffClosed = false; // Indicator if the socket was manually closed - don't autoreconnect if true.\n\n // Websocket.\n #socket = null;\n\n host;\n secure;\n apiKey;\n\n version;\n autoreconnect;\n\n initialized;\n\n // (config.host, config.apiKey, config.transport, config.secure), PROTOCOL_VERSION, true\n constructor(config, version_, autoreconnect_) {\n this.host = config.host;\n this.secure = config.secure;\n this.apiKey = config.apiKey;\n\n this.version = version_;\n this.autoreconnect = autoreconnect_;\n\n if (config.transport === 'lp') {\n // explicit request to use long polling\n this.#init_lp();\n this.initialized = 'lp';\n } else if (config.transport === 'ws') {\n // explicit request to use web socket\n // if websockets are not available, horrible things will happen\n this.#init_ws();\n this.initialized = 'ws';\n }\n\n if (!this.initialized) {\n // Invalid or undefined network transport.\n Connection.#log(\"Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.\");\n throw new Error(\"Unknown or invalid network transport. Running under Node? Call 'Tinode.setNetworkProviders()'.\");\n }\n }\n\n /**\n * To use Connection in a non browser context, supply WebSocket and XMLHttpRequest providers.\n * @static\n * @memberof Connection\n * @param wsProvider WebSocket provider, e.g. for nodeJS , require('ws')
.\n * @param xhrProvider XMLHttpRequest provider, e.g. for node require('xhr')
.\n */\n static setNetworkProviders(wsProvider, xhrProvider) {\n WebSocketProvider = wsProvider;\n XHRProvider = xhrProvider;\n }\n\n /**\n * Assign a non-default logger.\n * @static\n * @memberof Connection\n * @param {function} l variadic logging function.\n */\n static set logger(l) {\n Connection.#log = l;\n }\n\n /**\n * Initiate a new connection\n * @memberof Tinode.Connection#\n * @param {string} host_ Host name to connect to; if null
the old host name will be used.\n * @param {boolean} force Force new connection even if one already exists.\n * @return {Promise} Promise resolved/rejected when the connection call completes, resolution is called without\n * parameters, rejection passes the {Error} as parameter.\n */\n connect(host_, force) {\n return Promise.reject(null);\n }\n\n /**\n * Try to restore a network connection, also reset backoff.\n * @memberof Tinode.Connection#\n *\n * @param {boolean} force - reconnect even if there is a live connection already.\n */\n reconnect(force) {}\n\n /**\n * Terminate the network connection\n * @memberof Tinode.Connection#\n */\n disconnect() {}\n\n /**\n * Send a string to the server.\n * @memberof Tinode.Connection#\n *\n * @param {string} msg - String to send.\n * @throws Throws an exception if the underlying connection is not live.\n */\n sendText(msg) {}\n\n /**\n * Check if connection is alive.\n * @memberof Tinode.Connection#\n * @returns {boolean} true
if connection is live, false
otherwise.\n */\n isConnected() {\n return false;\n }\n\n /**\n * Get the name of the current network transport.\n * @memberof Tinode.Connection#\n * @returns {string} name of the transport such as \"ws\"
or \"lp\"
.\n */\n transport() {\n return this.initialized;\n }\n\n /**\n * Send network probe to check if connection is indeed live.\n * @memberof Tinode.Connection#\n */\n probe() {\n this.sendText('1');\n }\n\n /**\n * Reset autoreconnect counter to zero.\n * @memberof Tinode.Connection#\n */\n backoffReset() {\n this.#boffReset();\n }\n\n // Backoff implementation - reconnect after a timeout.\n #boffReconnect() {\n // Clear timer\n clearTimeout(this.#boffTimer);\n // Calculate when to fire the reconnect attempt\n const timeout = _BOFF_BASE * (Math.pow(2, this.#boffIteration) * (1.0 + _BOFF_JITTER * Math.random()));\n // Update iteration counter for future use\n this.#boffIteration = (this.#boffIteration >= _BOFF_MAX_ITER ? this.#boffIteration : this.#boffIteration + 1);\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(timeout);\n }\n\n this.#boffTimer = setTimeout(_ => {\n Connection.#log(`Reconnecting, iter=${this.#boffIteration}, timeout=${timeout}`);\n // Maybe the socket was closed while we waited for the timer?\n if (!this.#boffClosed) {\n const prom = this.connect();\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(0, prom);\n } else {\n // Suppress error if it's not used.\n prom.catch(_ => {\n /* do nothing */\n });\n }\n } else if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(-1);\n }\n }, timeout);\n }\n\n // Terminate auto-reconnect process.\n #boffStop() {\n clearTimeout(this.#boffTimer);\n this.#boffTimer = null;\n }\n\n // Reset auto-reconnect iteration counter.\n #boffReset() {\n this.#boffIteration = 0;\n }\n\n // Initialization for long polling.\n #init_lp() {\n const XDR_UNSENT = 0; // Client has been created. open() not called yet.\n const XDR_OPENED = 1; // open() has been called.\n const XDR_HEADERS_RECEIVED = 2; // send() has been called, and headers and status are available.\n const XDR_LOADING = 3; // Downloading; responseText holds partial data.\n const XDR_DONE = 4; // The operation is complete.\n\n // Fully composed endpoint URL, with API key & SID\n let _lpURL = null;\n\n let _poller = null;\n let _sender = null;\n\n let lp_sender = (url_) => {\n const sender = new XHRProvider();\n sender.onreadystatechange = (evt) => {\n if (sender.readyState == XDR_DONE && sender.status >= 400) {\n // Some sort of error response\n throw new CommError(\"LP sender failed\", sender.status);\n }\n };\n\n sender.open('POST', url_, true);\n return sender;\n }\n\n let lp_poller = (url_, resolve, reject) => {\n let poller = new XHRProvider();\n let promiseCompleted = false;\n\n poller.onreadystatechange = evt => {\n if (poller.readyState == XDR_DONE) {\n if (poller.status == 201) { // 201 == HTTP.Created, get SID\n let pkt = JSON.parse(poller.responseText, jsonParseHelper);\n _lpURL = url_ + '&sid=' + pkt.ctrl.params.sid;\n poller = lp_poller(_lpURL);\n poller.send(null);\n if (this.onOpen) {\n this.onOpen();\n }\n\n if (resolve) {\n promiseCompleted = true;\n resolve();\n }\n\n if (this.autoreconnect) {\n this.#boffStop();\n }\n } else if (poller.status > 0 && poller.status < 400) { // 0 = network error; 400 = HTTP.BadRequest\n if (this.onMessage) {\n this.onMessage(poller.responseText);\n }\n poller = lp_poller(_lpURL);\n poller.send(null);\n } else {\n // Don't throw an error here, gracefully handle server errors\n if (reject && !promiseCompleted) {\n promiseCompleted = true;\n reject(poller.responseText);\n }\n if (this.onMessage && poller.responseText) {\n this.onMessage(poller.responseText);\n }\n if (this.onDisconnect) {\n const code = poller.status || (this.#boffClosed ? NETWORK_USER : NETWORK_ERROR);\n const text = poller.responseText || (this.#boffClosed ? NETWORK_USER_TEXT : NETWORK_ERROR_TEXT);\n this.onDisconnect(new CommError(text, code), code);\n }\n\n // Polling has stopped. Indicate it by setting poller to null.\n poller = null;\n if (!this.#boffClosed && this.autoreconnect) {\n this.#boffReconnect();\n }\n }\n }\n };\n // Using POST to avoid caching response by service worker.\n poller.open('POST', url_, true);\n return poller;\n }\n\n this.connect = (host_, force) => {\n this.#boffClosed = false;\n\n if (_poller) {\n if (!force) {\n return Promise.resolve();\n }\n _poller.onreadystatechange = undefined;\n _poller.abort();\n _poller = null;\n }\n\n if (host_) {\n this.host = host_;\n }\n\n return new Promise((resolve, reject) => {\n const url = makeBaseUrl(this.host, this.secure ? 'https' : 'http', this.version, this.apiKey);\n Connection.#log(\"LP connecting to:\", url);\n _poller = lp_poller(url, resolve, reject);\n _poller.send(null);\n }).catch(err => {\n Connection.#log(\"LP connection failed:\", err);\n });\n };\n\n this.reconnect = force => {\n this.#boffStop();\n this.connect(null, force);\n };\n\n this.disconnect = _ => {\n this.#boffClosed = true;\n this.#boffStop();\n\n if (_sender) {\n _sender.onreadystatechange = undefined;\n _sender.abort();\n _sender = null;\n }\n if (_poller) {\n _poller.onreadystatechange = undefined;\n _poller.abort();\n _poller = null;\n }\n\n if (this.onDisconnect) {\n this.onDisconnect(new CommError(NETWORK_USER_TEXT, NETWORK_USER), NETWORK_USER);\n }\n // Ensure it's reconstructed\n _lpURL = null;\n };\n\n this.sendText = (msg) => {\n _sender = lp_sender(_lpURL);\n if (_sender && (_sender.readyState == XDR_OPENED)) { // 1 == OPENED\n _sender.send(msg);\n } else {\n throw new Error(\"Long poller failed to connect\");\n }\n };\n\n this.isConnected = _ => {\n return (_poller && true);\n };\n }\n\n // Initialization for Websocket\n #init_ws() {\n this.connect = (host_, force) => {\n this.#boffClosed = false;\n\n if (this.#socket) {\n if (!force && this.#socket.readyState == this.#socket.OPEN) {\n return Promise.resolve();\n }\n this.#socket.close();\n this.#socket = null;\n }\n\n if (host_) {\n this.host = host_;\n }\n\n return new Promise((resolve, reject) => {\n const url = makeBaseUrl(this.host, this.secure ? 'wss' : 'ws', this.version, this.apiKey);\n\n Connection.#log(\"WS connecting to: \", url);\n\n // It throws when the server is not accessible but the exception cannot be caught:\n // https://stackoverflow.com/questions/31002592/javascript-doesnt-catch-error-in-websocket-instantiation/31003057\n const conn = new WebSocketProvider(url);\n\n conn.onerror = err => {\n reject(err);\n };\n\n conn.onopen = _ => {\n if (this.autoreconnect) {\n this.#boffStop();\n }\n\n if (this.onOpen) {\n this.onOpen();\n }\n\n resolve();\n };\n\n conn.onclose = _ => {\n this.#socket = null;\n\n if (this.onDisconnect) {\n const code = this.#boffClosed ? NETWORK_USER : NETWORK_ERROR;\n this.onDisconnect(new CommError(this.#boffClosed ? NETWORK_USER_TEXT : NETWORK_ERROR_TEXT, code), code);\n }\n\n if (!this.#boffClosed && this.autoreconnect) {\n this.#boffReconnect();\n }\n };\n\n conn.onmessage = evt => {\n if (this.onMessage) {\n this.onMessage(evt.data);\n }\n };\n\n this.#socket = conn;\n });\n }\n\n this.reconnect = force => {\n this.#boffStop();\n this.connect(null, force);\n };\n\n this.disconnect = _ => {\n this.#boffClosed = true;\n this.#boffStop();\n\n if (!this.#socket) {\n return;\n }\n this.#socket.close();\n this.#socket = null;\n };\n\n this.sendText = msg => {\n if (this.#socket && (this.#socket.readyState == this.#socket.OPEN)) {\n this.#socket.send(msg);\n } else {\n throw new Error(\"Websocket is not connected\");\n }\n };\n\n this.isConnected = _ => {\n return (this.#socket && (this.#socket.readyState == this.#socket.OPEN));\n };\n }\n\n // Callbacks:\n\n /**\n * A callback to pass incoming messages to. See {@link Tinode.Connection#onMessage}.\n * @callback Tinode.Connection.OnMessage\n * @memberof Tinode.Connection\n * @param {string} message - Message to process.\n */\n onMessage = undefined;\n\n /**\n * A callback for reporting a dropped connection.\n * @type {function}\n * @memberof Tinode.Connection#\n */\n onDisconnect = undefined;\n\n /**\n * A callback called when the connection is ready to be used for sending. For websockets it's socket open,\n * for long polling it's readyState=1
(OPENED)\n * @type {function}\n * @memberof Tinode.Connection#\n */\n onOpen = undefined;\n\n /**\n * A callback to notify of reconnection attempts. See {@link Tinode.Connection#onAutoreconnectIteration}.\n * @memberof Tinode.Connection\n * @callback AutoreconnectIterationType\n * @param {string} timeout - time till the next reconnect attempt in milliseconds. -1
means reconnect was skipped.\n * @param {Promise} promise resolved or rejected when the reconnect attemp completes.\n *\n */\n /**\n * A callback to inform when the next attampt to reconnect will happen and to receive connection promise.\n * @memberof Tinode.Connection#\n * @type {Tinode.Connection.AutoreconnectIterationType}\n */\n onAutoreconnectIteration = undefined;\n}\n\nConnection.NETWORK_ERROR = NETWORK_ERROR;\nConnection.NETWORK_ERROR_TEXT = NETWORK_ERROR_TEXT;\nConnection.NETWORK_USER = NETWORK_USER;\nConnection.NETWORK_USER_TEXT = NETWORK_USER_TEXT;\n","/**\n * @file Helper methods for dealing with IndexedDB cache of messages, users, and topics.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nconst DB_VERSION = 1;\nconst DB_NAME = 'tinode-web';\n\nlet IDBProvider;\n\nexport default class DB {\n #onError = _ => {};\n #logger = _ => {};\n\n // Instance of IndexDB.\n db = null;\n // Indicator that the cache is disabled.\n disabled = true;\n\n constructor(onError, logger) {\n this.#onError = onError || this.#onError;\n this.#logger = logger || this.#logger;\n }\n\n #mapObjects(source, callback, context) {\n if (!this.db) {\n return disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction([source]);\n trx.onerror = event => {\n this.#logger('PCache', 'mapObjects', source, event.target.error);\n reject(event.target.error);\n };\n trx.objectStore(source).getAll().onsuccess = event => {\n if (callback) {\n event.target.result.forEach(topic => {\n callback.call(context, topic);\n });\n }\n resolve(event.target.result);\n };\n });\n }\n\n /**\n * Initialize persistent cache: open or create/upgrade if needed.\n * @returns {Promise} promise to be resolved/rejected when the DB is initialized.\n */\n initDatabase() {\n return new Promise((resolve, reject) => {\n // Open the database and initialize callbacks.\n const req = IDBProvider.open(DB_NAME, DB_VERSION);\n req.onsuccess = event => {\n this.db = event.target.result;\n this.disabled = false;\n resolve(this.db);\n };\n req.onerror = event => {\n this.#logger('PCache', \"failed to initialize\", event);\n reject(event.target.error);\n this.#onError(event.target.error);\n };\n req.onupgradeneeded = event => {\n this.db = event.target.result;\n\n this.db.onerror = event => {\n this.#logger('PCache', \"failed to create storage\", event);\n this.#onError(event.target.error);\n };\n\n // Individual object stores.\n // Object store (table) for topics. The primary key is topic name.\n this.db.createObjectStore('topic', {\n keyPath: 'name'\n });\n\n // Users object store. UID is the primary key.\n this.db.createObjectStore('user', {\n keyPath: 'uid'\n });\n\n // Subscriptions object store topic <-> user. Topic name + UID is the primary key.\n this.db.createObjectStore('subscription', {\n keyPath: ['topic', 'uid']\n });\n\n // Messages object store. The primary key is topic name + seq.\n this.db.createObjectStore('message', {\n keyPath: ['topic', 'seq']\n });\n };\n });\n }\n\n /**\n * Delete persistent cache.\n */\n deleteDatabase() {\n // Close connection, otherwise operations will fail with 'onblocked'.\n if (this.db) {\n this.db.close();\n this.db = null;\n }\n return new Promise((resolve, reject) => {\n const req = IDBProvider.deleteDatabase(DB_NAME);\n req.onblocked = _ => {\n if (this.db) {\n this.db.close();\n }\n const err = new Error(\"blocked\");\n this.#logger('PCache', 'deleteDatabase', err);\n reject(err);\n };\n req.onsuccess = _ => {\n this.db = null;\n this.disabled = true;\n resolve(true);\n };\n req.onerror = event => {\n this.#logger('PCache', 'deleteDatabase', event.target.error);\n reject(event.target.error);\n };\n });\n }\n\n /**\n * Check if persistent cache is ready for use.\n * @memberOf DB\n * @returns {boolean} true
if cache is ready, false
otherwise.\n */\n isReady() {\n return !!this.db;\n }\n\n // Topics.\n\n /**\n * Save to cache or update topic in persistent cache.\n * @memberOf DB\n * @param {Topic} topic - topic to be added or updated.\n * @returns {Promise} promise resolved/rejected on operation completion.\n */\n updTopic(topic) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updTopic', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('topic').get(topic.name);\n req.onsuccess = _ => {\n trx.objectStore('topic').put(DB.#serializeTopic(req.result, topic));\n trx.commit();\n };\n });\n }\n\n /**\n * Mark or unmark topic as deleted.\n * @memberOf DB\n * @param {string} name - name of the topic to mark or unmark.\n * @param {boolean} deleted - status\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n markTopicAsDeleted(name, deleted) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'markTopicAsDeleted', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('topic').get(name);\n req.onsuccess = event => {\n const topic = event.target.result;\n if (topic && topic._deleted != deleted) {\n topic._deleted = deleted;\n trx.objectStore('topic').put(topic);\n }\n trx.commit();\n };\n });\n }\n\n /**\n * Remove topic from persistent cache.\n * @memberOf DB\n * @param {string} name - name of the topic to remove from database.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remTopic(name) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['topic', 'subscription', 'message'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'remTopic', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('topic').delete(IDBKeyRange.only(name));\n trx.objectStore('subscription').delete(IDBKeyRange.bound([name, '-'], [name, '~']));\n trx.objectStore('message').delete(IDBKeyRange.bound([name, 0], [name, Number.MAX_SAFE_INTEGER]));\n trx.commit();\n });\n }\n\n /**\n * Execute a callback for each stored topic.\n * @memberOf DB\n * @param {function} callback - function to call for each topic.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapTopics(callback, context) {\n return this.#mapObjects('topic', callback, context);\n }\n\n /**\n * Copy data from serialized object to topic.\n * @memberOf DB\n * @param {Topic} topic - target to deserialize to.\n * @param {Object} src - serialized data to copy from.\n */\n deserializeTopic(topic, src) {\n DB.#deserializeTopic(topic, src);\n }\n\n // Users.\n /**\n * Add or update user object in the persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to save or update.\n * @param {Object} pub - user's public
information.\n * @returns {Promise} promise resolved/rejected on operation completion.\n */\n updUser(uid, pub) {\n if (arguments.length < 2 || pub === undefined) {\n // No point inupdating user with invalid data.\n return;\n }\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').put({\n uid: uid,\n public: pub\n });\n trx.commit();\n });\n }\n\n /**\n * Remove user from persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to remove from the cache.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remUser(uid) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'remUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').delete(IDBKeyRange.only(uid));\n trx.commit();\n });\n }\n\n /**\n * Execute a callback for each stored user.\n * @memberOf DB\n * @param {function} callback - function to call for each topic.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapUsers(callback, context) {\n return this.#mapObjects('user', callback, context);\n }\n\n /**\n * Read a single user from persistent cache.\n * @memberOf DB\n * @param {string} uid - ID of the user to fetch from cache.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n getUser(uid) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['user']);\n trx.oncomplete = event => {\n const user = event.target.result;\n resolve({\n user: user.uid,\n public: user.public\n });\n };\n trx.onerror = event => {\n this.#logger('PCache', 'getUser', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('user').get(uid);\n });\n }\n\n // Subscriptions.\n /**\n * Add or update subscription in persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {string} uid - ID of the subscribed user.\n * @param {Object} sub - subscription to save.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n updSubscription(topicName, uid, sub) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['subscription'], 'readwrite');\n trx.oncomplete = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updSubscription', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('subscription').get([topicName, uid]).onsuccess = (event) => {\n trx.objectStore('subscription').put(DB.#serializeSubscription(event.target.result, topicName, uid, sub));\n trx.commit();\n };\n });\n }\n\n /**\n * Execute a callback for each cached subscription in a given topic.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the subscriptions.\n * @param {function} callback - function to call for each subscription.\n * @param {Object} context - the value or this
inside the callback.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n mapSubscriptions(topicName, callback, context) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['subscription']);\n trx.onerror = (event) => {\n this.#logger('PCache', 'mapSubscriptions', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('subscription').getAll(IDBKeyRange.bound([topicName, '-'], [topicName, '~'])).onsuccess = (event) => {\n if (callback) {\n event.target.result.forEach((topic) => {\n callback.call(context, topic);\n });\n }\n resolve(event.target.result);\n };\n });\n }\n\n // Messages.\n\n /**\n * Save message to persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {Object} msg - message to save.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n addMessage(msg) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'addMessage', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('message').add(DB.#serializeMessage(null, msg));\n trx.commit();\n });\n }\n\n /**\n * Update delivery status of a message stored in persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {number} seq - ID of the message to update\n * @param {number} status - new delivery status of the message.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n updMessageStatus(topicName, seq, status) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = event => {\n resolve(event.target.result);\n };\n trx.onerror = event => {\n this.#logger('PCache', 'updMessageStatus', event.target.error);\n reject(event.target.error);\n };\n const req = trx.objectStore('message').get(IDBKeyRange.only([topicName, seq]));\n req.onsuccess = event => {\n const src = req.result || event.target.result;\n if (!src || src._status == status) {\n trx.commit();\n return;\n }\n trx.objectStore('message').put(DB.#serializeMessage(src, {\n topic: topicName,\n seq: seq,\n _status: status\n }));\n trx.commit();\n };\n });\n }\n\n /**\n * Remove one or more messages from persistent cache.\n * @memberOf DB\n * @param {string} topicName - name of the topic which owns the message.\n * @param {number} from - id of the message to remove or lower boundary when removing range (inclusive).\n * @param {number=} to - upper boundary (exclusive) when removing a range of messages.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n remMessages(topicName, from, to) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve() :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n if (!from && !to) {\n from = 0;\n to = Number.MAX_SAFE_INTEGER;\n }\n const range = to > 0 ? IDBKeyRange.bound([topicName, from], [topicName, to], false, true) :\n IDBKeyRange.only([topicName, from]);\n const trx = this.db.transaction(['message'], 'readwrite');\n trx.onsuccess = (event) => {\n resolve(event.target.result);\n };\n trx.onerror = (event) => {\n this.#logger('PCache', 'remMessages', event.target.error);\n reject(event.target.error);\n };\n trx.objectStore('message').delete(range);\n trx.commit();\n });\n }\n\n /**\n * Retrieve messages from persistent store.\n * @memberOf DB\n * @param {string} topicName - name of the topic to retrieve messages from.\n * @param {function} callback to call for each retrieved message.\n * @param {Object} query - parameters of the message range to retrieve.\n * @param {number=} query.since - the least message ID to retrieve (inclusive).\n * @param {number=} query.before - the greatest message ID to retrieve (exclusive).\n * @param {number=} query.limit - the maximum number of messages to retrieve.\n * @return {Promise} promise resolved/rejected on operation completion.\n */\n readMessages(topicName, query, callback, context) {\n if (!this.isReady()) {\n return this.disabled ?\n Promise.resolve([]) :\n Promise.reject(new Error(\"not initialized\"));\n }\n return new Promise((resolve, reject) => {\n query = query || {};\n const since = query.since > 0 ? query.since : 0;\n const before = query.before > 0 ? query.before : Number.MAX_SAFE_INTEGER;\n const limit = query.limit | 0;\n\n const result = [];\n const range = IDBKeyRange.bound([topicName, since], [topicName, before], false, true);\n const trx = this.db.transaction(['message']);\n trx.onerror = (event) => {\n this.#logger('PCache', 'readMessages', event.target.error);\n reject(event.target.error);\n };\n // Iterate in descending order.\n trx.objectStore('message').openCursor(range, 'prev').onsuccess = (event) => {\n const cursor = event.target.result;\n if (cursor) {\n if (callback) {\n callback.call(context, cursor.value);\n }\n result.push(cursor.value);\n if (limit <= 0 || result.length < limit) {\n cursor.continue();\n } else {\n resolve(result);\n }\n } else {\n resolve(result);\n }\n };\n });\n }\n\n // Private methods.\n\n // Serializable topic fields.\n static #topic_fields = ['created', 'updated', 'deleted', 'read', 'recv', 'seq', 'clear', 'defacs',\n 'creds', 'public', 'trusted', 'private', 'touched', '_deleted'\n ];\n\n // Copy data from src to Topic object.\n static #deserializeTopic(topic, src) {\n DB.#topic_fields.forEach((f) => {\n if (src.hasOwnProperty(f)) {\n topic[f] = src[f];\n }\n });\n if (Array.isArray(src.tags)) {\n topic._tags = src.tags;\n }\n if (src.acs) {\n topic.setAccessMode(src.acs);\n }\n topic.seq |= 0;\n topic.read |= 0;\n topic.unread = Math.max(0, topic.seq - topic.read);\n }\n\n // Copy values from 'src' to 'dst'. Allocate dst if it's null or undefined.\n static #serializeTopic(dst, src) {\n const res = dst || {\n name: src.name\n };\n DB.#topic_fields.forEach((f) => {\n if (src.hasOwnProperty(f)) {\n res[f] = src[f];\n }\n });\n if (Array.isArray(src._tags)) {\n res.tags = src._tags;\n }\n if (src.acs) {\n res.acs = src.getAccessMode().jsonHelper();\n }\n return res;\n }\n\n static #serializeSubscription(dst, topicName, uid, sub) {\n const fields = ['updated', 'mode', 'read', 'recv', 'clear', 'lastSeen', 'userAgent'];\n const res = dst || {\n topic: topicName,\n uid: uid\n };\n\n fields.forEach((f) => {\n if (sub.hasOwnProperty(f)) {\n res[f] = sub[f];\n }\n });\n\n return res;\n }\n\n static #serializeMessage(dst, msg) {\n // Serializable fields.\n const fields = ['topic', 'seq', 'ts', '_status', 'from', 'head', 'content'];\n const res = dst || {};\n fields.forEach((f) => {\n if (msg.hasOwnProperty(f)) {\n res[f] = msg[f];\n }\n });\n return res;\n }\n\n /**\n * To use DB in a non browser context, supply indexedDB provider.\n * @static\n * @memberof DB\n * @param idbProvider indexedDB provider, e.g. for node require('fake-indexeddb')
.\n */\n static setDatabaseProvider(idbProvider) {\n IDBProvider = idbProvider;\n }\n}\n","/**\n * @file Utilities for uploading and downloading files.\n *\n * @copyright 2015-2023 Tinode LLC.\n */\n'use strict';\n\nimport CommError from './comm-error.js';\nimport {\n isUrlRelative,\n jsonParseHelper\n} from './utils.js';\n\nlet XHRProvider;\n\nfunction addURLParam(relUrl, key, value) {\n const url = new URL(relUrl, window.location.origin);\n url.searchParams.append(key, value);\n return url.toString().substring(window.location.origin.length);\n}\n\n/**\n * @class LargeFileHelper - utilities for uploading and downloading files out of band.\n * Don't instantiate this class directly. Use {Tinode.getLargeFileHelper} instead.\n * @memberof Tinode\n *\n * @param {Tinode} tinode - the main Tinode object.\n * @param {string} version - protocol version, i.e. '0'.\n */\nexport default class LargeFileHelper {\n constructor(tinode, version) {\n this._tinode = tinode;\n this._version = version;\n\n this._apiKey = tinode._apiKey;\n this._authToken = tinode.getAuthToken();\n\n // Ongoing requests.\n this.xhr = [];\n }\n\n /**\n * Start uploading the file to an endpoint at baseUrl.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {string} baseUrl base URL of upload server.\n * @param {File|Blob} data data to upload.\n * @param {string} avatarFor topic name if the upload represents an avatar.\n * @param {Callback} onProgress callback. Takes one {float} parameter 0..1\n * @param {Callback} onSuccess callback. Called when the file is successfully uploaded.\n * @param {Callback} onFailure callback. Called in case of a failure.\n *\n * @returns {Promise} resolved/rejected when the upload is completed/failed.\n */\n uploadWithBaseUrl(baseUrl, data, avatarFor, onProgress, onSuccess, onFailure) {\n let url = `/v${this._version}/file/u/`;\n if (baseUrl) {\n let base = baseUrl;\n if (base.endsWith('/')) {\n // Removing trailing slash.\n base = base.slice(0, -1);\n }\n if (base.startsWith('http://') || base.startsWith('https://')) {\n url = base + url;\n } else {\n throw new Error(`Invalid base URL '${baseUrl}'`);\n }\n }\n\n const instance = this;\n const xhr = new XHRProvider();\n this.xhr.push(xhr);\n\n xhr.open('POST', url, true);\n xhr.setRequestHeader('X-Tinode-APIKey', this._apiKey);\n if (this._authToken) {\n xhr.setRequestHeader('X-Tinode-Auth', `Token ${this._authToken.token}`);\n }\n\n let toResolve = null;\n let toReject = null;\n\n const result = new Promise((resolve, reject) => {\n toResolve = resolve;\n toReject = reject;\n });\n\n xhr.upload.onprogress = e => {\n if (e.lengthComputable) {\n if (onProgress) {\n onProgress(e.loaded / e.total);\n }\n if (this.onProgress) {\n this.onProgress(e.loaded / e.total);\n }\n }\n };\n\n xhr.onload = function() {\n let pkt;\n try {\n pkt = JSON.parse(this.response, jsonParseHelper);\n } catch (err) {\n instance._tinode.logger(\"ERROR: Invalid server response in LargeFileHelper\", this.response);\n pkt = {\n ctrl: {\n code: this.status,\n text: this.statusText\n }\n };\n }\n\n if (this.status >= 200 && this.status < 300) {\n if (toResolve) {\n toResolve(pkt.ctrl.params.url);\n }\n if (onSuccess) {\n onSuccess(pkt.ctrl);\n }\n } else if (this.status >= 400) {\n if (toReject) {\n toReject(new CommError(pkt.ctrl.text, pkt.ctrl.code));\n }\n if (onFailure) {\n onFailure(pkt.ctrl);\n }\n } else {\n instance._tinode.logger(\"ERROR: Unexpected server response status\", this.status, this.response);\n }\n };\n\n xhr.onerror = function(e) {\n if (toReject) {\n toReject(e || new Error(\"failed\"));\n }\n if (onFailure) {\n onFailure(null);\n }\n };\n\n xhr.onabort = function(e) {\n if (toReject) {\n toReject(new Error(\"upload cancelled by user\"));\n }\n if (onFailure) {\n onFailure(null);\n }\n };\n\n try {\n const form = new FormData();\n form.append('file', data);\n form.set('id', this._tinode.getNextUniqueId());\n if (avatarFor) {\n form.set('topic', avatarFor);\n }\n xhr.send(form);\n } catch (err) {\n if (toReject) {\n toReject(err);\n }\n if (onFailure) {\n onFailure(null);\n }\n }\n\n return result;\n }\n /**\n * Start uploading the file to default endpoint.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {File|Blob} data to upload\n * @param {string} avatarFor topic name if the upload represents an avatar.\n * @param {Callback} onProgress callback. Takes one {float} parameter 0..1\n * @param {Callback} onSuccess callback. Called when the file is successfully uploaded.\n * @param {Callback} onFailure callback. Called in case of a failure.\n *\n * @returns {Promise} resolved/rejected when the upload is completed/failed.\n */\n upload(data, avatarFor, onProgress, onSuccess, onFailure) {\n const baseUrl = (this._tinode._secure ? 'https://' : 'http://') + this._tinode._host;\n return this.uploadWithBaseUrl(baseUrl, data, avatarFor, onProgress, onSuccess, onFailure);\n }\n /**\n * Download the file from a given URL using GET request. This method works with the Tinode server only.\n *\n * @memberof Tinode.LargeFileHelper#\n *\n * @param {string} relativeUrl - URL to download the file from. Must be relative url, i.e. must not contain the host.\n * @param {string=} filename - file name to use for the downloaded file.\n *\n * @returns {Promise} resolved/rejected when the download is completed/failed.\n */\n download(relativeUrl, filename, mimetype, onProgress, onError) {\n if (!isUrlRelative(relativeUrl)) {\n // As a security measure refuse to download from an absolute URL.\n if (onError) {\n onError(`The URL '${relativeUrl}' must be relative, not absolute`);\n }\n return;\n }\n if (!this._authToken) {\n if (onError) {\n onError(\"Must authenticate first\");\n }\n return;\n }\n const instance = this;\n\n const xhr = new XHRProvider();\n this.xhr.push(xhr);\n\n // Add '&asatt=1' to URL to request 'Content-Disposition: attachment' response header.\n relativeUrl = addURLParam(relativeUrl, 'asatt', '1');\n\n // Get data as blob (stored by the browser as a temporary file).\n xhr.open('GET', relativeUrl, true);\n xhr.setRequestHeader('X-Tinode-APIKey', this._apiKey);\n xhr.setRequestHeader('X-Tinode-Auth', 'Token ' + this._authToken.token);\n xhr.responseType = 'blob';\n\n xhr.onprogress = function(e) {\n if (onProgress) {\n // Passing e.loaded instead of e.loaded/e.total because e.total\n // is always 0 with gzip compression enabled by the server.\n onProgress(e.loaded);\n }\n };\n\n let toResolve = null;\n let toReject = null;\n\n const result = new Promise((resolve, reject) => {\n toResolve = resolve;\n toReject = reject;\n });\n\n // The blob needs to be saved as file. There is no known way to\n // save the blob as file other than to fake a click on an .\n xhr.onload = function() {\n if (this.status == 200) {\n const link = document.createElement('a');\n // URL.createObjectURL is not available in non-browser environment. This call will fail.\n link.href = window.URL.createObjectURL(new Blob([this.response], {\n type: mimetype\n }));\n link.style.display = 'none';\n link.setAttribute('download', filename);\n document.body.appendChild(link);\n link.click();\n document.body.removeChild(link);\n window.URL.revokeObjectURL(link.href);\n if (toResolve) {\n toResolve();\n }\n } else if (this.status >= 400 && toReject) {\n // The this.responseText is undefined, must use this.response which is a blob.\n // Need to convert this.response to JSON. The blob can only be accessed by the\n // FileReader.\n const reader = new FileReader();\n reader.onload = function() {\n try {\n const pkt = JSON.parse(this.result, jsonParseHelper);\n toReject(new CommError(pkt.ctrl.text, pkt.ctrl.code));\n } catch (err) {\n instance._tinode.logger(\"ERROR: Invalid server response in LargeFileHelper\", this.result);\n toReject(err);\n }\n };\n reader.readAsText(this.response);\n }\n };\n\n xhr.onerror = function(e) {\n if (toReject) {\n toReject(new Error(\"failed\"));\n }\n if (onError) {\n onError(e);\n }\n };\n\n xhr.onabort = function() {\n if (toReject) {\n toReject(null);\n }\n };\n\n try {\n xhr.send();\n } catch (err) {\n if (toReject) {\n toReject(err);\n }\n if (onError) {\n onError(err);\n }\n }\n\n return result;\n }\n /**\n * Try to cancel all ongoing uploads or downloads.\n * @memberof Tinode.LargeFileHelper#\n */\n cancel() {\n this.xhr.forEach(req => {\n if (req.readyState < 4) {\n req.abort();\n }\n });\n }\n /**\n * To use LargeFileHelper in a non browser context, supply XMLHttpRequest provider.\n * @static\n * @memberof LargeFileHelper\n * @param xhrProvider XMLHttpRequest provider, e.g. for node require('xhr')
.\n */\n static setNetworkProvider(xhrProvider) {\n XHRProvider = xhrProvider;\n }\n}\n","/**\n * @module tinode-sdk\n *\n * @copyright 2015-2022 Tinode LLC.\n * @summary Javascript bindings for Tinode.\n * @license Apache 2.0\n * @version 0.20\n *\n * See https://github.com/tinode/webapp for real-life usage.\n *\n * @example\n * \n * \n * \n *\n * \n * ...\n * \n * \n */\n'use strict';\n\n// NOTE TO DEVELOPERS:\n// Localizable strings should be double quoted \"строка на другом языке\",\n// non-localizable strings should be single quoted 'non-localized'.\n\nimport AccessMode from './access-mode.js';\nimport * as Const from './config.js';\nimport CommError from './comm-error.js';\nimport Connection from './connection.js';\nimport DBCache from './db.js';\nimport Drafty from './drafty.js';\nimport LargeFileHelper from './large-file.js';\nimport MetaGetBuilder from './meta-builder.js';\nimport {\n Topic,\n TopicMe,\n TopicFnd\n} from './topic.js';\n\nimport {\n isUrlRelative,\n jsonParseHelper,\n mergeObj,\n rfc3339DateString,\n simplify\n} from './utils.js';\n\n// Re-export AccessMode\nexport {\n AccessMode\n};\n\nlet WebSocketProvider;\nif (typeof WebSocket != 'undefined') {\n WebSocketProvider = WebSocket;\n}\n\nlet XHRProvider;\nif (typeof XMLHttpRequest != 'undefined') {\n XHRProvider = XMLHttpRequest;\n}\n\nlet IndexedDBProvider;\nif (typeof indexedDB != 'undefined') {\n IndexedDBProvider = indexedDB;\n}\n\n// Re-export Drafty.\nexport {\n Drafty\n}\n\ninitForNonBrowserApp();\n\n// Utility functions\n\n// Polyfill for non-browser context, e.g. NodeJs.\nfunction initForNonBrowserApp() {\n // Tinode requirement in native mode because react native doesn't provide Base64 method\n const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=';\n\n if (typeof btoa == 'undefined') {\n global.btoa = function(input = '') {\n let str = input;\n let output = '';\n\n for (let block = 0, charCode, i = 0, map = chars; str.charAt(i | 0) || (map = '=', i % 1); output += map.charAt(63 & block >> 8 - i % 1 * 8)) {\n\n charCode = str.charCodeAt(i += 3 / 4);\n\n if (charCode > 0xFF) {\n throw new Error(\"'btoa' failed: The string to be encoded contains characters outside of the Latin1 range.\");\n }\n block = block << 8 | charCode;\n }\n\n return output;\n };\n }\n\n if (typeof atob == 'undefined') {\n global.atob = function(input = '') {\n let str = input.replace(/=+$/, '');\n let output = '';\n\n if (str.length % 4 == 1) {\n throw new Error(\"'atob' failed: The string to be decoded is not correctly encoded.\");\n }\n for (let bc = 0, bs = 0, buffer, i = 0; buffer = str.charAt(i++);\n\n ~buffer && (bs = bc % 4 ? bs * 64 + buffer : buffer,\n bc++ % 4) ? output += String.fromCharCode(255 & bs >> (-2 * bc & 6)) : 0\n ) {\n buffer = chars.indexOf(buffer);\n }\n\n return output;\n };\n }\n\n if (typeof window == 'undefined') {\n global.window = {\n WebSocket: WebSocketProvider,\n XMLHttpRequest: XHRProvider,\n indexedDB: IndexedDBProvider,\n URL: {\n createObjectURL: function() {\n throw new Error(\"Unable to use URL.createObjectURL in a non-browser application\");\n }\n }\n }\n }\n\n Connection.setNetworkProviders(WebSocketProvider, XHRProvider);\n LargeFileHelper.setNetworkProvider(XHRProvider);\n DBCache.setDatabaseProvider(IndexedDBProvider);\n}\n\n// Detect find most useful network transport.\nfunction detectTransport() {\n if (typeof window == 'object') {\n if (window['WebSocket']) {\n return 'ws';\n } else if (window['XMLHttpRequest']) {\n // The browser or node has no websockets, using long polling.\n return 'lp';\n }\n }\n return null;\n}\n\n// btoa replacement. Stock btoa fails on on non-Latin1 strings.\nfunction b64EncodeUnicode(str) {\n // The encodeURIComponent percent-encodes UTF-8 string,\n // then the percent encoding is converted into raw bytes which\n // can be fed into btoa.\n return btoa(encodeURIComponent(str).replace(/%([0-9A-F]{2})/g,\n function toSolidBytes(match, p1) {\n return String.fromCharCode('0x' + p1);\n }));\n}\n\n// JSON stringify helper - pre-processor for JSON.stringify\nfunction jsonBuildHelper(key, val) {\n if (val instanceof Date) {\n // Convert javascript Date objects to rfc3339 strings\n val = rfc3339DateString(val);\n } else if (val instanceof AccessMode) {\n val = val.jsonHelper();\n } else if (val === undefined || val === null || val === false ||\n (Array.isArray(val) && val.length == 0) ||\n ((typeof val == 'object') && (Object.keys(val).length == 0))) {\n // strip out empty elements while serializing objects to JSON\n return undefined;\n }\n\n return val;\n};\n\n// Trims very long strings (encoded images) to make logged packets more readable.\nfunction jsonLoggerHelper(key, val) {\n if (typeof val == 'string' && val.length > 128) {\n return '<' + val.length + ', bytes: ' + val.substring(0, 12) + '...' + val.substring(val.length - 12) + '>';\n }\n return jsonBuildHelper(key, val);\n};\n\n// Parse browser user agent to extract browser name and version.\nfunction getBrowserInfo(ua, product) {\n ua = ua || '';\n let reactnative = '';\n // Check if this is a ReactNative app.\n if (/reactnative/i.test(product)) {\n reactnative = 'ReactNative; ';\n }\n let result;\n // Remove useless string.\n ua = ua.replace(' (KHTML, like Gecko)', '');\n // Test for WebKit-based browser.\n let m = ua.match(/(AppleWebKit\\/[.\\d]+)/i);\n if (m) {\n // List of common strings, from more useful to less useful.\n // All unknown strings get the highest (-1) priority.\n const priority = ['edg', 'chrome', 'safari', 'mobile', 'version'];\n let tmp = ua.substr(m.index + m[0].length).split(' ');\n let tokens = [];\n let version; // 1.0 in Version/1.0 or undefined;\n // Split string like 'Name/0.0.0' into ['Name', '0.0.0', 3] where the last element is the priority.\n for (let i = 0; i < tmp.length; i++) {\n let m2 = /([\\w.]+)[\\/]([\\.\\d]+)/.exec(tmp[i]);\n if (m2) {\n // Unknown values are highest priority (-1).\n tokens.push([m2[1], m2[2], priority.findIndex((e) => {\n return m2[1].toLowerCase().startsWith(e);\n })]);\n if (m2[1] == 'Version') {\n version = m2[2];\n }\n }\n }\n // Sort by priority: more interesting is earlier than less interesting.\n tokens.sort((a, b) => {\n return a[2] - b[2];\n });\n if (tokens.length > 0) {\n // Return the least common browser string and version.\n if (tokens[0][0].toLowerCase().startsWith('edg')) {\n tokens[0][0] = 'Edge';\n } else if (tokens[0][0] == 'OPR') {\n tokens[0][0] = 'Opera';\n } else if (tokens[0][0] == 'Safari' && version) {\n tokens[0][1] = version;\n }\n result = tokens[0][0] + '/' + tokens[0][1];\n } else {\n // Failed to ID the browser. Return the webkit version.\n result = m[1];\n }\n } else if (/firefox/i.test(ua)) {\n m = /Firefox\\/([.\\d]+)/g.exec(ua);\n if (m) {\n result = 'Firefox/' + m[1];\n } else {\n result = 'Firefox/?';\n }\n } else {\n // Neither AppleWebKit nor Firefox. Try the last resort.\n m = /([\\w.]+)\\/([.\\d]+)/.exec(ua);\n if (m) {\n result = m[1] + '/' + m[2];\n } else {\n m = ua.split(' ');\n result = m[0];\n }\n }\n\n // Shorten the version to one dot 'a.bb.ccc.d -> a.bb' at most.\n m = result.split('/');\n if (m.length > 1) {\n const v = m[1].split('.');\n const minor = v[1] ? '.' + v[1].substr(0, 2) : '';\n result = `${m[0]}/${v[0]}${minor}`;\n }\n return reactnative + result;\n}\n\n/**\n * The main class for interacting with Tinode server.\n */\nexport class Tinode {\n _host;\n _secure;\n\n _appName;\n\n // API Key.\n _apiKey;\n\n // Name and version of the browser.\n _browser = '';\n _platform;\n // Hardware\n _hwos = 'undefined';\n _humanLanguage = 'xx';\n\n // Logging to console enabled\n _loggingEnabled = false;\n // When logging, trip long strings (base64-encoded images) for readability\n _trimLongStrings = false;\n // UID of the currently authenticated user.\n _myUID = null;\n // Status of connection: authenticated or not.\n _authenticated = false;\n // Login used in the last successful basic authentication\n _login = null;\n // Token which can be used for login instead of login/password.\n _authToken = null;\n // Counter of received packets\n _inPacketCount = 0;\n // Counter for generating unique message IDs\n _messageId = Math.floor((Math.random() * 0xFFFF) + 0xFFFF);\n // Information about the server, if connected\n _serverInfo = null;\n // Push notification token. Called deviceToken for consistency with the Android SDK.\n _deviceToken = null;\n\n // Cache of pending promises by message id.\n _pendingPromises = {};\n // The Timeout object returned by the reject expired promises setInterval.\n _expirePromises = null;\n\n // Websocket or long polling connection.\n _connection = null;\n\n // Use indexDB for caching topics and messages.\n _persist = false;\n // IndexedDB wrapper object.\n _db = null;\n\n // Tinode's cache of objects\n _cache = {};\n\n /**\n * Create Tinode object.\n *\n * @param {Object} config - configuration parameters.\n * @param {string} config.appName - Name of the calling application to be reported in the User Agent.\n * @param {string} config.host - Host name and optional port number to connect to.\n * @param {string} config.apiKey - API key generated by keygen
.\n * @param {string} config.transport - See {@link Tinode.Connection#transport}.\n * @param {boolean} config.secure - Use Secure WebSocket if true
.\n * @param {string} config.platform - Optional platform identifier, one of \"ios\"
, \"web\"
, \"android\"
.\n * @param {boolen} config.persist - Use IndexedDB persistent storage.\n * @param {function} onComplete - callback to call when initialization is completed.\n */\n constructor(config, onComplete) {\n this._host = config.host;\n this._secure = config.secure;\n\n // Client-provided application name, format /\n this._appName = config.appName || \"Undefined\";\n\n // API Key.\n this._apiKey = config.apiKey;\n\n // Name and version of the browser.\n this._platform = config.platform || 'web';\n // Underlying OS.\n if (typeof navigator != 'undefined') {\n this._browser = getBrowserInfo(navigator.userAgent, navigator.product);\n this._hwos = navigator.platform;\n // This is the default language. It could be changed by client.\n this._humanLanguage = navigator.language || 'en-US';\n }\n\n Connection.logger = this.logger;\n Drafty.logger = this.logger;\n\n // WebSocket or long polling network connection.\n if (config.transport != 'lp' && config.transport != 'ws') {\n config.transport = detectTransport();\n }\n this._connection = new Connection(config, Const.PROTOCOL_VERSION, /* autoreconnect */ true);\n this._connection.onMessage = (data) => {\n // Call the main message dispatcher.\n this.#dispatchMessage(data);\n }\n\n // Ready to start sending.\n this._connection.onOpen = _ => this.#connectionOpen();\n this._connection.onDisconnect = (err, code) => this.#disconnected(err, code);\n\n // Wrapper for the reconnect iterator callback.\n this._connection.onAutoreconnectIteration = (timeout, promise) => {\n if (this.onAutoreconnectIteration) {\n this.onAutoreconnectIteration(timeout, promise);\n }\n }\n\n this._persist = config.persist;\n // Initialize object regardless. It simplifies the code.\n this._db = new DBCache(err => {\n this.logger('DB', err);\n }, this.logger);\n\n if (this._persist) {\n // Create the persistent cache.\n // Store promises to be resolved when messages load into memory.\n const prom = [];\n this._db.initDatabase().then(_ => {\n // First load topics into memory.\n return this._db.mapTopics((data) => {\n let topic = this.#cacheGet('topic', data.name);\n if (topic) {\n return;\n }\n if (data.name == Const.TOPIC_ME) {\n topic = new TopicMe();\n } else if (data.name == Const.TOPIC_FND) {\n topic = new TopicFnd();\n } else {\n topic = new Topic(data.name);\n }\n this._db.deserializeTopic(topic, data);\n this.#attachCacheToTopic(topic);\n topic._cachePutSelf();\n // Topic loaded from DB is not new.\n delete topic._new;\n // Request to load messages and save the promise.\n prom.push(topic._loadMessages(this._db));\n });\n }).then(_ => {\n // Then load users.\n return this._db.mapUsers((data) => {\n this.#cachePut('user', data.uid, mergeObj({}, data.public));\n });\n }).then(_ => {\n // Now wait for all messages to finish loading.\n return Promise.all(prom);\n }).then(_ => {\n if (onComplete) {\n onComplete();\n }\n this.logger(\"Persistent cache initialized.\");\n }).catch(err => {\n if (onComplete) {\n onComplete(err);\n }\n this.logger(\"Failed to initialize persistent cache:\", err);\n });\n } else {\n this._db.deleteDatabase().then(_ => {\n if (onComplete) {\n onComplete();\n }\n });\n }\n }\n\n // Private methods.\n\n // Console logger. Babel somehow fails to parse '...rest' parameter.\n logger(str, ...args) {\n if (this._loggingEnabled) {\n const d = new Date();\n const dateString = ('0' + d.getUTCHours()).slice(-2) + ':' +\n ('0' + d.getUTCMinutes()).slice(-2) + ':' +\n ('0' + d.getUTCSeconds()).slice(-2) + '.' +\n ('00' + d.getUTCMilliseconds()).slice(-3);\n\n console.log('[' + dateString + ']', str, args.join(' '));\n }\n }\n\n // Generator of default promises for sent packets.\n #makePromise(id) {\n let promise = null;\n if (id) {\n promise = new Promise((resolve, reject) => {\n // Stored callbacks will be called when the response packet with this Id arrives\n this._pendingPromises[id] = {\n 'resolve': resolve,\n 'reject': reject,\n 'ts': new Date()\n };\n });\n }\n return promise;\n };\n\n // Resolve or reject a pending promise.\n // Unresolved promises are stored in _pendingPromises.\n #execPromise(id, code, onOK, errorText) {\n const callbacks = this._pendingPromises[id];\n if (callbacks) {\n delete this._pendingPromises[id];\n if (code >= 200 && code < 400) {\n if (callbacks.resolve) {\n callbacks.resolve(onOK);\n }\n } else if (callbacks.reject) {\n callbacks.reject(new CommError(errorText, code));\n }\n }\n }\n\n // Send a packet. If packet id is provided return a promise.\n #send(pkt, id) {\n let promise;\n if (id) {\n promise = this.#makePromise(id);\n }\n pkt = simplify(pkt);\n let msg = JSON.stringify(pkt);\n this.logger(\"out: \" + (this._trimLongStrings ? JSON.stringify(pkt, jsonLoggerHelper) : msg));\n try {\n this._connection.sendText(msg);\n } catch (err) {\n // If sendText throws, wrap the error in a promise or rethrow.\n if (id) {\n this.#execPromise(id, Connection.NETWORK_ERROR, null, err.message);\n } else {\n throw err;\n }\n }\n return promise;\n }\n\n // The main message dispatcher.\n #dispatchMessage(data) {\n // Skip empty response. This happens when LP times out.\n if (!data)\n return;\n\n this._inPacketCount++;\n\n // Send raw message to listener\n if (this.onRawMessage) {\n this.onRawMessage(data);\n }\n\n if (data === '0') {\n // Server response to a network probe.\n if (this.onNetworkProbe) {\n this.onNetworkProbe();\n }\n // No processing is necessary.\n return;\n }\n\n let pkt = JSON.parse(data, jsonParseHelper);\n if (!pkt) {\n this.logger(\"in: \" + data);\n this.logger(\"ERROR: failed to parse data\");\n } else {\n this.logger(\"in: \" + (this._trimLongStrings ? JSON.stringify(pkt, jsonLoggerHelper) : data));\n\n // Send complete packet to listener\n if (this.onMessage) {\n this.onMessage(pkt);\n }\n\n if (pkt.ctrl) {\n // Handling {ctrl} message\n if (this.onCtrlMessage) {\n this.onCtrlMessage(pkt.ctrl);\n }\n\n // Resolve or reject a pending promise, if any\n if (pkt.ctrl.id) {\n this.#execPromise(pkt.ctrl.id, pkt.ctrl.code, pkt.ctrl, pkt.ctrl.text);\n }\n setTimeout(_ => {\n if (pkt.ctrl.code == 205 && pkt.ctrl.text == 'evicted') {\n // User evicted from topic.\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n topic._resetSub();\n if (pkt.ctrl.params && pkt.ctrl.params.unsub) {\n topic._gone();\n }\n }\n } else if (pkt.ctrl.code < 300 && pkt.ctrl.params) {\n if (pkt.ctrl.params.what == 'data') {\n // code=208, all messages received: \"params\":{\"count\":11,\"what\":\"data\"},\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n topic._allMessagesReceived(pkt.ctrl.params.count);\n }\n } else if (pkt.ctrl.params.what == 'sub') {\n // code=204, the topic has no (refreshed) subscriptions.\n const topic = this.#cacheGet('topic', pkt.ctrl.topic);\n if (topic) {\n // Trigger topic.onSubsUpdated.\n topic._processMetaSubs([]);\n }\n }\n }\n }, 0);\n } else {\n setTimeout(_ => {\n if (pkt.meta) {\n // Handling a {meta} message.\n // Preferred API: Route meta to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.meta.topic);\n if (topic) {\n topic._routeMeta(pkt.meta);\n }\n\n if (pkt.meta.id) {\n this.#execPromise(pkt.meta.id, 200, pkt.meta, 'META');\n }\n\n // Secondary API: callback\n if (this.onMetaMessage) {\n this.onMetaMessage(pkt.meta);\n }\n } else if (pkt.data) {\n // Handling {data} message\n // Preferred API: Route data to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.data.topic);\n if (topic) {\n topic._routeData(pkt.data);\n }\n\n // Secondary API: Call callback\n if (this.onDataMessage) {\n this.onDataMessage(pkt.data);\n }\n } else if (pkt.pres) {\n // Handling {pres} message\n // Preferred API: Route presence to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.pres.topic);\n if (topic) {\n topic._routePres(pkt.pres);\n }\n\n // Secondary API - callback\n if (this.onPresMessage) {\n this.onPresMessage(pkt.pres);\n }\n } else if (pkt.info) {\n // {info} message - read/received notifications and key presses\n // Preferred API: Route {info}} to topic, if one is registered\n const topic = this.#cacheGet('topic', pkt.info.topic);\n if (topic) {\n topic._routeInfo(pkt.info);\n }\n\n // Secondary API - callback\n if (this.onInfoMessage) {\n this.onInfoMessage(pkt.info);\n }\n } else {\n this.logger(\"ERROR: Unknown packet received.\");\n }\n }, 0);\n }\n }\n }\n\n // Connection open, ready to start sending.\n #connectionOpen() {\n if (!this._expirePromises) {\n // Reject promises which have not been resolved for too long.\n this._expirePromises = setInterval(_ => {\n const err = new CommError(\"timeout\", 504);\n const expires = new Date(new Date().getTime() - Const.EXPIRE_PROMISES_TIMEOUT);\n for (let id in this._pendingPromises) {\n let callbacks = this._pendingPromises[id];\n if (callbacks && callbacks.ts < expires) {\n this.logger(\"Promise expired\", id);\n delete this._pendingPromises[id];\n if (callbacks.reject) {\n callbacks.reject(err);\n }\n }\n }\n }, Const.EXPIRE_PROMISES_PERIOD);\n }\n this.hello();\n }\n\n #disconnected(err, code) {\n this._inPacketCount = 0;\n this._serverInfo = null;\n this._authenticated = false;\n\n if (this._expirePromises) {\n clearInterval(this._expirePromises);\n this._expirePromises = null;\n }\n\n // Mark all topics as unsubscribed\n this.#cacheMap('topic', (topic, key) => {\n topic._resetSub();\n });\n\n // Reject all pending promises\n for (let key in this._pendingPromises) {\n const callbacks = this._pendingPromises[key];\n if (callbacks && callbacks.reject) {\n callbacks.reject(err);\n }\n }\n this._pendingPromises = {};\n\n if (this.onDisconnect) {\n this.onDisconnect(err);\n }\n }\n\n // Get User Agent string\n #getUserAgent() {\n return this._appName + ' (' + (this._browser ? this._browser + '; ' : '') + this._hwos + '); ' + Const.LIBRARY;\n }\n\n // Generator of packets stubs\n #initPacket(type, topic) {\n switch (type) {\n case 'hi':\n return {\n 'hi': {\n 'id': this.getNextUniqueId(),\n 'ver': Const.VERSION,\n 'ua': this.#getUserAgent(),\n 'dev': this._deviceToken,\n 'lang': this._humanLanguage,\n 'platf': this._platform\n }\n };\n\n case 'acc':\n return {\n 'acc': {\n 'id': this.getNextUniqueId(),\n 'user': null,\n 'scheme': null,\n 'secret': null,\n 'tmpscheme': null,\n 'tmpsecret': null,\n 'login': false,\n 'tags': null,\n 'desc': {},\n 'cred': {}\n }\n };\n\n case 'login':\n return {\n 'login': {\n 'id': this.getNextUniqueId(),\n 'scheme': null,\n 'secret': null\n }\n };\n\n case 'sub':\n return {\n 'sub': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'set': {},\n 'get': {}\n }\n };\n\n case 'leave':\n return {\n 'leave': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'unsub': false\n }\n };\n\n case 'pub':\n return {\n 'pub': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'noecho': false,\n 'head': null,\n 'content': {}\n }\n };\n\n case 'get':\n return {\n 'get': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'what': null,\n 'desc': {},\n 'sub': {},\n 'data': {}\n }\n };\n\n case 'set':\n return {\n 'set': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'desc': {},\n 'sub': {},\n 'tags': [],\n 'ephemeral': {}\n }\n };\n\n case 'del':\n return {\n 'del': {\n 'id': this.getNextUniqueId(),\n 'topic': topic,\n 'what': null,\n 'delseq': null,\n 'user': null,\n 'hard': false\n }\n };\n\n case 'note':\n return {\n 'note': {\n // no id by design (except calls).\n 'topic': topic,\n 'what': null, // one of \"recv\", \"read\", \"kp\", \"call\"\n 'seq': undefined // the server-side message id acknowledged as received or read.\n }\n };\n\n default:\n throw new Error(`Unknown packet type requested: ${type}`);\n }\n }\n\n // Cache management\n #cachePut(type, name, obj) {\n this._cache[type + ':' + name] = obj;\n }\n #cacheGet(type, name) {\n return this._cache[type + ':' + name];\n }\n #cacheDel(type, name) {\n delete this._cache[type + ':' + name];\n }\n\n // Enumerate all items in cache, call func for each item.\n // Enumeration stops if func returns true.\n #cacheMap(type, func, context) {\n const key = type ? type + ':' : undefined;\n for (let idx in this._cache) {\n if (!key || idx.indexOf(key) == 0) {\n if (func.call(context, this._cache[idx], idx)) {\n break;\n }\n }\n }\n }\n\n // Make limited cache management available to topic.\n // Caching user.public only. Everything else is per-topic.\n #attachCacheToTopic(topic) {\n topic._tinode = this;\n\n topic._cacheGetUser = (uid) => {\n const pub = this.#cacheGet('user', uid);\n if (pub) {\n return {\n user: uid,\n public: mergeObj({}, pub)\n };\n }\n return undefined;\n };\n topic._cachePutUser = (uid, user) => {\n this.#cachePut('user', uid, mergeObj({}, user.public));\n };\n topic._cacheDelUser = (uid) => {\n this.#cacheDel('user', uid);\n };\n topic._cachePutSelf = _ => {\n this.#cachePut('topic', topic.name, topic);\n };\n topic._cacheDelSelf = _ => {\n this.#cacheDel('topic', topic.name);\n };\n }\n\n // On successful login save server-provided data.\n #loginSuccessful(ctrl) {\n if (!ctrl.params || !ctrl.params.user) {\n return ctrl;\n }\n // This is a response to a successful login,\n // extract UID and security token, save it in Tinode module\n this._myUID = ctrl.params.user;\n this._authenticated = (ctrl && ctrl.code >= 200 && ctrl.code < 300);\n if (ctrl.params && ctrl.params.token && ctrl.params.expires) {\n this._authToken = {\n token: ctrl.params.token,\n expires: ctrl.params.expires\n };\n } else {\n this._authToken = null;\n }\n\n if (this.onLogin) {\n this.onLogin(ctrl.code, ctrl.text);\n }\n\n return ctrl;\n }\n\n // Static methods.\n /**\n * Helper method to package account credential.\n *\n * @param {string | Credential} meth - validation method or object with validation data.\n * @param {string=} val - validation value (e.g. email or phone number).\n * @param {Object=} params - validation parameters.\n * @param {string=} resp - validation response.\n *\n * @returns {Array.} array with a single credential or null
if no valid credentials were given.\n */\n static credential(meth, val, params, resp) {\n if (typeof meth == 'object') {\n ({\n val,\n params,\n resp,\n meth\n } = meth);\n }\n if (meth && (val || resp)) {\n return [{\n 'meth': meth,\n 'val': val,\n 'resp': resp,\n 'params': params\n }];\n }\n return null;\n }\n\n /**\n * Determine topic type from topic's name: grp, p2p, me, fnd, sys.\n * @param {string} name - Name of the topic to test.\n * @returns {string} One of \"me\"
, \"fnd\"
, \"sys\"
, \"grp\"
,\n * \"p2p\"
or undefined
.\n */\n static topicType(name) {\n return Topic.topicType(name);\n }\n\n /**\n * Check if the given topic name is a name of a 'me' topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a 'me' topic, false
otherwise.\n */\n static isMeTopicName(name) {\n return Topic.isMeTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a group topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a group topic, false
otherwise.\n */\n static isGroupTopicName(name) {\n return Topic.isGroupTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a p2p topic.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p topic, false
otherwise.\n */\n static isP2PTopicName(name) {\n return Topic.isP2PTopicName(name);\n }\n /**\n * Check if the given topic name is a name of a communication topic, i.e. P2P or group.\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p or group topic, false
otherwise.\n */\n static isCommTopicName(name) {\n return Topic.isCommTopicName(name);\n }\n /**\n * Check if the topic name is a name of a new topic.\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a new topic, false
otherwise.\n */\n static isNewGroupTopicName(name) {\n return Topic.isNewGroupTopicName(name);\n }\n /**\n * Check if the topic name is a name of a channel.\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a channel, false
otherwise.\n */\n static isChannelTopicName(name) {\n return Topic.isChannelTopicName(name);\n }\n /**\n * Get information about the current version of this Tinode client library.\n * @returns {string} semantic version of the library, e.g. \"0.15.5-rc1\"
.\n */\n static getVersion() {\n return Const.VERSION;\n }\n /**\n * To use Tinode in a non browser context, supply WebSocket and XMLHttpRequest providers.\n * @static\n *\n * @param wsProvider WebSocket
provider, e.g. for nodeJS , require('ws')
.\n * @param xhrProvider XMLHttpRequest
provider, e.g. for node require('xhr')
.\n */\n static setNetworkProviders(wsProvider, xhrProvider) {\n WebSocketProvider = wsProvider;\n XHRProvider = xhrProvider;\n\n Connection.setNetworkProviders(WebSocketProvider, XHRProvider);\n LargeFileHelper.setNetworkProvider(XHRProvider);\n }\n /**\n * To use Tinode in a non browser context, supply indexedDB
provider.\n * @static\n *\n * @param idbProvider indexedDB
provider, e.g. for nodeJS , require('fake-indexeddb')
.\n */\n static setDatabaseProvider(idbProvider) {\n IndexedDBProvider = idbProvider;\n\n DBCache.setDatabaseProvider(IndexedDBProvider);\n }\n /**\n * Return information about the current name and version of this Tinode library.\n * @static\n *\n * @returns {string} the name of the library and it's version.\n */\n static getLibrary() {\n return Const.LIBRARY;\n }\n /**\n * Check if the given string represents NULL
value as defined by Tinode ('\\u2421'
).\n * @param {string} str - string to check for NULL
value.\n * @returns {boolean} true
if string represents NULL
value, false
otherwise.\n */\n static isNullValue(str) {\n return str === Const.DEL_CHAR;\n }\n\n // Instance methods.\n\n // Generates unique message IDs\n getNextUniqueId() {\n return (this._messageId != 0) ? '' + this._messageId++ : undefined;\n };\n\n /**\n * Connect to the server.\n *\n * @param {string} host_ - name of the host to connect to.\n * @return {Promise} Promise resolved/rejected when the connection call completes:\n * resolve()
is called without parameters, reject()
receives the\n * Error
as a single parameter.\n */\n connect(host_) {\n return this._connection.connect(host_);\n }\n\n /**\n * Attempt to reconnect to the server immediately.\n *\n * @param {string} force - if true
, reconnect even if there is a connection already.\n */\n reconnect(force) {\n this._connection.reconnect(force);\n }\n\n /**\n * Disconnect from the server.\n */\n disconnect() {\n this._connection.disconnect();\n }\n\n /**\n * Clear persistent cache: remove IndexedDB.\n *\n * @return {Promise} Promise resolved/rejected when the operation is completed.\n */\n clearStorage() {\n if (this._db.isReady()) {\n return this._db.deleteDatabase();\n }\n return Promise.resolve();\n }\n\n /**\n * Initialize persistent cache: create IndexedDB cache.\n *\n * @return {Promise} Promise resolved/rejected when the operation is completed.\n */\n initStorage() {\n if (!this._db.isReady()) {\n return this._db.initDatabase();\n }\n return Promise.resolve();\n }\n\n /**\n * Send a network probe message to make sure the connection is alive.\n */\n networkProbe() {\n this._connection.probe();\n }\n\n /**\n * Check for live connection to server.\n *\n * @returns {boolean} true
if there is a live connection, false
otherwise.\n */\n isConnected() {\n return this._connection.isConnected();\n }\n\n /**\n * Check if connection is authenticated (last login was successful).\n *\n * @returns {boolean} true
if authenticated, false
otherwise.\n */\n isAuthenticated() {\n return this._authenticated;\n }\n\n /**\n * Add API key and auth token to the relative URL making it usable for getting data\n * from the server in a simple HTTP GET
request.\n *\n * @param {string} URL - URL to wrap.\n * @returns {string} URL with appended API key and token, if valid token is present.\n */\n authorizeURL(url) {\n if (typeof url != 'string') {\n return url;\n }\n\n if (isUrlRelative(url)) {\n // Fake base to make the relative URL parseable.\n const base = 'scheme://host/';\n const parsed = new URL(url, base);\n if (this._apiKey) {\n parsed.searchParams.append('apikey', this._apiKey);\n }\n if (this._authToken && this._authToken.token) {\n parsed.searchParams.append('auth', 'token');\n parsed.searchParams.append('secret', this._authToken.token);\n }\n // Convert back to string and strip fake base URL except for the root slash.\n url = parsed.toString().substring(base.length - 1);\n }\n return url;\n }\n\n /**\n * @typedef AccountParams\n * @type {Object}\n * @property {DefAcs=} defacs - Default access parameters for user's me
topic.\n * @property {Object=} public - Public application-defined data exposed on me
topic.\n * @property {Object=} private - Private application-defined data accessible on me
topic.\n * @property {Object=} trusted - Trusted user data which can be set by a root user only.\n * @property {Array.} tags - array of string tags for user discovery.\n * @property {string} scheme - Temporary authentication scheme for password reset.\n * @property {string} secret - Temporary authentication secret for password reset.\n * @property {Array.=} attachments - Array of references to out of band attachments used in account description.\n */\n /**\n * @typedef DefAcs\n * @type {Object}\n * @property {string=} auth - Access mode for me
for authenticated users.\n * @property {string=} anon - Access mode for me
for anonymous users.\n */\n\n /**\n * Create or update an account.\n *\n * @param {string} uid - User id to update\n * @param {string} scheme - Authentication scheme; \"basic\"
and \"anonymous\"
are the currently supported schemes.\n * @param {string} secret - Authentication secret, assumed to be already base64 encoded.\n * @param {boolean=} login - Use new account to authenticate current session\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n account(uid, scheme, secret, login, params) {\n const pkt = this.#initPacket('acc');\n pkt.acc.user = uid;\n pkt.acc.scheme = scheme;\n pkt.acc.secret = secret;\n // Log in to the new account using selected scheme\n pkt.acc.login = login;\n\n if (params) {\n pkt.acc.desc.defacs = params.defacs;\n pkt.acc.desc.public = params.public;\n pkt.acc.desc.private = params.private;\n pkt.acc.desc.trusted = params.trusted;\n\n pkt.acc.tags = params.tags;\n pkt.acc.cred = params.cred;\n\n pkt.acc.tmpscheme = params.scheme;\n pkt.acc.tmpsecret = params.secret;\n\n if (Array.isArray(params.attachments) && params.attachments.length > 0) {\n pkt.extra = {\n attachments: params.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n }\n\n return this.#send(pkt, pkt.acc.id);\n }\n\n /**\n * Create a new user. Wrapper for {@link Tinode#account}.\n *\n * @param {string} scheme - Authentication scheme; \"basic\"
is the only currently supported scheme.\n * @param {string} secret - Authentication.\n * @param {boolean=} login - Use new account to authenticate current session\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n createAccount(scheme, secret, login, params) {\n let promise = this.account(Const.USER_NEW, scheme, secret, login, params);\n if (login) {\n promise = promise.then(ctrl => this.#loginSuccessful(ctrl));\n }\n return promise;\n }\n\n /**\n * Create user with 'basic'
authentication scheme and immediately\n * use it for authentication. Wrapper for {@link Tinode#account}.\n *\n * @param {string} username - Login to use for the new account.\n * @param {string} password - User's password.\n * @param {AccountParams=} params - User data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n createAccountBasic(username, password, params) {\n // Make sure we are not using 'null' or 'undefined';\n username = username || '';\n password = password || '';\n return this.createAccount('basic',\n b64EncodeUnicode(username + ':' + password), true, params);\n }\n\n /**\n * Update user's credentials for 'basic'
authentication scheme. Wrapper for {@link Tinode#account}.\n *\n * @param {string} uid - User ID to update.\n * @param {string} username - Login to use for the new account.\n * @param {string} password - User's password.\n * @param {AccountParams=} params - data to pass to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n updateAccountBasic(uid, username, password, params) {\n // Make sure we are not using 'null' or 'undefined';\n username = username || '';\n password = password || '';\n return this.account(uid, 'basic',\n b64EncodeUnicode(username + ':' + password), false, params);\n }\n\n /**\n * Send handshake to the server.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n hello() {\n const pkt = this.#initPacket('hi');\n\n return this.#send(pkt, pkt.hi.id)\n .then(ctrl => {\n // Reset backoff counter on successful connection.\n this._connection.backoffReset();\n\n // Server response contains server protocol version, build, constraints,\n // session ID for long polling. Save them.\n if (ctrl.params) {\n this._serverInfo = ctrl.params;\n }\n\n if (this.onConnect) {\n this.onConnect();\n }\n\n return ctrl;\n }).catch(err => {\n this._connection.reconnect(true);\n\n if (this.onDisconnect) {\n this.onDisconnect(err);\n }\n });\n }\n\n /**\n * Set or refresh the push notifications/device token. If the client is connected,\n * the deviceToken can be sent to the server.\n *\n * @param {string} dt - token obtained from the provider or false
,\n * null
or undefined
to clear the token.\n *\n * @returns true
if attempt was made to send the update to the server.\n */\n setDeviceToken(dt) {\n let sent = false;\n // Convert any falsish value to null.\n dt = dt || null;\n if (dt != this._deviceToken) {\n this._deviceToken = dt;\n if (this.isConnected() && this.isAuthenticated()) {\n this.#send({\n 'hi': {\n 'dev': dt || Tinode.DEL_CHAR\n }\n });\n sent = true;\n }\n }\n return sent;\n }\n\n /**\n * @typedef Credential\n * @type {Object}\n * @property {string} meth - validation method.\n * @property {string} val - value to validate (e.g. email or phone number).\n * @property {string} resp - validation response.\n * @property {Object} params - validation parameters.\n */\n /**\n * Authenticate current session.\n *\n * @param {string} scheme - Authentication scheme; \"basic\"
is the only currently supported scheme.\n * @param {string} secret - Authentication secret, assumed to be already base64 encoded.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected when server reply is received.\n */\n login(scheme, secret, cred) {\n const pkt = this.#initPacket('login');\n pkt.login.scheme = scheme;\n pkt.login.secret = secret;\n pkt.login.cred = cred;\n\n return this.#send(pkt, pkt.login.id)\n .then(ctrl => this.#loginSuccessful(ctrl));\n }\n\n /**\n * Wrapper for {@link Tinode#login} with basic authentication\n *\n * @param {string} uname - User name.\n * @param {string} password - Password.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n loginBasic(uname, password, cred) {\n return this.login('basic', b64EncodeUnicode(uname + ':' + password), cred)\n .then(ctrl => {\n this._login = uname;\n return ctrl;\n });\n }\n\n /**\n * Wrapper for {@link Tinode#login} with token authentication\n *\n * @param {string} token - Token received in response to earlier login.\n * @param {Credential=} cred - credential confirmation, if required.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n loginToken(token, cred) {\n return this.login('token', token, cred);\n }\n\n /**\n * Send a request for resetting an authentication secret.\n *\n * @param {string} scheme - authentication scheme to reset.\n * @param {string} method - method to use for resetting the secret, such as \"email\" or \"tel\".\n * @param {string} value - value of the credential to use, a specific email address or a phone number.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving the server reply.\n */\n requestResetAuthSecret(scheme, method, value) {\n return this.login('reset', b64EncodeUnicode(scheme + ':' + method + ':' + value));\n }\n\n /**\n * @typedef AuthToken\n * @type {Object}\n * @property {string} token - Token value.\n * @property {Date} expires - Token expiration time.\n */\n /**\n * Get stored authentication token.\n *\n * @returns {AuthToken} authentication token.\n */\n getAuthToken() {\n if (this._authToken && (this._authToken.expires.getTime() > Date.now())) {\n return this._authToken;\n } else {\n this._authToken = null;\n }\n return null;\n }\n\n /**\n * Application may provide a saved authentication token.\n *\n * @param {AuthToken} token - authentication token.\n */\n setAuthToken(token) {\n this._authToken = token;\n }\n\n /**\n * @typedef SetParams\n * @type {Object}\n * @property {SetDesc=} desc - Topic initialization parameters when creating a new topic or a new subscription.\n * @property {SetSub=} sub - Subscription initialization parameters.\n * @property {Array.=} attachments - URLs of out of band attachments used in parameters.\n */\n /**\n * @typedef SetDesc\n * @type {Object}\n * @property {DefAcs=} defacs - Default access mode.\n * @property {Object=} public - Free-form topic description, publically accessible.\n * @property {Object=} private - Free-form topic description accessible only to the owner.\n * @property {Object=} trusted - Trusted user data which can be set by a root user only.\n */\n /**\n * @typedef SetSub\n * @type {Object}\n * @property {string=} user - UID of the user affected by the request. Default (empty) - current user.\n * @property {string=} mode - User access mode, either requested or assigned dependent on context.\n */\n /**\n * Send a topic subscription request.\n *\n * @param {string} topic - Name of the topic to subscribe to.\n * @param {GetQuery=} getParams - Optional subscription metadata query\n * @param {SetParams=} setParams - Optional initialization parameters\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n subscribe(topicName, getParams, setParams) {\n const pkt = this.#initPacket('sub', topicName)\n if (!topicName) {\n topicName = Const.TOPIC_NEW;\n }\n\n pkt.sub.get = getParams;\n\n if (setParams) {\n if (setParams.sub) {\n pkt.sub.set.sub = setParams.sub;\n }\n\n if (setParams.desc) {\n const desc = setParams.desc;\n if (Tinode.isNewGroupTopicName(topicName)) {\n // Full set.desc params are used for new topics only\n pkt.sub.set.desc = desc;\n } else if (Tinode.isP2PTopicName(topicName) && desc.defacs) {\n // Use optional default permissions only.\n pkt.sub.set.desc = {\n defacs: desc.defacs\n };\n }\n }\n\n // See if external objects were used in topic description.\n if (Array.isArray(setParams.attachments) && setParams.attachments.length > 0) {\n pkt.extra = {\n attachments: setParams.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n\n if (setParams.tags) {\n pkt.sub.set.tags = setParams.tags;\n }\n }\n return this.#send(pkt, pkt.sub.id);\n }\n\n /**\n * Detach and optionally unsubscribe from the topic\n *\n * @param {string} topic - Topic to detach from.\n * @param {boolean} unsub - If true
, detach and unsubscribe, otherwise just detach.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n leave(topic, unsub) {\n const pkt = this.#initPacket('leave', topic);\n pkt.leave.unsub = unsub;\n\n return this.#send(pkt, pkt.leave.id);\n }\n\n /**\n * Create message draft without sending it to the server.\n *\n * @param {string} topic - Name of the topic to publish to.\n * @param {Object} content - Payload to publish.\n * @param {boolean=} noEcho - If true
, tell the server not to echo the message to the original session.\n *\n * @returns {Object} new message which can be sent to the server or otherwise used.\n */\n createMessage(topic, content, noEcho) {\n const pkt = this.#initPacket('pub', topic);\n\n let dft = typeof content == 'string' ? Drafty.parse(content) : content;\n if (dft && !Drafty.isPlainText(dft)) {\n pkt.pub.head = {\n mime: Drafty.getContentType()\n };\n content = dft;\n }\n pkt.pub.noecho = noEcho;\n pkt.pub.content = content;\n\n return pkt.pub;\n }\n\n /**\n * Publish {data} message to topic.\n *\n * @param {string} topicName - Name of the topic to publish to.\n * @param {Object} content - Payload to publish.\n * @param {boolean=} noEcho - If true
, tell the server not to echo the message to the original session.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n publish(topicName, content, noEcho) {\n return this.publishMessage(\n this.createMessage(topicName, content, noEcho)\n );\n }\n\n /**\n * Publish message to topic. The message should be created by {@link Tinode#createMessage}.\n *\n * @param {Object} pub - Message to publish.\n * @param {Array.=} attachments - array of URLs with attachments.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n publishMessage(pub, attachments) {\n // Make a shallow copy. Needed in order to clear locally-assigned temp values;\n pub = Object.assign({}, pub);\n pub.seq = undefined;\n pub.from = undefined;\n pub.ts = undefined;\n const msg = {\n pub: pub,\n };\n if (attachments) {\n msg.extra = {\n attachments: attachments.filter(ref => isUrlRelative(ref))\n };\n }\n return this.#send(msg, pub.id);\n }\n\n /**\n * Out of band notification: notify topic that an external (push) notification was recived by the client.\n *\n * @param {object} data - notification payload.\n * @param {string} data.what - notification type, 'msg', 'read', 'sub'.\n * @param {string} data.topic - name of the updated topic.\n * @param {number=} data.seq - seq ID of the affected message.\n * @param {string=} data.xfrom - UID of the sender.\n * @param {object=} data.given - new subscription 'given', e.g. 'ASWP...'.\n * @param {object=} data.want - new subscription 'want', e.g. 'RWJ...'.\n */\n oobNotification(data) {\n this.logger('oob: ' + (this._trimLongStrings ? JSON.stringify(data, jsonLoggerHelper) : data));\n\n switch (data.what) {\n case 'msg':\n if (!data.seq || data.seq < 1 || !data.topic) {\n // Server sent invalid data.\n break;\n }\n\n if (!this.isConnected()) {\n // Let's ignore the message if there is no connection: no connection means there are no open\n // tabs with Tinode.\n break;\n }\n\n const topic = this.#cacheGet('topic', data.topic);\n if (!topic) {\n // TODO: check if there is a case when a message can arrive from an unknown topic.\n break;\n }\n\n if (topic.isSubscribed()) {\n // No need to fetch: topic is already subscribed and got data through normal channel.\n break;\n }\n\n if (topic.maxMsgSeq() < data.seq) {\n if (topic.isChannelType()) {\n topic._updateReceived(data.seq, 'fake-uid');\n }\n\n // New message.\n if (data.xfrom && !this.#cacheGet('user', data.xfrom)) {\n // Message from unknown sender, fetch description from the server.\n // Sending asynchronously without a subscription.\n this.getMeta(data.xfrom, new MetaGetBuilder().withDesc().build()).catch(err => {\n this.logger(\"Failed to get the name of a new sender\", err);\n });\n }\n\n topic.subscribe(null).then(_ => {\n return topic.getMeta(new MetaGetBuilder(topic).withLaterData(24).withLaterDel(24).build());\n }).then(_ => {\n // Allow data fetch to complete and get processed successfully.\n topic.leaveDelayed(false, 1000);\n }).catch(err => {\n this.logger(\"On push data fetch failed\", err);\n }).finally(_ => {\n this.getMeTopic()._refreshContact('msg', topic);\n });\n }\n break;\n\n case 'read':\n this.getMeTopic()._routePres({\n what: 'read',\n seq: data.seq\n });\n break;\n\n case 'sub':\n if (!this.isMe(data.xfrom)) {\n // TODO: handle updates from other users.\n break;\n }\n\n const mode = {\n given: data.modeGiven,\n want: data.modeWant\n };\n const acs = new AccessMode(mode);\n const pres = (!acs.mode || acs.mode == AccessMode._NONE) ?\n // Subscription deleted.\n {\n what: 'gone',\n src: data.topic\n } :\n // New subscription or subscription updated.\n {\n what: 'acs',\n src: data.topic,\n dacs: mode\n };\n this.getMeTopic()._routePres(pres);\n break;\n\n default:\n this.logger(\"Unknown push type ignored\", data.what);\n }\n }\n\n /**\n * @typedef GetQuery\n * @type {Object}\n * @property {GetOptsType=} desc - If provided (even if empty), fetch topic description.\n * @property {GetOptsType=} sub - If provided (even if empty), fetch topic subscriptions.\n * @property {GetDataType=} data - If provided (even if empty), get messages.\n */\n\n /**\n * @typedef GetOptsType\n * @type {Object}\n * @property {Date=} ims - \"If modified since\", fetch data only it was was modified since stated date.\n * @property {number=} limit - Maximum number of results to return. Ignored when querying topic description.\n */\n\n /**\n * @typedef GetDataType\n * @type {Object}\n * @property {number=} since - Load messages with seq id equal or greater than this value.\n * @property {number=} before - Load messages with seq id lower than this number.\n * @property {number=} limit - Maximum number of results to return.\n */\n\n /**\n * Request topic metadata\n *\n * @param {string} topic - Name of the topic to query.\n * @param {GetQuery} params - Parameters of the query. Use {@link Tinode.MetaGetBuilder} to generate.\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n getMeta(topic, params) {\n const pkt = this.#initPacket('get', topic);\n\n pkt.get = mergeObj(pkt.get, params);\n\n return this.#send(pkt, pkt.get.id);\n }\n\n /**\n * Update topic's metadata: description, subscribtions.\n *\n * @param {string} topic - Topic to update.\n * @param {SetParams} params - topic metadata to update.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n setMeta(topic, params) {\n const pkt = this.#initPacket('set', topic);\n const what = [];\n\n if (params) {\n ['desc', 'sub', 'tags', 'cred', 'ephemeral'].forEach(function(key) {\n if (params.hasOwnProperty(key)) {\n what.push(key);\n pkt.set[key] = params[key];\n }\n });\n\n if (Array.isArray(params.attachments) && params.attachments.length > 0) {\n pkt.extra = {\n attachments: params.attachments.filter(ref => isUrlRelative(ref))\n };\n }\n }\n\n if (what.length == 0) {\n return Promise.reject(new Error(\"Invalid {set} parameters\"));\n }\n\n return this.#send(pkt, pkt.set.id);\n }\n\n /**\n * Range of message IDs to delete.\n *\n * @typedef DelRange\n * @type {Object}\n * @property {number} low - low end of the range, inclusive (closed).\n * @property {number=} hi - high end of the range, exclusive (open).\n */\n /**\n * Delete some or all messages in a topic.\n *\n * @param {string} topic - Topic name to delete messages from.\n * @param {DelRange[]} list - Ranges of message IDs to delete.\n * @param {boolean=} hard - Hard or soft delete\n *\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delMessages(topic, ranges, hard) {\n const pkt = this.#initPacket('del', topic);\n\n pkt.del.what = 'msg';\n pkt.del.delseq = ranges;\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete the topic alltogether. Requires Owner permission.\n *\n * @param {string} topicName - Name of the topic to delete\n * @param {boolean} hard - hard-delete topic.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delTopic(topicName, hard) {\n const pkt = this.#initPacket('del', topicName);\n pkt.del.what = 'topic';\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete subscription. Requires Share permission.\n *\n * @param {string} topicName - Name of the topic to delete\n * @param {string} user - User ID to remove.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delSubscription(topicName, user) {\n const pkt = this.#initPacket('del', topicName);\n pkt.del.what = 'sub';\n pkt.del.user = user;\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Delete credential. Always sent on 'me'
topic.\n *\n * @param {string} method - validation method such as 'email'
or 'tel'
.\n * @param {string} value - validation value, i.e. 'alice@example.com'
.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCredential(method, value) {\n const pkt = this.#initPacket('del', Const.TOPIC_ME);\n pkt.del.what = 'cred';\n pkt.del.cred = {\n meth: method,\n val: value\n };\n\n return this.#send(pkt, pkt.del.id);\n }\n\n /**\n * Request to delete account of the current user.\n *\n * @param {boolean} hard - hard-delete user.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCurrentUser(hard) {\n const pkt = this.#initPacket('del', null);\n pkt.del.what = 'user';\n pkt.del.hard = hard;\n\n return this.#send(pkt, pkt.del.id).then(_ => {\n this._myUID = null;\n });\n }\n\n /**\n * Notify server that a message or messages were read or received. Does NOT return promise.\n *\n * @param {string} topicName - Name of the topic where the mesage is being aknowledged.\n * @param {string} what - Action being aknowledged, either \"read\"
or \"recv\"
.\n * @param {number} seq - Maximum id of the message being acknowledged.\n */\n note(topicName, what, seq) {\n if (seq <= 0 || seq >= Const.LOCAL_SEQID) {\n throw new Error(`Invalid message id ${seq}`);\n }\n\n const pkt = this.#initPacket('note', topicName);\n pkt.note.what = what;\n pkt.note.seq = seq;\n this.#send(pkt);\n }\n\n /**\n * Broadcast a key-press notification to topic subscribers. Used to show\n * typing notifications \"user X is typing...\".\n *\n * @param {string} topicName - Name of the topic to broadcast to.\n * @param {string=} type - notification to send, default is 'kp'.\n */\n noteKeyPress(topicName, type) {\n const pkt = this.#initPacket('note', topicName);\n pkt.note.what = type || 'kp';\n this.#send(pkt);\n }\n\n /**\n * Send a video call notification to topic subscribers (including dialing,\n * hangup, etc.).\n *\n * @param {string} topicName - Name of the topic to broadcast to.\n * @param {int} seq - ID of the call message the event pertains to.\n * @param {string} evt - Call event.\n * @param {string} payload - Payload associated with this event (e.g. SDP string).\n *\n * @returns {Promise} Promise (for some call events) which will\n * be resolved/rejected on receiving server reply\n */\n videoCall(topicName, seq, evt, payload) {\n const pkt = this.#initPacket('note', topicName);\n pkt.note.seq = seq;\n pkt.note.what = 'call';\n pkt.note.event = evt;\n pkt.note.payload = payload;\n this.#send(pkt, pkt.note.id);\n }\n\n /**\n * Get a named topic, either pull it from cache or create a new instance.\n * There is a single instance of topic for each name.\n *\n * @param {string} topicName - Name of the topic to get.\n *\n * @returns {Topic} Requested or newly created topic or undefined
if topic name is invalid.\n */\n getTopic(topicName) {\n let topic = this.#cacheGet('topic', topicName);\n if (!topic && topicName) {\n if (topicName == Const.TOPIC_ME) {\n topic = new TopicMe();\n } else if (topicName == Const.TOPIC_FND) {\n topic = new TopicFnd();\n } else {\n topic = new Topic(topicName);\n }\n // Cache management.\n this.#attachCacheToTopic(topic);\n topic._cachePutSelf();\n // Don't save to DB here: a record will be added when the topic is subscribed.\n }\n return topic;\n }\n\n /**\n * Get a named topic from cache.\n *\n * @param {string} topicName - Name of the topic to get.\n *\n * @returns {Topic} Requested topic or undefined
if topic is not found in cache.\n */\n cacheGetTopic(topicName) {\n return this.#cacheGet('topic', topicName);\n }\n\n /**\n * Remove named topic from cache.\n *\n * @param {string} topicName - Name of the topic to remove from cache.\n */\n cacheRemTopic(topicName) {\n this.#cacheDel('topic', topicName);\n }\n\n /**\n * Iterate over cached topics.\n *\n * @param {Function} func - callback to call for each topic.\n * @param {Object} context - 'this' inside the 'func'.\n */\n mapTopics(func, context) {\n this.#cacheMap('topic', func, context);\n }\n\n /**\n * Check if named topic is already present in cache.\n *\n * @param {string} topicName - Name of the topic to check.\n * @returns {boolean} true if topic is found in cache, false otherwise.\n */\n isTopicCached(topicName) {\n return !!this.#cacheGet('topic', topicName);\n }\n\n /**\n * Generate unique name like 'new123456'
suitable for creating a new group topic.\n *\n * @param {boolean} isChan - if the topic is channel-enabled.\n * @returns {string} name which can be used for creating a new group topic.\n */\n newGroupTopicName(isChan) {\n return (isChan ? Const.TOPIC_NEW_CHAN : Const.TOPIC_NEW) + this.getNextUniqueId();\n }\n\n /**\n * Instantiate 'me'
topic or get it from cache.\n *\n * @returns {TopicMe} Instance of 'me'
topic.\n */\n getMeTopic() {\n return this.getTopic(Const.TOPIC_ME);\n }\n\n /**\n * Instantiate 'fnd'
(find) topic or get it from cache.\n *\n * @returns {Topic} Instance of 'fnd'
topic.\n */\n getFndTopic() {\n return this.getTopic(Const.TOPIC_FND);\n }\n\n /**\n * Create a new {@link LargeFileHelper} instance\n *\n * @returns {LargeFileHelper} instance of a {@link Tinode.LargeFileHelper}.\n */\n getLargeFileHelper() {\n return new LargeFileHelper(this, Const.PROTOCOL_VERSION);\n }\n\n /**\n * Get the UID of the the current authenticated user.\n *\n * @returns {string} UID of the current user or undefined
if the session is not yet\n * authenticated or if there is no session.\n */\n getCurrentUserID() {\n return this._myUID;\n }\n\n /**\n * Check if the given user ID is equal to the current user's UID.\n *\n * @param {string} uid - UID to check.\n *\n * @returns {boolean} true if the given UID belongs to the current logged in user.\n */\n isMe(uid) {\n return this._myUID === uid;\n }\n\n /**\n * Get login used for last successful authentication.\n *\n * @returns {string} login last used successfully or undefined
.\n */\n getCurrentLogin() {\n return this._login;\n }\n\n /**\n * Return information about the server: protocol version and build timestamp.\n *\n * @returns {Object} build and version of the server or null
if there is no connection or\n * if the first server response has not been received yet.\n */\n getServerInfo() {\n return this._serverInfo;\n }\n\n /**\n * Report a topic for abuse. Wrapper for {@link Tinode#publish}.\n *\n * @param {string} action - the only supported action is 'report'.\n * @param {string} target - name of the topic being reported.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n report(action, target) {\n return this.publish(Const.TOPIC_SYS, Drafty.attachJSON(null, {\n 'action': action,\n 'target': target\n }));\n }\n\n /**\n * Return server-provided configuration value.\n *\n * @param {string} name of the value to return.\n * @param {Object} defaultValue to return in case the parameter is not set or not found.\n *\n * @returns {Object} named value.\n */\n getServerParam(name, defaultValue) {\n return this._serverInfo && this._serverInfo[name] || defaultValue;\n }\n\n /**\n * Toggle console logging. Logging is off by default.\n *\n * @param {boolean} enabled - Set to true
to enable logging to console.\n * @param {boolean} trimLongStrings - Set to true
to trim long strings.\n */\n enableLogging(enabled, trimLongStrings) {\n this._loggingEnabled = enabled;\n this._trimLongStrings = enabled && trimLongStrings;\n }\n\n /**\n * Set UI language to report to the server. Must be called before 'hi'
is sent, otherwise it will not be used.\n *\n * @param {string} hl - human (UI) language, like \"en_US\"
or \"zh-Hans\"
.\n */\n setHumanLanguage(hl) {\n if (hl) {\n this._humanLanguage = hl;\n }\n }\n\n /**\n * Check if given topic is online.\n *\n * @param {string} name of the topic to test.\n * @returns {boolean} true if topic is online, false otherwise.\n */\n isTopicOnline(name) {\n const topic = this.#cacheGet('topic', name);\n return topic && topic.online;\n }\n\n /**\n * Get access mode for the given contact.\n *\n * @param {string} name of the topic to query.\n * @returns {AccessMode} access mode if topic is found, null otherwise.\n */\n getTopicAccessMode(name) {\n const topic = this.#cacheGet('topic', name);\n return topic ? topic.acs : null;\n }\n\n /**\n * Include message ID into all subsequest messages to server instructin it to send aknowledgemens.\n * Required for promises to function. Default is \"on\"
.\n *\n * @param {boolean} status - Turn aknowledgemens on or off.\n * @deprecated\n */\n wantAkn(status) {\n if (status) {\n this._messageId = Math.floor((Math.random() * 0xFFFFFF) + 0xFFFFFF);\n } else {\n this._messageId = 0;\n }\n }\n\n // Callbacks:\n /**\n * Callback to report when the websocket is opened. The callback has no parameters.\n *\n * @type {onWebsocketOpen}\n */\n onWebsocketOpen = undefined;\n\n /**\n * @typedef ServerParams\n *\n * @type {Object}\n * @property {string} ver - Server version\n * @property {string} build - Server build\n * @property {string=} sid - Session ID, long polling connections only.\n */\n\n /**\n * @callback onConnect\n * @param {number} code - Result code\n * @param {string} text - Text epxplaining the completion, i.e \"OK\" or an error message.\n * @param {ServerParams} params - Parameters returned by the server.\n */\n /**\n * Callback to report when connection with Tinode server is established.\n * @type {onConnect}\n */\n onConnect = undefined;\n\n /**\n * Callback to report when connection is lost. The callback has no parameters.\n * @type {onDisconnect}\n */\n onDisconnect = undefined;\n\n /**\n * @callback onLogin\n * @param {number} code - NUmeric completion code, same as HTTP status codes.\n * @param {string} text - Explanation of the completion code.\n */\n /**\n * Callback to report login completion.\n * @type {onLogin}\n */\n onLogin = undefined;\n\n /**\n * Callback to receive {ctrl}
(control) messages.\n * @type {onCtrlMessage}\n */\n onCtrlMessage = undefined;\n\n /**\n * Callback to recieve {data}
(content) messages.\n * @type {onDataMessage}\n */\n onDataMessage = undefined;\n\n /**\n * Callback to receive {pres}
(presence) messages.\n * @type {onPresMessage}\n */\n onPresMessage = undefined;\n\n /**\n * Callback to receive all messages as objects.\n * @type {onMessage}\n */\n onMessage = undefined;\n\n /**\n * Callback to receive all messages as unparsed text.\n * @type {onRawMessage}\n */\n onRawMessage = undefined;\n\n /**\n * Callback to receive server responses to network probes. See {@link Tinode#networkProbe}\n * @type {onNetworkProbe}\n */\n onNetworkProbe = undefined;\n\n /**\n * Callback to be notified when exponential backoff is iterating.\n * @type {onAutoreconnectIteration}\n */\n onAutoreconnectIteration = undefined;\n};\n\n// Exported constants\nTinode.MESSAGE_STATUS_NONE = Const.MESSAGE_STATUS_NONE;\nTinode.MESSAGE_STATUS_QUEUED = Const.MESSAGE_STATUS_QUEUED;\nTinode.MESSAGE_STATUS_SENDING = Const.MESSAGE_STATUS_SENDING;\nTinode.MESSAGE_STATUS_FAILED = Const.MESSAGE_STATUS_FAILED;\nTinode.MESSAGE_STATUS_FATAL = Const.MESSAGE_STATUS_FATAL;\nTinode.MESSAGE_STATUS_SENT = Const.MESSAGE_STATUS_SENT;\nTinode.MESSAGE_STATUS_RECEIVED = Const.MESSAGE_STATUS_RECEIVED;\nTinode.MESSAGE_STATUS_READ = Const.MESSAGE_STATUS_READ;\nTinode.MESSAGE_STATUS_TO_ME = Const.MESSAGE_STATUS_TO_ME;\n\n// Unicode [del] symbol.\nTinode.DEL_CHAR = Const.DEL_CHAR;\n\n// Names of keys to server-provided configuration limits.\nTinode.MAX_MESSAGE_SIZE = 'maxMessageSize';\nTinode.MAX_SUBSCRIBER_COUNT = 'maxSubscriberCount';\nTinode.MAX_TAG_COUNT = 'maxTagCount';\nTinode.MAX_FILE_UPLOAD_SIZE = 'maxFileUploadSize';\nTinode.REQ_CRED_VALIDATORS = 'reqCred';\n\n// Tinode URI topic ID prefix, 'scheme:path/'.\nTinode.URI_TOPIC_ID_PREFIX = 'tinode:topic/';\n","/**\n * @file Helper class for constructing {@link Tinode.GetQuery}.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n/**\n * Helper class for constructing {@link Tinode.GetQuery}.\n *\n * @class MetaGetBuilder\n * @memberof Tinode\n *\n * @param {Tinode.Topic} parent topic which instantiated this builder.\n */\nexport default class MetaGetBuilder {\n constructor(parent) {\n this.topic = parent;\n this.what = {};\n }\n\n // Get timestamp of the most recent desc update.\n #get_desc_ims() {\n return this.topic._deleted ? undefined : this.topic.updated;\n }\n\n // Get timestamp of the most recent subs update.\n #get_subs_ims() {\n if (this.topic.isP2PType()) {\n return this.#get_desc_ims();\n }\n return this.topic._deleted ? undefined : this.topic._lastSubsUpdate;\n }\n /**\n * Add query parameters to fetch messages within explicit limits.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} since - messages newer than this (inclusive);\n * @param {number=} before - older than this (exclusive)\n * @param {number=} limit - number of messages to fetch\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withData(since, before, limit) {\n this.what['data'] = {\n since: since,\n before: before,\n limit: limit\n };\n return this;\n }\n /**\n * Add query parameters to fetch messages newer than the latest saved message.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - number of messages to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterData(limit) {\n return this.withData(this.topic._maxSeq > 0 ? this.topic._maxSeq + 1 : undefined, undefined, limit);\n }\n /**\n * Add query parameters to fetch messages older than the earliest saved message.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - maximum number of messages to fetch.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withEarlierData(limit) {\n return this.withData(undefined, this.topic._minSeq > 0 ? this.topic._minSeq : undefined, limit);\n }\n /**\n * Add query parameters to fetch topic description if it's newer than the given timestamp.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch messages newer than this timestamp.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withDesc(ims) {\n this.what['desc'] = {\n ims: ims\n };\n return this;\n }\n /**\n * Add query parameters to fetch topic description if it's newer than the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterDesc() {\n return this.withDesc(this.#get_desc_ims());\n }\n /**\n * Add query parameters to fetch subscriptions.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch subscriptions modified more recently than this timestamp\n * @param {number=} limit - maximum number of subscriptions to fetch.\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withSub(ims, limit, userOrTopic) {\n const opts = {\n ims: ims,\n limit: limit\n };\n if (this.topic.getType() == 'me') {\n opts.topic = userOrTopic;\n } else {\n opts.user = userOrTopic;\n }\n this.what['sub'] = opts;\n return this;\n }\n /**\n * Add query parameters to fetch a single subscription.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {Date=} ims - fetch subscriptions modified more recently than this timestamp\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withOneSub(ims, userOrTopic) {\n return this.withSub(ims, undefined, userOrTopic);\n }\n /**\n * Add query parameters to fetch a single subscription if it's been updated since the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {string=} userOrTopic - user ID or topic name to fetch for fetching one subscription.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterOneSub(userOrTopic) {\n return this.withOneSub(this.topic._lastSubsUpdate, userOrTopic);\n }\n /**\n * Add query parameters to fetch subscriptions updated since the last update.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - maximum number of subscriptions to fetch.\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterSub(limit) {\n return this.withSub(this.#get_subs_ims(), limit);\n }\n /**\n * Add query parameters to fetch topic tags.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withTags() {\n this.what['tags'] = true;\n return this;\n }\n /**\n * Add query parameters to fetch user's credentials. 'me'
topic only.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withCred() {\n if (this.topic.getType() == 'me') {\n this.what['cred'] = true;\n } else {\n this.topic._tinode.logger(\"ERROR: Invalid topic type for MetaGetBuilder:withCreds\", this.topic.getType());\n }\n return this;\n }\n /**\n * Add query parameters to fetch deleted messages within explicit limits. Any/all parameters can be null.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} since - ids of messages deleted since this 'del' id (inclusive)\n * @param {number=} limit - number of deleted message ids to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withDel(since, limit) {\n if (since || limit) {\n this.what['del'] = {\n since: since,\n limit: limit\n };\n }\n return this;\n }\n /**\n * Add query parameters to fetch messages deleted after the saved 'del'
id.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @param {number=} limit - number of deleted message ids to fetch\n *\n * @returns {Tinode.MetaGetBuilder} this
object.\n */\n withLaterDel(limit) {\n // Specify 'since' only if we have already received some messages. If\n // we have no locally cached messages then we don't care if any messages were deleted.\n return this.withDel(this.topic._maxSeq > 0 ? this.topic._maxDel + 1 : undefined, limit);\n }\n\n /**\n * Extract subquery: get an object that contains specified subquery.\n * @memberof Tinode.MetaGetBuilder#\n * @param {string} what - subquery to return: one of 'data', 'sub', 'desc', 'tags', 'cred', 'del'.\n * @returns {Object} requested subquery or undefined
.\n */\n extract(what) {\n return this.what[what];\n }\n\n /**\n * Construct parameters.\n * @memberof Tinode.MetaGetBuilder#\n *\n * @returns {Tinode.GetQuery} Get query\n */\n build() {\n const what = [];\n let params = {};\n ['data', 'sub', 'desc', 'tags', 'cred', 'del'].forEach((key) => {\n if (this.what.hasOwnProperty(key)) {\n what.push(key);\n if (Object.getOwnPropertyNames(this.what[key]).length > 0) {\n params[key] = this.what[key];\n }\n }\n });\n if (what.length > 0) {\n params.what = what.join(' ');\n } else {\n params = undefined;\n }\n return params;\n }\n}\n","/**\n * @file In-memory sorted cache of objects.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\n/**\n * In-memory sorted cache of objects.\n *\n * @class CBuffer\n * @memberof Tinode\n * @protected\n *\n * @param {function} compare custom comparator of objects. Takes two parameters a
and b
;\n * returns -1
if a < b
, 0
if a == b
, 1
otherwise.\n * @param {boolean} unique enforce element uniqueness: when true
replace existing element with a new\n * one on conflict; when false
keep both elements.\n */\nexport default class CBuffer {\n #comparator = undefined;\n #unique = false;\n buffer = [];\n\n constructor(compare_, unique_) {\n this.#comparator = compare_ || ((a, b) => {\n return a === b ? 0 : a < b ? -1 : 1;\n });\n this.#unique = unique_;\n }\n\n #findNearest(elem, arr, exact) {\n let start = 0;\n let end = arr.length - 1;\n let pivot = 0;\n let diff = 0;\n let found = false;\n\n while (start <= end) {\n pivot = (start + end) / 2 | 0;\n diff = this.#comparator(arr[pivot], elem);\n if (diff < 0) {\n start = pivot + 1;\n } else if (diff > 0) {\n end = pivot - 1;\n } else {\n found = true;\n break;\n }\n }\n if (found) {\n return {\n idx: pivot,\n exact: true\n };\n }\n if (exact) {\n return {\n idx: -1\n };\n }\n // Not exact - insertion point\n return {\n idx: diff < 0 ? pivot + 1 : pivot\n };\n }\n\n // Insert element into a sorted array.\n #insertSorted(elem, arr) {\n const found = this.#findNearest(elem, arr, false);\n const count = (found.exact && this.#unique) ? 1 : 0;\n arr.splice(found.idx, count, elem);\n return arr;\n }\n\n /**\n * Get an element at the given position.\n * @memberof Tinode.CBuffer#\n * @param {number} at - Position to fetch from.\n * @returns {Object} Element at the given position or undefined
.\n */\n getAt(at) {\n return this.buffer[at];\n }\n\n /**\n * Convenience method for getting the element from the end of the buffer.\n * @memberof Tinode.CBuffer#\n * @param {number} at - position to fetch from, counting from the end;\n * undefined
or null
mean \"last\".\n * @returns {Object} The last element in the buffer or undefined
if buffer is empty.\n */\n getLast(at) {\n at |= 0;\n return this.buffer.length > at ? this.buffer[this.buffer.length - 1 - at] : undefined;\n }\n\n /**\n * Add new element(s) to the buffer. Variadic: takes one or more arguments. If an array is passed as a single\n * argument, its elements are inserted individually.\n * @memberof Tinode.CBuffer#\n *\n * @param {...Object|Array} - One or more objects to insert.\n */\n put() {\n let insert;\n // inspect arguments: if array, insert its elements, if one or more non-array arguments, insert them one by one\n if (arguments.length == 1 && Array.isArray(arguments[0])) {\n insert = arguments[0];\n } else {\n insert = arguments;\n }\n for (let idx in insert) {\n this.#insertSorted(insert[idx], this.buffer);\n }\n }\n\n /**\n * Remove element at the given position.\n * @memberof Tinode.CBuffer#\n * @param {number} at - Position to delete at.\n * @returns {Object} Element at the given position or undefined
.\n */\n delAt(at) {\n at |= 0;\n let r = this.buffer.splice(at, 1);\n if (r && r.length > 0) {\n return r[0];\n }\n return undefined;\n }\n\n /**\n * Remove elements between two positions.\n * @memberof Tinode.CBuffer#\n * @param {number} since - Position to delete from (inclusive).\n * @param {number} before - Position to delete to (exclusive).\n *\n * @returns {Array} array of removed elements (could be zero length).\n */\n delRange(since, before) {\n return this.buffer.splice(since, before - since);\n }\n\n /**\n * Return the number of elements the buffer holds.\n * @memberof Tinode.CBuffer#\n * @return {number} Number of elements in the buffer.\n */\n length() {\n return this.buffer.length;\n }\n\n /**\n * Reset the buffer discarding all elements\n * @memberof Tinode.CBuffer#\n */\n reset() {\n this.buffer = [];\n }\n\n /**\n * Callback for iterating contents of buffer. See {@link Tinode.CBuffer#forEach}.\n * @callback ForEachCallbackType\n * @memberof Tinode.CBuffer#\n * @param {Object} elem - Current element of the buffer.\n * @param {Object} prev - Previous element of the buffer.\n * @param {Object} next - Next element of the buffer.\n * @param {number} index - Index of the current element.\n */\n\n /**\n * Apply given callback
to all elements of the buffer.\n * @memberof Tinode.CBuffer#\n *\n * @param {Tinode.ForEachCallbackType} callback - Function to call for each element.\n * @param {number} startIdx - Optional index to start iterating from (inclusive).\n * @param {number} beforeIdx - Optional index to stop iterating before (exclusive).\n * @param {Object} context - calling context (i.e. value of this
in callback)\n */\n forEach(callback, startIdx, beforeIdx, context) {\n startIdx = startIdx | 0;\n beforeIdx = beforeIdx || this.buffer.length;\n\n for (let i = startIdx; i < beforeIdx; i++) {\n callback.call(context, this.buffer[i],\n (i > startIdx ? this.buffer[i - 1] : undefined),\n (i < beforeIdx - 1 ? this.buffer[i + 1] : undefined), i);\n }\n }\n\n /**\n * Find element in buffer using buffer's comparison function.\n * @memberof Tinode.CBuffer#\n *\n * @param {Object} elem - element to find.\n * @param {boolean=} nearest - when true and exact match is not found, return the nearest element (insertion point).\n * @returns {number} index of the element in the buffer or -1.\n */\n find(elem, nearest) {\n const {\n idx\n } = this.#findNearest(elem, this.buffer, !nearest);\n return idx;\n }\n\n /**\n * Callback for filtering the buffer. See {@link Tinode.CBuffer#filter}.\n * @callback FilterCallbackType\n * @memberof Tinode.CBuffer#\n * @param {Object} elem - Current element of the buffer.\n * @param {number} index - Index of the current element.\n * @returns {boolen} true
to keep the element, false
to remove.\n */\n\n /**\n * Remove all elements that do not pass the test implemented by the provided callback function.\n * @memberof Tinode.CBuffer#\n *\n * @param {Tinode.FilterCallbackType} callback - Function to call for each element.\n * @param {Object} context - calling context (i.e. value of this
in the callback)\n */\n filter(callback, context) {\n let count = 0;\n for (let i = 0; i < this.buffer.length; i++) {\n if (callback.call(context, this.buffer[i], i)) {\n this.buffer[count] = this.buffer[i];\n count++;\n }\n }\n\n this.buffer.splice(count);\n }\n\n /**\n * Check if buffer is empty.\n * @returns {boolean} true
if the buffer is empty, false
otherwise.\n */\n isEmpty() {\n return this.buffer.length == 0;\n }\n}\n","/**\n * @file Topic management.\n *\n * @copyright 2015-2022 Tinode LLC.\n */\n'use strict';\n\nimport AccessMode from './access-mode.js';\nimport CBuffer from './cbuffer.js';\nimport CommError from './comm-error.js';\nimport * as Const from './config.js';\nimport Drafty from './drafty.js';\nimport MetaGetBuilder from './meta-builder.js';\nimport {\n mergeObj,\n mergeToCache,\n normalizeArray\n} from './utils.js';\n\n/**\n * Topic is a class representing a logical communication channel.\n */\nexport class Topic {\n /**\n * @callback onData\n * @param {Data} data - Data packet\n */\n\n /**\n * Create topic.\n * @param {string} name - Name of the topic to create.\n * @param {Object=} callbacks - Object with various event callbacks.\n * @param {onData} callbacks.onData - Callback which receives a {data}
message.\n * @param {callback} callbacks.onMeta - Callback which receives a {meta}
message.\n * @param {callback} callbacks.onPres - Callback which receives a {pres}
message.\n * @param {callback} callbacks.onInfo - Callback which receives an {info}
message.\n * @param {callback} callbacks.onMetaDesc - Callback which receives changes to topic desctioption {@link desc}.\n * @param {callback} callbacks.onMetaSub - Called for a single subscription record change.\n * @param {callback} callbacks.onSubsUpdated - Called after a batch of subscription changes have been recieved and cached.\n * @param {callback} callbacks.onDeleteTopic - Called after the topic is deleted.\n * @param {callback} callbacls.onAllMessagesReceived - Called when all requested {data}
messages have been recived.\n */\n constructor(name, callbacks) {\n // Parent Tinode object.\n this._tinode = null;\n\n // Server-provided data, locally immutable.\n // topic name\n this.name = name;\n // Timestamp when the topic was created.\n this.created = null;\n // Timestamp when the topic was last updated.\n this.updated = null;\n // Timestamp of the last messages\n this.touched = new Date(0);\n // Access mode, see AccessMode\n this.acs = new AccessMode(null);\n // Per-topic private data (accessible by current user only).\n this.private = null;\n // Per-topic public data (accessible by all users).\n this.public = null;\n // Per-topic system-provided data (accessible by all users).\n this.trusted = null;\n\n // Locally cached data\n // Subscribed users, for tracking read/recv/msg notifications.\n this._users = {};\n\n // Current value of locally issued seqId, used for pending messages.\n this._queuedSeqId = Const.LOCAL_SEQID;\n\n // The maximum known {data.seq} value.\n this._maxSeq = 0;\n // The minimum known {data.seq} value.\n this._minSeq = 0;\n // Indicator that the last request for earlier messages returned 0.\n this._noEarlierMsgs = false;\n // The maximum known deletion ID.\n this._maxDel = 0;\n // Timer object used to send 'recv' notifications.\n this._recvNotificationTimer = null;\n\n // User discovery tags\n this._tags = [];\n // Credentials such as email or phone number.\n this._credentials = [];\n // Message versions cache (e.g. for edited messages).\n // Keys: original message seq ID.\n // Values: CBuffers containing newer versions of the original message\n // ordered by seq id.\n this._messageVersions = {};\n // Message cache, sorted by message seq values, from old to new.\n this._messages = new CBuffer((a, b) => {\n return a.seq - b.seq;\n }, true);\n // Boolean, true if the topic is currently live\n this._attached = false;\n // Timestap of the most recently updated subscription.\n this._lastSubsUpdate = new Date(0);\n // Topic created but not yet synced with the server. Used only during initialization.\n this._new = true;\n // The topic is deleted at the server, this is a local copy.\n this._deleted = false;\n\n // Timer used to trgger {leave} request after a delay.\n this._delayedLeaveTimer = null;\n\n // Callbacks\n if (callbacks) {\n this.onData = callbacks.onData;\n this.onMeta = callbacks.onMeta;\n this.onPres = callbacks.onPres;\n this.onInfo = callbacks.onInfo;\n // A single desc update;\n this.onMetaDesc = callbacks.onMetaDesc;\n // A single subscription record;\n this.onMetaSub = callbacks.onMetaSub;\n // All subscription records received;\n this.onSubsUpdated = callbacks.onSubsUpdated;\n this.onTagsUpdated = callbacks.onTagsUpdated;\n this.onCredsUpdated = callbacks.onCredsUpdated;\n this.onDeleteTopic = callbacks.onDeleteTopic;\n this.onAllMessagesReceived = callbacks.onAllMessagesReceived;\n }\n }\n\n // Static methods.\n\n /**\n * Determine topic type from topic's name: grp, p2p, me, fnd, sys.\n *\n * @param {string} name - Name of the topic to test.\n * @returns {string} One of \"me\"
, \"fnd\"
, \"sys\"
, \"grp\"
,\n * \"p2p\"
or undefined
.\n */\n static topicType(name) {\n const types = {\n 'me': Const.TOPIC_ME,\n 'fnd': Const.TOPIC_FND,\n 'grp': Const.TOPIC_GRP,\n 'new': Const.TOPIC_GRP,\n 'nch': Const.TOPIC_GRP,\n 'chn': Const.TOPIC_GRP,\n 'usr': Const.TOPIC_P2P,\n 'sys': Const.TOPIC_SYS\n };\n return types[(typeof name == 'string') ? name.substring(0, 3) : 'xxx'];\n }\n\n /**\n * Check if the given topic name is a name of a 'me' topic.\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a 'me' topic, false
otherwise.\n */\n static isMeTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_ME;\n }\n\n /**\n * Check if the given topic name is a name of a group topic.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a group topic, false
otherwise.\n */\n static isGroupTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_GRP;\n }\n\n /**\n * Check if the given topic name is a name of a p2p topic.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p topic, false
otherwise.\n */\n static isP2PTopicName(name) {\n return Topic.topicType(name) == Const.TOPIC_P2P;\n }\n\n /**\n * Check if the given topic name is a name of a communication topic, i.e. P2P or group.\n * @static\n *\n * @param {string} name - Name of the topic to test.\n * @returns {boolean} true
if the name is a name of a p2p or group topic, false
otherwise.\n */\n static isCommTopicName(name) {\n return Topic.isP2PTopicName(name) || Topic.isGroupTopicName(name);\n }\n\n /**\n * Check if the topic name is a name of a new topic.\n * @static\n *\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a new topic, false
otherwise.\n */\n static isNewGroupTopicName(name) {\n return (typeof name == 'string') &&\n (name.substring(0, 3) == Const.TOPIC_NEW || name.substring(0, 3) == Const.TOPIC_NEW_CHAN);\n }\n\n /**\n * Check if the topic name is a name of a channel.\n * @static\n *\n * @param {string} name - topic name to check.\n * @returns {boolean} true
if the name is a name of a channel, false
otherwise.\n */\n static isChannelTopicName(name) {\n return (typeof name == 'string') &&\n (name.substring(0, 3) == Const.TOPIC_CHAN || name.substring(0, 3) == Const.TOPIC_NEW_CHAN);\n }\n\n /**\n * Check if the topic is subscribed.\n * @returns {boolean} True is topic is attached/subscribed, false otherwise.\n */\n isSubscribed() {\n return this._attached;\n }\n\n /**\n * Request topic to subscribe. Wrapper for {@link Tinode#subscribe}.\n *\n * @param {Tinode.GetQuery=} getParams - get query parameters.\n * @param {Tinode.SetParams=} setParams - set parameters.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n subscribe(getParams, setParams) {\n // Clear request to leave topic.\n clearTimeout(this._delayedLeaveTimer);\n this._delayedLeaveTimer = null;\n\n // If the topic is already subscribed, return resolved promise\n if (this._attached) {\n return Promise.resolve(this);\n }\n\n // Send subscribe message, handle async response.\n // If topic name is explicitly provided, use it. If no name, then it's a new group topic,\n // use \"new\".\n return this._tinode.subscribe(this.name || Const.TOPIC_NEW, getParams, setParams).then(ctrl => {\n if (ctrl.code >= 300) {\n // Do nothing if subscription status has not changed.\n return ctrl;\n }\n\n this._attached = true;\n this._deleted = false;\n this.acs = (ctrl.params && ctrl.params.acs) ? ctrl.params.acs : this.acs;\n\n // Set topic name for new topics and add it to cache.\n if (this._new) {\n delete this._new;\n\n if (this.name != ctrl.topic) {\n // Name may change new123456 -> grpAbCdEf. Remove from cache under the old name.\n this._cacheDelSelf();\n this.name = ctrl.topic;\n }\n this._cachePutSelf();\n\n this.created = ctrl.ts;\n this.updated = ctrl.ts;\n\n if (this.name != Const.TOPIC_ME && this.name != Const.TOPIC_FND) {\n // Add the new topic to the list of contacts maintained by the 'me' topic.\n const me = this._tinode.getMeTopic();\n if (me.onMetaSub) {\n me.onMetaSub(this);\n }\n if (me.onSubsUpdated) {\n me.onSubsUpdated([this.name], 1);\n }\n }\n\n if (setParams && setParams.desc) {\n setParams.desc._noForwarding = true;\n this._processMetaDesc(setParams.desc);\n }\n }\n return ctrl;\n });\n }\n\n /**\n * Create a draft of a message without sending it to the server.\n * @memberof Tinode.Topic#\n *\n * @param {string | Object} data - Content to wrap in a draft.\n * @param {boolean=} noEcho - If true
server will not echo message back to originating\n * session. Otherwise the server will send a copy of the message to sender.\n *\n * @returns {Object} message draft.\n */\n createMessage(data, noEcho) {\n return this._tinode.createMessage(this.name, data, noEcho);\n }\n\n /**\n * Immediately publish data to topic. Wrapper for {@link Tinode#publish}.\n * @memberof Tinode.Topic#\n *\n * @param {string | Object} data - Message to publish, either plain string or a Drafty object.\n * @param {boolean=} noEcho - If true
server will not echo message back to originating\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n publish(data, noEcho) {\n return this.publishMessage(this.createMessage(data, noEcho));\n }\n\n /**\n * Publish message created by {@link Tinode.Topic#createMessage}.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub - {data} object to publish. Must be created by {@link Tinode.Topic#createMessage}\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n publishMessage(pub) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot publish on inactive topic\"));\n }\n if (this._sending) {\n return Promise.reject(new Error(\"The message is already being sent\"));\n }\n\n // Send data.\n pub._sending = true;\n pub._failed = false;\n\n // Extract refereces to attachments and out of band image records.\n let attachments = null;\n if (Drafty.hasEntities(pub.content)) {\n attachments = [];\n Drafty.entities(pub.content, data => {\n if (data) {\n if (data.ref) {\n attachments.push(data.ref);\n }\n if (data.preref) {\n attachments.push(data.preref);\n }\n }\n });\n if (attachments.length == 0) {\n attachments = null;\n }\n }\n\n return this._tinode.publishMessage(pub, attachments).then(ctrl => {\n pub._sending = false;\n pub.ts = ctrl.ts;\n this.swapMessageId(pub, ctrl.params.seq);\n this._maybeUpdateMessageVersionsCache(pub);\n this._routeData(pub);\n return ctrl;\n }).catch(err => {\n this._tinode.logger(\"WARNING: Message rejected by the server\", err);\n pub._sending = false;\n pub._failed = true;\n if (this.onData) {\n this.onData();\n }\n });\n }\n\n /**\n * Add message to local message cache, send to the server when the promise is resolved.\n * If promise is null or undefined, the message will be sent immediately.\n * The message is sent when the\n * The message should be created by {@link Tinode.Topic#createMessage}.\n * This is probably not the final API.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub - Message to use as a draft.\n * @param {Promise} prom - Message will be sent when this promise is resolved, discarded if rejected.\n *\n * @returns {Promise} derived promise.\n */\n publishDraft(pub, prom) {\n const seq = pub.seq || this._getQueuedSeqId();\n if (!pub._noForwarding) {\n // The 'seq', 'ts', and 'from' are added to mimic {data}. They are removed later\n // before the message is sent.\n pub._noForwarding = true;\n pub.seq = seq;\n pub.ts = new Date();\n pub.from = this._tinode.getCurrentUserID();\n\n // Don't need an echo message because the message is added to local cache right away.\n pub.noecho = true;\n // Add to cache.\n this._messages.put(pub);\n this._tinode._db.addMessage(pub);\n\n if (this.onData) {\n this.onData(pub);\n }\n }\n // If promise is provided, send the queued message when it's resolved.\n // If no promise is provided, create a resolved one and send immediately.\n return (prom || Promise.resolve())\n .then(_ => {\n if (pub._cancelled) {\n return {\n code: 300,\n text: \"cancelled\"\n };\n }\n return this.publishMessage(pub);\n }).catch(err => {\n this._tinode.logger(\"WARNING: Message draft rejected\", err);\n pub._sending = false;\n pub._failed = true;\n pub._fatal = err instanceof CommError ? (err.code >= 400 && err.code < 500) : false;\n if (this.onData) {\n this.onData();\n }\n // Rethrow to let caller know that the operation failed.\n throw err;\n });\n }\n\n /**\n * Leave the topic, optionally unsibscribe. Leaving the topic means the topic will stop\n * receiving updates from the server. Unsubscribing will terminate user's relationship with the topic.\n * Wrapper for {@link Tinode#leave}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean=} unsub - If true, unsubscribe, otherwise just leave.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n leave(unsub) {\n // It's possible to unsubscribe (unsub==true) from inactive topic.\n if (!this._attached && !unsub) {\n return Promise.reject(new Error(\"Cannot leave inactive topic\"));\n }\n\n // Send a 'leave' message, handle async response\n return this._tinode.leave(this.name, unsub).then(ctrl => {\n this._resetSub();\n if (unsub) {\n this._gone();\n }\n return ctrl;\n });\n }\n\n /**\n * Leave the topic, optionally unsibscribe after a delay. Leaving the topic means the topic will stop\n * receiving updates from the server. Unsubscribing will terminate user's relationship with the topic.\n * Wrapper for {@link Tinode#leave}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} unsub - If true, unsubscribe, otherwise just leave.\n * @param {number} delay - time in milliseconds to delay leave request.\n */\n leaveDelayed(unsub, delay) {\n clearTimeout(this._delayedLeaveTimer);\n this._delayedLeaveTimer = setTimeout(_ => {\n this._delayedLeaveTimer = null;\n this.leave(unsub)\n }, delay);\n }\n\n /**\n * Request topic metadata from the server.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.GetQuery} request parameters\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n getMeta(params) {\n // Send {get} message, return promise.\n return this._tinode.getMeta(this.name, params);\n }\n\n /**\n * Request more messages from the server\n * @memberof Tinode.Topic#\n *\n * @param {number} limit number of messages to get.\n * @param {boolean} forward if true, request newer messages.\n */\n getMessagesPage(limit, forward) {\n let query = forward ?\n this.startMetaQuery().withLaterData(limit) :\n this.startMetaQuery().withEarlierData(limit);\n\n // First try fetching from DB, then from the server.\n return this._loadMessages(this._tinode._db, query.extract('data'))\n .then((count) => {\n if (count == limit) {\n // Got enough messages from local cache.\n return Promise.resolve({\n topic: this.name,\n code: 200,\n params: {\n count: count\n }\n });\n }\n\n // Reduce the count of requested messages.\n limit -= count;\n // Update query with new values loaded from DB.\n query = forward ? this.startMetaQuery().withLaterData(limit) :\n this.startMetaQuery().withEarlierData(limit);\n let promise = this.getMeta(query.build());\n if (!forward) {\n promise = promise.then(ctrl => {\n if (ctrl && ctrl.params && !ctrl.params.count) {\n this._noEarlierMsgs = true;\n }\n });\n }\n return promise;\n });\n }\n /**\n * Update topic metadata.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.SetParams} params parameters to update.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n setMeta(params) {\n if (params.tags) {\n params.tags = normalizeArray(params.tags);\n }\n // Send Set message, handle async response.\n return this._tinode.setMeta(this.name, params)\n .then(ctrl => {\n if (ctrl && ctrl.code >= 300) {\n // Not modified\n return ctrl;\n }\n\n if (params.sub) {\n params.sub.topic = this.name;\n if (ctrl.params && ctrl.params.acs) {\n params.sub.acs = ctrl.params.acs;\n params.sub.updated = ctrl.ts;\n }\n if (!params.sub.user) {\n // This is a subscription update of the current user.\n // Assign user ID otherwise the update will be ignored by _processMetaSubs.\n params.sub.user = this._tinode.getCurrentUserID();\n if (!params.desc) {\n // Force update to topic's asc.\n params.desc = {};\n }\n }\n params.sub._noForwarding = true;\n this._processMetaSubs([params.sub]);\n }\n\n if (params.desc) {\n if (ctrl.params && ctrl.params.acs) {\n params.desc.acs = ctrl.params.acs;\n params.desc.updated = ctrl.ts;\n }\n this._processMetaDesc(params.desc);\n }\n\n if (params.tags) {\n this._processMetaTags(params.tags);\n }\n if (params.cred) {\n this._processMetaCreds([params.cred], true);\n }\n\n return ctrl;\n });\n }\n /**\n * Update access mode of the current user or of another topic subsriber.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - UID of the user to update or null to update current user.\n * @param {string} update - the update value, full or delta.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n updateMode(uid, update) {\n const user = uid ? this.subscriber(uid) : null;\n const am = user ?\n user.acs.updateGiven(update).getGiven() :\n this.getAccessMode().updateWant(update).getWant();\n\n return this.setMeta({\n sub: {\n user: uid,\n mode: am\n }\n });\n }\n /**\n * Create new topic subscription. Wrapper for {@link Tinode#setMeta}.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - ID of the user to invite\n * @param {string=} mode - Access mode. null
means to use default.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n invite(uid, mode) {\n return this.setMeta({\n sub: {\n user: uid,\n mode: mode\n }\n });\n }\n /**\n * Archive or un-archive the topic. Wrapper for {@link Tinode#setMeta}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} arch - true to archive the topic, false otherwise.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n archive(arch) {\n if (this.private && (!this.private.arch == !arch)) {\n return Promise.resolve(arch);\n }\n return this.setMeta({\n desc: {\n private: {\n arch: arch ? true : Const.DEL_CHAR\n }\n }\n });\n }\n /**\n * Delete messages. Hard-deleting messages requires Owner permission.\n * Wrapper for {@link Tinode#delMessages}.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.DelRange[]} ranges - Ranges of message IDs to delete.\n * @param {boolean=} hard - Hard or soft delete\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessages(ranges, hard) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete messages in inactive topic\"));\n }\n\n // Sort ranges in accending order by low, the descending by hi.\n ranges.sort((r1, r2) => {\n if (r1.low < r2.low) {\n return true;\n }\n if (r1.low == r2.low) {\n return !r2.hi || (r1.hi >= r2.hi);\n }\n return false;\n });\n\n // Remove pending messages from ranges possibly clipping some ranges.\n let tosend = ranges.reduce((out, r) => {\n if (r.low < Const.LOCAL_SEQID) {\n if (!r.hi || r.hi < Const.LOCAL_SEQID) {\n out.push(r);\n } else {\n // Clip hi to max allowed value.\n out.push({\n low: r.low,\n hi: this._maxSeq + 1\n });\n }\n }\n return out;\n }, []);\n\n // Send {del} message, return promise\n let result;\n if (tosend.length > 0) {\n result = this._tinode.delMessages(this.name, tosend, hard);\n } else {\n result = Promise.resolve({\n params: {\n del: 0\n }\n });\n }\n // Update local cache.\n return result.then(ctrl => {\n if (ctrl.params.del > this._maxDel) {\n this._maxDel = ctrl.params.del;\n }\n\n ranges.forEach((r) => {\n if (r.hi) {\n this.flushMessageRange(r.low, r.hi);\n } else {\n this.flushMessage(r.low);\n }\n });\n\n if (this.onData) {\n // Calling with no parameters to indicate the messages were deleted.\n this.onData();\n }\n return ctrl;\n });\n }\n /**\n * Delete all messages. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessagesAll(hardDel) {\n if (!this._maxSeq || this._maxSeq <= 0) {\n // There are no messages to delete.\n return Promise.resolve();\n }\n return this.delMessages([{\n low: 1,\n hi: this._maxSeq + 1,\n _all: true\n }], hardDel);\n }\n\n /**\n * Delete multiple messages defined by their IDs. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {Array.} list - list of seq IDs to delete.\n * @param {boolean=} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delMessagesList(list, hardDel) {\n // Sort the list in ascending order\n list.sort((a, b) => a - b);\n // Convert the array of IDs to ranges.\n let ranges = list.reduce((out, id) => {\n if (out.length == 0) {\n // First element.\n out.push({\n low: id\n });\n } else {\n let prev = out[out.length - 1];\n if ((!prev.hi && (id != prev.low + 1)) || (id > prev.hi)) {\n // New range.\n out.push({\n low: id\n });\n } else {\n // Expand existing range.\n prev.hi = prev.hi ? Math.max(prev.hi, id + 1) : id + 1;\n }\n }\n return out;\n }, []);\n // Send {del} message, return promise\n return this.delMessages(ranges, hardDel);\n }\n\n /**\n * Delete original message and edited variants. Hard-deleting messages requires Deleter permission.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - original seq ID of the message to delete.\n * @param {boolean=} hardDel - true if messages should be hard-deleted.\n *\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n delMessagesEdits(seq, hardDel) {\n const list = [seq];\n this.messageVersions(seq, msg => list.push(msg.seq));\n // Send {del} message, return promise\n return this.delMessagesList(list, hardDel);\n }\n\n /**\n * Delete topic. Requires Owner permission. Wrapper for {@link Tinode#delTopic}.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} hard - had-delete topic.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to the request.\n */\n delTopic(hard) {\n if (this._deleted) {\n // The topic is already deleted at the server, just remove from DB.\n this._gone();\n return Promise.resolve(null);\n }\n\n return this._tinode.delTopic(this.name, hard).then(ctrl => {\n this._deleted = true;\n this._resetSub();\n this._gone();\n return ctrl;\n });\n }\n /**\n * Delete subscription. Requires Share permission. Wrapper for {@link Tinode#delSubscription}.\n * @memberof Tinode.Topic#\n *\n * @param {string} user - ID of the user to remove subscription for.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n delSubscription(user) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete subscription in inactive topic\"));\n }\n // Send {del} message, return promise\n return this._tinode.delSubscription(this.name, user).then(ctrl => {\n // Remove the object from the subscription cache;\n delete this._users[user];\n // Notify listeners\n if (this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._users));\n }\n return ctrl;\n });\n }\n /**\n * Send a read/recv notification.\n * @memberof Tinode.Topic#\n *\n * @param {string} what - what notification to send: recv
, read
.\n * @param {number} seq - ID or the message read or received.\n */\n note(what, seq) {\n if (!this._attached) {\n // Cannot sending {note} on an inactive topic\".\n return;\n }\n\n // Update local cache with the new count.\n const user = this._users[this._tinode.getCurrentUserID()];\n let update = false;\n if (user) {\n // Self-subscription is found.\n if (!user[what] || user[what] < seq) {\n user[what] = seq;\n update = true;\n }\n } else {\n // Self-subscription is not found.\n update = (this[what] | 0) < seq;\n }\n\n if (update) {\n // Send notification to the server.\n this._tinode.note(this.name, what, seq);\n // Update locally cached contact with the new count.\n this._updateMyReadRecv(what, seq);\n\n if (this.acs != null && !this.acs.isMuted()) {\n const me = this._tinode.getMeTopic();\n // Sent a notification to 'me' listeners.\n me._refreshContact(what, this);\n }\n }\n }\n\n /**\n * Send a 'recv' receipt. Wrapper for {@link Tinode#noteRecv}.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - ID of the message to aknowledge.\n */\n noteRecv(seq) {\n this.note('recv', seq);\n }\n /**\n * Send a 'read' receipt. Wrapper for {@link Tinode#noteRead}.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - ID of the message to aknowledge or 0/undefined to acknowledge the latest messages.\n */\n noteRead(seq) {\n seq = seq || this._maxSeq;\n if (seq > 0) {\n this.note('read', seq);\n }\n }\n /**\n * Send a key-press notification. Wrapper for {@link Tinode#noteKeyPress}.\n * @memberof Tinode.Topic#\n */\n noteKeyPress() {\n if (this._attached) {\n this._tinode.noteKeyPress(this.name);\n } else {\n this._tinode.logger(\"INFO: Cannot send notification in inactive topic\");\n }\n }\n /**\n * Send a notification than a video or audio message is . Wrapper for {@link Tinode#noteKeyPress}.\n * @memberof Tinode.Topic#\n * @param audioOnly - true if the recording is audio-only, false if it's a video recording.\n */\n noteRecording(audioOnly) {\n if (this._attached) {\n this._tinode.noteKeyPress(this.name, audioOnly ? 'kpa' : 'kpv');\n } else {\n this._tinode.logger(\"INFO: Cannot send notification in inactive topic\");\n }\n }\n\n /**\n * Send a {note what='call'}. Wrapper for {@link Tinode#videoCall}.\n * @memberof Tinode#\n *\n * @param {string} evt - Call event.\n * @param {int} seq - ID of the call message the event pertains to.\n * @param {string} payload - Payload associated with this event (e.g. SDP string).\n *\n * @returns {Promise} Promise (for some call events) which will\n * be resolved/rejected on receiving server reply\n */\n videoCall(evt, seq, payload) {\n if (!this._attached && !['ringing', 'hang-up'].includes(evt)) {\n // Cannot {call} on an inactive topic\".\n return;\n }\n return this._tinode.videoCall(this.name, seq, evt, payload);\n }\n\n // Update cached read/recv/unread counts for the current user.\n _updateMyReadRecv(what, seq, ts) {\n let oldVal, doUpdate = false;\n\n seq = seq | 0;\n this.seq = this.seq | 0;\n this.read = this.read | 0;\n this.recv = this.recv | 0;\n switch (what) {\n case 'recv':\n oldVal = this.recv;\n this.recv = Math.max(this.recv, seq);\n doUpdate = (oldVal != this.recv);\n break;\n case 'read':\n oldVal = this.read;\n this.read = Math.max(this.read, seq);\n doUpdate = (oldVal != this.read);\n break;\n case 'msg':\n oldVal = this.seq;\n this.seq = Math.max(this.seq, seq);\n if (!this.touched || this.touched < ts) {\n this.touched = ts;\n }\n doUpdate = (oldVal != this.seq);\n break;\n }\n\n // Sanity checks.\n if (this.recv < this.read) {\n this.recv = this.read;\n doUpdate = true;\n }\n if (this.seq < this.recv) {\n this.seq = this.recv;\n if (!this.touched || this.touched < ts) {\n this.touched = ts;\n }\n doUpdate = true;\n }\n this.unread = this.seq - this.read;\n return doUpdate;\n }\n /**\n * Get user description from global cache. The user does not need to be a\n * subscriber of this topic.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - ID of the user to fetch.\n * @return {Object} user description or undefined.\n */\n userDesc(uid) {\n // TODO: handle asynchronous requests\n const user = this._cacheGetUser(uid);\n if (user) {\n return user; // Promise.resolve(user)\n }\n }\n /**\n * Get description of the p2p peer from subscription cache.\n * @memberof Tinode.Topic#\n *\n * @return {Object} peer's description or undefined.\n */\n p2pPeerDesc() {\n if (!this.isP2PType()) {\n return undefined;\n }\n return this._users[this.name];\n }\n /**\n * Iterate over cached subscribers. If callback is undefined, use this.onMetaSub.\n * @memberof Tinode.Topic#\n *\n * @param {function} callback - Callback which will receive subscribers one by one.\n * @param {Object=} context - Value of `this` inside the `callback`.\n */\n subscribers(callback, context) {\n const cb = (callback || this.onMetaSub);\n if (cb) {\n for (let idx in this._users) {\n cb.call(context, this._users[idx], idx, this._users);\n }\n }\n }\n /**\n * Get a copy of cached tags.\n * @memberof Tinode.Topic#\n *\n * @return {Array.} a copy of tags\n */\n tags() {\n // Return a copy.\n return this._tags.slice(0);\n }\n /**\n * Get cached subscription for the given user ID.\n * @memberof Tinode.Topic#\n *\n * @param {string} uid - id of the user to query for\n * @return user description or undefined.\n */\n subscriber(uid) {\n return this._users[uid];\n }\n /**\n * Iterate over versions of a message: call callback
for each version (excluding original).\n * If callback
is undefined, does nothing.\n * @memberof Tinode.Topic#\n *\n * @param {number} origSeq - seq ID of the original message.\n * @param {Tinode.ForEachCallbackType} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {Object} context - Value of `this` inside the `callback`.\n */\n messageVersions(origSeq, callback, context) {\n if (!callback) {\n // No callback? We are done then.\n return;\n }\n const versions = this._messageVersions[origSeq];\n if (!versions) {\n return;\n }\n versions.forEach(callback, undefined, undefined, context);\n }\n /**\n * Iterate over cached messages: call callback
for each message in the range [sinceIdx, beforeIdx).\n * If callback
is undefined, use this.onData
.\n * @memberof Tinode.Topic#\n *\n * @param {Tinode.ForEachCallbackType} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {number} sinceId - Optional seqId to start iterating from (inclusive).\n * @param {number} beforeId - Optional seqId to stop iterating before it is reached (exclusive).\n * @param {Object} context - Value of `this` inside the `callback`.\n */\n messages(callback, sinceId, beforeId, context) {\n const cb = (callback || this.onData);\n if (cb) {\n const startIdx = typeof sinceId == 'number' ? this._messages.find({\n seq: sinceId\n }, true) : undefined;\n const beforeIdx = typeof beforeId == 'number' ? this._messages.find({\n seq: beforeId\n }, true) : undefined;\n if (startIdx != -1 && beforeIdx != -1) {\n // Step 1. Filter out all replacement messages and\n // save displayable messages in a temporary buffer.\n let msgs = [];\n this._messages.forEach((msg, unused1, unused2, i) => {\n if (this._isReplacementMsg(msg)) {\n // Skip replacements.\n return;\n }\n // In case the massage was edited, replace timestamp of the version with the original's timestamp.\n const latest = this.latestMsgVersion(msg.seq) || msg;\n if (!latest._origTs) {\n latest._origTs = latest.ts;\n latest._origSeq = latest.seq;\n latest.ts = msg.ts;\n latest.seq = msg.seq;\n }\n msgs.push({\n data: latest,\n idx: i\n });\n }, startIdx, beforeIdx, {});\n // Step 2. Loop over displayble messages invoking cb on each of them.\n msgs.forEach((val, i) => {\n cb.call(context, val.data,\n (i > 0 ? msgs[i - 1].data : undefined),\n (i < msgs.length - 1 ? msgs[i + 1].data : undefined), val.idx);\n });\n }\n }\n }\n /**\n * Get the message from cache by seq
.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - message seqId to search for.\n * @returns {Object} the message with the given seq
or undefined
, if no such message is found.\n */\n findMessage(seq) {\n const idx = this._messages.find({\n seq: seq\n });\n if (idx >= 0) {\n return this._messages.getAt(idx);\n }\n return undefined;\n }\n /**\n * Get the most recent message from cache. This method counts all messages, including deleted ranges.\n * @memberof Tinode.Topic#\n *\n * @returns {Object} the most recent cached message or undefined
, if no messages are cached.\n */\n latestMessage() {\n return this._messages.getLast();\n }\n /**\n * Get the latest version for message.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - original seq ID of the message.\n * @returns {Object} the latest version of the message or null if message not found.\n */\n latestMsgVersion(seq) {\n const versions = this._messageVersions[seq];\n return versions ? versions.getLast() : null;\n }\n /**\n * Get the maximum cached seq ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the greatest seq ID in cache.\n */\n maxMsgSeq() {\n return this._maxSeq;\n }\n /**\n * Get the minimum cached seq ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the smallest seq ID in cache or 0.\n */\n minMsgSeq() {\n return this._minSeq;\n }\n /**\n * Get the maximum deletion ID.\n * @memberof Tinode.Topic#\n *\n * @returns {number} the greatest deletion ID.\n */\n maxClearId() {\n return this._maxDel;\n }\n /**\n * Get the number of messages in the cache.\n * @memberof Tinode.Topic#\n *\n * @returns {number} count of cached messages.\n */\n messageCount() {\n return this._messages.length();\n }\n /**\n * Iterate over cached unsent messages. Wraps {@link Tinode.Topic#messages}.\n * @memberof Tinode.Topic#\n *\n * @param {function} callback - Callback which will receive messages one by one. See {@link Tinode.CBuffer#forEach}\n * @param {Object} context - Value of this
inside the callback
.\n */\n queuedMessages(callback, context) {\n if (!callback) {\n throw new Error(\"Callback must be provided\");\n }\n this.messages(callback, Const.LOCAL_SEQID, undefined, context);\n }\n /**\n * Get the number of topic subscribers who marked this message as either recv or read\n * Current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {string} what - what action to consider: received \"recv\"
or read \"read\"
.\n * @param {number} seq - ID or the message read or received.\n *\n * @returns {number} the number of subscribers who marked the message with the given ID as read or received.\n */\n msgReceiptCount(what, seq) {\n let count = 0;\n if (seq > 0) {\n const me = this._tinode.getCurrentUserID();\n for (let idx in this._users) {\n const user = this._users[idx];\n if (user.user !== me && user[what] >= seq) {\n count++;\n }\n }\n }\n return count;\n }\n /**\n * Get the number of topic subscribers who marked this message (and all older messages) as read.\n * The current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - message id to check.\n * @returns {number} number of subscribers who claim to have received the message.\n */\n msgReadCount(seq) {\n return this.msgReceiptCount('read', seq);\n }\n /**\n * Get the number of topic subscribers who marked this message (and all older messages) as received.\n * The current user is excluded from the count.\n * @memberof Tinode.Topic#\n *\n * @param {number} seq - Message id to check.\n * @returns {number} Number of subscribers who claim to have received the message.\n */\n msgRecvCount(seq) {\n return this.msgReceiptCount('recv', seq);\n }\n /**\n * Check if cached message IDs indicate that the server may have more messages.\n * @memberof Tinode.Topic#\n *\n * @param {boolean} newer - if true
, check for newer messages only.\n */\n msgHasMoreMessages(newer) {\n return newer ? this.seq > this._maxSeq :\n // _minSeq could be more than 1, but earlier messages could have been deleted.\n (this._minSeq > 1 && !this._noEarlierMsgs);\n }\n /**\n * Check if the given seq Id is id of the most recent message.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to check\n */\n isNewMessage(seqId) {\n return this._maxSeq <= seqId;\n }\n /**\n * Remove one message from local cache.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to remove from cache.\n * @returns {Message} removed message or undefined if such message was not found.\n */\n flushMessage(seqId) {\n const idx = this._messages.find({\n seq: seqId\n });\n delete this._messageVersions[seqId];\n if (idx >= 0) {\n this._tinode._db.remMessages(this.name, seqId);\n return this._messages.delAt(idx);\n }\n return undefined;\n }\n /**\n * Remove a range of messages from the local cache.\n * @memberof Tinode.Topic#\n *\n * @param {number} fromId seq ID of the first message to remove (inclusive).\n * @param {number} untilId seqID of the last message to remove (exclusive).\n *\n * @returns {Message[]} array of removed messages (could be empty).\n */\n flushMessageRange(fromId, untilId) {\n // Remove range from persistent cache.\n this._tinode._db.remMessages(this.name, fromId, untilId);\n\n // Remove all versions keyed by IDs in the range.\n for (let i = fromId; i < untilId; i++) {\n delete this._messageVersions[i];\n }\n\n // start, end: find insertion points (nearest == true).\n const since = this._messages.find({\n seq: fromId\n }, true);\n return since >= 0 ? this._messages.delRange(since, this._messages.find({\n seq: untilId\n }, true)) : [];\n }\n /**\n * Update message's seqId.\n * @memberof Tinode.Topic#\n *\n * @param {Object} pub message object.\n * @param {number} newSeqId new seq id for pub.\n */\n swapMessageId(pub, newSeqId) {\n const idx = this._messages.find(pub);\n const numMessages = this._messages.length();\n if (0 <= idx && idx < numMessages) {\n // Remove message with the old seq ID.\n this._messages.delAt(idx);\n this._tinode._db.remMessages(this.name, pub.seq);\n // Add message with the new seq ID.\n pub.seq = newSeqId;\n this._messages.put(pub);\n this._tinode._db.addMessage(pub);\n }\n }\n /**\n * Attempt to stop message from being sent.\n * @memberof Tinode.Topic#\n *\n * @param {number} seqId id of the message to stop sending and remove from cache.\n *\n * @returns {boolean} true
if message was cancelled, false
otherwise.\n */\n cancelSend(seqId) {\n const idx = this._messages.find({\n seq: seqId\n });\n if (idx >= 0) {\n const msg = this._messages.getAt(idx);\n const status = this.msgStatus(msg);\n if (status == Const.MESSAGE_STATUS_QUEUED ||\n status == Const.MESSAGE_STATUS_FAILED ||\n status == Const.MESSAGE_STATUS_FATAL) {\n this._tinode._db.remMessages(this.name, seqId);\n msg._cancelled = true;\n this._messages.delAt(idx);\n if (this.onData) {\n // Calling with no parameters to indicate the message was deleted.\n this.onData();\n }\n return true;\n }\n }\n return false;\n }\n /**\n * Get type of the topic: me, p2p, grp, fnd...\n * @memberof Tinode.Topic#\n *\n * @returns {string} One of 'me', 'p2p', 'grp', 'fnd', 'sys' or undefined
.\n */\n getType() {\n return Topic.topicType(this.name);\n }\n /**\n * Get current user's access mode of the topic.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.AccessMode} - user's access mode\n */\n getAccessMode() {\n return this.acs;\n }\n /**\n * Set current user's access mode of the topic.\n * @memberof Tinode.Topic#\n *\n * @param {AccessMode | Object} acs - access mode to set.\n */\n setAccessMode(acs) {\n return this.acs = new AccessMode(acs);\n }\n /**\n * Get topic's default access mode.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.DefAcs} - access mode, such as {auth: `RWP`, anon: `N`}.\n */\n getDefaultAccess() {\n return this.defacs;\n }\n /**\n * Initialize new meta {@link Tinode.GetQuery} builder. The query is attched to the current topic.\n * It will not work correctly if used with a different topic.\n * @memberof Tinode.Topic#\n *\n * @returns {Tinode.MetaGetBuilder} query attached to the current topic.\n */\n startMetaQuery() {\n return new MetaGetBuilder(this);\n }\n /**\n * Check if topic is archived, i.e. private.arch == true.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is archived, false
otherwise.\n */\n isArchived() {\n return this.private && !!this.private.arch;\n }\n /**\n * Check if topic is a 'me' topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a 'me' topic, false
otherwise.\n */\n isMeType() {\n return Topic.isMeTopicName(this.name);\n }\n /**\n * Check if topic is a channel.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a channel, false
otherwise.\n */\n isChannelType() {\n return Topic.isChannelTopicName(this.name);\n }\n /**\n * Check if topic is a group topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a group, false
otherwise.\n */\n isGroupType() {\n return Topic.isGroupTopicName(this.name);\n }\n /**\n * Check if topic is a p2p topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a p2p topic, false
otherwise.\n */\n isP2PType() {\n return Topic.isP2PTopicName(this.name);\n }\n /**\n * Check if topic is a communication topic, i.e. a group or p2p topic.\n * @memberof Tinode.Topic#\n *\n * @returns {boolean} - true
if topic is a p2p or group topic, false
otherwise.\n */\n isCommType() {\n return Topic.isCommTopicName(this.name);\n }\n /**\n * Get status (queued, sent, received etc) of a given message in the context\n * of this topic.\n * @memberof Tinode.Topic#\n *\n * @param {Message} msg - message to check for status.\n * @param {boolean} upd - update chached message status.\n *\n * @returns message status constant.\n */\n msgStatus(msg, upd) {\n let status = Const.MESSAGE_STATUS_NONE;\n if (this._tinode.isMe(msg.from)) {\n if (msg._sending) {\n status = Const.MESSAGE_STATUS_SENDING;\n } else if (msg._fatal || msg._cancelled) {\n status = Const.MESSAGE_STATUS_FATAL;\n } else if (msg._failed) {\n status = Const.MESSAGE_STATUS_FAILED;\n } else if (msg.seq >= Const.LOCAL_SEQID) {\n status = Const.MESSAGE_STATUS_QUEUED;\n } else if (this.msgReadCount(msg.seq) > 0) {\n status = Const.MESSAGE_STATUS_READ;\n } else if (this.msgRecvCount(msg.seq) > 0) {\n status = Const.MESSAGE_STATUS_RECEIVED;\n } else if (msg.seq > 0) {\n status = Const.MESSAGE_STATUS_SENT;\n }\n } else {\n status = Const.MESSAGE_STATUS_TO_ME;\n }\n\n if (upd && msg._status != status) {\n msg._status = status;\n this._tinode._db.updMessageStatus(this.name, msg.seq, status);\n }\n\n return status;\n }\n\n // Returns true if pub is meant to replace another message (e.g. original message was edited).\n _isReplacementMsg(pub) {\n return pub.head && pub.head.replace;\n }\n\n // If msg is a replacement for another message, save the msg in the message versions cache\n // as a newer version for the message it's supposed to replace.\n _maybeUpdateMessageVersionsCache(msg) {\n if (!this._isReplacementMsg(msg)) {\n // Check if this message is the original in the chain of edits and if so\n // ensure all version have the same sender.\n if (this._messageVersions[msg.seq]) {\n // Remove versions with different 'from'.\n this._messageVersions[msg.seq].filter(version => version.from == msg.from);\n if (this._messageVersions[msg.seq].isEmpty()) {\n delete this._messageVersions[msg.seq];\n }\n }\n return;\n }\n\n const targetSeq = parseInt(msg.head.replace.split(':')[1]);\n if (targetSeq > msg.seq) {\n // Substitutes are supposed to have higher seq ids.\n return;\n }\n const targetMsg = this.findMessage(targetSeq);\n if (targetMsg && targetMsg.from != msg.from) {\n // Substitute cannot change the sender.\n return;\n }\n const versions = this._messageVersions[targetSeq] || new CBuffer((a, b) => {\n return a.seq - b.seq;\n }, true);\n versions.put(msg);\n this._messageVersions[targetSeq] = versions;\n }\n\n // Process data message\n _routeData(data) {\n if (data.content) {\n if (!this.touched || this.touched < data.ts) {\n this.touched = data.ts;\n this._tinode._db.updTopic(this);\n }\n }\n\n if (data.seq > this._maxSeq) {\n this._maxSeq = data.seq;\n this.msgStatus(data, true);\n // Ackn receiving the message.\n clearTimeout(this._recvNotificationTimer);\n this._recvNotificationTimer = setTimeout(_ => {\n this._recvNotificationTimer = null;\n this.noteRecv(this._maxSeq);\n }, Const.RECV_TIMEOUT);\n }\n\n if (data.seq < this._minSeq || this._minSeq == 0) {\n this._minSeq = data.seq;\n }\n\n const outgoing = ((!this.isChannelType() && !data.from) || this._tinode.isMe(data.from));\n\n if (data.head && data.head.webrtc && data.head.mime == Drafty.getContentType() && data.content) {\n // Rewrite VC body with info from the headers.\n data.content = Drafty.updateVideoCall(data.content, {\n state: data.head.webrtc,\n duration: data.head['webrtc-duration'],\n incoming: !outgoing,\n });\n }\n\n if (!data._noForwarding) {\n this._messages.put(data);\n this._tinode._db.addMessage(data);\n this._maybeUpdateMessageVersionsCache(data);\n }\n\n if (this.onData) {\n this.onData(data);\n }\n\n // Update locally cached contact with the new message count.\n const what = outgoing ? 'read' : 'msg';\n this._updateMyReadRecv(what, data.seq, data.ts);\n\n if (!outgoing && data.from) {\n // Mark messages as read by the sender.\n this._routeInfo({\n what: 'read',\n from: data.from,\n seq: data.seq,\n _noForwarding: true\n });\n }\n\n // Notify 'me' listeners of the change.\n this._tinode.getMeTopic()._refreshContact(what, this);\n }\n\n // Process metadata message\n _routeMeta(meta) {\n if (meta.desc) {\n this._processMetaDesc(meta.desc);\n }\n if (meta.sub && meta.sub.length > 0) {\n this._processMetaSubs(meta.sub);\n }\n if (meta.del) {\n this._processDelMessages(meta.del.clear, meta.del.delseq);\n }\n if (meta.tags) {\n this._processMetaTags(meta.tags);\n }\n if (meta.cred) {\n this._processMetaCreds(meta.cred);\n }\n if (this.onMeta) {\n this.onMeta(meta);\n }\n }\n // Process presence change message\n _routePres(pres) {\n let user, uid;\n switch (pres.what) {\n case 'del':\n // Delete cached messages.\n this._processDelMessages(pres.clear, pres.delseq);\n break;\n case 'on':\n case 'off':\n // Update online status of a subscription.\n user = this._users[pres.src];\n if (user) {\n user.online = pres.what == 'on';\n } else {\n this._tinode.logger(\"WARNING: Presence update for an unknown user\", this.name, pres.src);\n }\n break;\n case 'term':\n // Attachment to topic is terminated probably due to cluster rehashing.\n this._resetSub();\n break;\n case 'upd':\n // A topic subscriber has updated his description.\n // Issue {get sub} only if the current user has no p2p topics with the updated user (p2p name is not in cache).\n // Otherwise 'me' will issue a {get desc} request.\n if (pres.src && !this._tinode.isTopicCached(pres.src)) {\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n }\n break;\n case 'acs':\n uid = pres.src || this._tinode.getCurrentUserID();\n user = this._users[uid];\n if (!user) {\n // Update for an unknown user: notification of a new subscription.\n const acs = new AccessMode().updateAll(pres.dacs);\n if (acs && acs.mode != AccessMode._NONE) {\n user = this._cacheGetUser(uid);\n if (!user) {\n user = {\n user: uid,\n acs: acs\n };\n this.getMeta(this.startMetaQuery().withOneSub(undefined, uid).build());\n } else {\n user.acs = acs;\n }\n user.updated = new Date();\n this._processMetaSubs([user]);\n }\n } else {\n // Known user\n user.acs.updateAll(pres.dacs);\n // Update user's access mode.\n this._processMetaSubs([{\n user: uid,\n updated: new Date(),\n acs: user.acs\n }]);\n }\n break;\n default:\n this._tinode.logger(\"INFO: Ignored presence update\", pres.what);\n }\n\n if (this.onPres) {\n this.onPres(pres);\n }\n }\n // Process {info} message\n _routeInfo(info) {\n switch (info.what) {\n case 'recv':\n case 'read':\n const user = this._users[info.from];\n if (user) {\n user[info.what] = info.seq;\n if (user.recv < user.read) {\n user.recv = user.read;\n }\n }\n const msg = this.latestMessage();\n if (msg) {\n this.msgStatus(msg, true);\n }\n\n // If this is an update from the current user, update the cache with the new count.\n if (this._tinode.isMe(info.from) && !info._noForwarding) {\n this._updateMyReadRecv(info.what, info.seq);\n }\n\n // Notify 'me' listener of the status change.\n this._tinode.getMeTopic()._refreshContact(info.what, this);\n break;\n case 'kp':\n case 'kpa':\n case 'kpv':\n // Typing or audio/video recording notification. Do nothing.\n break;\n case 'call':\n // Do nothing here.\n break;\n default:\n this._tinode.logger(\"INFO: Ignored info update\", info.what);\n }\n\n if (this.onInfo) {\n this.onInfo(info);\n }\n }\n // Called by Tinode when meta.desc packet is received.\n // Called by 'me' topic on contact update (desc._noForwarding is true).\n _processMetaDesc(desc) {\n if (this.isP2PType()) {\n // Synthetic desc may include defacs for p2p topics which is useless.\n // Remove it.\n delete desc.defacs;\n\n // Update to p2p desc is the same as user update. Update cached user.\n this._tinode._db.updUser(this.name, desc.public);\n }\n\n // Copy parameters from desc object to this topic.\n mergeObj(this, desc);\n // Update persistent cache.\n this._tinode._db.updTopic(this);\n\n // Notify 'me' listener, if available:\n if (this.name !== Const.TOPIC_ME && !desc._noForwarding) {\n const me = this._tinode.getMeTopic();\n if (me.onMetaSub) {\n me.onMetaSub(this);\n }\n if (me.onSubsUpdated) {\n me.onSubsUpdated([this.name], 1);\n }\n }\n\n if (this.onMetaDesc) {\n this.onMetaDesc(this);\n }\n }\n // Called by Tinode when meta.sub is recived or in response to received\n // {ctrl} after setMeta-sub.\n _processMetaSubs(subs) {\n for (let idx in subs) {\n const sub = subs[idx];\n\n // Fill defaults.\n sub.online = !!sub.online;\n // Update timestamp of the most recent subscription update.\n this._lastSubsUpdate = new Date(Math.max(this._lastSubsUpdate, sub.updated));\n\n let user = null;\n if (!sub.deleted) {\n // If this is a change to user's own permissions, update them in topic too.\n // Desc will update 'me' topic.\n if (this._tinode.isMe(sub.user) && sub.acs) {\n this._processMetaDesc({\n updated: sub.updated,\n touched: sub.touched,\n acs: sub.acs\n });\n }\n user = this._updateCachedUser(sub.user, sub);\n } else {\n // Subscription is deleted, remove it from topic (but leave in Users cache)\n delete this._users[sub.user];\n user = sub;\n }\n\n if (this.onMetaSub) {\n this.onMetaSub(user);\n }\n }\n\n if (this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._users));\n }\n }\n // Called by Tinode when meta.tags is recived.\n _processMetaTags(tags) {\n if (tags.length == 1 && tags[0] == Const.DEL_CHAR) {\n tags = [];\n }\n this._tags = tags;\n if (this.onTagsUpdated) {\n this.onTagsUpdated(tags);\n }\n }\n // Do nothing for topics other than 'me'\n _processMetaCreds(creds) {}\n // Delete cached messages and update cached transaction IDs\n _processDelMessages(clear, delseq) {\n this._maxDel = Math.max(clear, this._maxDel);\n this.clear = Math.max(clear, this.clear);\n const topic = this;\n let count = 0;\n if (Array.isArray(delseq)) {\n delseq.forEach(function(range) {\n if (!range.hi) {\n count++;\n topic.flushMessage(range.low);\n } else {\n for (let i = range.low; i < range.hi; i++) {\n count++;\n topic.flushMessage(i);\n }\n }\n });\n }\n\n if (count > 0) {\n // this._updateDeletedRanges();\n\n if (this.onData) {\n this.onData();\n }\n }\n }\n // Topic is informed that the entire response to {get what=data} has been received.\n _allMessagesReceived(count) {\n\n if (this.onAllMessagesReceived) {\n this.onAllMessagesReceived(count);\n }\n }\n // Reset subscribed state\n _resetSub() {\n this._attached = false;\n }\n // This topic is either deleted or unsubscribed from.\n _gone() {\n this._messages.reset();\n this._tinode._db.remMessages(this.name);\n this._users = {};\n this.acs = new AccessMode(null);\n this.private = null;\n this.public = null;\n this.trusted = null;\n this._maxSeq = 0;\n this._minSeq = 0;\n this._attached = false;\n\n const me = this._tinode.getMeTopic();\n if (me) {\n me._routePres({\n _noForwarding: true,\n what: 'gone',\n topic: Const.TOPIC_ME,\n src: this.name\n });\n }\n if (this.onDeleteTopic) {\n this.onDeleteTopic();\n }\n }\n // Update global user cache and local subscribers cache.\n // Don't call this method for non-subscribers.\n _updateCachedUser(uid, obj) {\n // Fetch user object from the global cache.\n // This is a clone of the stored object\n let cached = this._cacheGetUser(uid);\n cached = mergeObj(cached || {}, obj);\n // Save to global cache\n this._cachePutUser(uid, cached);\n // Save to the list of topic subsribers.\n return mergeToCache(this._users, uid, cached);\n }\n // Get local seqId for a queued message.\n _getQueuedSeqId() {\n return this._queuedSeqId++;\n }\n\n // Load most recent messages from persistent cache.\n _loadMessages(db, params) {\n const {\n since,\n before,\n limit\n } = params || {};\n return db.readMessages(this.name, {\n since: since,\n before: before,\n limit: limit || Const.DEFAULT_MESSAGES_PAGE\n })\n .then(msgs => {\n msgs.forEach((data) => {\n if (data.seq > this._maxSeq) {\n this._maxSeq = data.seq;\n }\n if (data.seq < this._minSeq || this._minSeq == 0) {\n this._minSeq = data.seq;\n }\n this._messages.put(data);\n this._maybeUpdateMessageVersionsCache(data);\n });\n return msgs.length;\n });\n }\n // Push or {pres}: message received.\n _updateReceived(seq, act) {\n this.touched = new Date();\n this.seq = seq | 0;\n // Check if message is sent by the current user. If so it's been read already.\n if (!act || this._tinode.isMe(act)) {\n this.read = this.read ? Math.max(this.read, this.seq) : this.seq;\n this.recv = this.recv ? Math.max(this.read, this.recv) : this.read;\n }\n this.unread = this.seq - (this.read | 0);\n this._tinode._db.updTopic(this);\n }\n}\n\n/**\n * @class TopicMe - special case of {@link Tinode.Topic} for\n * managing data of the current user, including contact list.\n * @extends Tinode.Topic\n * @memberof Tinode\n *\n * @param {TopicMe.Callbacks} callbacks - Callbacks to receive various events.\n */\nexport class TopicMe extends Topic {\n onContactUpdate;\n\n constructor(callbacks) {\n super(Const.TOPIC_ME, callbacks);\n\n // me-specific callbacks\n if (callbacks) {\n this.onContactUpdate = callbacks.onContactUpdate;\n }\n }\n\n // Override the original Topic._processMetaDesc.\n _processMetaDesc(desc) {\n // Check if online contacts need to be turned off because P permission was removed.\n const turnOff = (desc.acs && !desc.acs.isPresencer()) && (this.acs && this.acs.isPresencer());\n\n // Copy parameters from desc object to this topic.\n mergeObj(this, desc);\n this._tinode._db.updTopic(this);\n // Update current user's record in the global cache.\n this._updateCachedUser(this._tinode._myUID, desc);\n\n // 'P' permission was removed. All topics are offline now.\n if (turnOff) {\n this._tinode.mapTopics((cont) => {\n if (cont.online) {\n cont.online = false;\n cont.seen = Object.assign(cont.seen || {}, {\n when: new Date()\n });\n this._refreshContact('off', cont);\n }\n });\n }\n\n if (this.onMetaDesc) {\n this.onMetaDesc(this);\n }\n }\n\n // Override the original Topic._processMetaSubs\n _processMetaSubs(subs) {\n let updateCount = 0;\n subs.forEach((sub) => {\n const topicName = sub.topic;\n // Don't show 'me' and 'fnd' topics in the list of contacts.\n if (topicName == Const.TOPIC_FND || topicName == Const.TOPIC_ME) {\n return;\n }\n sub.online = !!sub.online;\n\n let cont = null;\n if (sub.deleted) {\n cont = sub;\n this._tinode.cacheRemTopic(topicName);\n this._tinode._db.remTopic(topicName);\n } else {\n // Ensure the values are defined and are integers.\n if (typeof sub.seq != 'undefined') {\n sub.seq = sub.seq | 0;\n sub.recv = sub.recv | 0;\n sub.read = sub.read | 0;\n sub.unread = sub.seq - sub.read;\n }\n\n const topic = this._tinode.getTopic(topicName);\n if (topic._new) {\n delete topic._new;\n }\n\n cont = mergeObj(topic, sub);\n this._tinode._db.updTopic(cont);\n\n if (Topic.isP2PTopicName(topicName)) {\n this._cachePutUser(topicName, cont);\n this._tinode._db.updUser(topicName, cont.public);\n }\n // Notify topic of the update if it's an external update.\n if (!sub._noForwarding && topic) {\n sub._noForwarding = true;\n topic._processMetaDesc(sub);\n }\n }\n\n updateCount++;\n\n if (this.onMetaSub) {\n this.onMetaSub(cont);\n }\n });\n\n if (this.onSubsUpdated && updateCount > 0) {\n const keys = [];\n subs.forEach((s) => {\n keys.push(s.topic);\n });\n this.onSubsUpdated(keys, updateCount);\n }\n }\n\n // Called by Tinode when meta.sub is recived.\n _processMetaCreds(creds, upd) {\n if (creds.length == 1 && creds[0] == Const.DEL_CHAR) {\n creds = [];\n }\n if (upd) {\n creds.forEach((cr) => {\n if (cr.val) {\n // Adding a credential.\n let idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && el.val == cr.val;\n });\n if (idx < 0) {\n // Not found.\n if (!cr.done) {\n // Unconfirmed credential replaces previous unconfirmed credential of the same method.\n idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && !el.done;\n });\n if (idx >= 0) {\n // Remove previous unconfirmed credential.\n this._credentials.splice(idx, 1);\n }\n }\n this._credentials.push(cr);\n } else {\n // Found. Maybe change 'done' status.\n this._credentials[idx].done = cr.done;\n }\n } else if (cr.resp) {\n // Handle credential confirmation.\n const idx = this._credentials.findIndex((el) => {\n return el.meth == cr.meth && !el.done;\n });\n if (idx >= 0) {\n this._credentials[idx].done = true;\n }\n }\n });\n } else {\n this._credentials = creds;\n }\n if (this.onCredsUpdated) {\n this.onCredsUpdated(this._credentials);\n }\n }\n\n // Process presence change message\n _routePres(pres) {\n if (pres.what == 'term') {\n // The 'me' topic itself is detached. Mark as unsubscribed.\n this._resetSub();\n return;\n }\n\n if (pres.what == 'upd' && pres.src == Const.TOPIC_ME) {\n // Update to me's description. Request updated value.\n this.getMeta(this.startMetaQuery().withDesc().build());\n return;\n }\n\n const cont = this._tinode.cacheGetTopic(pres.src);\n if (cont) {\n switch (pres.what) {\n case 'on': // topic came online\n cont.online = true;\n break;\n case 'off': // topic went offline\n if (cont.online) {\n cont.online = false;\n cont.seen = Object.assign(cont.seen || {}, {\n when: new Date()\n });\n }\n break;\n case 'msg': // new message received\n cont._updateReceived(pres.seq, pres.act);\n break;\n case 'upd': // desc updated\n // Request updated subscription.\n this.getMeta(this.startMetaQuery().withLaterOneSub(pres.src).build());\n break;\n case 'acs': // access mode changed\n // If 'tgt' is not set then this is an update to the permissions of the current user.\n // Otherwise it's an update to group topic subscriber permissions while the topic is offline.\n // Just gnore it then.\n if (!pres.tgt) {\n if (cont.acs) {\n cont.acs.updateAll(pres.dacs);\n } else {\n cont.acs = new AccessMode().updateAll(pres.dacs);\n }\n cont.touched = new Date();\n }\n break;\n case 'ua':\n // user agent changed.\n cont.seen = {\n when: new Date(),\n ua: pres.ua\n };\n break;\n case 'recv':\n // user's other session marked some messges as received.\n pres.seq = pres.seq | 0;\n cont.recv = cont.recv ? Math.max(cont.recv, pres.seq) : pres.seq;\n break;\n case 'read':\n // user's other session marked some messages as read.\n pres.seq = pres.seq | 0;\n cont.read = cont.read ? Math.max(cont.read, pres.seq) : pres.seq;\n cont.recv = cont.recv ? Math.max(cont.read, cont.recv) : cont.recv;\n cont.unread = cont.seq - cont.read;\n break;\n case 'gone':\n // topic deleted or unsubscribed from.\n this._tinode.cacheRemTopic(pres.src);\n if (!cont._deleted) {\n cont._deleted = true;\n cont._attached = false;\n this._tinode._db.markTopicAsDeleted(pres.src, true);\n } else {\n this._tinode._db.remTopic(pres.src);\n }\n break;\n case 'del':\n // Update topic.del value.\n break;\n default:\n this._tinode.logger(\"INFO: Unsupported presence update in 'me'\", pres.what);\n }\n\n this._refreshContact(pres.what, cont);\n } else {\n if (pres.what == 'acs') {\n // New subscriptions and deleted/banned subscriptions have full\n // access mode (no + or - in the dacs string). Changes to known subscriptions are sent as\n // deltas, but they should not happen here.\n const acs = new AccessMode(pres.dacs);\n if (!acs || acs.mode == AccessMode._INVALID) {\n this._tinode.logger(\"ERROR: Invalid access mode update\", pres.src, pres.dacs);\n return;\n } else if (acs.mode == AccessMode._NONE) {\n this._tinode.logger(\"WARNING: Removing non-existent subscription\", pres.src, pres.dacs);\n return;\n } else {\n // New subscription. Send request for the full description.\n // Using .withOneSub (not .withLaterOneSub) to make sure IfModifiedSince is not set.\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n // Create a dummy entry to catch online status update.\n const dummy = this._tinode.getTopic(pres.src);\n dummy.online = false;\n dummy.acs = acs;\n this._tinode._db.updTopic(dummy);\n }\n } else if (pres.what == 'tags') {\n this.getMeta(this.startMetaQuery().withTags().build());\n } else if (pres.what == 'msg') {\n // Message received for un unknown (previously deleted) topic.\n this.getMeta(this.startMetaQuery().withOneSub(undefined, pres.src).build());\n // Create an entry to catch updates and messages.\n const dummy = this._tinode.getTopic(pres.src);\n dummy._deleted = false;\n this._tinode._db.updTopic(dummy);\n }\n\n this._refreshContact(pres.what, cont);\n }\n\n if (this.onPres) {\n this.onPres(pres);\n }\n }\n\n // Contact is updated, execute callbacks.\n _refreshContact(what, cont) {\n if (this.onContactUpdate) {\n this.onContactUpdate(what, cont);\n }\n }\n\n /**\n * Publishing to TopicMe is not supported. {@link Topic#publish} is overriden and thows an {Error} if called.\n * @memberof Tinode.TopicMe#\n * @throws {Error} Always throws an error.\n */\n publish() {\n return Promise.reject(new Error(\"Publishing to 'me' is not supported\"));\n }\n\n /**\n * Delete validation credential.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} topic - Name of the topic to delete\n * @param {string} user - User ID to remove.\n * @returns {Promise} Promise which will be resolved/rejected on receiving server reply.\n */\n delCredential(method, value) {\n if (!this._attached) {\n return Promise.reject(new Error(\"Cannot delete credential in inactive 'me' topic\"));\n }\n // Send {del} message, return promise\n return this._tinode.delCredential(method, value).then(ctrl => {\n // Remove deleted credential from the cache.\n const index = this._credentials.findIndex((el) => {\n return el.meth == method && el.val == value;\n });\n if (index > -1) {\n this._credentials.splice(index, 1);\n }\n // Notify listeners\n if (this.onCredsUpdated) {\n this.onCredsUpdated(this._credentials);\n }\n return ctrl;\n });\n }\n\n /**\n * @callback contactFilter\n * @param {Object} contact to check for inclusion.\n * @returns {boolean} true
if contact should be processed, false
to exclude it.\n */\n /**\n * Iterate over cached contacts.\n *\n * @function\n * @memberof Tinode.TopicMe#\n * @param {TopicMe.ContactCallback} callback - Callback to call for each contact.\n * @param {contactFilter=} filter - Optionally filter contacts; include all if filter is false-ish, otherwise\n * include those for which filter returns true-ish.\n * @param {Object=} context - Context to use for calling the `callback`, i.e. the value of `this` inside the callback.\n */\n contacts(callback, filter, context) {\n this._tinode.mapTopics((c, idx) => {\n if (c.isCommType() && (!filter || filter(c))) {\n callback.call(context, c, idx);\n }\n });\n }\n\n /**\n * Get a contact from cache.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to get, either a UID (for p2p topics) or a topic name.\n * @returns {Tinode.Contact} - Contact or `undefined`.\n */\n getContact(name) {\n return this._tinode.cacheGetTopic(name);\n }\n\n /**\n * Get access mode of a given contact from cache.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to get access mode for, either a UID (for p2p topics)\n * or a topic name; if missing, access mode for the 'me' topic itself.\n * @returns {string} - access mode, such as `RWP`.\n */\n getAccessMode(name) {\n if (name) {\n const cont = this._tinode.cacheGetTopic(name);\n return cont ? cont.acs : null;\n }\n return this.acs;\n }\n\n /**\n * Check if contact is archived, i.e. contact.private.arch == true.\n * @memberof Tinode.TopicMe#\n *\n * @param {string} name - Name of the contact to check archived status, either a UID (for p2p topics) or a topic name.\n * @returns {boolean} - true if contact is archived, false otherwise.\n */\n isArchived(name) {\n const cont = this._tinode.cacheGetTopic(name);\n return cont && cont.private && !!cont.private.arch;\n }\n\n /**\n * @typedef Tinode.Credential\n * @memberof Tinode\n * @type Object\n * @property {string} meth - validation method such as 'email' or 'tel'.\n * @property {string} val - credential value, i.e. 'jdoe@example.com' or '+17025551234'\n * @property {boolean} done - true if credential is validated.\n */\n /**\n * Get the user's credentials: email, phone, etc.\n * @memberof Tinode.TopicMe#\n *\n * @returns {Tinode.Credential[]} - array of credentials.\n */\n getCredentials() {\n return this._credentials;\n }\n}\n\n/**\n * Special case of {@link Tinode.Topic} for searching for contacts and group topics\n * @extends Tinode.Topic\n *\n */\nexport class TopicFnd extends Topic {\n // List of users and topics uid or topic_name -> Contact object)\n _contacts = {};\n\n /**\n * Create TopicFnd.\n *\n * @param {TopicFnd.Callbacks} callbacks - Callbacks to receive various events.\n */\n constructor(callbacks) {\n super(Const.TOPIC_FND, callbacks);\n }\n\n // Override the original Topic._processMetaSubs\n _processMetaSubs(subs) {\n let updateCount = Object.getOwnPropertyNames(this._contacts).length;\n // Reset contact list.\n this._contacts = {};\n for (let idx in subs) {\n let sub = subs[idx];\n const indexBy = sub.topic ? sub.topic : sub.user;\n\n sub = mergeToCache(this._contacts, indexBy, sub);\n updateCount++;\n\n if (this.onMetaSub) {\n this.onMetaSub(sub);\n }\n }\n\n if (updateCount > 0 && this.onSubsUpdated) {\n this.onSubsUpdated(Object.keys(this._contacts));\n }\n }\n\n /**\n * Publishing to TopicFnd is not supported. {@link Topic#publish} is overriden and thows an {Error} if called.\n * @memberof Tinode.TopicFnd#\n * @throws {Error} Always throws an error.\n */\n publish() {\n return Promise.reject(new Error(\"Publishing to 'fnd' is not supported\"));\n }\n\n /**\n * setMeta to TopicFnd resets contact list in addition to sending the message.\n * @memberof Tinode.TopicFnd#\n * @param {Tinode.SetParams} params parameters to update.\n * @returns {Promise} Promise to be resolved/rejected when the server responds to request.\n */\n setMeta(params) {\n return Object.getPrototypeOf(TopicFnd.prototype).setMeta.call(this, params).then(_ => {\n if (Object.keys(this._contacts).length > 0) {\n this._contacts = {};\n if (this.onSubsUpdated) {\n this.onSubsUpdated([]);\n }\n }\n });\n }\n\n /**\n * Iterate over found contacts. If callback is undefined, use {@link this.onMetaSub}.\n * @function\n * @memberof Tinode.TopicFnd#\n * @param {TopicFnd.ContactCallback} callback - Callback to call for each contact.\n * @param {Object} context - Context to use for calling the `callback`, i.e. the value of `this` inside the callback.\n */\n contacts(callback, context) {\n const cb = (callback || this.onMetaSub);\n if (cb) {\n for (let idx in this._contacts) {\n cb.call(context, this._contacts[idx], idx, this._contacts);\n }\n }\n }\n}\n"],"names":["root","factory","exports","module","define","amd","this","MAX_PREVIEW_DATA_SIZE","JSON_MIME_TYPE","ALLOWED_ENT_FIELDS","INLINE_STYLES","name","start","end","FMT_WEIGHT","ENTITY_TYPES","dataName","pack","val","test","url","re","slice","FORMAT_TAGS","AU","html_tag","md_tag","undefined","isVoid","BN","BR","CO","DL","EM","EX","FM","HD","HL","HT","IM","LN","MN","RW","QQ","ST","VC","VD","base64toObjectUrl","b64","contentType","logger","bin","atob","length","buf","ArrayBuffer","arr","Uint8Array","i","charCodeAt","URL","createObjectURL","Blob","type","err","message","base64toDataUrl","DECORATORS","open","_","close","data","props","href","target","id","act","ref","mime","Drafty","src","duration","size","tmpPreviewUrl","_tempPreview","previewUrl","downloadUrl","width","height","title","alt","state","preview","premime","poster","preref","txt","fmt","ent","chunkify","line","spans","chunks","span","at","push","chunk","tp","chld","children","toSpanTree","tree","last","draftyToTree","doc","Array","isArray","text","len","key","attachments","forEach","includes","sort","a","b","diff","indexOf","spansToTree","treeTopDown","node","child","parent","addNode","n","substring","att","subspans","inner","tag","treeToDrafty","keymap","c","Object","keys","newKey","transformer","context","dst","call","treeBottomUp","formatter","index","stack","values","pop","shortenTree","limit","tail","lightEntity","allow","copyEntData","lTrim","trimStart","shift","attachmentsToEnd","concat","draftify","startAt","plain","ranges","drafty","light","entries","dc","init","plainText","parse","content","lines","split","entityMap","entityIndex","blx","entities","block","original","re_start","re_end","result","exec","start_offset","lastIndexOf","end_offset","spannify","match","extracted","entity","offset","unique","idx","filter","el","extractEntities","map","s","append","first","second","insertImage","imageDesc","ex","refurl","bits","filename","urlPromise","_processing","then","insertVideo","videoDesc","urls","insertAudio","audioDesc","videoCall","audioOnly","aonly","updateVideoCall","params","assign","quote","header","uid","body","appendLineBreak","mention","appendLink","linkData","appendImage","appendAudio","attachFile","attachmentDesc","wrapInto","style","wrapAsForm","insertButton","actionType","actionValue","refUrl","appendButton","attachJSON","UNSAFE_toHTML","join","format","shorten","forwardedContent","replyContent","startsWith","forwarding","toPlainText","isPlainText","toMarkdown","def","isValid","txt_type","hasAttachments","callback","count","hasEntities","styles","sanitizeEntities","getDownloadUrl","entData","isProcessing","getPreviewUrl","getEntitySize","getEntityMimeType","tagName","attrValue","getContentType","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","__webpack_modules__","getter","__esModule","d","definition","o","defineProperty","enumerable","get","g","globalThis","Function","e","window","obj","prop","prototype","hasOwnProperty","r","Symbol","toStringTag","value","AccessMode","constructor","acs","given","decode","want","mode","side","flag","Error","str","_BITMASK","_NONE","bitmask","_JOIN","_READ","_WRITE","_PRES","_APPROVE","_SHARE","_DELETE","_OWNER","m0","bit","charAt","toUpperCase","encode","_INVALID","res","update","upd","action","val0","parts","a1","a2","toString","jsonHelper","setMode","m","updateMode","u","getMode","setGiven","updateGiven","getGiven","setWant","w","updateWant","getWant","getMissing","getExcessive","updateAll","isOwner","isPresencer","isMuted","isJoiner","isReader","isWriter","isApprover","isAdmin","isSharer","isDeleter","VERSION","LIBRARY","TOPIC_NEW","TOPIC_NEW_CHAN","TOPIC_ME","TOPIC_FND","TOPIC_GRP","LOCAL_SEQID","DEL_CHAR","CommError","code","super","jsonParseHelper","date","Date","isNaN","isUrlRelative","isValidDate","getTime","mergeObj","ignore","mergeToCache","cache","newval","simplify","getOwnPropertyNames","WebSocketProvider","XHRProvider","NETWORK_ERROR_TEXT","NETWORK_USER","NETWORK_USER_TEXT","makeBaseUrl","host","protocol","version","apiKey","Connection","static","secure","autoreconnect","initialized","config","version_","autoreconnect_","transport","setNetworkProviders","wsProvider","xhrProvider","l","connect","host_","force","Promise","reject","reconnect","disconnect","sendText","msg","isConnected","probe","backoffReset","clearTimeout","timeout","Math","pow","random","onAutoreconnectIteration","setTimeout","prom","catch","_lpURL","_poller","_sender","lp_poller","url_","resolve","poller","promiseCompleted","onreadystatechange","evt","readyState","status","pkt","JSON","responseText","ctrl","sid","send","onOpen","onMessage","onDisconnect","abort","sender","lp_sender","OPEN","conn","onerror","onopen","onclose","onmessage","NETWORK_ERROR","DB_NAME","IDBProvider","DB","db","disabled","onError","source","trx","transaction","event","error","objectStore","getAll","onsuccess","topic","initDatabase","req","onupgradeneeded","createObjectStore","keyPath","deleteDatabase","onblocked","isReady","updTopic","oncomplete","put","commit","markTopicAsDeleted","deleted","_deleted","remTopic","delete","IDBKeyRange","only","bound","Number","MAX_SAFE_INTEGER","mapTopics","deserializeTopic","updUser","pub","arguments","public","remUser","mapUsers","getUser","user","updSubscription","topicName","sub","mapSubscriptions","addMessage","add","updMessageStatus","seq","_status","remMessages","from","to","range","readMessages","query","since","before","openCursor","cursor","continue","f","tags","_tags","setAccessMode","read","unread","max","getAccessMode","setDatabaseProvider","idbProvider","IndexedDBProvider","LargeFileHelper","tinode","_tinode","_version","_apiKey","_authToken","getAuthToken","xhr","uploadWithBaseUrl","baseUrl","avatarFor","onProgress","onSuccess","onFailure","base","endsWith","instance","setRequestHeader","token","toResolve","toReject","upload","onprogress","lengthComputable","loaded","total","onload","response","statusText","onabort","form","FormData","set","getNextUniqueId","_secure","_host","download","relativeUrl","mimetype","relUrl","location","origin","searchParams","addURLParam","responseType","link","document","createElement","display","setAttribute","appendChild","click","removeChild","revokeObjectURL","reader","FileReader","readAsText","cancel","setNetworkProvider","MetaGetBuilder","what","updated","isP2PType","_lastSubsUpdate","withData","withLaterData","_maxSeq","withEarlierData","_minSeq","withDesc","ims","withLaterDesc","withSub","userOrTopic","opts","getType","withOneSub","withLaterOneSub","withLaterSub","withTags","withCred","withDel","withLaterDel","_maxDel","extract","build","CBuffer","buffer","compare_","unique_","elem","exact","pivot","found","splice","getAt","getLast","insert","delAt","delRange","reset","startIdx","beforeIdx","find","nearest","isEmpty","Topic","callbacks","created","touched","private","trusted","_users","_queuedSeqId","Const","_noEarlierMsgs","_recvNotificationTimer","_credentials","_messageVersions","_messages","_attached","_new","_delayedLeaveTimer","onData","onMeta","onPres","onInfo","onMetaDesc","onMetaSub","onSubsUpdated","onTagsUpdated","onCredsUpdated","onDeleteTopic","onAllMessagesReceived","topicType","isMeTopicName","isGroupTopicName","isP2PTopicName","isCommTopicName","isNewGroupTopicName","isChannelTopicName","isSubscribed","subscribe","getParams","setParams","_cacheDelSelf","_cachePutSelf","ts","me","getMeTopic","desc","_noForwarding","_processMetaDesc","createMessage","noEcho","publish","publishMessage","_sending","_failed","swapMessageId","_maybeUpdateMessageVersionsCache","_routeData","publishDraft","_getQueuedSeqId","getCurrentUserID","noecho","_db","_cancelled","_fatal","leave","unsub","_resetSub","_gone","leaveDelayed","delay","getMeta","getMessagesPage","forward","startMetaQuery","_loadMessages","promise","setMeta","out","t","trim","toLowerCase","item","pos","ary","normalizeArray","_processMetaSubs","_processMetaTags","cred","_processMetaCreds","subscriber","am","invite","archive","arch","delMessages","hard","r1","r2","low","hi","tosend","reduce","del","flushMessageRange","flushMessage","delMessagesAll","hardDel","_all","delMessagesList","list","prev","delMessagesEdits","messageVersions","delTopic","delSubscription","note","_updateMyReadRecv","_refreshContact","noteRecv","noteRead","noteKeyPress","noteRecording","payload","oldVal","doUpdate","recv","userDesc","_cacheGetUser","p2pPeerDesc","subscribers","cb","origSeq","versions","messages","sinceId","beforeId","msgs","unused1","unused2","_isReplacementMsg","latest","latestMsgVersion","_origTs","_origSeq","findMessage","latestMessage","maxMsgSeq","minMsgSeq","maxClearId","messageCount","queuedMessages","msgReceiptCount","msgReadCount","msgRecvCount","msgHasMoreMessages","newer","isNewMessage","seqId","fromId","untilId","newSeqId","numMessages","cancelSend","msgStatus","getDefaultAccess","defacs","isArchived","isMeType","isChannelType","isGroupType","isCommType","isMe","head","replace","targetSeq","parseInt","targetMsg","outgoing","webrtc","incoming","_routeInfo","_routeMeta","meta","_processDelMessages","clear","delseq","_routePres","pres","online","isTopicCached","dacs","info","subs","_updateCachedUser","creds","_allMessagesReceived","cached","_cachePutUser","_updateReceived","TopicMe","onContactUpdate","turnOff","_myUID","cont","seen","when","updateCount","cacheRemTopic","getTopic","cr","findIndex","meth","done","resp","cacheGetTopic","tgt","ua","dummy","delCredential","method","contacts","getContact","getCredentials","TopicFnd","_contacts","indexBy","getPrototypeOf","b64EncodeUnicode","btoa","encodeURIComponent","p1","String","fromCharCode","jsonBuildHelper","pad","sp","repeat","millis","getUTCMilliseconds","getUTCFullYear","getUTCMonth","getUTCDate","getUTCHours","getUTCMinutes","getUTCSeconds","rfc3339DateString","jsonLoggerHelper","WebSocket","XMLHttpRequest","indexedDB","chars","global","output","charCode","bc","bs","DBCache","initForNonBrowserApp","Tinode","_appName","_browser","_platform","_hwos","_humanLanguage","_loggingEnabled","_trimLongStrings","_authenticated","_login","_inPacketCount","_messageId","floor","_serverInfo","_deviceToken","_pendingPromises","_expirePromises","_connection","_persist","_cache","onComplete","appName","platform","navigator","product","reactnative","priority","tmp","substr","tokens","m2","v","minor","getBrowserInfo","userAgent","language","detectTransport","persist","all","dateString","_len","args","_key","console","log","onOK","errorText","stringify","onRawMessage","onNetworkProbe","onCtrlMessage","onMetaMessage","onDataMessage","onPresMessage","onInfoMessage","setInterval","expires","hello","clearInterval","func","_cacheDelUser","onLogin","credential","getVersion","getLibrary","isNullValue","clearStorage","initStorage","networkProbe","isAuthenticated","authorizeURL","parsed","account","scheme","secret","login","acc","tmpscheme","tmpsecret","extra","createAccount","createAccountBasic","username","password","updateAccountBasic","onConnect","setDeviceToken","dt","sent","loginBasic","uname","loginToken","requestResetAuthSecret","now","setAuthToken","dft","oobNotification","xfrom","finally","modeGiven","modeWant","delCurrentUser","newGroupTopicName","isChan","getFndTopic","getLargeFileHelper","getCurrentLogin","getServerInfo","report","getServerParam","defaultValue","enableLogging","enabled","trimLongStrings","setHumanLanguage","hl","isTopicOnline","getTopicAccessMode","wantAkn","onWebsocketOpen","MESSAGE_STATUS_NONE","MESSAGE_STATUS_QUEUED","MESSAGE_STATUS_SENDING","MESSAGE_STATUS_FAILED","MESSAGE_STATUS_FATAL","MESSAGE_STATUS_SENT","MESSAGE_STATUS_RECEIVED","MESSAGE_STATUS_READ","MESSAGE_STATUS_TO_ME","MAX_MESSAGE_SIZE","MAX_SUBSCRIBER_COUNT","MAX_TAG_COUNT","MAX_FILE_UPLOAD_SIZE","REQ_CRED_VALIDATORS","URI_TOPIC_ID_PREFIX"],"sourceRoot":""}
\ No newline at end of file
diff --git a/version.js b/version.js
index 3acf994..4409b3f 100644
--- a/version.js
+++ b/version.js
@@ -1 +1 @@
-export const PACKAGE_VERSION = "0.22.11";
+export const PACKAGE_VERSION = "0.22.12";