From 551b4c821fe670be75a5dafbf66a50ceec5e4fc7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Fabrice=20PENHO=C3=8BT?= Date: Fri, 3 Sep 2021 17:34:45 +0200 Subject: [PATCH] Nouvelle compilation du script du 1er exemple d'utilisation. --- public/JS/firstExample.app.js | 1074 +-------------------------------- 1 file changed, 30 insertions(+), 1044 deletions(-) diff --git a/public/JS/firstExample.app.js b/public/JS/firstExample.app.js index 39b6b3b..fb6d0e9 100644 --- a/public/JS/firstExample.app.js +++ b/public/JS/firstExample.app.js @@ -1,3 +1,11 @@ +/* + * ATTENTION: The "eval" devtool has been used (maybe by default in mode: "development"). + * This devtool is neither made for production nor for readable output files. + * It uses "eval()" calls to create a separate source file in the browser devtools. + * If you are trying to read the output file, select a different devtool (https://webpack.js.org/configuration/devtool/) + * or disable the default devtool with "devtool: false". + * If you are looking for production-ready output files, see mode: "production" (https://webpack.js.org/configuration/mode/). + */ /******/ (() => { // webpackBootstrap /******/ var __webpack_modules__ = ({ @@ -8,639 +16,7 @@ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; -__webpack_require__.r(__webpack_exports__); -/* harmony export */ __webpack_require__.d(__webpack_exports__, { -/* harmony export */ "orderBy": () => (/* binding */ orderBy), -/* harmony export */ "compare": () => (/* binding */ compare) -/* harmony export */ }); -var compareNumbers = function compareNumbers(numberA, numberB) { - if (numberA < numberB) { - return -1; - } - - if (numberA > numberB) { - return 1; - } - - return 0; -}; - -var RE_NUMBERS = /(^0x[\da-fA-F]+$|^([+-]?(?:\d+(?:\.\d*)?|\.\d+)(?:[eE][+-]?\d+)?(?!\.\d+)(?=\D|\s|$))|\d+)/g; -var RE_LEADING_OR_TRAILING_WHITESPACES = /^\s+|\s+$/g; // trim pre-post whitespace - -var RE_WHITESPACES = /\s+/g; // normalize all whitespace to single ' ' character - -var RE_INT_OR_FLOAT = /^[+-]?(?:\d+(?:\.\d*)?|\.\d+)(?:[eE][+-]?\d+)?$/; // identify integers and floats - -var RE_DATE = /(^([\w ]+,?[\w ]+)?[\w ]+,?[\w ]+\d+:\d+(:\d+)?[\w ]?|^\d{1,4}[/-]\d{1,4}[/-]\d{1,4}|^\w+, \w+ \d+, \d{4})/; // identify date strings - -var RE_LEADING_ZERO = /^0+[1-9]{1}[0-9]*$/; -var RE_UNICODE_CHARACTERS = /[^\x00-\x80]/; - -var compareUnicode = function compareUnicode(stringA, stringB) { - var result = stringA.localeCompare(stringB); - return result ? result / Math.abs(result) : 0; -}; - -var stringCompare = function stringCompare(stringA, stringB) { - if (stringA < stringB) { - return -1; - } - - if (stringA > stringB) { - return 1; - } - - return 0; -}; - -var compareChunks = function compareChunks(chunksA, chunksB) { - var lengthA = chunksA.length; - var lengthB = chunksB.length; - var size = Math.min(lengthA, lengthB); - - for (var i = 0; i < size; i++) { - var chunkA = chunksA[i]; - var chunkB = chunksB[i]; - - if (chunkA.normalizedString !== chunkB.normalizedString) { - if (chunkA.normalizedString === '' !== (chunkB.normalizedString === '')) { - // empty strings have lowest value - return chunkA.normalizedString === '' ? -1 : 1; - } - - if (chunkA.parsedNumber !== undefined && chunkB.parsedNumber !== undefined) { - // compare numbers - var result = compareNumbers(chunkA.parsedNumber, chunkB.parsedNumber); - - if (result === 0) { - // compare string value, if parsed numbers are equal - // Example: - // chunkA = { parsedNumber: 1, normalizedString: "001" } - // chunkB = { parsedNumber: 1, normalizedString: "01" } - // chunkA.parsedNumber === chunkB.parsedNumber - // chunkA.normalizedString < chunkB.normalizedString - return stringCompare(chunkA.normalizedString, chunkB.normalizedString); - } - - return result; - } else if (chunkA.parsedNumber !== undefined || chunkB.parsedNumber !== undefined) { - // number < string - return chunkA.parsedNumber !== undefined ? -1 : 1; - } else if (RE_UNICODE_CHARACTERS.test(chunkA.normalizedString + chunkB.normalizedString) && chunkA.normalizedString.localeCompare) { - // use locale comparison only if one of the chunks contains unicode characters - return compareUnicode(chunkA.normalizedString, chunkB.normalizedString); - } else { - // use common string comparison for performance reason - return stringCompare(chunkA.normalizedString, chunkB.normalizedString); - } - } - } // if the chunks are equal so far, the one which has more chunks is greater than the other one - - - if (lengthA > size || lengthB > size) { - return lengthA <= size ? -1 : 1; - } - - return 0; -}; - -var compareOtherTypes = function compareOtherTypes(valueA, valueB) { - if (!valueA.chunks ? valueB.chunks : !valueB.chunks) { - return !valueA.chunks ? 1 : -1; - } - - if (valueA.isNaN ? !valueB.isNaN : valueB.isNaN) { - return valueA.isNaN ? -1 : 1; - } - - if (valueA.isSymbol ? !valueB.isSymbol : valueB.isSymbol) { - return valueA.isSymbol ? -1 : 1; - } - - if (valueA.isObject ? !valueB.isObject : valueB.isObject) { - return valueA.isObject ? -1 : 1; - } - - if (valueA.isArray ? !valueB.isArray : valueB.isArray) { - return valueA.isArray ? -1 : 1; - } - - if (valueA.isFunction ? !valueB.isFunction : valueB.isFunction) { - return valueA.isFunction ? -1 : 1; - } - - if (valueA.isNull ? !valueB.isNull : valueB.isNull) { - return valueA.isNull ? -1 : 1; - } - - return 0; -}; - -var compareValues = function compareValues(valueA, valueB) { - if (valueA.value === valueB.value) { - return 0; - } - - if (valueA.parsedNumber !== undefined && valueB.parsedNumber !== undefined) { - return compareNumbers(valueA.parsedNumber, valueB.parsedNumber); - } - - if (valueA.chunks && valueB.chunks) { - return compareChunks(valueA.chunks, valueB.chunks); - } - - return compareOtherTypes(valueA, valueB); -}; - -var compareMultiple = function compareMultiple(recordA, recordB, orders) { - var indexA = recordA.index, - valuesA = recordA.values; - var indexB = recordB.index, - valuesB = recordB.values; - var length = valuesA.length; - var ordersLength = orders.length; - - for (var i = 0; i < length; i++) { - var order = i < ordersLength ? orders[i] : null; - - if (order && typeof order === 'function') { - var result = order(valuesA[i].value, valuesB[i].value); - - if (result) { - return result; - } - } else { - var _result = compareValues(valuesA[i], valuesB[i]); - - if (_result) { - return _result * (order === 'desc' ? -1 : 1); - } - } - } - - return indexA - indexB; -}; - -var createIdentifierFn = function createIdentifierFn(identifier) { - if (typeof identifier === 'function') { - // identifier is already a lookup function - return identifier; - } - - return function (value) { - if (Array.isArray(value)) { - var index = Number(identifier); - - if (Number.isInteger(index)) { - return value[index]; - } - } else if (value && typeof value === 'object' && typeof identifier !== 'function') { - return value[identifier]; - } - - return value; - }; -}; - -var stringify = function stringify(value) { - if (typeof value === 'boolean' || value instanceof Boolean) { - return Number(value).toString(); - } - - if (typeof value === 'number' || value instanceof Number) { - return value.toString(); - } - - if (value instanceof Date) { - return value.getTime().toString(); - } - - if (typeof value === 'string' || value instanceof String) { - return value.toLowerCase().replace(RE_LEADING_OR_TRAILING_WHITESPACES, ''); - } - - return ''; -}; - -var parseNumber = function parseNumber(value) { - if (value.length !== 0) { - var parsedNumber = Number(value); - - if (!Number.isNaN(parsedNumber)) { - return parsedNumber; - } - } - - return undefined; -}; - -var parseDate = function parseDate(value) { - if (RE_DATE.test(value)) { - var parsedDate = Date.parse(value); - - if (!Number.isNaN(parsedDate)) { - return parsedDate; - } - } - - return undefined; -}; - -var numberify = function numberify(value) { - var parsedNumber = parseNumber(value); - - if (parsedNumber !== undefined) { - return parsedNumber; - } - - return parseDate(value); -}; - -var createChunks = function createChunks(value) { - return value.replace(RE_NUMBERS, '\0$1\0').replace(/\0$/, '').replace(/^\0/, '').split('\0'); -}; - -var normalizeAlphaChunk = function normalizeAlphaChunk(chunk) { - return chunk.replace(RE_WHITESPACES, ' ').replace(RE_LEADING_OR_TRAILING_WHITESPACES, ''); -}; - -var normalizeNumericChunk = function normalizeNumericChunk(chunk, index, chunks) { - if (RE_INT_OR_FLOAT.test(chunk)) { - // don´t parse a number, if there´s a preceding decimal point - // to keep significance - // e.g. 1.0020, 1.020 - if (!RE_LEADING_ZERO.test(chunk) || index === 0 || chunks[index - 1] !== '.') { - return parseNumber(chunk) || 0; - } - } - - return undefined; -}; - -var createChunkMap = function createChunkMap(chunk, index, chunks) { - return { - parsedNumber: normalizeNumericChunk(chunk, index, chunks), - normalizedString: normalizeAlphaChunk(chunk) - }; -}; - -var createChunkMaps = function createChunkMaps(value) { - var chunksMaps = createChunks(value).map(createChunkMap); - return chunksMaps; -}; - -var isFunction = function isFunction(value) { - return typeof value === 'function'; -}; - -var isNaN = function isNaN(value) { - return Number.isNaN(value) || value instanceof Number && Number.isNaN(value.valueOf()); -}; - -var isNull = function isNull(value) { - return value === null; -}; - -var isObject = function isObject(value) { - return value !== null && typeof value === 'object' && !Array.isArray(value) && !(value instanceof Number) && !(value instanceof String) && !(value instanceof Boolean) && !(value instanceof Date); -}; - -var isSymbol = function isSymbol(value) { - return typeof value === 'symbol'; -}; - -var isUndefined = function isUndefined(value) { - return value === undefined; -}; - -var getMappedValueRecord = function getMappedValueRecord(value) { - if (typeof value === 'string' || value instanceof String || (typeof value === 'number' || value instanceof Number) && !isNaN(value) || typeof value === 'boolean' || value instanceof Boolean || value instanceof Date) { - var stringValue = stringify(value); - var parsedNumber = numberify(stringValue); - var chunks = createChunkMaps(parsedNumber ? "" + parsedNumber : stringValue); - return { - parsedNumber: parsedNumber, - chunks: chunks, - value: value - }; - } - - return { - isArray: Array.isArray(value), - isFunction: isFunction(value), - isNaN: isNaN(value), - isNull: isNull(value), - isObject: isObject(value), - isSymbol: isSymbol(value), - isUndefined: isUndefined(value), - value: value - }; -}; - -var getValueByIdentifier = function getValueByIdentifier(value, getValue) { - return getValue(value); -}; - -var getElementByIndex = function getElementByIndex(collection, index) { - return collection[index]; -}; - -var baseOrderBy = function baseOrderBy(collection, identifiers, orders) { - var identifierFns = identifiers.length ? identifiers.map(createIdentifierFn) : [function (value) { - return value; - }]; // temporary array holds elements with position and sort-values - - var mappedCollection = collection.map(function (element, index) { - var values = identifierFns.map(function (identifier) { - return getValueByIdentifier(element, identifier); - }).map(getMappedValueRecord); - return { - index: index, - values: values - }; - }); // iterate over values and compare values until a != b or last value reached - - mappedCollection.sort(function (recordA, recordB) { - return compareMultiple(recordA, recordB, orders); - }); - return mappedCollection.map(function (element) { - return getElementByIndex(collection, element.index); - }); -}; - -var getIdentifiers = function getIdentifiers(identifiers) { - if (!identifiers) { - return []; - } - - var identifierList = !Array.isArray(identifiers) ? [identifiers] : [].concat(identifiers); - - if (identifierList.some(function (identifier) { - return typeof identifier !== 'string' && typeof identifier !== 'number' && typeof identifier !== 'function'; - })) { - return []; - } - - return identifierList; -}; - -var getOrders = function getOrders(orders) { - if (!orders) { - return []; - } - - var orderList = !Array.isArray(orders) ? [orders] : [].concat(orders); - - if (orderList.some(function (order) { - return order !== 'asc' && order !== 'desc' && typeof order !== 'function'; - })) { - return []; - } - - return orderList; -}; - -/** - * Creates an array of elements, natural sorted by specified identifiers and - * the corresponding sort orders. This method implements a stable sort - * algorithm, which means the original sort order of equal elements is - * preserved. - * - * If `collection` is an array of primitives, `identifiers` may be unspecified. - * Otherwise, you should specify `identifiers` to sort by or `collection` will - * be returned unsorted. An identifier can expressed by: - * - * - an index position, if `collection` is a nested array, - * - a property name, if `collection` is an array of objects, - * - a function which returns a particular value from an element of a nested array or an array of objects. This function will be invoked by passing one element of `collection`. - * - * If `orders` is unspecified, all values are sorted in ascending order. - * Otherwise, specify an order of `'desc'` for descending or `'asc'` for - * ascending sort order of corresponding values. You may also specify a compare - * function for an order, which will be invoked by two arguments: - * `(valueA, valueB)`. It must return a number representing the sort order. - * - * @example - * - * import { orderBy } from 'natural-orderby'; - * - * const users = [ - * { - * username: 'Bamm-Bamm', - * ip: '192.168.5.2', - * datetime: 'Fri Jun 15 2018 16:48:00 GMT+0200 (CEST)' - * }, - * { - * username: 'Wilma', - * ip: '192.168.10.1', - * datetime: '14 Jun 2018 00:00:00 PDT' - * }, - * { - * username: 'dino', - * ip: '192.168.0.2', - * datetime: 'June 15, 2018 14:48:00' - * }, - * { - * username: 'Barney', - * ip: '192.168.1.1', - * datetime: 'Thu, 14 Jun 2018 07:00:00 GMT' - * }, - * { - * username: 'Pebbles', - * ip: '192.168.1.21', - * datetime: '15 June 2018 14:48 UTC' - * }, - * { - * username: 'Hoppy', - * ip: '192.168.5.10', - * datetime: '2018-06-15T14:48:00.000Z' - * }, - * ]; - * - * orderBy( - * users, - * [v => v.datetime, v => v.ip], - * ['desc', 'asc'] - * ); - * - * // => [ - * // { - * // username: 'dino', - * // ip: '192.168.0.2', - * // datetime: 'June 15, 2018 14:48:00', - * // }, - * // { - * // username: 'Pebbles', - * // ip: '192.168.1.21', - * // datetime: '15 June 2018 14:48 UTC', - * // }, - * // { - * // username: 'Bamm-Bamm', - * // ip: '192.168.5.2', - * // datetime: 'Fri Jun 15 2018 16:48:00 GMT+0200 (CEST)', - * // }, - * // { - * // username: 'Hoppy', - * // ip: '192.168.5.10', - * // datetime: '2018-06-15T14:48:00.000Z', - * // }, - * // { - * // username: 'Barney', - * // ip: '192.168.1.1', - * // datetime: 'Thu, 14 Jun 2018 07:00:00 GMT', - * // }, - * // { - * // username: 'Wilma', - * // ip: '192.168.10.1', - * // datetime: '14 Jun 2018 00:00:00 PDT', - * // }, - * // ] - */ -function orderBy(collection, identifiers, orders) { - if (!collection || !Array.isArray(collection)) { - return []; - } - - var validatedIdentifiers = getIdentifiers(identifiers); - var validatedOrders = getOrders(orders); - return baseOrderBy(collection, validatedIdentifiers, validatedOrders); -} - -var baseCompare = function baseCompare(options) { - return function (valueA, valueB) { - var a = getMappedValueRecord(valueA); - var b = getMappedValueRecord(valueB); - var result = compareValues(a, b); - return result * (options.order === 'desc' ? -1 : 1); - }; -}; - -var isValidOrder = function isValidOrder(value) { - return typeof value === 'string' && (value === 'asc' || value === 'desc'); -}; - -var getOptions = function getOptions(customOptions) { - var order = 'asc'; - - if (typeof customOptions === 'string' && isValidOrder(customOptions)) { - order = customOptions; - } else if (customOptions && typeof customOptions === 'object' && customOptions.order && isValidOrder(customOptions.order)) { - order = customOptions.order; - } - - return { - order: order - }; -}; - -/** - * Creates a compare function that defines the natural sort order considering - * the given `options` which may be passed to [`Array.prototype.sort()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort). - * - * If `options` or its property `order` is unspecified, values are sorted in - * ascending sort order. Otherwise, specify an order of `'desc'` for descending - * or `'asc'` for ascending sort order of values. - * - * @example - * - * import { compare } from 'natural-orderby'; - * - * const users = [ - * { - * username: 'Bamm-Bamm', - * lastLogin: { - * ip: '192.168.5.2', - * datetime: 'Fri Jun 15 2018 16:48:00 GMT+0200 (CEST)' - * }, - * }, - * { - * username: 'Wilma', - * lastLogin: { - * ip: '192.168.10.1', - * datetime: '14 Jun 2018 00:00:00 PDT' - * }, - * }, - * { - * username: 'dino', - * lastLogin: { - * ip: '192.168.0.2', - * datetime: 'June 15, 2018 14:48:00' - * }, - * }, - * { - * username: 'Barney', - * lastLogin: { - * ip: '192.168.1.1', - * datetime: 'Thu, 14 Jun 2018 07:00:00 GMT' - * }, - * }, - * { - * username: 'Pebbles', - * lastLogin: { - * ip: '192.168.1.21', - * datetime: '15 June 2018 14:48 UTC' - * }, - * }, - * { - * username: 'Hoppy', - * lastLogin: { - * ip: '192.168.5.10', - * datetime: '2018-06-15T14:48:00.000Z' - * }, - * }, - * ]; - * - * users.sort((a, b) => compare()(a.ip, b.ip)); - * - * // => [ - * // { - * // username: 'dino', - * // ip: '192.168.0.2', - * // datetime: 'June 15, 2018 14:48:00' - * // }, - * // { - * // username: 'Barney', - * // ip: '192.168.1.1', - * // datetime: 'Thu, 14 Jun 2018 07:00:00 GMT' - * // }, - * // { - * // username: 'Pebbles', - * // ip: '192.168.1.21', - * // datetime: '15 June 2018 14:48 UTC' - * // }, - * // { - * // username: 'Bamm-Bamm', - * // ip: '192.168.5.2', - * // datetime: 'Fri Jun 15 2018 16:48:00 GMT+0200 (CEST)' - * // }, - * // { - * // username: 'Hoppy', - * // ip: '192.168.5.10', - * // datetime: '2018-06-15T14:48:00.000Z' - * // }, - * // { - * // username: 'Wilma', - * // ip: '192.168.10.1', - * // datetime: '14 Jun 2018 00:00:00 PDT' - * // } - * // ] - */ -function compare(options) { - var validatedOptions = getOptions(options); - return baseCompare(validatedOptions); -} - -/* -* Javascript natural sort algorithm with unicode support -* based on chunking idea by Dave Koelle -* -* https://github.com/yobacca/natural-sort-order -* released under MIT License -*/ - - - +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"orderBy\": () => (/* binding */ orderBy),\n/* harmony export */ \"compare\": () => (/* binding */ compare)\n/* harmony export */ });\nvar compareNumbers = function compareNumbers(numberA, numberB) {\n if (numberA < numberB) {\n return -1;\n }\n\n if (numberA > numberB) {\n return 1;\n }\n\n return 0;\n};\n\nvar RE_NUMBERS = /(^0x[\\da-fA-F]+$|^([+-]?(?:\\d+(?:\\.\\d*)?|\\.\\d+)(?:[eE][+-]?\\d+)?(?!\\.\\d+)(?=\\D|\\s|$))|\\d+)/g;\nvar RE_LEADING_OR_TRAILING_WHITESPACES = /^\\s+|\\s+$/g; // trim pre-post whitespace\n\nvar RE_WHITESPACES = /\\s+/g; // normalize all whitespace to single ' ' character\n\nvar RE_INT_OR_FLOAT = /^[+-]?(?:\\d+(?:\\.\\d*)?|\\.\\d+)(?:[eE][+-]?\\d+)?$/; // identify integers and floats\n\nvar RE_DATE = /(^([\\w ]+,?[\\w ]+)?[\\w ]+,?[\\w ]+\\d+:\\d+(:\\d+)?[\\w ]?|^\\d{1,4}[/-]\\d{1,4}[/-]\\d{1,4}|^\\w+, \\w+ \\d+, \\d{4})/; // identify date strings\n\nvar RE_LEADING_ZERO = /^0+[1-9]{1}[0-9]*$/;\nvar RE_UNICODE_CHARACTERS = /[^\\x00-\\x80]/;\n\nvar compareUnicode = function compareUnicode(stringA, stringB) {\n var result = stringA.localeCompare(stringB);\n return result ? result / Math.abs(result) : 0;\n};\n\nvar stringCompare = function stringCompare(stringA, stringB) {\n if (stringA < stringB) {\n return -1;\n }\n\n if (stringA > stringB) {\n return 1;\n }\n\n return 0;\n};\n\nvar compareChunks = function compareChunks(chunksA, chunksB) {\n var lengthA = chunksA.length;\n var lengthB = chunksB.length;\n var size = Math.min(lengthA, lengthB);\n\n for (var i = 0; i < size; i++) {\n var chunkA = chunksA[i];\n var chunkB = chunksB[i];\n\n if (chunkA.normalizedString !== chunkB.normalizedString) {\n if (chunkA.normalizedString === '' !== (chunkB.normalizedString === '')) {\n // empty strings have lowest value\n return chunkA.normalizedString === '' ? -1 : 1;\n }\n\n if (chunkA.parsedNumber !== undefined && chunkB.parsedNumber !== undefined) {\n // compare numbers\n var result = compareNumbers(chunkA.parsedNumber, chunkB.parsedNumber);\n\n if (result === 0) {\n // compare string value, if parsed numbers are equal\n // Example:\n // chunkA = { parsedNumber: 1, normalizedString: \"001\" }\n // chunkB = { parsedNumber: 1, normalizedString: \"01\" }\n // chunkA.parsedNumber === chunkB.parsedNumber\n // chunkA.normalizedString < chunkB.normalizedString\n return stringCompare(chunkA.normalizedString, chunkB.normalizedString);\n }\n\n return result;\n } else if (chunkA.parsedNumber !== undefined || chunkB.parsedNumber !== undefined) {\n // number < string\n return chunkA.parsedNumber !== undefined ? -1 : 1;\n } else if (RE_UNICODE_CHARACTERS.test(chunkA.normalizedString + chunkB.normalizedString) && chunkA.normalizedString.localeCompare) {\n // use locale comparison only if one of the chunks contains unicode characters\n return compareUnicode(chunkA.normalizedString, chunkB.normalizedString);\n } else {\n // use common string comparison for performance reason\n return stringCompare(chunkA.normalizedString, chunkB.normalizedString);\n }\n }\n } // if the chunks are equal so far, the one which has more chunks is greater than the other one\n\n\n if (lengthA > size || lengthB > size) {\n return lengthA <= size ? -1 : 1;\n }\n\n return 0;\n};\n\nvar compareOtherTypes = function compareOtherTypes(valueA, valueB) {\n if (!valueA.chunks ? valueB.chunks : !valueB.chunks) {\n return !valueA.chunks ? 1 : -1;\n }\n\n if (valueA.isNaN ? !valueB.isNaN : valueB.isNaN) {\n return valueA.isNaN ? -1 : 1;\n }\n\n if (valueA.isSymbol ? !valueB.isSymbol : valueB.isSymbol) {\n return valueA.isSymbol ? -1 : 1;\n }\n\n if (valueA.isObject ? !valueB.isObject : valueB.isObject) {\n return valueA.isObject ? -1 : 1;\n }\n\n if (valueA.isArray ? !valueB.isArray : valueB.isArray) {\n return valueA.isArray ? -1 : 1;\n }\n\n if (valueA.isFunction ? !valueB.isFunction : valueB.isFunction) {\n return valueA.isFunction ? -1 : 1;\n }\n\n if (valueA.isNull ? !valueB.isNull : valueB.isNull) {\n return valueA.isNull ? -1 : 1;\n }\n\n return 0;\n};\n\nvar compareValues = function compareValues(valueA, valueB) {\n if (valueA.value === valueB.value) {\n return 0;\n }\n\n if (valueA.parsedNumber !== undefined && valueB.parsedNumber !== undefined) {\n return compareNumbers(valueA.parsedNumber, valueB.parsedNumber);\n }\n\n if (valueA.chunks && valueB.chunks) {\n return compareChunks(valueA.chunks, valueB.chunks);\n }\n\n return compareOtherTypes(valueA, valueB);\n};\n\nvar compareMultiple = function compareMultiple(recordA, recordB, orders) {\n var indexA = recordA.index,\n valuesA = recordA.values;\n var indexB = recordB.index,\n valuesB = recordB.values;\n var length = valuesA.length;\n var ordersLength = orders.length;\n\n for (var i = 0; i < length; i++) {\n var order = i < ordersLength ? orders[i] : null;\n\n if (order && typeof order === 'function') {\n var result = order(valuesA[i].value, valuesB[i].value);\n\n if (result) {\n return result;\n }\n } else {\n var _result = compareValues(valuesA[i], valuesB[i]);\n\n if (_result) {\n return _result * (order === 'desc' ? -1 : 1);\n }\n }\n }\n\n return indexA - indexB;\n};\n\nvar createIdentifierFn = function createIdentifierFn(identifier) {\n if (typeof identifier === 'function') {\n // identifier is already a lookup function\n return identifier;\n }\n\n return function (value) {\n if (Array.isArray(value)) {\n var index = Number(identifier);\n\n if (Number.isInteger(index)) {\n return value[index];\n }\n } else if (value && typeof value === 'object' && typeof identifier !== 'function') {\n return value[identifier];\n }\n\n return value;\n };\n};\n\nvar stringify = function stringify(value) {\n if (typeof value === 'boolean' || value instanceof Boolean) {\n return Number(value).toString();\n }\n\n if (typeof value === 'number' || value instanceof Number) {\n return value.toString();\n }\n\n if (value instanceof Date) {\n return value.getTime().toString();\n }\n\n if (typeof value === 'string' || value instanceof String) {\n return value.toLowerCase().replace(RE_LEADING_OR_TRAILING_WHITESPACES, '');\n }\n\n return '';\n};\n\nvar parseNumber = function parseNumber(value) {\n if (value.length !== 0) {\n var parsedNumber = Number(value);\n\n if (!Number.isNaN(parsedNumber)) {\n return parsedNumber;\n }\n }\n\n return undefined;\n};\n\nvar parseDate = function parseDate(value) {\n if (RE_DATE.test(value)) {\n var parsedDate = Date.parse(value);\n\n if (!Number.isNaN(parsedDate)) {\n return parsedDate;\n }\n }\n\n return undefined;\n};\n\nvar numberify = function numberify(value) {\n var parsedNumber = parseNumber(value);\n\n if (parsedNumber !== undefined) {\n return parsedNumber;\n }\n\n return parseDate(value);\n};\n\nvar createChunks = function createChunks(value) {\n return value.replace(RE_NUMBERS, '\\0$1\\0').replace(/\\0$/, '').replace(/^\\0/, '').split('\\0');\n};\n\nvar normalizeAlphaChunk = function normalizeAlphaChunk(chunk) {\n return chunk.replace(RE_WHITESPACES, ' ').replace(RE_LEADING_OR_TRAILING_WHITESPACES, '');\n};\n\nvar normalizeNumericChunk = function normalizeNumericChunk(chunk, index, chunks) {\n if (RE_INT_OR_FLOAT.test(chunk)) {\n // don´t parse a number, if there´s a preceding decimal point\n // to keep significance\n // e.g. 1.0020, 1.020\n if (!RE_LEADING_ZERO.test(chunk) || index === 0 || chunks[index - 1] !== '.') {\n return parseNumber(chunk) || 0;\n }\n }\n\n return undefined;\n};\n\nvar createChunkMap = function createChunkMap(chunk, index, chunks) {\n return {\n parsedNumber: normalizeNumericChunk(chunk, index, chunks),\n normalizedString: normalizeAlphaChunk(chunk)\n };\n};\n\nvar createChunkMaps = function createChunkMaps(value) {\n var chunksMaps = createChunks(value).map(createChunkMap);\n return chunksMaps;\n};\n\nvar isFunction = function isFunction(value) {\n return typeof value === 'function';\n};\n\nvar isNaN = function isNaN(value) {\n return Number.isNaN(value) || value instanceof Number && Number.isNaN(value.valueOf());\n};\n\nvar isNull = function isNull(value) {\n return value === null;\n};\n\nvar isObject = function isObject(value) {\n return value !== null && typeof value === 'object' && !Array.isArray(value) && !(value instanceof Number) && !(value instanceof String) && !(value instanceof Boolean) && !(value instanceof Date);\n};\n\nvar isSymbol = function isSymbol(value) {\n return typeof value === 'symbol';\n};\n\nvar isUndefined = function isUndefined(value) {\n return value === undefined;\n};\n\nvar getMappedValueRecord = function getMappedValueRecord(value) {\n if (typeof value === 'string' || value instanceof String || (typeof value === 'number' || value instanceof Number) && !isNaN(value) || typeof value === 'boolean' || value instanceof Boolean || value instanceof Date) {\n var stringValue = stringify(value);\n var parsedNumber = numberify(stringValue);\n var chunks = createChunkMaps(parsedNumber ? \"\" + parsedNumber : stringValue);\n return {\n parsedNumber: parsedNumber,\n chunks: chunks,\n value: value\n };\n }\n\n return {\n isArray: Array.isArray(value),\n isFunction: isFunction(value),\n isNaN: isNaN(value),\n isNull: isNull(value),\n isObject: isObject(value),\n isSymbol: isSymbol(value),\n isUndefined: isUndefined(value),\n value: value\n };\n};\n\nvar getValueByIdentifier = function getValueByIdentifier(value, getValue) {\n return getValue(value);\n};\n\nvar getElementByIndex = function getElementByIndex(collection, index) {\n return collection[index];\n};\n\nvar baseOrderBy = function baseOrderBy(collection, identifiers, orders) {\n var identifierFns = identifiers.length ? identifiers.map(createIdentifierFn) : [function (value) {\n return value;\n }]; // temporary array holds elements with position and sort-values\n\n var mappedCollection = collection.map(function (element, index) {\n var values = identifierFns.map(function (identifier) {\n return getValueByIdentifier(element, identifier);\n }).map(getMappedValueRecord);\n return {\n index: index,\n values: values\n };\n }); // iterate over values and compare values until a != b or last value reached\n\n mappedCollection.sort(function (recordA, recordB) {\n return compareMultiple(recordA, recordB, orders);\n });\n return mappedCollection.map(function (element) {\n return getElementByIndex(collection, element.index);\n });\n};\n\nvar getIdentifiers = function getIdentifiers(identifiers) {\n if (!identifiers) {\n return [];\n }\n\n var identifierList = !Array.isArray(identifiers) ? [identifiers] : [].concat(identifiers);\n\n if (identifierList.some(function (identifier) {\n return typeof identifier !== 'string' && typeof identifier !== 'number' && typeof identifier !== 'function';\n })) {\n return [];\n }\n\n return identifierList;\n};\n\nvar getOrders = function getOrders(orders) {\n if (!orders) {\n return [];\n }\n\n var orderList = !Array.isArray(orders) ? [orders] : [].concat(orders);\n\n if (orderList.some(function (order) {\n return order !== 'asc' && order !== 'desc' && typeof order !== 'function';\n })) {\n return [];\n }\n\n return orderList;\n};\n\n/**\n * Creates an array of elements, natural sorted by specified identifiers and\n * the corresponding sort orders. This method implements a stable sort\n * algorithm, which means the original sort order of equal elements is\n * preserved.\n *\n * If `collection` is an array of primitives, `identifiers` may be unspecified.\n * Otherwise, you should specify `identifiers` to sort by or `collection` will\n * be returned unsorted. An identifier can expressed by:\n *\n * - an index position, if `collection` is a nested array,\n * - a property name, if `collection` is an array of objects,\n * - a function which returns a particular value from an element of a nested array or an array of objects. This function will be invoked by passing one element of `collection`.\n *\n * If `orders` is unspecified, all values are sorted in ascending order.\n * Otherwise, specify an order of `'desc'` for descending or `'asc'` for\n * ascending sort order of corresponding values. You may also specify a compare\n * function for an order, which will be invoked by two arguments:\n * `(valueA, valueB)`. It must return a number representing the sort order.\n *\n * @example\n *\n * import { orderBy } from 'natural-orderby';\n *\n * const users = [\n * {\n * username: 'Bamm-Bamm',\n * ip: '192.168.5.2',\n * datetime: 'Fri Jun 15 2018 16:48:00 GMT+0200 (CEST)'\n * },\n * {\n * username: 'Wilma',\n * ip: '192.168.10.1',\n * datetime: '14 Jun 2018 00:00:00 PDT'\n * },\n * {\n * username: 'dino',\n * ip: '192.168.0.2',\n * datetime: 'June 15, 2018 14:48:00'\n * },\n * {\n * username: 'Barney',\n * ip: '192.168.1.1',\n * datetime: 'Thu, 14 Jun 2018 07:00:00 GMT'\n * },\n * {\n * username: 'Pebbles',\n * ip: '192.168.1.21',\n * datetime: '15 June 2018 14:48 UTC'\n * },\n * {\n * username: 'Hoppy',\n * ip: '192.168.5.10',\n * datetime: '2018-06-15T14:48:00.000Z'\n * },\n * ];\n *\n * orderBy(\n * users,\n * [v => v.datetime, v => v.ip],\n * ['desc', 'asc']\n * );\n *\n * // => [\n * // {\n * // username: 'dino',\n * // ip: '192.168.0.2',\n * // datetime: 'June 15, 2018 14:48:00',\n * // },\n * // {\n * // username: 'Pebbles',\n * // ip: '192.168.1.21',\n * // datetime: '15 June 2018 14:48 UTC',\n * // },\n * // {\n * // username: 'Bamm-Bamm',\n * // ip: '192.168.5.2',\n * // datetime: 'Fri Jun 15 2018 16:48:00 GMT+0200 (CEST)',\n * // },\n * // {\n * // username: 'Hoppy',\n * // ip: '192.168.5.10',\n * // datetime: '2018-06-15T14:48:00.000Z',\n * // },\n * // {\n * // username: 'Barney',\n * // ip: '192.168.1.1',\n * // datetime: 'Thu, 14 Jun 2018 07:00:00 GMT',\n * // },\n * // {\n * // username: 'Wilma',\n * // ip: '192.168.10.1',\n * // datetime: '14 Jun 2018 00:00:00 PDT',\n * // },\n * // ]\n */\nfunction orderBy(collection, identifiers, orders) {\n if (!collection || !Array.isArray(collection)) {\n return [];\n }\n\n var validatedIdentifiers = getIdentifiers(identifiers);\n var validatedOrders = getOrders(orders);\n return baseOrderBy(collection, validatedIdentifiers, validatedOrders);\n}\n\nvar baseCompare = function baseCompare(options) {\n return function (valueA, valueB) {\n var a = getMappedValueRecord(valueA);\n var b = getMappedValueRecord(valueB);\n var result = compareValues(a, b);\n return result * (options.order === 'desc' ? -1 : 1);\n };\n};\n\nvar isValidOrder = function isValidOrder(value) {\n return typeof value === 'string' && (value === 'asc' || value === 'desc');\n};\n\nvar getOptions = function getOptions(customOptions) {\n var order = 'asc';\n\n if (typeof customOptions === 'string' && isValidOrder(customOptions)) {\n order = customOptions;\n } else if (customOptions && typeof customOptions === 'object' && customOptions.order && isValidOrder(customOptions.order)) {\n order = customOptions.order;\n }\n\n return {\n order: order\n };\n};\n\n/**\n * Creates a compare function that defines the natural sort order considering\n * the given `options` which may be passed to [`Array.prototype.sort()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort).\n *\n * If `options` or its property `order` is unspecified, values are sorted in\n * ascending sort order. Otherwise, specify an order of `'desc'` for descending\n * or `'asc'` for ascending sort order of values.\n *\n * @example\n *\n * import { compare } from 'natural-orderby';\n *\n * const users = [\n * {\n * username: 'Bamm-Bamm',\n * lastLogin: {\n * ip: '192.168.5.2',\n * datetime: 'Fri Jun 15 2018 16:48:00 GMT+0200 (CEST)'\n * },\n * },\n * {\n * username: 'Wilma',\n * lastLogin: {\n * ip: '192.168.10.1',\n * datetime: '14 Jun 2018 00:00:00 PDT'\n * },\n * },\n * {\n * username: 'dino',\n * lastLogin: {\n * ip: '192.168.0.2',\n * datetime: 'June 15, 2018 14:48:00'\n * },\n * },\n * {\n * username: 'Barney',\n * lastLogin: {\n * ip: '192.168.1.1',\n * datetime: 'Thu, 14 Jun 2018 07:00:00 GMT'\n * },\n * },\n * {\n * username: 'Pebbles',\n * lastLogin: {\n * ip: '192.168.1.21',\n * datetime: '15 June 2018 14:48 UTC'\n * },\n * },\n * {\n * username: 'Hoppy',\n * lastLogin: {\n * ip: '192.168.5.10',\n * datetime: '2018-06-15T14:48:00.000Z'\n * },\n * },\n * ];\n *\n * users.sort((a, b) => compare()(a.ip, b.ip));\n *\n * // => [\n * // {\n * // username: 'dino',\n * // ip: '192.168.0.2',\n * // datetime: 'June 15, 2018 14:48:00'\n * // },\n * // {\n * // username: 'Barney',\n * // ip: '192.168.1.1',\n * // datetime: 'Thu, 14 Jun 2018 07:00:00 GMT'\n * // },\n * // {\n * // username: 'Pebbles',\n * // ip: '192.168.1.21',\n * // datetime: '15 June 2018 14:48 UTC'\n * // },\n * // {\n * // username: 'Bamm-Bamm',\n * // ip: '192.168.5.2',\n * // datetime: 'Fri Jun 15 2018 16:48:00 GMT+0200 (CEST)'\n * // },\n * // {\n * // username: 'Hoppy',\n * // ip: '192.168.5.10',\n * // datetime: '2018-06-15T14:48:00.000Z'\n * // },\n * // {\n * // username: 'Wilma',\n * // ip: '192.168.10.1',\n * // datetime: '14 Jun 2018 00:00:00 PDT'\n * // }\n * // ]\n */\nfunction compare(options) {\n var validatedOptions = getOptions(options);\n return baseCompare(validatedOptions);\n}\n\n/*\n* Javascript natural sort algorithm with unicode support\n* based on chunking idea by Dave Koelle\n*\n* https://github.com/yobacca/natural-sort-order\n* released under MIT License\n*/\n\n\n\n\n//# sourceURL=webpack://freedatas2html/./node_modules/natural-orderby/esm/natural-orderby.js?"); /***/ }), @@ -650,16 +26,18 @@ function compare(options) { \*************************************************/ /***/ (function(module, exports) { -var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_DEFINE_RESULT__;/* @license -Papa Parse -v5.3.1 -https://github.com/mholt/PapaParse -License: MIT -*/ -!function(e,t){ true?!(__WEBPACK_AMD_DEFINE_ARRAY__ = [], __WEBPACK_AMD_DEFINE_FACTORY__ = (t), - __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? - (__WEBPACK_AMD_DEFINE_FACTORY__.apply(exports, __WEBPACK_AMD_DEFINE_ARRAY__)) : __WEBPACK_AMD_DEFINE_FACTORY__), - __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__)):0}(this,function s(){"use strict";var f="undefined"!=typeof self?self:"undefined"!=typeof window?window:void 0!==f?f:{};var n=!f.document&&!!f.postMessage,o=n&&/blob:/i.test((f.location||{}).protocol),a={},h=0,b={parse:function(e,t){var i=(t=t||{}).dynamicTyping||!1;M(i)&&(t.dynamicTypingFunction=i,i={});if(t.dynamicTyping=i,t.transform=!!M(t.transform)&&t.transform,t.worker&&b.WORKERS_SUPPORTED){var r=function(){if(!b.WORKERS_SUPPORTED)return!1;var e=(i=f.URL||f.webkitURL||null,r=s.toString(),b.BLOB_URL||(b.BLOB_URL=i.createObjectURL(new Blob(["(",r,")();"],{type:"text/javascript"})))),t=new f.Worker(e);var i,r;return t.onmessage=_,t.id=h++,a[t.id]=t}();return r.userStep=t.step,r.userChunk=t.chunk,r.userComplete=t.complete,r.userError=t.error,t.step=M(t.step),t.chunk=M(t.chunk),t.complete=M(t.complete),t.error=M(t.error),delete t.worker,void r.postMessage({input:e,config:t,workerId:r.id})}var n=null;b.NODE_STREAM_INPUT,"string"==typeof e?n=t.download?new l(t):new p(t):!0===e.readable&&M(e.read)&&M(e.on)?n=new g(t):(f.File&&e instanceof File||e instanceof Object)&&(n=new c(t));return n.stream(e)},unparse:function(e,t){var n=!1,_=!0,m=",",y="\r\n",s='"',a=s+s,i=!1,r=null,o=!1;!function(){if("object"!=typeof t)return;"string"!=typeof t.delimiter||b.BAD_DELIMITERS.filter(function(e){return-1!==t.delimiter.indexOf(e)}).length||(m=t.delimiter);("boolean"==typeof t.quotes||"function"==typeof t.quotes||Array.isArray(t.quotes))&&(n=t.quotes);"boolean"!=typeof t.skipEmptyLines&&"string"!=typeof t.skipEmptyLines||(i=t.skipEmptyLines);"string"==typeof t.newline&&(y=t.newline);"string"==typeof t.quoteChar&&(s=t.quoteChar);"boolean"==typeof t.header&&(_=t.header);if(Array.isArray(t.columns)){if(0===t.columns.length)throw new Error("Option columns is empty");r=t.columns}void 0!==t.escapeChar&&(a=t.escapeChar+s);"boolean"==typeof t.escapeFormulae&&(o=t.escapeFormulae)}();var h=new RegExp(j(s),"g");"string"==typeof e&&(e=JSON.parse(e));if(Array.isArray(e)){if(!e.length||Array.isArray(e[0]))return u(null,e,i);if("object"==typeof e[0])return u(r||Object.keys(e[0]),e,i)}else if("object"==typeof e)return"string"==typeof e.data&&(e.data=JSON.parse(e.data)),Array.isArray(e.data)&&(e.fields||(e.fields=e.meta&&e.meta.fields),e.fields||(e.fields=Array.isArray(e.data[0])?e.fields:"object"==typeof e.data[0]?Object.keys(e.data[0]):[]),Array.isArray(e.data[0])||"object"==typeof e.data[0]||(e.data=[e.data])),u(e.fields||[],e.data||[],i);throw new Error("Unable to serialize unrecognized input");function u(e,t,i){var r="";"string"==typeof e&&(e=JSON.parse(e)),"string"==typeof t&&(t=JSON.parse(t));var n=Array.isArray(e)&&0=this._config.preview;if(o)f.postMessage({results:n,workerId:b.WORKER_ID,finished:a});else if(M(this._config.chunk)&&!t){if(this._config.chunk(n,this._handle),this._handle.paused()||this._handle.aborted())return void(this._halted=!0);n=void 0,this._completeResults=void 0}return this._config.step||this._config.chunk||(this._completeResults.data=this._completeResults.data.concat(n.data),this._completeResults.errors=this._completeResults.errors.concat(n.errors),this._completeResults.meta=n.meta),this._completed||!a||!M(this._config.complete)||n&&n.meta.aborted||(this._config.complete(this._completeResults,this._input),this._completed=!0),a||n&&n.meta.paused||this._nextChunk(),n}this._halted=!0},this._sendError=function(e){M(this._config.error)?this._config.error(e):o&&this._config.error&&f.postMessage({workerId:b.WORKER_ID,error:e,finished:!1})}}function l(e){var r;(e=e||{}).chunkSize||(e.chunkSize=b.RemoteChunkSize),u.call(this,e),this._nextChunk=n?function(){this._readChunk(),this._chunkLoaded()}:function(){this._readChunk()},this.stream=function(e){this._input=e,this._nextChunk()},this._readChunk=function(){if(this._finished)this._chunkLoaded();else{if(r=new XMLHttpRequest,this._config.withCredentials&&(r.withCredentials=this._config.withCredentials),n||(r.onload=v(this._chunkLoaded,this),r.onerror=v(this._chunkError,this)),r.open(this._config.downloadRequestBody?"POST":"GET",this._input,!n),this._config.downloadRequestHeaders){var e=this._config.downloadRequestHeaders;for(var t in e)r.setRequestHeader(t,e[t])}if(this._config.chunkSize){var i=this._start+this._config.chunkSize-1;r.setRequestHeader("Range","bytes="+this._start+"-"+i)}try{r.send(this._config.downloadRequestBody)}catch(e){this._chunkError(e.message)}n&&0===r.status&&this._chunkError()}},this._chunkLoaded=function(){4===r.readyState&&(r.status<200||400<=r.status?this._chunkError():(this._start+=this._config.chunkSize?this._config.chunkSize:r.responseText.length,this._finished=!this._config.chunkSize||this._start>=function(e){var t=e.getResponseHeader("Content-Range");if(null===t)return-1;return parseInt(t.substring(t.lastIndexOf("/")+1))}(r),this.parseChunk(r.responseText)))},this._chunkError=function(e){var t=r.statusText||e;this._sendError(new Error(t))}}function c(e){var r,n;(e=e||{}).chunkSize||(e.chunkSize=b.LocalChunkSize),u.call(this,e);var s="undefined"!=typeof FileReader;this.stream=function(e){this._input=e,n=e.slice||e.webkitSlice||e.mozSlice,s?((r=new FileReader).onload=v(this._chunkLoaded,this),r.onerror=v(this._chunkError,this)):r=new FileReaderSync,this._nextChunk()},this._nextChunk=function(){this._finished||this._config.preview&&!(this._rowCount=this._input.size,this.parseChunk(e.target.result)},this._chunkError=function(){this._sendError(r.error)}}function p(e){var i;u.call(this,e=e||{}),this.stream=function(e){return i=e,this._nextChunk()},this._nextChunk=function(){if(!this._finished){var e,t=this._config.chunkSize;return t?(e=i.substring(0,t),i=i.substring(t)):(e=i,i=""),this._finished=!i,this.parseChunk(e)}}}function g(e){u.call(this,e=e||{});var t=[],i=!0,r=!1;this.pause=function(){u.prototype.pause.apply(this,arguments),this._input.pause()},this.resume=function(){u.prototype.resume.apply(this,arguments),this._input.resume()},this.stream=function(e){this._input=e,this._input.on("data",this._streamData),this._input.on("end",this._streamEnd),this._input.on("error",this._streamError)},this._checkIsFinished=function(){r&&1===t.length&&(this._finished=!0)},this._nextChunk=function(){this._checkIsFinished(),t.length?this.parseChunk(t.shift()):i=!0},this._streamData=v(function(e){try{t.push("string"==typeof e?e:e.toString(this._config.encoding)),i&&(i=!1,this._checkIsFinished(),this.parseChunk(t.shift()))}catch(e){this._streamError(e)}},this),this._streamError=v(function(e){this._streamCleanUp(),this._sendError(e)},this),this._streamEnd=v(function(){this._streamCleanUp(),r=!0,this._streamData("")},this),this._streamCleanUp=v(function(){this._input.removeListener("data",this._streamData),this._input.removeListener("end",this._streamEnd),this._input.removeListener("error",this._streamError)},this)}function i(m){var a,o,h,r=Math.pow(2,53),n=-r,s=/^\s*-?(\d+\.?|\.\d+|\d+\.\d+)([eE][-+]?\d+)?\s*$/,u=/^(\d{4}-[01]\d-[0-3]\dT[0-2]\d:[0-5]\d:[0-5]\d\.\d+([+-][0-2]\d:[0-5]\d|Z))|(\d{4}-[01]\d-[0-3]\dT[0-2]\d:[0-5]\d:[0-5]\d([+-][0-2]\d:[0-5]\d|Z))|(\d{4}-[01]\d-[0-3]\dT[0-2]\d:[0-5]\d([+-][0-2]\d:[0-5]\d|Z))$/,t=this,i=0,f=0,d=!1,e=!1,l=[],c={data:[],errors:[],meta:{}};if(M(m.step)){var p=m.step;m.step=function(e){if(c=e,_())g();else{if(g(),0===c.data.length)return;i+=e.data.length,m.preview&&i>m.preview?o.abort():(c.data=c.data[0],p(c,t))}}}function y(e){return"greedy"===m.skipEmptyLines?""===e.join("").trim():1===e.length&&0===e[0].length}function g(){if(c&&h&&(k("Delimiter","UndetectableDelimiter","Unable to auto-detect delimiting character; defaulted to '"+b.DefaultDelimiter+"'"),h=!1),m.skipEmptyLines)for(var e=0;e=l.length?"__parsed_extra":l[i]),m.transform&&(s=m.transform(s,n)),s=v(n,s),"__parsed_extra"===n?(r[n]=r[n]||[],r[n].push(s)):r[n]=s}return m.header&&(i>l.length?k("FieldMismatch","TooManyFields","Too many fields: expected "+l.length+" fields but parsed "+i,f+t):i=r.length/2?"\r\n":"\r"}(e,r)),h=!1,m.delimiter)M(m.delimiter)&&(m.delimiter=m.delimiter(e),c.meta.delimiter=m.delimiter);else{var n=function(e,t,i,r,n){var s,a,o,h;n=n||[",","\t","|",";",b.RECORD_SEP,b.UNIT_SEP];for(var u=0;u=D)return C(!0)}else for(m=F,F++;;){if(-1===(m=r.indexOf(S,m+1)))return i||u.push({type:"Quotes",code:"MissingQuotes",message:"Quoted field unterminated",row:h.length,index:F}),E();if(m===n-1)return E(r.substring(F,m).replace(_,S));if(S!==L||r[m+1]!==L){if(S===L||0===m||r[m-1]!==L){-1!==p&&p=D)return C(!0);break}u.push({type:"Quotes",code:"InvalidQuotes",message:"Trailing quote on quoted field is malformed",row:h.length,index:F}),m++}}else m++}return E();function k(e){h.push(e),d=F}function b(e){var t=0;if(-1!==e){var i=r.substring(m+1,e);i&&""===i.trim()&&(t=i.length)}return t}function E(e){return i||(void 0===e&&(e=r.substring(F)),f.push(e),F=n,k(f),o&&R()),C()}function w(e){F=e,k(f),f=[],g=r.indexOf(x,F)}function C(e){return{data:h,errors:u,meta:{delimiter:O,linebreak:x,aborted:z,truncated:!!e,cursor:d+(t||0)}}}function R(){T(C()),h=[],u=[]}},this.abort=function(){z=!0},this.getCharIndex=function(){return F}}function _(e){var t=e.data,i=a[t.workerId],r=!1;if(t.error)i.userError(t.error,t.file);else if(t.results&&t.results.data){var n={abort:function(){r=!0,m(t.workerId,{data:[],errors:[],meta:{aborted:!0}})},pause:y,resume:y};if(M(i.userStep)){for(var s=0;s=this._config.preview;if(o)f.postMessage({results:n,workerId:b.WORKER_ID,finished:a});else if(M(this._config.chunk)&&!t){if(this._config.chunk(n,this._handle),this._handle.paused()||this._handle.aborted())return void(this._halted=!0);n=void 0,this._completeResults=void 0}return this._config.step||this._config.chunk||(this._completeResults.data=this._completeResults.data.concat(n.data),this._completeResults.errors=this._completeResults.errors.concat(n.errors),this._completeResults.meta=n.meta),this._completed||!a||!M(this._config.complete)||n&&n.meta.aborted||(this._config.complete(this._completeResults,this._input),this._completed=!0),a||n&&n.meta.paused||this._nextChunk(),n}this._halted=!0},this._sendError=function(e){M(this._config.error)?this._config.error(e):o&&this._config.error&&f.postMessage({workerId:b.WORKER_ID,error:e,finished:!1})}}function l(e){var r;(e=e||{}).chunkSize||(e.chunkSize=b.RemoteChunkSize),u.call(this,e),this._nextChunk=n?function(){this._readChunk(),this._chunkLoaded()}:function(){this._readChunk()},this.stream=function(e){this._input=e,this._nextChunk()},this._readChunk=function(){if(this._finished)this._chunkLoaded();else{if(r=new XMLHttpRequest,this._config.withCredentials&&(r.withCredentials=this._config.withCredentials),n||(r.onload=v(this._chunkLoaded,this),r.onerror=v(this._chunkError,this)),r.open(this._config.downloadRequestBody?\"POST\":\"GET\",this._input,!n),this._config.downloadRequestHeaders){var e=this._config.downloadRequestHeaders;for(var t in e)r.setRequestHeader(t,e[t])}if(this._config.chunkSize){var i=this._start+this._config.chunkSize-1;r.setRequestHeader(\"Range\",\"bytes=\"+this._start+\"-\"+i)}try{r.send(this._config.downloadRequestBody)}catch(e){this._chunkError(e.message)}n&&0===r.status&&this._chunkError()}},this._chunkLoaded=function(){4===r.readyState&&(r.status<200||400<=r.status?this._chunkError():(this._start+=this._config.chunkSize?this._config.chunkSize:r.responseText.length,this._finished=!this._config.chunkSize||this._start>=function(e){var t=e.getResponseHeader(\"Content-Range\");if(null===t)return-1;return parseInt(t.substring(t.lastIndexOf(\"/\")+1))}(r),this.parseChunk(r.responseText)))},this._chunkError=function(e){var t=r.statusText||e;this._sendError(new Error(t))}}function c(e){var r,n;(e=e||{}).chunkSize||(e.chunkSize=b.LocalChunkSize),u.call(this,e);var s=\"undefined\"!=typeof FileReader;this.stream=function(e){this._input=e,n=e.slice||e.webkitSlice||e.mozSlice,s?((r=new FileReader).onload=v(this._chunkLoaded,this),r.onerror=v(this._chunkError,this)):r=new FileReaderSync,this._nextChunk()},this._nextChunk=function(){this._finished||this._config.preview&&!(this._rowCount=this._input.size,this.parseChunk(e.target.result)},this._chunkError=function(){this._sendError(r.error)}}function p(e){var i;u.call(this,e=e||{}),this.stream=function(e){return i=e,this._nextChunk()},this._nextChunk=function(){if(!this._finished){var e,t=this._config.chunkSize;return t?(e=i.substring(0,t),i=i.substring(t)):(e=i,i=\"\"),this._finished=!i,this.parseChunk(e)}}}function g(e){u.call(this,e=e||{});var t=[],i=!0,r=!1;this.pause=function(){u.prototype.pause.apply(this,arguments),this._input.pause()},this.resume=function(){u.prototype.resume.apply(this,arguments),this._input.resume()},this.stream=function(e){this._input=e,this._input.on(\"data\",this._streamData),this._input.on(\"end\",this._streamEnd),this._input.on(\"error\",this._streamError)},this._checkIsFinished=function(){r&&1===t.length&&(this._finished=!0)},this._nextChunk=function(){this._checkIsFinished(),t.length?this.parseChunk(t.shift()):i=!0},this._streamData=v(function(e){try{t.push(\"string\"==typeof e?e:e.toString(this._config.encoding)),i&&(i=!1,this._checkIsFinished(),this.parseChunk(t.shift()))}catch(e){this._streamError(e)}},this),this._streamError=v(function(e){this._streamCleanUp(),this._sendError(e)},this),this._streamEnd=v(function(){this._streamCleanUp(),r=!0,this._streamData(\"\")},this),this._streamCleanUp=v(function(){this._input.removeListener(\"data\",this._streamData),this._input.removeListener(\"end\",this._streamEnd),this._input.removeListener(\"error\",this._streamError)},this)}function i(m){var a,o,h,r=Math.pow(2,53),n=-r,s=/^\\s*-?(\\d+\\.?|\\.\\d+|\\d+\\.\\d+)([eE][-+]?\\d+)?\\s*$/,u=/^(\\d{4}-[01]\\d-[0-3]\\dT[0-2]\\d:[0-5]\\d:[0-5]\\d\\.\\d+([+-][0-2]\\d:[0-5]\\d|Z))|(\\d{4}-[01]\\d-[0-3]\\dT[0-2]\\d:[0-5]\\d:[0-5]\\d([+-][0-2]\\d:[0-5]\\d|Z))|(\\d{4}-[01]\\d-[0-3]\\dT[0-2]\\d:[0-5]\\d([+-][0-2]\\d:[0-5]\\d|Z))$/,t=this,i=0,f=0,d=!1,e=!1,l=[],c={data:[],errors:[],meta:{}};if(M(m.step)){var p=m.step;m.step=function(e){if(c=e,_())g();else{if(g(),0===c.data.length)return;i+=e.data.length,m.preview&&i>m.preview?o.abort():(c.data=c.data[0],p(c,t))}}}function y(e){return\"greedy\"===m.skipEmptyLines?\"\"===e.join(\"\").trim():1===e.length&&0===e[0].length}function g(){if(c&&h&&(k(\"Delimiter\",\"UndetectableDelimiter\",\"Unable to auto-detect delimiting character; defaulted to '\"+b.DefaultDelimiter+\"'\"),h=!1),m.skipEmptyLines)for(var e=0;e=l.length?\"__parsed_extra\":l[i]),m.transform&&(s=m.transform(s,n)),s=v(n,s),\"__parsed_extra\"===n?(r[n]=r[n]||[],r[n].push(s)):r[n]=s}return m.header&&(i>l.length?k(\"FieldMismatch\",\"TooManyFields\",\"Too many fields: expected \"+l.length+\" fields but parsed \"+i,f+t):i=r.length/2?\"\\r\\n\":\"\\r\"}(e,r)),h=!1,m.delimiter)M(m.delimiter)&&(m.delimiter=m.delimiter(e),c.meta.delimiter=m.delimiter);else{var n=function(e,t,i,r,n){var s,a,o,h;n=n||[\",\",\"\\t\",\"|\",\";\",b.RECORD_SEP,b.UNIT_SEP];for(var u=0;u=D)return C(!0)}else for(m=F,F++;;){if(-1===(m=r.indexOf(S,m+1)))return i||u.push({type:\"Quotes\",code:\"MissingQuotes\",message:\"Quoted field unterminated\",row:h.length,index:F}),E();if(m===n-1)return E(r.substring(F,m).replace(_,S));if(S!==L||r[m+1]!==L){if(S===L||0===m||r[m-1]!==L){-1!==p&&p=D)return C(!0);break}u.push({type:\"Quotes\",code:\"InvalidQuotes\",message:\"Trailing quote on quoted field is malformed\",row:h.length,index:F}),m++}}else m++}return E();function k(e){h.push(e),d=F}function b(e){var t=0;if(-1!==e){var i=r.substring(m+1,e);i&&\"\"===i.trim()&&(t=i.length)}return t}function E(e){return i||(void 0===e&&(e=r.substring(F)),f.push(e),F=n,k(f),o&&R()),C()}function w(e){F=e,k(f),f=[],g=r.indexOf(x,F)}function C(e){return{data:h,errors:u,meta:{delimiter:O,linebreak:x,aborted:z,truncated:!!e,cursor:d+(t||0)}}}function R(){T(C()),h=[],u=[]}},this.abort=function(){z=!0},this.getCharIndex=function(){return F}}function _(e){var t=e.data,i=a[t.workerId],r=!1;if(t.error)i.userError(t.error,t.file);else if(t.results&&t.results.data){var n={abort:function(){r=!0,m(t.workerId,{data:[],errors:[],meta:{aborted:!0}})},pause:y,resume:y};if(M(i.userStep)){for(var s=0;s { + +"use strict"; +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _freeDatas2HTML__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./freeDatas2HTML */ \"./src/freeDatas2HTML.ts\");\nvar __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\nvar __generator = (undefined && undefined.__generator) || function (thisArg, body) {\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\n function verb(n) { return function (v) { return step([n, v]); }; }\n function step(op) {\n if (f) throw new TypeError(\"Generator is already executing.\");\n while (_) try {\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\n if (y = 0, t) op = [op[0] & 2, t.value];\n switch (op[0]) {\n case 0: case 1: t = op; break;\n case 4: _.label++; return { value: op[1], done: false };\n case 5: _.label++; y = op[1]; op = [0]; continue;\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\n default:\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\n if (t[2]) _.ops.pop();\n _.trys.pop(); continue;\n }\n op = body.call(thisArg, _);\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\n }\n};\n\nvar initialise = function () { return __awaiter(void 0, void 0, void 0, function () {\n var converter, e_1;\n return __generator(this, function (_a) {\n switch (_a.label) {\n case 0:\n _a.trys.push([0, 2, , 3]);\n converter = new _freeDatas2HTML__WEBPACK_IMPORTED_MODULE_0__.freeDatas2HTML();\n converter.datasViewElt = { id: \"datas\" };\n converter.datasSelectors = [{ datasFieldNb: 3, id: \"filtre1\" }, { datasFieldNb: 4, id: \"filtre2\" }, { datasFieldNb: 5, id: \"filtre3\", separator: \",\" }];\n converter.datasSortingColumns = [{ datasFieldNb: 0 }, { datasFieldNb: 1 }, { datasFieldNb: 2 }];\n converter.datasSourceUrl = \"http://localhost:8080/datas/elements-chimiques.csv\";\n return [4, converter.run()];\n case 1:\n _a.sent();\n return [3, 3];\n case 2:\n e_1 = _a.sent();\n console.error(e_1);\n if (document.getElementById(\"datas\") !== null)\n document.getElementById(\"datas\").innerHTML = \"Désolé, mais un problème technique empêche l'affichage des données.\";\n return [3, 3];\n case 3: return [2];\n }\n });\n}); };\ninitialise();\n\n\n//# sourceURL=webpack://freedatas2html/./src/firstExample.ts?"); /***/ }), @@ -670,319 +48,7 @@ License: MIT /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { "use strict"; -__webpack_require__.r(__webpack_exports__); -/* harmony export */ __webpack_require__.d(__webpack_exports__, { -/* harmony export */ "freeDatas2HTML": () => (/* binding */ freeDatas2HTML) -/* harmony export */ }); -var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) { - function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } - return new (P || (P = Promise))(function (resolve, reject) { - function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } - function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } - function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } - step((generator = generator.apply(thisArg, _arguments || [])).next()); - }); -}; -var __generator = (undefined && undefined.__generator) || function (thisArg, body) { - var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; - return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g; - function verb(n) { return function (v) { return step([n, v]); }; } - function step(op) { - if (f) throw new TypeError("Generator is already executing."); - while (_) try { - if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t; - if (y = 0, t) op = [op[0] & 2, t.value]; - switch (op[0]) { - case 0: case 1: t = op; break; - case 4: _.label++; return { value: op[1], done: false }; - case 5: _.label++; y = op[1]; op = [0]; continue; - case 7: op = _.ops.pop(); _.trys.pop(); continue; - default: - if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; } - if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; } - if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; } - if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; } - if (t[2]) _.ops.pop(); - _.trys.pop(); continue; - } - op = body.call(thisArg, _); - } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; } - if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true }; - } -}; -var Papa = __webpack_require__(/*! papaparse */ "./node_modules/papaparse/papaparse.min.js"); -var errors = __webpack_require__(/*! ./errors.js */ "./src/errors.js"); -var compare = __webpack_require__(/*! natural-orderby */ "./node_modules/natural-orderby/esm/natural-orderby.js").compare; -var freeDatas2HTML = (function () { - function freeDatas2HTML() { - this._datasViewElt = { id: "", eltDOM: undefined }; - this._datasSourceUrl = ""; - this._datasSelectors = []; - this._datasSortingColumns = []; - this.parseMeta = undefined; - this.parseDatas = []; - this.parseErrors = []; - this.datasHTML = ""; - this.stopIfParseErrors = false; - } - Object.defineProperty(freeDatas2HTML.prototype, "datasViewElt", { - set: function (elt) { - var checkContainerExist = document.getElementById(elt.id); - if (checkContainerExist === null) - throw new Error(errors.elementNotFound + elt.id); - else { - this._datasViewElt.id = elt.id; - this._datasViewElt.eltDOM = checkContainerExist; - } - }, - enumerable: false, - configurable: true - }); - Object.defineProperty(freeDatas2HTML.prototype, "datasSourceUrl", { - set: function (url) { - if (url.trim().length === 0) - throw new Error(errors.needUrl); - else - this._datasSourceUrl = url.trim(); - }, - enumerable: false, - configurable: true - }); - Object.defineProperty(freeDatas2HTML.prototype, "datasSelectors", { - get: function () { - return this._datasSelectors; - }, - set: function (selectionElts) { - this._datasSelectors = []; - var checkContainerExist; - for (var i = 0; i < selectionElts.length; i++) { - checkContainerExist = document.getElementById(selectionElts[i].id); - if (checkContainerExist === null) - console.error(errors.elementNotFound + selectionElts[i].id); - else if (Number.isInteger(selectionElts[i].datasFieldNb) === false || selectionElts[i].datasFieldNb < 0) - console.error(errors.needNaturalNumber); - else { - selectionElts[i].eltDOM = checkContainerExist; - if (selectionElts[i].separator !== undefined && selectionElts[i].separator === "") - selectionElts[i].separator = undefined; - this._datasSelectors.push(selectionElts[i]); - } - } - }, - enumerable: false, - configurable: true - }); - Object.defineProperty(freeDatas2HTML.prototype, "datasSortingColumns", { - get: function () { - return this._datasSortingColumns; - }, - set: function (sortingColumns) { - this._datasSortingColumns = []; - for (var i = 0; i < sortingColumns.length; i++) { - if (Number.isInteger(sortingColumns[i].datasFieldNb) === false || sortingColumns[i].datasFieldNb < 0) - console.error(errors.needNaturalNumber); - else { - sortingColumns[i].order = undefined; - this._datasSortingColumns.push(sortingColumns[i]); - } - } - }, - enumerable: false, - configurable: true - }); - freeDatas2HTML.prototype.parse = function () { - return __awaiter(this, void 0, void 0, function () { - var converter; - return __generator(this, function (_a) { - converter = this; - return [2, new Promise(function (resolve, reject) { - if (converter._datasSourceUrl !== "") { - Papa.parse(converter._datasSourceUrl, { - quoteChar: '"', - header: true, - complete: function (results) { - converter.parseErrors = results.errors; - converter.parseDatas = results.data; - var realFields = []; - for (var i in results.meta.fields) { - if (results.meta.fields[i].trim() !== "") - realFields.push(results.meta.fields[i]); - } - results.meta.fields = realFields; - converter.parseMeta = results.meta; - resolve(true); - }, - error: function (error) { - reject(new Error(errors.parserFail)); - }, - download: true, - skipEmptyLines: true, - }); - } - else - reject(new Error(errors.needUrl)); - })]; - }); - }); - }; - freeDatas2HTML.prototype.run = function () { - return __awaiter(this, void 0, void 0, function () { - var converter_1, selectorsHTML, i, values, colName, row, checkedValue, checkedValues, i_1, checkedValue, j, selectElement, i; - return __generator(this, function (_a) { - switch (_a.label) { - case 0: - if (this._datasViewElt.eltDOM === undefined) - throw new Error(errors.needDatasElt); - if (this._datasSourceUrl === "") - throw new Error(errors.needUrl); - return [4, this.parse()]; - case 1: - _a.sent(); - if (this.parseDatas.length === 0 || this.parseMeta.fields === undefined) - throw new Error(errors.datasNotFound); - else if (this.stopIfParseErrors && this.parseErrors.length !== 0) - console.error(this.parseErrors); - else { - converter_1 = this; - if (this._datasSelectors.length > 0) { - selectorsHTML = []; - for (i in this._datasSelectors) { - if (this._datasSelectors[i].datasFieldNb > (this.parseMeta.fields.length - 1)) - throw new Error(errors.selectorFieldNotFound); - else { - values = [], colName = this.parseMeta.fields[this._datasSelectors[i].datasFieldNb]; - for (row in this.parseDatas) { - if (this._datasSelectors[i].separator === undefined) { - checkedValue = this.parseDatas[row][colName].trim(); - if (checkedValue !== "" && values.indexOf(checkedValue) === -1) - values.push(checkedValue); - } - else { - checkedValues = this.parseDatas[row][colName].split(this._datasSelectors[i].separator); - for (i_1 in checkedValues) { - checkedValue = checkedValues[i_1].trim(); - if (checkedValue !== "" && values.indexOf(checkedValue) === -1) - values.push(checkedValue); - } - } - } - if (values.length > 0) { - values.sort(compare()); - this._datasSelectors[i].name = colName; - this._datasSelectors[i].values = values; - selectorsHTML[i] = ""; - this._datasSelectors[i].eltDOM.innerHTML = selectorsHTML[i]; - selectElement = document.getElementById("freeDatas2HTMLSelector" + i); - selectElement.addEventListener('change', function (e) { - converter_1.datasHTML = converter_1.createDatasHTML(converter_1.parseMeta.fields, converter_1.parseDatas); - converter_1.refreshView(); - }); - } - } - } - } - for (i in this._datasSortingColumns) { - if (this._datasSortingColumns[i].datasFieldNb > (this.parseMeta.fields.length - 1)) - throw new Error(errors.sortingColumnsFieldNotFound); - } - this.datasHTML = this.createDatasHTML(this.parseMeta.fields, this.parseDatas); - this.refreshView(); - return [2, true]; - } - return [2]; - } - }); - }); - }; - freeDatas2HTML.prototype.refreshView = function () { - if (this._datasViewElt.eltDOM !== undefined) { - var converter_2 = this; - this._datasViewElt.eltDOM.innerHTML = this.datasHTML; - if (this._datasSortingColumns.length > 0) { - var getTableTh = document.querySelectorAll("table th"); - if (getTableTh !== null) { - var _loop_1 = function (i) { - var datasFieldNb = this_1._datasSortingColumns[i].datasFieldNb; - var htmlContent = getTableTh[datasFieldNb].innerHTML; - htmlContent = "" + htmlContent + ""; - getTableTh[datasFieldNb].innerHTML = htmlContent; - var sortingElement = document.getElementById("freeDatas2HTMLSorting" + datasFieldNb); - sortingElement.addEventListener('click', function (e) { - e.preventDefault(); - var order = converter_2.datasSortingColumns[i].order; - if (order === undefined || order === "desc") - converter_2.datasSortingColumns[i].order = "asc"; - else - converter_2.datasSortingColumns[i].order = "desc"; - converter_2._datasSortedColumn = converter_2.datasSortingColumns[i]; - converter_2.datasHTML = converter_2.createDatasHTML(converter_2.parseMeta.fields, converter_2.parseDatas); - converter_2.refreshView(); - }); - }; - var this_1 = this; - for (var i in this._datasSortingColumns) { - _loop_1(i); - } - } - } - } - }; - freeDatas2HTML.prototype.createDatasHTML = function (fields, datas) { - var checkSelectorExist, filters = []; - for (var i in this._datasSelectors) { - checkSelectorExist = document.querySelector("#" + this._datasSelectors[i].id + " select"); - if (checkSelectorExist != null && checkSelectorExist.selectedIndex != 0) - filters.push({ field: this._datasSelectors[i].name, value: this._datasSelectors[i].values[checkSelectorExist.selectedIndex - 1], separator: this._datasSelectors[i].separator }); - } - if (this._datasSortedColumn !== undefined) { - var col_1 = fields[this._datasSortedColumn.datasFieldNb]; - var colOrder_1 = this._datasSortedColumn.order; - datas.sort(function (a, b) { return compare({ order: colOrder_1 })(a[col_1], b[col_1]); }); - } - var datasHTML = ""; - for (var i in fields) - datasHTML += ""; - datasHTML += ""; - for (var row in datas) { - var visible = true; - if (filters.length !== 0) { - for (var i in filters) { - if (filters[i].separator === undefined) { - if (datas[row][filters[i].field].trim() != filters[i].value) - visible = false; - } - else { - var checkedValues = datas[row][filters[i].field].split(filters[i].separator), finded = false; - for (var j in checkedValues) { - if (checkedValues[j].trim() === filters[i].value) { - finded = true; - break; - } - } - if (!finded) - visible = false; - } - } - } - if (visible) { - datasHTML += ""; - for (var field in datas[row]) { - if (fields.indexOf(field) !== -1) - datasHTML += ""; - } - datasHTML += ""; - } - } - datasHTML += "
" + fields[i] + "
" + datas[row][field] + "
"; - return datasHTML; - }; - return freeDatas2HTML; -}()); - - +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"freeDatas2HTML\": () => (/* binding */ freeDatas2HTML)\n/* harmony export */ });\nvar __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) {\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\n return new (P || (P = Promise))(function (resolve, reject) {\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\n step((generator = generator.apply(thisArg, _arguments || [])).next());\n });\n};\nvar __generator = (undefined && undefined.__generator) || function (thisArg, body) {\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\n function verb(n) { return function (v) { return step([n, v]); }; }\n function step(op) {\n if (f) throw new TypeError(\"Generator is already executing.\");\n while (_) try {\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\n if (y = 0, t) op = [op[0] & 2, t.value];\n switch (op[0]) {\n case 0: case 1: t = op; break;\n case 4: _.label++; return { value: op[1], done: false };\n case 5: _.label++; y = op[1]; op = [0]; continue;\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\n default:\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\n if (t[2]) _.ops.pop();\n _.trys.pop(); continue;\n }\n op = body.call(thisArg, _);\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\n }\n};\nvar Papa = __webpack_require__(/*! papaparse */ \"./node_modules/papaparse/papaparse.min.js\");\nvar errors = __webpack_require__(/*! ./errors.js */ \"./src/errors.js\");\nvar compare = __webpack_require__(/*! natural-orderby */ \"./node_modules/natural-orderby/esm/natural-orderby.js\").compare;\nvar freeDatas2HTML = (function () {\n function freeDatas2HTML() {\n this._datasViewElt = { id: \"\", eltDOM: undefined };\n this._datasSourceUrl = \"\";\n this._datasSelectors = [];\n this._datasSortingColumns = [];\n this.parseMeta = undefined;\n this.parseDatas = [];\n this.parseErrors = [];\n this.datasHTML = \"\";\n this.stopIfParseErrors = false;\n }\n Object.defineProperty(freeDatas2HTML.prototype, \"datasViewElt\", {\n set: function (elt) {\n var checkContainerExist = document.getElementById(elt.id);\n if (checkContainerExist === null)\n throw new Error(errors.elementNotFound + elt.id);\n else {\n this._datasViewElt.id = elt.id;\n this._datasViewElt.eltDOM = checkContainerExist;\n }\n },\n enumerable: false,\n configurable: true\n });\n Object.defineProperty(freeDatas2HTML.prototype, \"datasSourceUrl\", {\n set: function (url) {\n if (url.trim().length === 0)\n throw new Error(errors.needUrl);\n else\n this._datasSourceUrl = url.trim();\n },\n enumerable: false,\n configurable: true\n });\n Object.defineProperty(freeDatas2HTML.prototype, \"datasSelectors\", {\n get: function () {\n return this._datasSelectors;\n },\n set: function (selectionElts) {\n this._datasSelectors = [];\n var checkContainerExist;\n for (var i = 0; i < selectionElts.length; i++) {\n checkContainerExist = document.getElementById(selectionElts[i].id);\n if (checkContainerExist === null)\n console.error(errors.elementNotFound + selectionElts[i].id);\n else if (Number.isInteger(selectionElts[i].datasFieldNb) === false || selectionElts[i].datasFieldNb < 0)\n console.error(errors.needNaturalNumber);\n else {\n selectionElts[i].eltDOM = checkContainerExist;\n if (selectionElts[i].separator !== undefined && selectionElts[i].separator === \"\")\n selectionElts[i].separator = undefined;\n this._datasSelectors.push(selectionElts[i]);\n }\n }\n },\n enumerable: false,\n configurable: true\n });\n Object.defineProperty(freeDatas2HTML.prototype, \"datasSortingColumns\", {\n get: function () {\n return this._datasSortingColumns;\n },\n set: function (sortingColumns) {\n this._datasSortingColumns = [];\n for (var i = 0; i < sortingColumns.length; i++) {\n if (Number.isInteger(sortingColumns[i].datasFieldNb) === false || sortingColumns[i].datasFieldNb < 0)\n console.error(errors.needNaturalNumber);\n else {\n sortingColumns[i].order = undefined;\n this._datasSortingColumns.push(sortingColumns[i]);\n }\n }\n },\n enumerable: false,\n configurable: true\n });\n freeDatas2HTML.prototype.parse = function () {\n return __awaiter(this, void 0, void 0, function () {\n var converter;\n return __generator(this, function (_a) {\n converter = this;\n return [2, new Promise(function (resolve, reject) {\n if (converter._datasSourceUrl !== \"\") {\n Papa.parse(converter._datasSourceUrl, {\n quoteChar: '\"',\n header: true,\n complete: function (results) {\n converter.parseErrors = results.errors;\n converter.parseDatas = results.data;\n var realFields = [];\n for (var i in results.meta.fields) {\n if (results.meta.fields[i].trim() !== \"\")\n realFields.push(results.meta.fields[i]);\n }\n results.meta.fields = realFields;\n converter.parseMeta = results.meta;\n resolve(true);\n },\n error: function (error) {\n reject(new Error(errors.parserFail));\n },\n download: true,\n skipEmptyLines: true,\n });\n }\n else\n reject(new Error(errors.needUrl));\n })];\n });\n });\n };\n freeDatas2HTML.prototype.run = function () {\n return __awaiter(this, void 0, void 0, function () {\n var converter_1, selectorsHTML, i, values, colName, row, checkedValue, checkedValues, i_1, checkedValue, j, selectElement, i;\n return __generator(this, function (_a) {\n switch (_a.label) {\n case 0:\n if (this._datasViewElt.eltDOM === undefined)\n throw new Error(errors.needDatasElt);\n if (this._datasSourceUrl === \"\")\n throw new Error(errors.needUrl);\n return [4, this.parse()];\n case 1:\n _a.sent();\n if (this.parseDatas.length === 0 || this.parseMeta.fields === undefined)\n throw new Error(errors.datasNotFound);\n else if (this.stopIfParseErrors && this.parseErrors.length !== 0)\n console.error(this.parseErrors);\n else {\n converter_1 = this;\n if (this._datasSelectors.length > 0) {\n selectorsHTML = [];\n for (i in this._datasSelectors) {\n if (this._datasSelectors[i].datasFieldNb > (this.parseMeta.fields.length - 1))\n throw new Error(errors.selectorFieldNotFound);\n else {\n values = [], colName = this.parseMeta.fields[this._datasSelectors[i].datasFieldNb];\n for (row in this.parseDatas) {\n if (this._datasSelectors[i].separator === undefined) {\n checkedValue = this.parseDatas[row][colName].trim();\n if (checkedValue !== \"\" && values.indexOf(checkedValue) === -1)\n values.push(checkedValue);\n }\n else {\n checkedValues = this.parseDatas[row][colName].split(this._datasSelectors[i].separator);\n for (i_1 in checkedValues) {\n checkedValue = checkedValues[i_1].trim();\n if (checkedValue !== \"\" && values.indexOf(checkedValue) === -1)\n values.push(checkedValue);\n }\n }\n }\n if (values.length > 0) {\n values.sort(compare());\n this._datasSelectors[i].name = colName;\n this._datasSelectors[i].values = values;\n selectorsHTML[i] = \"\";\n this._datasSelectors[i].eltDOM.innerHTML = selectorsHTML[i];\n selectElement = document.getElementById(\"freeDatas2HTMLSelector\" + i);\n selectElement.addEventListener('change', function (e) {\n converter_1.datasHTML = converter_1.createDatasHTML(converter_1.parseMeta.fields, converter_1.parseDatas);\n converter_1.refreshView();\n });\n }\n }\n }\n }\n for (i in this._datasSortingColumns) {\n if (this._datasSortingColumns[i].datasFieldNb > (this.parseMeta.fields.length - 1))\n throw new Error(errors.sortingColumnsFieldNotFound);\n }\n this.datasHTML = this.createDatasHTML(this.parseMeta.fields, this.parseDatas);\n this.refreshView();\n return [2, true];\n }\n return [2];\n }\n });\n });\n };\n freeDatas2HTML.prototype.refreshView = function () {\n if (this._datasViewElt.eltDOM !== undefined) {\n var converter_2 = this;\n this._datasViewElt.eltDOM.innerHTML = this.datasHTML;\n if (this._datasSortingColumns.length > 0) {\n var getTableTh = document.querySelectorAll(\"table th\");\n if (getTableTh !== null) {\n var _loop_1 = function (i) {\n var datasFieldNb = this_1._datasSortingColumns[i].datasFieldNb;\n var htmlContent = getTableTh[datasFieldNb].innerHTML;\n htmlContent = \"\" + htmlContent + \"\";\n getTableTh[datasFieldNb].innerHTML = htmlContent;\n var sortingElement = document.getElementById(\"freeDatas2HTMLSorting\" + datasFieldNb);\n sortingElement.addEventListener('click', function (e) {\n e.preventDefault();\n var order = converter_2.datasSortingColumns[i].order;\n if (order === undefined || order === \"desc\")\n converter_2.datasSortingColumns[i].order = \"asc\";\n else\n converter_2.datasSortingColumns[i].order = \"desc\";\n converter_2._datasSortedColumn = converter_2.datasSortingColumns[i];\n converter_2.datasHTML = converter_2.createDatasHTML(converter_2.parseMeta.fields, converter_2.parseDatas);\n converter_2.refreshView();\n });\n };\n var this_1 = this;\n for (var i in this._datasSortingColumns) {\n _loop_1(i);\n }\n }\n }\n }\n };\n freeDatas2HTML.prototype.createDatasHTML = function (fields, datas) {\n var checkSelectorExist, filters = [];\n for (var i in this._datasSelectors) {\n checkSelectorExist = document.querySelector(\"#\" + this._datasSelectors[i].id + \" select\");\n if (checkSelectorExist != null && checkSelectorExist.selectedIndex != 0)\n filters.push({ field: this._datasSelectors[i].name, value: this._datasSelectors[i].values[checkSelectorExist.selectedIndex - 1], separator: this._datasSelectors[i].separator });\n }\n if (this._datasSortedColumn !== undefined) {\n var col_1 = fields[this._datasSortedColumn.datasFieldNb];\n var colOrder_1 = this._datasSortedColumn.order;\n datas.sort(function (a, b) { return compare({ order: colOrder_1 })(a[col_1], b[col_1]); });\n }\n var datasHTML = \"\";\n for (var i in fields)\n datasHTML += \"\";\n datasHTML += \"\";\n for (var row in datas) {\n var visible = true;\n if (filters.length !== 0) {\n for (var i in filters) {\n if (filters[i].separator === undefined) {\n if (datas[row][filters[i].field].trim() != filters[i].value)\n visible = false;\n }\n else {\n var checkedValues = datas[row][filters[i].field].split(filters[i].separator), finded = false;\n for (var j in checkedValues) {\n if (checkedValues[j].trim() === filters[i].value) {\n finded = true;\n break;\n }\n }\n if (!finded)\n visible = false;\n }\n }\n }\n if (visible) {\n datasHTML += \"\";\n for (var field in datas[row]) {\n if (fields.indexOf(field) !== -1)\n datasHTML += \"\";\n }\n datasHTML += \"\";\n }\n }\n datasHTML += \"
\" + fields[i] + \"
\" + datas[row][field] + \"
\";\n return datasHTML;\n };\n return freeDatas2HTML;\n}());\n\n\n\n//# sourceURL=webpack://freedatas2html/./src/freeDatas2HTML.ts?"); /***/ }), @@ -992,17 +58,7 @@ var freeDatas2HTML = (function () { \***********************/ /***/ ((module) => { -module.exports = -{ - datasNotFound : "Aucune donnée n'a été trouvée.", - elementNotFound : "Aucun élément HTML n'a été trouvé ayant comme \"id\" : ", - needDatasElt: "Merci de fournir un id valide pour l'élément où afficher les données.", - needNaturalNumber: "Merci de fournir un nombre entier supérieur ou égal à zéro pour désigner chaque colonne.", - needUrl: "Merci de fournir une url valide pour le fichier CSV à parser.", - parserFail: "La lecture des données du fichier a échoué.", - selectorFieldNotFound: "Au moins une des colonnes devant servir à filtrer les données n'existe pas dans le fichier.", - sortingColumnsFieldNotFound: "Au moins une des colonnes devant servir à classer les données n'existe pas dans le fichier.", -}; +eval("module.exports =\n{\n datasNotFound : \"Aucune donnée n'a été trouvée.\",\n elementNotFound : \"Aucun élément HTML n'a été trouvé ayant comme \\\"id\\\" : \",\n needDatasElt: \"Merci de fournir un id valide pour l'élément où afficher les données.\",\n needNaturalNumber: \"Merci de fournir un nombre entier supérieur ou égal à zéro pour désigner chaque colonne.\",\n needUrl: \"Merci de fournir une url valide pour le fichier CSV à parser.\",\n parserFail: \"La lecture des données du fichier a échoué.\",\n selectorFieldNotFound: \"Au moins une des colonnes devant servir à filtrer les données n'existe pas dans le fichier.\",\n sortingColumnsFieldNotFound: \"Au moins une des colonnes devant servir à classer les données n'existe pas dans le fichier.\",\n};\n\n//# sourceURL=webpack://freedatas2html/./src/errors.js?"); /***/ }) @@ -1062,81 +118,11 @@ module.exports = /******/ })(); /******/ /************************************************************************/ -var __webpack_exports__ = {}; -// This entry need to be wrapped in an IIFE because it need to be in strict mode. -(() => { -"use strict"; -/*!*****************************!*\ - !*** ./src/firstExample.ts ***! - \*****************************/ -__webpack_require__.r(__webpack_exports__); -/* harmony import */ var _freeDatas2HTML__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./freeDatas2HTML */ "./src/freeDatas2HTML.ts"); -var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) { - function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } - return new (P || (P = Promise))(function (resolve, reject) { - function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } - function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } - function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } - step((generator = generator.apply(thisArg, _arguments || [])).next()); - }); -}; -var __generator = (undefined && undefined.__generator) || function (thisArg, body) { - var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; - return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g; - function verb(n) { return function (v) { return step([n, v]); }; } - function step(op) { - if (f) throw new TypeError("Generator is already executing."); - while (_) try { - if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t; - if (y = 0, t) op = [op[0] & 2, t.value]; - switch (op[0]) { - case 0: case 1: t = op; break; - case 4: _.label++; return { value: op[1], done: false }; - case 5: _.label++; y = op[1]; op = [0]; continue; - case 7: op = _.ops.pop(); _.trys.pop(); continue; - default: - if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; } - if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; } - if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; } - if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; } - if (t[2]) _.ops.pop(); - _.trys.pop(); continue; - } - op = body.call(thisArg, _); - } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; } - if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true }; - } -}; - -var initialise = function () { return __awaiter(void 0, void 0, void 0, function () { - var converter, e_1; - return __generator(this, function (_a) { - switch (_a.label) { - case 0: - _a.trys.push([0, 2, , 3]); - converter = new _freeDatas2HTML__WEBPACK_IMPORTED_MODULE_0__.freeDatas2HTML(); - converter.datasViewElt = { id: "datas" }; - converter.datasSelectors = [{ datasFieldNb: 3, id: "filtre1" }, { datasFieldNb: 4, id: "filtre2" }, { datasFieldNb: 5, id: "filtre3", separator: "," }]; - converter.datasSortingColumns = [{ datasFieldNb: 0 }, { datasFieldNb: 1 }, { datasFieldNb: 2 }]; - converter.datasSourceUrl = "http://localhost:8080/datas/elements-chimiques.csv"; - return [4, converter.run()]; - case 1: - _a.sent(); - return [3, 3]; - case 2: - e_1 = _a.sent(); - console.error(e_1); - if (document.getElementById("datas") !== null) - document.getElementById("datas").innerHTML = "Désolé, mais un problème technique empêche l'affichage des données."; - return [3, 3]; - case 3: return [2]; - } - }); -}); }; -initialise(); - -})(); - +/******/ +/******/ // startup +/******/ // Load entry module and return exports +/******/ // This entry module can't be inlined because the eval devtool is used. +/******/ var __webpack_exports__ = __webpack_require__("./src/firstExample.ts"); +/******/ /******/ })() -; -//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9KUy9maXJzdEV4YW1wbGUuYXBwLmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7OztBQUFBO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBOztBQUVBO0FBQ0EsdURBQXVEOztBQUV2RCw2QkFBNkI7O0FBRTdCLHlFQUF5RTs7QUFFekUseUVBQXlFLElBQUksT0FBTyxJQUFJLE9BQU8sSUFBSSxtQkFBbUIsRUFBRSxJQUFJOztBQUU1SCxnQ0FBZ0MsRUFBRTtBQUNsQzs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQSxrQkFBa0IsVUFBVTtBQUM1QjtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBLHdCQUF3QjtBQUN4Qix3QkFBd0I7QUFDeEI7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQSxRQUFRO0FBQ1I7QUFDQTtBQUNBLFFBQVE7QUFDUjtBQUNBO0FBQ0EsUUFBUTtBQUNSO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsSUFBSTs7O0FBR0o7QUFDQTtBQUNBOztBQUVBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUEsa0JBQWtCLFlBQVk7QUFDOUI7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQSxNQUFNO0FBQ047O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBLE1BQU07QUFDTjtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBOztBQUVBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0EsR0FBRyxHQUFHOztBQUVOO0FBQ0E7QUFDQTtBQUNBLEtBQUs7QUFDTDtBQUNBO0FBQ0E7QUFDQTtBQUNBLEdBQUcsR0FBRzs7QUFFTjtBQUNBO0FBQ0EsR0FBRztBQUNIO0FBQ0E7QUFDQSxHQUFHO0FBQ0g7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7O0FBRUE7QUFDQTtBQUNBLEdBQUc7QUFDSDtBQUNBOztBQUVBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7O0FBRUE7QUFDQTtBQUNBLEdBQUc7QUFDSDtBQUNBOztBQUVBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxZQUFZLFVBQVU7QUFDdEI7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsTUFBTTtBQUNOO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsTUFBTTtBQUNOO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsTUFBTTtBQUNOO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsTUFBTTtBQUNOO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsTUFBTTtBQUNOO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsTUFBTTtBQUNOO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsWUFBWTtBQUNaO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsWUFBWTtBQUNaO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsWUFBWTtBQUNaO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsWUFBWTtBQUNaO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsWUFBWTtBQUNaO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsWUFBWTtBQUNaO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBOztBQUVBO0FBQ0E7QUFDQSxJQUFJO0FBQ0o7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLFlBQVksVUFBVTtBQUN0QjtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNB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QjtBQUM1RztBQUNBLG9DQUFvQyxNQUFNLCtCQUErQixZQUFZO0FBQ3JGLG1DQUFtQyxNQUFNLG1DQUFtQyxZQUFZO0FBQ3hGLGdDQUFnQztBQUNoQztBQUNBLEtBQUs7QUFDTDtBQUNBLG1CQUFtQixTQUFJLElBQUksU0FBSTtBQUMvQixjQUFjLDZCQUE2QiwwQkFBMEIsY0FBYyxxQkFBcUI7QUFDeEcsaUJBQWlCLG9EQUFvRCxxRUFBcUUsY0FBYztBQUN4Six1QkFBdUIsc0JBQXNCO0FBQzdDO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLHdDQUF3QztBQUN4QyxtQ0FBbUMsU0FBUztBQUM1QyxtQ0FBbUMsV0FBVyxVQUFVO0FBQ3hELDBDQUEwQyxjQUFjO0FBQ3hEO0FBQ0EsOEdBQThHLE9BQU87QUFDckgsaUZBQWlGLGlCQUFpQjtBQUNsRyx5REFBeUQsZ0JBQWdCLFFBQVE7QUFDakYsK0NBQStDLGdCQUFnQixnQkFBZ0I7QUFDL0U7QUFDQSxrQ0FBa0M7QUFDbEM7QUFDQTtBQUNBLFVBQVUsWUFBWSxhQUFhLFNBQVMsVUFBVTtBQUN0RCxvQ0FBb0MsU0FBUztBQUM3QztBQUNBO0FBQ2tEO0FBQ2xELCtCQUErQjtBQUMvQjtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsZ0NBQWdDLDJEQUFjO0FBQzlDLDJDQUEyQztBQUMzQyw4Q0FBOEMsZ0NBQWdDLElBQUksZ0NBQWdDLElBQUksZ0RBQWdEO0FBQ3RLLG1EQUFtRCxpQkFBaUIsSUFBSSxpQkFBaUIsSUFBSSxpQkFBaUI7QUFDOUc7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxLQUFLO0FBQ0wsQ0FBQztBQUNEIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vZnJlZWRhdGFzMmh0bWwvLi9ub2RlX21vZHVsZXMvbmF0dXJhbC1vcmRlcmJ5L2VzbS9uYXR1cmFsLW9yZGVyYnkuanMiLCJ3ZWJwYWNrOi8vZnJlZWRhdGFzMmh0bWwvLi9ub2RlX21vZHVsZXMvcGFwYXBhcnNlL3BhcGFwYXJzZS5taW4uanMiLCJ3ZWJwYWNrOi8vZnJlZWRhdGFzMmh0bWwvLi9zcmMvZnJlZURhdGFzMkhUTUwudHMiLCJ3ZWJwYWNrOi8vZnJlZWRhdGFzMmh0bWwvLi9zcmMvZXJyb3JzLmpzIiwid2VicGFjazovL2ZyZWVkYXRhczJodG1sL3dlYnBhY2svYm9vdHN0cmFwIiwid2VicGFjazovL2ZyZWVkYXRhczJodG1sL3dlYnBhY2svcnVudGltZS9kZWZpbmUgcHJvcGVydHkgZ2V0dGVycyIsIndlYnBhY2s6Ly9mcmVlZGF0YXMyaHRtbC93ZWJwYWNrL3J1bnRpbWUvaGFzT3duUHJvcGVydHkgc2hvcnRoYW5kIiwid2VicGFjazovL2ZyZWVkYXRhczJodG1sL3dlYnBhY2svcnVudGltZS9tYWtlIG5hbWVzcGFjZSBvYmplY3QiLCJ3ZWJwYWNrOi8vZnJlZWRhdGFzMmh0bWwvLi9zcmMvZmlyc3RFeGFtcGxlLnRzIl0sInNvdXJjZXNDb250ZW50IjpbInZhciBjb21wYXJlTnVtYmVycyA9IGZ1bmN0aW9uIGNvbXBhcmVOdW1iZXJzKG51bWJlckEsIG51bWJlckIpIHtcbiAgaWYgKG51bWJlckEgPCBudW1iZXJCKSB7XG4gICAgcmV0dXJuIC0xO1xuICB9XG5cbiAgaWYgKG51bWJlckEgPiBudW1iZXJCKSB7XG4gICAgcmV0dXJuIDE7XG4gIH1cblxuICByZXR1cm4gMDtcbn07XG5cbnZhciBSRV9OVU1CRVJTID0gLyheMHhbXFxkYS1mQS1GXSskfF4oWystXT8oPzpcXGQrKD86XFwuXFxkKik/fFxcLlxcZCspKD86W2VFXVsrLV0/XFxkKyk/KD8hXFwuXFxkKykoPz1cXER8XFxzfCQpKXxcXGQrKS9nO1xudmFyIFJFX0xFQURJTkdfT1JfVFJBSUxJTkdfV0hJVEVTUEFDRVMgPSAvXlxccyt8XFxzKyQvZzsgLy8gdHJpbSBwcmUtcG9zdCB3aGl0ZXNwYWNlXG5cbnZhciBSRV9XSElURVNQQUNFUyA9IC9cXHMrL2c7IC8vIG5vcm1hbGl6ZSBhbGwgd2hpdGVzcGFjZSB0byBzaW5nbGUgJyAnIGNoYXJhY3RlclxuXG52YXIgUkVfSU5UX09SX0ZMT0FUID0gL15bKy1dPyg/OlxcZCsoPzpcXC5cXGQqKT98XFwuXFxkKykoPzpbZUVdWystXT9cXGQrKT8kLzsgLy8gaWRlbnRpZnkgaW50ZWdlcnMgYW5kIGZsb2F0c1xuXG52YXIgUkVfREFURSA9IC8oXihbXFx3IF0rLD9bXFx3IF0rKT9bXFx3IF0rLD9bXFx3IF0rXFxkKzpcXGQrKDpcXGQrKT9bXFx3IF0/fF5cXGR7MSw0fVsvLV1cXGR7MSw0fVsvLV1cXGR7MSw0fXxeXFx3KywgXFx3KyBcXGQrLCBcXGR7NH0pLzsgLy8gaWRlbnRpZnkgZGF0ZSBzdHJpbmdzXG5cbnZhciBSRV9MRUFESU5HX1pFUk8gPSAvXjArWzEtOV17MX1bMC05XSokLztcbnZhciBSRV9VTklDT0RFX0NIQVJBQ1RFUlMgPSAvW15cXHgwMC1cXHg4MF0vO1xuXG52YXIgY29tcGFyZVVuaWNvZGUgPSBmdW5jdGlvbiBjb21wYXJlVW5pY29kZShzdHJpbmdBLCBzdHJpbmdCKSB7XG4gIHZhciByZXN1bHQgPSBzdHJpbmdBLmxvY2FsZUNvbXBhcmUoc3RyaW5nQik7XG4gIHJldHVybiByZXN1bHQgPyByZXN1bHQgLyBNYXRoLmFicyhyZXN1bHQpIDogMDtcbn07XG5cbnZhciBzdHJpbmdDb21wYXJlID0gZnVuY3Rpb24gc3RyaW5nQ29tcGFyZShzdHJpbmdBLCBzdHJpbmdCKSB7XG4gIGlmIChzdHJpbmdBIDwgc3RyaW5nQikge1xuICAgIHJldHVybiAtMTtcbiAgfVxuXG4gIGlmIChzdHJpbmdBID4gc3RyaW5nQikge1xuICAgIHJldHVybiAxO1xuICB9XG5cbiAgcmV0dXJuIDA7XG59O1xuXG52YXIgY29tcGFyZUNodW5rcyA9IGZ1bmN0aW9uIGNvbXBhcmVDaHVua3MoY2h1bmtzQSwgY2h1bmtzQikge1xuICB2YXIgbGVuZ3RoQSA9IGNodW5rc0EubGVuZ3RoO1xuICB2YXIgbGVuZ3RoQiA9IGNodW5rc0IubGVuZ3RoO1xuICB2YXIgc2l6ZSA9IE1hdGgubWluKGxlbmd0aEEsIGxlbmd0aEIpO1xuXG4gIGZvciAodmFyIGkgPSAwOyBpIDwgc2l6ZTsgaSsrKSB7XG4gICAgdmFyIGNodW5rQSA9IGNodW5rc0FbaV07XG4gICAgdmFyIGNodW5rQiA9IGNodW5rc0JbaV07XG5cbiAgICBpZiAoY2h1bmtBLm5vcm1hbGl6ZWRTdHJpbmcgIT09IGNodW5rQi5ub3JtYWxpemVkU3RyaW5nKSB7XG4gICAgICBpZiAoY2h1bmtBLm5vcm1hbGl6ZWRTdHJpbmcgPT09ICcnICE9PSAoY2h1bmtCLm5vcm1hbGl6ZWRTdHJpbmcgPT09ICcnKSkge1xuICAgICAgICAvLyBlbXB0eSBzdHJpbmdzIGhhdmUgbG93ZXN0IHZhbHVlXG4gICAgICAgIHJldHVybiBjaHVua0Eubm9ybWFsaXplZFN0cmluZyA9PT0gJycgPyAtMSA6IDE7XG4gICAgICB9XG5cbiAgICAgIGlmIChjaHVua0EucGFyc2VkTnVtYmVyICE9PSB1bmRlZmluZWQgJiYgY2h1bmtCLnBhcnNlZE51bWJlciAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIC8vIGNvbXBhcmUgbnVtYmVyc1xuICAgICAgICB2YXIgcmVzdWx0ID0gY29tcGFyZU51bWJlcnMoY2h1bmtBLnBhcnNlZE51bWJlciwgY2h1bmtCLnBhcnNlZE51bWJlcik7XG5cbiAgICAgICAgaWYgKHJlc3VsdCA9PT0gMCkge1xuICAgICAgICAgIC8vIGNvbXBhcmUgc3RyaW5nIHZhbHVlLCBpZiBwYXJzZWQgbnVtYmVycyBhcmUgZXF1YWxcbiAgICAgICAgICAvLyBFeGFtcGxlOlxuICAgICAgICAgIC8vIGNodW5rQSA9IHsgcGFyc2VkTnVtYmVyOiAxLCBub3JtYWxpemVkU3RyaW5nOiBcIjAwMVwiIH1cbiAgICAgICAgICAvLyBjaHVua0IgPSB7IHBhcnNlZE51bWJlcjogMSwgbm9ybWFsaXplZFN0cmluZzogXCIwMVwiIH1cbiAgICAgICAgICAvLyBjaHVua0EucGFyc2VkTnVtYmVyID09PSBjaHVua0IucGFyc2VkTnVtYmVyXG4gICAgICAgICAgLy8gY2h1bmtBLm5vcm1hbGl6ZWRTdHJpbmcgPCBjaHVua0Iubm9ybWFsaXplZFN0cmluZ1xuICAgICAgICAgIHJldHVybiBzdHJpbmdDb21wYXJlKGNodW5rQS5ub3JtYWxpemVkU3RyaW5nLCBjaHVua0Iubm9ybWFsaXplZFN0cmluZyk7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gcmVzdWx0O1xuICAgICAgfSBlbHNlIGlmIChjaHVua0EucGFyc2VkTnVtYmVyICE9PSB1bmRlZmluZWQgfHwgY2h1bmtCLnBhcnNlZE51bWJlciAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIC8vIG51bWJlciA8IHN0cmluZ1xuICAgICAgICByZXR1cm4gY2h1bmtBLnBhcnNlZE51bWJlciAhPT0gdW5kZWZpbmVkID8gLTEgOiAxO1xuICAgICAgfSBlbHNlIGlmIChSRV9VTklDT0RFX0NIQVJBQ1RFUlMudGVzdChjaHVua0Eubm9ybWFsaXplZFN0cmluZyArIGNodW5rQi5ub3JtYWxpemVkU3RyaW5nKSAmJiBjaHVua0Eubm9ybWFsaXplZFN0cmluZy5sb2NhbGVDb21wYXJlKSB7XG4gICAgICAgIC8vIHVzZSBsb2NhbGUgY29tcGFyaXNvbiBvbmx5IGlmIG9uZSBvZiB0aGUgY2h1bmtzIGNvbnRhaW5zIHVuaWNvZGUgY2hhcmFjdGVyc1xuICAgICAgICByZXR1cm4gY29tcGFyZVVuaWNvZGUoY2h1bmtBLm5vcm1hbGl6ZWRTdHJpbmcsIGNodW5rQi5ub3JtYWxpemVkU3RyaW5nKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIHVzZSBjb21tb24gc3RyaW5nIGNvbXBhcmlzb24gZm9yIHBlcmZvcm1hbmNlIHJlYXNvblxuICAgICAgICByZXR1cm4gc3RyaW5nQ29tcGFyZShjaHVua0Eubm9ybWFsaXplZFN0cmluZywgY2h1bmtCLm5vcm1hbGl6ZWRTdHJpbmcpO1xuICAgICAgfVxuICAgIH1cbiAgfSAvLyBpZiB0aGUgY2h1bmtzIGFyZSBlcXVhbCBzbyBmYXIsIHRoZSBvbmUgd2hpY2ggaGFzIG1vcmUgY2h1bmtzIGlzIGdyZWF0ZXIgdGhhbiB0aGUgb3RoZXIgb25lXG5cblxuICBpZiAobGVuZ3RoQSA+IHNpemUgfHwgbGVuZ3RoQiA+IHNpemUpIHtcbiAgICByZXR1cm4gbGVuZ3RoQSA8PSBzaXplID8gLTEgOiAxO1xuICB9XG5cbiAgcmV0dXJuIDA7XG59O1xuXG52YXIgY29tcGFyZU90aGVyVHlwZXMgPSBmdW5jdGlvbiBjb21wYXJlT3RoZXJUeXBlcyh2YWx1ZUEsIHZhbHVlQikge1xuICBpZiAoIXZhbHVlQS5jaHVua3MgPyB2YWx1ZUIuY2h1bmtzIDogIXZhbHVlQi5jaHVua3MpIHtcbiAgICByZXR1cm4gIXZhbHVlQS5jaHVua3MgPyAxIDogLTE7XG4gIH1cblxuICBpZiAodmFsdWVBLmlzTmFOID8gIXZhbHVlQi5pc05hTiA6IHZhbHVlQi5pc05hTikge1xuICAgIHJldHVybiB2YWx1ZUEuaXNOYU4gPyAtMSA6IDE7XG4gIH1cblxuICBpZiAodmFsdWVBLmlzU3ltYm9sID8gIXZhbHVlQi5pc1N5bWJvbCA6IHZhbHVlQi5pc1N5bWJvbCkge1xuICAgIHJldHVybiB2YWx1ZUEuaXNTeW1ib2wgPyAtMSA6IDE7XG4gIH1cblxuICBpZiAodmFsdWVBLmlzT2JqZWN0ID8gIXZhbHVlQi5pc09iamVjdCA6IHZhbHVlQi5pc09iamVjdCkge1xuICAgIHJldHVybiB2YWx1ZUEuaXNPYmplY3QgPyAtMSA6IDE7XG4gIH1cblxuICBpZiAodmFsdWVBLmlzQXJyYXkgPyAhdmFsdWVCLmlzQXJyYXkgOiB2YWx1ZUIuaXNBcnJheSkge1xuICAgIHJldHVybiB2YWx1ZUEuaXNBcnJheSA/IC0xIDogMTtcbiAgfVxuXG4gIGlmICh2YWx1ZUEuaXNGdW5jdGlvbiA/ICF2YWx1ZUIuaXNGdW5jdGlvbiA6IHZhbHVlQi5pc0Z1bmN0aW9uKSB7XG4gICAgcmV0dXJuIHZhbHVlQS5pc0Z1bmN0aW9uID8gLTEgOiAxO1xuICB9XG5cbiAgaWYgKHZhbHVlQS5pc051bGwgPyAhdmFsdWVCLmlzTnVsbCA6IHZhbHVlQi5pc051bGwpIHtcbiAgICByZXR1cm4gdmFsdWVBLmlzTnVsbCA/IC0xIDogMTtcbiAgfVxuXG4gIHJldHVybiAwO1xufTtcblxudmFyIGNvbXBhcmVWYWx1ZXMgPSBmdW5jdGlvbiBjb21wYXJlVmFsdWVzKHZhbHVlQSwgdmFsdWVCKSB7XG4gIGlmICh2YWx1ZUEudmFsdWUgPT09IHZhbHVlQi52YWx1ZSkge1xuICAgIHJldHVybiAwO1xuICB9XG5cbiAgaWYgKHZhbHVlQS5wYXJzZWROdW1iZXIgIT09IHVuZGVmaW5lZCAmJiB2YWx1ZUIucGFyc2VkTnVtYmVyICE9PSB1bmRlZmluZWQpIHtcbiAgICByZXR1cm4gY29tcGFyZU51bWJlcnModmFsdWVBLnBhcnNlZE51bWJlciwgdmFsdWVCLnBhcnNlZE51bWJlcik7XG4gIH1cblxuICBpZiAodmFsdWVBLmNodW5rcyAmJiB2YWx1ZUIuY2h1bmtzKSB7XG4gICAgcmV0dXJuIGNvbXBhcmVDaHVua3ModmFsdWVBLmNodW5rcywgdmFsdWVCLmNodW5rcyk7XG4gIH1cblxuICByZXR1cm4gY29tcGFyZU90aGVyVHlwZXModmFsdWVBLCB2YWx1ZUIpO1xufTtcblxudmFyIGNvbXBhcmVNdWx0aXBsZSA9IGZ1bmN0aW9uIGNvbXBhcmVNdWx0aXBsZShyZWNvcmRBLCByZWNvcmRCLCBvcmRlcnMpIHtcbiAgdmFyIGluZGV4QSA9IHJlY29yZEEuaW5kZXgsXG4gICAgICB2YWx1ZXNBID0gcmVjb3JkQS52YWx1ZXM7XG4gIHZhciBpbmRleEIgPSByZWNvcmRCLmluZGV4LFxuICAgICAgdmFsdWVzQiA9IHJlY29yZEIudmFsdWVzO1xuICB2YXIgbGVuZ3RoID0gdmFsdWVzQS5sZW5ndGg7XG4gIHZhciBvcmRlcnNMZW5ndGggPSBvcmRlcnMubGVuZ3RoO1xuXG4gIGZvciAodmFyIGkgPSAwOyBpIDwgbGVuZ3RoOyBpKyspIHtcbiAgICB2YXIgb3JkZXIgPSBpIDwgb3JkZXJzTGVuZ3RoID8gb3JkZXJzW2ldIDogbnVsbDtcblxuICAgIGlmIChvcmRlciAmJiB0eXBlb2Ygb3JkZXIgPT09ICdmdW5jdGlvbicpIHtcbiAgICAgIHZhciByZXN1bHQgPSBvcmRlcih2YWx1ZXNBW2ldLnZhbHVlLCB2YWx1ZXNCW2ldLnZhbHVlKTtcblxuICAgICAgaWYgKHJlc3VsdCkge1xuICAgICAgICByZXR1cm4gcmVzdWx0O1xuICAgICAgfVxuICAgIH0gZWxzZSB7XG4gICAgICB2YXIgX3Jlc3VsdCA9IGNvbXBhcmVWYWx1ZXModmFsdWVzQVtpXSwgdmFsdWVzQltpXSk7XG5cbiAgICAgIGlmIChfcmVzdWx0KSB7XG4gICAgICAgIHJldHVybiBfcmVzdWx0ICogKG9yZGVyID09PSAnZGVzYycgPyAtMSA6IDEpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIHJldHVybiBpbmRleEEgLSBpbmRleEI7XG59O1xuXG52YXIgY3JlYXRlSWRlbnRpZmllckZuID0gZnVuY3Rpb24gY3JlYXRlSWRlbnRpZmllckZuKGlkZW50aWZpZXIpIHtcbiAgaWYgKHR5cGVvZiBpZGVudGlmaWVyID09PSAnZnVuY3Rpb24nKSB7XG4gICAgLy8gaWRlbnRpZmllciBpcyBhbHJlYWR5IGEgbG9va3VwIGZ1bmN0aW9uXG4gICAgcmV0dXJuIGlkZW50aWZpZXI7XG4gIH1cblxuICByZXR1cm4gZnVuY3Rpb24gKHZhbHVlKSB7XG4gICAgaWYgKEFycmF5LmlzQXJyYXkodmFsdWUpKSB7XG4gICAgICB2YXIgaW5kZXggPSBOdW1iZXIoaWRlbnRpZmllcik7XG5cbiAgICAgIGlmIChOdW1iZXIuaXNJbnRlZ2VyKGluZGV4KSkge1xuICAgICAgICByZXR1cm4gdmFsdWVbaW5kZXhdO1xuICAgICAgfVxuICAgIH0gZWxzZSBpZiAodmFsdWUgJiYgdHlwZW9mIHZhbHVlID09PSAnb2JqZWN0JyAmJiB0eXBlb2YgaWRlbnRpZmllciAhPT0gJ2Z1bmN0aW9uJykge1xuICAgICAgcmV0dXJuIHZhbHVlW2lkZW50aWZpZXJdO1xuICAgIH1cblxuICAgIHJldHVybiB2YWx1ZTtcbiAgfTtcbn07XG5cbnZhciBzdHJpbmdpZnkgPSBmdW5jdGlvbiBzdHJpbmdpZnkodmFsdWUpIHtcbiAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gJ2Jvb2xlYW4nIHx8IHZhbHVlIGluc3RhbmNlb2YgQm9vbGVhbikge1xuICAgIHJldHVybiBOdW1iZXIodmFsdWUpLnRvU3RyaW5nKCk7XG4gIH1cblxuICBpZiAodHlwZW9mIHZhbHVlID09PSAnbnVtYmVyJyB8fCB2YWx1ZSBpbnN0YW5jZW9mIE51bWJlcikge1xuICAgIHJldHVybiB2YWx1ZS50b1N0cmluZygpO1xuICB9XG5cbiAgaWYgKHZhbHVlIGluc3RhbmNlb2YgRGF0ZSkge1xuICAgIHJldHVybiB2YWx1ZS5nZXRUaW1lKCkudG9TdHJpbmcoKTtcbiAgfVxuXG4gIGlmICh0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnIHx8IHZhbHVlIGluc3RhbmNlb2YgU3RyaW5nKSB7XG4gICAgcmV0dXJuIHZhbHVlLnRvTG93ZXJDYXNlKCkucmVwbGFjZShSRV9MRUFESU5HX09SX1RSQUlMSU5HX1dISVRFU1BBQ0VTLCAnJyk7XG4gIH1cblxuICByZXR1cm4gJyc7XG59O1xuXG52YXIgcGFyc2VOdW1iZXIgPSBmdW5jdGlvbiBwYXJzZU51bWJlcih2YWx1ZSkge1xuICBpZiAodmFsdWUubGVuZ3RoICE9PSAwKSB7XG4gICAgdmFyIHBhcnNlZE51bWJlciA9IE51bWJlcih2YWx1ZSk7XG5cbiAgICBpZiAoIU51bWJlci5pc05hTihwYXJzZWROdW1iZXIpKSB7XG4gICAgICByZXR1cm4gcGFyc2VkTnVtYmVyO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiB1bmRlZmluZWQ7XG59O1xuXG52YXIgcGFyc2VEYXRlID0gZnVuY3Rpb24gcGFyc2VEYXRlKHZhbHVlKSB7XG4gIGlmIChSRV9EQVRFLnRlc3QodmFsdWUpKSB7XG4gICAgdmFyIHBhcnNlZERhdGUgPSBEYXRlLnBhcnNlKHZhbHVlKTtcblxuICAgIGlmICghTnVtYmVyLmlzTmFOKHBhcnNlZERhdGUpKSB7XG4gICAgICByZXR1cm4gcGFyc2VkRGF0ZTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gdW5kZWZpbmVkO1xufTtcblxudmFyIG51bWJlcmlmeSA9IGZ1bmN0aW9uIG51bWJlcmlmeSh2YWx1ZSkge1xuICB2YXIgcGFyc2VkTnVtYmVyID0gcGFyc2VOdW1iZXIodmFsdWUpO1xuXG4gIGlmIChwYXJzZWROdW1iZXIgIT09IHVuZGVmaW5lZCkge1xuICAgIHJldHVybiBwYXJzZWROdW1iZXI7XG4gIH1cblxuICByZXR1cm4gcGFyc2VEYXRlKHZhbHVlKTtcbn07XG5cbnZhciBjcmVhdGVDaHVua3MgPSBmdW5jdGlvbiBjcmVhdGVDaHVua3ModmFsdWUpIHtcbiAgcmV0dXJuIHZhbHVlLnJlcGxhY2UoUkVfTlVNQkVSUywgJ1xcMCQxXFwwJykucmVwbGFjZSgvXFwwJC8sICcnKS5yZXBsYWNlKC9eXFwwLywgJycpLnNwbGl0KCdcXDAnKTtcbn07XG5cbnZhciBub3JtYWxpemVBbHBoYUNodW5rID0gZnVuY3Rpb24gbm9ybWFsaXplQWxwaGFDaHVuayhjaHVuaykge1xuICByZXR1cm4gY2h1bmsucmVwbGFjZShSRV9XSElURVNQQUNFUywgJyAnKS5yZXBsYWNlKFJFX0xFQURJTkdfT1JfVFJBSUxJTkdfV0hJVEVTUEFDRVMsICcnKTtcbn07XG5cbnZhciBub3JtYWxpemVOdW1lcmljQ2h1bmsgPSBmdW5jdGlvbiBub3JtYWxpemVOdW1lcmljQ2h1bmsoY2h1bmssIGluZGV4LCBjaHVua3MpIHtcbiAgaWYgKFJFX0lOVF9PUl9GTE9BVC50ZXN0KGNodW5rKSkge1xuICAgIC8vIGRvbsK0dCBwYXJzZSBhIG51bWJlciwgaWYgdGhlcmXCtHMgYSBwcmVjZWRpbmcgZGVjaW1hbCBwb2ludFxuICAgIC8vIHRvIGtlZXAgc2lnbmlmaWNhbmNlXG4gICAgLy8gZS5nLiAxLjAwMjAsIDEuMDIwXG4gICAgaWYgKCFSRV9MRUFESU5HX1pFUk8udGVzdChjaHVuaykgfHwgaW5kZXggPT09IDAgfHwgY2h1bmtzW2luZGV4IC0gMV0gIT09ICcuJykge1xuICAgICAgcmV0dXJuIHBhcnNlTnVtYmVyKGNodW5rKSB8fCAwO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiB1bmRlZmluZWQ7XG59O1xuXG52YXIgY3JlYXRlQ2h1bmtNYXAgPSBmdW5jdGlvbiBjcmVhdGVDaHVua01hcChjaHVuaywgaW5kZXgsIGNodW5rcykge1xuICByZXR1cm4ge1xuICAgIHBhcnNlZE51bWJlcjogbm9ybWFsaXplTnVtZXJpY0NodW5rKGNodW5rLCBpbmRleCwgY2h1bmtzKSxcbiAgICBub3JtYWxpemVkU3RyaW5nOiBub3JtYWxpemVBbHBoYUNodW5rKGNodW5rKVxuICB9O1xufTtcblxudmFyIGNyZWF0ZUNodW5rTWFwcyA9IGZ1bmN0aW9uIGNyZWF0ZUNodW5rTWFwcyh2YWx1ZSkge1xuICB2YXIgY2h1bmtzTWFwcyA9IGNyZWF0ZUNodW5rcyh2YWx1ZSkubWFwKGNyZWF0ZUNodW5rTWFwKTtcbiAgcmV0dXJuIGNodW5rc01hcHM7XG59O1xuXG52YXIgaXNGdW5jdGlvbiA9IGZ1bmN0aW9uIGlzRnVuY3Rpb24odmFsdWUpIHtcbiAgcmV0dXJuIHR5cGVvZiB2YWx1ZSA9PT0gJ2Z1bmN0aW9uJztcbn07XG5cbnZhciBpc05hTiA9IGZ1bmN0aW9uIGlzTmFOKHZhbHVlKSB7XG4gIHJldHVybiBOdW1iZXIuaXNOYU4odmFsdWUpIHx8IHZhbHVlIGluc3RhbmNlb2YgTnVtYmVyICYmIE51bWJlci5pc05hTih2YWx1ZS52YWx1ZU9mKCkpO1xufTtcblxudmFyIGlzTnVsbCA9IGZ1bmN0aW9uIGlzTnVsbCh2YWx1ZSkge1xuICByZXR1cm4gdmFsdWUgPT09IG51bGw7XG59O1xuXG52YXIgaXNPYmplY3QgPSBmdW5jdGlvbiBpc09iamVjdCh2YWx1ZSkge1xuICByZXR1cm4gdmFsdWUgIT09IG51bGwgJiYgdHlwZW9mIHZhbHVlID09PSAnb2JqZWN0JyAmJiAhQXJyYXkuaXNBcnJheSh2YWx1ZSkgJiYgISh2YWx1ZSBpbnN0YW5jZW9mIE51bWJlcikgJiYgISh2YWx1ZSBpbnN0YW5jZW9mIFN0cmluZykgJiYgISh2YWx1ZSBpbnN0YW5jZW9mIEJvb2xlYW4pICYmICEodmFsdWUgaW5zdGFuY2VvZiBEYXRlKTtcbn07XG5cbnZhciBpc1N5bWJvbCA9IGZ1bmN0aW9uIGlzU3ltYm9sKHZhbHVlKSB7XG4gIHJldHVybiB0eXBlb2YgdmFsdWUgPT09ICdzeW1ib2wnO1xufTtcblxudmFyIGlzVW5kZWZpbmVkID0gZnVuY3Rpb24gaXNVbmRlZmluZWQodmFsdWUpIHtcbiAgcmV0dXJuIHZhbHVlID09PSB1bmRlZmluZWQ7XG59O1xuXG52YXIgZ2V0TWFwcGVkVmFsdWVSZWNvcmQgPSBmdW5jdGlvbiBnZXRNYXBwZWRWYWx1ZVJlY29yZCh2YWx1ZSkge1xuICBpZiAodHlwZW9mIHZhbHVlID09PSAnc3RyaW5nJyB8fCB2YWx1ZSBpbnN0YW5jZW9mIFN0cmluZyB8fCAodHlwZW9mIHZhbHVlID09PSAnbnVtYmVyJyB8fCB2YWx1ZSBpbnN0YW5jZW9mIE51bWJlcikgJiYgIWlzTmFOKHZhbHVlKSB8fCB0eXBlb2YgdmFsdWUgPT09ICdib29sZWFuJyB8fCB2YWx1ZSBpbnN0YW5jZW9mIEJvb2xlYW4gfHwgdmFsdWUgaW5zdGFuY2VvZiBEYXRlKSB7XG4gICAgdmFyIHN0cmluZ1ZhbHVlID0gc3RyaW5naWZ5KHZhbHVlKTtcbiAgICB2YXIgcGFyc2VkTnVtYmVyID0gbnVtYmVyaWZ5KHN0cmluZ1ZhbHVlKTtcbiAgICB2YXIgY2h1bmtzID0gY3JlYXRlQ2h1bmtNYXBzKHBhcnNlZE51bWJlciA/IFwiXCIgKyBwYXJzZWROdW1iZXIgOiBzdHJpbmdWYWx1ZSk7XG4gICAgcmV0dXJuIHtcbiAgICAgIHBhcnNlZE51bWJlcjogcGFyc2VkTnVtYmVyLFxuICAgICAgY2h1bmtzOiBjaHVua3MsXG4gICAgICB2YWx1ZTogdmFsdWVcbiAgICB9O1xuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBpc0FycmF5OiBBcnJheS5pc0FycmF5KHZhbHVlKSxcbiAgICBpc0Z1bmN0aW9uOiBpc0Z1bmN0aW9uKHZhbHVlKSxcbiAgICBpc05hTjogaXNOYU4odmFsdWUpLFxuICAgIGlzTnVsbDogaXNOdWxsKHZhbHVlKSxcbiAgICBpc09iamVjdDogaXNPYmplY3QodmFsdWUpLFxuICAgIGlzU3ltYm9sOiBpc1N5bWJvbCh2YWx1ZSksXG4gICAgaXNVbmRlZmluZWQ6IGlzVW5kZWZpbmVkKHZhbHVlKSxcbiAgICB2YWx1ZTogdmFsdWVcbiAgfTtcbn07XG5cbnZhciBnZXRWYWx1ZUJ5SWRlbnRpZmllciA9IGZ1bmN0aW9uIGdldFZhbHVlQnlJZGVudGlmaWVyKHZhbHVlLCBnZXRWYWx1ZSkge1xuICByZXR1cm4gZ2V0VmFsdWUodmFsdWUpO1xufTtcblxudmFyIGdldEVsZW1lbnRCeUluZGV4ID0gZnVuY3Rpb24gZ2V0RWxlbWVudEJ5SW5kZXgoY29sbGVjdGlvbiwgaW5kZXgpIHtcbiAgcmV0dXJuIGNvbGxlY3Rpb25baW5kZXhdO1xufTtcblxudmFyIGJhc2VPcmRlckJ5ID0gZnVuY3Rpb24gYmFzZU9yZGVyQnkoY29sbGVjdGlvbiwgaWRlbnRpZmllcnMsIG9yZGVycykge1xuICB2YXIgaWRlbnRpZmllckZucyA9IGlkZW50aWZpZXJzLmxlbmd0aCA/IGlkZW50aWZpZXJzLm1hcChjcmVhdGVJZGVudGlmaWVyRm4pIDogW2Z1bmN0aW9uICh2YWx1ZSkge1xuICAgIHJldHVybiB2YWx1ZTtcbiAgfV07IC8vIHRlbXBvcmFyeSBhcnJheSBob2xkcyBlbGVtZW50cyB3aXRoIHBvc2l0aW9uIGFuZCBzb3J0LXZhbHVlc1xuXG4gIHZhciBtYXBwZWRDb2xsZWN0aW9uID0gY29sbGVjdGlvbi5tYXAoZnVuY3Rpb24gKGVsZW1lbnQsIGluZGV4KSB7XG4gICAgdmFyIHZhbHVlcyA9IGlkZW50aWZpZXJGbnMubWFwKGZ1bmN0aW9uIChpZGVudGlmaWVyKSB7XG4gICAgICByZXR1cm4gZ2V0VmFsdWVCeUlkZW50aWZpZXIoZWxlbWVudCwgaWRlbnRpZmllcik7XG4gICAgfSkubWFwKGdldE1hcHBlZFZhbHVlUmVjb3JkKTtcbiAgICByZXR1cm4ge1xuICAgICAgaW5kZXg6IGluZGV4LFxuICAgICAgdmFsdWVzOiB2YWx1ZXNcbiAgICB9O1xuICB9KTsgLy8gaXRlcmF0ZSBvdmVyIHZhbHVlcyBhbmQgY29tcGFyZSB2YWx1ZXMgdW50aWwgYSAhPSBiIG9yIGxhc3QgdmFsdWUgcmVhY2hlZFxuXG4gIG1hcHBlZENvbGxlY3Rpb24uc29ydChmdW5jdGlvbiAocmVjb3JkQSwgcmVjb3JkQikge1xuICAgIHJldHVybiBjb21wYXJlTXVsdGlwbGUocmVjb3JkQSwgcmVjb3JkQiwgb3JkZXJzKTtcbiAgfSk7XG4gIHJldHVybiBtYXBwZWRDb2xsZWN0aW9uLm1hcChmdW5jdGlvbiAoZWxlbWVudCkge1xuICAgIHJldHVybiBnZXRFbGVtZW50QnlJbmRleChjb2xsZWN0aW9uLCBlbGVtZW50LmluZGV4KTtcbiAgfSk7XG59O1xuXG52YXIgZ2V0SWRlbnRpZmllcnMgPSBmdW5jdGlvbiBnZXRJZGVudGlmaWVycyhpZGVudGlmaWVycykge1xuICBpZiAoIWlkZW50aWZpZXJzKSB7XG4gICAgcmV0dXJuIFtdO1xuICB9XG5cbiAgdmFyIGlkZW50aWZpZXJMaXN0ID0gIUFycmF5LmlzQXJyYXkoaWRlbnRpZmllcnMpID8gW2lkZW50aWZpZXJzXSA6IFtdLmNvbmNhdChpZGVudGlmaWVycyk7XG5cbiAgaWYgKGlkZW50aWZpZXJMaXN0LnNvbWUoZnVuY3Rpb24gKGlkZW50aWZpZXIpIHtcbiAgICByZXR1cm4gdHlwZW9mIGlkZW50aWZpZXIgIT09ICdzdHJpbmcnICYmIHR5cGVvZiBpZGVudGlmaWVyICE9PSAnbnVtYmVyJyAmJiB0eXBlb2YgaWRlbnRpZmllciAhPT0gJ2Z1bmN0aW9uJztcbiAgfSkpIHtcbiAgICByZXR1cm4gW107XG4gIH1cblxuICByZXR1cm4gaWRlbnRpZmllckxpc3Q7XG59O1xuXG52YXIgZ2V0T3JkZXJzID0gZnVuY3Rpb24gZ2V0T3JkZXJzKG9yZGVycykge1xuICBpZiAoIW9yZGVycykge1xuICAgIHJldHVybiBbXTtcbiAgfVxuXG4gIHZhciBvcmRlckxpc3QgPSAhQXJyYXkuaXNBcnJheShvcmRlcnMpID8gW29yZGVyc10gOiBbXS5jb25jYXQob3JkZXJzKTtcblxuICBpZiAob3JkZXJMaXN0LnNvbWUoZnVuY3Rpb24gKG9yZGVyKSB7XG4gICAgcmV0dXJuIG9yZGVyICE9PSAnYXNjJyAmJiBvcmRlciAhPT0gJ2Rlc2MnICYmIHR5cGVvZiBvcmRlciAhPT0gJ2Z1bmN0aW9uJztcbiAgfSkpIHtcbiAgICByZXR1cm4gW107XG4gIH1cblxuICByZXR1cm4gb3JkZXJMaXN0O1xufTtcblxuLyoqXG4gKiBDcmVhdGVzIGFuIGFycmF5IG9mIGVsZW1lbnRzLCBuYXR1cmFsIHNvcnRlZCBieSBzcGVjaWZpZWQgaWRlbnRpZmllcnMgYW5kXG4gKiB0aGUgY29ycmVzcG9uZGluZyBzb3J0IG9yZGVycy4gVGhpcyBtZXRob2QgaW1wbGVtZW50cyBhIHN0YWJsZSBzb3J0XG4gKiBhbGdvcml0aG0sIHdoaWNoIG1lYW5zIHRoZSBvcmlnaW5hbCBzb3J0IG9yZGVyIG9mIGVxdWFsIGVsZW1lbnRzIGlzXG4gKiBwcmVzZXJ2ZWQuXG4gKlxuICogSWYgYGNvbGxlY3Rpb25gIGlzIGFuIGFycmF5IG9mIHByaW1pdGl2ZXMsIGBpZGVudGlmaWVyc2AgbWF5IGJlIHVuc3BlY2lmaWVkLlxuICogT3RoZXJ3aXNlLCB5b3Ugc2hvdWxkIHNwZWNpZnkgYGlkZW50aWZpZXJzYCB0byBzb3J0IGJ5IG9yIGBjb2xsZWN0aW9uYCB3aWxsXG4gKiBiZSByZXR1cm5lZCB1bnNvcnRlZC4gQW4gaWRlbnRpZmllciBjYW4gZXhwcmVzc2VkIGJ5OlxuICpcbiAqIC0gYW4gaW5kZXggcG9zaXRpb24sIGlmIGBjb2xsZWN0aW9uYCBpcyBhIG5lc3RlZCBhcnJheSxcbiAqIC0gYSBwcm9wZXJ0eSBuYW1lLCBpZiBgY29sbGVjdGlvbmAgaXMgYW4gYXJyYXkgb2Ygb2JqZWN0cyxcbiAqIC0gYSBmdW5jdGlvbiB3aGljaCByZXR1cm5zIGEgcGFydGljdWxhciB2YWx1ZSBmcm9tIGFuIGVsZW1lbnQgb2YgYSBuZXN0ZWQgYXJyYXkgb3IgYW4gYXJyYXkgb2Ygb2JqZWN0cy4gVGhpcyBmdW5jdGlvbiB3aWxsIGJlIGludm9rZWQgYnkgcGFzc2luZyBvbmUgZWxlbWVudCBvZiBgY29sbGVjdGlvbmAuXG4gKlxuICogSWYgYG9yZGVyc2AgaXMgdW5zcGVjaWZpZWQsIGFsbCB2YWx1ZXMgYXJlIHNvcnRlZCBpbiBhc2NlbmRpbmcgb3JkZXIuXG4gKiBPdGhlcndpc2UsIHNwZWNpZnkgYW4gb3JkZXIgb2YgYCdkZXNjJ2AgZm9yIGRlc2NlbmRpbmcgb3IgYCdhc2MnYCBmb3JcbiAqIGFzY2VuZGluZyBzb3J0IG9yZGVyIG9mIGNvcnJlc3BvbmRpbmcgdmFsdWVzLiBZb3UgbWF5IGFsc28gc3BlY2lmeSBhIGNvbXBhcmVcbiAqIGZ1bmN0aW9uIGZvciBhbiBvcmRlciwgd2hpY2ggd2lsbCBiZSBpbnZva2VkIGJ5IHR3byBhcmd1bWVudHM6XG4gKiBgKHZhbHVlQSwgdmFsdWVCKWAuIEl0IG11c3QgcmV0dXJuIGEgbnVtYmVyIHJlcHJlc2VudGluZyB0aGUgc29ydCBvcmRlci5cbiAqXG4gKiBAZXhhbXBsZVxuICpcbiAqIGltcG9ydCB7IG9yZGVyQnkgfSBmcm9tICduYXR1cmFsLW9yZGVyYnknO1xuICpcbiAqIGNvbnN0IHVzZXJzID0gW1xuICogICB7XG4gKiAgICAgdXNlcm5hbWU6ICdCYW1tLUJhbW0nLFxuICogICAgIGlwOiAnMTkyLjE2OC41LjInLFxuICogICAgIGRhdGV0aW1lOiAnRnJpIEp1biAxNSAyMDE4IDE2OjQ4OjAwIEdNVCswMjAwIChDRVNUKSdcbiAqICAgfSxcbiAqICAge1xuICogICAgIHVzZXJuYW1lOiAnV2lsbWEnLFxuICogICAgIGlwOiAnMTkyLjE2OC4xMC4xJyxcbiAqICAgICBkYXRldGltZTogJzE0IEp1biAyMDE4IDAwOjAwOjAwIFBEVCdcbiAqICAgfSxcbiAqICAge1xuICogICAgIHVzZXJuYW1lOiAnZGlubycsXG4gKiAgICAgaXA6ICcxOTIuMTY4LjAuMicsXG4gKiAgICAgZGF0ZXRpbWU6ICdKdW5lIDE1LCAyMDE4IDE0OjQ4OjAwJ1xuICogICB9LFxuICogICB7XG4gKiAgICAgdXNlcm5hbWU6ICdCYXJuZXknLFxuICogICAgIGlwOiAnMTkyLjE2OC4xLjEnLFxuICogICAgIGRhdGV0aW1lOiAnVGh1LCAxNCBKdW4gMjAxOCAwNzowMDowMCBHTVQnXG4gKiAgIH0sXG4gKiAgIHtcbiAqICAgICB1c2VybmFtZTogJ1BlYmJsZXMnLFxuICogICAgIGlwOiAnMTkyLjE2OC4xLjIxJyxcbiAqICAgICBkYXRldGltZTogJzE1IEp1bmUgMjAxOCAxNDo0OCBVVEMnXG4gKiAgIH0sXG4gKiAgIHtcbiAqICAgICB1c2VybmFtZTogJ0hvcHB5JyxcbiAqICAgICBpcDogJzE5Mi4xNjguNS4xMCcsXG4gKiAgICAgZGF0ZXRpbWU6ICcyMDE4LTA2LTE1VDE0OjQ4OjAwLjAwMFonXG4gKiAgIH0sXG4gKiBdO1xuICpcbiAqIG9yZGVyQnkoXG4gKiAgIHVzZXJzLFxuICogICBbdiA9PiB2LmRhdGV0aW1lLCB2ID0+IHYuaXBdLFxuICogICBbJ2Rlc2MnLCAnYXNjJ11cbiAqICk7XG4gKlxuICogLy8gPT4gW1xuICogLy8gICAgICB7XG4gKiAvLyAgICAgICAgdXNlcm5hbWU6ICdkaW5vJyxcbiAqIC8vICAgICAgICBpcDogJzE5Mi4xNjguMC4yJyxcbiAqIC8vICAgICAgICBkYXRldGltZTogJ0p1bmUgMTUsIDIwMTggMTQ6NDg6MDAnLFxuICogLy8gICAgICB9LFxuICogLy8gICAgICB7XG4gKiAvLyAgICAgICAgdXNlcm5hbWU6ICdQZWJibGVzJyxcbiAqIC8vICAgICAgICBpcDogJzE5Mi4xNjguMS4yMScsXG4gKiAvLyAgICAgICAgZGF0ZXRpbWU6ICcxNSBKdW5lIDIwMTggMTQ6NDggVVRDJyxcbiAqIC8vICAgICAgfSxcbiAqIC8vICAgICAge1xuICogLy8gICAgICAgIHVzZXJuYW1lOiAnQmFtbS1CYW1tJyxcbiAqIC8vICAgICAgICBpcDogJzE5Mi4xNjguNS4yJyxcbiAqIC8vICAgICAgICBkYXRldGltZTogJ0ZyaSBKdW4gMTUgMjAxOCAxNjo0ODowMCBHTVQrMDIwMCAoQ0VTVCknLFxuICogLy8gICAgICB9LFxuICogLy8gICAgICB7XG4gKiAvLyAgICAgICAgdXNlcm5hbWU6ICdIb3BweScsXG4gKiAvLyAgICAgICAgaXA6ICcxOTIuMTY4LjUuMTAnLFxuICogLy8gICAgICAgIGRhdGV0aW1lOiAnMjAxOC0wNi0xNVQxNDo0ODowMC4wMDBaJyxcbiAqIC8vICAgICAgfSxcbiAqIC8vICAgICAge1xuICogLy8gICAgICAgIHVzZXJuYW1lOiAnQmFybmV5JyxcbiAqIC8vICAgICAgICBpcDogJzE5Mi4xNjguMS4xJyxcbiAqIC8vICAgICAgICBkYXRldGltZTogJ1RodSwgMTQgSnVuIDIwMTggMDc6MDA6MDAgR01UJyxcbiAqIC8vICAgICAgfSxcbiAqIC8vICAgICAge1xuICogLy8gICAgICAgIHVzZXJuYW1lOiAnV2lsbWEnLFxuICogLy8gICAgICAgIGlwOiAnMTkyLjE2OC4xMC4xJyxcbiAqIC8vICAgICAgICBkYXRldGltZTogJzE0IEp1biAyMDE4IDAwOjAwOjAwIFBEVCcsXG4gKiAvLyAgICAgIH0sXG4gKiAvLyAgICBdXG4gKi9cbmZ1bmN0aW9uIG9yZGVyQnkoY29sbGVjdGlvbiwgaWRlbnRpZmllcnMsIG9yZGVycykge1xuICBpZiAoIWNvbGxlY3Rpb24gfHwgIUFycmF5LmlzQXJyYXkoY29sbGVjdGlvbikpIHtcbiAgICByZXR1cm4gW107XG4gIH1cblxuICB2YXIgdmFsaWRhdGVkSWRlbnRpZmllcnMgPSBnZXRJZGVudGlmaWVycyhpZGVudGlmaWVycyk7XG4gIHZhciB2YWxpZGF0ZWRPcmRlcnMgPSBnZXRPcmRlcnMob3JkZXJzKTtcbiAgcmV0dXJuIGJhc2VPcmRlckJ5KGNvbGxlY3Rpb24sIHZhbGlkYXRlZElkZW50aWZpZXJzLCB2YWxpZGF0ZWRPcmRlcnMpO1xufVxuXG52YXIgYmFzZUNvbXBhcmUgPSBmdW5jdGlvbiBiYXNlQ29tcGFyZShvcHRpb25zKSB7XG4gIHJldHVybiBmdW5jdGlvbiAodmFsdWVBLCB2YWx1ZUIpIHtcbiAgICB2YXIgYSA9IGdldE1hcHBlZFZhbHVlUmVjb3JkKHZhbHVlQSk7XG4gICAgdmFyIGIgPSBnZXRNYXBwZWRWYWx1ZVJlY29yZCh2YWx1ZUIpO1xuICAgIHZhciByZXN1bHQgPSBjb21wYXJlVmFsdWVzKGEsIGIpO1xuICAgIHJldHVybiByZXN1bHQgKiAob3B0aW9ucy5vcmRlciA9PT0gJ2Rlc2MnID8gLTEgOiAxKTtcbiAgfTtcbn07XG5cbnZhciBpc1ZhbGlkT3JkZXIgPSBmdW5jdGlvbiBpc1ZhbGlkT3JkZXIodmFsdWUpIHtcbiAgcmV0dXJuIHR5cGVvZiB2YWx1ZSA9PT0gJ3N0cmluZycgJiYgKHZhbHVlID09PSAnYXNjJyB8fCB2YWx1ZSA9PT0gJ2Rlc2MnKTtcbn07XG5cbnZhciBnZXRPcHRpb25zID0gZnVuY3Rpb24gZ2V0T3B0aW9ucyhjdXN0b21PcHRpb25zKSB7XG4gIHZhciBvcmRlciA9ICdhc2MnO1xuXG4gIGlmICh0eXBlb2YgY3VzdG9tT3B0aW9ucyA9PT0gJ3N0cmluZycgJiYgaXNWYWxpZE9yZGVyKGN1c3RvbU9wdGlvbnMpKSB7XG4gICAgb3JkZXIgPSBjdXN0b21PcHRpb25zO1xuICB9IGVsc2UgaWYgKGN1c3RvbU9wdGlvbnMgJiYgdHlwZW9mIGN1c3RvbU9wdGlvbnMgPT09ICdvYmplY3QnICYmIGN1c3RvbU9wdGlvbnMub3JkZXIgJiYgaXNWYWxpZE9yZGVyKGN1c3RvbU9wdGlvbnMub3JkZXIpKSB7XG4gICAgb3JkZXIgPSBjdXN0b21PcHRpb25zLm9yZGVyO1xuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBvcmRlcjogb3JkZXJcbiAgfTtcbn07XG5cbi8qKlxuICogQ3JlYXRlcyBhIGNvbXBhcmUgZnVuY3Rpb24gdGhhdCBkZWZpbmVzIHRoZSBuYXR1cmFsIHNvcnQgb3JkZXIgY29uc2lkZXJpbmdcbiAqIHRoZSBnaXZlbiBgb3B0aW9uc2Agd2hpY2ggbWF5IGJlIHBhc3NlZCB0byBbYEFycmF5LnByb3RvdHlwZS5zb3J0KClgXShodHRwczovL2RldmVsb3Blci5tb3ppbGxhLm9yZy9lbi1VUy9kb2NzL1dlYi9KYXZhU2NyaXB0L1JlZmVyZW5jZS9HbG9iYWxfT2JqZWN0cy9BcnJheS9zb3J0KS5cbiAqXG4gKiBJZiBgb3B0aW9uc2Agb3IgaXRzIHByb3BlcnR5IGBvcmRlcmAgaXMgdW5zcGVjaWZpZWQsIHZhbHVlcyBhcmUgc29ydGVkIGluXG4gKiBhc2NlbmRpbmcgc29ydCBvcmRlci4gT3RoZXJ3aXNlLCBzcGVjaWZ5IGFuIG9yZGVyIG9mIGAnZGVzYydgIGZvciBkZXNjZW5kaW5nXG4gKiBvciBgJ2FzYydgIGZvciBhc2NlbmRpbmcgc29ydCBvcmRlciBvZiB2YWx1ZXMuXG4gKlxuICogQGV4YW1wbGVcbiAqXG4gKiBpbXBvcnQgeyBjb21wYXJlIH0gZnJvbSAnbmF0dXJhbC1vcmRlcmJ5JztcbiAqXG4gKiBjb25zdCB1c2VycyA9IFtcbiAqICAge1xuICogICAgIHVzZXJuYW1lOiAnQmFtbS1CYW1tJyxcbiAqICAgICBsYXN0TG9naW46IHtcbiAqICAgICAgIGlwOiAnMTkyLjE2OC41LjInLFxuICogICAgICAgZGF0ZXRpbWU6ICdGcmkgSnVuIDE1IDIwMTggMTY6NDg6MDAgR01UKzAyMDAgKENFU1QpJ1xuICogICAgIH0sXG4gKiAgIH0sXG4gKiAgIHtcbiAqICAgICB1c2VybmFtZTogJ1dpbG1hJyxcbiAqICAgICBsYXN0TG9naW46IHtcbiAqICAgICAgIGlwOiAnMTkyLjE2OC4xMC4xJyxcbiAqICAgICAgIGRhdGV0aW1lOiAnMTQgSnVuIDIwMTggMDA6MDA6MDAgUERUJ1xuICogICAgIH0sXG4gKiAgIH0sXG4gKiAgIHtcbiAqICAgICB1c2VybmFtZTogJ2Rpbm8nLFxuICogICAgIGxhc3RMb2dpbjoge1xuICogICAgICAgaXA6ICcxOTIuMTY4LjAuMicsXG4gKiAgICAgICBkYXRldGltZTogJ0p1bmUgMTUsIDIwMTggMTQ6NDg6MDAnXG4gKiAgICAgfSxcbiAqICAgfSxcbiAqICAge1xuICogICAgIHVzZXJuYW1lOiAnQmFybmV5JyxcbiAqICAgICBsYXN0TG9naW46IHtcbiAqICAgICAgIGlwOiAnMTkyLjE2OC4xLjEnLFxuICogICAgICAgZGF0ZXRpbWU6ICdUaHUsIDE0IEp1biAyMDE4IDA3OjAwOjAwIEdNVCdcbiAqICAgICB9LFxuICogICB9LFxuICogICB7XG4gKiAgICAgdXNlcm5hbWU6ICdQZWJibGVzJyxcbiAqICAgICBsYXN0TG9naW46IHtcbiAqICAgICAgIGlwOiAnMTkyLjE2OC4xLjIxJyxcbiAqICAgICAgIGRhdGV0aW1lOiAnMTUgSnVuZSAyMDE4IDE0OjQ4IFVUQydcbiAqICAgICB9LFxuICogICB9LFxuICogICB7XG4gKiAgICAgdXNlcm5hbWU6ICdIb3BweScsXG4gKiAgICAgbGFzdExvZ2luOiB7XG4gKiAgICAgICBpcDogJzE5Mi4xNjguNS4xMCcsXG4gKiAgICAgICBkYXRldGltZTogJzIwMTgtMDYtMTVUMTQ6NDg6MDAuMDAwWidcbiAqICAgICB9LFxuICogICB9LFxuICogXTtcbiAqXG4gKiB1c2Vycy5zb3J0KChhLCBiKSA9PiBjb21wYXJlKCkoYS5pcCwgYi5pcCkpO1xuICpcbiAqIC8vID0+IFtcbiAqIC8vICAgICAge1xuICogLy8gICAgICAgIHVzZXJuYW1lOiAnZGlubycsXG4gKiAvLyAgICAgICAgaXA6ICcxOTIuMTY4LjAuMicsXG4gKiAvLyAgICAgICAgZGF0ZXRpbWU6ICdKdW5lIDE1LCAyMDE4IDE0OjQ4OjAwJ1xuICogLy8gICAgICB9LFxuICogLy8gICAgICB7XG4gKiAvLyAgICAgICAgdXNlcm5hbWU6ICdCYXJuZXknLFxuICogLy8gICAgICAgIGlwOiAnMTkyLjE2OC4xLjEnLFxuICogLy8gICAgICAgIGRhdGV0aW1lOiAnVGh1LCAxNCBKdW4gMjAxOCAwNzowMDowMCBHTVQnXG4gKiAvLyAgICAgIH0sXG4gKiAvLyAgICAgIHtcbiAqIC8vICAgICAgICB1c2VybmFtZTogJ1BlYmJsZXMnLFxuICogLy8gICAgICAgIGlwOiAnMTkyLjE2OC4xLjIxJyxcbiAqIC8vICAgICAgICBkYXRldGltZTogJzE1IEp1bmUgMjAxOCAxNDo0OCBVVEMnXG4gKiAvLyAgICAgIH0sXG4gKiAvLyAgICAgIHtcbiAqIC8vICAgICAgICB1c2VybmFtZTogJ0JhbW0tQmFtbScsXG4gKiAvLyAgICAgICAgaXA6ICcxOTIuMTY4LjUuMicsXG4gKiAvLyAgICAgICAgZGF0ZXRpbWU6ICdGcmkgSnVuIDE1IDIwMTggMTY6NDg6MDAgR01UKzAyMDAgKENFU1QpJ1xuICogLy8gICAgICB9LFxuICogLy8gICAgICB7XG4gKiAvLyAgICAgICAgdXNlcm5hbWU6ICdIb3BweScsXG4gKiAvLyAgICAgICAgaXA6ICcxOTIuMTY4LjUuMTAnLFxuICogLy8gICAgICAgIGRhdGV0aW1lOiAnMjAxOC0wNi0xNVQxNDo0ODowMC4wMDBaJ1xuICogLy8gICAgICB9LFxuICogLy8gICAgICB7XG4gKiAvLyAgICAgICAgdXNlcm5hbWU6ICdXaWxtYScsXG4gKiAvLyAgICAgICAgaXA6ICcxOTIuMTY4LjEwLjEnLFxuICogLy8gICAgICAgIGRhdGV0aW1lOiAnMTQgSnVuIDIwMTggMDA6MDA6MDAgUERUJ1xuICogLy8gICAgICB9XG4gKiAvLyAgICBdXG4gKi9cbmZ1bmN0aW9uIGNvbXBhcmUob3B0aW9ucykge1xuICB2YXIgdmFsaWRhdGVkT3B0aW9ucyA9IGdldE9wdGlvbnMob3B0aW9ucyk7XG4gIHJldHVybiBiYXNlQ29tcGFyZSh2YWxpZGF0ZWRPcHRpb25zKTtcbn1cblxuLypcbiogSmF2YXNjcmlwdCBuYXR1cmFsIHNvcnQgYWxnb3JpdGhtIHdpdGggdW5pY29kZSBzdXBwb3J0XG4qIGJhc2VkIG9uIGNodW5raW5nIGlkZWEgYnkgRGF2ZSBLb2VsbGVcbipcbiogaHR0cHM6Ly9naXRodWIuY29tL3lvYmFjY2EvbmF0dXJhbC1zb3J0LW9yZGVyXG4qIHJlbGVhc2VkIHVuZGVyIE1JVCBMaWNlbnNlXG4qL1xuXG5leHBvcnQgeyBvcmRlckJ5LCBjb21wYXJlIH07XG4iLCIvKiBAbGljZW5zZVxuUGFwYSBQYXJzZVxudjUuMy4xXG5odHRwczovL2dpdGh1Yi5jb20vbWhvbHQvUGFwYVBhcnNlXG5MaWNlbnNlOiBNSVRcbiovXG4hZnVuY3Rpb24oZSx0KXtcImZ1bmN0aW9uXCI9PXR5cGVvZiBkZWZpbmUmJmRlZmluZS5hbWQ/ZGVmaW5lKFtdLHQpOlwib2JqZWN0XCI9PXR5cGVvZiBtb2R1bGUmJlwidW5kZWZpbmVkXCIhPXR5cGVvZiBleHBvcnRzP21vZHVsZS5leHBvcnRzPXQoKTplLlBhcGE9dCgpfSh0aGlzLGZ1bmN0aW9uIHMoKXtcInVzZSBzdHJpY3RcIjt2YXIgZj1cInVuZGVmaW5lZFwiIT10eXBlb2Ygc2VsZj9zZWxmOlwidW5kZWZpbmVkXCIhPXR5cGVvZiB3aW5kb3c/d2luZG93OnZvaWQgMCE9PWY/Zjp7fTt2YXIgbj0hZi5kb2N1bWVudCYmISFmLnBvc3RNZXNzYWdlLG89biYmL2Jsb2I6L2kudGVzdCgoZi5sb2NhdGlvbnx8e30pLnByb3RvY29sKSxhPXt9LGg9MCxiPXtwYXJzZTpmdW5jdGlvbihlLHQpe3ZhciBpPSh0PXR8fHt9KS5keW5hbWljVHlwaW5nfHwhMTtNKGkpJiYodC5keW5hbWljVHlwaW5nRnVuY3Rpb249aSxpPXt9KTtpZih0LmR5bmFtaWNUeXBpbmc9aSx0LnRyYW5zZm9ybT0hIU0odC50cmFuc2Zvcm0pJiZ0LnRyYW5zZm9ybSx0LndvcmtlciYmYi5XT1JLRVJTX1NVUFBPUlRFRCl7dmFyIHI9ZnVuY3Rpb24oKXtpZighYi5XT1JLRVJTX1NVUFBPUlRFRClyZXR1cm4hMTt2YXIgZT0oaT1mLlVSTHx8Zi53ZWJraXRVUkx8fG51bGwscj1zLnRvU3RyaW5nKCksYi5CTE9CX1VSTHx8KGIuQkxPQl9VUkw9aS5jcmVhdGVPYmplY3RVUkwobmV3IEJsb2IoW1wiKFwiLHIsXCIpKCk7XCJdLHt0eXBlOlwidGV4dC9qYXZhc2NyaXB0XCJ9KSkpKSx0PW5ldyBmLldvcmtlcihlKTt2YXIgaSxyO3JldHVybiB0Lm9ubWVzc2FnZT1fLHQuaWQ9aCsrLGFbdC5pZF09dH0oKTtyZXR1cm4gci51c2VyU3RlcD10LnN0ZXAsci51c2VyQ2h1bms9dC5jaHVuayxyLnVzZXJDb21wbGV0ZT10LmNvbXBsZXRlLHIudXNlckVycm9yPXQuZXJyb3IsdC5zdGVwPU0odC5zdGVwKSx0LmNodW5rPU0odC5jaHVuayksdC5jb21wbGV0ZT1NKHQuY29tcGxldGUpLHQuZXJyb3I9TSh0LmVycm9yKSxkZWxldGUgdC53b3JrZXIsdm9pZCByLnBvc3RNZXNzYWdlKHtpbnB1dDplLGNvbmZpZzp0LHdvcmtlcklkOnIuaWR9KX12YXIgbj1udWxsO2IuTk9ERV9TVFJFQU1fSU5QVVQsXCJzdHJpbmdcIj09dHlwZW9mIGU/bj10LmRvd25sb2FkP25ldyBsKHQpOm5ldyBwKHQpOiEwPT09ZS5yZWFkYWJsZSYmTShlLnJlYWQpJiZNKGUub24pP249bmV3IGcodCk6KGYuRmlsZSYmZSBpbnN0YW5jZW9mIEZpbGV8fGUgaW5zdGFuY2VvZiBPYmplY3QpJiYobj1uZXcgYyh0KSk7cmV0dXJuIG4uc3RyZWFtKGUpfSx1bnBhcnNlOmZ1bmN0aW9uKGUsdCl7dmFyIG49ITEsXz0hMCxtPVwiLFwiLHk9XCJcXHJcXG5cIixzPSdcIicsYT1zK3MsaT0hMSxyPW51bGwsbz0hMTshZnVuY3Rpb24oKXtpZihcIm9iamVjdFwiIT10eXBlb2YgdClyZXR1cm47XCJzdHJpbmdcIiE9dHlwZW9mIHQuZGVsaW1pdGVyfHxiLkJBRF9ERUxJTUlURVJTLmZpbHRlcihmdW5jdGlvbihlKXtyZXR1cm4tMSE9PXQuZGVsaW1pdGVyLmluZGV4T2YoZSl9KS5sZW5ndGh8fChtPXQuZGVsaW1pdGVyKTsoXCJib29sZWFuXCI9PXR5cGVvZiB0LnF1b3Rlc3x8XCJmdW5jdGlvblwiPT10eXBlb2YgdC5xdW90ZXN8fEFycmF5LmlzQXJyYXkodC5xdW90ZXMpKSYmKG49dC5xdW90ZXMpO1wiYm9vbGVhblwiIT10eXBlb2YgdC5za2lwRW1wdHlMaW5lcyYmXCJzdHJpbmdcIiE9dHlwZW9mIHQuc2tpcEVtcHR5TGluZXN8fChpPXQuc2tpcEVtcHR5TGluZXMpO1wic3RyaW5nXCI9PXR5cGVvZiB0Lm5ld2xpbmUmJih5PXQubmV3bGluZSk7XCJzdHJpbmdcIj09dHlwZW9mIHQucXVvdGVDaGFyJiYocz10LnF1b3RlQ2hhcik7XCJib29sZWFuXCI9PXR5cGVvZiB0LmhlYWRlciYmKF89dC5oZWFkZXIpO2lmKEFycmF5LmlzQXJyYXkodC5jb2x1bW5zKSl7aWYoMD09PXQuY29sdW1ucy5sZW5ndGgpdGhyb3cgbmV3IEVycm9yKFwiT3B0aW9uIGNvbHVtbnMgaXMgZW1wdHlcIik7cj10LmNvbHVtbnN9dm9pZCAwIT09dC5lc2NhcGVDaGFyJiYoYT10LmVzY2FwZUNoYXIrcyk7XCJib29sZWFuXCI9PXR5cGVvZiB0LmVzY2FwZUZvcm11bGFlJiYobz10LmVzY2FwZUZvcm11bGFlKX0oKTt2YXIgaD1uZXcgUmVnRXhwKGoocyksXCJnXCIpO1wic3RyaW5nXCI9PXR5cGVvZiBlJiYoZT1KU09OLnBhcnNlKGUpKTtpZihBcnJheS5pc0FycmF5KGUpKXtpZighZS5sZW5ndGh8fEFycmF5LmlzQXJyYXkoZVswXSkpcmV0dXJuIHUobnVsbCxlLGkpO2lmKFwib2JqZWN0XCI9PXR5cGVvZiBlWzBdKXJldHVybiB1KHJ8fE9iamVjdC5rZXlzKGVbMF0pLGUsaSl9ZWxzZSBpZihcIm9iamVjdFwiPT10eXBlb2YgZSlyZXR1cm5cInN0cmluZ1wiPT10eXBlb2YgZS5kYXRhJiYoZS5kYXRhPUpTT04ucGFyc2UoZS5kYXRhKSksQXJyYXkuaXNBcnJheShlLmRhdGEpJiYoZS5maWVsZHN8fChlLmZpZWxkcz1lLm1ldGEmJmUubWV0YS5maWVsZHMpLGUuZmllbGRzfHwoZS5maWVsZHM9QXJyYXkuaXNBcnJheShlLmRhdGFbMF0pP2UuZmllbGRzOlwib2JqZWN0XCI9PXR5cGVvZiBlLmRhdGFbMF0/T2JqZWN0LmtleXMoZS5kYXRhWzBdKTpbXSksQXJyYXkuaXNBcnJheShlLmRhdGFbMF0pfHxcIm9iamVjdFwiPT10eXBlb2YgZS5kYXRhWzBdfHwoZS5kYXRhPVtlLmRhdGFdKSksdShlLmZpZWxkc3x8W10sZS5kYXRhfHxbXSxpKTt0aHJvdyBuZXcgRXJyb3IoXCJVbmFibGUgdG8gc2VyaWFsaXplIHVucmVjb2duaXplZCBpbnB1dFwiKTtmdW5jdGlvbiB1KGUsdCxpKXt2YXIgcj1cIlwiO1wic3RyaW5nXCI9PXR5cGVvZiBlJiYoZT1KU09OLnBhcnNlKGUpKSxcInN0cmluZ1wiPT10eXBlb2YgdCYmKHQ9SlNPTi5wYXJzZSh0KSk7dmFyIG49QXJyYXkuaXNBcnJheShlKSYmMDxlLmxlbmd0aCxzPSFBcnJheS5pc0FycmF5KHRbMF0pO2lmKG4mJl8pe2Zvcih2YXIgYT0wO2E8ZS5sZW5ndGg7YSsrKTA8YSYmKHIrPW0pLHIrPXYoZVthXSxhKTswPHQubGVuZ3RoJiYocis9eSl9Zm9yKHZhciBvPTA7bzx0Lmxlbmd0aDtvKyspe3ZhciBoPW4/ZS5sZW5ndGg6dFtvXS5sZW5ndGgsdT0hMSxmPW4/MD09PU9iamVjdC5rZXlzKHRbb10pLmxlbmd0aDowPT09dFtvXS5sZW5ndGg7aWYoaSYmIW4mJih1PVwiZ3JlZWR5XCI9PT1pP1wiXCI9PT10W29dLmpvaW4oXCJcIikudHJpbSgpOjE9PT10W29dLmxlbmd0aCYmMD09PXRbb11bMF0ubGVuZ3RoKSxcImdyZWVkeVwiPT09aSYmbil7Zm9yKHZhciBkPVtdLGw9MDtsPGg7bCsrKXt2YXIgYz1zP2VbbF06bDtkLnB1c2godFtvXVtjXSl9dT1cIlwiPT09ZC5qb2luKFwiXCIpLnRyaW0oKX1pZighdSl7Zm9yKHZhciBwPTA7cDxoO3ArKyl7MDxwJiYhZiYmKHIrPW0pO3ZhciBnPW4mJnM/ZVtwXTpwO3IrPXYodFtvXVtnXSxwKX1vPHQubGVuZ3RoLTEmJighaXx8MDxoJiYhZikmJihyKz15KX19cmV0dXJuIHJ9ZnVuY3Rpb24gdihlLHQpe2lmKG51bGw9PWUpcmV0dXJuXCJcIjtpZihlLmNvbnN0cnVjdG9yPT09RGF0ZSlyZXR1cm4gSlNPTi5zdHJpbmdpZnkoZSkuc2xpY2UoMSwyNSk7ITA9PT1vJiZcInN0cmluZ1wiPT10eXBlb2YgZSYmbnVsbCE9PWUubWF0Y2goL15bPStcXC1AXS4qJC8pJiYoZT1cIidcIitlKTt2YXIgaT1lLnRvU3RyaW5nKCkucmVwbGFjZShoLGEpLHI9XCJib29sZWFuXCI9PXR5cGVvZiBuJiZufHxcImZ1bmN0aW9uXCI9PXR5cGVvZiBuJiZuKGUsdCl8fEFycmF5LmlzQXJyYXkobikmJm5bdF18fGZ1bmN0aW9uKGUsdCl7Zm9yKHZhciBpPTA7aTx0Lmxlbmd0aDtpKyspaWYoLTE8ZS5pbmRleE9mKHRbaV0pKXJldHVybiEwO3JldHVybiExfShpLGIuQkFEX0RFTElNSVRFUlMpfHwtMTxpLmluZGV4T2YobSl8fFwiIFwiPT09aS5jaGFyQXQoMCl8fFwiIFwiPT09aS5jaGFyQXQoaS5sZW5ndGgtMSk7cmV0dXJuIHI/cytpK3M6aX19fTtpZihiLlJFQ09SRF9TRVA9U3RyaW5nLmZyb21DaGFyQ29kZSgzMCksYi5VTklUX1NFUD1TdHJpbmcuZnJvbUNoYXJDb2RlKDMxKSxiLkJZVEVfT1JERVJfTUFSSz1cIlxcdWZlZmZcIixiLkJBRF9ERUxJTUlURVJTPVtcIlxcclwiLFwiXFxuXCIsJ1wiJyxiLkJZVEVfT1JERVJfTUFSS10sYi5XT1JLRVJTX1NVUFBPUlRFRD0hbiYmISFmLldvcmtlcixiLk5PREVfU1RSRUFNX0lOUFVUPTEsYi5Mb2NhbENodW5rU2l6ZT0xMDQ4NTc2MCxiLlJlbW90ZUNodW5rU2l6ZT01MjQyODgwLGIuRGVmYXVsdERlbGltaXRlcj1cIixcIixiLlBhcnNlcj1FLGIuUGFyc2VySGFuZGxlPWksYi5OZXR3b3JrU3RyZWFtZXI9bCxiLkZpbGVTdHJlYW1lcj1jLGIuU3RyaW5nU3RyZWFtZXI9cCxiLlJlYWRhYmxlU3RyZWFtU3RyZWFtZXI9ZyxmLmpRdWVyeSl7dmFyIGQ9Zi5qUXVlcnk7ZC5mbi5wYXJzZT1mdW5jdGlvbihvKXt2YXIgaT1vLmNvbmZpZ3x8e30saD1bXTtyZXR1cm4gdGhpcy5lYWNoKGZ1bmN0aW9uKGUpe2lmKCEoXCJJTlBVVFwiPT09ZCh0aGlzKS5wcm9wKFwidGFnTmFtZVwiKS50b1VwcGVyQ2FzZSgpJiZcImZpbGVcIj09PWQodGhpcykuYXR0cihcInR5cGVcIikudG9Mb3dlckNhc2UoKSYmZi5GaWxlUmVhZGVyKXx8IXRoaXMuZmlsZXN8fDA9PT10aGlzLmZpbGVzLmxlbmd0aClyZXR1cm4hMDtmb3IodmFyIHQ9MDt0PHRoaXMuZmlsZXMubGVuZ3RoO3QrKyloLnB1c2goe2ZpbGU6dGhpcy5maWxlc1t0XSxpbnB1dEVsZW06dGhpcyxpbnN0YW5jZUNvbmZpZzpkLmV4dGVuZCh7fSxpKX0pfSksZSgpLHRoaXM7ZnVuY3Rpb24gZSgpe2lmKDAhPT1oLmxlbmd0aCl7dmFyIGUsdCxpLHIsbj1oWzBdO2lmKE0oby5iZWZvcmUpKXt2YXIgcz1vLmJlZm9yZShuLmZpbGUsbi5pbnB1dEVsZW0pO2lmKFwib2JqZWN0XCI9PXR5cGVvZiBzKXtpZihcImFib3J0XCI9PT1zLmFjdGlvbilyZXR1cm4gZT1cIkFib3J0RXJyb3JcIix0PW4uZmlsZSxpPW4uaW5wdXRFbGVtLHI9cy5yZWFzb24sdm9pZChNKG8uZXJyb3IpJiZvLmVycm9yKHtuYW1lOmV9LHQsaSxyKSk7aWYoXCJza2lwXCI9PT1zLmFjdGlvbilyZXR1cm4gdm9pZCB1KCk7XCJvYmplY3RcIj09dHlwZW9mIHMuY29uZmlnJiYobi5pbnN0YW5jZUNvbmZpZz1kLmV4dGVuZChuLmluc3RhbmNlQ29uZmlnLHMuY29uZmlnKSl9ZWxzZSBpZihcInNraXBcIj09PXMpcmV0dXJuIHZvaWQgdSgpfXZhciBhPW4uaW5zdGFuY2VDb25maWcuY29tcGxldGU7bi5pbnN0YW5jZUNvbmZpZy5jb21wbGV0ZT1mdW5jdGlvbihlKXtNKGEpJiZhKGUsbi5maWxlLG4uaW5wdXRFbGVtKSx1KCl9LGIucGFyc2Uobi5maWxlLG4uaW5zdGFuY2VDb25maWcpfWVsc2UgTShvLmNvbXBsZXRlKSYmby5jb21wbGV0ZSgpfWZ1bmN0aW9uIHUoKXtoLnNwbGljZSgwLDEpLGUoKX19fWZ1bmN0aW9uIHUoZSl7dGhpcy5faGFuZGxlPW51bGwsdGhpcy5fZmluaXNoZWQ9ITEsdGhpcy5fY29tcGxldGVkPSExLHRoaXMuX2hhbHRlZD0hMSx0aGlzLl9pbnB1dD1udWxsLHRoaXMuX2Jhc2VJbmRleD0wLHRoaXMuX3BhcnRpYWxMaW5lPVwiXCIsdGhpcy5fcm93Q291bnQ9MCx0aGlzLl9zdGFydD0wLHRoaXMuX25leHRDaHVuaz1udWxsLHRoaXMuaXNGaXJzdENodW5rPSEwLHRoaXMuX2NvbXBsZXRlUmVzdWx0cz17ZGF0YTpbXSxlcnJvcnM6W10sbWV0YTp7fX0sZnVuY3Rpb24oZSl7dmFyIHQ9dyhlKTt0LmNodW5rU2l6ZT1wYXJzZUludCh0LmNodW5rU2l6ZSksZS5zdGVwfHxlLmNodW5rfHwodC5jaHVua1NpemU9bnVsbCk7dGhpcy5faGFuZGxlPW5ldyBpKHQpLCh0aGlzLl9oYW5kbGUuc3RyZWFtZXI9dGhpcykuX2NvbmZpZz10fS5jYWxsKHRoaXMsZSksdGhpcy5wYXJzZUNodW5rPWZ1bmN0aW9uKGUsdCl7aWYodGhpcy5pc0ZpcnN0Q2h1bmsmJk0odGhpcy5fY29uZmlnLmJlZm9yZUZpcnN0Q2h1bmspKXt2YXIgaT10aGlzLl9jb25maWcuYmVmb3JlRmlyc3RDaHVuayhlKTt2b2lkIDAhPT1pJiYoZT1pKX10aGlzLmlzRmlyc3RDaHVuaz0hMSx0aGlzLl9oYWx0ZWQ9ITE7dmFyIHI9dGhpcy5fcGFydGlhbExpbmUrZTt0aGlzLl9wYXJ0aWFsTGluZT1cIlwiO3ZhciBuPXRoaXMuX2hhbmRsZS5wYXJzZShyLHRoaXMuX2Jhc2VJbmRleCwhdGhpcy5fZmluaXNoZWQpO2lmKCF0aGlzLl9oYW5kbGUucGF1c2VkKCkmJiF0aGlzLl9oYW5kbGUuYWJvcnRlZCgpKXt2YXIgcz1uLm1ldGEuY3Vyc29yO3RoaXMuX2ZpbmlzaGVkfHwodGhpcy5fcGFydGlhbExpbmU9ci5zdWJzdHJpbmcocy10aGlzLl9iYXNlSW5kZXgpLHRoaXMuX2Jhc2VJbmRleD1zKSxuJiZuLmRhdGEmJih0aGlzLl9yb3dDb3VudCs9bi5kYXRhLmxlbmd0aCk7dmFyIGE9dGhpcy5fZmluaXNoZWR8fHRoaXMuX2NvbmZpZy5wcmV2aWV3JiZ0aGlzLl9yb3dDb3VudD49dGhpcy5fY29uZmlnLnByZXZpZXc7aWYobylmLnBvc3RNZXNzYWdlKHtyZXN1bHRzOm4sd29ya2VySWQ6Yi5XT1JLRVJfSUQsZmluaXNoZWQ6YX0pO2Vsc2UgaWYoTSh0aGlzLl9jb25maWcuY2h1bmspJiYhdCl7aWYodGhpcy5fY29uZmlnLmNodW5rKG4sdGhpcy5faGFuZGxlKSx0aGlzLl9oYW5kbGUucGF1c2VkKCl8fHRoaXMuX2hhbmRsZS5hYm9ydGVkKCkpcmV0dXJuIHZvaWQodGhpcy5faGFsdGVkPSEwKTtuPXZvaWQgMCx0aGlzLl9jb21wbGV0ZVJlc3VsdHM9dm9pZCAwfXJldHVybiB0aGlzLl9jb25maWcuc3RlcHx8dGhpcy5fY29uZmlnLmNodW5rfHwodGhpcy5fY29tcGxldGVSZXN1bHRzLmRhdGE9dGhpcy5fY29tcGxldGVSZXN1bHRzLmRhdGEuY29uY2F0KG4uZGF0YSksdGhpcy5fY29tcGxldGVSZXN1bHRzLmVycm9ycz10aGlzLl9jb21wbGV0ZVJlc3VsdHMuZXJyb3JzLmNvbmNhdChuLmVycm9ycyksdGhpcy5fY29tcGxldGVSZXN1bHRzLm1ldGE9bi5tZXRhKSx0aGlzLl9jb21wbGV0ZWR8fCFhfHwhTSh0aGlzLl9jb25maWcuY29tcGxldGUpfHxuJiZuLm1ldGEuYWJvcnRlZHx8KHRoaXMuX2NvbmZpZy5jb21wbGV0ZSh0aGlzLl9jb21wbGV0ZVJlc3VsdHMsdGhpcy5faW5wdXQpLHRoaXMuX2NvbXBsZXRlZD0hMCksYXx8biYmbi5tZXRhLnBhdXNlZHx8dGhpcy5fbmV4dENodW5rKCksbn10aGlzLl9oYWx0ZWQ9ITB9LHRoaXMuX3NlbmRFcnJvcj1mdW5jdGlvbihlKXtNKHRoaXMuX2NvbmZpZy5lcnJvcik/dGhpcy5fY29uZmlnLmVycm9yKGUpOm8mJnRoaXMuX2NvbmZpZy5lcnJvciYmZi5wb3N0TWVzc2FnZSh7d29ya2VySWQ6Yi5XT1JLRVJfSUQsZXJyb3I6ZSxmaW5pc2hlZDohMX0pfX1mdW5jdGlvbiBsKGUpe3ZhciByOyhlPWV8fHt9KS5jaHVua1NpemV8fChlLmNodW5rU2l6ZT1iLlJlbW90ZUNodW5rU2l6ZSksdS5jYWxsKHRoaXMsZSksdGhpcy5fbmV4dENodW5rPW4/ZnVuY3Rpb24oKXt0aGlzLl9yZWFkQ2h1bmsoKSx0aGlzLl9jaHVua0xvYWRlZCgpfTpmdW5jdGlvbigpe3RoaXMuX3JlYWRDaHVuaygpfSx0aGlzLnN0cmVhbT1mdW5jdGlvbihlKXt0aGlzLl9pbnB1dD1lLHRoaXMuX25leHRDaHVuaygpfSx0aGlzLl9yZWFkQ2h1bms9ZnVuY3Rpb24oKXtpZih0aGlzLl9maW5pc2hlZCl0aGlzLl9jaHVua0xvYWRlZCgpO2Vsc2V7aWYocj1uZXcgWE1MSHR0cFJlcXVlc3QsdGhpcy5fY29uZmlnLndpdGhDcmVkZW50aWFscyYmKHIud2l0aENyZWRlbnRpYWxzPXRoaXMuX2NvbmZpZy53aXRoQ3JlZGVudGlhbHMpLG58fChyLm9ubG9hZD12KHRoaXMuX2NodW5rTG9hZGVkLHRoaXMpLHIub25lcnJvcj12KHRoaXMuX2NodW5rRXJyb3IsdGhpcykpLHIub3Blbih0aGlzLl9jb25maWcuZG93bmxvYWRSZXF1ZXN0Qm9keT9cIlBPU1RcIjpcIkdFVFwiLHRoaXMuX2lucHV0LCFuKSx0aGlzLl9jb25maWcuZG93bmxvYWRSZXF1ZXN0SGVhZGVycyl7dmFyIGU9dGhpcy5fY29uZmlnLmRvd25sb2FkUmVxdWVzdEhlYWRlcnM7Zm9yKHZhciB0IGluIGUpci5zZXRSZXF1ZXN0SGVhZGVyKHQsZVt0XSl9aWYodGhpcy5fY29uZmlnLmNodW5rU2l6ZSl7dmFyIGk9dGhpcy5fc3RhcnQrdGhpcy5fY29uZmlnLmNodW5rU2l6ZS0xO3Iuc2V0UmVxdWVzdEhlYWRlcihcIlJhbmdlXCIsXCJieXRlcz1cIit0aGlzLl9zdGFydCtcIi1cIitpKX10cnl7ci5zZW5kKHRoaXMuX2NvbmZpZy5kb3dubG9hZFJlcXVlc3RCb2R5KX1jYXRjaChlKXt0aGlzLl9jaHVua0Vycm9yKGUubWVzc2FnZSl9biYmMD09PXIuc3RhdHVzJiZ0aGlzLl9jaHVua0Vycm9yKCl9fSx0aGlzLl9jaHVua0xvYWRlZD1mdW5jdGlvbigpezQ9PT1yLnJlYWR5U3RhdGUmJihyLnN0YXR1czwyMDB8fDQwMDw9ci5zdGF0dXM/dGhpcy5fY2h1bmtFcnJvcigpOih0aGlzLl9zdGFydCs9dGhpcy5fY29uZmlnLmNodW5rU2l6ZT90aGlzLl9jb25maWcuY2h1bmtTaXplOnIucmVzcG9uc2VUZXh0Lmxlbmd0aCx0aGlzLl9maW5pc2hlZD0hdGhpcy5fY29uZmlnLmNodW5rU2l6ZXx8dGhpcy5fc3RhcnQ+PWZ1bmN0aW9uKGUpe3ZhciB0PWUuZ2V0UmVzcG9uc2VIZWFkZXIoXCJDb250ZW50LVJhbmdlXCIpO2lmKG51bGw9PT10KXJldHVybi0xO3JldHVybiBwYXJzZUludCh0LnN1YnN0cmluZyh0Lmxhc3RJbmRleE9mKFwiL1wiKSsxKSl9KHIpLHRoaXMucGFyc2VDaHVuayhyLnJlc3BvbnNlVGV4dCkpKX0sdGhpcy5fY2h1bmtFcnJvcj1mdW5jdGlvbihlKXt2YXIgdD1yLnN0YXR1c1RleHR8fGU7dGhpcy5fc2VuZEVycm9yKG5ldyBFcnJvcih0KSl9fWZ1bmN0aW9uIGMoZSl7dmFyIHIsbjsoZT1lfHx7fSkuY2h1bmtTaXplfHwoZS5jaHVua1NpemU9Yi5Mb2NhbENodW5rU2l6ZSksdS5jYWxsKHRoaXMsZSk7dmFyIHM9XCJ1bmRlZmluZWRcIiE9dHlwZW9mIEZpbGVSZWFkZXI7dGhpcy5zdHJlYW09ZnVuY3Rpb24oZSl7dGhpcy5faW5wdXQ9ZSxuPWUuc2xpY2V8fGUud2Via2l0U2xpY2V8fGUubW96U2xpY2Uscz8oKHI9bmV3IEZpbGVSZWFkZXIpLm9ubG9hZD12KHRoaXMuX2NodW5rTG9hZGVkLHRoaXMpLHIub25lcnJvcj12KHRoaXMuX2NodW5rRXJyb3IsdGhpcykpOnI9bmV3IEZpbGVSZWFkZXJTeW5jLHRoaXMuX25leHRDaHVuaygpfSx0aGlzLl9uZXh0Q2h1bms9ZnVuY3Rpb24oKXt0aGlzLl9maW5pc2hlZHx8dGhpcy5fY29uZmlnLnByZXZpZXcmJiEodGhpcy5fcm93Q291bnQ8dGhpcy5fY29uZmlnLnByZXZpZXcpfHx0aGlzLl9yZWFkQ2h1bmsoKX0sdGhpcy5fcmVhZENodW5rPWZ1bmN0aW9uKCl7dmFyIGU9dGhpcy5faW5wdXQ7aWYodGhpcy5fY29uZmlnLmNodW5rU2l6ZSl7dmFyIHQ9TWF0aC5taW4odGhpcy5fc3RhcnQrdGhpcy5fY29uZmlnLmNodW5rU2l6ZSx0aGlzLl9pbnB1dC5zaXplKTtlPW4uY2FsbChlLHRoaXMuX3N0YXJ0LHQpfXZhciBpPXIucmVhZEFzVGV4dChlLHRoaXMuX2NvbmZpZy5lbmNvZGluZyk7c3x8dGhpcy5fY2h1bmtMb2FkZWQoe3RhcmdldDp7cmVzdWx0Oml9fSl9LHRoaXMuX2NodW5rTG9hZGVkPWZ1bmN0aW9uKGUpe3RoaXMuX3N0YXJ0Kz10aGlzLl9jb25maWcuY2h1bmtTaXplLHRoaXMuX2ZpbmlzaGVkPSF0aGlzLl9jb25maWcuY2h1bmtTaXplfHx0aGlzLl9zdGFydD49dGhpcy5faW5wdXQuc2l6ZSx0aGlzLnBhcnNlQ2h1bmsoZS50YXJnZXQucmVzdWx0KX0sdGhpcy5fY2h1bmtFcnJvcj1mdW5jdGlvbigpe3RoaXMuX3NlbmRFcnJvcihyLmVycm9yKX19ZnVuY3Rpb24gcChlKXt2YXIgaTt1LmNhbGwodGhpcyxlPWV8fHt9KSx0aGlzLnN0cmVhbT1mdW5jdGlvbihlKXtyZXR1cm4gaT1lLHRoaXMuX25leHRDaHVuaygpfSx0aGlzLl9uZXh0Q2h1bms9ZnVuY3Rpb24oKXtpZighdGhpcy5fZmluaXNoZWQpe3ZhciBlLHQ9dGhpcy5fY29uZmlnLmNodW5rU2l6ZTtyZXR1cm4gdD8oZT1pLnN1YnN0cmluZygwLHQpLGk9aS5zdWJzdHJpbmcodCkpOihlPWksaT1cIlwiKSx0aGlzLl9maW5pc2hlZD0haSx0aGlzLnBhcnNlQ2h1bmsoZSl9fX1mdW5jdGlvbiBnKGUpe3UuY2FsbCh0aGlzLGU9ZXx8e30pO3ZhciB0PVtdLGk9ITAscj0hMTt0aGlzLnBhdXNlPWZ1bmN0aW9uKCl7dS5wcm90b3R5cGUucGF1c2UuYXBwbHkodGhpcyxhcmd1bWVudHMpLHRoaXMuX2lucHV0LnBhdXNlKCl9LHRoaXMucmVzdW1lPWZ1bmN0aW9uKCl7dS5wcm90b3R5cGUucmVzdW1lLmFwcGx5KHRoaXMsYXJndW1lbnRzKSx0aGlzLl9pbnB1dC5yZXN1bWUoKX0sdGhpcy5zdHJlYW09ZnVuY3Rpb24oZSl7dGhpcy5faW5wdXQ9ZSx0aGlzLl9pbnB1dC5vbihcImRhdGFcIix0aGlzLl9zdHJlYW1EYXRhKSx0aGlzLl9pbnB1dC5vbihcImVuZFwiLHRoaXMuX3N0cmVhbUVuZCksdGhpcy5faW5wdXQub24oXCJlcnJvclwiLHRoaXMuX3N0cmVhbUVycm9yKX0sdGhpcy5fY2hlY2tJc0ZpbmlzaGVkPWZ1bmN0aW9uKCl7ciYmMT09PXQubGVuZ3RoJiYodGhpcy5fZmluaXNoZWQ9ITApfSx0aGlzLl9uZXh0Q2h1bms9ZnVuY3Rpb24oKXt0aGlzLl9jaGVja0lzRmluaXNoZWQoKSx0Lmxlbmd0aD90aGlzLnBhcnNlQ2h1bmsodC5zaGlmdCgpKTppPSEwfSx0aGlzLl9zdHJlYW1EYXRhPXYoZnVuY3Rpb24oZSl7dHJ5e3QucHVzaChcInN0cmluZ1wiPT10eXBlb2YgZT9lOmUudG9TdHJpbmcodGhpcy5fY29uZmlnLmVuY29kaW5nKSksaSYmKGk9ITEsdGhpcy5fY2hlY2tJc0ZpbmlzaGVkKCksdGhpcy5wYXJzZUNodW5rKHQuc2hpZnQoKSkpfWNhdGNoKGUpe3RoaXMuX3N0cmVhbUVycm9yKGUpfX0sdGhpcyksdGhpcy5fc3RyZWFtRXJyb3I9dihmdW5jdGlvbihlKXt0aGlzLl9zdHJlYW1DbGVhblVwKCksdGhpcy5fc2VuZEVycm9yKGUpfSx0aGlzKSx0aGlzLl9zdHJlYW1FbmQ9dihmdW5jdGlvbigpe3RoaXMuX3N0cmVhbUNsZWFuVXAoKSxyPSEwLHRoaXMuX3N0cmVhbURhdGEoXCJcIil9LHRoaXMpLHRoaXMuX3N0cmVhbUNsZWFuVXA9dihmdW5jdGlvbigpe3RoaXMuX2lucHV0LnJlbW92ZUxpc3RlbmVyKFwiZGF0YVwiLHRoaXMuX3N0cmVhbURhdGEpLHRoaXMuX2lucHV0LnJlbW92ZUxpc3RlbmVyKFwiZW5kXCIsdGhpcy5fc3RyZWFtRW5kKSx0aGlzLl9pbnB1dC5yZW1vdmVMaXN0ZW5lcihcImVycm9yXCIsdGhpcy5fc3RyZWFtRXJyb3IpfSx0aGlzKX1mdW5jdGlvbiBpKG0pe3ZhciBhLG8saCxyPU1hdGgucG93KDIsNTMpLG49LXIscz0vXlxccyotPyhcXGQrXFwuP3xcXC5cXGQrfFxcZCtcXC5cXGQrKShbZUVdWy0rXT9cXGQrKT9cXHMqJC8sdT0vXihcXGR7NH0tWzAxXVxcZC1bMC0zXVxcZFRbMC0yXVxcZDpbMC01XVxcZDpbMC01XVxcZFxcLlxcZCsoWystXVswLTJdXFxkOlswLTVdXFxkfFopKXwoXFxkezR9LVswMV1cXGQtWzAtM11cXGRUWzAtMl1cXGQ6WzAtNV1cXGQ6WzAtNV1cXGQoWystXVswLTJdXFxkOlswLTVdXFxkfFopKXwoXFxkezR9LVswMV1cXGQtWzAtM11cXGRUWzAtMl1cXGQ6WzAtNV1cXGQoWystXVswLTJdXFxkOlswLTVdXFxkfFopKSQvLHQ9dGhpcyxpPTAsZj0wLGQ9ITEsZT0hMSxsPVtdLGM9e2RhdGE6W10sZXJyb3JzOltdLG1ldGE6e319O2lmKE0obS5zdGVwKSl7dmFyIHA9bS5zdGVwO20uc3RlcD1mdW5jdGlvbihlKXtpZihjPWUsXygpKWcoKTtlbHNle2lmKGcoKSwwPT09Yy5kYXRhLmxlbmd0aClyZXR1cm47aSs9ZS5kYXRhLmxlbmd0aCxtLnByZXZpZXcmJmk+bS5wcmV2aWV3P28uYWJvcnQoKTooYy5kYXRhPWMuZGF0YVswXSxwKGMsdCkpfX19ZnVuY3Rpb24geShlKXtyZXR1cm5cImdyZWVkeVwiPT09bS5za2lwRW1wdHlMaW5lcz9cIlwiPT09ZS5qb2luKFwiXCIpLnRyaW0oKToxPT09ZS5sZW5ndGgmJjA9PT1lWzBdLmxlbmd0aH1mdW5jdGlvbiBnKCl7aWYoYyYmaCYmKGsoXCJEZWxpbWl0ZXJcIixcIlVuZGV0ZWN0YWJsZURlbGltaXRlclwiLFwiVW5hYmxlIHRvIGF1dG8tZGV0ZWN0IGRlbGltaXRpbmcgY2hhcmFjdGVyOyBkZWZhdWx0ZWQgdG8gJ1wiK2IuRGVmYXVsdERlbGltaXRlcitcIidcIiksaD0hMSksbS5za2lwRW1wdHlMaW5lcylmb3IodmFyIGU9MDtlPGMuZGF0YS5sZW5ndGg7ZSsrKXkoYy5kYXRhW2VdKSYmYy5kYXRhLnNwbGljZShlLS0sMSk7cmV0dXJuIF8oKSYmZnVuY3Rpb24oKXtpZighYylyZXR1cm47ZnVuY3Rpb24gZShlLHQpe00obS50cmFuc2Zvcm1IZWFkZXIpJiYoZT1tLnRyYW5zZm9ybUhlYWRlcihlLHQpKSxsLnB1c2goZSl9aWYoQXJyYXkuaXNBcnJheShjLmRhdGFbMF0pKXtmb3IodmFyIHQ9MDtfKCkmJnQ8Yy5kYXRhLmxlbmd0aDt0KyspYy5kYXRhW3RdLmZvckVhY2goZSk7Yy5kYXRhLnNwbGljZSgwLDEpfWVsc2UgYy5kYXRhLmZvckVhY2goZSl9KCksZnVuY3Rpb24oKXtpZighY3x8IW0uaGVhZGVyJiYhbS5keW5hbWljVHlwaW5nJiYhbS50cmFuc2Zvcm0pcmV0dXJuIGM7ZnVuY3Rpb24gZShlLHQpe3ZhciBpLHI9bS5oZWFkZXI/e306W107Zm9yKGk9MDtpPGUubGVuZ3RoO2krKyl7dmFyIG49aSxzPWVbaV07bS5oZWFkZXImJihuPWk+PWwubGVuZ3RoP1wiX19wYXJzZWRfZXh0cmFcIjpsW2ldKSxtLnRyYW5zZm9ybSYmKHM9bS50cmFuc2Zvcm0ocyxuKSkscz12KG4scyksXCJfX3BhcnNlZF9leHRyYVwiPT09bj8ocltuXT1yW25dfHxbXSxyW25dLnB1c2gocykpOnJbbl09c31yZXR1cm4gbS5oZWFkZXImJihpPmwubGVuZ3RoP2soXCJGaWVsZE1pc21hdGNoXCIsXCJUb29NYW55RmllbGRzXCIsXCJUb28gbWFueSBmaWVsZHM6IGV4cGVjdGVkIFwiK2wubGVuZ3RoK1wiIGZpZWxkcyBidXQgcGFyc2VkIFwiK2ksZit0KTppPGwubGVuZ3RoJiZrKFwiRmllbGRNaXNtYXRjaFwiLFwiVG9vRmV3RmllbGRzXCIsXCJUb28gZmV3IGZpZWxkczogZXhwZWN0ZWQgXCIrbC5sZW5ndGgrXCIgZmllbGRzIGJ1dCBwYXJzZWQgXCIraSxmK3QpKSxyfXZhciB0PTE7IWMuZGF0YS5sZW5ndGh8fEFycmF5LmlzQXJyYXkoYy5kYXRhWzBdKT8oYy5kYXRhPWMuZGF0YS5tYXAoZSksdD1jLmRhdGEubGVuZ3RoKTpjLmRhdGE9ZShjLmRhdGEsMCk7bS5oZWFkZXImJmMubWV0YSYmKGMubWV0YS5maWVsZHM9bCk7cmV0dXJuIGYrPXQsY30oKX1mdW5jdGlvbiBfKCl7cmV0dXJuIG0uaGVhZGVyJiYwPT09bC5sZW5ndGh9ZnVuY3Rpb24gdihlLHQpe3JldHVybiBpPWUsbS5keW5hbWljVHlwaW5nRnVuY3Rpb24mJnZvaWQgMD09PW0uZHluYW1pY1R5cGluZ1tpXSYmKG0uZHluYW1pY1R5cGluZ1tpXT1tLmR5bmFtaWNUeXBpbmdGdW5jdGlvbihpKSksITA9PT0obS5keW5hbWljVHlwaW5nW2ldfHxtLmR5bmFtaWNUeXBpbmcpP1widHJ1ZVwiPT09dHx8XCJUUlVFXCI9PT10fHxcImZhbHNlXCIhPT10JiZcIkZBTFNFXCIhPT10JiYoZnVuY3Rpb24oZSl7aWYocy50ZXN0KGUpKXt2YXIgdD1wYXJzZUZsb2F0KGUpO2lmKG48dCYmdDxyKXJldHVybiEwfXJldHVybiExfSh0KT9wYXJzZUZsb2F0KHQpOnUudGVzdCh0KT9uZXcgRGF0ZSh0KTpcIlwiPT09dD9udWxsOnQpOnQ7dmFyIGl9ZnVuY3Rpb24gayhlLHQsaSxyKXt2YXIgbj17dHlwZTplLGNvZGU6dCxtZXNzYWdlOml9O3ZvaWQgMCE9PXImJihuLnJvdz1yKSxjLmVycm9ycy5wdXNoKG4pfXRoaXMucGFyc2U9ZnVuY3Rpb24oZSx0LGkpe3ZhciByPW0ucXVvdGVDaGFyfHwnXCInO2lmKG0ubmV3bGluZXx8KG0ubmV3bGluZT1mdW5jdGlvbihlLHQpe2U9ZS5zdWJzdHJpbmcoMCwxMDQ4NTc2KTt2YXIgaT1uZXcgUmVnRXhwKGoodCkrXCIoW15dKj8pXCIraih0KSxcImdtXCIpLHI9KGU9ZS5yZXBsYWNlKGksXCJcIikpLnNwbGl0KFwiXFxyXCIpLG49ZS5zcGxpdChcIlxcblwiKSxzPTE8bi5sZW5ndGgmJm5bMF0ubGVuZ3RoPHJbMF0ubGVuZ3RoO2lmKDE9PT1yLmxlbmd0aHx8cylyZXR1cm5cIlxcblwiO2Zvcih2YXIgYT0wLG89MDtvPHIubGVuZ3RoO28rKylcIlxcblwiPT09cltvXVswXSYmYSsrO3JldHVybiBhPj1yLmxlbmd0aC8yP1wiXFxyXFxuXCI6XCJcXHJcIn0oZSxyKSksaD0hMSxtLmRlbGltaXRlcilNKG0uZGVsaW1pdGVyKSYmKG0uZGVsaW1pdGVyPW0uZGVsaW1pdGVyKGUpLGMubWV0YS5kZWxpbWl0ZXI9bS5kZWxpbWl0ZXIpO2Vsc2V7dmFyIG49ZnVuY3Rpb24oZSx0LGkscixuKXt2YXIgcyxhLG8saDtuPW58fFtcIixcIixcIlxcdFwiLFwifFwiLFwiO1wiLGIuUkVDT1JEX1NFUCxiLlVOSVRfU0VQXTtmb3IodmFyIHU9MDt1PG4ubGVuZ3RoO3UrKyl7dmFyIGY9blt1XSxkPTAsbD0wLGM9MDtvPXZvaWQgMDtmb3IodmFyIHA9bmV3IEUoe2NvbW1lbnRzOnIsZGVsaW1pdGVyOmYsbmV3bGluZTp0LHByZXZpZXc6MTB9KS5wYXJzZShlKSxnPTA7ZzxwLmRhdGEubGVuZ3RoO2crKylpZihpJiZ5KHAuZGF0YVtnXSkpYysrO2Vsc2V7dmFyIF89cC5kYXRhW2ddLmxlbmd0aDtsKz1fLHZvaWQgMCE9PW8/MDxfJiYoZCs9TWF0aC5hYnMoXy1vKSxvPV8pOm89X30wPHAuZGF0YS5sZW5ndGgmJihsLz1wLmRhdGEubGVuZ3RoLWMpLCh2b2lkIDA9PT1hfHxkPD1hKSYmKHZvaWQgMD09PWh8fGg8bCkmJjEuOTk8bCYmKGE9ZCxzPWYsaD1sKX1yZXR1cm57c3VjY2Vzc2Z1bDohIShtLmRlbGltaXRlcj1zKSxiZXN0RGVsaW1pdGVyOnN9fShlLG0ubmV3bGluZSxtLnNraXBFbXB0eUxpbmVzLG0uY29tbWVudHMsbS5kZWxpbWl0ZXJzVG9HdWVzcyk7bi5zdWNjZXNzZnVsP20uZGVsaW1pdGVyPW4uYmVzdERlbGltaXRlcjooaD0hMCxtLmRlbGltaXRlcj1iLkRlZmF1bHREZWxpbWl0ZXIpLGMubWV0YS5kZWxpbWl0ZXI9bS5kZWxpbWl0ZXJ9dmFyIHM9dyhtKTtyZXR1cm4gbS5wcmV2aWV3JiZtLmhlYWRlciYmcy5wcmV2aWV3KyssYT1lLG89bmV3IEUocyksYz1vLnBhcnNlKGEsdCxpKSxnKCksZD97bWV0YTp7cGF1c2VkOiEwfX06Y3x8e21ldGE6e3BhdXNlZDohMX19fSx0aGlzLnBhdXNlZD1mdW5jdGlvbigpe3JldHVybiBkfSx0aGlzLnBhdXNlPWZ1bmN0aW9uKCl7ZD0hMCxvLmFib3J0KCksYT1NKG0uY2h1bmspP1wiXCI6YS5zdWJzdHJpbmcoby5nZXRDaGFySW5kZXgoKSl9LHRoaXMucmVzdW1lPWZ1bmN0aW9uKCl7dC5zdHJlYW1lci5faGFsdGVkPyhkPSExLHQuc3RyZWFtZXIucGFyc2VDaHVuayhhLCEwKSk6c2V0VGltZW91dCh0LnJlc3VtZSwzKX0sdGhpcy5hYm9ydGVkPWZ1bmN0aW9uKCl7cmV0dXJuIGV9LHRoaXMuYWJvcnQ9ZnVuY3Rpb24oKXtlPSEwLG8uYWJvcnQoKSxjLm1ldGEuYWJvcnRlZD0hMCxNKG0uY29tcGxldGUpJiZtLmNvbXBsZXRlKGMpLGE9XCJcIn19ZnVuY3Rpb24gaihlKXtyZXR1cm4gZS5yZXBsYWNlKC9bLiorP14ke30oKXxbXFxdXFxcXF0vZyxcIlxcXFwkJlwiKX1mdW5jdGlvbiBFKGUpe3ZhciBTLE89KGU9ZXx8e30pLmRlbGltaXRlcix4PWUubmV3bGluZSxJPWUuY29tbWVudHMsVD1lLnN0ZXAsRD1lLnByZXZpZXcsQT1lLmZhc3RNb2RlLEw9Uz12b2lkIDA9PT1lLnF1b3RlQ2hhcj8nXCInOmUucXVvdGVDaGFyO2lmKHZvaWQgMCE9PWUuZXNjYXBlQ2hhciYmKEw9ZS5lc2NhcGVDaGFyKSwoXCJzdHJpbmdcIiE9dHlwZW9mIE98fC0xPGIuQkFEX0RFTElNSVRFUlMuaW5kZXhPZihPKSkmJihPPVwiLFwiKSxJPT09Tyl0aHJvdyBuZXcgRXJyb3IoXCJDb21tZW50IGNoYXJhY3RlciBzYW1lIGFzIGRlbGltaXRlclwiKTshMD09PUk/ST1cIiNcIjooXCJzdHJpbmdcIiE9dHlwZW9mIEl8fC0xPGIuQkFEX0RFTElNSVRFUlMuaW5kZXhPZihJKSkmJihJPSExKSxcIlxcblwiIT09eCYmXCJcXHJcIiE9PXgmJlwiXFxyXFxuXCIhPT14JiYoeD1cIlxcblwiKTt2YXIgRj0wLHo9ITE7dGhpcy5wYXJzZT1mdW5jdGlvbihyLHQsaSl7aWYoXCJzdHJpbmdcIiE9dHlwZW9mIHIpdGhyb3cgbmV3IEVycm9yKFwiSW5wdXQgbXVzdCBiZSBhIHN0cmluZ1wiKTt2YXIgbj1yLmxlbmd0aCxlPU8ubGVuZ3RoLHM9eC5sZW5ndGgsYT1JLmxlbmd0aCxvPU0oVCksaD1bXSx1PVtdLGY9W10sZD1GPTA7aWYoIXIpcmV0dXJuIEMoKTtpZihBfHwhMSE9PUEmJi0xPT09ci5pbmRleE9mKFMpKXtmb3IodmFyIGw9ci5zcGxpdCh4KSxjPTA7YzxsLmxlbmd0aDtjKyspe2lmKGY9bFtjXSxGKz1mLmxlbmd0aCxjIT09bC5sZW5ndGgtMSlGKz14Lmxlbmd0aDtlbHNlIGlmKGkpcmV0dXJuIEMoKTtpZighSXx8Zi5zdWJzdHJpbmcoMCxhKSE9PUkpe2lmKG8pe2lmKGg9W10sayhmLnNwbGl0KE8pKSxSKCkseilyZXR1cm4gQygpfWVsc2UgayhmLnNwbGl0KE8pKTtpZihEJiZEPD1jKXJldHVybiBoPWguc2xpY2UoMCxEKSxDKCEwKX19cmV0dXJuIEMoKX1mb3IodmFyIHA9ci5pbmRleE9mKE8sRiksZz1yLmluZGV4T2YoeCxGKSxfPW5ldyBSZWdFeHAoaihMKStqKFMpLFwiZ1wiKSxtPXIuaW5kZXhPZihTLEYpOzspaWYocltGXSE9PVMpaWYoSSYmMD09PWYubGVuZ3RoJiZyLnN1YnN0cmluZyhGLEYrYSk9PT1JKXtpZigtMT09PWcpcmV0dXJuIEMoKTtGPWcrcyxnPXIuaW5kZXhPZih4LEYpLHA9ci5pbmRleE9mKE8sRil9ZWxzZSBpZigtMSE9PXAmJihwPGd8fC0xPT09ZykpZi5wdXNoKHIuc3Vic3RyaW5nKEYscCkpLEY9cCtlLHA9ci5pbmRleE9mKE8sRik7ZWxzZXtpZigtMT09PWcpYnJlYWs7aWYoZi5wdXNoKHIuc3Vic3RyaW5nKEYsZykpLHcoZytzKSxvJiYoUigpLHopKXJldHVybiBDKCk7aWYoRCYmaC5sZW5ndGg+PUQpcmV0dXJuIEMoITApfWVsc2UgZm9yKG09RixGKys7Oyl7aWYoLTE9PT0obT1yLmluZGV4T2YoUyxtKzEpKSlyZXR1cm4gaXx8dS5wdXNoKHt0eXBlOlwiUXVvdGVzXCIsY29kZTpcIk1pc3NpbmdRdW90ZXNcIixtZXNzYWdlOlwiUXVvdGVkIGZpZWxkIHVudGVybWluYXRlZFwiLHJvdzpoLmxlbmd0aCxpbmRleDpGfSksRSgpO2lmKG09PT1uLTEpcmV0dXJuIEUoci5zdWJzdHJpbmcoRixtKS5yZXBsYWNlKF8sUykpO2lmKFMhPT1MfHxyW20rMV0hPT1MKXtpZihTPT09THx8MD09PW18fHJbbS0xXSE9PUwpey0xIT09cCYmcDxtKzEmJihwPXIuaW5kZXhPZihPLG0rMSkpLC0xIT09ZyYmZzxtKzEmJihnPXIuaW5kZXhPZih4LG0rMSkpO3ZhciB5PWIoLTE9PT1nP3A6TWF0aC5taW4ocCxnKSk7aWYoclttKzEreV09PT1PKXtmLnB1c2goci5zdWJzdHJpbmcoRixtKS5yZXBsYWNlKF8sUykpLHJbRj1tKzEreStlXSE9PVMmJihtPXIuaW5kZXhPZihTLEYpKSxwPXIuaW5kZXhPZihPLEYpLGc9ci5pbmRleE9mKHgsRik7YnJlYWt9dmFyIHY9YihnKTtpZihyLnN1YnN0cmluZyhtKzErdixtKzErditzKT09PXgpe2lmKGYucHVzaChyLnN1YnN0cmluZyhGLG0pLnJlcGxhY2UoXyxTKSksdyhtKzErditzKSxwPXIuaW5kZXhPZihPLEYpLG09ci5pbmRleE9mKFMsRiksbyYmKFIoKSx6KSlyZXR1cm4gQygpO2lmKEQmJmgubGVuZ3RoPj1EKXJldHVybiBDKCEwKTticmVha311LnB1c2goe3R5cGU6XCJRdW90ZXNcIixjb2RlOlwiSW52YWxpZFF1b3Rlc1wiLG1lc3NhZ2U6XCJUcmFpbGluZyBxdW90ZSBvbiBxdW90ZWQgZmllbGQgaXMgbWFsZm9ybWVkXCIscm93OmgubGVuZ3RoLGluZGV4OkZ9KSxtKyt9fWVsc2UgbSsrfXJldHVybiBFKCk7ZnVuY3Rpb24gayhlKXtoLnB1c2goZSksZD1GfWZ1bmN0aW9uIGIoZSl7dmFyIHQ9MDtpZigtMSE9PWUpe3ZhciBpPXIuc3Vic3RyaW5nKG0rMSxlKTtpJiZcIlwiPT09aS50cmltKCkmJih0PWkubGVuZ3RoKX1yZXR1cm4gdH1mdW5jdGlvbiBFKGUpe3JldHVybiBpfHwodm9pZCAwPT09ZSYmKGU9ci5zdWJzdHJpbmcoRikpLGYucHVzaChlKSxGPW4sayhmKSxvJiZSKCkpLEMoKX1mdW5jdGlvbiB3KGUpe0Y9ZSxrKGYpLGY9W10sZz1yLmluZGV4T2YoeCxGKX1mdW5jdGlvbiBDKGUpe3JldHVybntkYXRhOmgsZXJyb3JzOnUsbWV0YTp7ZGVsaW1pdGVyOk8sbGluZWJyZWFrOngsYWJvcnRlZDp6LHRydW5jYXRlZDohIWUsY3Vyc29yOmQrKHR8fDApfX19ZnVuY3Rpb24gUigpe1QoQygpKSxoPVtdLHU9W119fSx0aGlzLmFib3J0PWZ1bmN0aW9uKCl7ej0hMH0sdGhpcy5nZXRDaGFySW5kZXg9ZnVuY3Rpb24oKXtyZXR1cm4gRn19ZnVuY3Rpb24gXyhlKXt2YXIgdD1lLmRhdGEsaT1hW3Qud29ya2VySWRdLHI9ITE7aWYodC5lcnJvcilpLnVzZXJFcnJvcih0LmVycm9yLHQuZmlsZSk7ZWxzZSBpZih0LnJlc3VsdHMmJnQucmVzdWx0cy5kYXRhKXt2YXIgbj17YWJvcnQ6ZnVuY3Rpb24oKXtyPSEwLG0odC53b3JrZXJJZCx7ZGF0YTpbXSxlcnJvcnM6W10sbWV0YTp7YWJvcnRlZDohMH19KX0scGF1c2U6eSxyZXN1bWU6eX07aWYoTShpLnVzZXJTdGVwKSl7Zm9yKHZhciBzPTA7czx0LnJlc3VsdHMuZGF0YS5sZW5ndGgmJihpLnVzZXJTdGVwKHtkYXRhOnQucmVzdWx0cy5kYXRhW3NdLGVycm9yczp0LnJlc3VsdHMuZXJyb3JzLG1ldGE6dC5yZXN1bHRzLm1ldGF9LG4pLCFyKTtzKyspO2RlbGV0ZSB0LnJlc3VsdHN9ZWxzZSBNKGkudXNlckNodW5rKSYmKGkudXNlckNodW5rKHQucmVzdWx0cyxuLHQuZmlsZSksZGVsZXRlIHQucmVzdWx0cyl9dC5maW5pc2hlZCYmIXImJm0odC53b3JrZXJJZCx0LnJlc3VsdHMpfWZ1bmN0aW9uIG0oZSx0KXt2YXIgaT1hW2VdO00oaS51c2VyQ29tcGxldGUpJiZpLnVzZXJDb21wbGV0ZSh0KSxpLnRlcm1pbmF0ZSgpLGRlbGV0ZSBhW2VdfWZ1bmN0aW9uIHkoKXt0aHJvdyBuZXcgRXJyb3IoXCJOb3QgaW1wbGVtZW50ZWQuXCIpfWZ1bmN0aW9uIHcoZSl7aWYoXCJvYmplY3RcIiE9dHlwZW9mIGV8fG51bGw9PT1lKXJldHVybiBlO3ZhciB0PUFycmF5LmlzQXJyYXkoZSk/W106e307Zm9yKHZhciBpIGluIGUpdFtpXT13KGVbaV0pO3JldHVybiB0fWZ1bmN0aW9uIHYoZSx0KXtyZXR1cm4gZnVuY3Rpb24oKXtlLmFwcGx5KHQsYXJndW1lbnRzKX19ZnVuY3Rpb24gTShlKXtyZXR1cm5cImZ1bmN0aW9uXCI9PXR5cGVvZiBlfXJldHVybiBvJiYoZi5vbm1lc3NhZ2U9ZnVuY3Rpb24oZSl7dmFyIHQ9ZS5kYXRhO3ZvaWQgMD09PWIuV09SS0VSX0lEJiZ0JiYoYi5XT1JLRVJfSUQ9dC53b3JrZXJJZCk7aWYoXCJzdHJpbmdcIj09dHlwZW9mIHQuaW5wdXQpZi5wb3N0TWVzc2FnZSh7d29ya2VySWQ6Yi5XT1JLRVJfSUQscmVzdWx0czpiLnBhcnNlKHQuaW5wdXQsdC5jb25maWcpLGZpbmlzaGVkOiEwfSk7ZWxzZSBpZihmLkZpbGUmJnQuaW5wdXQgaW5zdGFuY2VvZiBGaWxlfHx0LmlucHV0IGluc3RhbmNlb2YgT2JqZWN0KXt2YXIgaT1iLnBhcnNlKHQuaW5wdXQsdC5jb25maWcpO2kmJmYucG9zdE1lc3NhZ2Uoe3dvcmtlcklkOmIuV09SS0VSX0lELHJlc3VsdHM6aSxmaW5pc2hlZDohMH0pfX0pLChsLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHUucHJvdG90eXBlKSkuY29uc3RydWN0b3I9bCwoYy5wcm90b3R5cGU9T2JqZWN0LmNyZWF0ZSh1LnByb3RvdHlwZSkpLmNvbnN0cnVjdG9yPWMsKHAucHJvdG90eXBlPU9iamVjdC5jcmVhdGUocC5wcm90b3R5cGUpKS5jb25zdHJ1Y3Rvcj1wLChnLnByb3RvdHlwZT1PYmplY3QuY3JlYXRlKHUucHJvdG90eXBlKSkuY29uc3RydWN0b3I9ZyxifSk7IiwidmFyIF9fYXdhaXRlciA9ICh0aGlzICYmIHRoaXMuX19hd2FpdGVyKSB8fCBmdW5jdGlvbiAodGhpc0FyZywgX2FyZ3VtZW50cywgUCwgZ2VuZXJhdG9yKSB7XG4gICAgZnVuY3Rpb24gYWRvcHQodmFsdWUpIHsgcmV0dXJuIHZhbHVlIGluc3RhbmNlb2YgUCA/IHZhbHVlIDogbmV3IFAoZnVuY3Rpb24gKHJlc29sdmUpIHsgcmVzb2x2ZSh2YWx1ZSk7IH0pOyB9XG4gICAgcmV0dXJuIG5ldyAoUCB8fCAoUCA9IFByb21pc2UpKShmdW5jdGlvbiAocmVzb2x2ZSwgcmVqZWN0KSB7XG4gICAgICAgIGZ1bmN0aW9uIGZ1bGZpbGxlZCh2YWx1ZSkgeyB0cnkgeyBzdGVwKGdlbmVyYXRvci5uZXh0KHZhbHVlKSk7IH0gY2F0Y2ggKGUpIHsgcmVqZWN0KGUpOyB9IH1cbiAgICAgICAgZnVuY3Rpb24gcmVqZWN0ZWQodmFsdWUpIHsgdHJ5IHsgc3RlcChnZW5lcmF0b3JbXCJ0aHJvd1wiXSh2YWx1ZSkpOyB9IGNhdGNoIChlKSB7IHJlamVjdChlKTsgfSB9XG4gICAgICAgIGZ1bmN0aW9uIHN0ZXAocmVzdWx0KSB7IHJlc3VsdC5kb25lID8gcmVzb2x2ZShyZXN1bHQudmFsdWUpIDogYWRvcHQocmVzdWx0LnZhbHVlKS50aGVuKGZ1bGZpbGxlZCwgcmVqZWN0ZWQpOyB9XG4gICAgICAgIHN0ZXAoKGdlbmVyYXRvciA9IGdlbmVyYXRvci5hcHBseSh0aGlzQXJnLCBfYXJndW1lbnRzIHx8IFtdKSkubmV4dCgpKTtcbiAgICB9KTtcbn07XG52YXIgX19nZW5lcmF0b3IgPSAodGhpcyAmJiB0aGlzLl9fZ2VuZXJhdG9yKSB8fCBmdW5jdGlvbiAodGhpc0FyZywgYm9keSkge1xuICAgIHZhciBfID0geyBsYWJlbDogMCwgc2VudDogZnVuY3Rpb24oKSB7IGlmICh0WzBdICYgMSkgdGhyb3cgdFsxXTsgcmV0dXJuIHRbMV07IH0sIHRyeXM6IFtdLCBvcHM6IFtdIH0sIGYsIHksIHQsIGc7XG4gICAgcmV0dXJuIGcgPSB7IG5leHQ6IHZlcmIoMCksIFwidGhyb3dcIjogdmVyYigxKSwgXCJyZXR1cm5cIjogdmVyYigyKSB9LCB0eXBlb2YgU3ltYm9sID09PSBcImZ1bmN0aW9uXCIgJiYgKGdbU3ltYm9sLml0ZXJhdG9yXSA9IGZ1bmN0aW9uKCkgeyByZXR1cm4gdGhpczsgfSksIGc7XG4gICAgZnVuY3Rpb24gdmVyYihuKSB7IHJldHVybiBmdW5jdGlvbiAodikgeyByZXR1cm4gc3RlcChbbiwgdl0pOyB9OyB9XG4gICAgZnVuY3Rpb24gc3RlcChvcCkge1xuICAgICAgICBpZiAoZikgdGhyb3cgbmV3IFR5cGVFcnJvcihcIkdlbmVyYXRvciBpcyBhbHJlYWR5IGV4ZWN1dGluZy5cIik7XG4gICAgICAgIHdoaWxlIChfKSB0cnkge1xuICAgICAgICAgICAgaWYgKGYgPSAxLCB5ICYmICh0ID0gb3BbMF0gJiAyID8geVtcInJldHVyblwiXSA6IG9wWzBdID8geVtcInRocm93XCJdIHx8ICgodCA9IHlbXCJyZXR1cm5cIl0pICYmIHQuY2FsbCh5KSwgMCkgOiB5Lm5leHQpICYmICEodCA9IHQuY2FsbCh5LCBvcFsxXSkpLmRvbmUpIHJldHVybiB0O1xuICAgICAgICAgICAgaWYgKHkgPSAwLCB0KSBvcCA9IFtvcFswXSAmIDIsIHQudmFsdWVdO1xuICAgICAgICAgICAgc3dpdGNoIChvcFswXSkge1xuICAgICAgICAgICAgICAgIGNhc2UgMDogY2FzZSAxOiB0ID0gb3A7IGJyZWFrO1xuICAgICAgICAgICAgICAgIGNhc2UgNDogXy5sYWJlbCsrOyByZXR1cm4geyB2YWx1ZTogb3BbMV0sIGRvbmU6IGZhbHNlIH07XG4gICAgICAgICAgICAgICAgY2FzZSA1OiBfLmxhYmVsKys7IHkgPSBvcFsxXTsgb3AgPSBbMF07IGNvbnRpbnVlO1xuICAgICAgICAgICAgICAgIGNhc2UgNzogb3AgPSBfLm9wcy5wb3AoKTsgXy50cnlzLnBvcCgpOyBjb250aW51ZTtcbiAgICAgICAgICAgICAgICBkZWZhdWx0OlxuICAgICAgICAgICAgICAgICAgICBpZiAoISh0ID0gXy50cnlzLCB0ID0gdC5sZW5ndGggPiAwICYmIHRbdC5sZW5ndGggLSAxXSkgJiYgKG9wWzBdID09PSA2IHx8IG9wWzBdID09PSAyKSkgeyBfID0gMDsgY29udGludWU7IH1cbiAgICAgICAgICAgICAgICAgICAgaWYgKG9wWzBdID09PSAzICYmICghdCB8fCAob3BbMV0gPiB0WzBdICYmIG9wWzFdIDwgdFszXSkpKSB7IF8ubGFiZWwgPSBvcFsxXTsgYnJlYWs7IH1cbiAgICAgICAgICAgICAgICAgICAgaWYgKG9wWzBdID09PSA2ICYmIF8ubGFiZWwgPCB0WzFdKSB7IF8ubGFiZWwgPSB0WzFdOyB0ID0gb3A7IGJyZWFrOyB9XG4gICAgICAgICAgICAgICAgICAgIGlmICh0ICYmIF8ubGFiZWwgPCB0WzJdKSB7IF8ubGFiZWwgPSB0WzJdOyBfLm9wcy5wdXNoKG9wKTsgYnJlYWs7IH1cbiAgICAgICAgICAgICAgICAgICAgaWYgKHRbMl0pIF8ub3BzLnBvcCgpO1xuICAgICAgICAgICAgICAgICAgICBfLnRyeXMucG9wKCk7IGNvbnRpbnVlO1xuICAgICAgICAgICAgfVxuICAgICAgICAgICAgb3AgPSBib2R5LmNhbGwodGhpc0FyZywgXyk7XG4gICAgICAgIH0gY2F0Y2ggKGUpIHsgb3AgPSBbNiwgZV07IHkgPSAwOyB9IGZpbmFsbHkgeyBmID0gdCA9IDA7IH1cbiAgICAgICAgaWYgKG9wWzBdICYgNSkgdGhyb3cgb3BbMV07IHJldHVybiB7IHZhbHVlOiBvcFswXSA/IG9wWzFdIDogdm9pZCAwLCBkb25lOiB0cnVlIH07XG4gICAgfVxufTtcbnZhciBQYXBhID0gcmVxdWlyZShcInBhcGFwYXJzZVwiKTtcbnZhciBlcnJvcnMgPSByZXF1aXJlKFwiLi9lcnJvcnMuanNcIik7XG52YXIgY29tcGFyZSA9IHJlcXVpcmUoJ25hdHVyYWwtb3JkZXJieScpLmNvbXBhcmU7XG52YXIgZnJlZURhdGFzMkhUTUwgPSAoZnVuY3Rpb24gKCkge1xuICAgIGZ1bmN0aW9uIGZyZWVEYXRhczJIVE1MKCkge1xuICAgICAgICB0aGlzLl9kYXRhc1ZpZXdFbHQgPSB7IGlkOiBcIlwiLCBlbHRET006IHVuZGVmaW5lZCB9O1xuICAgICAgICB0aGlzLl9kYXRhc1NvdXJjZVVybCA9IFwiXCI7XG4gICAgICAgIHRoaXMuX2RhdGFzU2VsZWN0b3JzID0gW107XG4gICAgICAgIHRoaXMuX2RhdGFzU29ydGluZ0NvbHVtbnMgPSBbXTtcbiAgICAgICAgdGhpcy5wYXJzZU1ldGEgPSB1bmRlZmluZWQ7XG4gICAgICAgIHRoaXMucGFyc2VEYXRhcyA9IFtdO1xuICAgICAgICB0aGlzLnBhcnNlRXJyb3JzID0gW107XG4gICAgICAgIHRoaXMuZGF0YXNIVE1MID0gXCJcIjtcbiAgICAgICAgdGhpcy5zdG9wSWZQYXJzZUVycm9ycyA9IGZhbHNlO1xuICAgIH1cbiAgICBPYmplY3QuZGVmaW5lUHJvcGVydHkoZnJlZURhdGFzMkhUTUwucHJvdG90eXBlLCBcImRhdGFzVmlld0VsdFwiLCB7XG4gICAgICAgIHNldDogZnVuY3Rpb24gKGVsdCkge1xuICAgICAgICAgICAgdmFyIGNoZWNrQ29udGFpbmVyRXhpc3QgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZChlbHQuaWQpO1xuICAgICAgICAgICAgaWYgKGNoZWNrQ29udGFpbmVyRXhpc3QgPT09IG51bGwpXG4gICAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGVycm9ycy5lbGVtZW50Tm90Rm91bmQgKyBlbHQuaWQpO1xuICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgdGhpcy5fZGF0YXNWaWV3RWx0LmlkID0gZWx0LmlkO1xuICAgICAgICAgICAgICAgIHRoaXMuX2RhdGFzVmlld0VsdC5lbHRET00gPSBjaGVja0NvbnRhaW5lckV4aXN0O1xuICAgICAgICAgICAgfVxuICAgICAgICB9LFxuICAgICAgICBlbnVtZXJhYmxlOiBmYWxzZSxcbiAgICAgICAgY29uZmlndXJhYmxlOiB0cnVlXG4gICAgfSk7XG4gICAgT2JqZWN0LmRlZmluZVByb3BlcnR5KGZyZWVEYXRhczJIVE1MLnByb3RvdHlwZSwgXCJkYXRhc1NvdXJjZVVybFwiLCB7XG4gICAgICAgIHNldDogZnVuY3Rpb24gKHVybCkge1xuICAgICAgICAgICAgaWYgKHVybC50cmltKCkubGVuZ3RoID09PSAwKVxuICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihlcnJvcnMubmVlZFVybCk7XG4gICAgICAgICAgICBlbHNlXG4gICAgICAgICAgICAgICAgdGhpcy5fZGF0YXNTb3VyY2VVcmwgPSB1cmwudHJpbSgpO1xuICAgICAgICB9LFxuICAgICAgICBlbnVtZXJhYmxlOiBmYWxzZSxcbiAgICAgICAgY29uZmlndXJhYmxlOiB0cnVlXG4gICAgfSk7XG4gICAgT2JqZWN0LmRlZmluZVByb3BlcnR5KGZyZWVEYXRhczJIVE1MLnByb3RvdHlwZSwgXCJkYXRhc1NlbGVjdG9yc1wiLCB7XG4gICAgICAgIGdldDogZnVuY3Rpb24gKCkge1xuICAgICAgICAgICAgcmV0dXJuIHRoaXMuX2RhdGFzU2VsZWN0b3JzO1xuICAgICAgICB9LFxuICAgICAgICBzZXQ6IGZ1bmN0aW9uIChzZWxlY3Rpb25FbHRzKSB7XG4gICAgICAgICAgICB0aGlzLl9kYXRhc1NlbGVjdG9ycyA9IFtdO1xuICAgICAgICAgICAgdmFyIGNoZWNrQ29udGFpbmVyRXhpc3Q7XG4gICAgICAgICAgICBmb3IgKHZhciBpID0gMDsgaSA8IHNlbGVjdGlvbkVsdHMubGVuZ3RoOyBpKyspIHtcbiAgICAgICAgICAgICAgICBjaGVja0NvbnRhaW5lckV4aXN0ID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoc2VsZWN0aW9uRWx0c1tpXS5pZCk7XG4gICAgICAgICAgICAgICAgaWYgKGNoZWNrQ29udGFpbmVyRXhpc3QgPT09IG51bGwpXG4gICAgICAgICAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3JzLmVsZW1lbnROb3RGb3VuZCArIHNlbGVjdGlvbkVsdHNbaV0uaWQpO1xuICAgICAgICAgICAgICAgIGVsc2UgaWYgKE51bWJlci5pc0ludGVnZXIoc2VsZWN0aW9uRWx0c1tpXS5kYXRhc0ZpZWxkTmIpID09PSBmYWxzZSB8fCBzZWxlY3Rpb25FbHRzW2ldLmRhdGFzRmllbGROYiA8IDApXG4gICAgICAgICAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3JzLm5lZWROYXR1cmFsTnVtYmVyKTtcbiAgICAgICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgc2VsZWN0aW9uRWx0c1tpXS5lbHRET00gPSBjaGVja0NvbnRhaW5lckV4aXN0O1xuICAgICAgICAgICAgICAgICAgICBpZiAoc2VsZWN0aW9uRWx0c1tpXS5zZXBhcmF0b3IgIT09IHVuZGVmaW5lZCAmJiBzZWxlY3Rpb25FbHRzW2ldLnNlcGFyYXRvciA9PT0gXCJcIilcbiAgICAgICAgICAgICAgICAgICAgICAgIHNlbGVjdGlvbkVsdHNbaV0uc2VwYXJhdG9yID0gdW5kZWZpbmVkO1xuICAgICAgICAgICAgICAgICAgICB0aGlzLl9kYXRhc1NlbGVjdG9ycy5wdXNoKHNlbGVjdGlvbkVsdHNbaV0pO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgfSxcbiAgICAgICAgZW51bWVyYWJsZTogZmFsc2UsXG4gICAgICAgIGNvbmZpZ3VyYWJsZTogdHJ1ZVxuICAgIH0pO1xuICAgIE9iamVjdC5kZWZpbmVQcm9wZXJ0eShmcmVlRGF0YXMySFRNTC5wcm90b3R5cGUsIFwiZGF0YXNTb3J0aW5nQ29sdW1uc1wiLCB7XG4gICAgICAgIGdldDogZnVuY3Rpb24gKCkge1xuICAgICAgICAgICAgcmV0dXJuIHRoaXMuX2RhdGFzU29ydGluZ0NvbHVtbnM7XG4gICAgICAgIH0sXG4gICAgICAgIHNldDogZnVuY3Rpb24gKHNvcnRpbmdDb2x1bW5zKSB7XG4gICAgICAgICAgICB0aGlzLl9kYXRhc1NvcnRpbmdDb2x1bW5zID0gW107XG4gICAgICAgICAgICBmb3IgKHZhciBpID0gMDsgaSA8IHNvcnRpbmdDb2x1bW5zLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgICAgICAgaWYgKE51bWJlci5pc0ludGVnZXIoc29ydGluZ0NvbHVtbnNbaV0uZGF0YXNGaWVsZE5iKSA9PT0gZmFsc2UgfHwgc29ydGluZ0NvbHVtbnNbaV0uZGF0YXNGaWVsZE5iIDwgMClcbiAgICAgICAgICAgICAgICAgICAgY29uc29sZS5lcnJvcihlcnJvcnMubmVlZE5hdHVyYWxOdW1iZXIpO1xuICAgICAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICBzb3J0aW5nQ29sdW1uc1tpXS5vcmRlciA9IHVuZGVmaW5lZDtcbiAgICAgICAgICAgICAgICAgICAgdGhpcy5fZGF0YXNTb3J0aW5nQ29sdW1ucy5wdXNoKHNvcnRpbmdDb2x1bW5zW2ldKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgIH0sXG4gICAgICAgIGVudW1lcmFibGU6IGZhbHNlLFxuICAgICAgICBjb25maWd1cmFibGU6IHRydWVcbiAgICB9KTtcbiAgICBmcmVlRGF0YXMySFRNTC5wcm90b3R5cGUucGFyc2UgPSBmdW5jdGlvbiAoKSB7XG4gICAgICAgIHJldHVybiBfX2F3YWl0ZXIodGhpcywgdm9pZCAwLCB2b2lkIDAsIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICAgIHZhciBjb252ZXJ0ZXI7XG4gICAgICAgICAgICByZXR1cm4gX19nZW5lcmF0b3IodGhpcywgZnVuY3Rpb24gKF9hKSB7XG4gICAgICAgICAgICAgICAgY29udmVydGVyID0gdGhpcztcbiAgICAgICAgICAgICAgICByZXR1cm4gWzIsIG5ldyBQcm9taXNlKGZ1bmN0aW9uIChyZXNvbHZlLCByZWplY3QpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjb252ZXJ0ZXIuX2RhdGFzU291cmNlVXJsICE9PSBcIlwiKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgUGFwYS5wYXJzZShjb252ZXJ0ZXIuX2RhdGFzU291cmNlVXJsLCB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHF1b3RlQ2hhcjogJ1wiJyxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaGVhZGVyOiB0cnVlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb21wbGV0ZTogZnVuY3Rpb24gKHJlc3VsdHMpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnZlcnRlci5wYXJzZUVycm9ycyA9IHJlc3VsdHMuZXJyb3JzO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29udmVydGVyLnBhcnNlRGF0YXMgPSByZXN1bHRzLmRhdGE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2YXIgcmVhbEZpZWxkcyA9IFtdO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yICh2YXIgaSBpbiByZXN1bHRzLm1ldGEuZmllbGRzKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHJlc3VsdHMubWV0YS5maWVsZHNbaV0udHJpbSgpICE9PSBcIlwiKVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZWFsRmllbGRzLnB1c2gocmVzdWx0cy5tZXRhLmZpZWxkc1tpXSk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXN1bHRzLm1ldGEuZmllbGRzID0gcmVhbEZpZWxkcztcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnZlcnRlci5wYXJzZU1ldGEgPSByZXN1bHRzLm1ldGE7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXNvbHZlKHRydWUpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBlcnJvcjogZnVuY3Rpb24gKGVycm9yKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZWplY3QobmV3IEVycm9yKGVycm9ycy5wYXJzZXJGYWlsKSk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2FkOiB0cnVlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBza2lwRW1wdHlMaW5lczogdHJ1ZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgIGVsc2VcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICByZWplY3QobmV3IEVycm9yKGVycm9ycy5uZWVkVXJsKSk7XG4gICAgICAgICAgICAgICAgICAgIH0pXTtcbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9KTtcbiAgICB9O1xuICAgIGZyZWVEYXRhczJIVE1MLnByb3RvdHlwZS5ydW4gPSBmdW5jdGlvbiAoKSB7XG4gICAgICAgIHJldHVybiBfX2F3YWl0ZXIodGhpcywgdm9pZCAwLCB2b2lkIDAsIGZ1bmN0aW9uICgpIHtcbiAgICAgICAgICAgIHZhciBjb252ZXJ0ZXJfMSwgc2VsZWN0b3JzSFRNTCwgaSwgdmFsdWVzLCBjb2xOYW1lLCByb3csIGNoZWNrZWRWYWx1ZSwgY2hlY2tlZFZhbHVlcywgaV8xLCBjaGVja2VkVmFsdWUsIGosIHNlbGVjdEVsZW1lbnQsIGk7XG4gICAgICAgICAgICByZXR1cm4gX19nZW5lcmF0b3IodGhpcywgZnVuY3Rpb24gKF9hKSB7XG4gICAgICAgICAgICAgICAgc3dpdGNoIChfYS5sYWJlbCkge1xuICAgICAgICAgICAgICAgICAgICBjYXNlIDA6XG4gICAgICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy5fZGF0YXNWaWV3RWx0LmVsdERPTSA9PT0gdW5kZWZpbmVkKVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihlcnJvcnMubmVlZERhdGFzRWx0KTtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmICh0aGlzLl9kYXRhc1NvdXJjZVVybCA9PT0gXCJcIilcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoZXJyb3JzLm5lZWRVcmwpO1xuICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIFs0LCB0aGlzLnBhcnNlKCldO1xuICAgICAgICAgICAgICAgICAgICBjYXNlIDE6XG4gICAgICAgICAgICAgICAgICAgICAgICBfYS5zZW50KCk7XG4gICAgICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy5wYXJzZURhdGFzLmxlbmd0aCA9PT0gMCB8fCB0aGlzLnBhcnNlTWV0YS5maWVsZHMgPT09IHVuZGVmaW5lZClcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoZXJyb3JzLmRhdGFzTm90Rm91bmQpO1xuICAgICAgICAgICAgICAgICAgICAgICAgZWxzZSBpZiAodGhpcy5zdG9wSWZQYXJzZUVycm9ycyAmJiB0aGlzLnBhcnNlRXJyb3JzLmxlbmd0aCAhPT0gMClcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zb2xlLmVycm9yKHRoaXMucGFyc2VFcnJvcnMpO1xuICAgICAgICAgICAgICAgICAgICAgICAgZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29udmVydGVyXzEgPSB0aGlzO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmICh0aGlzLl9kYXRhc1NlbGVjdG9ycy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNlbGVjdG9yc0hUTUwgPSBbXTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpIGluIHRoaXMuX2RhdGFzU2VsZWN0b3JzKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy5fZGF0YXNTZWxlY3RvcnNbaV0uZGF0YXNGaWVsZE5iID4gKHRoaXMucGFyc2VNZXRhLmZpZWxkcy5sZW5ndGggLSAxKSlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgRXJyb3IoZXJyb3JzLnNlbGVjdG9yRmllbGROb3RGb3VuZCk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB2YWx1ZXMgPSBbXSwgY29sTmFtZSA9IHRoaXMucGFyc2VNZXRhLmZpZWxkc1t0aGlzLl9kYXRhc1NlbGVjdG9yc1tpXS5kYXRhc0ZpZWxkTmJdO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZvciAocm93IGluIHRoaXMucGFyc2VEYXRhcykge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAodGhpcy5fZGF0YXNTZWxlY3RvcnNbaV0uc2VwYXJhdG9yID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNoZWNrZWRWYWx1ZSA9IHRoaXMucGFyc2VEYXRhc1tyb3ddW2NvbE5hbWVdLnRyaW0oKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlmIChjaGVja2VkVmFsdWUgIT09IFwiXCIgJiYgdmFsdWVzLmluZGV4T2YoY2hlY2tlZFZhbHVlKSA9PT0gLTEpXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFsdWVzLnB1c2goY2hlY2tlZFZhbHVlKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNoZWNrZWRWYWx1ZXMgPSB0aGlzLnBhcnNlRGF0YXNbcm93XVtjb2xOYW1lXS5zcGxpdCh0aGlzLl9kYXRhc1NlbGVjdG9yc1tpXS5zZXBhcmF0b3IpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpXzEgaW4gY2hlY2tlZFZhbHVlcykge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNoZWNrZWRWYWx1ZSA9IGNoZWNrZWRWYWx1ZXNbaV8xXS50cmltKCk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGNoZWNrZWRWYWx1ZSAhPT0gXCJcIiAmJiB2YWx1ZXMuaW5kZXhPZihjaGVja2VkVmFsdWUpID09PSAtMSlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFsdWVzLnB1c2goY2hlY2tlZFZhbHVlKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAodmFsdWVzLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFsdWVzLnNvcnQoY29tcGFyZSgpKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fZGF0YXNTZWxlY3RvcnNbaV0ubmFtZSA9IGNvbE5hbWU7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRoaXMuX2RhdGFzU2VsZWN0b3JzW2ldLnZhbHVlcyA9IHZhbHVlcztcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2VsZWN0b3JzSFRNTFtpXSA9IFwiPGxhYmVsIGZvcj0nZnJlZURhdGFzMkhUTUxTZWxlY3RvclwiICsgaSArIFwiJz5cIiArIGNvbE5hbWUgKyBcIiA6IDwvbGFiZWw+PHNlbGVjdCBuYW1lPSdmcmVlRGF0YXMySFRNTFNlbGVjdG9yXCIgKyBpICsgXCInIGlkPSdmcmVlRGF0YXMySFRNTFNlbGVjdG9yXCIgKyBpICsgXCInPjxvcHRpb24gdmFsdWU9JzAnPi0tLS08L29wdGlvbj5cIjtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChqIGluIHZhbHVlcylcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNlbGVjdG9yc0hUTUxbaV0gKz0gXCI8b3B0aW9uIHZhbHVlPSdcIiArIChOdW1iZXIoaikgKyAxKSArIFwiJz5cIiArIHZhbHVlc1tqXSArIFwiPC9vcHRpb24+XCI7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNlbGVjdG9yc0hUTUxbaV0gKz0gXCI8L3NlbGVjdD5cIjtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5fZGF0YXNTZWxlY3RvcnNbaV0uZWx0RE9NLmlubmVySFRNTCA9IHNlbGVjdG9yc0hUTUxbaV07XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNlbGVjdEVsZW1lbnQgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZChcImZyZWVEYXRhczJIVE1MU2VsZWN0b3JcIiArIGkpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzZWxlY3RFbGVtZW50LmFkZEV2ZW50TGlzdGVuZXIoJ2NoYW5nZScsIGZ1bmN0aW9uIChlKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb252ZXJ0ZXJfMS5kYXRhc0hUTUwgPSBjb252ZXJ0ZXJfMS5jcmVhdGVEYXRhc0hUTUwoY29udmVydGVyXzEucGFyc2VNZXRhLmZpZWxkcywgY29udmVydGVyXzEucGFyc2VEYXRhcyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb252ZXJ0ZXJfMS5yZWZyZXNoVmlldygpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgZm9yIChpIGluIHRoaXMuX2RhdGFzU29ydGluZ0NvbHVtbnMpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKHRoaXMuX2RhdGFzU29ydGluZ0NvbHVtbnNbaV0uZGF0YXNGaWVsZE5iID4gKHRoaXMucGFyc2VNZXRhLmZpZWxkcy5sZW5ndGggLSAxKSlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihlcnJvcnMuc29ydGluZ0NvbHVtbnNGaWVsZE5vdEZvdW5kKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5kYXRhc0hUTUwgPSB0aGlzLmNyZWF0ZURhdGFzSFRNTCh0aGlzLnBhcnNlTWV0YS5maWVsZHMsIHRoaXMucGFyc2VEYXRhcyk7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdGhpcy5yZWZyZXNoVmlldygpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiBbMiwgdHJ1ZV07XG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gWzJdO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9KTtcbiAgICB9O1xuICAgIGZyZWVEYXRhczJIVE1MLnByb3RvdHlwZS5yZWZyZXNoVmlldyA9IGZ1bmN0aW9uICgpIHtcbiAgICAgICAgaWYgKHRoaXMuX2RhdGFzVmlld0VsdC5lbHRET00gIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgdmFyIGNvbnZlcnRlcl8yID0gdGhpcztcbiAgICAgICAgICAgIHRoaXMuX2RhdGFzVmlld0VsdC5lbHRET00uaW5uZXJIVE1MID0gdGhpcy5kYXRhc0hUTUw7XG4gICAgICAgICAgICBpZiAodGhpcy5fZGF0YXNTb3J0aW5nQ29sdW1ucy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgICAgICAgdmFyIGdldFRhYmxlVGggPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsKFwidGFibGUgdGhcIik7XG4gICAgICAgICAgICAgICAgaWYgKGdldFRhYmxlVGggIT09IG51bGwpIHtcbiAgICAgICAgICAgICAgICAgICAgdmFyIF9sb29wXzEgPSBmdW5jdGlvbiAoaSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgdmFyIGRhdGFzRmllbGROYiA9IHRoaXNfMS5fZGF0YXNTb3J0aW5nQ29sdW1uc1tpXS5kYXRhc0ZpZWxkTmI7XG4gICAgICAgICAgICAgICAgICAgICAgICB2YXIgaHRtbENvbnRlbnQgPSBnZXRUYWJsZVRoW2RhdGFzRmllbGROYl0uaW5uZXJIVE1MO1xuICAgICAgICAgICAgICAgICAgICAgICAgaHRtbENvbnRlbnQgPSBcIjxhIGhyZWY9JyNmcmVlRGF0YXMySFRNTFNvcnRpbmdcIiArIGRhdGFzRmllbGROYiArIFwiJyBpZD0nZnJlZURhdGFzMkhUTUxTb3J0aW5nXCIgKyBkYXRhc0ZpZWxkTmIgKyBcIic+XCIgKyBodG1sQ29udGVudCArIFwiPC9hPlwiO1xuICAgICAgICAgICAgICAgICAgICAgICAgZ2V0VGFibGVUaFtkYXRhc0ZpZWxkTmJdLmlubmVySFRNTCA9IGh0bWxDb250ZW50O1xuICAgICAgICAgICAgICAgICAgICAgICAgdmFyIHNvcnRpbmdFbGVtZW50ID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoXCJmcmVlRGF0YXMySFRNTFNvcnRpbmdcIiArIGRhdGFzRmllbGROYik7XG4gICAgICAgICAgICAgICAgICAgICAgICBzb3J0aW5nRWxlbWVudC5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIGZ1bmN0aW9uIChlKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgZS5wcmV2ZW50RGVmYXVsdCgpO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHZhciBvcmRlciA9IGNvbnZlcnRlcl8yLmRhdGFzU29ydGluZ0NvbHVtbnNbaV0ub3JkZXI7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKG9yZGVyID09PSB1bmRlZmluZWQgfHwgb3JkZXIgPT09IFwiZGVzY1wiKVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb252ZXJ0ZXJfMi5kYXRhc1NvcnRpbmdDb2x1bW5zW2ldLm9yZGVyID0gXCJhc2NcIjtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBlbHNlXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnZlcnRlcl8yLmRhdGFzU29ydGluZ0NvbHVtbnNbaV0ub3JkZXIgPSBcImRlc2NcIjtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb252ZXJ0ZXJfMi5fZGF0YXNTb3J0ZWRDb2x1bW4gPSBjb252ZXJ0ZXJfMi5kYXRhc1NvcnRpbmdDb2x1bW5zW2ldO1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnZlcnRlcl8yLmRhdGFzSFRNTCA9IGNvbnZlcnRlcl8yLmNyZWF0ZURhdGFzSFRNTChjb252ZXJ0ZXJfMi5wYXJzZU1ldGEuZmllbGRzLCBjb252ZXJ0ZXJfMi5wYXJzZURhdGFzKTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb252ZXJ0ZXJfMi5yZWZyZXNoVmlldygpO1xuICAgICAgICAgICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgICAgICAgIH07XG4gICAgICAgICAgICAgICAgICAgIHZhciB0aGlzXzEgPSB0aGlzO1xuICAgICAgICAgICAgICAgICAgICBmb3IgKHZhciBpIGluIHRoaXMuX2RhdGFzU29ydGluZ0NvbHVtbnMpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIF9sb29wXzEoaSk7XG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICB9O1xuICAgIGZyZWVEYXRhczJIVE1MLnByb3RvdHlwZS5jcmVhdGVEYXRhc0hUTUwgPSBmdW5jdGlvbiAoZmllbGRzLCBkYXRhcykge1xuICAgICAgICB2YXIgY2hlY2tTZWxlY3RvckV4aXN0LCBmaWx0ZXJzID0gW107XG4gICAgICAgIGZvciAodmFyIGkgaW4gdGhpcy5fZGF0YXNTZWxlY3RvcnMpIHtcbiAgICAgICAgICAgIGNoZWNrU2VsZWN0b3JFeGlzdCA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3IoXCIjXCIgKyB0aGlzLl9kYXRhc1NlbGVjdG9yc1tpXS5pZCArIFwiIHNlbGVjdFwiKTtcbiAgICAgICAgICAgIGlmIChjaGVja1NlbGVjdG9yRXhpc3QgIT0gbnVsbCAmJiBjaGVja1NlbGVjdG9yRXhpc3Quc2VsZWN0ZWRJbmRleCAhPSAwKVxuICAgICAgICAgICAgICAgIGZpbHRlcnMucHVzaCh7IGZpZWxkOiB0aGlzLl9kYXRhc1NlbGVjdG9yc1tpXS5uYW1lLCB2YWx1ZTogdGhpcy5fZGF0YXNTZWxlY3RvcnNbaV0udmFsdWVzW2NoZWNrU2VsZWN0b3JFeGlzdC5zZWxlY3RlZEluZGV4IC0gMV0sIHNlcGFyYXRvcjogdGhpcy5fZGF0YXNTZWxlY3RvcnNbaV0uc2VwYXJhdG9yIH0pO1xuICAgICAgICB9XG4gICAgICAgIGlmICh0aGlzLl9kYXRhc1NvcnRlZENvbHVtbiAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgICB2YXIgY29sXzEgPSBmaWVsZHNbdGhpcy5fZGF0YXNTb3J0ZWRDb2x1bW4uZGF0YXNGaWVsZE5iXTtcbiAgICAgICAgICAgIHZhciBjb2xPcmRlcl8xID0gdGhpcy5fZGF0YXNTb3J0ZWRDb2x1bW4ub3JkZXI7XG4gICAgICAgICAgICBkYXRhcy5zb3J0KGZ1bmN0aW9uIChhLCBiKSB7IHJldHVybiBjb21wYXJlKHsgb3JkZXI6IGNvbE9yZGVyXzEgfSkoYVtjb2xfMV0sIGJbY29sXzFdKTsgfSk7XG4gICAgICAgIH1cbiAgICAgICAgdmFyIGRhdGFzSFRNTCA9IFwiPHRhYmxlPjx0aGVhZD5cIjtcbiAgICAgICAgZm9yICh2YXIgaSBpbiBmaWVsZHMpXG4gICAgICAgICAgICBkYXRhc0hUTUwgKz0gXCI8dGg+XCIgKyBmaWVsZHNbaV0gKyBcIjwvdGg+XCI7XG4gICAgICAgIGRhdGFzSFRNTCArPSBcIjwvdGhlYWQ+PHRib2R5PlwiO1xuICAgICAgICBmb3IgKHZhciByb3cgaW4gZGF0YXMpIHtcbiAgICAgICAgICAgIHZhciB2aXNpYmxlID0gdHJ1ZTtcbiAgICAgICAgICAgIGlmIChmaWx0ZXJzLmxlbmd0aCAhPT0gMCkge1xuICAgICAgICAgICAgICAgIGZvciAodmFyIGkgaW4gZmlsdGVycykge1xuICAgICAgICAgICAgICAgICAgICBpZiAoZmlsdGVyc1tpXS5zZXBhcmF0b3IgPT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGRhdGFzW3Jvd11bZmlsdGVyc1tpXS5maWVsZF0udHJpbSgpICE9IGZpbHRlcnNbaV0udmFsdWUpXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdmlzaWJsZSA9IGZhbHNlO1xuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgICAgICAgICAgdmFyIGNoZWNrZWRWYWx1ZXMgPSBkYXRhc1tyb3ddW2ZpbHRlcnNbaV0uZmllbGRdLnNwbGl0KGZpbHRlcnNbaV0uc2VwYXJhdG9yKSwgZmluZGVkID0gZmFsc2U7XG4gICAgICAgICAgICAgICAgICAgICAgICBmb3IgKHZhciBqIGluIGNoZWNrZWRWYWx1ZXMpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoY2hlY2tlZFZhbHVlc1tqXS50cmltKCkgPT09IGZpbHRlcnNbaV0udmFsdWUpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmluZGVkID0gdHJ1ZTtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgaWYgKCFmaW5kZWQpXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdmlzaWJsZSA9IGZhbHNlO1xuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgaWYgKHZpc2libGUpIHtcbiAgICAgICAgICAgICAgICBkYXRhc0hUTUwgKz0gXCI8dHI+XCI7XG4gICAgICAgICAgICAgICAgZm9yICh2YXIgZmllbGQgaW4gZGF0YXNbcm93XSkge1xuICAgICAgICAgICAgICAgICAgICBpZiAoZmllbGRzLmluZGV4T2YoZmllbGQpICE9PSAtMSlcbiAgICAgICAgICAgICAgICAgICAgICAgIGRhdGFzSFRNTCArPSBcIjx0ZD5cIiArIGRhdGFzW3Jvd11bZmllbGRdICsgXCI8L3RkPlwiO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBkYXRhc0hUTUwgKz0gXCI8L3RyPlwiO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIGRhdGFzSFRNTCArPSBcIjwvdGJvZHk+PC90YWJsZT5cIjtcbiAgICAgICAgcmV0dXJuIGRhdGFzSFRNTDtcbiAgICB9O1xuICAgIHJldHVybiBmcmVlRGF0YXMySFRNTDtcbn0oKSk7XG5leHBvcnQgeyBmcmVlRGF0YXMySFRNTCB9O1xuIiwibW9kdWxlLmV4cG9ydHMgPVxue1xuICAgIGRhdGFzTm90Rm91bmQgOiBcIkF1Y3VuZSBkb25uw6llIG4nYSDDqXTDqSB0cm91dsOpZS5cIixcbiAgICBlbGVtZW50Tm90Rm91bmQgOiBcIkF1Y3VuIMOpbMOpbWVudCBIVE1MIG4nYSDDqXTDqSB0cm91dsOpIGF5YW50IGNvbW1lIFxcXCJpZFxcXCIgOiBcIixcbiAgICBuZWVkRGF0YXNFbHQ6IFwiTWVyY2kgZGUgZm91cm5pciB1biBpZCB2YWxpZGUgcG91ciBsJ8OpbMOpbWVudCBvw7kgYWZmaWNoZXIgbGVzIGRvbm7DqWVzLlwiLFxuICAgIG5lZWROYXR1cmFsTnVtYmVyOiBcIk1lcmNpIGRlIGZvdXJuaXIgdW4gbm9tYnJlIGVudGllciBzdXDDqXJpZXVyIG91IMOpZ2FsIMOgIHrDqXJvIHBvdXIgZMOpc2lnbmVyIGNoYXF1ZSBjb2xvbm5lLlwiLFxuICAgIG5lZWRVcmw6IFwiTWVyY2kgZGUgZm91cm5pciB1bmUgdXJsIHZhbGlkZSBwb3VyIGxlIGZpY2hpZXIgQ1NWIMOgIHBhcnNlci5cIixcbiAgICBwYXJzZXJGYWlsOiBcIkxhIGxlY3R1cmUgZGVzIGRvbm7DqWVzIGR1IGZpY2hpZXIgYSDDqWNob3XDqS5cIixcbiAgICBzZWxlY3RvckZpZWxkTm90Rm91bmQ6IFwiQXUgbW9pbnMgdW5lIGRlcyBjb2xvbm5lcyBkZXZhbnQgc2VydmlyIMOgIGZpbHRyZXIgbGVzIGRvbm7DqWVzIG4nZXhpc3RlIHBhcyBkYW5zIGxlIGZpY2hpZXIuXCIsXG4gICAgc29ydGluZ0NvbHVtbnNGaWVsZE5vdEZvdW5kOiBcIkF1IG1vaW5zIHVuZSBkZXMgY29sb25uZXMgZGV2YW50IHNlcnZpciDDoCBjbGFzc2VyIGxlcyBkb25uw6llcyBuJ2V4aXN0ZSBwYXMgZGFucyBsZSBmaWNoaWVyLlwiLFxufTsiLCIvLyBUaGUgbW9kdWxlIGNhY2hlXG52YXIgX193ZWJwYWNrX21vZHVsZV9jYWNoZV9fID0ge307XG5cbi8vIFRoZSByZXF1aXJlIGZ1bmN0aW9uXG5mdW5jdGlvbiBfX3dlYnBhY2tfcmVxdWlyZV9fKG1vZHVsZUlkKSB7XG5cdC8vIENoZWNrIGlmIG1vZHVsZSBpcyBpbiBjYWNoZVxuXHR2YXIgY2FjaGVkTW9kdWxlID0gX193ZWJwYWNrX21vZHVsZV9jYWNoZV9fW21vZHVsZUlkXTtcblx0aWYgKGNhY2hlZE1vZHVsZSAhPT0gdW5kZWZpbmVkKSB7XG5cdFx0cmV0dXJuIGNhY2hlZE1vZHVsZS5leHBvcnRzO1xuXHR9XG5cdC8vIENyZWF0ZSBhIG5ldyBtb2R1bGUgKGFuZCBwdXQgaXQgaW50byB0aGUgY2FjaGUpXG5cdHZhciBtb2R1bGUgPSBfX3dlYnBhY2tfbW9kdWxlX2NhY2hlX19bbW9kdWxlSWRdID0ge1xuXHRcdC8vIG5vIG1vZHVsZS5pZCBuZWVkZWRcblx0XHQvLyBubyBtb2R1bGUubG9hZGVkIG5lZWRlZFxuXHRcdGV4cG9ydHM6IHt9XG5cdH07XG5cblx0Ly8gRXhlY3V0ZSB0aGUgbW9kdWxlIGZ1bmN0aW9uXG5cdF9fd2VicGFja19tb2R1bGVzX19bbW9kdWxlSWRdLmNhbGwobW9kdWxlLmV4cG9ydHMsIG1vZHVsZSwgbW9kdWxlLmV4cG9ydHMsIF9fd2VicGFja19yZXF1aXJlX18pO1xuXG5cdC8vIFJldHVybiB0aGUgZXhwb3J0cyBvZiB0aGUgbW9kdWxlXG5cdHJldHVybiBtb2R1bGUuZXhwb3J0cztcbn1cblxuIiwiLy8gZGVmaW5lIGdldHRlciBmdW5jdGlvbnMgZm9yIGhhcm1vbnkgZXhwb3J0c1xuX193ZWJwYWNrX3JlcXVpcmVfXy5kID0gKGV4cG9ydHMsIGRlZmluaXRpb24pID0+IHtcblx0Zm9yKHZhciBrZXkgaW4gZGVmaW5pdGlvbikge1xuXHRcdGlmKF9fd2VicGFja19yZXF1aXJlX18ubyhkZWZpbml0aW9uLCBrZXkpICYmICFfX3dlYnBhY2tfcmVxdWlyZV9fLm8oZXhwb3J0cywga2V5KSkge1xuXHRcdFx0T2JqZWN0LmRlZmluZVByb3BlcnR5KGV4cG9ydHMsIGtleSwgeyBlbnVtZXJhYmxlOiB0cnVlLCBnZXQ6IGRlZmluaXRpb25ba2V5XSB9KTtcblx0XHR9XG5cdH1cbn07IiwiX193ZWJwYWNrX3JlcXVpcmVfXy5vID0gKG9iaiwgcHJvcCkgPT4gKE9iamVjdC5wcm90b3R5cGUuaGFzT3duUHJvcGVydHkuY2FsbChvYmosIHByb3ApKSIsIi8vIGRlZmluZSBfX2VzTW9kdWxlIG9uIGV4cG9ydHNcbl9fd2VicGFja19yZXF1aXJlX18uciA9IChleHBvcnRzKSA9PiB7XG5cdGlmKHR5cGVvZiBTeW1ib2wgIT09ICd1bmRlZmluZWQnICYmIFN5bWJvbC50b1N0cmluZ1RhZykge1xuXHRcdE9iamVjdC5kZWZpbmVQcm9wZXJ0eShleHBvcnRzLCBTeW1ib2wudG9TdHJpbmdUYWcsIHsgdmFsdWU6ICdNb2R1bGUnIH0pO1xuXHR9XG5cdE9iamVjdC5kZWZpbmVQcm9wZXJ0eShleHBvcnRzLCAnX19lc01vZHVsZScsIHsgdmFsdWU6IHRydWUgfSk7XG59OyIsInZhciBfX2F3YWl0ZXIgPSAodGhpcyAmJiB0aGlzLl9fYXdhaXRlcikgfHwgZnVuY3Rpb24gKHRoaXNBcmcsIF9hcmd1bWVudHMsIFAsIGdlbmVyYXRvcikge1xuICAgIGZ1bmN0aW9uIGFkb3B0KHZhbHVlKSB7IHJldHVybiB2YWx1ZSBpbnN0YW5jZW9mIFAgPyB2YWx1ZSA6IG5ldyBQKGZ1bmN0aW9uIChyZXNvbHZlKSB7IHJlc29sdmUodmFsdWUpOyB9KTsgfVxuICAgIHJldHVybiBuZXcgKFAgfHwgKFAgPSBQcm9taXNlKSkoZnVuY3Rpb24gKHJlc29sdmUsIHJlamVjdCkge1xuICAgICAgICBmdW5jdGlvbiBmdWxmaWxsZWQodmFsdWUpIHsgdHJ5IHsgc3RlcChnZW5lcmF0b3IubmV4dCh2YWx1ZSkpOyB9IGNhdGNoIChlKSB7IHJlamVjdChlKTsgfSB9XG4gICAgICAgIGZ1bmN0aW9uIHJlamVjdGVkKHZhbHVlKSB7IHRyeSB7IHN0ZXAoZ2VuZXJhdG9yW1widGhyb3dcIl0odmFsdWUpKTsgfSBjYXRjaCAoZSkgeyByZWplY3QoZSk7IH0gfVxuICAgICAgICBmdW5jdGlvbiBzdGVwKHJlc3VsdCkgeyByZXN1bHQuZG9uZSA/IHJlc29sdmUocmVzdWx0LnZhbHVlKSA6IGFkb3B0KHJlc3VsdC52YWx1ZSkudGhlbihmdWxmaWxsZWQsIHJlamVjdGVkKTsgfVxuICAgICAgICBzdGVwKChnZW5lcmF0b3IgPSBnZW5lcmF0b3IuYXBwbHkodGhpc0FyZywgX2FyZ3VtZW50cyB8fCBbXSkpLm5leHQoKSk7XG4gICAgfSk7XG59O1xudmFyIF9fZ2VuZXJhdG9yID0gKHRoaXMgJiYgdGhpcy5fX2dlbmVyYXRvcikgfHwgZnVuY3Rpb24gKHRoaXNBcmcsIGJvZHkpIHtcbiAgICB2YXIgXyA9IHsgbGFiZWw6IDAsIHNlbnQ6IGZ1bmN0aW9uKCkgeyBpZiAodFswXSAmIDEpIHRocm93IHRbMV07IHJldHVybiB0WzFdOyB9LCB0cnlzOiBbXSwgb3BzOiBbXSB9LCBmLCB5LCB0LCBnO1xuICAgIHJldHVybiBnID0geyBuZXh0OiB2ZXJiKDApLCBcInRocm93XCI6IHZlcmIoMSksIFwicmV0dXJuXCI6IHZlcmIoMikgfSwgdHlwZW9mIFN5bWJvbCA9PT0gXCJmdW5jdGlvblwiICYmIChnW1N5bWJvbC5pdGVyYXRvcl0gPSBmdW5jdGlvbigpIHsgcmV0dXJuIHRoaXM7IH0pLCBnO1xuICAgIGZ1bmN0aW9uIHZlcmIobikgeyByZXR1cm4gZnVuY3Rpb24gKHYpIHsgcmV0dXJuIHN0ZXAoW24sIHZdKTsgfTsgfVxuICAgIGZ1bmN0aW9uIHN0ZXAob3ApIHtcbiAgICAgICAgaWYgKGYpIHRocm93IG5ldyBUeXBlRXJyb3IoXCJHZW5lcmF0b3IgaXMgYWxyZWFkeSBleGVjdXRpbmcuXCIpO1xuICAgICAgICB3aGlsZSAoXykgdHJ5IHtcbiAgICAgICAgICAgIGlmIChmID0gMSwgeSAmJiAodCA9IG9wWzBdICYgMiA/IHlbXCJyZXR1cm5cIl0gOiBvcFswXSA/IHlbXCJ0aHJvd1wiXSB8fCAoKHQgPSB5W1wicmV0dXJuXCJdKSAmJiB0LmNhbGwoeSksIDApIDogeS5uZXh0KSAmJiAhKHQgPSB0LmNhbGwoeSwgb3BbMV0pKS5kb25lKSByZXR1cm4gdDtcbiAgICAgICAgICAgIGlmICh5ID0gMCwgdCkgb3AgPSBbb3BbMF0gJiAyLCB0LnZhbHVlXTtcbiAgICAgICAgICAgIHN3aXRjaCAob3BbMF0pIHtcbiAgICAgICAgICAgICAgICBjYXNlIDA6IGNhc2UgMTogdCA9IG9wOyBicmVhaztcbiAgICAgICAgICAgICAgICBjYXNlIDQ6IF8ubGFiZWwrKzsgcmV0dXJuIHsgdmFsdWU6IG9wWzFdLCBkb25lOiBmYWxzZSB9O1xuICAgICAgICAgICAgICAgIGNhc2UgNTogXy5sYWJlbCsrOyB5ID0gb3BbMV07IG9wID0gWzBdOyBjb250aW51ZTtcbiAgICAgICAgICAgICAgICBjYXNlIDc6IG9wID0gXy5vcHMucG9wKCk7IF8udHJ5cy5wb3AoKTsgY29udGludWU7XG4gICAgICAgICAgICAgICAgZGVmYXVsdDpcbiAgICAgICAgICAgICAgICAgICAgaWYgKCEodCA9IF8udHJ5cywgdCA9IHQubGVuZ3RoID4gMCAmJiB0W3QubGVuZ3RoIC0gMV0pICYmIChvcFswXSA9PT0gNiB8fCBvcFswXSA9PT0gMikpIHsgXyA9IDA7IGNvbnRpbnVlOyB9XG4gICAgICAgICAgICAgICAgICAgIGlmIChvcFswXSA9PT0gMyAmJiAoIXQgfHwgKG9wWzFdID4gdFswXSAmJiBvcFsxXSA8IHRbM10pKSkgeyBfLmxhYmVsID0gb3BbMV07IGJyZWFrOyB9XG4gICAgICAgICAgICAgICAgICAgIGlmIChvcFswXSA9PT0gNiAmJiBfLmxhYmVsIDwgdFsxXSkgeyBfLmxhYmVsID0gdFsxXTsgdCA9IG9wOyBicmVhazsgfVxuICAgICAgICAgICAgICAgICAgICBpZiAodCAmJiBfLmxhYmVsIDwgdFsyXSkgeyBfLmxhYmVsID0gdFsyXTsgXy5vcHMucHVzaChvcCk7IGJyZWFrOyB9XG4gICAgICAgICAgICAgICAgICAgIGlmICh0WzJdKSBfLm9wcy5wb3AoKTtcbiAgICAgICAgICAgICAgICAgICAgXy50cnlzLnBvcCgpOyBjb250aW51ZTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIG9wID0gYm9keS5jYWxsKHRoaXNBcmcsIF8pO1xuICAgICAgICB9IGNhdGNoIChlKSB7IG9wID0gWzYsIGVdOyB5ID0gMDsgfSBmaW5hbGx5IHsgZiA9IHQgPSAwOyB9XG4gICAgICAgIGlmIChvcFswXSAmIDUpIHRocm93IG9wWzFdOyByZXR1cm4geyB2YWx1ZTogb3BbMF0gPyBvcFsxXSA6IHZvaWQgMCwgZG9uZTogdHJ1ZSB9O1xuICAgIH1cbn07XG5pbXBvcnQgeyBmcmVlRGF0YXMySFRNTCB9IGZyb20gXCIuL2ZyZWVEYXRhczJIVE1MXCI7XG52YXIgaW5pdGlhbGlzZSA9IGZ1bmN0aW9uICgpIHsgcmV0dXJuIF9fYXdhaXRlcih2b2lkIDAsIHZvaWQgMCwgdm9pZCAwLCBmdW5jdGlvbiAoKSB7XG4gICAgdmFyIGNvbnZlcnRlciwgZV8xO1xuICAgIHJldHVybiBfX2dlbmVyYXRvcih0aGlzLCBmdW5jdGlvbiAoX2EpIHtcbiAgICAgICAgc3dpdGNoIChfYS5sYWJlbCkge1xuICAgICAgICAgICAgY2FzZSAwOlxuICAgICAgICAgICAgICAgIF9hLnRyeXMucHVzaChbMCwgMiwgLCAzXSk7XG4gICAgICAgICAgICAgICAgY29udmVydGVyID0gbmV3IGZyZWVEYXRhczJIVE1MKCk7XG4gICAgICAgICAgICAgICAgY29udmVydGVyLmRhdGFzVmlld0VsdCA9IHsgaWQ6IFwiZGF0YXNcIiB9O1xuICAgICAgICAgICAgICAgIGNvbnZlcnRlci5kYXRhc1NlbGVjdG9ycyA9IFt7IGRhdGFzRmllbGROYjogMywgaWQ6IFwiZmlsdHJlMVwiIH0sIHsgZGF0YXNGaWVsZE5iOiA0LCBpZDogXCJmaWx0cmUyXCIgfSwgeyBkYXRhc0ZpZWxkTmI6IDUsIGlkOiBcImZpbHRyZTNcIiwgc2VwYXJhdG9yOiBcIixcIiB9XTtcbiAgICAgICAgICAgICAgICBjb252ZXJ0ZXIuZGF0YXNTb3J0aW5nQ29sdW1ucyA9IFt7IGRhdGFzRmllbGROYjogMCB9LCB7IGRhdGFzRmllbGROYjogMSB9LCB7IGRhdGFzRmllbGROYjogMiB9XTtcbiAgICAgICAgICAgICAgICBjb252ZXJ0ZXIuZGF0YXNTb3VyY2VVcmwgPSBcImh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9kYXRhcy9lbGVtZW50cy1jaGltaXF1ZXMuY3N2XCI7XG4gICAgICAgICAgICAgICAgcmV0dXJuIFs0LCBjb252ZXJ0ZXIucnVuKCldO1xuICAgICAgICAgICAgY2FzZSAxOlxuICAgICAgICAgICAgICAgIF9hLnNlbnQoKTtcbiAgICAgICAgICAgICAgICByZXR1cm4gWzMsIDNdO1xuICAgICAgICAgICAgY2FzZSAyOlxuICAgICAgICAgICAgICAgIGVfMSA9IF9hLnNlbnQoKTtcbiAgICAgICAgICAgICAgICBjb25zb2xlLmVycm9yKGVfMSk7XG4gICAgICAgICAgICAgICAgaWYgKGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKFwiZGF0YXNcIikgIT09IG51bGwpXG4gICAgICAgICAgICAgICAgICAgIGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKFwiZGF0YXNcIikuaW5uZXJIVE1MID0gXCI8c3Ryb25nPkTDqXNvbMOpLCBtYWlzIHVuIHByb2Jsw6htZSB0ZWNobmlxdWUgZW1ww6pjaGUgbCdhZmZpY2hhZ2UgZGVzIGRvbm7DqWVzLjwvc3Ryb25nPlwiO1xuICAgICAgICAgICAgICAgIHJldHVybiBbMywgM107XG4gICAgICAgICAgICBjYXNlIDM6IHJldHVybiBbMl07XG4gICAgICAgIH1cbiAgICB9KTtcbn0pOyB9O1xuaW5pdGlhbGlzZSgpO1xuIl0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9 \ No newline at end of file +; \ No newline at end of file