UNPKG

@tensorflow/tfjs-converter

Version:

Tensorflow model converter for javascript

629 lines 97.2 kB
/** * @license * Copyright 2018 Google LLC. All Rights Reserved. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ============================================================================= */ import { env, keep, tidy, util } from '@tensorflow/tfjs-core'; import { getNodeNameAndIndex, getParamValue, getTensor, getTensorsForCurrentContext, parseNodeName } from '../operations/executors/utils'; import { executeOp } from '../operations/operation_executor'; import { ExecutionContext } from './execution_context'; import { getExecutionSubgraph, getNodeLiveUntilMap, getNodesInTopologicalOrder, isControlFlow } from './model_analysis'; export class GraphExecutor { get weightIds() { return this.parent ? this.parent.weightIds : this._weightIds; } get functionExecutorMap() { return this.parent ? this.parent.functionExecutorMap : this._functionExecutorMap; } get weightMap() { return this.parent ? this.parent.weightMap : this._weightMap; } set weightMap(weightMap) { const weightIds = Object.keys(weightMap).map(key => weightMap[key].map(tensor => tensor.id)); this._weightIds = [].concat(...weightIds); this._weightMap = weightMap; } /** * Set `ResourceManager` shared by executors of a model. * @param resourceManager: `ResourceManager` of the `GraphModel`. */ set resourceManager(resourceManager) { this._resourceManager = resourceManager; } get inputs() { return this._inputs.map(node => { return { name: node.name, shape: node.attrParams['shape'] ? node.attrParams['shape'].value : undefined, dtype: node.attrParams['dtype'] ? node.attrParams['dtype'].value : undefined }; }); } get outputs() { return this._outputs.map(node => { return { name: node.name, shape: node.attrParams['shape'] ? node.attrParams['shape'].value : undefined, dtype: node.attrParams['dtype'] ? node.attrParams['dtype'].value : undefined }; }); } get inputNodes() { return this._inputs.map(node => node.signatureKey || node.name); } get outputNodes() { return this._outputs.map((node) => { const name = node.signatureKey || node.name; return node.defaultOutput ? (`${name}:${node.defaultOutput}`) : name; }); } get functions() { return Object.keys(this._functions).reduce((map, key) => { map[key] = this._functions[key].signature; return map; }, {}); } /** * * @param graph Graph the model or function graph to be executed. * @param parent When building function exector you need to set the parent * executor. Since the weights and function executor maps are set at parant * level, that function executor can access the function maps and weight maps * through the parent. */ constructor(graph, parent) { this.graph = graph; this.parent = parent; this.compiledMap = new Map(); this.parseNodeNameCache = new Map(); this._weightMap = {}; this.SEPARATOR = ','; this._functions = {}; this._functionExecutorMap = {}; this.keepIntermediateTensors = false; this._outputs = graph.outputs; this._inputs = graph.inputs; this._initNodes = graph.initNodes; this._signature = graph.signature; this._functions = graph.functions; // create sub-graph executors if (graph.functions != null) { Object.keys(graph.functions).forEach(name => { this._functionExecutorMap[name] = new GraphExecutor(graph.functions[name], this); }); } } getCompilationKey(inputs, outputs) { const sortedInputs = inputs.map(node => node.name).sort(); const sortedOutputs = outputs.map(node => node.name).sort(); return sortedInputs.join(this.SEPARATOR) + '--' + sortedOutputs.join(this.SEPARATOR); } /** * Compiles the inference graph and returns the minimal set of nodes that are * required for execution, in the correct execution order. * @returns {Object} compilation The compile result. * @returns {Node[]} compilation.orderedNodes Nodes in the correct execution * order. * @returns {Map<string, Node[]>} compilation.nodeLiveUntilMap A map from node * to disposable nodes after its execution. That is, for a node `x`, * `nodeLiveUntilMap[x]` indicates all nodes whose intermediate * tensors should be disposed after `x` is executed. */ compile(inputs, outputs) { const executionInfo = getExecutionSubgraph(inputs, outputs, this.weightMap, this._initNodes); const { missingInputs, dynamicNode, syncInputs } = executionInfo; if (dynamicNode != null) { throw new Error(`This execution contains the node '${dynamicNode.name}', which has ` + `the dynamic op '${dynamicNode.op}'. Please use ` + `model.executeAsync() instead. Alternatively, to avoid the ` + `dynamic ops, specify the inputs [${syncInputs}]`); } if (missingInputs.length > 0) { const outNames = outputs.map(n => n.name); const inNames = Object.keys(inputs); throw new Error(`Cannot compute the outputs [${outNames}] from the provided inputs ` + `[${inNames}]. Missing the following inputs: [${missingInputs}]`); } const orderedNodes = getNodesInTopologicalOrder(this.graph, executionInfo); const nodeLiveUntilMap = getNodeLiveUntilMap(orderedNodes); return { orderedNodes, nodeLiveUntilMap }; } cloneAndKeepTensor(tensor) { if (tensor == null) { return null; } const clone = tensor.clone(); // Keep the clone because`model.execute()` may be called within // a `tidy()`, but the user may inspect these tensors after the // tidy. keep(clone); return clone; } cloneTensorList(tensors) { if (!tensors) { return null; } const clonedTensor = tensors.map(tensor => { return this.cloneAndKeepTensor(tensor); }); return clonedTensor; } cloneTensorMap(tensorsMap) { return Object.fromEntries(Object.entries(tensorsMap).map(([name, tensorsList]) => { return [name, this.cloneTensorList(tensorsList)]; })); } /** * Executes the inference for given input tensors. * @param inputs Tensor map for the model inputs, keyed by the input node * names. * @param outputs Optional. output node name from the Tensorflow model, if * no outputs are specified, the default outputs of the model would be used. * You can inspect intermediate nodes of the model by adding them to the * outputs array. */ execute(inputs, outputs) { // Dispose any tensors from a prior run to avoid leaking them. this.disposeIntermediateTensors(); inputs = this.mapInputs(inputs); const names = Object.keys(inputs).sort(); this.checkInputs(inputs); this.checkInputShapeAndType(inputs); outputs = this.mapOutputs(outputs); this.checkOutputs(outputs); const inputNodes = names.map(name => this.graph.nodes[parseNodeName(name)[0]]); const outputNodeNames = outputs.map(name => parseNodeName(name)[0]); const outputNodeNameSet = new Set(outputNodeNames); let outputNodes = outputNodeNames.map(name => this.graph.nodes[name]); // If no outputs are specified, then use the default outputs of the model. if (outputNodes.length === 0) { outputNodes = this._outputs; } const compilationKey = this.getCompilationKey(inputNodes, outputNodes); // Do nothing if the compiled graph cache contains the input. let compilation = this.compiledMap.get(compilationKey); if (compilation == null) { compilation = this.compile(inputs, outputNodes); this.compiledMap.set(compilationKey, compilation); } // Keep tensors if KEEP_INTERMEDIATE_TENSORS is on. try { this.keepIntermediateTensors = env().getBool('KEEP_INTERMEDIATE_TENSORS'); } catch (e) { this.keepIntermediateTensors = false; console.warn(e.message); } const tensorArrayMap = {}; const tensorListMap = {}; return tidy(() => { const context = new ExecutionContext(this.weightMap, tensorArrayMap, tensorListMap, this.functionExecutorMap, this.parseNodeNameCache); const tensorsMap = Object.assign({}, this.weightMap); if (this.keepIntermediateTensors) { this.clonedTensorsMap = this.cloneTensorMap(this.weightMap); } Object.keys(inputs).forEach(name => { const [nodeName, index] = parseNodeName(name, context); const tensors = []; tensors[index] = inputs[name]; tensorsMap[nodeName] = tensors; if (this.keepIntermediateTensors) { this.clonedTensorsMap[nodeName] = this.cloneTensorList(tensors); } }); const tensorsToKeep = this.getFrozenTensorIds(tensorsMap); const { orderedNodes, nodeLiveUntilMap } = compilation; for (const node of orderedNodes) { if (tensorsMap[node.name]) { continue; } const tensors = executeOp(node, tensorsMap, context, this._resourceManager); if (util.isPromise(tensors)) { throw new Error(`The execution of the op '${node.op}' returned a promise. ` + `Please use model.executeAsync() instead.`); } tensorsMap[node.name] = tensors; if (this.keepIntermediateTensors) { this.clonedTensorsMap[node.name] = this.cloneTensorList(tensors); } this.checkTensorForDisposalWithNodeLiveUntilInfo(node, tensorsMap, context, tensorsToKeep, outputNodeNameSet, nodeLiveUntilMap.get(node.name)); } // dispose the context for the root executor if (this.parent == null) { context.dispose(tensorsToKeep); } return outputs.map(name => getTensor(name, tensorsMap, context)); }); } getFrozenTensorIds(tensorMap) { const ids = [].concat.apply([], Object.keys(tensorMap) .map(key => tensorMap[key]) .map(tensors => tensors.map(tensor => tensor.id))); return new Set(ids); } checkTensorForDisposal(nodeName, node, tensorMap, context, tensorsToKeep, outputNodeNameSet, intermediateTensorConsumerCount) { // Skip output nodes and any control flow nodes, since its dependency is // tricky to track correctly. if (isControlFlow(node) || outputNodeNameSet.has(nodeName)) { return; } for (const tensor of tensorMap[nodeName]) { if (tensor == null) { continue; } intermediateTensorConsumerCount[tensor.id] = (intermediateTensorConsumerCount[tensor.id] || 0) + node.children.length; } for (const input of node.inputs) { // Skip any control flow nodes, since its dependency is tricky to track // correctly. if (isControlFlow(input)) { continue; } const tensors = getTensorsForCurrentContext(input.name, tensorMap, context); if (tensors == null) { continue; } for (const tensor of tensors) { if (!tensor || tensor.kept || tensorsToKeep.has(tensor.id)) { continue; } // Only intermediate nodes' tensors have counts set, not marked as // kept, and not in `tensorsToKeep`. // Input and weight nodes' tensors should exist in `tensorsToKeep`. // Output and control flow nodes' tensors should never have count set. const count = intermediateTensorConsumerCount[tensor.id]; if (count === 1) { tensor.dispose(); delete intermediateTensorConsumerCount[tensor.id]; } else if (count != null) { intermediateTensorConsumerCount[tensor.id]--; } } } } checkTensorForDisposalWithNodeLiveUntilInfo(node, tensorMap, context, tensorsToKeep, outputNodeNameSet, liveUntilNodes) { function isNonDisposableNode(node) { // Skip output nodes and any control flow nodes, since its dependency is // tricky to track correctly. return isControlFlow(node) || outputNodeNameSet.has(node.name); } if (isControlFlow(node) || liveUntilNodes == null) { return; } for (const nodeToDispose of liveUntilNodes) { if (isNonDisposableNode(nodeToDispose)) { continue; } const tensors = getTensorsForCurrentContext(nodeToDispose.name, tensorMap, context); for (const tensor of tensors) { if (!tensor || tensor.kept || tensorsToKeep.has(tensor.id)) { continue; } tensor.dispose(); } } } /** * Executes the inference for given input tensors in Async fashion. * @param inputs Tensor map for the model inputs, keyed by the input node * names. * @param outputs output node name from the Tensorflow model, if no outputs * are specified, the default outputs of the model would be used. You can * inspect intermediate nodes of the model by adding them to the outputs * array. */ async executeAsync(inputs, outputs) { return this._executeAsync(inputs, outputs); } disposeIntermediateTensors() { if (!this.clonedTensorsMap) { return; } Object.values(this.clonedTensorsMap).forEach(tensorsList => { for (const tensor of tensorsList) { if (tensor && !tensor.isDisposed) { tensor.dispose(); } } }); this.clonedTensorsMap = null; } getIntermediateTensors() { return this.clonedTensorsMap; } /** * Executes the inference for given input tensors in Async fashion. * @param inputs Tensor map for the model inputs, keyed by the input node * names. * @param outputs Optional. output node name from the Tensorflow model, * if no outputs are specified, the default outputs of the model would be * used. You can inspect intermediate nodes of the model by adding them to * the outputs array. * @param isFunctionExecution Optional. Flag for executing a function. * @param tensorArrayMap Optional, global TensorArray map by id. Used for * function execution. * @param tensorArrayMap Optional global TensorList map by id. Used for * function execution. */ async _executeAsync(inputs, outputs, isFunctionExecution = false, tensorArrayMap = {}, tensorListMap = {}) { // Dispose any tensors from a prior run to avoid leaking them. this.disposeIntermediateTensors(); if (!isFunctionExecution) { inputs = this.mapInputs(inputs); this.checkInputs(inputs); this.checkInputShapeAndType(inputs); outputs = this.mapOutputs(outputs); this.checkOutputs(outputs); } // Keep tensors if KEEP_INTERMEDIATE_TENSORS is on. try { this.keepIntermediateTensors = env().getBool('KEEP_INTERMEDIATE_TENSORS'); } catch (e) { this.keepIntermediateTensors = false; console.warn(e.message); } const context = new ExecutionContext(this.weightMap, tensorArrayMap, tensorListMap, this.functionExecutorMap, this.parseNodeNameCache); if (this.keepIntermediateTensors) { this.clonedTensorsMap = this.cloneTensorMap(this.weightMap); } // Graph with control flow op requires runtime evaluation of the execution // order, while without control flow the execution order is pre-determined // in the compile method. const tensorsMap = await this.executeWithControlFlow(inputs, context, outputs, isFunctionExecution); const results = outputs.map(name => getTensor(name, tensorsMap, context)); // dispose all the intermediate tensors const outputIds = results.map(t => t.id); const inputIds = Object.keys(inputs).map(name => inputs[name].id); const keepIds = new Set([...outputIds, ...inputIds, ...this.weightIds]); Object.values(tensorsMap).forEach(tensorsList => { tensorsList.forEach(tensor => { if (tensor && !tensor.isDisposed && !keepIds.has(tensor.id)) { tensor.dispose(); } }); }); // dispose the context for the root executor if (this.parent == null) { context.dispose(keepIds); } return results; } async executeFunctionAsync(inputs, tensorArrayMap, tensorListMap) { const mappedInputs = inputs.reduce((map, tensor, index) => { map[this.inputs[index].name] = tensor; return map; }, {}); return this._executeAsync(mappedInputs, this.outputNodes, true, tensorArrayMap, tensorListMap); } /** * When there are control flow nodes in the graph, the graph execution use * ExecutionContext to keep track of the frames and loop iterators. * @param inputs placeholder tensors for the graph. * @param context the execution context object for current execution. * @param outputNames Optional. output node name from the Tensorflow model, * if no outputs are specified, the default outputs of the model would be * used. You can inspect intermediate nodes of the model by adding them to * the outputs array. * @param isFunctionExecution Flag for executing a function. */ async executeWithControlFlow(inputs, context, outputNames, isFunctionExecution) { const names = Object.keys(inputs); const inputNodes = names.map(name => this.graph.nodes[parseNodeName(name)[0]]); const outputNodeNames = outputNames.map(name => parseNodeName(name)[0]); const outputNodeNameSet = new Set(outputNodeNames); let outputNodes = outputNodeNames.map(name => this.graph.nodes[name]); // If no outputs are specified, then use the default outputs of the model. if (outputNodes.length === 0) { outputNodes = this._outputs; } const { usedNodes, missingInputs, dynamicNode, syncInputs } = getExecutionSubgraph(inputs, outputNodes, this.weightMap, this._initNodes); // First nodes to execute include inputNodes, weights, and initNodes. const stack = [ ...inputNodes, ...this.graph.weights, ...(this._initNodes || []) ].map(node => { return { node, contexts: context.currentContext }; }); const tensorsMap = Object.assign({}, this.weightMap); Object.keys(inputs).forEach(name => { const [nodeName, index] = parseNodeName(name); const tensors = []; tensors[index] = inputs[name]; tensorsMap[nodeName] = tensors; }); const intermediateTensorConsumerCount = {}; const tensorsToKeep = this.getFrozenTensorIds(tensorsMap); const added = {}; while (stack.length > 0) { const promises = this.processStack(inputNodes, stack, context, tensorsMap, added, tensorsToKeep, outputNodeNameSet, intermediateTensorConsumerCount, usedNodes); await Promise.all(promises); } if (dynamicNode == null && !isFunctionExecution) { console.warn(`This model execution did not contain any nodes with control flow ` + `or dynamic output shapes. You can use model.execute() instead.`); } const missingOutputs = outputNodes .filter(node => !isControlFlow(node) && !getTensor(node.name, tensorsMap, context)) .map(node => node.name); if (missingOutputs.length > 0) { let alternativeMsg = ''; if (dynamicNode != null) { alternativeMsg = `Alternatively, to avoid the dynamic ops, use model.execute() ` + `and specify the inputs [${syncInputs}]`; } throw new Error(`Cannot compute the outputs [${missingOutputs}] from the provided ` + `inputs [${names}]. Consider providing the following inputs: ` + `[${missingInputs}]. ${alternativeMsg}`); } return tensorsMap; } processStack(inputNodes, stack, context, tensorMap, added, tensorsToKeep, outputNodeNameSet, intermediateTensorConsumerCount, usedNodes) { const promises = []; while (stack.length > 0) { const item = stack.pop(); context.currentContext = item.contexts; let nodeName = ''; // The tensor of the Enter op with isConstant set should be set // in the parent scope, so it will be available as constant for the // whole loop. if (item.node.op === 'Enter' && getParamValue('isConstant', item.node, tensorMap, context)) { [nodeName] = getNodeNameAndIndex(item.node.name, context); } // only process nodes that are not in the tensorMap yet, this include // inputNodes and internal initNodes. if (tensorMap[item.node.name] == null) { const tensors = executeOp(item.node, tensorMap, context, this._resourceManager); if (!nodeName) { [nodeName] = getNodeNameAndIndex(item.node.name, context); } const currentContext = context.currentContext; if (util.isPromise(tensors)) { promises.push(tensors.then(t => { tensorMap[nodeName] = t; if (this.keepIntermediateTensors) { this.clonedTensorsMap[nodeName] = this.cloneTensorList(t); } context.currentContext = currentContext; this.checkTensorForDisposal(nodeName, item.node, tensorMap, context, tensorsToKeep, outputNodeNameSet, intermediateTensorConsumerCount); this.processChildNodes(item.node, stack, context, tensorMap, added, usedNodes); return t; })); } else { tensorMap[nodeName] = tensors; if (this.keepIntermediateTensors) { this.clonedTensorsMap[nodeName] = this.cloneTensorList(tensors); } this.checkTensorForDisposal(nodeName, item.node, tensorMap, context, tensorsToKeep, outputNodeNameSet, intermediateTensorConsumerCount); this.processChildNodes(item.node, stack, context, tensorMap, added, usedNodes); } } else { this.processChildNodes(item.node, stack, context, tensorMap, added, usedNodes); } } return promises; } processChildNodes(node, stack, context, tensorMap, added, usedNodes) { node.children.forEach((childNode) => { const [nodeName,] = getNodeNameAndIndex(childNode.name, context); if (added[nodeName] || !usedNodes.has(childNode.name)) { return; } // Merge op can be pushed if any of its inputs has value. if (childNode.op === 'Merge') { if (childNode.inputNames.some(name => { return !!getTensor(name, tensorMap, context); })) { added[nodeName] = true; stack.push({ contexts: context.currentContext, node: childNode }); } } else // Otherwise all inputs must to have value. if (childNode.inputNames.every(name => { return !!getTensor(name, tensorMap, context); })) { added[nodeName] = true; stack.push({ contexts: context.currentContext, node: childNode }); } }); } /** * Releases the memory used by the weight tensors. */ dispose() { Object.keys(this.weightMap) .forEach(key => this.weightMap[key].forEach(tensor => tensor.dispose())); } checkInputShapeAndType(inputs) { Object.keys(inputs).forEach(name => { const input = inputs[name]; const [nodeName,] = parseNodeName(name); const node = this.graph.nodes[nodeName]; if (node.attrParams['shape'] && node.attrParams['shape'].value) { const shape = node.attrParams['shape'].value; const match = shape.length === input.shape.length && input.shape.every((dim, index) => shape[index] === -1 || shape[index] === dim); util.assert(match, () => `The shape of dict['${node.name}'] provided in ` + `model.execute(dict) must be [${shape}], but was ` + `[${input.shape}]`); } if (node.attrParams['dtype'] && node.attrParams['dtype'].value) { util.assert(input.dtype === node.attrParams['dtype'].value, () => `The dtype of dict['${node.name}'] provided in ` + `model.execute(dict) must be ` + `${node.attrParams['dtype'].value}, but was ${input.dtype}`); } }); } mapInputs(inputs) { var _a, _b; const result = {}; for (const inputName in inputs) { const tensor = (_b = (_a = this._signature) === null || _a === void 0 ? void 0 : _a.inputs) === null || _b === void 0 ? void 0 : _b[inputName]; if (tensor != null) { result[tensor.name] = inputs[inputName]; } else { result[inputName] = inputs[inputName]; } } return result; } checkInputs(inputs) { const notInGraph = Object.keys(inputs).filter(name => { const [nodeName] = parseNodeName(name); return this.graph.nodes[nodeName] == null; }); if (notInGraph.length > 0) { throw new Error(`The dict provided in model.execute(dict) has ` + `keys: [${notInGraph}] that are not part of graph`); } } mapOutputs(outputs) { return outputs.map(name => { var _a, _b; const tensor = (_b = (_a = this._signature) === null || _a === void 0 ? void 0 : _a.outputs) === null || _b === void 0 ? void 0 : _b[name]; if (tensor != null) { return tensor.name; } return name; }, {}); } checkOutputs(outputs) { outputs.forEach(name => { const [normalizedName] = parseNodeName(name); if (!this.graph.nodes[normalizedName]) { throw new Error(`The output '${name}' is not found in the graph`); } }); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"graph_executor.js","sourceRoot":"","sources":["../../../../../../tfjs-converter/src/executor/graph_executor.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;GAeG;AAEH,OAAO,EAAW,GAAG,EAAE,IAAI,EAA0B,IAAI,EAAE,IAAI,EAAC,MAAM,uBAAuB,CAAC;AAI9F,OAAO,EAAC,mBAAmB,EAAE,aAAa,EAAE,SAAS,EAAE,2BAA2B,EAAE,aAAa,EAAC,MAAM,+BAA+B,CAAC;AACxI,OAAO,EAAC,SAAS,EAAC,MAAM,kCAAkC,CAAC;AAG3D,OAAO,EAAC,gBAAgB,EAAuB,MAAM,qBAAqB,CAAC;AAC3E,OAAO,EAAC,oBAAoB,EAAE,mBAAmB,EAAE,0BAA0B,EAAE,aAAa,EAAC,MAAM,kBAAkB,CAAC;AAStH,MAAM,OAAO,aAAa;IAgBxB,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC;IAC/D,CAAC;IAED,IAAI,mBAAmB;QACrB,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC;YACjC,IAAI,CAAC,oBAAoB,CAAC;IACjD,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC;IAC/D,CAAC;IAED,IAAI,SAAS,CAAC,SAA0B;QACtC,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,GAAG,CACxC,GAAG,CAAC,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QACpD,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC,MAAM,CAAC,GAAG,SAAS,CAAC,CAAC;QAC1C,IAAI,CAAC,UAAU,GAAG,SAAS,CAAC;IAC9B,CAAC;IAED;;;OAGG;IACH,IAAI,eAAe,CAAC,eAAgC;QAClD,IAAI,CAAC,gBAAgB,GAAG,eAAe,CAAC;IAC1C,CAAC;IAED,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE;YAC7B,OAAO;gBACL,IAAI,EAAE,IAAI,CAAC,IAAI;gBACf,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC;oBAC7B,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,KAAiB,CAAC,CAAC;oBAC5C,SAAS;gBACb,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC;oBAC7B,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,KAAiB,CAAC,CAAC;oBAC5C,SAAS;aACd,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE;YAC9B,OAAO;gBACL,IAAI,EAAE,IAAI,CAAC,IAAI;gBACf,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC;oBAC7B,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,KAAiB,CAAC,CAAC;oBAC5C,SAAS;gBACb,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC;oBAC7B,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC,KAAiB,CAAC,CAAC;oBAC5C,SAAS;aACd,CAAC;QACJ,CAAC,CAAC,CAAC;IACL,CAAC;IAED,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC;IAClE,CAAC;IAED,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;YAChC,MAAM,IAAI,GAAG,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,IAAI,CAAC;YAC5C,OAAO,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QACvE,CAAC,CAAC,CAAC;IACL,CAAC;IAED,IAAI,SAAS;QACX,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;YACtD,GAAG,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC;YAC1C,OAAO,GAAG,CAAC;QACb,CAAC,EAAE,EAAoC,CAAC,CAAC;IAC3C,CAAC;IAED;;;;;;;OAOG;IACH,YAAoB,KAAY,EAAU,MAAsB;QAA5C,UAAK,GAAL,KAAK,CAAO;QAAU,WAAM,GAAN,MAAM,CAAgB;QAjGxD,gBAAW,GAAG,IAAI,GAAG,EAA2C,CAAC;QACjE,uBAAkB,GAAG,IAAI,GAAG,EAAqC,CAAC;QAClE,eAAU,GAAoB,EAAE,CAAC;QAMjC,cAAS,GAAG,GAAG,CAAC;QAChB,eAAU,GAA2B,EAAE,CAAC;QACxC,yBAAoB,GAAsC,EAAE,CAAC;QAG7D,4BAAuB,GAAG,KAAK,CAAC;QAqFtC,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC,OAAO,CAAC;QAC9B,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC,MAAM,CAAC;QAC5B,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,SAAS,CAAC;QAClC,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,SAAS,CAAC;QAClC,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,SAAS,CAAC;QAClC,6BAA6B;QAC7B,IAAI,KAAK,CAAC,SAAS,IAAI,IAAI,EAAE;YAC3B,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;gBAC1C,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC;oBAC3B,IAAI,aAAa,CAAC,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,IAAI,CAAC,CAAC;YACrD,CAAC,CAAC,CAAC;SACJ;IACH,CAAC;IAEO,iBAAiB,CAAC,MAAc,EAAE,OAAe;QACvD,MAAM,YAAY,GAAG,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC;QAC1D,MAAM,aAAa,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC;QAC5D,OAAO,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,IAAI;YAC3C,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACzC,CAAC;IAED;;;;;;;;;;OAUG;IACK,OAAO,CAAC,MAAsB,EAAE,OAAe;QAErD,MAAM,aAAa,GACf,oBAAoB,CAAC,MAAM,EAAE,OAAO,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;QAC3E,MAAM,EAAC,aAAa,EAAE,WAAW,EAAE,UAAU,EAAC,GAAG,aAAa,CAAC;QAC/D,IAAI,WAAW,IAAI,IAAI,EAAE;YACvB,MAAM,IAAI,KAAK,CACX,qCAAqC,WAAW,CAAC,IAAI,eAAe;gBACpE,mBAAmB,WAAW,CAAC,EAAE,gBAAgB;gBACjD,4DAA4D;gBAC5D,oCAAoC,UAAU,GAAG,CAAC,CAAC;SACxD;QAED,IAAI,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE;YAC5B,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YAC1C,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YACpC,MAAM,IAAI,KAAK,CACX,+BAA+B,QAAQ,6BAA6B;gBACpE,IAAI,OAAO,qCAAqC,aAAa,GAAG,CAAC,CAAC;SACvE;QAED,MAAM,YAAY,GAAG,0BAA0B,CAAC,IAAI,CAAC,KAAK,EAAE,aAAa,CAAC,CAAC;QAC3E,MAAM,gBAAgB,GAAG,mBAAmB,CAAC,YAAY,CAAC,CAAC;QAC3D,OAAO,EAAC,YAAY,EAAE,gBAAgB,EAAC,CAAC;IAC1C,CAAC;IAEO,kBAAkB,CAAC,MAAc;QACvC,IAAI,MAAM,IAAI,IAAI,EAAE;YAClB,OAAO,IAAI,CAAC;SACb;QACD,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,EAAE,CAAC;QAC7B,+DAA+D;QAC/D,+DAA+D;QAC/D,QAAQ;QACR,IAAI,CAAC,KAAK,CAAC,CAAC;QACZ,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,eAAe,CAAC,OAAiB;QACvC,IAAI,CAAC,OAAO,EAAE;YACZ,OAAO,IAAI,CAAC;SACb;QACD,MAAM,YAAY,GAAG,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE;YACxC,OAAO,IAAI,CAAC,kBAAkB,CAAC,MAAM,CAAC,CAAC;QACzC,CAAC,CAAC,CAAC;QACH,OAAO,YAAY,CAAC;IACtB,CAAC;IAEO,cAAc,CAAC,UAA2B;QAChD,OAAO,MAAM,CAAC,WAAW,CACrB,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,EAAE,WAAW,CAAC,EAAE,EAAE;YACrD,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,eAAe,CAAC,WAAW,CAAC,CAAC,CAAC;QACnD,CAAC,CAAC,CAAC,CAAC;IACV,CAAC;IAED;;;;;;;;OAQG;IACH,OAAO,CAAC,MAAsB,EAAE,OAAkB;QAChD,8DAA8D;QAC9D,IAAI,CAAC,0BAA0B,EAAE,CAAC;QAClC,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;QAChC,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,CAAC;QACzC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;QACzB,IAAI,CAAC,sBAAsB,CAAC,MAAM,CAAC,CAAC;QACpC,OAAO,GAAG,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QACnC,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;QAC3B,MAAM,UAAU,GACZ,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAChE,MAAM,eAAe,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACpE,MAAM,iBAAiB,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC,CAAC;QACnD,IAAI,WAAW,GAAG,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;QACtE,0EAA0E;QAC1E,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE;YAC5B,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC;SAC7B;QAED,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;QAEvE,6DAA6D;QAC7D,IAAI,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;QACvD,IAAI,WAAW,IAAI,IAAI,EAAE;YACvB,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;YAChD,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,cAAc,EAAE,WAAW,CAAC,CAAC;SACnD;QAED,mDAAmD;QACnD,IAAI;YACF,IAAI,CAAC,uBAAuB,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,2BAA2B,CAAC,CAAC;SAC3E;QAAC,OAAO,CAAC,EAAE;YACV,IAAI,CAAC,uBAAuB,GAAG,KAAK,CAAC;YACrC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;SACzB;QACD,MAAM,cAAc,GAAmB,EAAE,CAAC;QAC1C,MAAM,aAAa,GAAkB,EAAE,CAAC;QAExC,OAAO,IAAI,CAAC,GAAG,EAAE;YACf,MAAM,OAAO,GAAG,IAAI,gBAAgB,CAChC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,aAAa,EAC7C,IAAI,CAAC,mBAAmB,EAAE,IAAI,CAAC,kBAAkB,CAAC,CAAC;YACvD,MAAM,UAAU,qBAAwB,IAAI,CAAC,SAAS,CAAC,CAAC;YACxD,IAAI,IAAI,CAAC,uBAAuB,EAAE;gBAChC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;aAC7D;YAED,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;gBACjC,MAAM,CAAC,QAAQ,EAAE,KAAK,CAAC,GAAG,aAAa,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;gBACvD,MAAM,OAAO,GAAa,EAAE,CAAC;gBAC7B,OAAO,CAAC,KAAK,CAAC,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;gBAC9B,UAAU,CAAC,QAAQ,CAAC,GAAG,OAAO,CAAC;gBAC/B,IAAI,IAAI,CAAC,uBAAuB,EAAE;oBAChC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC;iBACjE;YACH,CAAC,CAAC,CAAC;YAEH,MAAM,aAAa,GAAG,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;YAC1D,MAAM,EAAC,YAAY,EAAE,gBAAgB,EAAC,GAAG,WAAW,CAAC;YACrD,KAAK,MAAM,IAAI,IAAI,YAAY,EAAE;gBAC/B,IAAI,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;oBACzB,SAAS;iBACV;gBACD,MAAM,OAAO,GACT,SAAS,CAAC,IAAI,EAAE,UAAU,EAAE,OAAO,EAAE,IAAI,CAAC,gBAAgB,CAClD,CAAC;gBACb,IAAI,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,EAAE;oBAC3B,MAAM,IAAI,KAAK,CACX,4BAA4B,IAAI,CAAC,EAAE,wBAAwB;wBAC3D,0CAA0C,CAAC,CAAC;iBACjD;gBACD,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,OAAO,CAAC;gBAChC,IAAI,IAAI,CAAC,uBAAuB,EAAE;oBAChC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC;iBAClE;gBACD,IAAI,CAAC,2CAA2C,CAC5C,IAAI,EAAE,UAAU,EAAE,OAAO,EAAE,aAAa,EAAE,iBAAiB,EAC3D,gBAAgB,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;aACtC;YAED,4CAA4C;YAC5C,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,EAAE;gBACvB,OAAO,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;aAChC;YAED,OAAO,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,IAAI,EAAE,UAAU,EAAE,OAAO,CAAC,CAAC,CAAC;QACnE,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,kBAAkB,CAAC,SAA0B;QACnD,MAAM,GAAG,GAAG,EAAE,CAAC,MAAM,CAAC,KAAK,CACvB,EAAE,EACF,MAAM,CAAC,IAAI,CAAC,SAAS,CAAC;aACjB,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;aAC1B,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QAC3D,OAAO,IAAI,GAAG,CAAC,GAAG,CAAC,CAAC;IACtB,CAAC;IAEO,sBAAsB,CAC1B,QAAgB,EAAE,IAAU,EAAE,SAA0B,EACxD,OAAyB,EAAE,aAA0B,EACrD,iBAA8B,EAC9B,+BAAwD;QAC1D,wEAAwE;QACxE,6BAA6B;QAC7B,IAAI,aAAa,CAAC,IAAI,CAAC,IAAI,iBAAiB,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE;YAC1D,OAAO;SACR;QAED,KAAK,MAAM,MAAM,IAAI,SAAS,CAAC,QAAQ,CAAC,EAAE;YACxC,IAAI,MAAM,IAAI,IAAI,EAAE;gBAClB,SAAS;aACV;YACD,+BAA+B,CAAC,MAAM,CAAC,EAAE,CAAC;gBACtC,CAAC,+BAA+B,CAAC,MAAM,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;oBACjD,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC;SAC1B;QAED,KAAK,MAAM,KAAK,IAAI,IAAI,CAAC,MAAM,EAAE;YAC/B,uEAAuE;YACvE,aAAa;YACb,IAAI,aAAa,CAAC,KAAK,CAAC,EAAE;gBACxB,SAAS;aACV;YAED,MAAM,OAAO,GACT,2BAA2B,CAAC,KAAK,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;YAChE,IAAI,OAAO,IAAI,IAAI,EAAE;gBACnB,SAAS;aACV;YAED,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE;gBAC5B,IAAI,CAAC,MAAM,IAAI,MAAM,CAAC,IAAI,IAAI,aAAa,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE;oBAC1D,SAAS;iBACV;gBAED,kEAAkE;gBAClE,oCAAoC;gBACpC,mEAAmE;gBACnE,sEAAsE;gBACtE,MAAM,KAAK,GAAG,+BAA+B,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;gBACzD,IAAI,KAAK,KAAK,CAAC,EAAE;oBACf,MAAM,CAAC,OAAO,EAAE,CAAC;oBACjB,OAAO,+BAA+B,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;iBACnD;qBAAM,IAAI,KAAK,IAAI,IAAI,EAAE;oBACxB,+BAA+B,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC;iBAC9C;aACF;SACF;IACH,CAAC;IAEO,2CAA2C,CAC/C,IAAU,EAAE,SAA0B,EAAE,OAAyB,EACjE,aAA0B,EAAE,iBAA8B,EAC1D,cAAuB;QACzB,SAAS,mBAAmB,CAAC,IAAU;YACrC,wEAAwE;YACxE,6BAA6B;YAC7B,OAAO,aAAa,CAAC,IAAI,CAAC,IAAI,iBAAiB,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACjE,CAAC;QAED,IAAI,aAAa,CAAC,IAAI,CAAC,IAAI,cAAc,IAAI,IAAI,EAAE;YACjD,OAAO;SACR;QAED,KAAK,MAAM,aAAa,IAAI,cAAc,EAAE;YAC1C,IAAI,mBAAmB,CAAC,aAAa,CAAC,EAAE;gBACtC,SAAS;aACV;YACD,MAAM,OAAO,GAAG,2BAA2B,CACvC,aAAa,CAAC,IAAI,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;YAC5C,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE;gBAC5B,IAAI,CAAC,MAAM,IAAI,MAAM,CAAC,IAAI,IAAI,aAAa,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE;oBAC1D,SAAS;iBACV;gBACD,MAAM,CAAC,OAAO,EAAE,CAAC;aAClB;SACF;IACH,CAAC;IAED;;;;;;;;OAQG;IACH,KAAK,CAAC,YAAY,CAAC,MAAsB,EAAE,OAAkB;QAE3D,OAAO,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAC7C,CAAC;IAED,0BAA0B;QACxB,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE;YAC1B,OAAO;SACR;QACD,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC,EAAE;YACzD,KAAK,MAAM,MAAM,IAAI,WAAW,EAAE;gBAChC,IAAI,MAAM,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE;oBAChC,MAAM,CAAC,OAAO,EAAE,CAAC;iBAClB;aACF;QACH,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;IAC/B,CAAC;IAED,sBAAsB;QACpB,OAAO,IAAI,CAAC,gBAAgB,CAAC;IAC/B,CAAC;IAED;;;;;;;;;;;;;OAaG;IACK,KAAK,CAAC,aAAa,CACvB,MAAsB,EAAE,OAAkB,EAAE,mBAAmB,GAAG,KAAK,EACvE,iBAAiC,EAAE,EACnC,gBAA+B,EAAE;QACnC,8DAA8D;QAC9D,IAAI,CAAC,0BAA0B,EAAE,CAAC;QAClC,IAAI,CAAC,mBAAmB,EAAE;YACxB,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;YAChC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;YACzB,IAAI,CAAC,sBAAsB,CAAC,MAAM,CAAC,CAAC;YACpC,OAAO,GAAG,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;YACnC,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;SAC5B;QAED,mDAAmD;QACnD,IAAI;YACF,IAAI,CAAC,uBAAuB,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,2BAA2B,CAAC,CAAC;SAC3E;QAAC,OAAO,CAAC,EAAE;YACV,IAAI,CAAC,uBAAuB,GAAG,KAAK,CAAC;YACrC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;SACzB;QAED,MAAM,OAAO,GAAG,IAAI,gBAAgB,CAChC,IAAI,CAAC,SAAS,EAAE,cAAc,EAAE,aAAa,EAAE,IAAI,CAAC,mBAAmB,EACvE,IAAI,CAAC,kBAAkB,CAAC,CAAC;QAE7B,IAAI,IAAI,CAAC,uBAAuB,EAAE;YAChC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;SAC7D;QAED,0EAA0E;QAC1E,0EAA0E;QAC1E,yBAAyB;QACzB,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,sBAAsB,CAChD,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,mBAAmB,CAAC,CAAC;QACnD,MAAM,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,IAAI,EAAE,UAAU,EAAE,OAAO,CAAC,CAAC,CAAC;QAE1E,uCAAuC;QACvC,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;QACzC,MAAM,QAAQ,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC,CAAC;QAClE,MAAM,OAAO,GACT,IAAI,GAAG,CAAS,CAAC,GAAG,SAAS,EAAE,GAAG,QAAQ,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC;QAEpE,MAAM,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,OAAO,CAAC,WAAW,CAAC,EAAE;YAC9C,WAAW,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;gBAC3B,IAAI,MAAM,IAAI,CAAC,MAAM,CAAC,UAAU,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE;oBAC3D,MAAM,CAAC,OAAO,EAAE,CAAC;iBAClB;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,4CAA4C;QAC5C,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,EAAE;YACvB,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;SAC1B;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,KAAK,CAAC,oBAAoB,CACtB,MAAgB,EAAE,cAA8B,EAChD,aAA4B;QAC9B,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,EAAE;YACxD,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC;YACtC,OAAO,GAAG,CAAC;QACb,CAAC,EAAE,EAAoB,CAAC,CAAC;QAEzB,OAAO,IAAI,CAAC,aAAa,CACrB,YAAY,EAAE,IAAI,CAAC,WAAW,EAAE,IAAI,EAAE,cAAc,EAAE,aAAa,CAAC,CAAC;IAC3E,CAAC;IAED;;;;;;;;;;OAUG;IACK,KAAK,CAAC,sBAAsB,CAChC,MAAsB,EAAE,OAAyB,EAAE,WAAsB,EACzE,mBAA6B;QAC/B,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAClC,MAAM,UAAU,GACZ,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAChE,MAAM,eAAe,GAAG,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACxE,MAAM,iBAAiB,GAAG,IAAI,GAAG,CAAC,eAAe,CAAC,CAAC;QACnD,IAAI,WAAW,GAAG,eAAe,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;QAEtE,0EAA0E;QAC1E,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE;YAC5B,WAAW,GAAG,IAAI,CAAC,QAAQ,CAAC;SAC7B;QAED,MAAM,EAAC,SAAS,EAAE,aAAa,EAAE,WAAW,EAAE,UAAU,EAAC,GACrD,oBAAoB,CAChB,MAAM,EAAE,WAAW,EAAE,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;QAE9D,qEAAqE;QACrE,MAAM,KAAK,GAAuB;YAChC,GAAG,UAAU,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,CAAC,IAAI,CAAC,UAAU,IAAI,EAAE,CAAC;SACjE,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE;YACX,OAAO,EAAC,IAAI,EAAE,QAAQ,EAAE,OAAO,CAAC,cAAc,EAAC,CAAC;QAClD,CAAC,CAAC,CAAC;QACH,MAAM,UAAU,qBAAwB,IAAI,CAAC,SAAS,CAAC,CAAC;QACxD,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;YACjC,MAAM,CAAC,QAAQ,EAAE,KAAK,CAAC,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC;YAC9C,MAAM,OAAO,GAAa,EAAE,CAAC;YAC7B,OAAO,CAAC,KAAK,CAAC,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;YAC9B,UAAU,CAAC,QAAQ,CAAC,GAAG,OAAO,CAAC;QACjC,CAAC,CAAC,CAAC;QACH,MAAM,+BAA+B,GAA4B,EAAE,CAAC;QACpE,MAAM,aAAa,GAAG,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QAC1D,MAAM,KAAK,GAA6B,EAAE,CAAC;QAC3C,OAAO,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE;YACvB,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,CAC9B,UAAU,EAAE,KAAK,EAAE,OAAO,EAAE,UAAU,EAAE,KAAK,EAAE,aAAa,EAC5D,iBAAiB,EAAE,+BAA+B,EAAE,SAAS,CAAC,CAAC;YACnE,MAAM,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;SAC7B;QACD,IAAI,WAAW,IAAI,IAAI,IAAI,CAAC,mBAAmB,EAAE;YAC/C,OAAO,CAAC,IAAI,CACR,mEAAmE;gBACnE,gEAAgE,CAAC,CAAC;SACvE;QACD,MAAM,cAAc,GAChB,WAAW;aACN,MAAM,CACH,IAAI,CAAC,EAAE,CAAC,CAAC,aAAa,CAAC,IAAI,CAAC;YACxB,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,EAAE,UAAU,EAAE,OAAO,CAAC,CAAC;aAClD,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAChC,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE;YAC7B,IAAI,cAAc,GAAG,EAAE,CAAC;YACxB,IAAI,WAAW,IAAI,IAAI,EAAE;gBACvB,cAAc;oBACV,+DAA+D;wBAC/D,2BAA2B,UAAU,GAAG,CAAC;aAC9C;YACD,MAAM,IAAI,KAAK,CACX,+BAA+B,cAAc,sBAAsB;gBACnE,WAAW,KAAK,8CAA8C;gBAC9