import * as crypto from "crypto"; import * as fs from "fs"; import * as http from "http"; import * as https from "https"; import * as path from "path"; import * as stream from "stream"; import * as async from 'async'; import BlockStream2 from 'block-stream2'; import { isBrowser } from 'browser-or-node'; import _ from 'lodash'; import * as qs from 'query-string'; import xml2js from 'xml2js'; import { CredentialProvider } from "../CredentialProvider.mjs"; import * as errors from "../errors.mjs"; import { CopyDestinationOptions, CopySourceOptions, DEFAULT_REGION, LEGAL_HOLD_STATUS, PRESIGN_EXPIRY_DAYS_MAX, RETENTION_MODES, RETENTION_VALIDITY_UNITS } from "../helpers.mjs"; import { NotificationConfig, NotificationPoller } from "../notification.mjs"; import { postPresignSignatureV4, presignSignatureV4, signV4 } from "../signing.mjs"; import { fsp, streamPromise } from "./async.mjs"; import { CopyConditions } from "./copy-conditions.mjs"; import { Extensions } from "./extensions.mjs"; import { calculateEvenSplits, extractMetadata, getContentLength, getScope, getSourceVersionId, getVersionId, hashBinary, insertContentType, isAmazonEndpoint, isBoolean, isDefined, isEmpty, isNumber, isObject, isPlainObject, isReadableStream, isString, isValidBucketName, isValidEndpoint, isValidObjectName, isValidPort, isValidPrefix, isVirtualHostStyle, makeDateLong, PART_CONSTRAINTS, partsRequired, prependXAMZMeta, readableStream, sanitizeETag, toMd5, toSha256, uriEscape, uriResourceEscape } from "./helper.mjs"; import { joinHostPort } from "./join-host-port.mjs"; import { PostPolicy } from "./post-policy.mjs"; import { requestWithRetry } from "./request.mjs"; import { drainResponse, readAsBuffer, readAsString } from "./response.mjs"; import { getS3Endpoint } from "./s3-endpoints.mjs"; import { parseBucketNotification, parseCompleteMultipart, parseInitiateMultipart, parseListObjects, parseListObjectsV2, parseObjectLegalHoldConfig, parseSelectObjectContentResponse, uploadPartParser } from "./xml-parser.mjs"; import * as xmlParsers from "./xml-parser.mjs"; const xml = new xml2js.Builder({ renderOpts: { pretty: false }, headless: true }); // will be replaced by bundler. const Package = { version: "8.0.7" || 'development' }; const requestOptionProperties = ['agent', 'ca', 'cert', 'ciphers', 'clientCertEngine', 'crl', 'dhparam', 'ecdhCurve', 'family', 'honorCipherOrder', 'key', 'passphrase', 'pfx', 'rejectUnauthorized', 'secureOptions', 'secureProtocol', 'servername', 'sessionIdContext']; export class TypedClient { partSize = 64 * 1024 * 1024; maximumPartSize = 5 * 1024 * 1024 * 1024; maxObjectSize = 5 * 1024 * 1024 * 1024 * 1024; constructor(params) { // @ts-expect-error deprecated property if (params.secure !== undefined) { throw new Error('"secure" option deprecated, "useSSL" should be used instead'); } // Default values if not specified. if (params.useSSL === undefined) { params.useSSL = true; } if (!params.port) { params.port = 0; } // Validate input params. if (!isValidEndpoint(params.endPoint)) { throw new errors.InvalidEndpointError(`Invalid endPoint : ${params.endPoint}`); } if (!isValidPort(params.port)) { throw new errors.InvalidArgumentError(`Invalid port : ${params.port}`); } if (!isBoolean(params.useSSL)) { throw new errors.InvalidArgumentError(`Invalid useSSL flag type : ${params.useSSL}, expected to be of type "boolean"`); } // Validate region only if its set. if (params.region) { if (!isString(params.region)) { throw new errors.InvalidArgumentError(`Invalid region : ${params.region}`); } } const host = params.endPoint.toLowerCase(); let port = params.port; let protocol; let transport; let transportAgent; // Validate if configuration is not using SSL // for constructing relevant endpoints. if (params.useSSL) { // Defaults to secure. transport = https; protocol = 'https:'; port = port || 443; transportAgent = https.globalAgent; } else { transport = http; protocol = 'http:'; port = port || 80; transportAgent = http.globalAgent; } // if custom transport is set, use it. if (params.transport) { if (!isObject(params.transport)) { throw new errors.InvalidArgumentError(`Invalid transport type : ${params.transport}, expected to be type "object"`); } transport = params.transport; } // if custom transport agent is set, use it. if (params.transportAgent) { if (!isObject(params.transportAgent)) { throw new errors.InvalidArgumentError(`Invalid transportAgent type: ${params.transportAgent}, expected to be type "object"`); } transportAgent = params.transportAgent; } // User Agent should always following the below style. // Please open an issue to discuss any new changes here. // // MinIO (OS; ARCH) LIB/VER APP/VER // const libraryComments = `(${process.platform}; ${process.arch})`; const libraryAgent = `MinIO ${libraryComments} minio-js/${Package.version}`; // User agent block ends. this.transport = transport; this.transportAgent = transportAgent; this.host = host; this.port = port; this.protocol = protocol; this.userAgent = `${libraryAgent}`; // Default path style is true if (params.pathStyle === undefined) { this.pathStyle = true; } else { this.pathStyle = params.pathStyle; } this.accessKey = params.accessKey ?? ''; this.secretKey = params.secretKey ?? ''; this.sessionToken = params.sessionToken; this.anonymous = !this.accessKey || !this.secretKey; if (params.credentialsProvider) { this.anonymous = false; this.credentialsProvider = params.credentialsProvider; } this.regionMap = {}; if (params.region) { this.region = params.region; } if (params.partSize) { this.partSize = params.partSize; this.overRidePartSize = true; } if (this.partSize < 5 * 1024 * 1024) { throw new errors.InvalidArgumentError(`Part size should be greater than 5MB`); } if (this.partSize > 5 * 1024 * 1024 * 1024) { throw new errors.InvalidArgumentError(`Part size should be less than 5GB`); } // SHA256 is enabled only for authenticated http requests. If the request is authenticated // and the connection is https we use x-amz-content-sha256=UNSIGNED-PAYLOAD // header for signature calculation. this.enableSHA256 = !this.anonymous && !params.useSSL; this.s3AccelerateEndpoint = params.s3AccelerateEndpoint || undefined; this.reqOptions = {}; this.clientExtensions = new Extensions(this); if (params.retryOptions) { if (!isObject(params.retryOptions)) { throw new errors.InvalidArgumentError(`Invalid retryOptions type: ${params.retryOptions}, expected to be type "object"`); } this.retryOptions = params.retryOptions; } else { this.retryOptions = { disableRetry: false }; } } /** * Minio extensions that aren't necessary present for Amazon S3 compatible storage servers */ get extensions() { return this.clientExtensions; } /** * @param endPoint - valid S3 acceleration end point */ setS3TransferAccelerate(endPoint) { this.s3AccelerateEndpoint = endPoint; } /** * Sets the supported request options. */ setRequestOptions(options) { if (!isObject(options)) { throw new TypeError('request options should be of type "object"'); } this.reqOptions = _.pick(options, requestOptionProperties); } /** * This is s3 Specific and does not hold validity in any other Object storage. */ getAccelerateEndPointIfSet(bucketName, objectName) { if (!isEmpty(this.s3AccelerateEndpoint) && !isEmpty(bucketName) && !isEmpty(objectName)) { // http://docs.aws.amazon.com/AmazonS3/latest/dev/transfer-acceleration.html // Disable transfer acceleration for non-compliant bucket names. if (bucketName.includes('.')) { throw new Error(`Transfer Acceleration is not supported for non compliant bucket:${bucketName}`); } // If transfer acceleration is requested set new host. // For more details about enabling transfer acceleration read here. // http://docs.aws.amazon.com/AmazonS3/latest/dev/transfer-acceleration.html return this.s3AccelerateEndpoint; } return false; } /** * Set application specific information. * Generates User-Agent in the following style. * MinIO (OS; ARCH) LIB/VER APP/VER */ setAppInfo(appName, appVersion) { if (!isString(appName)) { throw new TypeError(`Invalid appName: ${appName}`); } if (appName.trim() === '') { throw new errors.InvalidArgumentError('Input appName cannot be empty.'); } if (!isString(appVersion)) { throw new TypeError(`Invalid appVersion: ${appVersion}`); } if (appVersion.trim() === '') { throw new errors.InvalidArgumentError('Input appVersion cannot be empty.'); } this.userAgent = `${this.userAgent} ${appName}/${appVersion}`; } /** * returns options object that can be used with http.request() * Takes care of constructing virtual-host-style or path-style hostname */ getRequestOptions(opts) { const method = opts.method; const region = opts.region; const bucketName = opts.bucketName; let objectName = opts.objectName; const headers = opts.headers; const query = opts.query; let reqOptions = { method, headers: {}, protocol: this.protocol, // If custom transportAgent was supplied earlier, we'll inject it here agent: this.transportAgent }; // Verify if virtual host supported. let virtualHostStyle; if (bucketName) { virtualHostStyle = isVirtualHostStyle(this.host, this.protocol, bucketName, this.pathStyle); } let path = '/'; let host = this.host; let port; if (this.port) { port = this.port; } if (objectName) { objectName = uriResourceEscape(objectName); } // For Amazon S3 endpoint, get endpoint based on region. if (isAmazonEndpoint(host)) { const accelerateEndPoint = this.getAccelerateEndPointIfSet(bucketName, objectName); if (accelerateEndPoint) { host = `${accelerateEndPoint}`; } else { host = getS3Endpoint(region); } } if (virtualHostStyle && !opts.pathStyle) { // For all hosts which support virtual host style, `bucketName` // is part of the hostname in the following format: // // var host = 'bucketName.example.com' // if (bucketName) { host = `${bucketName}.${host}`; } if (objectName) { path = `/${objectName}`; } } else { // For all S3 compatible storage services we will fallback to // path style requests, where `bucketName` is part of the URI // path. if (bucketName) { path = `/${bucketName}`; } if (objectName) { path = `/${bucketName}/${objectName}`; } } if (query) { path += `?${query}`; } reqOptions.headers.host = host; if (reqOptions.protocol === 'http:' && port !== 80 || reqOptions.protocol === 'https:' && port !== 443) { reqOptions.headers.host = joinHostPort(host, port); } reqOptions.headers['user-agent'] = this.userAgent; if (headers) { // have all header keys in lower case - to make signing easy for (const [k, v] of Object.entries(headers)) { reqOptions.headers[k.toLowerCase()] = v; } } // Use any request option specified in minioClient.setRequestOptions() reqOptions = Object.assign({}, this.reqOptions, reqOptions); return { ...reqOptions, headers: _.mapValues(_.pickBy(reqOptions.headers, isDefined), v => v.toString()), host, port, path }; } async setCredentialsProvider(credentialsProvider) { if (!(credentialsProvider instanceof CredentialProvider)) { throw new Error('Unable to get credentials. Expected instance of CredentialProvider'); } this.credentialsProvider = credentialsProvider; await this.checkAndRefreshCreds(); } async checkAndRefreshCreds() { if (this.credentialsProvider) { try { const credentialsConf = await this.credentialsProvider.getCredentials(); this.accessKey = credentialsConf.getAccessKey(); this.secretKey = credentialsConf.getSecretKey(); this.sessionToken = credentialsConf.getSessionToken(); } catch (e) { throw new Error(`Unable to get credentials: ${e}`, { cause: e }); } } } /** * log the request, response, error */ logHTTP(reqOptions, response, err) { // if no logStream available return. if (!this.logStream) { return; } if (!isObject(reqOptions)) { throw new TypeError('reqOptions should be of type "object"'); } if (response && !isReadableStream(response)) { throw new TypeError('response should be of type "Stream"'); } if (err && !(err instanceof Error)) { throw new TypeError('err should be of type "Error"'); } const logStream = this.logStream; const logHeaders = headers => { Object.entries(headers).forEach(([k, v]) => { if (k == 'authorization') { if (isString(v)) { const redactor = new RegExp('Signature=([0-9a-f]+)'); v = v.replace(redactor, 'Signature=**REDACTED**'); } } logStream.write(`${k}: ${v}\n`); }); logStream.write('\n'); }; logStream.write(`REQUEST: ${reqOptions.method} ${reqOptions.path}\n`); logHeaders(reqOptions.headers); if (response) { this.logStream.write(`RESPONSE: ${response.statusCode}\n`); logHeaders(response.headers); } if (err) { logStream.write('ERROR BODY:\n'); const errJSON = JSON.stringify(err, null, '\t'); logStream.write(`${errJSON}\n`); } } /** * Enable tracing */ traceOn(stream) { if (!stream) { stream = process.stdout; } this.logStream = stream; } /** * Disable tracing */ traceOff() { this.logStream = undefined; } /** * makeRequest is the primitive used by the apis for making S3 requests. * payload can be empty string in case of no payload. * statusCode is the expected statusCode. If response.statusCode does not match * we parse the XML error and call the callback with the error message. * * A valid region is passed by the calls - listBuckets, makeBucket and getBucketRegion. * * @internal */ async makeRequestAsync(options, payload = '', expectedCodes = [200], region = '') { if (!isObject(options)) { throw new TypeError('options should be of type "object"'); } if (!isString(payload) && !isObject(payload)) { // Buffer is of type 'object' throw new TypeError('payload should be of type "string" or "Buffer"'); } expectedCodes.forEach(statusCode => { if (!isNumber(statusCode)) { throw new TypeError('statusCode should be of type "number"'); } }); if (!isString(region)) { throw new TypeError('region should be of type "string"'); } if (!options.headers) { options.headers = {}; } if (options.method === 'POST' || options.method === 'PUT' || options.method === 'DELETE') { options.headers['content-length'] = payload.length.toString(); } const sha256sum = this.enableSHA256 ? toSha256(payload) : ''; return this.makeRequestStreamAsync(options, payload, sha256sum, expectedCodes, region); } /** * new request with promise * * No need to drain response, response body is not valid */ async makeRequestAsyncOmit(options, payload = '', statusCodes = [200], region = '') { const res = await this.makeRequestAsync(options, payload, statusCodes, region); await drainResponse(res); return res; } /** * makeRequestStream will be used directly instead of makeRequest in case the payload * is available as a stream. for ex. putObject * * @internal */ async makeRequestStreamAsync(options, body, sha256sum, statusCodes, region) { if (!isObject(options)) { throw new TypeError('options should be of type "object"'); } if (!(Buffer.isBuffer(body) || typeof body === 'string' || isReadableStream(body))) { throw new errors.InvalidArgumentError(`stream should be a Buffer, string or readable Stream, got ${typeof body} instead`); } if (!isString(sha256sum)) { throw new TypeError('sha256sum should be of type "string"'); } statusCodes.forEach(statusCode => { if (!isNumber(statusCode)) { throw new TypeError('statusCode should be of type "number"'); } }); if (!isString(region)) { throw new TypeError('region should be of type "string"'); } // sha256sum will be empty for anonymous or https requests if (!this.enableSHA256 && sha256sum.length !== 0) { throw new errors.InvalidArgumentError(`sha256sum expected to be empty for anonymous or https requests`); } // sha256sum should be valid for non-anonymous http requests. if (this.enableSHA256 && sha256sum.length !== 64) { throw new errors.InvalidArgumentError(`Invalid sha256sum : ${sha256sum}`); } await this.checkAndRefreshCreds(); // eslint-disable-next-line @typescript-eslint/no-non-null-assertion region = region || (await this.getBucketRegionAsync(options.bucketName)); const reqOptions = this.getRequestOptions({ ...options, region }); if (!this.anonymous) { // For non-anonymous https requests sha256sum is 'UNSIGNED-PAYLOAD' for signature calculation. if (!this.enableSHA256) { sha256sum = 'UNSIGNED-PAYLOAD'; } const date = new Date(); reqOptions.headers['x-amz-date'] = makeDateLong(date); reqOptions.headers['x-amz-content-sha256'] = sha256sum; if (this.sessionToken) { reqOptions.headers['x-amz-security-token'] = this.sessionToken; } reqOptions.headers.authorization = signV4(reqOptions, this.accessKey, this.secretKey, region, date, sha256sum); } const response = await requestWithRetry(this.transport, reqOptions, body, this.retryOptions.disableRetry === true ? 0 : this.retryOptions.maximumRetryCount, this.retryOptions.baseDelayMs, this.retryOptions.maximumDelayMs); if (!response.statusCode) { throw new Error("BUG: response doesn't have a statusCode"); } if (!statusCodes.includes(response.statusCode)) { // For an incorrect region, S3 server always sends back 400. // But we will do cache invalidation for all errors so that, // in future, if AWS S3 decides to send a different status code or // XML error code we will still work fine. // eslint-disable-next-line @typescript-eslint/no-non-null-assertion delete this.regionMap[options.bucketName]; const err = await xmlParsers.parseResponseError(response); this.logHTTP(reqOptions, response, err); throw err; } this.logHTTP(reqOptions, response); return response; } /** * gets the region of the bucket * * @param bucketName * */ async getBucketRegionAsync(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name : ${bucketName}`); } // Region is set with constructor, return the region right here. if (this.region) { return this.region; } const cached = this.regionMap[bucketName]; if (cached) { return cached; } const extractRegionAsync = async response => { const body = await readAsString(response); const region = xmlParsers.parseBucketRegion(body) || DEFAULT_REGION; this.regionMap[bucketName] = region; return region; }; const method = 'GET'; const query = 'location'; // `getBucketLocation` behaves differently in following ways for // different environments. // // - For nodejs env we default to path style requests. // - For browser env path style requests on buckets yields CORS // error. To circumvent this problem we make a virtual host // style request signed with 'us-east-1'. This request fails // with an error 'AuthorizationHeaderMalformed', additionally // the error XML also provides Region of the bucket. To validate // this region is proper we retry the same request with the newly // obtained region. const pathStyle = this.pathStyle && !isBrowser; let region; try { const res = await this.makeRequestAsync({ method, bucketName, query, pathStyle }, '', [200], DEFAULT_REGION); return extractRegionAsync(res); } catch (e) { // make alignment with mc cli if (e instanceof errors.S3Error) { const errCode = e.code; const errRegion = e.region; if (errCode === 'AccessDenied' && !errRegion) { return DEFAULT_REGION; } } // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore if (!(e.name === 'AuthorizationHeaderMalformed')) { throw e; } // @ts-expect-error we set extra properties on error object region = e.Region; if (!region) { throw e; } } const res = await this.makeRequestAsync({ method, bucketName, query, pathStyle }, '', [200], region); return await extractRegionAsync(res); } /** * makeRequest is the primitive used by the apis for making S3 requests. * payload can be empty string in case of no payload. * statusCode is the expected statusCode. If response.statusCode does not match * we parse the XML error and call the callback with the error message. * A valid region is passed by the calls - listBuckets, makeBucket and * getBucketRegion. * * @deprecated use `makeRequestAsync` instead */ makeRequest(options, payload = '', expectedCodes = [200], region = '', returnResponse, cb) { let prom; if (returnResponse) { prom = this.makeRequestAsync(options, payload, expectedCodes, region); } else { // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-expect-error compatible for old behaviour prom = this.makeRequestAsyncOmit(options, payload, expectedCodes, region); } prom.then(result => cb(null, result), err => { // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore cb(err); }); } /** * makeRequestStream will be used directly instead of makeRequest in case the payload * is available as a stream. for ex. putObject * * @deprecated use `makeRequestStreamAsync` instead */ makeRequestStream(options, stream, sha256sum, statusCodes, region, returnResponse, cb) { const executor = async () => { const res = await this.makeRequestStreamAsync(options, stream, sha256sum, statusCodes, region); if (!returnResponse) { await drainResponse(res); } return res; }; executor().then(result => cb(null, result), // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore err => cb(err)); } /** * @deprecated use `getBucketRegionAsync` instead */ getBucketRegion(bucketName, cb) { return this.getBucketRegionAsync(bucketName).then(result => cb(null, result), // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore err => cb(err)); } // Bucket operations /** * Creates the bucket `bucketName`. * */ async makeBucket(bucketName, region = '', makeOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } // Backward Compatibility if (isObject(region)) { makeOpts = region; region = ''; } if (!isString(region)) { throw new TypeError('region should be of type "string"'); } if (makeOpts && !isObject(makeOpts)) { throw new TypeError('makeOpts should be of type "object"'); } let payload = ''; // Region already set in constructor, validate if // caller requested bucket location is same. if (region && this.region) { if (region !== this.region) { throw new errors.InvalidArgumentError(`Configured region ${this.region}, requested ${region}`); } } // sending makeBucket request with XML containing 'us-east-1' fails. For // default region server expects the request without body if (region && region !== DEFAULT_REGION) { payload = xml.buildObject({ CreateBucketConfiguration: { $: { xmlns: 'http://s3.amazonaws.com/doc/2006-03-01/' }, LocationConstraint: region } }); } const method = 'PUT'; const headers = {}; if (makeOpts && makeOpts.ObjectLocking) { headers['x-amz-bucket-object-lock-enabled'] = true; } // For custom region clients default to custom region specified in client constructor const finalRegion = this.region || region || DEFAULT_REGION; const requestOpt = { method, bucketName, headers }; try { await this.makeRequestAsyncOmit(requestOpt, payload, [200], finalRegion); } catch (err) { if (region === '' || region === DEFAULT_REGION) { if (err instanceof errors.S3Error) { const errCode = err.code; const errRegion = err.region; if (errCode === 'AuthorizationHeaderMalformed' && errRegion !== '') { // Retry with region returned as part of error await this.makeRequestAsyncOmit(requestOpt, payload, [200], errCode); } } } throw err; } } /** * To check if a bucket already exists. */ async bucketExists(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'HEAD'; try { await this.makeRequestAsyncOmit({ method, bucketName }); } catch (err) { // @ts-ignore if (err.code === 'NoSuchBucket' || err.code === 'NotFound') { return false; } throw err; } return true; } /** * @deprecated use promise style API */ async removeBucket(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'DELETE'; await this.makeRequestAsyncOmit({ method, bucketName }, '', [204]); delete this.regionMap[bucketName]; } /** * Callback is called with readable stream of the object content. */ async getObject(bucketName, objectName, getOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } return this.getPartialObject(bucketName, objectName, 0, 0, getOpts); } /** * Callback is called with readable stream of the partial object content. * @param bucketName * @param objectName * @param offset * @param length - length of the object that will be read in the stream (optional, if not specified we read the rest of the file from the offset) * @param getOpts */ async getPartialObject(bucketName, objectName, offset, length = 0, getOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isNumber(offset)) { throw new TypeError('offset should be of type "number"'); } if (!isNumber(length)) { throw new TypeError('length should be of type "number"'); } let range = ''; if (offset || length) { if (offset) { range = `bytes=${+offset}-`; } else { range = 'bytes=0-'; offset = 0; } if (length) { range += `${+length + offset - 1}`; } } let query = ''; let headers = { ...(range !== '' && { range }) }; if (getOpts) { const sseHeaders = { ...(getOpts.SSECustomerAlgorithm && { 'X-Amz-Server-Side-Encryption-Customer-Algorithm': getOpts.SSECustomerAlgorithm }), ...(getOpts.SSECustomerKey && { 'X-Amz-Server-Side-Encryption-Customer-Key': getOpts.SSECustomerKey }), ...(getOpts.SSECustomerKeyMD5 && { 'X-Amz-Server-Side-Encryption-Customer-Key-MD5': getOpts.SSECustomerKeyMD5 }) }; query = qs.stringify(getOpts); headers = { ...prependXAMZMeta(sseHeaders), ...headers }; } const expectedStatusCodes = [200]; if (range) { expectedStatusCodes.push(206); } const method = 'GET'; return await this.makeRequestAsync({ method, bucketName, objectName, headers, query }, '', expectedStatusCodes); } /** * download object content to a file. * This method will create a temp file named `${filename}.${base64(etag)}.part.minio` when downloading. * * @param bucketName - name of the bucket * @param objectName - name of the object * @param filePath - path to which the object data will be written to * @param getOpts - Optional object get option */ async fGetObject(bucketName, objectName, filePath, getOpts) { // Input validation. if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isString(filePath)) { throw new TypeError('filePath should be of type "string"'); } const downloadToTmpFile = async () => { let partFileStream; const objStat = await this.statObject(bucketName, objectName, getOpts); const encodedEtag = Buffer.from(objStat.etag).toString('base64'); const partFile = `${filePath}.${encodedEtag}.part.minio`; await fsp.mkdir(path.dirname(filePath), { recursive: true }); let offset = 0; try { const stats = await fsp.stat(partFile); if (objStat.size === stats.size) { return partFile; } offset = stats.size; partFileStream = fs.createWriteStream(partFile, { flags: 'a' }); } catch (e) { if (e instanceof Error && e.code === 'ENOENT') { // file not exist partFileStream = fs.createWriteStream(partFile, { flags: 'w' }); } else { // other error, maybe access deny throw e; } } const downloadStream = await this.getPartialObject(bucketName, objectName, offset, 0, getOpts); await streamPromise.pipeline(downloadStream, partFileStream); const stats = await fsp.stat(partFile); if (stats.size === objStat.size) { return partFile; } throw new Error('Size mismatch between downloaded file and the object'); }; const partFile = await downloadToTmpFile(); await fsp.rename(partFile, filePath); } /** * Stat information of the object. */ async statObject(bucketName, objectName, statOpts) { const statOptDef = statOpts || {}; if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isObject(statOptDef)) { throw new errors.InvalidArgumentError('statOpts should be of type "object"'); } const query = qs.stringify(statOptDef); const method = 'HEAD'; const res = await this.makeRequestAsyncOmit({ method, bucketName, objectName, query }); return { size: parseInt(res.headers['content-length']), metaData: extractMetadata(res.headers), lastModified: new Date(res.headers['last-modified']), versionId: getVersionId(res.headers), etag: sanitizeETag(res.headers.etag) }; } async removeObject(bucketName, objectName, removeOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (removeOpts && !isObject(removeOpts)) { throw new errors.InvalidArgumentError('removeOpts should be of type "object"'); } const method = 'DELETE'; const headers = {}; if (removeOpts !== null && removeOpts !== void 0 && removeOpts.governanceBypass) { headers['X-Amz-Bypass-Governance-Retention'] = true; } if (removeOpts !== null && removeOpts !== void 0 && removeOpts.forceDelete) { headers['x-minio-force-delete'] = true; } const queryParams = {}; if (removeOpts !== null && removeOpts !== void 0 && removeOpts.versionId) { queryParams.versionId = `${removeOpts.versionId}`; } const query = qs.stringify(queryParams); await this.makeRequestAsyncOmit({ method, bucketName, objectName, headers, query }, '', [200, 204]); } // Calls implemented below are related to multipart. listIncompleteUploads(bucket, prefix, recursive) { if (prefix === undefined) { prefix = ''; } if (recursive === undefined) { recursive = false; } if (!isValidBucketName(bucket)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucket); } if (!isValidPrefix(prefix)) { throw new errors.InvalidPrefixError(`Invalid prefix : ${prefix}`); } if (!isBoolean(recursive)) { throw new TypeError('recursive should be of type "boolean"'); } const delimiter = recursive ? '' : '/'; let keyMarker = ''; let uploadIdMarker = ''; const uploads = []; let ended = false; // TODO: refactor this with async/await and `stream.Readable.from` const readStream = new stream.Readable({ objectMode: true }); readStream._read = () => { // push one upload info per _read() if (uploads.length) { return readStream.push(uploads.shift()); } if (ended) { return readStream.push(null); } this.listIncompleteUploadsQuery(bucket, prefix, keyMarker, uploadIdMarker, delimiter).then(result => { // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore result.prefixes.forEach(prefix => uploads.push(prefix)); async.eachSeries(result.uploads, (upload, cb) => { // for each incomplete upload add the sizes of its uploaded parts // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore this.listParts(bucket, upload.key, upload.uploadId).then(parts => { // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore upload.size = parts.reduce((acc, item) => acc + item.size, 0); uploads.push(upload); cb(); }, err => cb(err)); }, err => { if (err) { readStream.emit('error', err); return; } if (result.isTruncated) { keyMarker = result.nextKeyMarker; uploadIdMarker = result.nextUploadIdMarker; } else { ended = true; } // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore readStream._read(); }); }, e => { readStream.emit('error', e); }); }; return readStream; } /** * Called by listIncompleteUploads to fetch a batch of incomplete uploads. */ async listIncompleteUploadsQuery(bucketName, prefix, keyMarker, uploadIdMarker, delimiter) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isString(prefix)) { throw new TypeError('prefix should be of type "string"'); } if (!isString(keyMarker)) { throw new TypeError('keyMarker should be of type "string"'); } if (!isString(uploadIdMarker)) { throw new TypeError('uploadIdMarker should be of type "string"'); } if (!isString(delimiter)) { throw new TypeError('delimiter should be of type "string"'); } const queries = []; queries.push(`prefix=${uriEscape(prefix)}`); queries.push(`delimiter=${uriEscape(delimiter)}`); if (keyMarker) { queries.push(`key-marker=${uriEscape(keyMarker)}`); } if (uploadIdMarker) { queries.push(`upload-id-marker=${uploadIdMarker}`); } const maxUploads = 1000; queries.push(`max-uploads=${maxUploads}`); queries.sort(); queries.unshift('uploads'); let query = ''; if (queries.length > 0) { query = `${queries.join('&')}`; } const method = 'GET'; const res = await this.makeRequestAsync({ method, bucketName, query }); const body = await readAsString(res); return xmlParsers.parseListMultipart(body); } /** * Initiate a new multipart upload. * @internal */ async initiateNewMultipartUpload(bucketName, objectName, headers) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isObject(headers)) { throw new errors.InvalidObjectNameError('contentType should be of type "object"'); } const method = 'POST'; const query = 'uploads'; const res = await this.makeRequestAsync({ method, bucketName, objectName, query, headers }); const body = await readAsBuffer(res); return parseInitiateMultipart(body.toString()); } /** * Internal Method to abort a multipart upload request in case of any errors. * * @param bucketName - Bucket Name * @param objectName - Object Name * @param uploadId - id of a multipart upload to cancel during compose object sequence. */ async abortMultipartUpload(bucketName, objectName, uploadId) { const method = 'DELETE'; const query = `uploadId=${uploadId}`; const requestOptions = { method, bucketName, objectName: objectName, query }; await this.makeRequestAsyncOmit(requestOptions, '', [204]); } async findUploadId(bucketName, objectName) { var _latestUpload; if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } let latestUpload; let keyMarker = ''; let uploadIdMarker = ''; for (;;) { const result = await this.listIncompleteUploadsQuery(bucketName, objectName, keyMarker, uploadIdMarker, ''); for (const upload of result.uploads) { if (upload.key === objectName) { if (!latestUpload || upload.initiated.getTime() > latestUpload.initiated.getTime()) { latestUpload = upload; } } } if (result.isTruncated) { keyMarker = result.nextKeyMarker; uploadIdMarker = result.nextUploadIdMarker; continue; } break; } return (_latestUpload = latestUpload) === null || _latestUpload === void 0 ? void 0 : _latestUpload.uploadId; } /** * this call will aggregate the parts on the server into a single object. */ async completeMultipartUpload(bucketName, objectName, uploadId, etags) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isString(uploadId)) { throw new TypeError('uploadId should be of type "string"'); } if (!isObject(etags)) { throw new TypeError('etags should be of type "Array"'); } if (!uploadId) { throw new errors.InvalidArgumentError('uploadId cannot be empty'); } const method = 'POST'; const query = `uploadId=${uriEscape(uploadId)}`; const builder = new xml2js.Builder(); const payload = builder.buildObject({ CompleteMultipartUpload: { $: { xmlns: 'http://s3.amazonaws.com/doc/2006-03-01/' }, Part: etags.map(etag => { return { PartNumber: etag.part, ETag: etag.etag }; }) } }); const res = await this.makeRequestAsync({ method, bucketName, objectName, query }, payload); const body = await readAsBuffer(res); const result = parseCompleteMultipart(body.toString()); if (!result) { throw new Error('BUG: failed to parse server response'); } if (result.errCode) { // Multipart Complete API returns an error XML after a 200 http status throw new errors.S3Error(result.errMessage); } return { // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore etag: result.etag, versionId: getVersionId(res.headers) }; } /** * Get part-info of all parts of an incomplete upload specified by uploadId. */ async listParts(bucketName, objectName, uploadId) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isString(uploadId)) { throw new TypeError('uploadId should be of type "string"'); } if (!uploadId) { throw new errors.InvalidArgumentError('uploadId cannot be empty'); } const parts = []; let marker = 0; let result; do { result = await this.listPartsQuery(bucketName, objectName, uploadId, marker); marker = result.marker; parts.push(...result.parts); } while (result.isTruncated); return parts; } /** * Called by listParts to fetch a batch of part-info */ async listPartsQuery(bucketName, objectName, uploadId, marker) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isString(uploadId)) { throw new TypeError('uploadId should be of type "string"'); } if (!isNumber(marker)) { throw new TypeError('marker should be of type "number"'); } if (!uploadId) { throw new errors.InvalidArgumentError('uploadId cannot be empty'); } let query = `uploadId=${uriEscape(uploadId)}`; if (marker) { query += `&part-number-marker=${marker}`; } const method = 'GET'; const res = await this.makeRequestAsync({ method, bucketName, objectName, query }); return xmlParsers.parseListParts(await readAsString(res)); } async listBuckets() { const method = 'GET'; const regionConf = this.region || DEFAULT_REGION; const httpRes = await this.makeRequestAsync({ method }, '', [200], regionConf); const xmlResult = await readAsString(httpRes); return xmlParsers.parseListBucket(xmlResult); } /** * Calculate part size given the object size. Part size will be atleast this.partSize */ calculatePartSize(size) { if (!isNumber(size)) { throw new TypeError('size should be of type "number"'); } if (size > this.maxObjectSize) { throw new TypeError(`size should not be more than ${this.maxObjectSize}`); } if (this.overRidePartSize) { return this.partSize; } let partSize = this.partSize; for (;;) { // while(true) {...} throws linting error. // If partSize is big enough to accomodate the object size, then use it. if (partSize * 10000 > size) { return partSize; } // Try part sizes as 64MB, 80MB, 96MB etc. partSize += 16 * 1024 * 1024; } } /** * Uploads the object using contents from a file */ async fPutObject(bucketName, objectName, filePath, metaData) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isString(filePath)) { throw new TypeError('filePath should be of type "string"'); } if (metaData && !isObject(metaData)) { throw new TypeError('metaData should be of type "object"'); } // Inserts correct `content-type` attribute based on metaData and filePath metaData = insertContentType(metaData || {}, filePath); const stat = await fsp.stat(filePath); return await this.putObject(bucketName, objectName, fs.createReadStream(filePath), stat.size, metaData); } /** * Uploading a stream, "Buffer" or "string". * It's recommended to pass `size` argument with stream. */ async putObject(bucketName, objectName, stream, size, metaData) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } // We'll need to shift arguments to the left because of metaData // and size being optional. if (isObject(size)) { metaData = size; } // Ensures Metadata has appropriate prefix for A3 API const headers = prependXAMZMeta(metaData); if (typeof stream === 'string' || stream instanceof Buffer) { // Adapts the non-stream interface into a stream. size = stream.length; stream = readableStream(stream); } else if (!isReadableStream(stream)) { throw new TypeError('third argument should be of type "stream.Readable" or "Buffer" or "string"'); } if (isNumber(size) && size < 0) { throw new errors.InvalidArgumentError(`size cannot be negative, given size: ${size}`); } // Get the part size and forward that to the BlockStream. Default to the // largest block size possible if necessary. if (!isNumber(size)) { size = this.maxObjectSize; } // Get the part size and forward that to the BlockStream. Default to the // largest block size possible if necessary. if (size === undefined) { const statSize = await getContentLength(stream); if (statSize !== null) { size = statSize; } } if (!isNumber(size)) { // Backward compatibility size = this.maxObjectSize; } if (size === 0) { return this.uploadBuffer(bucketName, objectName, headers, Buffer.from('')); } const partSize = this.calculatePartSize(size); if (typeof stream === 'string' || Buffer.isBuffer(stream) || size <= partSize) { const buf = isReadableStream(stream) ? await readAsBuffer(stream) : Buffer.from(stream); return this.uploadBuffer(bucketName, objectName, headers, buf); } return this.uploadStream(bucketName, objectName, headers, stream, partSize); } /** * method to upload buffer in one call * @private */ async uploadBuffer(bucketName, objectName, headers, buf) { const { md5sum, sha256sum } = hashBinary(buf, this.enableSHA256); headers['Content-Length'] = buf.length; if (!this.enableSHA256) { headers['Content-MD5'] = md5sum; } const res = await this.makeRequestStreamAsync({ method: 'PUT', bucketName, objectName, headers }, buf, sha256sum, [200], ''); await drainResponse(res); return { etag: sanitizeETag(res.headers.etag), versionId: getVersionId(res.headers) }; } /** * upload stream with MultipartUpload * @private */ async uploadStream(bucketName, objectName, headers, body, partSize) { // A map of the previously uploaded chunks, for resuming a file upload. This // will be null if we aren't resuming an upload. const oldParts = {}; // Keep track of the etags for aggregating the chunks together later. Each // etag represents a single chunk of the file. const eTags = []; const previousUploadId = await this.findUploadId(bucketName, objectName); let uploadId; if (!previousUploadId) { uploadId = await this.initiateNewMultipartUpload(bucketName, objectName, headers); } else { uploadId = previousUploadId; const oldTags = await this.listParts(bucketName, objectName, previousUploadId); oldTags.forEach(e => { oldParts[e.part] = e; }); } const chunkier = new BlockStream2({ size: partSize, zeroPadding: false }); // eslint-disable-next-line @typescript-eslint/no-unused-vars const [_, o] = await Promise.all([new Promise((resolve, reject) => { body.pipe(chunkier).on('error', reject); chunkier.on('end', resolve).on('error', reject); }), (async () => { let partNumber = 1; for await (const chunk of chunkier) { const md5 = crypto.createHash('md5').update(chunk).digest(); const oldPart = oldParts[partNumber]; if (oldPart) { if (oldPart.etag === md5.toString('hex')) { eTags.push({ part: partNumber, etag: oldPart.etag }); partNumber++; continue; } } partNumber++; // now start to upload missing part const options = { method: 'PUT', query: qs.stringify({ partNumber, uploadId }), headers: { 'Content-Length': chunk.length, 'Content-MD5': md5.toString('base64') }, bucketName, objectName }; const response = await this.makeRequestAsyncOmit(options, chunk); let etag = response.headers.etag; if (etag) { etag = etag.replace(/^"/, '').replace(/"$/, ''); } else { etag = ''; } eTags.push({ part: partNumber, etag }); } return await this.completeMultipartUpload(bucketName, objectName, uploadId, eTags); })()]); return o; } async removeBucketReplication(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'DELETE'; const query = 'replication'; await this.makeRequestAsyncOmit({ method, bucketName, query }, '', [200, 204], ''); } async setBucketReplication(bucketName, replicationConfig) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isObject(replicationConfig)) { throw new errors.InvalidArgumentError('replicationConfig should be of type "object"'); } else { if (_.isEmpty(replicationConfig.role)) { throw new errors.InvalidArgumentError('Role cannot be empty'); } else if (replicationConfig.role && !isString(replicationConfig.role)) { throw new errors.InvalidArgumentError('Invalid value for role', replicationConfig.role); } if (_.isEmpty(replicationConfig.rules)) { throw new errors.InvalidArgumentError('Minimum one replication rule must be specified'); } } const method = 'PUT'; const query = 'replication'; const headers = {}; const replicationParamsConfig = { ReplicationConfiguration: { Role: replicationConfig.role, Rule: replicationConfig.rules } }; const builder = new xml2js.Builder({ renderOpts: { pretty: false }, headless: true }); const payload = builder.buildObject(replicationParamsConfig); headers['Content-MD5'] = toMd5(payload); await this.makeRequestAsyncOmit({ method, bucketName, query, headers }, payload); } async getBucketReplication(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'GET'; const query = 'replication'; const httpRes = await this.makeRequestAsync({ method, bucketName, query }, '', [200, 204]); const xmlResult = await readAsString(httpRes); return xmlParsers.parseReplicationConfig(xmlResult); } async getObjectLegalHold(bucketName, objectName, getOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (getOpts) { if (!isObject(getOpts)) { throw new TypeError('getOpts should be of type "Object"'); } else if (Object.keys(getOpts).length > 0 && getOpts.versionId && !isString(getOpts.versionId)) { throw new TypeError('versionId should be of type string.:', getOpts.versionId); } } const method = 'GET'; let query = 'legal-hold'; if (getOpts !== null && getOpts !== void 0 && getOpts.versionId) { query += `&versionId=${getOpts.versionId}`; } const httpRes = await this.makeRequestAsync({ method, bucketName, objectName, query }, '', [200]); const strRes = await readAsString(httpRes); return parseObjectLegalHoldConfig(strRes); } async setObjectLegalHold(bucketName, objectName, setOpts = { status: LEGAL_HOLD_STATUS.ENABLED }) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isObject(setOpts)) { throw new TypeError('setOpts should be of type "Object"'); } else { if (![LEGAL_HOLD_STATUS.ENABLED, LEGAL_HOLD_STATUS.DISABLED].includes(setOpts === null || setOpts === void 0 ? void 0 : setOpts.status)) { throw new TypeError('Invalid status: ' + setOpts.status); } if (setOpts.versionId && !setOpts.versionId.length) { throw new TypeError('versionId should be of type string.:' + setOpts.versionId); } } const method = 'PUT'; let query = 'legal-hold'; if (setOpts.versionId) { query += `&versionId=${setOpts.versionId}`; } const config = { Status: setOpts.status }; const builder = new xml2js.Builder({ rootName: 'LegalHold', renderOpts: { pretty: false }, headless: true }); const payload = builder.buildObject(config); const headers = {}; headers['Content-MD5'] = toMd5(payload); await this.makeRequestAsyncOmit({ method, bucketName, objectName, query, headers }, payload); } /** * Get Tags associated with a Bucket */ async getBucketTagging(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`); } const method = 'GET'; const query = 'tagging'; const requestOptions = { method, bucketName, query }; const response = await this.makeRequestAsync(requestOptions); const body = await readAsString(response); return xmlParsers.parseTagging(body); } /** * Get the tags associated with a bucket OR an object */ async getObjectTagging(bucketName, objectName, getOpts) { const method = 'GET'; let query = 'tagging'; if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidBucketNameError('Invalid object name: ' + objectName); } if (getOpts && !isObject(getOpts)) { throw new errors.InvalidArgumentError('getOpts should be of type "object"'); } if (getOpts && getOpts.versionId) { query = `${query}&versionId=${getOpts.versionId}`; } const requestOptions = { method, bucketName, query }; if (objectName) { requestOptions['objectName'] = objectName; } const response = await this.makeRequestAsync(requestOptions); const body = await readAsString(response); return xmlParsers.parseTagging(body); } /** * Set the policy on a bucket or an object prefix. */ async setBucketPolicy(bucketName, policy) { // Validate arguments. if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`); } if (!isString(policy)) { throw new errors.InvalidBucketPolicyError(`Invalid bucket policy: ${policy} - must be "string"`); } const query = 'policy'; let method = 'DELETE'; if (policy) { method = 'PUT'; } await this.makeRequestAsyncOmit({ method, bucketName, query }, policy, [204], ''); } /** * Get the policy on a bucket or an object prefix. */ async getBucketPolicy(bucketName) { // Validate arguments. if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`); } const method = 'GET'; const query = 'policy'; const res = await this.makeRequestAsync({ method, bucketName, query }); return await readAsString(res); } async putObjectRetention(bucketName, objectName, retentionOpts = {}) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!isObject(retentionOpts)) { throw new errors.InvalidArgumentError('retentionOpts should be of type "object"'); } else { if (retentionOpts.governanceBypass && !isBoolean(retentionOpts.governanceBypass)) { throw new errors.InvalidArgumentError(`Invalid value for governanceBypass: ${retentionOpts.governanceBypass}`); } if (retentionOpts.mode && ![RETENTION_MODES.COMPLIANCE, RETENTION_MODES.GOVERNANCE].includes(retentionOpts.mode)) { throw new errors.InvalidArgumentError(`Invalid object retention mode: ${retentionOpts.mode}`); } if (retentionOpts.retainUntilDate && !isString(retentionOpts.retainUntilDate)) { throw new errors.InvalidArgumentError(`Invalid value for retainUntilDate: ${retentionOpts.retainUntilDate}`); } if (retentionOpts.versionId && !isString(retentionOpts.versionId)) { throw new errors.InvalidArgumentError(`Invalid value for versionId: ${retentionOpts.versionId}`); } } const method = 'PUT'; let query = 'retention'; const headers = {}; if (retentionOpts.governanceBypass) { headers['X-Amz-Bypass-Governance-Retention'] = true; } const builder = new xml2js.Builder({ rootName: 'Retention', renderOpts: { pretty: false }, headless: true }); const params = {}; if (retentionOpts.mode) { params.Mode = retentionOpts.mode; } if (retentionOpts.retainUntilDate) { params.RetainUntilDate = retentionOpts.retainUntilDate; } if (retentionOpts.versionId) { query += `&versionId=${retentionOpts.versionId}`; } const payload = builder.buildObject(params); headers['Content-MD5'] = toMd5(payload); await this.makeRequestAsyncOmit({ method, bucketName, objectName, query, headers }, payload, [200, 204]); } async getObjectLockConfig(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'GET'; const query = 'object-lock'; const httpRes = await this.makeRequestAsync({ method, bucketName, query }); const xmlResult = await readAsString(httpRes); return xmlParsers.parseObjectLockConfig(xmlResult); } async setObjectLockConfig(bucketName, lockConfigOpts) { const retentionModes = [RETENTION_MODES.COMPLIANCE, RETENTION_MODES.GOVERNANCE]; const validUnits = [RETENTION_VALIDITY_UNITS.DAYS, RETENTION_VALIDITY_UNITS.YEARS]; if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (lockConfigOpts.mode && !retentionModes.includes(lockConfigOpts.mode)) { throw new TypeError(`lockConfigOpts.mode should be one of ${retentionModes}`); } if (lockConfigOpts.unit && !validUnits.includes(lockConfigOpts.unit)) { throw new TypeError(`lockConfigOpts.unit should be one of ${validUnits}`); } if (lockConfigOpts.validity && !isNumber(lockConfigOpts.validity)) { throw new TypeError(`lockConfigOpts.validity should be a number`); } const method = 'PUT'; const query = 'object-lock'; const config = { ObjectLockEnabled: 'Enabled' }; const configKeys = Object.keys(lockConfigOpts); const isAllKeysSet = ['unit', 'mode', 'validity'].every(lck => configKeys.includes(lck)); // Check if keys are present and all keys are present. if (configKeys.length > 0) { if (!isAllKeysSet) { throw new TypeError(`lockConfigOpts.mode,lockConfigOpts.unit,lockConfigOpts.validity all the properties should be specified.`); } else { config.Rule = { DefaultRetention: {} }; if (lockConfigOpts.mode) { config.Rule.DefaultRetention.Mode = lockConfigOpts.mode; } if (lockConfigOpts.unit === RETENTION_VALIDITY_UNITS.DAYS) { config.Rule.DefaultRetention.Days = lockConfigOpts.validity; } else if (lockConfigOpts.unit === RETENTION_VALIDITY_UNITS.YEARS) { config.Rule.DefaultRetention.Years = lockConfigOpts.validity; } } } const builder = new xml2js.Builder({ rootName: 'ObjectLockConfiguration', renderOpts: { pretty: false }, headless: true }); const payload = builder.buildObject(config); const headers = {}; headers['Content-MD5'] = toMd5(payload); await this.makeRequestAsyncOmit({ method, bucketName, query, headers }, payload); } async getBucketVersioning(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'GET'; const query = 'versioning'; const httpRes = await this.makeRequestAsync({ method, bucketName, query }); const xmlResult = await readAsString(httpRes); return await xmlParsers.parseBucketVersioningConfig(xmlResult); } async setBucketVersioning(bucketName, versionConfig) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!Object.keys(versionConfig).length) { throw new errors.InvalidArgumentError('versionConfig should be of type "object"'); } const method = 'PUT'; const query = 'versioning'; const builder = new xml2js.Builder({ rootName: 'VersioningConfiguration', renderOpts: { pretty: false }, headless: true }); const payload = builder.buildObject(versionConfig); await this.makeRequestAsyncOmit({ method, bucketName, query }, payload); } async setTagging(taggingParams) { const { bucketName, objectName, tags, putOpts } = taggingParams; const method = 'PUT'; let query = 'tagging'; if (putOpts && putOpts !== null && putOpts !== void 0 && putOpts.versionId) { query = `${query}&versionId=${putOpts.versionId}`; } const tagsList = []; for (const [key, value] of Object.entries(tags)) { tagsList.push({ Key: key, Value: value }); } const taggingConfig = { Tagging: { TagSet: { Tag: tagsList } } }; const headers = {}; const builder = new xml2js.Builder({ headless: true, renderOpts: { pretty: false } }); const payloadBuf = Buffer.from(builder.buildObject(taggingConfig)); const requestOptions = { method, bucketName, query, headers, ...(objectName && { objectName: objectName }) }; headers['Content-MD5'] = toMd5(payloadBuf); await this.makeRequestAsyncOmit(requestOptions, payloadBuf); } async removeTagging({ bucketName, objectName, removeOpts }) { const method = 'DELETE'; let query = 'tagging'; if (removeOpts && Object.keys(removeOpts).length && removeOpts.versionId) { query = `${query}&versionId=${removeOpts.versionId}`; } const requestOptions = { method, bucketName, objectName, query }; if (objectName) { requestOptions['objectName'] = objectName; } await this.makeRequestAsync(requestOptions, '', [200, 204]); } async setBucketTagging(bucketName, tags) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isPlainObject(tags)) { throw new errors.InvalidArgumentError('tags should be of type "object"'); } if (Object.keys(tags).length > 10) { throw new errors.InvalidArgumentError('maximum tags allowed is 10"'); } await this.setTagging({ bucketName, tags }); } async removeBucketTagging(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } await this.removeTagging({ bucketName }); } async setObjectTagging(bucketName, objectName, tags, putOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidBucketNameError('Invalid object name: ' + objectName); } if (!isPlainObject(tags)) { throw new errors.InvalidArgumentError('tags should be of type "object"'); } if (Object.keys(tags).length > 10) { throw new errors.InvalidArgumentError('Maximum tags allowed is 10"'); } await this.setTagging({ bucketName, objectName, tags, putOpts }); } async removeObjectTagging(bucketName, objectName, removeOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidBucketNameError('Invalid object name: ' + objectName); } if (removeOpts && Object.keys(removeOpts).length && !isObject(removeOpts)) { throw new errors.InvalidArgumentError('removeOpts should be of type "object"'); } await this.removeTagging({ bucketName, objectName, removeOpts }); } async selectObjectContent(bucketName, objectName, selectOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (!_.isEmpty(selectOpts)) { if (!isString(selectOpts.expression)) { throw new TypeError('sqlExpression should be of type "string"'); } if (!_.isEmpty(selectOpts.inputSerialization)) { if (!isObject(selectOpts.inputSerialization)) { throw new TypeError('inputSerialization should be of type "object"'); } } else { throw new TypeError('inputSerialization is required'); } if (!_.isEmpty(selectOpts.outputSerialization)) { if (!isObject(selectOpts.outputSerialization)) { throw new TypeError('outputSerialization should be of type "object"'); } } else { throw new TypeError('outputSerialization is required'); } } else { throw new TypeError('valid select configuration is required'); } const method = 'POST'; const query = `select&select-type=2`; const config = [{ Expression: selectOpts.expression }, { ExpressionType: selectOpts.expressionType || 'SQL' }, { InputSerialization: [selectOpts.inputSerialization] }, { OutputSerialization: [selectOpts.outputSerialization] }]; // Optional if (selectOpts.requestProgress) { config.push({ RequestProgress: selectOpts === null || selectOpts === void 0 ? void 0 : selectOpts.requestProgress }); } // Optional if (selectOpts.scanRange) { config.push({ ScanRange: selectOpts.scanRange }); } const builder = new xml2js.Builder({ rootName: 'SelectObjectContentRequest', renderOpts: { pretty: false }, headless: true }); const payload = builder.buildObject(config); const res = await this.makeRequestAsync({ method, bucketName, objectName, query }, payload); const body = await readAsBuffer(res); return parseSelectObjectContentResponse(body); } async applyBucketLifecycle(bucketName, policyConfig) { const method = 'PUT'; const query = 'lifecycle'; const headers = {}; const builder = new xml2js.Builder({ rootName: 'LifecycleConfiguration', headless: true, renderOpts: { pretty: false } }); const payload = builder.buildObject(policyConfig); headers['Content-MD5'] = toMd5(payload); await this.makeRequestAsyncOmit({ method, bucketName, query, headers }, payload); } async removeBucketLifecycle(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'DELETE'; const query = 'lifecycle'; await this.makeRequestAsyncOmit({ method, bucketName, query }, '', [204]); } async setBucketLifecycle(bucketName, lifeCycleConfig) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (_.isEmpty(lifeCycleConfig)) { await this.removeBucketLifecycle(bucketName); } else { await this.applyBucketLifecycle(bucketName, lifeCycleConfig); } } async getBucketLifecycle(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'GET'; const query = 'lifecycle'; const res = await this.makeRequestAsync({ method, bucketName, query }); const body = await readAsString(res); return xmlParsers.parseLifecycleConfig(body); } async setBucketEncryption(bucketName, encryptionConfig) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!_.isEmpty(encryptionConfig) && encryptionConfig.Rule.length > 1) { throw new errors.InvalidArgumentError('Invalid Rule length. Only one rule is allowed.: ' + encryptionConfig.Rule); } let encryptionObj = encryptionConfig; if (_.isEmpty(encryptionConfig)) { encryptionObj = { // Default MinIO Server Supported Rule Rule: [{ ApplyServerSideEncryptionByDefault: { SSEAlgorithm: 'AES256' } }] }; } const method = 'PUT'; const query = 'encryption'; const builder = new xml2js.Builder({ rootName: 'ServerSideEncryptionConfiguration', renderOpts: { pretty: false }, headless: true }); const payload = builder.buildObject(encryptionObj); const headers = {}; headers['Content-MD5'] = toMd5(payload); await this.makeRequestAsyncOmit({ method, bucketName, query, headers }, payload); } async getBucketEncryption(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'GET'; const query = 'encryption'; const res = await this.makeRequestAsync({ method, bucketName, query }); const body = await readAsString(res); return xmlParsers.parseBucketEncryptionConfig(body); } async removeBucketEncryption(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'DELETE'; const query = 'encryption'; await this.makeRequestAsyncOmit({ method, bucketName, query }, '', [204]); } async getObjectRetention(bucketName, objectName, getOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } if (getOpts && !isObject(getOpts)) { throw new errors.InvalidArgumentError('getOpts should be of type "object"'); } else if (getOpts !== null && getOpts !== void 0 && getOpts.versionId && !isString(getOpts.versionId)) { throw new errors.InvalidArgumentError('versionId should be of type "string"'); } const method = 'GET'; let query = 'retention'; if (getOpts !== null && getOpts !== void 0 && getOpts.versionId) { query += `&versionId=${getOpts.versionId}`; } const res = await this.makeRequestAsync({ method, bucketName, objectName, query }); const body = await readAsString(res); return xmlParsers.parseObjectRetentionConfig(body); } async removeObjects(bucketName, objectsList) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!Array.isArray(objectsList)) { throw new errors.InvalidArgumentError('objectsList should be a list'); } const runDeleteObjects = async batch => { const delObjects = batch.map(value => { return isObject(value) ? { Key: value.name, VersionId: value.versionId } : { Key: value }; }); const remObjects = { Delete: { Quiet: true, Object: delObjects } }; const payload = Buffer.from(new xml2js.Builder({ headless: true }).buildObject(remObjects)); const headers = { 'Content-MD5': toMd5(payload) }; const res = await this.makeRequestAsync({ method: 'POST', bucketName, query: 'delete', headers }, payload); const body = await readAsString(res); return xmlParsers.removeObjectsParser(body); }; const maxEntries = 1000; // max entries accepted in server for DeleteMultipleObjects API. // Client side batching const batches = []; for (let i = 0; i < objectsList.length; i += maxEntries) { batches.push(objectsList.slice(i, i + maxEntries)); } const batchResults = await Promise.all(batches.map(runDeleteObjects)); return batchResults.flat(); } async removeIncompleteUpload(bucketName, objectName) { if (!isValidBucketName(bucketName)) { throw new errors.IsValidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } const removeUploadId = await this.findUploadId(bucketName, objectName); const method = 'DELETE'; const query = `uploadId=${removeUploadId}`; await this.makeRequestAsyncOmit({ method, bucketName, objectName, query }, '', [204]); } async copyObjectV1(targetBucketName, targetObjectName, sourceBucketNameAndObjectName, conditions) { if (typeof conditions == 'function') { conditions = null; } if (!isValidBucketName(targetBucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + targetBucketName); } if (!isValidObjectName(targetObjectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${targetObjectName}`); } if (!isString(sourceBucketNameAndObjectName)) { throw new TypeError('sourceBucketNameAndObjectName should be of type "string"'); } if (sourceBucketNameAndObjectName === '') { throw new errors.InvalidPrefixError(`Empty source prefix`); } if (conditions != null && !(conditions instanceof CopyConditions)) { throw new TypeError('conditions should be of type "CopyConditions"'); } const headers = {}; headers['x-amz-copy-source'] = uriResourceEscape(sourceBucketNameAndObjectName); if (conditions) { if (conditions.modified !== '') { headers['x-amz-copy-source-if-modified-since'] = conditions.modified; } if (conditions.unmodified !== '') { headers['x-amz-copy-source-if-unmodified-since'] = conditions.unmodified; } if (conditions.matchETag !== '') { headers['x-amz-copy-source-if-match'] = conditions.matchETag; } if (conditions.matchETagExcept !== '') { headers['x-amz-copy-source-if-none-match'] = conditions.matchETagExcept; } } const method = 'PUT'; const res = await this.makeRequestAsync({ method, bucketName: targetBucketName, objectName: targetObjectName, headers }); const body = await readAsString(res); return xmlParsers.parseCopyObject(body); } async copyObjectV2(sourceConfig, destConfig) { if (!(sourceConfig instanceof CopySourceOptions)) { throw new errors.InvalidArgumentError('sourceConfig should of type CopySourceOptions '); } if (!(destConfig instanceof CopyDestinationOptions)) { throw new errors.InvalidArgumentError('destConfig should of type CopyDestinationOptions '); } if (!destConfig.validate()) { return Promise.reject(); } if (!destConfig.validate()) { return Promise.reject(); } const headers = Object.assign({}, sourceConfig.getHeaders(), destConfig.getHeaders()); const bucketName = destConfig.Bucket; const objectName = destConfig.Object; const method = 'PUT'; const res = await this.makeRequestAsync({ method, bucketName, objectName, headers }); const body = await readAsString(res); const copyRes = xmlParsers.parseCopyObject(body); const resHeaders = res.headers; const sizeHeaderValue = resHeaders && resHeaders['content-length']; const size = typeof sizeHeaderValue === 'number' ? sizeHeaderValue : undefined; return { Bucket: destConfig.Bucket, Key: destConfig.Object, LastModified: copyRes.lastModified, MetaData: extractMetadata(resHeaders), VersionId: getVersionId(resHeaders), SourceVersionId: getSourceVersionId(resHeaders), Etag: sanitizeETag(resHeaders.etag), Size: size }; } async copyObject(...allArgs) { if (typeof allArgs[0] === 'string') { const [targetBucketName, targetObjectName, sourceBucketNameAndObjectName, conditions] = allArgs; return await this.copyObjectV1(targetBucketName, targetObjectName, sourceBucketNameAndObjectName, conditions); } const [source, dest] = allArgs; return await this.copyObjectV2(source, dest); } async uploadPart(partConfig, payload) { const { bucketName, objectName, uploadID, partNumber, headers } = partConfig; const method = 'PUT'; const query = `uploadId=${uploadID}&partNumber=${partNumber}`; const requestOptions = { method, bucketName, objectName: objectName, query, headers }; const res = await this.makeRequestAsync(requestOptions, payload); const body = await readAsString(res); const partRes = uploadPartParser(body); const partEtagVal = sanitizeETag(res.headers.etag) || sanitizeETag(partRes.ETag); return { etag: partEtagVal, key: objectName, part: partNumber }; } async composeObject(destObjConfig, sourceObjList, { maxConcurrency = 10 } = {}) { const sourceFilesLength = sourceObjList.length; if (!Array.isArray(sourceObjList)) { throw new errors.InvalidArgumentError('sourceConfig should an array of CopySourceOptions '); } if (!(destObjConfig instanceof CopyDestinationOptions)) { throw new errors.InvalidArgumentError('destConfig should of type CopyDestinationOptions '); } if (sourceFilesLength < 1 || sourceFilesLength > PART_CONSTRAINTS.MAX_PARTS_COUNT) { throw new errors.InvalidArgumentError(`"There must be as least one and up to ${PART_CONSTRAINTS.MAX_PARTS_COUNT} source objects.`); } for (let i = 0; i < sourceFilesLength; i++) { const sObj = sourceObjList[i]; if (!sObj.validate()) { return false; } } if (!destObjConfig.validate()) { return false; } const getStatOptions = srcConfig => { let statOpts = {}; if (!_.isEmpty(srcConfig.VersionID)) { statOpts = { versionId: srcConfig.VersionID }; } return statOpts; }; const srcObjectSizes = []; let totalSize = 0; let totalParts = 0; const sourceObjStats = sourceObjList.map(srcItem => this.statObject(srcItem.Bucket, srcItem.Object, getStatOptions(srcItem))); const srcObjectInfos = await Promise.all(sourceObjStats); const validatedStats = srcObjectInfos.map((resItemStat, index) => { const srcConfig = sourceObjList[index]; let srcCopySize = resItemStat.size; // Check if a segment is specified, and if so, is the // segment within object bounds? if (srcConfig && srcConfig.MatchRange) { // Since range is specified, // 0 <= src.srcStart <= src.srcEnd // so only invalid case to check is: const srcStart = srcConfig.Start; const srcEnd = srcConfig.End; if (srcEnd >= srcCopySize || srcStart < 0) { throw new errors.InvalidArgumentError(`CopySrcOptions ${index} has invalid segment-to-copy [${srcStart}, ${srcEnd}] (size is ${srcCopySize})`); } srcCopySize = srcEnd - srcStart + 1; } // Only the last source may be less than `absMinPartSize` if (srcCopySize < PART_CONSTRAINTS.ABS_MIN_PART_SIZE && index < sourceFilesLength - 1) { throw new errors.InvalidArgumentError(`CopySrcOptions ${index} is too small (${srcCopySize}) and it is not the last part.`); } // Is data to copy too large? totalSize += srcCopySize; if (totalSize > PART_CONSTRAINTS.MAX_MULTIPART_PUT_OBJECT_SIZE) { throw new errors.InvalidArgumentError(`Cannot compose an object of size ${totalSize} (> 5TiB)`); } // record source size srcObjectSizes[index] = srcCopySize; // calculate parts needed for current source totalParts += partsRequired(srcCopySize); // Do we need more parts than we are allowed? if (totalParts > PART_CONSTRAINTS.MAX_PARTS_COUNT) { throw new errors.InvalidArgumentError(`Your proposed compose object requires more than ${PART_CONSTRAINTS.MAX_PARTS_COUNT} parts`); } return resItemStat; }); if (totalParts === 1 && totalSize <= PART_CONSTRAINTS.MAX_PART_SIZE || totalSize === 0) { return await this.copyObject(sourceObjList[0], destObjConfig); // use copyObjectV2 } // preserve etag to avoid modification of object while copying. for (let i = 0; i < sourceFilesLength; i++) { ; sourceObjList[i].MatchETag = validatedStats[i].etag; } const splitPartSizeList = validatedStats.map((resItemStat, idx) => { return calculateEvenSplits(srcObjectSizes[idx], sourceObjList[idx]); }); const getUploadPartConfigList = uploadId => { const uploadPartConfigList = []; splitPartSizeList.forEach((splitSize, splitIndex) => { if (splitSize) { const { startIndex: startIdx, endIndex: endIdx, objInfo: objConfig } = splitSize; const partIndex = splitIndex + 1; // part index starts from 1. const totalUploads = Array.from(startIdx); const headers = sourceObjList[splitIndex].getHeaders(); totalUploads.forEach((splitStart, upldCtrIdx) => { const splitEnd = endIdx[upldCtrIdx]; const sourceObj = `${objConfig.Bucket}/${objConfig.Object}`; headers['x-amz-copy-source'] = `${sourceObj}`; headers['x-amz-copy-source-range'] = `bytes=${splitStart}-${splitEnd}`; const uploadPartConfig = { bucketName: destObjConfig.Bucket, objectName: destObjConfig.Object, uploadID: uploadId, partNumber: partIndex, headers: headers, sourceObj: sourceObj }; uploadPartConfigList.push(uploadPartConfig); }); } }); return uploadPartConfigList; }; const uploadAllParts = async uploadList => { const partUploads = []; // Process upload parts in batches to avoid too many concurrent requests for (const batch of _.chunk(uploadList, maxConcurrency)) { const batchResults = await Promise.all(batch.map(item => this.uploadPart(item))); partUploads.push(...batchResults); } // Process results here if needed return partUploads; }; const performUploadParts = async uploadId => { const uploadList = getUploadPartConfigList(uploadId); const partsRes = await uploadAllParts(uploadList); return partsRes.map(partCopy => ({ etag: partCopy.etag, part: partCopy.part })); }; const newUploadHeaders = destObjConfig.getHeaders(); const uploadId = await this.initiateNewMultipartUpload(destObjConfig.Bucket, destObjConfig.Object, newUploadHeaders); try { const partsDone = await performUploadParts(uploadId); return await this.completeMultipartUpload(destObjConfig.Bucket, destObjConfig.Object, uploadId, partsDone); } catch (err) { return await this.abortMultipartUpload(destObjConfig.Bucket, destObjConfig.Object, uploadId); } } async presignedUrl(method, bucketName, objectName, expires, reqParams, requestDate) { var _requestDate; if (this.anonymous) { throw new errors.AnonymousRequestError(`Presigned ${method} url cannot be generated for anonymous requests`); } if (!expires) { expires = PRESIGN_EXPIRY_DAYS_MAX; } if (!reqParams) { reqParams = {}; } if (!requestDate) { requestDate = new Date(); } // Type assertions if (expires && typeof expires !== 'number') { throw new TypeError('expires should be of type "number"'); } if (reqParams && typeof reqParams !== 'object') { throw new TypeError('reqParams should be of type "object"'); } if (requestDate && !(requestDate instanceof Date) || requestDate && isNaN((_requestDate = requestDate) === null || _requestDate === void 0 ? void 0 : _requestDate.getTime())) { throw new TypeError('requestDate should be of type "Date" and valid'); } const query = reqParams ? qs.stringify(reqParams) : undefined; try { const region = await this.getBucketRegionAsync(bucketName); await this.checkAndRefreshCreds(); const reqOptions = this.getRequestOptions({ method, region, bucketName, objectName, query }); return presignSignatureV4(reqOptions, this.accessKey, this.secretKey, this.sessionToken, region, requestDate, expires); } catch (err) { if (err instanceof errors.InvalidBucketNameError) { throw new errors.InvalidArgumentError(`Unable to get bucket region for ${bucketName}.`); } throw err; } } async presignedGetObject(bucketName, objectName, expires, respHeaders, requestDate) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } const validRespHeaders = ['response-content-type', 'response-content-language', 'response-expires', 'response-cache-control', 'response-content-disposition', 'response-content-encoding']; validRespHeaders.forEach(header => { // @ts-ignore if (respHeaders !== undefined && respHeaders[header] !== undefined && !isString(respHeaders[header])) { throw new TypeError(`response header ${header} should be of type "string"`); } }); return this.presignedUrl('GET', bucketName, objectName, expires, respHeaders, requestDate); } async presignedPutObject(bucketName, objectName, expires) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`); } if (!isValidObjectName(objectName)) { throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`); } return this.presignedUrl('PUT', bucketName, objectName, expires); } newPostPolicy() { return new PostPolicy(); } async presignedPostPolicy(postPolicy) { if (this.anonymous) { throw new errors.AnonymousRequestError('Presigned POST policy cannot be generated for anonymous requests'); } if (!isObject(postPolicy)) { throw new TypeError('postPolicy should be of type "object"'); } const bucketName = postPolicy.formData.bucket; try { const region = await this.getBucketRegionAsync(bucketName); const date = new Date(); const dateStr = makeDateLong(date); await this.checkAndRefreshCreds(); if (!postPolicy.policy.expiration) { // 'expiration' is mandatory field for S3. // Set default expiration date of 7 days. const expires = new Date(); expires.setSeconds(PRESIGN_EXPIRY_DAYS_MAX); postPolicy.setExpires(expires); } postPolicy.policy.conditions.push(['eq', '$x-amz-date', dateStr]); postPolicy.formData['x-amz-date'] = dateStr; postPolicy.policy.conditions.push(['eq', '$x-amz-algorithm', 'AWS4-HMAC-SHA256']); postPolicy.formData['x-amz-algorithm'] = 'AWS4-HMAC-SHA256'; postPolicy.policy.conditions.push(['eq', '$x-amz-credential', this.accessKey + '/' + getScope(region, date)]); postPolicy.formData['x-amz-credential'] = this.accessKey + '/' + getScope(region, date); if (this.sessionToken) { postPolicy.policy.conditions.push(['eq', '$x-amz-security-token', this.sessionToken]); postPolicy.formData['x-amz-security-token'] = this.sessionToken; } const policyBase64 = Buffer.from(JSON.stringify(postPolicy.policy)).toString('base64'); postPolicy.formData.policy = policyBase64; postPolicy.formData['x-amz-signature'] = postPresignSignatureV4(region, date, this.secretKey, policyBase64); const opts = { region: region, bucketName: bucketName, method: 'POST' }; const reqOptions = this.getRequestOptions(opts); const portStr = this.port == 80 || this.port === 443 ? '' : `:${this.port.toString()}`; const urlStr = `${reqOptions.protocol}//${reqOptions.host}${portStr}${reqOptions.path}`; return { postURL: urlStr, formData: postPolicy.formData }; } catch (err) { if (err instanceof errors.InvalidBucketNameError) { throw new errors.InvalidArgumentError(`Unable to get bucket region for ${bucketName}.`); } throw err; } } // list a batch of objects async listObjectsQuery(bucketName, prefix, marker, listQueryOpts) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isString(prefix)) { throw new TypeError('prefix should be of type "string"'); } if (marker && !isString(marker)) { throw new TypeError('marker should be of type "string"'); } if (listQueryOpts && !isObject(listQueryOpts)) { throw new TypeError('listQueryOpts should be of type "object"'); } let { Delimiter, MaxKeys, IncludeVersion, versionIdMarker, keyMarker } = listQueryOpts; if (!isString(Delimiter)) { throw new TypeError('Delimiter should be of type "string"'); } if (!isNumber(MaxKeys)) { throw new TypeError('MaxKeys should be of type "number"'); } const queries = []; // escape every value in query string, except maxKeys queries.push(`prefix=${uriEscape(prefix)}`); queries.push(`delimiter=${uriEscape(Delimiter)}`); queries.push(`encoding-type=url`); if (IncludeVersion) { queries.push(`versions`); } if (IncludeVersion) { // v1 version listing.. if (keyMarker) { queries.push(`key-marker=${keyMarker}`); } if (versionIdMarker) { queries.push(`version-id-marker=${versionIdMarker}`); } } else if (marker) { marker = uriEscape(marker); queries.push(`marker=${marker}`); } // no need to escape maxKeys if (MaxKeys) { if (MaxKeys >= 1000) { MaxKeys = 1000; } queries.push(`max-keys=${MaxKeys}`); } queries.sort(); let query = ''; if (queries.length > 0) { query = `${queries.join('&')}`; } const method = 'GET'; const res = await this.makeRequestAsync({ method, bucketName, query }); const body = await readAsString(res); const listQryList = parseListObjects(body); return listQryList; } listObjects(bucketName, prefix, recursive, listOpts) { if (prefix === undefined) { prefix = ''; } if (recursive === undefined) { recursive = false; } if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidPrefix(prefix)) { throw new errors.InvalidPrefixError(`Invalid prefix : ${prefix}`); } if (!isString(prefix)) { throw new TypeError('prefix should be of type "string"'); } if (!isBoolean(recursive)) { throw new TypeError('recursive should be of type "boolean"'); } if (listOpts && !isObject(listOpts)) { throw new TypeError('listOpts should be of type "object"'); } let marker = ''; let keyMarker = ''; let versionIdMarker = ''; let objects = []; let ended = false; const readStream = new stream.Readable({ objectMode: true }); readStream._read = async () => { // push one object per _read() if (objects.length) { readStream.push(objects.shift()); return; } if (ended) { return readStream.push(null); } try { const listQueryOpts = { Delimiter: recursive ? '' : '/', // if recursive is false set delimiter to '/' MaxKeys: 1000, IncludeVersion: listOpts === null || listOpts === void 0 ? void 0 : listOpts.IncludeVersion, // version listing specific options keyMarker: keyMarker, versionIdMarker: versionIdMarker }; const result = await this.listObjectsQuery(bucketName, prefix, marker, listQueryOpts); if (result.isTruncated) { marker = result.nextMarker || undefined; if (result.keyMarker) { keyMarker = result.keyMarker; } if (result.versionIdMarker) { versionIdMarker = result.versionIdMarker; } } else { ended = true; } if (result.objects) { objects = result.objects; } // @ts-ignore readStream._read(); } catch (err) { readStream.emit('error', err); } }; return readStream; } async listObjectsV2Query(bucketName, prefix, continuationToken, delimiter, maxKeys, startAfter) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isString(prefix)) { throw new TypeError('prefix should be of type "string"'); } if (!isString(continuationToken)) { throw new TypeError('continuationToken should be of type "string"'); } if (!isString(delimiter)) { throw new TypeError('delimiter should be of type "string"'); } if (!isNumber(maxKeys)) { throw new TypeError('maxKeys should be of type "number"'); } if (!isString(startAfter)) { throw new TypeError('startAfter should be of type "string"'); } const queries = []; queries.push(`list-type=2`); queries.push(`encoding-type=url`); queries.push(`prefix=${uriEscape(prefix)}`); queries.push(`delimiter=${uriEscape(delimiter)}`); if (continuationToken) { queries.push(`continuation-token=${uriEscape(continuationToken)}`); } if (startAfter) { queries.push(`start-after=${uriEscape(startAfter)}`); } if (maxKeys) { if (maxKeys >= 1000) { maxKeys = 1000; } queries.push(`max-keys=${maxKeys}`); } queries.sort(); let query = ''; if (queries.length > 0) { query = `${queries.join('&')}`; } const method = 'GET'; const res = await this.makeRequestAsync({ method, bucketName, query }); const body = await readAsString(res); return parseListObjectsV2(body); } listObjectsV2(bucketName, prefix, recursive, startAfter) { if (prefix === undefined) { prefix = ''; } if (recursive === undefined) { recursive = false; } if (startAfter === undefined) { startAfter = ''; } if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isValidPrefix(prefix)) { throw new errors.InvalidPrefixError(`Invalid prefix : ${prefix}`); } if (!isString(prefix)) { throw new TypeError('prefix should be of type "string"'); } if (!isBoolean(recursive)) { throw new TypeError('recursive should be of type "boolean"'); } if (!isString(startAfter)) { throw new TypeError('startAfter should be of type "string"'); } const delimiter = recursive ? '' : '/'; const prefixStr = prefix; const startAfterStr = startAfter; let continuationToken = ''; let objects = []; let ended = false; const readStream = new stream.Readable({ objectMode: true }); readStream._read = async () => { if (objects.length) { readStream.push(objects.shift()); return; } if (ended) { return readStream.push(null); } try { const result = await this.listObjectsV2Query(bucketName, prefixStr, continuationToken, delimiter, 1000, startAfterStr); if (result.isTruncated) { continuationToken = result.nextContinuationToken; } else { ended = true; } objects = result.objects; // @ts-ignore readStream._read(); } catch (err) { readStream.emit('error', err); } }; return readStream; } async setBucketNotification(bucketName, config) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } if (!isObject(config)) { throw new TypeError('notification config should be of type "Object"'); } const method = 'PUT'; const query = 'notification'; const builder = new xml2js.Builder({ rootName: 'NotificationConfiguration', renderOpts: { pretty: false }, headless: true }); const payload = builder.buildObject(config); await this.makeRequestAsyncOmit({ method, bucketName, query }, payload); } async removeAllBucketNotification(bucketName) { await this.setBucketNotification(bucketName, new NotificationConfig()); } async getBucketNotification(bucketName) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName); } const method = 'GET'; const query = 'notification'; const res = await this.makeRequestAsync({ method, bucketName, query }); const body = await readAsString(res); return parseBucketNotification(body); } listenBucketNotification(bucketName, prefix, suffix, events) { if (!isValidBucketName(bucketName)) { throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`); } if (!isString(prefix)) { throw new TypeError('prefix must be of type string'); } if (!isString(suffix)) { throw new TypeError('suffix must be of type string'); } if (!Array.isArray(events)) { throw new TypeError('events must be of type Array'); } const listener = new NotificationPoller(this, bucketName, prefix, suffix, events); listener.start(); return listener; } } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["crypto","fs","http","https","path","stream","async","BlockStream2","isBrowser","_","qs","xml2js","CredentialProvider","errors","CopyDestinationOptions","CopySourceOptions","DEFAULT_REGION","LEGAL_HOLD_STATUS","PRESIGN_EXPIRY_DAYS_MAX","RETENTION_MODES","RETENTION_VALIDITY_UNITS","NotificationConfig","NotificationPoller","postPresignSignatureV4","presignSignatureV4","signV4","fsp","streamPromise","CopyConditions","Extensions","calculateEvenSplits","extractMetadata","getContentLength","getScope","getSourceVersionId","getVersionId","hashBinary","insertContentType","isAmazonEndpoint","isBoolean","isDefined","isEmpty","isNumber","isObject","isPlainObject","isReadableStream","isString","isValidBucketName","isValidEndpoint","isValidObjectName","isValidPort","isValidPrefix","isVirtualHostStyle","makeDateLong","PART_CONSTRAINTS","partsRequired","prependXAMZMeta","readableStream","sanitizeETag","toMd5","toSha256","uriEscape","uriResourceEscape","joinHostPort","PostPolicy","requestWithRetry","drainResponse","readAsBuffer","readAsString","getS3Endpoint","parseBucketNotification","parseCompleteMultipart","parseInitiateMultipart","parseListObjects","parseListObjectsV2","parseObjectLegalHoldConfig","parseSelectObjectContentResponse","uploadPartParser","xmlParsers","xml","Builder","renderOpts","pretty","headless","Package","version","requestOptionProperties","TypedClient","partSize","maximumPartSize","maxObjectSize","constructor","params","secure","undefined","Error","useSSL","port","endPoint","InvalidEndpointError","InvalidArgumentError","region","host","toLowerCase","protocol","transport","transportAgent","globalAgent","libraryComments","process","platform","arch","libraryAgent","userAgent","pathStyle","accessKey","secretKey","sessionToken","anonymous","credentialsProvider","regionMap","overRidePartSize","enableSHA256","s3AccelerateEndpoint","reqOptions","clientExtensions","retryOptions","disableRetry","extensions","setS3TransferAccelerate","setRequestOptions","options","TypeError","pick","getAccelerateEndPointIfSet","bucketName","objectName","includes","setAppInfo","appName","appVersion","trim","getRequestOptions","opts","method","headers","query","agent","virtualHostStyle","accelerateEndPoint","k","v","Object","entries","assign","mapValues","pickBy","toString","setCredentialsProvider","checkAndRefreshCreds","credentialsConf","getCredentials","getAccessKey","getSecretKey","getSessionToken","e","cause","logHTTP","response","err","logStream","logHeaders","forEach","redactor","RegExp","replace","write","statusCode","errJSON","JSON","stringify","traceOn","stdout","traceOff","makeRequestAsync","payload","expectedCodes","length","sha256sum","makeRequestStreamAsync","makeRequestAsyncOmit","statusCodes","res","body","Buffer","isBuffer","getBucketRegionAsync","date","Date","authorization","maximumRetryCount","baseDelayMs","maximumDelayMs","parseResponseError","InvalidBucketNameError","cached","extractRegionAsync","parseBucketRegion","S3Error","errCode","code","errRegion","name","Region","makeRequest","returnResponse","cb","prom","then","result","makeRequestStream","executor","getBucketRegion","makeBucket","makeOpts","buildObject","CreateBucketConfiguration","$","xmlns","LocationConstraint","ObjectLocking","finalRegion","requestOpt","bucketExists","removeBucket","getObject","getOpts","InvalidObjectNameError","getPartialObject","offset","range","sseHeaders","SSECustomerAlgorithm","SSECustomerKey","SSECustomerKeyMD5","expectedStatusCodes","push","fGetObject","filePath","downloadToTmpFile","partFileStream","objStat","statObject","encodedEtag","from","etag","partFile","mkdir","dirname","recursive","stats","stat","size","createWriteStream","flags","downloadStream","pipeline","rename","statOpts","statOptDef","parseInt","metaData","lastModified","versionId","removeObject","removeOpts","governanceBypass","forceDelete","queryParams","listIncompleteUploads","bucket","prefix","InvalidPrefixError","delimiter","keyMarker","uploadIdMarker","uploads","ended","readStream","Readable","objectMode","_read","shift","listIncompleteUploadsQuery","prefixes","eachSeries","upload","listParts","key","uploadId","parts","reduce","acc","item","emit","isTruncated","nextKeyMarker","nextUploadIdMarker","queries","maxUploads","sort","unshift","join","parseListMultipart","initiateNewMultipartUpload","abortMultipartUpload","requestOptions","findUploadId","_latestUpload","latestUpload","initiated","getTime","completeMultipartUpload","etags","builder","CompleteMultipartUpload","Part","map","PartNumber","part","ETag","errMessage","marker","listPartsQuery","parseListParts","listBuckets","regionConf","httpRes","xmlResult","parseListBucket","calculatePartSize","fPutObject","putObject","createReadStream","statSize","uploadBuffer","buf","uploadStream","md5sum","oldParts","eTags","previousUploadId","oldTags","chunkier","zeroPadding","o","Promise","all","resolve","reject","pipe","on","partNumber","chunk","md5","createHash","update","digest","oldPart","removeBucketReplication","setBucketReplication","replicationConfig","role","rules","replicationParamsConfig","ReplicationConfiguration","Role","Rule","getBucketReplication","parseReplicationConfig","getObjectLegalHold","keys","strRes","setObjectLegalHold","setOpts","status","ENABLED","DISABLED","config","Status","rootName","getBucketTagging","parseTagging","getObjectTagging","setBucketPolicy","policy","InvalidBucketPolicyError","getBucketPolicy","putObjectRetention","retentionOpts","mode","COMPLIANCE","GOVERNANCE","retainUntilDate","Mode","RetainUntilDate","getObjectLockConfig","parseObjectLockConfig","setObjectLockConfig","lockConfigOpts","retentionModes","validUnits","DAYS","YEARS","unit","validity","ObjectLockEnabled","configKeys","isAllKeysSet","every","lck","DefaultRetention","Days","Years","getBucketVersioning","parseBucketVersioningConfig","setBucketVersioning","versionConfig","setTagging","taggingParams","tags","putOpts","tagsList","value","Key","Value","taggingConfig","Tagging","TagSet","Tag","payloadBuf","removeTagging","setBucketTagging","removeBucketTagging","setObjectTagging","removeObjectTagging","selectObjectContent","selectOpts","expression","inputSerialization","outputSerialization","Expression","ExpressionType","expressionType","InputSerialization","OutputSerialization","requestProgress","RequestProgress","scanRange","ScanRange","applyBucketLifecycle","policyConfig","removeBucketLifecycle","setBucketLifecycle","lifeCycleConfig","getBucketLifecycle","parseLifecycleConfig","setBucketEncryption","encryptionConfig","encryptionObj","ApplyServerSideEncryptionByDefault","SSEAlgorithm","getBucketEncryption","parseBucketEncryptionConfig","removeBucketEncryption","getObjectRetention","parseObjectRetentionConfig","removeObjects","objectsList","Array","isArray","runDeleteObjects","batch","delObjects","VersionId","remObjects","Delete","Quiet","removeObjectsParser","maxEntries","batches","i","slice","batchResults","flat","removeIncompleteUpload","IsValidBucketNameError","removeUploadId","copyObjectV1","targetBucketName","targetObjectName","sourceBucketNameAndObjectName","conditions","modified","unmodified","matchETag","matchETagExcept","parseCopyObject","copyObjectV2","sourceConfig","destConfig","validate","getHeaders","Bucket","copyRes","resHeaders","sizeHeaderValue","LastModified","MetaData","SourceVersionId","Etag","Size","copyObject","allArgs","source","dest","uploadPart","partConfig","uploadID","partRes","partEtagVal","composeObject","destObjConfig","sourceObjList","maxConcurrency","sourceFilesLength","MAX_PARTS_COUNT","sObj","getStatOptions","srcConfig","VersionID","srcObjectSizes","totalSize","totalParts","sourceObjStats","srcItem","srcObjectInfos","validatedStats","resItemStat","index","srcCopySize","MatchRange","srcStart","Start","srcEnd","End","ABS_MIN_PART_SIZE","MAX_MULTIPART_PUT_OBJECT_SIZE","MAX_PART_SIZE","MatchETag","splitPartSizeList","idx","getUploadPartConfigList","uploadPartConfigList","splitSize","splitIndex","startIndex","startIdx","endIndex","endIdx","objInfo","objConfig","partIndex","totalUploads","splitStart","upldCtrIdx","splitEnd","sourceObj","uploadPartConfig","uploadAllParts","uploadList","partUploads","performUploadParts","partsRes","partCopy","newUploadHeaders","partsDone","presignedUrl","expires","reqParams","requestDate","_requestDate","AnonymousRequestError","isNaN","presignedGetObject","respHeaders","validRespHeaders","header","presignedPutObject","newPostPolicy","presignedPostPolicy","postPolicy","formData","dateStr","expiration","setSeconds","setExpires","policyBase64","portStr","urlStr","postURL","listObjectsQuery","listQueryOpts","Delimiter","MaxKeys","IncludeVersion","versionIdMarker","listQryList","listObjects","listOpts","objects","nextMarker","listObjectsV2Query","continuationToken","maxKeys","startAfter","listObjectsV2","prefixStr","startAfterStr","nextContinuationToken","setBucketNotification","removeAllBucketNotification","getBucketNotification","listenBucketNotification","suffix","events","listener","start"],"sources":["client.ts"],"sourcesContent":["import * as crypto from 'node:crypto'\nimport * as fs from 'node:fs'\nimport type { IncomingHttpHeaders } from 'node:http'\nimport * as http from 'node:http'\nimport * as https from 'node:https'\nimport * as path from 'node:path'\nimport * as stream from 'node:stream'\n\nimport * as async from 'async'\nimport BlockStream2 from 'block-stream2'\nimport { isBrowser } from 'browser-or-node'\nimport _ from 'lodash'\nimport * as qs from 'query-string'\nimport xml2js from 'xml2js'\n\nimport { CredentialProvider } from '../CredentialProvider.ts'\nimport * as errors from '../errors.ts'\nimport type { SelectResults } from '../helpers.ts'\nimport {\n  CopyDestinationOptions,\n  CopySourceOptions,\n  DEFAULT_REGION,\n  LEGAL_HOLD_STATUS,\n  PRESIGN_EXPIRY_DAYS_MAX,\n  RETENTION_MODES,\n  RETENTION_VALIDITY_UNITS,\n} from '../helpers.ts'\nimport type { NotificationEvent } from '../notification.ts'\nimport { NotificationConfig, NotificationPoller } from '../notification.ts'\nimport { postPresignSignatureV4, presignSignatureV4, signV4 } from '../signing.ts'\nimport { fsp, streamPromise } from './async.ts'\nimport { CopyConditions } from './copy-conditions.ts'\nimport { Extensions } from './extensions.ts'\nimport {\n  calculateEvenSplits,\n  extractMetadata,\n  getContentLength,\n  getScope,\n  getSourceVersionId,\n  getVersionId,\n  hashBinary,\n  insertContentType,\n  isAmazonEndpoint,\n  isBoolean,\n  isDefined,\n  isEmpty,\n  isNumber,\n  isObject,\n  isPlainObject,\n  isReadableStream,\n  isString,\n  isValidBucketName,\n  isValidEndpoint,\n  isValidObjectName,\n  isValidPort,\n  isValidPrefix,\n  isVirtualHostStyle,\n  makeDateLong,\n  PART_CONSTRAINTS,\n  partsRequired,\n  prependXAMZMeta,\n  readableStream,\n  sanitizeETag,\n  toMd5,\n  toSha256,\n  uriEscape,\n  uriResourceEscape,\n} from './helper.ts'\nimport { joinHostPort } from './join-host-port.ts'\nimport { PostPolicy } from './post-policy.ts'\nimport { requestWithRetry } from './request.ts'\nimport { drainResponse, readAsBuffer, readAsString } from './response.ts'\nimport type { Region } from './s3-endpoints.ts'\nimport { getS3Endpoint } from './s3-endpoints.ts'\nimport type {\n  Binary,\n  BucketItem,\n  BucketItemFromList,\n  BucketItemStat,\n  BucketStream,\n  BucketVersioningConfiguration,\n  CopyObjectParams,\n  CopyObjectResult,\n  CopyObjectResultV2,\n  EncryptionConfig,\n  GetObjectLegalHoldOptions,\n  GetObjectOpts,\n  GetObjectRetentionOpts,\n  IncompleteUploadedBucketItem,\n  IRequest,\n  ItemBucketMetadata,\n  LifecycleConfig,\n  LifeCycleConfigParam,\n  ListObjectQueryOpts,\n  ListObjectQueryRes,\n  ListObjectV2Res,\n  NotificationConfigResult,\n  ObjectInfo,\n  ObjectLockConfigParam,\n  ObjectLockInfo,\n  ObjectMetaData,\n  ObjectRetentionInfo,\n  PostPolicyResult,\n  PreSignRequestParams,\n  PutObjectLegalHoldOptions,\n  PutTaggingParams,\n  RemoveObjectsParam,\n  RemoveObjectsRequestEntry,\n  RemoveObjectsResponse,\n  RemoveTaggingParams,\n  ReplicationConfig,\n  ReplicationConfigOpts,\n  RequestHeaders,\n  ResponseHeader,\n  ResultCallback,\n  Retention,\n  SelectOptions,\n  StatObjectOpts,\n  Tag,\n  TaggingOpts,\n  Tags,\n  Transport,\n  UploadedObjectInfo,\n  UploadPartConfig,\n} from './type.ts'\nimport type { ListMultipartResult, UploadedPart } from './xml-parser.ts'\nimport {\n  parseBucketNotification,\n  parseCompleteMultipart,\n  parseInitiateMultipart,\n  parseListObjects,\n  parseListObjectsV2,\n  parseObjectLegalHoldConfig,\n  parseSelectObjectContentResponse,\n  uploadPartParser,\n} from './xml-parser.ts'\nimport * as xmlParsers from './xml-parser.ts'\n\nconst xml = new xml2js.Builder({ renderOpts: { pretty: false }, headless: true })\n\n// will be replaced by bundler.\nconst Package = { version: process.env.MINIO_JS_PACKAGE_VERSION || 'development' }\n\nconst requestOptionProperties = [\n  'agent',\n  'ca',\n  'cert',\n  'ciphers',\n  'clientCertEngine',\n  'crl',\n  'dhparam',\n  'ecdhCurve',\n  'family',\n  'honorCipherOrder',\n  'key',\n  'passphrase',\n  'pfx',\n  'rejectUnauthorized',\n  'secureOptions',\n  'secureProtocol',\n  'servername',\n  'sessionIdContext',\n] as const\n\nexport interface RetryOptions {\n  /**\n   * If this set to true, it will take precedence over all other retry options.\n   * @default false\n   */\n  disableRetry?: boolean\n  /**\n   * The maximum amount of retries for a request.\n   * @default 1\n   */\n  maximumRetryCount?: number\n  /**\n   * The minimum duration (in milliseconds) for the exponential backoff algorithm.\n   * @default 100\n   */\n  baseDelayMs?: number\n  /**\n   * The maximum duration (in milliseconds) for the exponential backoff algorithm.\n   * @default 60000\n   */\n  maximumDelayMs?: number\n}\n\nexport interface ClientOptions {\n  endPoint: string\n  accessKey?: string\n  secretKey?: string\n  useSSL?: boolean\n  port?: number\n  region?: Region\n  transport?: Transport\n  sessionToken?: string\n  partSize?: number\n  pathStyle?: boolean\n  credentialsProvider?: CredentialProvider\n  s3AccelerateEndpoint?: string\n  transportAgent?: http.Agent\n  retryOptions?: RetryOptions\n}\n\nexport type RequestOption = Partial<IRequest> & {\n  method: string\n  bucketName?: string\n  objectName?: string\n  query?: string\n  pathStyle?: boolean\n}\n\nexport type NoResultCallback = (error: unknown) => void\n\nexport interface MakeBucketOpt {\n  ObjectLocking?: boolean\n}\n\nexport interface RemoveOptions {\n  versionId?: string\n  governanceBypass?: boolean\n  forceDelete?: boolean\n}\n\ntype Part = {\n  part: number\n  etag: string\n}\n\nexport class TypedClient {\n  protected transport: Transport\n  protected host: string\n  protected port: number\n  protected protocol: string\n  protected accessKey: string\n  protected secretKey: string\n  protected sessionToken?: string\n  protected userAgent: string\n  protected anonymous: boolean\n  protected pathStyle: boolean\n  protected regionMap: Record<string, string>\n  public region?: string\n  protected credentialsProvider?: CredentialProvider\n  partSize: number = 64 * 1024 * 1024\n  protected overRidePartSize?: boolean\n  protected retryOptions: RetryOptions\n\n  protected maximumPartSize = 5 * 1024 * 1024 * 1024\n  protected maxObjectSize = 5 * 1024 * 1024 * 1024 * 1024\n  public enableSHA256: boolean\n  protected s3AccelerateEndpoint?: string\n  protected reqOptions: Record<string, unknown>\n\n  protected transportAgent: http.Agent\n  private readonly clientExtensions: Extensions\n\n  constructor(params: ClientOptions) {\n    // @ts-expect-error deprecated property\n    if (params.secure !== undefined) {\n      throw new Error('\"secure\" option deprecated, \"useSSL\" should be used instead')\n    }\n    // Default values if not specified.\n    if (params.useSSL === undefined) {\n      params.useSSL = true\n    }\n    if (!params.port) {\n      params.port = 0\n    }\n    // Validate input params.\n    if (!isValidEndpoint(params.endPoint)) {\n      throw new errors.InvalidEndpointError(`Invalid endPoint : ${params.endPoint}`)\n    }\n    if (!isValidPort(params.port)) {\n      throw new errors.InvalidArgumentError(`Invalid port : ${params.port}`)\n    }\n    if (!isBoolean(params.useSSL)) {\n      throw new errors.InvalidArgumentError(\n        `Invalid useSSL flag type : ${params.useSSL}, expected to be of type \"boolean\"`,\n      )\n    }\n\n    // Validate region only if its set.\n    if (params.region) {\n      if (!isString(params.region)) {\n        throw new errors.InvalidArgumentError(`Invalid region : ${params.region}`)\n      }\n    }\n\n    const host = params.endPoint.toLowerCase()\n    let port = params.port\n    let protocol: string\n    let transport\n    let transportAgent: http.Agent\n    // Validate if configuration is not using SSL\n    // for constructing relevant endpoints.\n    if (params.useSSL) {\n      // Defaults to secure.\n      transport = https\n      protocol = 'https:'\n      port = port || 443\n      transportAgent = https.globalAgent\n    } else {\n      transport = http\n      protocol = 'http:'\n      port = port || 80\n      transportAgent = http.globalAgent\n    }\n\n    // if custom transport is set, use it.\n    if (params.transport) {\n      if (!isObject(params.transport)) {\n        throw new errors.InvalidArgumentError(\n          `Invalid transport type : ${params.transport}, expected to be type \"object\"`,\n        )\n      }\n      transport = params.transport\n    }\n\n    // if custom transport agent is set, use it.\n    if (params.transportAgent) {\n      if (!isObject(params.transportAgent)) {\n        throw new errors.InvalidArgumentError(\n          `Invalid transportAgent type: ${params.transportAgent}, expected to be type \"object\"`,\n        )\n      }\n\n      transportAgent = params.transportAgent\n    }\n\n    // User Agent should always following the below style.\n    // Please open an issue to discuss any new changes here.\n    //\n    //       MinIO (OS; ARCH) LIB/VER APP/VER\n    //\n    const libraryComments = `(${process.platform}; ${process.arch})`\n    const libraryAgent = `MinIO ${libraryComments} minio-js/${Package.version}`\n    // User agent block ends.\n\n    this.transport = transport\n    this.transportAgent = transportAgent\n    this.host = host\n    this.port = port\n    this.protocol = protocol\n    this.userAgent = `${libraryAgent}`\n\n    // Default path style is true\n    if (params.pathStyle === undefined) {\n      this.pathStyle = true\n    } else {\n      this.pathStyle = params.pathStyle\n    }\n\n    this.accessKey = params.accessKey ?? ''\n    this.secretKey = params.secretKey ?? ''\n    this.sessionToken = params.sessionToken\n    this.anonymous = !this.accessKey || !this.secretKey\n\n    if (params.credentialsProvider) {\n      this.anonymous = false\n      this.credentialsProvider = params.credentialsProvider\n    }\n\n    this.regionMap = {}\n    if (params.region) {\n      this.region = params.region\n    }\n\n    if (params.partSize) {\n      this.partSize = params.partSize\n      this.overRidePartSize = true\n    }\n    if (this.partSize < 5 * 1024 * 1024) {\n      throw new errors.InvalidArgumentError(`Part size should be greater than 5MB`)\n    }\n    if (this.partSize > 5 * 1024 * 1024 * 1024) {\n      throw new errors.InvalidArgumentError(`Part size should be less than 5GB`)\n    }\n\n    // SHA256 is enabled only for authenticated http requests. If the request is authenticated\n    // and the connection is https we use x-amz-content-sha256=UNSIGNED-PAYLOAD\n    // header for signature calculation.\n    this.enableSHA256 = !this.anonymous && !params.useSSL\n\n    this.s3AccelerateEndpoint = params.s3AccelerateEndpoint || undefined\n    this.reqOptions = {}\n    this.clientExtensions = new Extensions(this)\n\n    if (params.retryOptions) {\n      if (!isObject(params.retryOptions)) {\n        throw new errors.InvalidArgumentError(\n          `Invalid retryOptions type: ${params.retryOptions}, expected to be type \"object\"`,\n        )\n      }\n\n      this.retryOptions = params.retryOptions\n    } else {\n      this.retryOptions = {\n        disableRetry: false,\n      }\n    }\n  }\n  /**\n   * Minio extensions that aren't necessary present for Amazon S3 compatible storage servers\n   */\n  get extensions() {\n    return this.clientExtensions\n  }\n\n  /**\n   * @param endPoint - valid S3 acceleration end point\n   */\n  setS3TransferAccelerate(endPoint: string) {\n    this.s3AccelerateEndpoint = endPoint\n  }\n\n  /**\n   * Sets the supported request options.\n   */\n  public setRequestOptions(options: Pick<https.RequestOptions, (typeof requestOptionProperties)[number]>) {\n    if (!isObject(options)) {\n      throw new TypeError('request options should be of type \"object\"')\n    }\n    this.reqOptions = _.pick(options, requestOptionProperties)\n  }\n\n  /**\n   *  This is s3 Specific and does not hold validity in any other Object storage.\n   */\n  private getAccelerateEndPointIfSet(bucketName?: string, objectName?: string) {\n    if (!isEmpty(this.s3AccelerateEndpoint) && !isEmpty(bucketName) && !isEmpty(objectName)) {\n      // http://docs.aws.amazon.com/AmazonS3/latest/dev/transfer-acceleration.html\n      // Disable transfer acceleration for non-compliant bucket names.\n      if (bucketName.includes('.')) {\n        throw new Error(`Transfer Acceleration is not supported for non compliant bucket:${bucketName}`)\n      }\n      // If transfer acceleration is requested set new host.\n      // For more details about enabling transfer acceleration read here.\n      // http://docs.aws.amazon.com/AmazonS3/latest/dev/transfer-acceleration.html\n      return this.s3AccelerateEndpoint\n    }\n    return false\n  }\n\n  /**\n   *   Set application specific information.\n   *   Generates User-Agent in the following style.\n   *   MinIO (OS; ARCH) LIB/VER APP/VER\n   */\n  setAppInfo(appName: string, appVersion: string) {\n    if (!isString(appName)) {\n      throw new TypeError(`Invalid appName: ${appName}`)\n    }\n    if (appName.trim() === '') {\n      throw new errors.InvalidArgumentError('Input appName cannot be empty.')\n    }\n    if (!isString(appVersion)) {\n      throw new TypeError(`Invalid appVersion: ${appVersion}`)\n    }\n    if (appVersion.trim() === '') {\n      throw new errors.InvalidArgumentError('Input appVersion cannot be empty.')\n    }\n    this.userAgent = `${this.userAgent} ${appName}/${appVersion}`\n  }\n\n  /**\n   * returns options object that can be used with http.request()\n   * Takes care of constructing virtual-host-style or path-style hostname\n   */\n  protected getRequestOptions(\n    opts: RequestOption & {\n      region: string\n    },\n  ): IRequest & {\n    host: string\n    headers: Record<string, string>\n  } {\n    const method = opts.method\n    const region = opts.region\n    const bucketName = opts.bucketName\n    let objectName = opts.objectName\n    const headers = opts.headers\n    const query = opts.query\n\n    let reqOptions = {\n      method,\n      headers: {} as RequestHeaders,\n      protocol: this.protocol,\n      // If custom transportAgent was supplied earlier, we'll inject it here\n      agent: this.transportAgent,\n    }\n\n    // Verify if virtual host supported.\n    let virtualHostStyle\n    if (bucketName) {\n      virtualHostStyle = isVirtualHostStyle(this.host, this.protocol, bucketName, this.pathStyle)\n    }\n\n    let path = '/'\n    let host = this.host\n\n    let port: undefined | number\n    if (this.port) {\n      port = this.port\n    }\n\n    if (objectName) {\n      objectName = uriResourceEscape(objectName)\n    }\n\n    // For Amazon S3 endpoint, get endpoint based on region.\n    if (isAmazonEndpoint(host)) {\n      const accelerateEndPoint = this.getAccelerateEndPointIfSet(bucketName, objectName)\n      if (accelerateEndPoint) {\n        host = `${accelerateEndPoint}`\n      } else {\n        host = getS3Endpoint(region)\n      }\n    }\n\n    if (virtualHostStyle && !opts.pathStyle) {\n      // For all hosts which support virtual host style, `bucketName`\n      // is part of the hostname in the following format:\n      //\n      //  var host = 'bucketName.example.com'\n      //\n      if (bucketName) {\n        host = `${bucketName}.${host}`\n      }\n      if (objectName) {\n        path = `/${objectName}`\n      }\n    } else {\n      // For all S3 compatible storage services we will fallback to\n      // path style requests, where `bucketName` is part of the URI\n      // path.\n      if (bucketName) {\n        path = `/${bucketName}`\n      }\n      if (objectName) {\n        path = `/${bucketName}/${objectName}`\n      }\n    }\n\n    if (query) {\n      path += `?${query}`\n    }\n    reqOptions.headers.host = host\n    if ((reqOptions.protocol === 'http:' && port !== 80) || (reqOptions.protocol === 'https:' && port !== 443)) {\n      reqOptions.headers.host = joinHostPort(host, port)\n    }\n\n    reqOptions.headers['user-agent'] = this.userAgent\n    if (headers) {\n      // have all header keys in lower case - to make signing easy\n      for (const [k, v] of Object.entries(headers)) {\n        reqOptions.headers[k.toLowerCase()] = v\n      }\n    }\n\n    // Use any request option specified in minioClient.setRequestOptions()\n    reqOptions = Object.assign({}, this.reqOptions, reqOptions)\n\n    return {\n      ...reqOptions,\n      headers: _.mapValues(_.pickBy(reqOptions.headers, isDefined), (v) => v.toString()),\n      host,\n      port,\n      path,\n    } satisfies https.RequestOptions\n  }\n\n  public async setCredentialsProvider(credentialsProvider: CredentialProvider) {\n    if (!(credentialsProvider instanceof CredentialProvider)) {\n      throw new Error('Unable to get credentials. Expected instance of CredentialProvider')\n    }\n    this.credentialsProvider = credentialsProvider\n    await this.checkAndRefreshCreds()\n  }\n\n  private async checkAndRefreshCreds() {\n    if (this.credentialsProvider) {\n      try {\n        const credentialsConf = await this.credentialsProvider.getCredentials()\n        this.accessKey = credentialsConf.getAccessKey()\n        this.secretKey = credentialsConf.getSecretKey()\n        this.sessionToken = credentialsConf.getSessionToken()\n      } catch (e) {\n        throw new Error(`Unable to get credentials: ${e}`, { cause: e })\n      }\n    }\n  }\n\n  private logStream?: stream.Writable\n\n  /**\n   * log the request, response, error\n   */\n  private logHTTP(reqOptions: IRequest, response: http.IncomingMessage | null, err?: unknown) {\n    // if no logStream available return.\n    if (!this.logStream) {\n      return\n    }\n    if (!isObject(reqOptions)) {\n      throw new TypeError('reqOptions should be of type \"object\"')\n    }\n    if (response && !isReadableStream(response)) {\n      throw new TypeError('response should be of type \"Stream\"')\n    }\n    if (err && !(err instanceof Error)) {\n      throw new TypeError('err should be of type \"Error\"')\n    }\n    const logStream = this.logStream\n    const logHeaders = (headers: RequestHeaders) => {\n      Object.entries(headers).forEach(([k, v]) => {\n        if (k == 'authorization') {\n          if (isString(v)) {\n            const redactor = new RegExp('Signature=([0-9a-f]+)')\n            v = v.replace(redactor, 'Signature=**REDACTED**')\n          }\n        }\n        logStream.write(`${k}: ${v}\\n`)\n      })\n      logStream.write('\\n')\n    }\n    logStream.write(`REQUEST: ${reqOptions.method} ${reqOptions.path}\\n`)\n    logHeaders(reqOptions.headers)\n    if (response) {\n      this.logStream.write(`RESPONSE: ${response.statusCode}\\n`)\n      logHeaders(response.headers as RequestHeaders)\n    }\n    if (err) {\n      logStream.write('ERROR BODY:\\n')\n      const errJSON = JSON.stringify(err, null, '\\t')\n      logStream.write(`${errJSON}\\n`)\n    }\n  }\n\n  /**\n   * Enable tracing\n   */\n  public traceOn(stream?: stream.Writable) {\n    if (!stream) {\n      stream = process.stdout\n    }\n    this.logStream = stream\n  }\n\n  /**\n   * Disable tracing\n   */\n  public traceOff() {\n    this.logStream = undefined\n  }\n\n  /**\n   * makeRequest is the primitive used by the apis for making S3 requests.\n   * payload can be empty string in case of no payload.\n   * statusCode is the expected statusCode. If response.statusCode does not match\n   * we parse the XML error and call the callback with the error message.\n   *\n   * A valid region is passed by the calls - listBuckets, makeBucket and getBucketRegion.\n   *\n   * @internal\n   */\n  async makeRequestAsync(\n    options: RequestOption,\n    payload: Binary = '',\n    expectedCodes: number[] = [200],\n    region = '',\n  ): Promise<http.IncomingMessage> {\n    if (!isObject(options)) {\n      throw new TypeError('options should be of type \"object\"')\n    }\n    if (!isString(payload) && !isObject(payload)) {\n      // Buffer is of type 'object'\n      throw new TypeError('payload should be of type \"string\" or \"Buffer\"')\n    }\n    expectedCodes.forEach((statusCode) => {\n      if (!isNumber(statusCode)) {\n        throw new TypeError('statusCode should be of type \"number\"')\n      }\n    })\n    if (!isString(region)) {\n      throw new TypeError('region should be of type \"string\"')\n    }\n    if (!options.headers) {\n      options.headers = {}\n    }\n    if (options.method === 'POST' || options.method === 'PUT' || options.method === 'DELETE') {\n      options.headers['content-length'] = payload.length.toString()\n    }\n    const sha256sum = this.enableSHA256 ? toSha256(payload) : ''\n    return this.makeRequestStreamAsync(options, payload, sha256sum, expectedCodes, region)\n  }\n\n  /**\n   * new request with promise\n   *\n   * No need to drain response, response body is not valid\n   */\n  async makeRequestAsyncOmit(\n    options: RequestOption,\n    payload: Binary = '',\n    statusCodes: number[] = [200],\n    region = '',\n  ): Promise<Omit<http.IncomingMessage, 'on'>> {\n    const res = await this.makeRequestAsync(options, payload, statusCodes, region)\n    await drainResponse(res)\n    return res\n  }\n\n  /**\n   * makeRequestStream will be used directly instead of makeRequest in case the payload\n   * is available as a stream. for ex. putObject\n   *\n   * @internal\n   */\n  async makeRequestStreamAsync(\n    options: RequestOption,\n    body: stream.Readable | Binary,\n    sha256sum: string,\n    statusCodes: number[],\n    region: string,\n  ): Promise<http.IncomingMessage> {\n    if (!isObject(options)) {\n      throw new TypeError('options should be of type \"object\"')\n    }\n    if (!(Buffer.isBuffer(body) || typeof body === 'string' || isReadableStream(body))) {\n      throw new errors.InvalidArgumentError(\n        `stream should be a Buffer, string or readable Stream, got ${typeof body} instead`,\n      )\n    }\n    if (!isString(sha256sum)) {\n      throw new TypeError('sha256sum should be of type \"string\"')\n    }\n    statusCodes.forEach((statusCode) => {\n      if (!isNumber(statusCode)) {\n        throw new TypeError('statusCode should be of type \"number\"')\n      }\n    })\n    if (!isString(region)) {\n      throw new TypeError('region should be of type \"string\"')\n    }\n    // sha256sum will be empty for anonymous or https requests\n    if (!this.enableSHA256 && sha256sum.length !== 0) {\n      throw new errors.InvalidArgumentError(`sha256sum expected to be empty for anonymous or https requests`)\n    }\n    // sha256sum should be valid for non-anonymous http requests.\n    if (this.enableSHA256 && sha256sum.length !== 64) {\n      throw new errors.InvalidArgumentError(`Invalid sha256sum : ${sha256sum}`)\n    }\n\n    await this.checkAndRefreshCreds()\n\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n    region = region || (await this.getBucketRegionAsync(options.bucketName!))\n\n    const reqOptions = this.getRequestOptions({ ...options, region })\n    if (!this.anonymous) {\n      // For non-anonymous https requests sha256sum is 'UNSIGNED-PAYLOAD' for signature calculation.\n      if (!this.enableSHA256) {\n        sha256sum = 'UNSIGNED-PAYLOAD'\n      }\n      const date = new Date()\n      reqOptions.headers['x-amz-date'] = makeDateLong(date)\n      reqOptions.headers['x-amz-content-sha256'] = sha256sum\n      if (this.sessionToken) {\n        reqOptions.headers['x-amz-security-token'] = this.sessionToken\n      }\n      reqOptions.headers.authorization = signV4(reqOptions, this.accessKey, this.secretKey, region, date, sha256sum)\n    }\n\n    const response = await requestWithRetry(\n      this.transport,\n      reqOptions,\n      body,\n      this.retryOptions.disableRetry === true ? 0 : this.retryOptions.maximumRetryCount,\n      this.retryOptions.baseDelayMs,\n      this.retryOptions.maximumDelayMs,\n    )\n    if (!response.statusCode) {\n      throw new Error(\"BUG: response doesn't have a statusCode\")\n    }\n\n    if (!statusCodes.includes(response.statusCode)) {\n      // For an incorrect region, S3 server always sends back 400.\n      // But we will do cache invalidation for all errors so that,\n      // in future, if AWS S3 decides to send a different status code or\n      // XML error code we will still work fine.\n      // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n      delete this.regionMap[options.bucketName!]\n\n      const err = await xmlParsers.parseResponseError(response)\n      this.logHTTP(reqOptions, response, err)\n      throw err\n    }\n\n    this.logHTTP(reqOptions, response)\n\n    return response\n  }\n\n  /**\n   * gets the region of the bucket\n   *\n   * @param bucketName\n   *\n   */\n  async getBucketRegionAsync(bucketName: string): Promise<string> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name : ${bucketName}`)\n    }\n\n    // Region is set with constructor, return the region right here.\n    if (this.region) {\n      return this.region\n    }\n\n    const cached = this.regionMap[bucketName]\n    if (cached) {\n      return cached\n    }\n\n    const extractRegionAsync = async (response: http.IncomingMessage) => {\n      const body = await readAsString(response)\n      const region = xmlParsers.parseBucketRegion(body) || DEFAULT_REGION\n      this.regionMap[bucketName] = region\n      return region\n    }\n\n    const method = 'GET'\n    const query = 'location'\n    // `getBucketLocation` behaves differently in following ways for\n    // different environments.\n    //\n    // - For nodejs env we default to path style requests.\n    // - For browser env path style requests on buckets yields CORS\n    //   error. To circumvent this problem we make a virtual host\n    //   style request signed with 'us-east-1'. This request fails\n    //   with an error 'AuthorizationHeaderMalformed', additionally\n    //   the error XML also provides Region of the bucket. To validate\n    //   this region is proper we retry the same request with the newly\n    //   obtained region.\n    const pathStyle = this.pathStyle && !isBrowser\n    let region: string\n    try {\n      const res = await this.makeRequestAsync({ method, bucketName, query, pathStyle }, '', [200], DEFAULT_REGION)\n      return extractRegionAsync(res)\n    } catch (e) {\n      // make alignment with mc cli\n      if (e instanceof errors.S3Error) {\n        const errCode = e.code\n        const errRegion = e.region\n        if (errCode === 'AccessDenied' && !errRegion) {\n          return DEFAULT_REGION\n        }\n      }\n      // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n      // @ts-ignore\n      if (!(e.name === 'AuthorizationHeaderMalformed')) {\n        throw e\n      }\n      // @ts-expect-error we set extra properties on error object\n      region = e.Region as string\n      if (!region) {\n        throw e\n      }\n    }\n\n    const res = await this.makeRequestAsync({ method, bucketName, query, pathStyle }, '', [200], region)\n    return await extractRegionAsync(res)\n  }\n\n  /**\n   * makeRequest is the primitive used by the apis for making S3 requests.\n   * payload can be empty string in case of no payload.\n   * statusCode is the expected statusCode. If response.statusCode does not match\n   * we parse the XML error and call the callback with the error message.\n   * A valid region is passed by the calls - listBuckets, makeBucket and\n   * getBucketRegion.\n   *\n   * @deprecated use `makeRequestAsync` instead\n   */\n  makeRequest(\n    options: RequestOption,\n    payload: Binary = '',\n    expectedCodes: number[] = [200],\n    region = '',\n    returnResponse: boolean,\n    cb: (cb: unknown, result: http.IncomingMessage) => void,\n  ) {\n    let prom: Promise<http.IncomingMessage>\n    if (returnResponse) {\n      prom = this.makeRequestAsync(options, payload, expectedCodes, region)\n    } else {\n      // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n      // @ts-expect-error compatible for old behaviour\n      prom = this.makeRequestAsyncOmit(options, payload, expectedCodes, region)\n    }\n\n    prom.then(\n      (result) => cb(null, result),\n      (err) => {\n        // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n        // @ts-ignore\n        cb(err)\n      },\n    )\n  }\n\n  /**\n   * makeRequestStream will be used directly instead of makeRequest in case the payload\n   * is available as a stream. for ex. putObject\n   *\n   * @deprecated use `makeRequestStreamAsync` instead\n   */\n  makeRequestStream(\n    options: RequestOption,\n    stream: stream.Readable | Buffer,\n    sha256sum: string,\n    statusCodes: number[],\n    region: string,\n    returnResponse: boolean,\n    cb: (cb: unknown, result: http.IncomingMessage) => void,\n  ) {\n    const executor = async () => {\n      const res = await this.makeRequestStreamAsync(options, stream, sha256sum, statusCodes, region)\n      if (!returnResponse) {\n        await drainResponse(res)\n      }\n\n      return res\n    }\n\n    executor().then(\n      (result) => cb(null, result),\n      // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n      // @ts-ignore\n      (err) => cb(err),\n    )\n  }\n\n  /**\n   * @deprecated use `getBucketRegionAsync` instead\n   */\n  getBucketRegion(bucketName: string, cb: (err: unknown, region: string) => void) {\n    return this.getBucketRegionAsync(bucketName).then(\n      (result) => cb(null, result),\n      // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n      // @ts-ignore\n      (err) => cb(err),\n    )\n  }\n\n  // Bucket operations\n\n  /**\n   * Creates the bucket `bucketName`.\n   *\n   */\n  async makeBucket(bucketName: string, region: Region = '', makeOpts?: MakeBucketOpt): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    // Backward Compatibility\n    if (isObject(region)) {\n      makeOpts = region\n      region = ''\n    }\n\n    if (!isString(region)) {\n      throw new TypeError('region should be of type \"string\"')\n    }\n    if (makeOpts && !isObject(makeOpts)) {\n      throw new TypeError('makeOpts should be of type \"object\"')\n    }\n\n    let payload = ''\n\n    // Region already set in constructor, validate if\n    // caller requested bucket location is same.\n    if (region && this.region) {\n      if (region !== this.region) {\n        throw new errors.InvalidArgumentError(`Configured region ${this.region}, requested ${region}`)\n      }\n    }\n    // sending makeBucket request with XML containing 'us-east-1' fails. For\n    // default region server expects the request without body\n    if (region && region !== DEFAULT_REGION) {\n      payload = xml.buildObject({\n        CreateBucketConfiguration: {\n          $: { xmlns: 'http://s3.amazonaws.com/doc/2006-03-01/' },\n          LocationConstraint: region,\n        },\n      })\n    }\n    const method = 'PUT'\n    const headers: RequestHeaders = {}\n\n    if (makeOpts && makeOpts.ObjectLocking) {\n      headers['x-amz-bucket-object-lock-enabled'] = true\n    }\n\n    // For custom region clients  default to custom region specified in client constructor\n    const finalRegion = this.region || region || DEFAULT_REGION\n\n    const requestOpt: RequestOption = { method, bucketName, headers }\n\n    try {\n      await this.makeRequestAsyncOmit(requestOpt, payload, [200], finalRegion)\n    } catch (err: unknown) {\n      if (region === '' || region === DEFAULT_REGION) {\n        if (err instanceof errors.S3Error) {\n          const errCode = err.code\n          const errRegion = err.region\n          if (errCode === 'AuthorizationHeaderMalformed' && errRegion !== '') {\n            // Retry with region returned as part of error\n            await this.makeRequestAsyncOmit(requestOpt, payload, [200], errCode)\n          }\n        }\n      }\n      throw err\n    }\n  }\n\n  /**\n   * To check if a bucket already exists.\n   */\n  async bucketExists(bucketName: string): Promise<boolean> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'HEAD'\n    try {\n      await this.makeRequestAsyncOmit({ method, bucketName })\n    } catch (err) {\n      // @ts-ignore\n      if (err.code === 'NoSuchBucket' || err.code === 'NotFound') {\n        return false\n      }\n      throw err\n    }\n\n    return true\n  }\n\n  async removeBucket(bucketName: string): Promise<void>\n\n  /**\n   * @deprecated use promise style API\n   */\n  removeBucket(bucketName: string, callback: NoResultCallback): void\n\n  async removeBucket(bucketName: string): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'DELETE'\n    await this.makeRequestAsyncOmit({ method, bucketName }, '', [204])\n    delete this.regionMap[bucketName]\n  }\n\n  /**\n   * Callback is called with readable stream of the object content.\n   */\n  async getObject(bucketName: string, objectName: string, getOpts?: GetObjectOpts): Promise<stream.Readable> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    return this.getPartialObject(bucketName, objectName, 0, 0, getOpts)\n  }\n\n  /**\n   * Callback is called with readable stream of the partial object content.\n   * @param bucketName\n   * @param objectName\n   * @param offset\n   * @param length - length of the object that will be read in the stream (optional, if not specified we read the rest of the file from the offset)\n   * @param getOpts\n   */\n  async getPartialObject(\n    bucketName: string,\n    objectName: string,\n    offset: number,\n    length = 0,\n    getOpts?: GetObjectOpts,\n  ): Promise<stream.Readable> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    if (!isNumber(offset)) {\n      throw new TypeError('offset should be of type \"number\"')\n    }\n    if (!isNumber(length)) {\n      throw new TypeError('length should be of type \"number\"')\n    }\n\n    let range = ''\n    if (offset || length) {\n      if (offset) {\n        range = `bytes=${+offset}-`\n      } else {\n        range = 'bytes=0-'\n        offset = 0\n      }\n      if (length) {\n        range += `${+length + offset - 1}`\n      }\n    }\n\n    let query = ''\n    let headers: RequestHeaders = {\n      ...(range !== '' && { range }),\n    }\n\n    if (getOpts) {\n      const sseHeaders: Record<string, string> = {\n        ...(getOpts.SSECustomerAlgorithm && {\n          'X-Amz-Server-Side-Encryption-Customer-Algorithm': getOpts.SSECustomerAlgorithm,\n        }),\n        ...(getOpts.SSECustomerKey && { 'X-Amz-Server-Side-Encryption-Customer-Key': getOpts.SSECustomerKey }),\n        ...(getOpts.SSECustomerKeyMD5 && {\n          'X-Amz-Server-Side-Encryption-Customer-Key-MD5': getOpts.SSECustomerKeyMD5,\n        }),\n      }\n      query = qs.stringify(getOpts)\n      headers = {\n        ...prependXAMZMeta(sseHeaders),\n        ...headers,\n      }\n    }\n\n    const expectedStatusCodes = [200]\n    if (range) {\n      expectedStatusCodes.push(206)\n    }\n    const method = 'GET'\n\n    return await this.makeRequestAsync({ method, bucketName, objectName, headers, query }, '', expectedStatusCodes)\n  }\n\n  /**\n   * download object content to a file.\n   * This method will create a temp file named `${filename}.${base64(etag)}.part.minio` when downloading.\n   *\n   * @param bucketName - name of the bucket\n   * @param objectName - name of the object\n   * @param filePath - path to which the object data will be written to\n   * @param getOpts - Optional object get option\n   */\n  async fGetObject(bucketName: string, objectName: string, filePath: string, getOpts?: GetObjectOpts): Promise<void> {\n    // Input validation.\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    if (!isString(filePath)) {\n      throw new TypeError('filePath should be of type \"string\"')\n    }\n\n    const downloadToTmpFile = async (): Promise<string> => {\n      let partFileStream: stream.Writable\n      const objStat = await this.statObject(bucketName, objectName, getOpts)\n      const encodedEtag = Buffer.from(objStat.etag).toString('base64')\n      const partFile = `${filePath}.${encodedEtag}.part.minio`\n\n      await fsp.mkdir(path.dirname(filePath), { recursive: true })\n\n      let offset = 0\n      try {\n        const stats = await fsp.stat(partFile)\n        if (objStat.size === stats.size) {\n          return partFile\n        }\n        offset = stats.size\n        partFileStream = fs.createWriteStream(partFile, { flags: 'a' })\n      } catch (e) {\n        if (e instanceof Error && (e as unknown as { code: string }).code === 'ENOENT') {\n          // file not exist\n          partFileStream = fs.createWriteStream(partFile, { flags: 'w' })\n        } else {\n          // other error, maybe access deny\n          throw e\n        }\n      }\n\n      const downloadStream = await this.getPartialObject(bucketName, objectName, offset, 0, getOpts)\n\n      await streamPromise.pipeline(downloadStream, partFileStream)\n      const stats = await fsp.stat(partFile)\n      if (stats.size === objStat.size) {\n        return partFile\n      }\n\n      throw new Error('Size mismatch between downloaded file and the object')\n    }\n\n    const partFile = await downloadToTmpFile()\n    await fsp.rename(partFile, filePath)\n  }\n\n  /**\n   * Stat information of the object.\n   */\n  async statObject(bucketName: string, objectName: string, statOpts?: StatObjectOpts): Promise<BucketItemStat> {\n    const statOptDef = statOpts || {}\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n\n    if (!isObject(statOptDef)) {\n      throw new errors.InvalidArgumentError('statOpts should be of type \"object\"')\n    }\n\n    const query = qs.stringify(statOptDef)\n    const method = 'HEAD'\n    const res = await this.makeRequestAsyncOmit({ method, bucketName, objectName, query })\n\n    return {\n      size: parseInt(res.headers['content-length'] as string),\n      metaData: extractMetadata(res.headers as ResponseHeader),\n      lastModified: new Date(res.headers['last-modified'] as string),\n      versionId: getVersionId(res.headers as ResponseHeader),\n      etag: sanitizeETag(res.headers.etag),\n    }\n  }\n\n  async removeObject(bucketName: string, objectName: string, removeOpts?: RemoveOptions): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n\n    if (removeOpts && !isObject(removeOpts)) {\n      throw new errors.InvalidArgumentError('removeOpts should be of type \"object\"')\n    }\n\n    const method = 'DELETE'\n\n    const headers: RequestHeaders = {}\n    if (removeOpts?.governanceBypass) {\n      headers['X-Amz-Bypass-Governance-Retention'] = true\n    }\n    if (removeOpts?.forceDelete) {\n      headers['x-minio-force-delete'] = true\n    }\n\n    const queryParams: Record<string, string> = {}\n    if (removeOpts?.versionId) {\n      queryParams.versionId = `${removeOpts.versionId}`\n    }\n    const query = qs.stringify(queryParams)\n\n    await this.makeRequestAsyncOmit({ method, bucketName, objectName, headers, query }, '', [200, 204])\n  }\n\n  // Calls implemented below are related to multipart.\n\n  listIncompleteUploads(\n    bucket: string,\n    prefix: string,\n    recursive: boolean,\n  ): BucketStream<IncompleteUploadedBucketItem> {\n    if (prefix === undefined) {\n      prefix = ''\n    }\n    if (recursive === undefined) {\n      recursive = false\n    }\n    if (!isValidBucketName(bucket)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucket)\n    }\n    if (!isValidPrefix(prefix)) {\n      throw new errors.InvalidPrefixError(`Invalid prefix : ${prefix}`)\n    }\n    if (!isBoolean(recursive)) {\n      throw new TypeError('recursive should be of type \"boolean\"')\n    }\n    const delimiter = recursive ? '' : '/'\n    let keyMarker = ''\n    let uploadIdMarker = ''\n    const uploads: unknown[] = []\n    let ended = false\n\n    // TODO: refactor this with async/await and `stream.Readable.from`\n    const readStream = new stream.Readable({ objectMode: true })\n    readStream._read = () => {\n      // push one upload info per _read()\n      if (uploads.length) {\n        return readStream.push(uploads.shift())\n      }\n      if (ended) {\n        return readStream.push(null)\n      }\n      this.listIncompleteUploadsQuery(bucket, prefix, keyMarker, uploadIdMarker, delimiter).then(\n        (result) => {\n          // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n          // @ts-ignore\n          result.prefixes.forEach((prefix) => uploads.push(prefix))\n          async.eachSeries(\n            result.uploads,\n            (upload, cb) => {\n              // for each incomplete upload add the sizes of its uploaded parts\n              // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n              // @ts-ignore\n              this.listParts(bucket, upload.key, upload.uploadId).then(\n                (parts: Part[]) => {\n                  // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n                  // @ts-ignore\n                  upload.size = parts.reduce((acc, item) => acc + item.size, 0)\n                  uploads.push(upload)\n                  cb()\n                },\n                (err: Error) => cb(err),\n              )\n            },\n            (err) => {\n              if (err) {\n                readStream.emit('error', err)\n                return\n              }\n              if (result.isTruncated) {\n                keyMarker = result.nextKeyMarker\n                uploadIdMarker = result.nextUploadIdMarker\n              } else {\n                ended = true\n              }\n\n              // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n              // @ts-ignore\n              readStream._read()\n            },\n          )\n        },\n        (e) => {\n          readStream.emit('error', e)\n        },\n      )\n    }\n    return readStream\n  }\n\n  /**\n   * Called by listIncompleteUploads to fetch a batch of incomplete uploads.\n   */\n  async listIncompleteUploadsQuery(\n    bucketName: string,\n    prefix: string,\n    keyMarker: string,\n    uploadIdMarker: string,\n    delimiter: string,\n  ): Promise<ListMultipartResult> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isString(prefix)) {\n      throw new TypeError('prefix should be of type \"string\"')\n    }\n    if (!isString(keyMarker)) {\n      throw new TypeError('keyMarker should be of type \"string\"')\n    }\n    if (!isString(uploadIdMarker)) {\n      throw new TypeError('uploadIdMarker should be of type \"string\"')\n    }\n    if (!isString(delimiter)) {\n      throw new TypeError('delimiter should be of type \"string\"')\n    }\n    const queries = []\n    queries.push(`prefix=${uriEscape(prefix)}`)\n    queries.push(`delimiter=${uriEscape(delimiter)}`)\n\n    if (keyMarker) {\n      queries.push(`key-marker=${uriEscape(keyMarker)}`)\n    }\n    if (uploadIdMarker) {\n      queries.push(`upload-id-marker=${uploadIdMarker}`)\n    }\n\n    const maxUploads = 1000\n    queries.push(`max-uploads=${maxUploads}`)\n    queries.sort()\n    queries.unshift('uploads')\n    let query = ''\n    if (queries.length > 0) {\n      query = `${queries.join('&')}`\n    }\n    const method = 'GET'\n    const res = await this.makeRequestAsync({ method, bucketName, query })\n    const body = await readAsString(res)\n    return xmlParsers.parseListMultipart(body)\n  }\n\n  /**\n   * Initiate a new multipart upload.\n   * @internal\n   */\n  async initiateNewMultipartUpload(bucketName: string, objectName: string, headers: RequestHeaders): Promise<string> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    if (!isObject(headers)) {\n      throw new errors.InvalidObjectNameError('contentType should be of type \"object\"')\n    }\n    const method = 'POST'\n    const query = 'uploads'\n    const res = await this.makeRequestAsync({ method, bucketName, objectName, query, headers })\n    const body = await readAsBuffer(res)\n    return parseInitiateMultipart(body.toString())\n  }\n\n  /**\n   * Internal Method to abort a multipart upload request in case of any errors.\n   *\n   * @param bucketName - Bucket Name\n   * @param objectName - Object Name\n   * @param uploadId - id of a multipart upload to cancel during compose object sequence.\n   */\n  async abortMultipartUpload(bucketName: string, objectName: string, uploadId: string): Promise<void> {\n    const method = 'DELETE'\n    const query = `uploadId=${uploadId}`\n\n    const requestOptions = { method, bucketName, objectName: objectName, query }\n    await this.makeRequestAsyncOmit(requestOptions, '', [204])\n  }\n\n  async findUploadId(bucketName: string, objectName: string): Promise<string | undefined> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n\n    let latestUpload: ListMultipartResult['uploads'][number] | undefined\n    let keyMarker = ''\n    let uploadIdMarker = ''\n    for (;;) {\n      const result = await this.listIncompleteUploadsQuery(bucketName, objectName, keyMarker, uploadIdMarker, '')\n      for (const upload of result.uploads) {\n        if (upload.key === objectName) {\n          if (!latestUpload || upload.initiated.getTime() > latestUpload.initiated.getTime()) {\n            latestUpload = upload\n          }\n        }\n      }\n      if (result.isTruncated) {\n        keyMarker = result.nextKeyMarker\n        uploadIdMarker = result.nextUploadIdMarker\n        continue\n      }\n\n      break\n    }\n    return latestUpload?.uploadId\n  }\n\n  /**\n   * this call will aggregate the parts on the server into a single object.\n   */\n  async completeMultipartUpload(\n    bucketName: string,\n    objectName: string,\n    uploadId: string,\n    etags: {\n      part: number\n      etag?: string\n    }[],\n  ): Promise<{ etag: string; versionId: string | null }> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    if (!isString(uploadId)) {\n      throw new TypeError('uploadId should be of type \"string\"')\n    }\n    if (!isObject(etags)) {\n      throw new TypeError('etags should be of type \"Array\"')\n    }\n\n    if (!uploadId) {\n      throw new errors.InvalidArgumentError('uploadId cannot be empty')\n    }\n\n    const method = 'POST'\n    const query = `uploadId=${uriEscape(uploadId)}`\n\n    const builder = new xml2js.Builder()\n    const payload = builder.buildObject({\n      CompleteMultipartUpload: {\n        $: {\n          xmlns: 'http://s3.amazonaws.com/doc/2006-03-01/',\n        },\n        Part: etags.map((etag) => {\n          return {\n            PartNumber: etag.part,\n            ETag: etag.etag,\n          }\n        }),\n      },\n    })\n\n    const res = await this.makeRequestAsync({ method, bucketName, objectName, query }, payload)\n    const body = await readAsBuffer(res)\n    const result = parseCompleteMultipart(body.toString())\n    if (!result) {\n      throw new Error('BUG: failed to parse server response')\n    }\n\n    if (result.errCode) {\n      // Multipart Complete API returns an error XML after a 200 http status\n      throw new errors.S3Error(result.errMessage)\n    }\n\n    return {\n      // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n      // @ts-ignore\n      etag: result.etag as string,\n      versionId: getVersionId(res.headers as ResponseHeader),\n    }\n  }\n\n  /**\n   * Get part-info of all parts of an incomplete upload specified by uploadId.\n   */\n  protected async listParts(bucketName: string, objectName: string, uploadId: string): Promise<UploadedPart[]> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    if (!isString(uploadId)) {\n      throw new TypeError('uploadId should be of type \"string\"')\n    }\n    if (!uploadId) {\n      throw new errors.InvalidArgumentError('uploadId cannot be empty')\n    }\n\n    const parts: UploadedPart[] = []\n    let marker = 0\n    let result\n    do {\n      result = await this.listPartsQuery(bucketName, objectName, uploadId, marker)\n      marker = result.marker\n      parts.push(...result.parts)\n    } while (result.isTruncated)\n\n    return parts\n  }\n\n  /**\n   * Called by listParts to fetch a batch of part-info\n   */\n  private async listPartsQuery(bucketName: string, objectName: string, uploadId: string, marker: number) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    if (!isString(uploadId)) {\n      throw new TypeError('uploadId should be of type \"string\"')\n    }\n    if (!isNumber(marker)) {\n      throw new TypeError('marker should be of type \"number\"')\n    }\n    if (!uploadId) {\n      throw new errors.InvalidArgumentError('uploadId cannot be empty')\n    }\n\n    let query = `uploadId=${uriEscape(uploadId)}`\n    if (marker) {\n      query += `&part-number-marker=${marker}`\n    }\n\n    const method = 'GET'\n    const res = await this.makeRequestAsync({ method, bucketName, objectName, query })\n    return xmlParsers.parseListParts(await readAsString(res))\n  }\n\n  async listBuckets(): Promise<BucketItemFromList[]> {\n    const method = 'GET'\n    const regionConf = this.region || DEFAULT_REGION\n    const httpRes = await this.makeRequestAsync({ method }, '', [200], regionConf)\n    const xmlResult = await readAsString(httpRes)\n    return xmlParsers.parseListBucket(xmlResult)\n  }\n\n  /**\n   * Calculate part size given the object size. Part size will be atleast this.partSize\n   */\n  calculatePartSize(size: number) {\n    if (!isNumber(size)) {\n      throw new TypeError('size should be of type \"number\"')\n    }\n    if (size > this.maxObjectSize) {\n      throw new TypeError(`size should not be more than ${this.maxObjectSize}`)\n    }\n    if (this.overRidePartSize) {\n      return this.partSize\n    }\n    let partSize = this.partSize\n    for (;;) {\n      // while(true) {...} throws linting error.\n      // If partSize is big enough to accomodate the object size, then use it.\n      if (partSize * 10000 > size) {\n        return partSize\n      }\n      // Try part sizes as 64MB, 80MB, 96MB etc.\n      partSize += 16 * 1024 * 1024\n    }\n  }\n\n  /**\n   * Uploads the object using contents from a file\n   */\n  async fPutObject(bucketName: string, objectName: string, filePath: string, metaData?: ObjectMetaData) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n\n    if (!isString(filePath)) {\n      throw new TypeError('filePath should be of type \"string\"')\n    }\n    if (metaData && !isObject(metaData)) {\n      throw new TypeError('metaData should be of type \"object\"')\n    }\n\n    // Inserts correct `content-type` attribute based on metaData and filePath\n    metaData = insertContentType(metaData || {}, filePath)\n    const stat = await fsp.stat(filePath)\n    return await this.putObject(bucketName, objectName, fs.createReadStream(filePath), stat.size, metaData)\n  }\n\n  /**\n   *  Uploading a stream, \"Buffer\" or \"string\".\n   *  It's recommended to pass `size` argument with stream.\n   */\n  async putObject(\n    bucketName: string,\n    objectName: string,\n    stream: stream.Readable | Buffer | string,\n    size?: number,\n    metaData?: ItemBucketMetadata,\n  ): Promise<UploadedObjectInfo> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n\n    // We'll need to shift arguments to the left because of metaData\n    // and size being optional.\n    if (isObject(size)) {\n      metaData = size\n    }\n    // Ensures Metadata has appropriate prefix for A3 API\n    const headers = prependXAMZMeta(metaData)\n    if (typeof stream === 'string' || stream instanceof Buffer) {\n      // Adapts the non-stream interface into a stream.\n      size = stream.length\n      stream = readableStream(stream)\n    } else if (!isReadableStream(stream)) {\n      throw new TypeError('third argument should be of type \"stream.Readable\" or \"Buffer\" or \"string\"')\n    }\n\n    if (isNumber(size) && size < 0) {\n      throw new errors.InvalidArgumentError(`size cannot be negative, given size: ${size}`)\n    }\n\n    // Get the part size and forward that to the BlockStream. Default to the\n    // largest block size possible if necessary.\n    if (!isNumber(size)) {\n      size = this.maxObjectSize\n    }\n\n    // Get the part size and forward that to the BlockStream. Default to the\n    // largest block size possible if necessary.\n    if (size === undefined) {\n      const statSize = await getContentLength(stream)\n      if (statSize !== null) {\n        size = statSize\n      }\n    }\n\n    if (!isNumber(size)) {\n      // Backward compatibility\n      size = this.maxObjectSize\n    }\n    if (size === 0) {\n      return this.uploadBuffer(bucketName, objectName, headers, Buffer.from(''))\n    }\n\n    const partSize = this.calculatePartSize(size)\n    if (typeof stream === 'string' || Buffer.isBuffer(stream) || size <= partSize) {\n      const buf = isReadableStream(stream) ? await readAsBuffer(stream) : Buffer.from(stream)\n      return this.uploadBuffer(bucketName, objectName, headers, buf)\n    }\n\n    return this.uploadStream(bucketName, objectName, headers, stream, partSize)\n  }\n\n  /**\n   * method to upload buffer in one call\n   * @private\n   */\n  private async uploadBuffer(\n    bucketName: string,\n    objectName: string,\n    headers: RequestHeaders,\n    buf: Buffer,\n  ): Promise<UploadedObjectInfo> {\n    const { md5sum, sha256sum } = hashBinary(buf, this.enableSHA256)\n    headers['Content-Length'] = buf.length\n    if (!this.enableSHA256) {\n      headers['Content-MD5'] = md5sum\n    }\n    const res = await this.makeRequestStreamAsync(\n      {\n        method: 'PUT',\n        bucketName,\n        objectName,\n        headers,\n      },\n      buf,\n      sha256sum,\n      [200],\n      '',\n    )\n    await drainResponse(res)\n    return {\n      etag: sanitizeETag(res.headers.etag),\n      versionId: getVersionId(res.headers as ResponseHeader),\n    }\n  }\n\n  /**\n   * upload stream with MultipartUpload\n   * @private\n   */\n  private async uploadStream(\n    bucketName: string,\n    objectName: string,\n    headers: RequestHeaders,\n    body: stream.Readable,\n    partSize: number,\n  ): Promise<UploadedObjectInfo> {\n    // A map of the previously uploaded chunks, for resuming a file upload. This\n    // will be null if we aren't resuming an upload.\n    const oldParts: Record<number, Part> = {}\n\n    // Keep track of the etags for aggregating the chunks together later. Each\n    // etag represents a single chunk of the file.\n    const eTags: Part[] = []\n\n    const previousUploadId = await this.findUploadId(bucketName, objectName)\n    let uploadId: string\n    if (!previousUploadId) {\n      uploadId = await this.initiateNewMultipartUpload(bucketName, objectName, headers)\n    } else {\n      uploadId = previousUploadId\n      const oldTags = await this.listParts(bucketName, objectName, previousUploadId)\n      oldTags.forEach((e) => {\n        oldParts[e.part] = e\n      })\n    }\n\n    const chunkier = new BlockStream2({ size: partSize, zeroPadding: false })\n\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    const [_, o] = await Promise.all([\n      new Promise((resolve, reject) => {\n        body.pipe(chunkier).on('error', reject)\n        chunkier.on('end', resolve).on('error', reject)\n      }),\n      (async () => {\n        let partNumber = 1\n\n        for await (const chunk of chunkier) {\n          const md5 = crypto.createHash('md5').update(chunk).digest()\n\n          const oldPart = oldParts[partNumber]\n          if (oldPart) {\n            if (oldPart.etag === md5.toString('hex')) {\n              eTags.push({ part: partNumber, etag: oldPart.etag })\n              partNumber++\n              continue\n            }\n          }\n\n          partNumber++\n\n          // now start to upload missing part\n          const options: RequestOption = {\n            method: 'PUT',\n            query: qs.stringify({ partNumber, uploadId }),\n            headers: {\n              'Content-Length': chunk.length,\n              'Content-MD5': md5.toString('base64'),\n            },\n            bucketName,\n            objectName,\n          }\n\n          const response = await this.makeRequestAsyncOmit(options, chunk)\n\n          let etag = response.headers.etag\n          if (etag) {\n            etag = etag.replace(/^\"/, '').replace(/\"$/, '')\n          } else {\n            etag = ''\n          }\n\n          eTags.push({ part: partNumber, etag })\n        }\n\n        return await this.completeMultipartUpload(bucketName, objectName, uploadId, eTags)\n      })(),\n    ])\n\n    return o\n  }\n\n  async removeBucketReplication(bucketName: string): Promise<void>\n  removeBucketReplication(bucketName: string, callback: NoResultCallback): void\n  async removeBucketReplication(bucketName: string): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'DELETE'\n    const query = 'replication'\n    await this.makeRequestAsyncOmit({ method, bucketName, query }, '', [200, 204], '')\n  }\n\n  setBucketReplication(bucketName: string, replicationConfig: ReplicationConfigOpts): void\n  async setBucketReplication(bucketName: string, replicationConfig: ReplicationConfigOpts): Promise<void>\n  async setBucketReplication(bucketName: string, replicationConfig: ReplicationConfigOpts) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isObject(replicationConfig)) {\n      throw new errors.InvalidArgumentError('replicationConfig should be of type \"object\"')\n    } else {\n      if (_.isEmpty(replicationConfig.role)) {\n        throw new errors.InvalidArgumentError('Role cannot be empty')\n      } else if (replicationConfig.role && !isString(replicationConfig.role)) {\n        throw new errors.InvalidArgumentError('Invalid value for role', replicationConfig.role)\n      }\n      if (_.isEmpty(replicationConfig.rules)) {\n        throw new errors.InvalidArgumentError('Minimum one replication rule must be specified')\n      }\n    }\n    const method = 'PUT'\n    const query = 'replication'\n    const headers: Record<string, string> = {}\n\n    const replicationParamsConfig = {\n      ReplicationConfiguration: {\n        Role: replicationConfig.role,\n        Rule: replicationConfig.rules,\n      },\n    }\n\n    const builder = new xml2js.Builder({ renderOpts: { pretty: false }, headless: true })\n    const payload = builder.buildObject(replicationParamsConfig)\n    headers['Content-MD5'] = toMd5(payload)\n    await this.makeRequestAsyncOmit({ method, bucketName, query, headers }, payload)\n  }\n\n  getBucketReplication(bucketName: string): void\n  async getBucketReplication(bucketName: string): Promise<ReplicationConfig>\n  async getBucketReplication(bucketName: string) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'GET'\n    const query = 'replication'\n\n    const httpRes = await this.makeRequestAsync({ method, bucketName, query }, '', [200, 204])\n    const xmlResult = await readAsString(httpRes)\n    return xmlParsers.parseReplicationConfig(xmlResult)\n  }\n\n  getObjectLegalHold(\n    bucketName: string,\n    objectName: string,\n    getOpts?: GetObjectLegalHoldOptions,\n    callback?: ResultCallback<LEGAL_HOLD_STATUS>,\n  ): Promise<LEGAL_HOLD_STATUS>\n  async getObjectLegalHold(\n    bucketName: string,\n    objectName: string,\n    getOpts?: GetObjectLegalHoldOptions,\n  ): Promise<LEGAL_HOLD_STATUS> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n\n    if (getOpts) {\n      if (!isObject(getOpts)) {\n        throw new TypeError('getOpts should be of type \"Object\"')\n      } else if (Object.keys(getOpts).length > 0 && getOpts.versionId && !isString(getOpts.versionId)) {\n        throw new TypeError('versionId should be of type string.:', getOpts.versionId)\n      }\n    }\n\n    const method = 'GET'\n    let query = 'legal-hold'\n\n    if (getOpts?.versionId) {\n      query += `&versionId=${getOpts.versionId}`\n    }\n\n    const httpRes = await this.makeRequestAsync({ method, bucketName, objectName, query }, '', [200])\n    const strRes = await readAsString(httpRes)\n    return parseObjectLegalHoldConfig(strRes)\n  }\n\n  setObjectLegalHold(bucketName: string, objectName: string, setOpts?: PutObjectLegalHoldOptions): void\n  async setObjectLegalHold(\n    bucketName: string,\n    objectName: string,\n    setOpts = {\n      status: LEGAL_HOLD_STATUS.ENABLED,\n    } as PutObjectLegalHoldOptions,\n  ): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n\n    if (!isObject(setOpts)) {\n      throw new TypeError('setOpts should be of type \"Object\"')\n    } else {\n      if (![LEGAL_HOLD_STATUS.ENABLED, LEGAL_HOLD_STATUS.DISABLED].includes(setOpts?.status)) {\n        throw new TypeError('Invalid status: ' + setOpts.status)\n      }\n      if (setOpts.versionId && !setOpts.versionId.length) {\n        throw new TypeError('versionId should be of type string.:' + setOpts.versionId)\n      }\n    }\n\n    const method = 'PUT'\n    let query = 'legal-hold'\n\n    if (setOpts.versionId) {\n      query += `&versionId=${setOpts.versionId}`\n    }\n\n    const config = {\n      Status: setOpts.status,\n    }\n\n    const builder = new xml2js.Builder({ rootName: 'LegalHold', renderOpts: { pretty: false }, headless: true })\n    const payload = builder.buildObject(config)\n    const headers: Record<string, string> = {}\n    headers['Content-MD5'] = toMd5(payload)\n\n    await this.makeRequestAsyncOmit({ method, bucketName, objectName, query, headers }, payload)\n  }\n\n  /**\n   * Get Tags associated with a Bucket\n   */\n  async getBucketTagging(bucketName: string): Promise<Tag[]> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`)\n    }\n\n    const method = 'GET'\n    const query = 'tagging'\n    const requestOptions = { method, bucketName, query }\n\n    const response = await this.makeRequestAsync(requestOptions)\n    const body = await readAsString(response)\n    return xmlParsers.parseTagging(body)\n  }\n\n  /**\n   *  Get the tags associated with a bucket OR an object\n   */\n  async getObjectTagging(bucketName: string, objectName: string, getOpts?: GetObjectOpts): Promise<Tag[]> {\n    const method = 'GET'\n    let query = 'tagging'\n\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidBucketNameError('Invalid object name: ' + objectName)\n    }\n    if (getOpts && !isObject(getOpts)) {\n      throw new errors.InvalidArgumentError('getOpts should be of type \"object\"')\n    }\n\n    if (getOpts && getOpts.versionId) {\n      query = `${query}&versionId=${getOpts.versionId}`\n    }\n    const requestOptions: RequestOption = { method, bucketName, query }\n    if (objectName) {\n      requestOptions['objectName'] = objectName\n    }\n\n    const response = await this.makeRequestAsync(requestOptions)\n    const body = await readAsString(response)\n    return xmlParsers.parseTagging(body)\n  }\n\n  /**\n   *  Set the policy on a bucket or an object prefix.\n   */\n  async setBucketPolicy(bucketName: string, policy: string): Promise<void> {\n    // Validate arguments.\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`)\n    }\n    if (!isString(policy)) {\n      throw new errors.InvalidBucketPolicyError(`Invalid bucket policy: ${policy} - must be \"string\"`)\n    }\n\n    const query = 'policy'\n\n    let method = 'DELETE'\n    if (policy) {\n      method = 'PUT'\n    }\n\n    await this.makeRequestAsyncOmit({ method, bucketName, query }, policy, [204], '')\n  }\n\n  /**\n   * Get the policy on a bucket or an object prefix.\n   */\n  async getBucketPolicy(bucketName: string): Promise<string> {\n    // Validate arguments.\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`)\n    }\n\n    const method = 'GET'\n    const query = 'policy'\n    const res = await this.makeRequestAsync({ method, bucketName, query })\n    return await readAsString(res)\n  }\n\n  async putObjectRetention(bucketName: string, objectName: string, retentionOpts: Retention = {}): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    if (!isObject(retentionOpts)) {\n      throw new errors.InvalidArgumentError('retentionOpts should be of type \"object\"')\n    } else {\n      if (retentionOpts.governanceBypass && !isBoolean(retentionOpts.governanceBypass)) {\n        throw new errors.InvalidArgumentError(`Invalid value for governanceBypass: ${retentionOpts.governanceBypass}`)\n      }\n      if (\n        retentionOpts.mode &&\n        ![RETENTION_MODES.COMPLIANCE, RETENTION_MODES.GOVERNANCE].includes(retentionOpts.mode)\n      ) {\n        throw new errors.InvalidArgumentError(`Invalid object retention mode: ${retentionOpts.mode}`)\n      }\n      if (retentionOpts.retainUntilDate && !isString(retentionOpts.retainUntilDate)) {\n        throw new errors.InvalidArgumentError(`Invalid value for retainUntilDate: ${retentionOpts.retainUntilDate}`)\n      }\n      if (retentionOpts.versionId && !isString(retentionOpts.versionId)) {\n        throw new errors.InvalidArgumentError(`Invalid value for versionId: ${retentionOpts.versionId}`)\n      }\n    }\n\n    const method = 'PUT'\n    let query = 'retention'\n\n    const headers: RequestHeaders = {}\n    if (retentionOpts.governanceBypass) {\n      headers['X-Amz-Bypass-Governance-Retention'] = true\n    }\n\n    const builder = new xml2js.Builder({ rootName: 'Retention', renderOpts: { pretty: false }, headless: true })\n    const params: Record<string, string> = {}\n\n    if (retentionOpts.mode) {\n      params.Mode = retentionOpts.mode\n    }\n    if (retentionOpts.retainUntilDate) {\n      params.RetainUntilDate = retentionOpts.retainUntilDate\n    }\n    if (retentionOpts.versionId) {\n      query += `&versionId=${retentionOpts.versionId}`\n    }\n\n    const payload = builder.buildObject(params)\n\n    headers['Content-MD5'] = toMd5(payload)\n    await this.makeRequestAsyncOmit({ method, bucketName, objectName, query, headers }, payload, [200, 204])\n  }\n\n  getObjectLockConfig(bucketName: string, callback: ResultCallback<ObjectLockInfo>): void\n  getObjectLockConfig(bucketName: string): void\n  async getObjectLockConfig(bucketName: string): Promise<ObjectLockInfo>\n  async getObjectLockConfig(bucketName: string) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'GET'\n    const query = 'object-lock'\n\n    const httpRes = await this.makeRequestAsync({ method, bucketName, query })\n    const xmlResult = await readAsString(httpRes)\n    return xmlParsers.parseObjectLockConfig(xmlResult)\n  }\n\n  setObjectLockConfig(bucketName: string, lockConfigOpts: Omit<ObjectLockInfo, 'objectLockEnabled'>): void\n  async setObjectLockConfig(\n    bucketName: string,\n    lockConfigOpts: Omit<ObjectLockInfo, 'objectLockEnabled'>,\n  ): Promise<void>\n  async setObjectLockConfig(bucketName: string, lockConfigOpts: Omit<ObjectLockInfo, 'objectLockEnabled'>) {\n    const retentionModes = [RETENTION_MODES.COMPLIANCE, RETENTION_MODES.GOVERNANCE]\n    const validUnits = [RETENTION_VALIDITY_UNITS.DAYS, RETENTION_VALIDITY_UNITS.YEARS]\n\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n\n    if (lockConfigOpts.mode && !retentionModes.includes(lockConfigOpts.mode)) {\n      throw new TypeError(`lockConfigOpts.mode should be one of ${retentionModes}`)\n    }\n    if (lockConfigOpts.unit && !validUnits.includes(lockConfigOpts.unit)) {\n      throw new TypeError(`lockConfigOpts.unit should be one of ${validUnits}`)\n    }\n    if (lockConfigOpts.validity && !isNumber(lockConfigOpts.validity)) {\n      throw new TypeError(`lockConfigOpts.validity should be a number`)\n    }\n\n    const method = 'PUT'\n    const query = 'object-lock'\n\n    const config: ObjectLockConfigParam = {\n      ObjectLockEnabled: 'Enabled',\n    }\n    const configKeys = Object.keys(lockConfigOpts)\n\n    const isAllKeysSet = ['unit', 'mode', 'validity'].every((lck) => configKeys.includes(lck))\n    // Check if keys are present and all keys are present.\n    if (configKeys.length > 0) {\n      if (!isAllKeysSet) {\n        throw new TypeError(\n          `lockConfigOpts.mode,lockConfigOpts.unit,lockConfigOpts.validity all the properties should be specified.`,\n        )\n      } else {\n        config.Rule = {\n          DefaultRetention: {},\n        }\n        if (lockConfigOpts.mode) {\n          config.Rule.DefaultRetention.Mode = lockConfigOpts.mode\n        }\n        if (lockConfigOpts.unit === RETENTION_VALIDITY_UNITS.DAYS) {\n          config.Rule.DefaultRetention.Days = lockConfigOpts.validity\n        } else if (lockConfigOpts.unit === RETENTION_VALIDITY_UNITS.YEARS) {\n          config.Rule.DefaultRetention.Years = lockConfigOpts.validity\n        }\n      }\n    }\n\n    const builder = new xml2js.Builder({\n      rootName: 'ObjectLockConfiguration',\n      renderOpts: { pretty: false },\n      headless: true,\n    })\n    const payload = builder.buildObject(config)\n\n    const headers: RequestHeaders = {}\n    headers['Content-MD5'] = toMd5(payload)\n\n    await this.makeRequestAsyncOmit({ method, bucketName, query, headers }, payload)\n  }\n\n  async getBucketVersioning(bucketName: string): Promise<BucketVersioningConfiguration> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'GET'\n    const query = 'versioning'\n\n    const httpRes = await this.makeRequestAsync({ method, bucketName, query })\n    const xmlResult = await readAsString(httpRes)\n    return await xmlParsers.parseBucketVersioningConfig(xmlResult)\n  }\n\n  async setBucketVersioning(bucketName: string, versionConfig: BucketVersioningConfiguration): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!Object.keys(versionConfig).length) {\n      throw new errors.InvalidArgumentError('versionConfig should be of type \"object\"')\n    }\n\n    const method = 'PUT'\n    const query = 'versioning'\n    const builder = new xml2js.Builder({\n      rootName: 'VersioningConfiguration',\n      renderOpts: { pretty: false },\n      headless: true,\n    })\n    const payload = builder.buildObject(versionConfig)\n\n    await this.makeRequestAsyncOmit({ method, bucketName, query }, payload)\n  }\n\n  private async setTagging(taggingParams: PutTaggingParams): Promise<void> {\n    const { bucketName, objectName, tags, putOpts } = taggingParams\n    const method = 'PUT'\n    let query = 'tagging'\n\n    if (putOpts && putOpts?.versionId) {\n      query = `${query}&versionId=${putOpts.versionId}`\n    }\n    const tagsList = []\n    for (const [key, value] of Object.entries(tags)) {\n      tagsList.push({ Key: key, Value: value })\n    }\n    const taggingConfig = {\n      Tagging: {\n        TagSet: {\n          Tag: tagsList,\n        },\n      },\n    }\n    const headers = {} as RequestHeaders\n    const builder = new xml2js.Builder({ headless: true, renderOpts: { pretty: false } })\n    const payloadBuf = Buffer.from(builder.buildObject(taggingConfig))\n    const requestOptions = {\n      method,\n      bucketName,\n      query,\n      headers,\n\n      ...(objectName && { objectName: objectName }),\n    }\n\n    headers['Content-MD5'] = toMd5(payloadBuf)\n\n    await this.makeRequestAsyncOmit(requestOptions, payloadBuf)\n  }\n\n  private async removeTagging({ bucketName, objectName, removeOpts }: RemoveTaggingParams): Promise<void> {\n    const method = 'DELETE'\n    let query = 'tagging'\n\n    if (removeOpts && Object.keys(removeOpts).length && removeOpts.versionId) {\n      query = `${query}&versionId=${removeOpts.versionId}`\n    }\n    const requestOptions = { method, bucketName, objectName, query }\n\n    if (objectName) {\n      requestOptions['objectName'] = objectName\n    }\n    await this.makeRequestAsync(requestOptions, '', [200, 204])\n  }\n\n  async setBucketTagging(bucketName: string, tags: Tags): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isPlainObject(tags)) {\n      throw new errors.InvalidArgumentError('tags should be of type \"object\"')\n    }\n    if (Object.keys(tags).length > 10) {\n      throw new errors.InvalidArgumentError('maximum tags allowed is 10\"')\n    }\n\n    await this.setTagging({ bucketName, tags })\n  }\n\n  async removeBucketTagging(bucketName: string) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    await this.removeTagging({ bucketName })\n  }\n\n  async setObjectTagging(bucketName: string, objectName: string, tags: Tags, putOpts?: TaggingOpts) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidBucketNameError('Invalid object name: ' + objectName)\n    }\n\n    if (!isPlainObject(tags)) {\n      throw new errors.InvalidArgumentError('tags should be of type \"object\"')\n    }\n    if (Object.keys(tags).length > 10) {\n      throw new errors.InvalidArgumentError('Maximum tags allowed is 10\"')\n    }\n\n    await this.setTagging({ bucketName, objectName, tags, putOpts })\n  }\n\n  async removeObjectTagging(bucketName: string, objectName: string, removeOpts: TaggingOpts) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidBucketNameError('Invalid object name: ' + objectName)\n    }\n    if (removeOpts && Object.keys(removeOpts).length && !isObject(removeOpts)) {\n      throw new errors.InvalidArgumentError('removeOpts should be of type \"object\"')\n    }\n\n    await this.removeTagging({ bucketName, objectName, removeOpts })\n  }\n\n  async selectObjectContent(\n    bucketName: string,\n    objectName: string,\n    selectOpts: SelectOptions,\n  ): Promise<SelectResults | undefined> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    if (!_.isEmpty(selectOpts)) {\n      if (!isString(selectOpts.expression)) {\n        throw new TypeError('sqlExpression should be of type \"string\"')\n      }\n      if (!_.isEmpty(selectOpts.inputSerialization)) {\n        if (!isObject(selectOpts.inputSerialization)) {\n          throw new TypeError('inputSerialization should be of type \"object\"')\n        }\n      } else {\n        throw new TypeError('inputSerialization is required')\n      }\n      if (!_.isEmpty(selectOpts.outputSerialization)) {\n        if (!isObject(selectOpts.outputSerialization)) {\n          throw new TypeError('outputSerialization should be of type \"object\"')\n        }\n      } else {\n        throw new TypeError('outputSerialization is required')\n      }\n    } else {\n      throw new TypeError('valid select configuration is required')\n    }\n\n    const method = 'POST'\n    const query = `select&select-type=2`\n\n    const config: Record<string, unknown>[] = [\n      {\n        Expression: selectOpts.expression,\n      },\n      {\n        ExpressionType: selectOpts.expressionType || 'SQL',\n      },\n      {\n        InputSerialization: [selectOpts.inputSerialization],\n      },\n      {\n        OutputSerialization: [selectOpts.outputSerialization],\n      },\n    ]\n\n    // Optional\n    if (selectOpts.requestProgress) {\n      config.push({ RequestProgress: selectOpts?.requestProgress })\n    }\n    // Optional\n    if (selectOpts.scanRange) {\n      config.push({ ScanRange: selectOpts.scanRange })\n    }\n\n    const builder = new xml2js.Builder({\n      rootName: 'SelectObjectContentRequest',\n      renderOpts: { pretty: false },\n      headless: true,\n    })\n    const payload = builder.buildObject(config)\n\n    const res = await this.makeRequestAsync({ method, bucketName, objectName, query }, payload)\n    const body = await readAsBuffer(res)\n    return parseSelectObjectContentResponse(body)\n  }\n\n  private async applyBucketLifecycle(bucketName: string, policyConfig: LifeCycleConfigParam): Promise<void> {\n    const method = 'PUT'\n    const query = 'lifecycle'\n\n    const headers: RequestHeaders = {}\n    const builder = new xml2js.Builder({\n      rootName: 'LifecycleConfiguration',\n      headless: true,\n      renderOpts: { pretty: false },\n    })\n    const payload = builder.buildObject(policyConfig)\n    headers['Content-MD5'] = toMd5(payload)\n\n    await this.makeRequestAsyncOmit({ method, bucketName, query, headers }, payload)\n  }\n\n  async removeBucketLifecycle(bucketName: string): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'DELETE'\n    const query = 'lifecycle'\n    await this.makeRequestAsyncOmit({ method, bucketName, query }, '', [204])\n  }\n\n  async setBucketLifecycle(bucketName: string, lifeCycleConfig: LifeCycleConfigParam): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (_.isEmpty(lifeCycleConfig)) {\n      await this.removeBucketLifecycle(bucketName)\n    } else {\n      await this.applyBucketLifecycle(bucketName, lifeCycleConfig)\n    }\n  }\n\n  async getBucketLifecycle(bucketName: string): Promise<LifecycleConfig | null> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'GET'\n    const query = 'lifecycle'\n\n    const res = await this.makeRequestAsync({ method, bucketName, query })\n    const body = await readAsString(res)\n    return xmlParsers.parseLifecycleConfig(body)\n  }\n\n  async setBucketEncryption(bucketName: string, encryptionConfig?: EncryptionConfig): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!_.isEmpty(encryptionConfig) && encryptionConfig.Rule.length > 1) {\n      throw new errors.InvalidArgumentError('Invalid Rule length. Only one rule is allowed.: ' + encryptionConfig.Rule)\n    }\n\n    let encryptionObj = encryptionConfig\n    if (_.isEmpty(encryptionConfig)) {\n      encryptionObj = {\n        // Default MinIO Server Supported Rule\n        Rule: [\n          {\n            ApplyServerSideEncryptionByDefault: {\n              SSEAlgorithm: 'AES256',\n            },\n          },\n        ],\n      }\n    }\n\n    const method = 'PUT'\n    const query = 'encryption'\n    const builder = new xml2js.Builder({\n      rootName: 'ServerSideEncryptionConfiguration',\n      renderOpts: { pretty: false },\n      headless: true,\n    })\n    const payload = builder.buildObject(encryptionObj)\n\n    const headers: RequestHeaders = {}\n    headers['Content-MD5'] = toMd5(payload)\n\n    await this.makeRequestAsyncOmit({ method, bucketName, query, headers }, payload)\n  }\n\n  async getBucketEncryption(bucketName: string) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'GET'\n    const query = 'encryption'\n\n    const res = await this.makeRequestAsync({ method, bucketName, query })\n    const body = await readAsString(res)\n    return xmlParsers.parseBucketEncryptionConfig(body)\n  }\n\n  async removeBucketEncryption(bucketName: string) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'DELETE'\n    const query = 'encryption'\n\n    await this.makeRequestAsyncOmit({ method, bucketName, query }, '', [204])\n  }\n\n  async getObjectRetention(\n    bucketName: string,\n    objectName: string,\n    getOpts?: GetObjectRetentionOpts,\n  ): Promise<ObjectRetentionInfo | null | undefined> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    if (getOpts && !isObject(getOpts)) {\n      throw new errors.InvalidArgumentError('getOpts should be of type \"object\"')\n    } else if (getOpts?.versionId && !isString(getOpts.versionId)) {\n      throw new errors.InvalidArgumentError('versionId should be of type \"string\"')\n    }\n\n    const method = 'GET'\n    let query = 'retention'\n    if (getOpts?.versionId) {\n      query += `&versionId=${getOpts.versionId}`\n    }\n    const res = await this.makeRequestAsync({ method, bucketName, objectName, query })\n    const body = await readAsString(res)\n    return xmlParsers.parseObjectRetentionConfig(body)\n  }\n\n  async removeObjects(bucketName: string, objectsList: RemoveObjectsParam): Promise<RemoveObjectsResponse[]> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!Array.isArray(objectsList)) {\n      throw new errors.InvalidArgumentError('objectsList should be a list')\n    }\n\n    const runDeleteObjects = async (batch: RemoveObjectsParam): Promise<RemoveObjectsResponse[]> => {\n      const delObjects: RemoveObjectsRequestEntry[] = batch.map((value) => {\n        return isObject(value) ? { Key: value.name, VersionId: value.versionId } : { Key: value }\n      })\n\n      const remObjects = { Delete: { Quiet: true, Object: delObjects } }\n      const payload = Buffer.from(new xml2js.Builder({ headless: true }).buildObject(remObjects))\n      const headers: RequestHeaders = { 'Content-MD5': toMd5(payload) }\n\n      const res = await this.makeRequestAsync({ method: 'POST', bucketName, query: 'delete', headers }, payload)\n      const body = await readAsString(res)\n      return xmlParsers.removeObjectsParser(body)\n    }\n\n    const maxEntries = 1000 // max entries accepted in server for DeleteMultipleObjects API.\n    // Client side batching\n    const batches = []\n    for (let i = 0; i < objectsList.length; i += maxEntries) {\n      batches.push(objectsList.slice(i, i + maxEntries))\n    }\n\n    const batchResults = await Promise.all(batches.map(runDeleteObjects))\n    return batchResults.flat()\n  }\n\n  async removeIncompleteUpload(bucketName: string, objectName: string): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.IsValidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n    const removeUploadId = await this.findUploadId(bucketName, objectName)\n    const method = 'DELETE'\n    const query = `uploadId=${removeUploadId}`\n    await this.makeRequestAsyncOmit({ method, bucketName, objectName, query }, '', [204])\n  }\n\n  private async copyObjectV1(\n    targetBucketName: string,\n    targetObjectName: string,\n    sourceBucketNameAndObjectName: string,\n    conditions?: null | CopyConditions,\n  ) {\n    if (typeof conditions == 'function') {\n      conditions = null\n    }\n\n    if (!isValidBucketName(targetBucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + targetBucketName)\n    }\n    if (!isValidObjectName(targetObjectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${targetObjectName}`)\n    }\n    if (!isString(sourceBucketNameAndObjectName)) {\n      throw new TypeError('sourceBucketNameAndObjectName should be of type \"string\"')\n    }\n    if (sourceBucketNameAndObjectName === '') {\n      throw new errors.InvalidPrefixError(`Empty source prefix`)\n    }\n\n    if (conditions != null && !(conditions instanceof CopyConditions)) {\n      throw new TypeError('conditions should be of type \"CopyConditions\"')\n    }\n\n    const headers: RequestHeaders = {}\n    headers['x-amz-copy-source'] = uriResourceEscape(sourceBucketNameAndObjectName)\n\n    if (conditions) {\n      if (conditions.modified !== '') {\n        headers['x-amz-copy-source-if-modified-since'] = conditions.modified\n      }\n      if (conditions.unmodified !== '') {\n        headers['x-amz-copy-source-if-unmodified-since'] = conditions.unmodified\n      }\n      if (conditions.matchETag !== '') {\n        headers['x-amz-copy-source-if-match'] = conditions.matchETag\n      }\n      if (conditions.matchETagExcept !== '') {\n        headers['x-amz-copy-source-if-none-match'] = conditions.matchETagExcept\n      }\n    }\n\n    const method = 'PUT'\n\n    const res = await this.makeRequestAsync({\n      method,\n      bucketName: targetBucketName,\n      objectName: targetObjectName,\n      headers,\n    })\n    const body = await readAsString(res)\n    return xmlParsers.parseCopyObject(body)\n  }\n\n  private async copyObjectV2(\n    sourceConfig: CopySourceOptions,\n    destConfig: CopyDestinationOptions,\n  ): Promise<CopyObjectResultV2> {\n    if (!(sourceConfig instanceof CopySourceOptions)) {\n      throw new errors.InvalidArgumentError('sourceConfig should of type CopySourceOptions ')\n    }\n    if (!(destConfig instanceof CopyDestinationOptions)) {\n      throw new errors.InvalidArgumentError('destConfig should of type CopyDestinationOptions ')\n    }\n    if (!destConfig.validate()) {\n      return Promise.reject()\n    }\n    if (!destConfig.validate()) {\n      return Promise.reject()\n    }\n\n    const headers = Object.assign({}, sourceConfig.getHeaders(), destConfig.getHeaders())\n\n    const bucketName = destConfig.Bucket\n    const objectName = destConfig.Object\n\n    const method = 'PUT'\n\n    const res = await this.makeRequestAsync({ method, bucketName, objectName, headers })\n    const body = await readAsString(res)\n    const copyRes = xmlParsers.parseCopyObject(body)\n    const resHeaders: IncomingHttpHeaders = res.headers\n\n    const sizeHeaderValue = resHeaders && resHeaders['content-length']\n    const size = typeof sizeHeaderValue === 'number' ? sizeHeaderValue : undefined\n\n    return {\n      Bucket: destConfig.Bucket,\n      Key: destConfig.Object,\n      LastModified: copyRes.lastModified,\n      MetaData: extractMetadata(resHeaders as ResponseHeader),\n      VersionId: getVersionId(resHeaders as ResponseHeader),\n      SourceVersionId: getSourceVersionId(resHeaders as ResponseHeader),\n      Etag: sanitizeETag(resHeaders.etag),\n      Size: size,\n    }\n  }\n\n  async copyObject(source: CopySourceOptions, dest: CopyDestinationOptions): Promise<CopyObjectResult>\n  async copyObject(\n    targetBucketName: string,\n    targetObjectName: string,\n    sourceBucketNameAndObjectName: string,\n    conditions?: CopyConditions,\n  ): Promise<CopyObjectResult>\n  async copyObject(...allArgs: CopyObjectParams): Promise<CopyObjectResult> {\n    if (typeof allArgs[0] === 'string') {\n      const [targetBucketName, targetObjectName, sourceBucketNameAndObjectName, conditions] = allArgs as [\n        string,\n        string,\n        string,\n        CopyConditions?,\n      ]\n      return await this.copyObjectV1(targetBucketName, targetObjectName, sourceBucketNameAndObjectName, conditions)\n    }\n    const [source, dest] = allArgs as [CopySourceOptions, CopyDestinationOptions]\n    return await this.copyObjectV2(source, dest)\n  }\n\n  async uploadPart(\n    partConfig: {\n      bucketName: string\n      objectName: string\n      uploadID: string\n      partNumber: number\n      headers: RequestHeaders\n    },\n    payload?: Binary,\n  ) {\n    const { bucketName, objectName, uploadID, partNumber, headers } = partConfig\n\n    const method = 'PUT'\n    const query = `uploadId=${uploadID}&partNumber=${partNumber}`\n    const requestOptions = { method, bucketName, objectName: objectName, query, headers }\n    const res = await this.makeRequestAsync(requestOptions, payload)\n    const body = await readAsString(res)\n    const partRes = uploadPartParser(body)\n    const partEtagVal = sanitizeETag(res.headers.etag) || sanitizeETag(partRes.ETag)\n    return {\n      etag: partEtagVal,\n      key: objectName,\n      part: partNumber,\n    }\n  }\n\n  async composeObject(\n    destObjConfig: CopyDestinationOptions,\n    sourceObjList: CopySourceOptions[],\n    { maxConcurrency = 10 } = {},\n  ): Promise<boolean | { etag: string; versionId: string | null } | Promise<void> | CopyObjectResult> {\n    const sourceFilesLength = sourceObjList.length\n\n    if (!Array.isArray(sourceObjList)) {\n      throw new errors.InvalidArgumentError('sourceConfig should an array of CopySourceOptions ')\n    }\n    if (!(destObjConfig instanceof CopyDestinationOptions)) {\n      throw new errors.InvalidArgumentError('destConfig should of type CopyDestinationOptions ')\n    }\n\n    if (sourceFilesLength < 1 || sourceFilesLength > PART_CONSTRAINTS.MAX_PARTS_COUNT) {\n      throw new errors.InvalidArgumentError(\n        `\"There must be as least one and up to ${PART_CONSTRAINTS.MAX_PARTS_COUNT} source objects.`,\n      )\n    }\n\n    for (let i = 0; i < sourceFilesLength; i++) {\n      const sObj = sourceObjList[i] as CopySourceOptions\n      if (!sObj.validate()) {\n        return false\n      }\n    }\n\n    if (!(destObjConfig as CopyDestinationOptions).validate()) {\n      return false\n    }\n\n    const getStatOptions = (srcConfig: CopySourceOptions) => {\n      let statOpts = {}\n      if (!_.isEmpty(srcConfig.VersionID)) {\n        statOpts = {\n          versionId: srcConfig.VersionID,\n        }\n      }\n      return statOpts\n    }\n    const srcObjectSizes: number[] = []\n    let totalSize = 0\n    let totalParts = 0\n\n    const sourceObjStats = sourceObjList.map((srcItem) =>\n      this.statObject(srcItem.Bucket, srcItem.Object, getStatOptions(srcItem)),\n    )\n\n    const srcObjectInfos = await Promise.all(sourceObjStats)\n\n    const validatedStats = srcObjectInfos.map((resItemStat, index) => {\n      const srcConfig: CopySourceOptions | undefined = sourceObjList[index]\n\n      let srcCopySize = resItemStat.size\n      // Check if a segment is specified, and if so, is the\n      // segment within object bounds?\n      if (srcConfig && srcConfig.MatchRange) {\n        // Since range is specified,\n        //    0 <= src.srcStart <= src.srcEnd\n        // so only invalid case to check is:\n        const srcStart = srcConfig.Start\n        const srcEnd = srcConfig.End\n        if (srcEnd >= srcCopySize || srcStart < 0) {\n          throw new errors.InvalidArgumentError(\n            `CopySrcOptions ${index} has invalid segment-to-copy [${srcStart}, ${srcEnd}] (size is ${srcCopySize})`,\n          )\n        }\n        srcCopySize = srcEnd - srcStart + 1\n      }\n\n      // Only the last source may be less than `absMinPartSize`\n      if (srcCopySize < PART_CONSTRAINTS.ABS_MIN_PART_SIZE && index < sourceFilesLength - 1) {\n        throw new errors.InvalidArgumentError(\n          `CopySrcOptions ${index} is too small (${srcCopySize}) and it is not the last part.`,\n        )\n      }\n\n      // Is data to copy too large?\n      totalSize += srcCopySize\n      if (totalSize > PART_CONSTRAINTS.MAX_MULTIPART_PUT_OBJECT_SIZE) {\n        throw new errors.InvalidArgumentError(`Cannot compose an object of size ${totalSize} (> 5TiB)`)\n      }\n\n      // record source size\n      srcObjectSizes[index] = srcCopySize\n\n      // calculate parts needed for current source\n      totalParts += partsRequired(srcCopySize)\n      // Do we need more parts than we are allowed?\n      if (totalParts > PART_CONSTRAINTS.MAX_PARTS_COUNT) {\n        throw new errors.InvalidArgumentError(\n          `Your proposed compose object requires more than ${PART_CONSTRAINTS.MAX_PARTS_COUNT} parts`,\n        )\n      }\n\n      return resItemStat\n    })\n\n    if ((totalParts === 1 && totalSize <= PART_CONSTRAINTS.MAX_PART_SIZE) || totalSize === 0) {\n      return await this.copyObject(sourceObjList[0] as CopySourceOptions, destObjConfig) // use copyObjectV2\n    }\n\n    // preserve etag to avoid modification of object while copying.\n    for (let i = 0; i < sourceFilesLength; i++) {\n      ;(sourceObjList[i] as CopySourceOptions).MatchETag = (validatedStats[i] as BucketItemStat).etag\n    }\n\n    const splitPartSizeList = validatedStats.map((resItemStat, idx) => {\n      return calculateEvenSplits(srcObjectSizes[idx] as number, sourceObjList[idx] as CopySourceOptions)\n    })\n\n    const getUploadPartConfigList = (uploadId: string) => {\n      const uploadPartConfigList: UploadPartConfig[] = []\n\n      splitPartSizeList.forEach((splitSize, splitIndex: number) => {\n        if (splitSize) {\n          const { startIndex: startIdx, endIndex: endIdx, objInfo: objConfig } = splitSize\n\n          const partIndex = splitIndex + 1 // part index starts from 1.\n          const totalUploads = Array.from(startIdx)\n\n          const headers = (sourceObjList[splitIndex] as CopySourceOptions).getHeaders()\n\n          totalUploads.forEach((splitStart, upldCtrIdx) => {\n            const splitEnd = endIdx[upldCtrIdx]\n\n            const sourceObj = `${objConfig.Bucket}/${objConfig.Object}`\n            headers['x-amz-copy-source'] = `${sourceObj}`\n            headers['x-amz-copy-source-range'] = `bytes=${splitStart}-${splitEnd}`\n\n            const uploadPartConfig = {\n              bucketName: destObjConfig.Bucket,\n              objectName: destObjConfig.Object,\n              uploadID: uploadId,\n              partNumber: partIndex,\n              headers: headers,\n              sourceObj: sourceObj,\n            }\n\n            uploadPartConfigList.push(uploadPartConfig)\n          })\n        }\n      })\n\n      return uploadPartConfigList\n    }\n\n    const uploadAllParts = async (uploadList: UploadPartConfig[]) => {\n      const partUploads: Awaited<ReturnType<typeof this.uploadPart>>[] = []\n\n      // Process upload parts in batches to avoid too many concurrent requests\n      for (const batch of _.chunk(uploadList, maxConcurrency)) {\n        const batchResults = await Promise.all(batch.map((item) => this.uploadPart(item)))\n\n        partUploads.push(...batchResults)\n      }\n\n      // Process results here if needed\n      return partUploads\n    }\n\n    const performUploadParts = async (uploadId: string) => {\n      const uploadList = getUploadPartConfigList(uploadId)\n      const partsRes = await uploadAllParts(uploadList)\n      return partsRes.map((partCopy) => ({ etag: partCopy.etag, part: partCopy.part }))\n    }\n\n    const newUploadHeaders = destObjConfig.getHeaders()\n\n    const uploadId = await this.initiateNewMultipartUpload(destObjConfig.Bucket, destObjConfig.Object, newUploadHeaders)\n    try {\n      const partsDone = await performUploadParts(uploadId)\n      return await this.completeMultipartUpload(destObjConfig.Bucket, destObjConfig.Object, uploadId, partsDone)\n    } catch (err) {\n      return await this.abortMultipartUpload(destObjConfig.Bucket, destObjConfig.Object, uploadId)\n    }\n  }\n\n  async presignedUrl(\n    method: string,\n    bucketName: string,\n    objectName: string,\n    expires?: number | PreSignRequestParams | undefined,\n    reqParams?: PreSignRequestParams | Date,\n    requestDate?: Date,\n  ): Promise<string> {\n    if (this.anonymous) {\n      throw new errors.AnonymousRequestError(`Presigned ${method} url cannot be generated for anonymous requests`)\n    }\n\n    if (!expires) {\n      expires = PRESIGN_EXPIRY_DAYS_MAX\n    }\n    if (!reqParams) {\n      reqParams = {}\n    }\n    if (!requestDate) {\n      requestDate = new Date()\n    }\n\n    // Type assertions\n    if (expires && typeof expires !== 'number') {\n      throw new TypeError('expires should be of type \"number\"')\n    }\n    if (reqParams && typeof reqParams !== 'object') {\n      throw new TypeError('reqParams should be of type \"object\"')\n    }\n    if ((requestDate && !(requestDate instanceof Date)) || (requestDate && isNaN(requestDate?.getTime()))) {\n      throw new TypeError('requestDate should be of type \"Date\" and valid')\n    }\n\n    const query = reqParams ? qs.stringify(reqParams) : undefined\n\n    try {\n      const region = await this.getBucketRegionAsync(bucketName)\n      await this.checkAndRefreshCreds()\n      const reqOptions = this.getRequestOptions({ method, region, bucketName, objectName, query })\n\n      return presignSignatureV4(\n        reqOptions,\n        this.accessKey,\n        this.secretKey,\n        this.sessionToken,\n        region,\n        requestDate,\n        expires,\n      )\n    } catch (err) {\n      if (err instanceof errors.InvalidBucketNameError) {\n        throw new errors.InvalidArgumentError(`Unable to get bucket region for ${bucketName}.`)\n      }\n\n      throw err\n    }\n  }\n\n  async presignedGetObject(\n    bucketName: string,\n    objectName: string,\n    expires?: number,\n    respHeaders?: PreSignRequestParams | Date,\n    requestDate?: Date,\n  ): Promise<string> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n\n    const validRespHeaders = [\n      'response-content-type',\n      'response-content-language',\n      'response-expires',\n      'response-cache-control',\n      'response-content-disposition',\n      'response-content-encoding',\n    ]\n    validRespHeaders.forEach((header) => {\n      // @ts-ignore\n      if (respHeaders !== undefined && respHeaders[header] !== undefined && !isString(respHeaders[header])) {\n        throw new TypeError(`response header ${header} should be of type \"string\"`)\n      }\n    })\n    return this.presignedUrl('GET', bucketName, objectName, expires, respHeaders, requestDate)\n  }\n\n  async presignedPutObject(bucketName: string, objectName: string, expires?: number): Promise<string> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`)\n    }\n    if (!isValidObjectName(objectName)) {\n      throw new errors.InvalidObjectNameError(`Invalid object name: ${objectName}`)\n    }\n\n    return this.presignedUrl('PUT', bucketName, objectName, expires)\n  }\n\n  newPostPolicy(): PostPolicy {\n    return new PostPolicy()\n  }\n\n  async presignedPostPolicy(postPolicy: PostPolicy): Promise<PostPolicyResult> {\n    if (this.anonymous) {\n      throw new errors.AnonymousRequestError('Presigned POST policy cannot be generated for anonymous requests')\n    }\n    if (!isObject(postPolicy)) {\n      throw new TypeError('postPolicy should be of type \"object\"')\n    }\n    const bucketName = postPolicy.formData.bucket as string\n    try {\n      const region = await this.getBucketRegionAsync(bucketName)\n\n      const date = new Date()\n      const dateStr = makeDateLong(date)\n      await this.checkAndRefreshCreds()\n\n      if (!postPolicy.policy.expiration) {\n        // 'expiration' is mandatory field for S3.\n        // Set default expiration date of 7 days.\n        const expires = new Date()\n        expires.setSeconds(PRESIGN_EXPIRY_DAYS_MAX)\n        postPolicy.setExpires(expires)\n      }\n\n      postPolicy.policy.conditions.push(['eq', '$x-amz-date', dateStr])\n      postPolicy.formData['x-amz-date'] = dateStr\n\n      postPolicy.policy.conditions.push(['eq', '$x-amz-algorithm', 'AWS4-HMAC-SHA256'])\n      postPolicy.formData['x-amz-algorithm'] = 'AWS4-HMAC-SHA256'\n\n      postPolicy.policy.conditions.push(['eq', '$x-amz-credential', this.accessKey + '/' + getScope(region, date)])\n      postPolicy.formData['x-amz-credential'] = this.accessKey + '/' + getScope(region, date)\n\n      if (this.sessionToken) {\n        postPolicy.policy.conditions.push(['eq', '$x-amz-security-token', this.sessionToken])\n        postPolicy.formData['x-amz-security-token'] = this.sessionToken\n      }\n\n      const policyBase64 = Buffer.from(JSON.stringify(postPolicy.policy)).toString('base64')\n\n      postPolicy.formData.policy = policyBase64\n\n      postPolicy.formData['x-amz-signature'] = postPresignSignatureV4(region, date, this.secretKey, policyBase64)\n      const opts = {\n        region: region,\n        bucketName: bucketName,\n        method: 'POST',\n      }\n      const reqOptions = this.getRequestOptions(opts)\n      const portStr = this.port == 80 || this.port === 443 ? '' : `:${this.port.toString()}`\n      const urlStr = `${reqOptions.protocol}//${reqOptions.host}${portStr}${reqOptions.path}`\n      return { postURL: urlStr, formData: postPolicy.formData }\n    } catch (err) {\n      if (err instanceof errors.InvalidBucketNameError) {\n        throw new errors.InvalidArgumentError(`Unable to get bucket region for ${bucketName}.`)\n      }\n\n      throw err\n    }\n  }\n  // list a batch of objects\n  async listObjectsQuery(bucketName: string, prefix?: string, marker?: string, listQueryOpts?: ListObjectQueryOpts) {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isString(prefix)) {\n      throw new TypeError('prefix should be of type \"string\"')\n    }\n    if (marker && !isString(marker)) {\n      throw new TypeError('marker should be of type \"string\"')\n    }\n\n    if (listQueryOpts && !isObject(listQueryOpts)) {\n      throw new TypeError('listQueryOpts should be of type \"object\"')\n    }\n    let { Delimiter, MaxKeys, IncludeVersion, versionIdMarker, keyMarker } = listQueryOpts as ListObjectQueryOpts\n\n    if (!isString(Delimiter)) {\n      throw new TypeError('Delimiter should be of type \"string\"')\n    }\n    if (!isNumber(MaxKeys)) {\n      throw new TypeError('MaxKeys should be of type \"number\"')\n    }\n\n    const queries = []\n    // escape every value in query string, except maxKeys\n    queries.push(`prefix=${uriEscape(prefix)}`)\n    queries.push(`delimiter=${uriEscape(Delimiter)}`)\n    queries.push(`encoding-type=url`)\n\n    if (IncludeVersion) {\n      queries.push(`versions`)\n    }\n\n    if (IncludeVersion) {\n      // v1 version listing..\n      if (keyMarker) {\n        queries.push(`key-marker=${keyMarker}`)\n      }\n      if (versionIdMarker) {\n        queries.push(`version-id-marker=${versionIdMarker}`)\n      }\n    } else if (marker) {\n      marker = uriEscape(marker)\n      queries.push(`marker=${marker}`)\n    }\n\n    // no need to escape maxKeys\n    if (MaxKeys) {\n      if (MaxKeys >= 1000) {\n        MaxKeys = 1000\n      }\n      queries.push(`max-keys=${MaxKeys}`)\n    }\n    queries.sort()\n    let query = ''\n    if (queries.length > 0) {\n      query = `${queries.join('&')}`\n    }\n\n    const method = 'GET'\n    const res = await this.makeRequestAsync({ method, bucketName, query })\n    const body = await readAsString(res)\n    const listQryList = parseListObjects(body)\n    return listQryList\n  }\n\n  listObjects(\n    bucketName: string,\n    prefix?: string,\n    recursive?: boolean,\n    listOpts?: ListObjectQueryOpts | undefined,\n  ): BucketStream<ObjectInfo> {\n    if (prefix === undefined) {\n      prefix = ''\n    }\n    if (recursive === undefined) {\n      recursive = false\n    }\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidPrefix(prefix)) {\n      throw new errors.InvalidPrefixError(`Invalid prefix : ${prefix}`)\n    }\n    if (!isString(prefix)) {\n      throw new TypeError('prefix should be of type \"string\"')\n    }\n    if (!isBoolean(recursive)) {\n      throw new TypeError('recursive should be of type \"boolean\"')\n    }\n    if (listOpts && !isObject(listOpts)) {\n      throw new TypeError('listOpts should be of type \"object\"')\n    }\n    let marker: string | undefined = ''\n    let keyMarker: string | undefined = ''\n    let versionIdMarker: string | undefined = ''\n    let objects: ObjectInfo[] = []\n    let ended = false\n    const readStream: stream.Readable = new stream.Readable({ objectMode: true })\n    readStream._read = async () => {\n      // push one object per _read()\n      if (objects.length) {\n        readStream.push(objects.shift())\n        return\n      }\n      if (ended) {\n        return readStream.push(null)\n      }\n\n      try {\n        const listQueryOpts = {\n          Delimiter: recursive ? '' : '/', // if recursive is false set delimiter to '/'\n          MaxKeys: 1000,\n          IncludeVersion: listOpts?.IncludeVersion,\n          // version listing specific options\n          keyMarker: keyMarker,\n          versionIdMarker: versionIdMarker,\n        }\n\n        const result: ListObjectQueryRes = await this.listObjectsQuery(bucketName, prefix, marker, listQueryOpts)\n        if (result.isTruncated) {\n          marker = result.nextMarker || undefined\n          if (result.keyMarker) {\n            keyMarker = result.keyMarker\n          }\n          if (result.versionIdMarker) {\n            versionIdMarker = result.versionIdMarker\n          }\n        } else {\n          ended = true\n        }\n        if (result.objects) {\n          objects = result.objects\n        }\n        // @ts-ignore\n        readStream._read()\n      } catch (err) {\n        readStream.emit('error', err)\n      }\n    }\n    return readStream\n  }\n\n  async listObjectsV2Query(\n    bucketName: string,\n    prefix: string,\n    continuationToken: string,\n    delimiter: string,\n    maxKeys: number,\n    startAfter: string,\n  ): Promise<ListObjectV2Res> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isString(prefix)) {\n      throw new TypeError('prefix should be of type \"string\"')\n    }\n    if (!isString(continuationToken)) {\n      throw new TypeError('continuationToken should be of type \"string\"')\n    }\n    if (!isString(delimiter)) {\n      throw new TypeError('delimiter should be of type \"string\"')\n    }\n    if (!isNumber(maxKeys)) {\n      throw new TypeError('maxKeys should be of type \"number\"')\n    }\n    if (!isString(startAfter)) {\n      throw new TypeError('startAfter should be of type \"string\"')\n    }\n\n    const queries = []\n    queries.push(`list-type=2`)\n    queries.push(`encoding-type=url`)\n    queries.push(`prefix=${uriEscape(prefix)}`)\n    queries.push(`delimiter=${uriEscape(delimiter)}`)\n\n    if (continuationToken) {\n      queries.push(`continuation-token=${uriEscape(continuationToken)}`)\n    }\n    if (startAfter) {\n      queries.push(`start-after=${uriEscape(startAfter)}`)\n    }\n    if (maxKeys) {\n      if (maxKeys >= 1000) {\n        maxKeys = 1000\n      }\n      queries.push(`max-keys=${maxKeys}`)\n    }\n    queries.sort()\n    let query = ''\n    if (queries.length > 0) {\n      query = `${queries.join('&')}`\n    }\n\n    const method = 'GET'\n    const res = await this.makeRequestAsync({ method, bucketName, query })\n    const body = await readAsString(res)\n    return parseListObjectsV2(body)\n  }\n\n  listObjectsV2(\n    bucketName: string,\n    prefix?: string,\n    recursive?: boolean,\n    startAfter?: string,\n  ): BucketStream<BucketItem> {\n    if (prefix === undefined) {\n      prefix = ''\n    }\n    if (recursive === undefined) {\n      recursive = false\n    }\n    if (startAfter === undefined) {\n      startAfter = ''\n    }\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isValidPrefix(prefix)) {\n      throw new errors.InvalidPrefixError(`Invalid prefix : ${prefix}`)\n    }\n    if (!isString(prefix)) {\n      throw new TypeError('prefix should be of type \"string\"')\n    }\n    if (!isBoolean(recursive)) {\n      throw new TypeError('recursive should be of type \"boolean\"')\n    }\n    if (!isString(startAfter)) {\n      throw new TypeError('startAfter should be of type \"string\"')\n    }\n\n    const delimiter = recursive ? '' : '/'\n    const prefixStr = prefix\n    const startAfterStr = startAfter\n    let continuationToken = ''\n    let objects: BucketItem[] = []\n    let ended = false\n    const readStream: stream.Readable = new stream.Readable({ objectMode: true })\n    readStream._read = async () => {\n      if (objects.length) {\n        readStream.push(objects.shift())\n        return\n      }\n      if (ended) {\n        return readStream.push(null)\n      }\n\n      try {\n        const result = await this.listObjectsV2Query(\n          bucketName,\n          prefixStr,\n          continuationToken,\n          delimiter,\n          1000,\n          startAfterStr,\n        )\n        if (result.isTruncated) {\n          continuationToken = result.nextContinuationToken\n        } else {\n          ended = true\n        }\n        objects = result.objects\n        // @ts-ignore\n        readStream._read()\n      } catch (err) {\n        readStream.emit('error', err)\n      }\n    }\n    return readStream\n  }\n\n  async setBucketNotification(bucketName: string, config: NotificationConfig): Promise<void> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    if (!isObject(config)) {\n      throw new TypeError('notification config should be of type \"Object\"')\n    }\n    const method = 'PUT'\n    const query = 'notification'\n    const builder = new xml2js.Builder({\n      rootName: 'NotificationConfiguration',\n      renderOpts: { pretty: false },\n      headless: true,\n    })\n    const payload = builder.buildObject(config)\n    await this.makeRequestAsyncOmit({ method, bucketName, query }, payload)\n  }\n\n  async removeAllBucketNotification(bucketName: string): Promise<void> {\n    await this.setBucketNotification(bucketName, new NotificationConfig())\n  }\n\n  async getBucketNotification(bucketName: string): Promise<NotificationConfigResult> {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError('Invalid bucket name: ' + bucketName)\n    }\n    const method = 'GET'\n    const query = 'notification'\n    const res = await this.makeRequestAsync({ method, bucketName, query })\n    const body = await readAsString(res)\n    return parseBucketNotification(body)\n  }\n\n  listenBucketNotification(\n    bucketName: string,\n    prefix: string,\n    suffix: string,\n    events: NotificationEvent[],\n  ): NotificationPoller {\n    if (!isValidBucketName(bucketName)) {\n      throw new errors.InvalidBucketNameError(`Invalid bucket name: ${bucketName}`)\n    }\n    if (!isString(prefix)) {\n      throw new TypeError('prefix must be of type string')\n    }\n    if (!isString(suffix)) {\n      throw new TypeError('suffix must be of type string')\n    }\n    if (!Array.isArray(events)) {\n      throw new TypeError('events must be of type Array')\n    }\n    const listener = new NotificationPoller(this, bucketName, prefix, suffix, events)\n    listener.start()\n    return listener\n  }\n}\n"],"mappings":"AAAA,OAAO,KAAKA,MAAM;AAClB,OAAO,KAAKC,EAAE;AAEd,OAAO,KAAKC,IAAI;AAChB,OAAO,KAAKC,KAAK;AACjB,OAAO,KAAKC,IAAI;AAChB,OAAO,KAAKC,MAAM;AAElB,OAAO,KAAKC,KAAK,MAAM,OAAO;AAC9B,OAAOC,YAAY,MAAM,eAAe;AACxC,SAASC,SAAS,QAAQ,iBAAiB;AAC3C,OAAOC,CAAC,MAAM,QAAQ;AACtB,OAAO,KAAKC,EAAE,MAAM,cAAc;AAClC,OAAOC,MAAM,MAAM,QAAQ;AAE3B,SAASC,kBAAkB,QAAQ,2BAA0B;AAC7D,OAAO,KAAKC,MAAM,MAAM,eAAc;AAEtC,SACEC,sBAAsB,EACtBC,iBAAiB,EACjBC,cAAc,EACdC,iBAAiB,EACjBC,uBAAuB,EACvBC,eAAe,EACfC,wBAAwB,QACnB,gBAAe;AAEtB,SAASC,kBAAkB,EAAEC,kBAAkB,QAAQ,qBAAoB;AAC3E,SAASC,sBAAsB,EAAEC,kBAAkB,EAAEC,MAAM,QAAQ,gBAAe;AAClF,SAASC,GAAG,EAAEC,aAAa,QAAQ,aAAY;AAC/C,SAASC,cAAc,QAAQ,uBAAsB;AACrD,SAASC,UAAU,QAAQ,kBAAiB;AAC5C,SACEC,mBAAmB,EACnBC,eAAe,EACfC,gBAAgB,EAChBC,QAAQ,EACRC,kBAAkB,EAClBC,YAAY,EACZC,UAAU,EACVC,iBAAiB,EACjBC,gBAAgB,EAChBC,SAAS,EACTC,SAAS,EACTC,OAAO,EACPC,QAAQ,EACRC,QAAQ,EACRC,aAAa,EACbC,gBAAgB,EAChBC,QAAQ,EACRC,iBAAiB,EACjBC,eAAe,EACfC,iBAAiB,EACjBC,WAAW,EACXC,aAAa,EACbC,kBAAkB,EAClBC,YAAY,EACZC,gBAAgB,EAChBC,aAAa,EACbC,eAAe,EACfC,cAAc,EACdC,YAAY,EACZC,KAAK,EACLC,QAAQ,EACRC,SAAS,EACTC,iBAAiB,QACZ,cAAa;AACpB,SAASC,YAAY,QAAQ,sBAAqB;AAClD,SAASC,UAAU,QAAQ,mBAAkB;AAC7C,SAASC,gBAAgB,QAAQ,eAAc;AAC/C,SAASC,aAAa,EAAEC,YAAY,EAAEC,YAAY,QAAQ,gBAAe;AAEzE,SAASC,aAAa,QAAQ,oBAAmB;AAqDjD,SACEC,uBAAuB,EACvBC,sBAAsB,EACtBC,sBAAsB,EACtBC,gBAAgB,EAChBC,kBAAkB,EAClBC,0BAA0B,EAC1BC,gCAAgC,EAChCC,gBAAgB,QACX,kBAAiB;AACxB,OAAO,KAAKC,UAAU,MAAM,kBAAiB;AAE7C,MAAMC,GAAG,GAAG,IAAIpE,MAAM,CAACqE,OAAO,CAAC;EAAEC,UAAU,EAAE;IAAEC,MAAM,EAAE;EAAM,CAAC;EAAEC,QAAQ,EAAE;AAAK,CAAC,CAAC;;AAEjF;AACA,MAAMC,OAAO,GAAG;EAAEC,OAAO,EA7IzB,OAAO,IA6I4D;AAAc,CAAC;AAElF,MAAMC,uBAAuB,GAAG,CAC9B,OAAO,EACP,IAAI,EACJ,MAAM,EACN,SAAS,EACT,kBAAkB,EAClB,KAAK,EACL,SAAS,EACT,WAAW,EACX,QAAQ,EACR,kBAAkB,EAClB,KAAK,EACL,YAAY,EACZ,KAAK,EACL,oBAAoB,EACpB,eAAe,EACf,gBAAgB,EAChB,YAAY,EACZ,kBAAkB,CACV;AAmEV,OAAO,MAAMC,WAAW,CAAC;EAcvBC,QAAQ,GAAW,EAAE,GAAG,IAAI,GAAG,IAAI;EAIzBC,eAAe,GAAG,CAAC,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI;EACxCC,aAAa,GAAG,CAAC,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI;EAQvDC,WAAWA,CAACC,MAAqB,EAAE;IACjC;IACA,IAAIA,MAAM,CAACC,MAAM,KAAKC,SAAS,EAAE;MAC/B,MAAM,IAAIC,KAAK,CAAC,6DAA6D,CAAC;IAChF;IACA;IACA,IAAIH,MAAM,CAACI,MAAM,KAAKF,SAAS,EAAE;MAC/BF,MAAM,CAACI,MAAM,GAAG,IAAI;IACtB;IACA,IAAI,CAACJ,MAAM,CAACK,IAAI,EAAE;MAChBL,MAAM,CAACK,IAAI,GAAG,CAAC;IACjB;IACA;IACA,IAAI,CAACjD,eAAe,CAAC4C,MAAM,CAACM,QAAQ,CAAC,EAAE;MACrC,MAAM,IAAIrF,MAAM,CAACsF,oBAAoB,CAAE,sBAAqBP,MAAM,CAACM,QAAS,EAAC,CAAC;IAChF;IACA,IAAI,CAAChD,WAAW,CAAC0C,MAAM,CAACK,IAAI,CAAC,EAAE;MAC7B,MAAM,IAAIpF,MAAM,CAACuF,oBAAoB,CAAE,kBAAiBR,MAAM,CAACK,IAAK,EAAC,CAAC;IACxE;IACA,IAAI,CAAC1D,SAAS,CAACqD,MAAM,CAACI,MAAM,CAAC,EAAE;MAC7B,MAAM,IAAInF,MAAM,CAACuF,oBAAoB,CAClC,8BAA6BR,MAAM,CAACI,MAAO,oCAC9C,CAAC;IACH;;IAEA;IACA,IAAIJ,MAAM,CAACS,MAAM,EAAE;MACjB,IAAI,CAACvD,QAAQ,CAAC8C,MAAM,CAACS,MAAM,CAAC,EAAE;QAC5B,MAAM,IAAIxF,MAAM,CAACuF,oBAAoB,CAAE,oBAAmBR,MAAM,CAACS,MAAO,EAAC,CAAC;MAC5E;IACF;IAEA,MAAMC,IAAI,GAAGV,MAAM,CAACM,QAAQ,CAACK,WAAW,CAAC,CAAC;IAC1C,IAAIN,IAAI,GAAGL,MAAM,CAACK,IAAI;IACtB,IAAIO,QAAgB;IACpB,IAAIC,SAAS;IACb,IAAIC,cAA0B;IAC9B;IACA;IACA,IAAId,MAAM,CAACI,MAAM,EAAE;MACjB;MACAS,SAAS,GAAGtG,KAAK;MACjBqG,QAAQ,GAAG,QAAQ;MACnBP,IAAI,GAAGA,IAAI,IAAI,GAAG;MAClBS,cAAc,GAAGvG,KAAK,CAACwG,WAAW;IACpC,CAAC,MAAM;MACLF,SAAS,GAAGvG,IAAI;MAChBsG,QAAQ,GAAG,OAAO;MAClBP,IAAI,GAAGA,IAAI,IAAI,EAAE;MACjBS,cAAc,GAAGxG,IAAI,CAACyG,WAAW;IACnC;;IAEA;IACA,IAAIf,MAAM,CAACa,SAAS,EAAE;MACpB,IAAI,CAAC9D,QAAQ,CAACiD,MAAM,CAACa,SAAS,CAAC,EAAE;QAC/B,MAAM,IAAI5F,MAAM,CAACuF,oBAAoB,CAClC,4BAA2BR,MAAM,CAACa,SAAU,gCAC/C,CAAC;MACH;MACAA,SAAS,GAAGb,MAAM,CAACa,SAAS;IAC9B;;IAEA;IACA,IAAIb,MAAM,CAACc,cAAc,EAAE;MACzB,IAAI,CAAC/D,QAAQ,CAACiD,MAAM,CAACc,cAAc,CAAC,EAAE;QACpC,MAAM,IAAI7F,MAAM,CAACuF,oBAAoB,CAClC,gCAA+BR,MAAM,CAACc,cAAe,gCACxD,CAAC;MACH;MAEAA,cAAc,GAAGd,MAAM,CAACc,cAAc;IACxC;;IAEA;IACA;IACA;IACA;IACA;IACA,MAAME,eAAe,GAAI,IAAGC,OAAO,CAACC,QAAS,KAAID,OAAO,CAACE,IAAK,GAAE;IAChE,MAAMC,YAAY,GAAI,SAAQJ,eAAgB,aAAYxB,OAAO,CAACC,OAAQ,EAAC;IAC3E;;IAEA,IAAI,CAACoB,SAAS,GAAGA,SAAS;IAC1B,IAAI,CAACC,cAAc,GAAGA,cAAc;IACpC,IAAI,CAACJ,IAAI,GAAGA,IAAI;IAChB,IAAI,CAACL,IAAI,GAAGA,IAAI;IAChB,IAAI,CAACO,QAAQ,GAAGA,QAAQ;IACxB,IAAI,CAACS,SAAS,GAAI,GAAED,YAAa,EAAC;;IAElC;IACA,IAAIpB,MAAM,CAACsB,SAAS,KAAKpB,SAAS,EAAE;MAClC,IAAI,CAACoB,SAAS,GAAG,IAAI;IACvB,CAAC,MAAM;MACL,IAAI,CAACA,SAAS,GAAGtB,MAAM,CAACsB,SAAS;IACnC;IAEA,IAAI,CAACC,SAAS,GAAGvB,MAAM,CAACuB,SAAS,IAAI,EAAE;IACvC,IAAI,CAACC,SAAS,GAAGxB,MAAM,CAACwB,SAAS,IAAI,EAAE;IACvC,IAAI,CAACC,YAAY,GAAGzB,MAAM,CAACyB,YAAY;IACvC,IAAI,CAACC,SAAS,GAAG,CAAC,IAAI,CAACH,SAAS,IAAI,CAAC,IAAI,CAACC,SAAS;IAEnD,IAAIxB,MAAM,CAAC2B,mBAAmB,EAAE;MAC9B,IAAI,CAACD,SAAS,GAAG,KAAK;MACtB,IAAI,CAACC,mBAAmB,GAAG3B,MAAM,CAAC2B,mBAAmB;IACvD;IAEA,IAAI,CAACC,SAAS,GAAG,CAAC,CAAC;IACnB,IAAI5B,MAAM,CAACS,MAAM,EAAE;MACjB,IAAI,CAACA,MAAM,GAAGT,MAAM,CAACS,MAAM;IAC7B;IAEA,IAAIT,MAAM,CAACJ,QAAQ,EAAE;MACnB,IAAI,CAACA,QAAQ,GAAGI,MAAM,CAACJ,QAAQ;MAC/B,IAAI,CAACiC,gBAAgB,GAAG,IAAI;IAC9B;IACA,IAAI,IAAI,CAACjC,QAAQ,GAAG,CAAC,GAAG,IAAI,GAAG,IAAI,EAAE;MACnC,MAAM,IAAI3E,MAAM,CAACuF,oBAAoB,CAAE,sCAAqC,CAAC;IAC/E;IACA,IAAI,IAAI,CAACZ,QAAQ,GAAG,CAAC,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI,EAAE;MAC1C,MAAM,IAAI3E,MAAM,CAACuF,oBAAoB,CAAE,mCAAkC,CAAC;IAC5E;;IAEA;IACA;IACA;IACA,IAAI,CAACsB,YAAY,GAAG,CAAC,IAAI,CAACJ,SAAS,IAAI,CAAC1B,MAAM,CAACI,MAAM;IAErD,IAAI,CAAC2B,oBAAoB,GAAG/B,MAAM,CAAC+B,oBAAoB,IAAI7B,SAAS;IACpE,IAAI,CAAC8B,UAAU,GAAG,CAAC,CAAC;IACpB,IAAI,CAACC,gBAAgB,GAAG,IAAIhG,UAAU,CAAC,IAAI,CAAC;IAE5C,IAAI+D,MAAM,CAACkC,YAAY,EAAE;MACvB,IAAI,CAACnF,QAAQ,CAACiD,MAAM,CAACkC,YAAY,CAAC,EAAE;QAClC,MAAM,IAAIjH,MAAM,CAACuF,oBAAoB,CAClC,8BAA6BR,MAAM,CAACkC,YAAa,gCACpD,CAAC;MACH;MAEA,IAAI,CAACA,YAAY,GAAGlC,MAAM,CAACkC,YAAY;IACzC,CAAC,MAAM;MACL,IAAI,CAACA,YAAY,GAAG;QAClBC,YAAY,EAAE;MAChB,CAAC;IACH;EACF;EACA;AACF;AACA;EACE,IAAIC,UAAUA,CAAA,EAAG;IACf,OAAO,IAAI,CAACH,gBAAgB;EAC9B;;EAEA;AACF;AACA;EACEI,uBAAuBA,CAAC/B,QAAgB,EAAE;IACxC,IAAI,CAACyB,oBAAoB,GAAGzB,QAAQ;EACtC;;EAEA;AACF;AACA;EACSgC,iBAAiBA,CAACC,OAA6E,EAAE;IACtG,IAAI,CAACxF,QAAQ,CAACwF,OAAO,CAAC,EAAE;MACtB,MAAM,IAAIC,SAAS,CAAC,4CAA4C,CAAC;IACnE;IACA,IAAI,CAACR,UAAU,GAAGnH,CAAC,CAAC4H,IAAI,CAACF,OAAO,EAAE7C,uBAAuB,CAAC;EAC5D;;EAEA;AACF;AACA;EACUgD,0BAA0BA,CAACC,UAAmB,EAAEC,UAAmB,EAAE;IAC3E,IAAI,CAAC/F,OAAO,CAAC,IAAI,CAACkF,oBAAoB,CAAC,IAAI,CAAClF,OAAO,CAAC8F,UAAU,CAAC,IAAI,CAAC9F,OAAO,CAAC+F,UAAU,CAAC,EAAE;MACvF;MACA;MACA,IAAID,UAAU,CAACE,QAAQ,CAAC,GAAG,CAAC,EAAE;QAC5B,MAAM,IAAI1C,KAAK,CAAE,mEAAkEwC,UAAW,EAAC,CAAC;MAClG;MACA;MACA;MACA;MACA,OAAO,IAAI,CAACZ,oBAAoB;IAClC;IACA,OAAO,KAAK;EACd;;EAEA;AACF;AACA;AACA;AACA;EACEe,UAAUA,CAACC,OAAe,EAAEC,UAAkB,EAAE;IAC9C,IAAI,CAAC9F,QAAQ,CAAC6F,OAAO,CAAC,EAAE;MACtB,MAAM,IAAIP,SAAS,CAAE,oBAAmBO,OAAQ,EAAC,CAAC;IACpD;IACA,IAAIA,OAAO,CAACE,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE;MACzB,MAAM,IAAIhI,MAAM,CAACuF,oBAAoB,CAAC,gCAAgC,CAAC;IACzE;IACA,IAAI,CAACtD,QAAQ,CAAC8F,UAAU,CAAC,EAAE;MACzB,MAAM,IAAIR,SAAS,CAAE,uBAAsBQ,UAAW,EAAC,CAAC;IAC1D;IACA,IAAIA,UAAU,CAACC,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE;MAC5B,MAAM,IAAIhI,MAAM,CAACuF,oBAAoB,CAAC,mCAAmC,CAAC;IAC5E;IACA,IAAI,CAACa,SAAS,GAAI,GAAE,IAAI,CAACA,SAAU,IAAG0B,OAAQ,IAAGC,UAAW,EAAC;EAC/D;;EAEA;AACF;AACA;AACA;EACYE,iBAAiBA,CACzBC,IAEC,EAID;IACA,MAAMC,MAAM,GAAGD,IAAI,CAACC,MAAM;IAC1B,MAAM3C,MAAM,GAAG0C,IAAI,CAAC1C,MAAM;IAC1B,MAAMkC,UAAU,GAAGQ,IAAI,CAACR,UAAU;IAClC,IAAIC,UAAU,GAAGO,IAAI,CAACP,UAAU;IAChC,MAAMS,OAAO,GAAGF,IAAI,CAACE,OAAO;IAC5B,MAAMC,KAAK,GAAGH,IAAI,CAACG,KAAK;IAExB,IAAItB,UAAU,GAAG;MACfoB,MAAM;MACNC,OAAO,EAAE,CAAC,CAAmB;MAC7BzC,QAAQ,EAAE,IAAI,CAACA,QAAQ;MACvB;MACA2C,KAAK,EAAE,IAAI,CAACzC;IACd,CAAC;;IAED;IACA,IAAI0C,gBAAgB;IACpB,IAAIb,UAAU,EAAE;MACda,gBAAgB,GAAGhG,kBAAkB,CAAC,IAAI,CAACkD,IAAI,EAAE,IAAI,CAACE,QAAQ,EAAE+B,UAAU,EAAE,IAAI,CAACrB,SAAS,CAAC;IAC7F;IAEA,IAAI9G,IAAI,GAAG,GAAG;IACd,IAAIkG,IAAI,GAAG,IAAI,CAACA,IAAI;IAEpB,IAAIL,IAAwB;IAC5B,IAAI,IAAI,CAACA,IAAI,EAAE;MACbA,IAAI,GAAG,IAAI,CAACA,IAAI;IAClB;IAEA,IAAIuC,UAAU,EAAE;MACdA,UAAU,GAAG1E,iBAAiB,CAAC0E,UAAU,CAAC;IAC5C;;IAEA;IACA,IAAIlG,gBAAgB,CAACgE,IAAI,CAAC,EAAE;MAC1B,MAAM+C,kBAAkB,GAAG,IAAI,CAACf,0BAA0B,CAACC,UAAU,EAAEC,UAAU,CAAC;MAClF,IAAIa,kBAAkB,EAAE;QACtB/C,IAAI,GAAI,GAAE+C,kBAAmB,EAAC;MAChC,CAAC,MAAM;QACL/C,IAAI,GAAGjC,aAAa,CAACgC,MAAM,CAAC;MAC9B;IACF;IAEA,IAAI+C,gBAAgB,IAAI,CAACL,IAAI,CAAC7B,SAAS,EAAE;MACvC;MACA;MACA;MACA;MACA;MACA,IAAIqB,UAAU,EAAE;QACdjC,IAAI,GAAI,GAAEiC,UAAW,IAAGjC,IAAK,EAAC;MAChC;MACA,IAAIkC,UAAU,EAAE;QACdpI,IAAI,GAAI,IAAGoI,UAAW,EAAC;MACzB;IACF,CAAC,MAAM;MACL;MACA;MACA;MACA,IAAID,UAAU,EAAE;QACdnI,IAAI,GAAI,IAAGmI,UAAW,EAAC;MACzB;MACA,IAAIC,UAAU,EAAE;QACdpI,IAAI,GAAI,IAAGmI,UAAW,IAAGC,UAAW,EAAC;MACvC;IACF;IAEA,IAAIU,KAAK,EAAE;MACT9I,IAAI,IAAK,IAAG8I,KAAM,EAAC;IACrB;IACAtB,UAAU,CAACqB,OAAO,CAAC3C,IAAI,GAAGA,IAAI;IAC9B,IAAKsB,UAAU,CAACpB,QAAQ,KAAK,OAAO,IAAIP,IAAI,KAAK,EAAE,IAAM2B,UAAU,CAACpB,QAAQ,KAAK,QAAQ,IAAIP,IAAI,KAAK,GAAI,EAAE;MAC1G2B,UAAU,CAACqB,OAAO,CAAC3C,IAAI,GAAGvC,YAAY,CAACuC,IAAI,EAAEL,IAAI,CAAC;IACpD;IAEA2B,UAAU,CAACqB,OAAO,CAAC,YAAY,CAAC,GAAG,IAAI,CAAChC,SAAS;IACjD,IAAIgC,OAAO,EAAE;MACX;MACA,KAAK,MAAM,CAACK,CAAC,EAAEC,CAAC,CAAC,IAAIC,MAAM,CAACC,OAAO,CAACR,OAAO,CAAC,EAAE;QAC5CrB,UAAU,CAACqB,OAAO,CAACK,CAAC,CAAC/C,WAAW,CAAC,CAAC,CAAC,GAAGgD,CAAC;MACzC;IACF;;IAEA;IACA3B,UAAU,GAAG4B,MAAM,CAACE,MAAM,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC9B,UAAU,EAAEA,UAAU,CAAC;IAE3D,OAAO;MACL,GAAGA,UAAU;MACbqB,OAAO,EAAExI,CAAC,CAACkJ,SAAS,CAAClJ,CAAC,CAACmJ,MAAM,CAAChC,UAAU,CAACqB,OAAO,EAAEzG,SAAS,CAAC,EAAG+G,CAAC,IAAKA,CAAC,CAACM,QAAQ,CAAC,CAAC,CAAC;MAClFvD,IAAI;MACJL,IAAI;MACJ7F;IACF,CAAC;EACH;EAEA,MAAa0J,sBAAsBA,CAACvC,mBAAuC,EAAE;IAC3E,IAAI,EAAEA,mBAAmB,YAAY3G,kBAAkB,CAAC,EAAE;MACxD,MAAM,IAAImF,KAAK,CAAC,oEAAoE,CAAC;IACvF;IACA,IAAI,CAACwB,mBAAmB,GAAGA,mBAAmB;IAC9C,MAAM,IAAI,CAACwC,oBAAoB,CAAC,CAAC;EACnC;EAEA,MAAcA,oBAAoBA,CAAA,EAAG;IACnC,IAAI,IAAI,CAACxC,mBAAmB,EAAE;MAC5B,IAAI;QACF,MAAMyC,eAAe,GAAG,MAAM,IAAI,CAACzC,mBAAmB,CAAC0C,cAAc,CAAC,CAAC;QACvE,IAAI,CAAC9C,SAAS,GAAG6C,eAAe,CAACE,YAAY,CAAC,CAAC;QAC/C,IAAI,CAAC9C,SAAS,GAAG4C,eAAe,CAACG,YAAY,CAAC,CAAC;QAC/C,IAAI,CAAC9C,YAAY,GAAG2C,eAAe,CAACI,eAAe,CAAC,CAAC;MACvD,CAAC,CAAC,OAAOC,CAAC,EAAE;QACV,MAAM,IAAItE,KAAK,CAAE,8BAA6BsE,CAAE,EAAC,EAAE;UAAEC,KAAK,EAAED;QAAE,CAAC,CAAC;MAClE;IACF;EACF;EAIA;AACF;AACA;EACUE,OAAOA,CAAC3C,UAAoB,EAAE4C,QAAqC,EAAEC,GAAa,EAAE;IAC1F;IACA,IAAI,CAAC,IAAI,CAACC,SAAS,EAAE;MACnB;IACF;IACA,IAAI,CAAC/H,QAAQ,CAACiF,UAAU,CAAC,EAAE;MACzB,MAAM,IAAIQ,SAAS,CAAC,uCAAuC,CAAC;IAC9D;IACA,IAAIoC,QAAQ,IAAI,CAAC3H,gBAAgB,CAAC2H,QAAQ,CAAC,EAAE;MAC3C,MAAM,IAAIpC,SAAS,CAAC,qCAAqC,CAAC;IAC5D;IACA,IAAIqC,GAAG,IAAI,EAAEA,GAAG,YAAY1E,KAAK,CAAC,EAAE;MAClC,MAAM,IAAIqC,SAAS,CAAC,+BAA+B,CAAC;IACtD;IACA,MAAMsC,SAAS,GAAG,IAAI,CAACA,SAAS;IAChC,MAAMC,UAAU,GAAI1B,OAAuB,IAAK;MAC9CO,MAAM,CAACC,OAAO,CAACR,OAAO,CAAC,CAAC2B,OAAO,CAAC,CAAC,CAACtB,CAAC,EAAEC,CAAC,CAAC,KAAK;QAC1C,IAAID,CAAC,IAAI,eAAe,EAAE;UACxB,IAAIxG,QAAQ,CAACyG,CAAC,CAAC,EAAE;YACf,MAAMsB,QAAQ,GAAG,IAAIC,MAAM,CAAC,uBAAuB,CAAC;YACpDvB,CAAC,GAAGA,CAAC,CAACwB,OAAO,CAACF,QAAQ,EAAE,wBAAwB,CAAC;UACnD;QACF;QACAH,SAAS,CAACM,KAAK,CAAE,GAAE1B,CAAE,KAAIC,CAAE,IAAG,CAAC;MACjC,CAAC,CAAC;MACFmB,SAAS,CAACM,KAAK,CAAC,IAAI,CAAC;IACvB,CAAC;IACDN,SAAS,CAACM,KAAK,CAAE,YAAWpD,UAAU,CAACoB,MAAO,IAAGpB,UAAU,CAACxH,IAAK,IAAG,CAAC;IACrEuK,UAAU,CAAC/C,UAAU,CAACqB,OAAO,CAAC;IAC9B,IAAIuB,QAAQ,EAAE;MACZ,IAAI,CAACE,SAAS,CAACM,KAAK,CAAE,aAAYR,QAAQ,CAACS,UAAW,IAAG,CAAC;MAC1DN,UAAU,CAACH,QAAQ,CAACvB,OAAyB,CAAC;IAChD;IACA,IAAIwB,GAAG,EAAE;MACPC,SAAS,CAACM,KAAK,CAAC,eAAe,CAAC;MAChC,MAAME,OAAO,GAAGC,IAAI,CAACC,SAAS,CAACX,GAAG,EAAE,IAAI,EAAE,IAAI,CAAC;MAC/CC,SAAS,CAACM,KAAK,CAAE,GAAEE,OAAQ,IAAG,CAAC;IACjC;EACF;;EAEA;AACF;AACA;EACSG,OAAOA,CAAChL,MAAwB,EAAE;IACvC,IAAI,CAACA,MAAM,EAAE;MACXA,MAAM,GAAGwG,OAAO,CAACyE,MAAM;IACzB;IACA,IAAI,CAACZ,SAAS,GAAGrK,MAAM;EACzB;;EAEA;AACF;AACA;EACSkL,QAAQA,CAAA,EAAG;IAChB,IAAI,CAACb,SAAS,GAAG5E,SAAS;EAC5B;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,MAAM0F,gBAAgBA,CACpBrD,OAAsB,EACtBsD,OAAe,GAAG,EAAE,EACpBC,aAAuB,GAAG,CAAC,GAAG,CAAC,EAC/BrF,MAAM,GAAG,EAAE,EACoB;IAC/B,IAAI,CAAC1D,QAAQ,CAACwF,OAAO,CAAC,EAAE;MACtB,MAAM,IAAIC,SAAS,CAAC,oCAAoC,CAAC;IAC3D;IACA,IAAI,CAACtF,QAAQ,CAAC2I,OAAO,CAAC,IAAI,CAAC9I,QAAQ,CAAC8I,OAAO,CAAC,EAAE;MAC5C;MACA,MAAM,IAAIrD,SAAS,CAAC,gDAAgD,CAAC;IACvE;IACAsD,aAAa,CAACd,OAAO,CAAEK,UAAU,IAAK;MACpC,IAAI,CAACvI,QAAQ,CAACuI,UAAU,CAAC,EAAE;QACzB,MAAM,IAAI7C,SAAS,CAAC,uCAAuC,CAAC;MAC9D;IACF,CAAC,CAAC;IACF,IAAI,CAACtF,QAAQ,CAACuD,MAAM,CAAC,EAAE;MACrB,MAAM,IAAI+B,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA,IAAI,CAACD,OAAO,CAACc,OAAO,EAAE;MACpBd,OAAO,CAACc,OAAO,GAAG,CAAC,CAAC;IACtB;IACA,IAAId,OAAO,CAACa,MAAM,KAAK,MAAM,IAAIb,OAAO,CAACa,MAAM,KAAK,KAAK,IAAIb,OAAO,CAACa,MAAM,KAAK,QAAQ,EAAE;MACxFb,OAAO,CAACc,OAAO,CAAC,gBAAgB,CAAC,GAAGwC,OAAO,CAACE,MAAM,CAAC9B,QAAQ,CAAC,CAAC;IAC/D;IACA,MAAM+B,SAAS,GAAG,IAAI,CAAClE,YAAY,GAAG9D,QAAQ,CAAC6H,OAAO,CAAC,GAAG,EAAE;IAC5D,OAAO,IAAI,CAACI,sBAAsB,CAAC1D,OAAO,EAAEsD,OAAO,EAAEG,SAAS,EAAEF,aAAa,EAAErF,MAAM,CAAC;EACxF;;EAEA;AACF;AACA;AACA;AACA;EACE,MAAMyF,oBAAoBA,CACxB3D,OAAsB,EACtBsD,OAAe,GAAG,EAAE,EACpBM,WAAqB,GAAG,CAAC,GAAG,CAAC,EAC7B1F,MAAM,GAAG,EAAE,EACgC;IAC3C,MAAM2F,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAACrD,OAAO,EAAEsD,OAAO,EAAEM,WAAW,EAAE1F,MAAM,CAAC;IAC9E,MAAMnC,aAAa,CAAC8H,GAAG,CAAC;IACxB,OAAOA,GAAG;EACZ;;EAEA;AACF;AACA;AACA;AACA;AACA;EACE,MAAMH,sBAAsBA,CAC1B1D,OAAsB,EACtB8D,IAA8B,EAC9BL,SAAiB,EACjBG,WAAqB,EACrB1F,MAAc,EACiB;IAC/B,IAAI,CAAC1D,QAAQ,CAACwF,OAAO,CAAC,EAAE;MACtB,MAAM,IAAIC,SAAS,CAAC,oCAAoC,CAAC;IAC3D;IACA,IAAI,EAAE8D,MAAM,CAACC,QAAQ,CAACF,IAAI,CAAC,IAAI,OAAOA,IAAI,KAAK,QAAQ,IAAIpJ,gBAAgB,CAACoJ,IAAI,CAAC,CAAC,EAAE;MAClF,MAAM,IAAIpL,MAAM,CAACuF,oBAAoB,CAClC,6DAA4D,OAAO6F,IAAK,UAC3E,CAAC;IACH;IACA,IAAI,CAACnJ,QAAQ,CAAC8I,SAAS,CAAC,EAAE;MACxB,MAAM,IAAIxD,SAAS,CAAC,sCAAsC,CAAC;IAC7D;IACA2D,WAAW,CAACnB,OAAO,CAAEK,UAAU,IAAK;MAClC,IAAI,CAACvI,QAAQ,CAACuI,UAAU,CAAC,EAAE;QACzB,MAAM,IAAI7C,SAAS,CAAC,uCAAuC,CAAC;MAC9D;IACF,CAAC,CAAC;IACF,IAAI,CAACtF,QAAQ,CAACuD,MAAM,CAAC,EAAE;MACrB,MAAM,IAAI+B,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA;IACA,IAAI,CAAC,IAAI,CAACV,YAAY,IAAIkE,SAAS,CAACD,MAAM,KAAK,CAAC,EAAE;MAChD,MAAM,IAAI9K,MAAM,CAACuF,oBAAoB,CAAE,gEAA+D,CAAC;IACzG;IACA;IACA,IAAI,IAAI,CAACsB,YAAY,IAAIkE,SAAS,CAACD,MAAM,KAAK,EAAE,EAAE;MAChD,MAAM,IAAI9K,MAAM,CAACuF,oBAAoB,CAAE,uBAAsBwF,SAAU,EAAC,CAAC;IAC3E;IAEA,MAAM,IAAI,CAAC7B,oBAAoB,CAAC,CAAC;;IAEjC;IACA1D,MAAM,GAAGA,MAAM,KAAK,MAAM,IAAI,CAAC+F,oBAAoB,CAACjE,OAAO,CAACI,UAAW,CAAC,CAAC;IAEzE,MAAMX,UAAU,GAAG,IAAI,CAACkB,iBAAiB,CAAC;MAAE,GAAGX,OAAO;MAAE9B;IAAO,CAAC,CAAC;IACjE,IAAI,CAAC,IAAI,CAACiB,SAAS,EAAE;MACnB;MACA,IAAI,CAAC,IAAI,CAACI,YAAY,EAAE;QACtBkE,SAAS,GAAG,kBAAkB;MAChC;MACA,MAAMS,IAAI,GAAG,IAAIC,IAAI,CAAC,CAAC;MACvB1E,UAAU,CAACqB,OAAO,CAAC,YAAY,CAAC,GAAG5F,YAAY,CAACgJ,IAAI,CAAC;MACrDzE,UAAU,CAACqB,OAAO,CAAC,sBAAsB,CAAC,GAAG2C,SAAS;MACtD,IAAI,IAAI,CAACvE,YAAY,EAAE;QACrBO,UAAU,CAACqB,OAAO,CAAC,sBAAsB,CAAC,GAAG,IAAI,CAAC5B,YAAY;MAChE;MACAO,UAAU,CAACqB,OAAO,CAACsD,aAAa,GAAG9K,MAAM,CAACmG,UAAU,EAAE,IAAI,CAACT,SAAS,EAAE,IAAI,CAACC,SAAS,EAAEf,MAAM,EAAEgG,IAAI,EAAET,SAAS,CAAC;IAChH;IAEA,MAAMpB,QAAQ,GAAG,MAAMvG,gBAAgB,CACrC,IAAI,CAACwC,SAAS,EACdmB,UAAU,EACVqE,IAAI,EACJ,IAAI,CAACnE,YAAY,CAACC,YAAY,KAAK,IAAI,GAAG,CAAC,GAAG,IAAI,CAACD,YAAY,CAAC0E,iBAAiB,EACjF,IAAI,CAAC1E,YAAY,CAAC2E,WAAW,EAC7B,IAAI,CAAC3E,YAAY,CAAC4E,cACpB,CAAC;IACD,IAAI,CAAClC,QAAQ,CAACS,UAAU,EAAE;MACxB,MAAM,IAAIlF,KAAK,CAAC,yCAAyC,CAAC;IAC5D;IAEA,IAAI,CAACgG,WAAW,CAACtD,QAAQ,CAAC+B,QAAQ,CAACS,UAAU,CAAC,EAAE;MAC9C;MACA;MACA;MACA;MACA;MACA,OAAO,IAAI,CAACzD,SAAS,CAACW,OAAO,CAACI,UAAU,CAAE;MAE1C,MAAMkC,GAAG,GAAG,MAAM3F,UAAU,CAAC6H,kBAAkB,CAACnC,QAAQ,CAAC;MACzD,IAAI,CAACD,OAAO,CAAC3C,UAAU,EAAE4C,QAAQ,EAAEC,GAAG,CAAC;MACvC,MAAMA,GAAG;IACX;IAEA,IAAI,CAACF,OAAO,CAAC3C,UAAU,EAAE4C,QAAQ,CAAC;IAElC,OAAOA,QAAQ;EACjB;;EAEA;AACF;AACA;AACA;AACA;AACA;EACE,MAAM4B,oBAAoBA,CAAC7D,UAAkB,EAAmB;IAC9D,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,yBAAwBrE,UAAW,EAAC,CAAC;IAChF;;IAEA;IACA,IAAI,IAAI,CAAClC,MAAM,EAAE;MACf,OAAO,IAAI,CAACA,MAAM;IACpB;IAEA,MAAMwG,MAAM,GAAG,IAAI,CAACrF,SAAS,CAACe,UAAU,CAAC;IACzC,IAAIsE,MAAM,EAAE;MACV,OAAOA,MAAM;IACf;IAEA,MAAMC,kBAAkB,GAAG,MAAOtC,QAA8B,IAAK;MACnE,MAAMyB,IAAI,GAAG,MAAM7H,YAAY,CAACoG,QAAQ,CAAC;MACzC,MAAMnE,MAAM,GAAGvB,UAAU,CAACiI,iBAAiB,CAACd,IAAI,CAAC,IAAIjL,cAAc;MACnE,IAAI,CAACwG,SAAS,CAACe,UAAU,CAAC,GAAGlC,MAAM;MACnC,OAAOA,MAAM;IACf,CAAC;IAED,MAAM2C,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,UAAU;IACxB;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA,MAAMhC,SAAS,GAAG,IAAI,CAACA,SAAS,IAAI,CAAC1G,SAAS;IAC9C,IAAI6F,MAAc;IAClB,IAAI;MACF,MAAM2F,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;QAAExC,MAAM;QAAET,UAAU;QAAEW,KAAK;QAAEhC;MAAU,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,EAAElG,cAAc,CAAC;MAC5G,OAAO8L,kBAAkB,CAACd,GAAG,CAAC;IAChC,CAAC,CAAC,OAAO3B,CAAC,EAAE;MACV;MACA,IAAIA,CAAC,YAAYxJ,MAAM,CAACmM,OAAO,EAAE;QAC/B,MAAMC,OAAO,GAAG5C,CAAC,CAAC6C,IAAI;QACtB,MAAMC,SAAS,GAAG9C,CAAC,CAAChE,MAAM;QAC1B,IAAI4G,OAAO,KAAK,cAAc,IAAI,CAACE,SAAS,EAAE;UAC5C,OAAOnM,cAAc;QACvB;MACF;MACA;MACA;MACA,IAAI,EAAEqJ,CAAC,CAAC+C,IAAI,KAAK,8BAA8B,CAAC,EAAE;QAChD,MAAM/C,CAAC;MACT;MACA;MACAhE,MAAM,GAAGgE,CAAC,CAACgD,MAAgB;MAC3B,IAAI,CAAChH,MAAM,EAAE;QACX,MAAMgE,CAAC;MACT;IACF;IAEA,MAAM2B,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW,KAAK;MAAEhC;IAAU,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,EAAEb,MAAM,CAAC;IACpG,OAAO,MAAMyG,kBAAkB,CAACd,GAAG,CAAC;EACtC;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACEsB,WAAWA,CACTnF,OAAsB,EACtBsD,OAAe,GAAG,EAAE,EACpBC,aAAuB,GAAG,CAAC,GAAG,CAAC,EAC/BrF,MAAM,GAAG,EAAE,EACXkH,cAAuB,EACvBC,EAAuD,EACvD;IACA,IAAIC,IAAmC;IACvC,IAAIF,cAAc,EAAE;MAClBE,IAAI,GAAG,IAAI,CAACjC,gBAAgB,CAACrD,OAAO,EAAEsD,OAAO,EAAEC,aAAa,EAAErF,MAAM,CAAC;IACvE,CAAC,MAAM;MACL;MACA;MACAoH,IAAI,GAAG,IAAI,CAAC3B,oBAAoB,CAAC3D,OAAO,EAAEsD,OAAO,EAAEC,aAAa,EAAErF,MAAM,CAAC;IAC3E;IAEAoH,IAAI,CAACC,IAAI,CACNC,MAAM,IAAKH,EAAE,CAAC,IAAI,EAAEG,MAAM,CAAC,EAC3BlD,GAAG,IAAK;MACP;MACA;MACA+C,EAAE,CAAC/C,GAAG,CAAC;IACT,CACF,CAAC;EACH;;EAEA;AACF;AACA;AACA;AACA;AACA;EACEmD,iBAAiBA,CACfzF,OAAsB,EACtB9H,MAAgC,EAChCuL,SAAiB,EACjBG,WAAqB,EACrB1F,MAAc,EACdkH,cAAuB,EACvBC,EAAuD,EACvD;IACA,MAAMK,QAAQ,GAAG,MAAAA,CAAA,KAAY;MAC3B,MAAM7B,GAAG,GAAG,MAAM,IAAI,CAACH,sBAAsB,CAAC1D,OAAO,EAAE9H,MAAM,EAAEuL,SAAS,EAAEG,WAAW,EAAE1F,MAAM,CAAC;MAC9F,IAAI,CAACkH,cAAc,EAAE;QACnB,MAAMrJ,aAAa,CAAC8H,GAAG,CAAC;MAC1B;MAEA,OAAOA,GAAG;IACZ,CAAC;IAED6B,QAAQ,CAAC,CAAC,CAACH,IAAI,CACZC,MAAM,IAAKH,EAAE,CAAC,IAAI,EAAEG,MAAM,CAAC;IAC5B;IACA;IACClD,GAAG,IAAK+C,EAAE,CAAC/C,GAAG,CACjB,CAAC;EACH;;EAEA;AACF;AACA;EACEqD,eAAeA,CAACvF,UAAkB,EAAEiF,EAA0C,EAAE;IAC9E,OAAO,IAAI,CAACpB,oBAAoB,CAAC7D,UAAU,CAAC,CAACmF,IAAI,CAC9CC,MAAM,IAAKH,EAAE,CAAC,IAAI,EAAEG,MAAM,CAAC;IAC5B;IACA;IACClD,GAAG,IAAK+C,EAAE,CAAC/C,GAAG,CACjB,CAAC;EACH;;EAEA;;EAEA;AACF;AACA;AACA;EACE,MAAMsD,UAAUA,CAACxF,UAAkB,EAAElC,MAAc,GAAG,EAAE,EAAE2H,QAAwB,EAAiB;IACjG,IAAI,CAACjL,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA;IACA,IAAI5F,QAAQ,CAAC0D,MAAM,CAAC,EAAE;MACpB2H,QAAQ,GAAG3H,MAAM;MACjBA,MAAM,GAAG,EAAE;IACb;IAEA,IAAI,CAACvD,QAAQ,CAACuD,MAAM,CAAC,EAAE;MACrB,MAAM,IAAI+B,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA,IAAI4F,QAAQ,IAAI,CAACrL,QAAQ,CAACqL,QAAQ,CAAC,EAAE;MACnC,MAAM,IAAI5F,SAAS,CAAC,qCAAqC,CAAC;IAC5D;IAEA,IAAIqD,OAAO,GAAG,EAAE;;IAEhB;IACA;IACA,IAAIpF,MAAM,IAAI,IAAI,CAACA,MAAM,EAAE;MACzB,IAAIA,MAAM,KAAK,IAAI,CAACA,MAAM,EAAE;QAC1B,MAAM,IAAIxF,MAAM,CAACuF,oBAAoB,CAAE,qBAAoB,IAAI,CAACC,MAAO,eAAcA,MAAO,EAAC,CAAC;MAChG;IACF;IACA;IACA;IACA,IAAIA,MAAM,IAAIA,MAAM,KAAKrF,cAAc,EAAE;MACvCyK,OAAO,GAAG1G,GAAG,CAACkJ,WAAW,CAAC;QACxBC,yBAAyB,EAAE;UACzBC,CAAC,EAAE;YAAEC,KAAK,EAAE;UAA0C,CAAC;UACvDC,kBAAkB,EAAEhI;QACtB;MACF,CAAC,CAAC;IACJ;IACA,MAAM2C,MAAM,GAAG,KAAK;IACpB,MAAMC,OAAuB,GAAG,CAAC,CAAC;IAElC,IAAI+E,QAAQ,IAAIA,QAAQ,CAACM,aAAa,EAAE;MACtCrF,OAAO,CAAC,kCAAkC,CAAC,GAAG,IAAI;IACpD;;IAEA;IACA,MAAMsF,WAAW,GAAG,IAAI,CAAClI,MAAM,IAAIA,MAAM,IAAIrF,cAAc;IAE3D,MAAMwN,UAAyB,GAAG;MAAExF,MAAM;MAAET,UAAU;MAAEU;IAAQ,CAAC;IAEjE,IAAI;MACF,MAAM,IAAI,CAAC6C,oBAAoB,CAAC0C,UAAU,EAAE/C,OAAO,EAAE,CAAC,GAAG,CAAC,EAAE8C,WAAW,CAAC;IAC1E,CAAC,CAAC,OAAO9D,GAAY,EAAE;MACrB,IAAIpE,MAAM,KAAK,EAAE,IAAIA,MAAM,KAAKrF,cAAc,EAAE;QAC9C,IAAIyJ,GAAG,YAAY5J,MAAM,CAACmM,OAAO,EAAE;UACjC,MAAMC,OAAO,GAAGxC,GAAG,CAACyC,IAAI;UACxB,MAAMC,SAAS,GAAG1C,GAAG,CAACpE,MAAM;UAC5B,IAAI4G,OAAO,KAAK,8BAA8B,IAAIE,SAAS,KAAK,EAAE,EAAE;YAClE;YACA,MAAM,IAAI,CAACrB,oBAAoB,CAAC0C,UAAU,EAAE/C,OAAO,EAAE,CAAC,GAAG,CAAC,EAAEwB,OAAO,CAAC;UACtE;QACF;MACF;MACA,MAAMxC,GAAG;IACX;EACF;;EAEA;AACF;AACA;EACE,MAAMgE,YAAYA,CAAClG,UAAkB,EAAoB;IACvD,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,MAAM;IACrB,IAAI;MACF,MAAM,IAAI,CAAC8C,oBAAoB,CAAC;QAAE9C,MAAM;QAAET;MAAW,CAAC,CAAC;IACzD,CAAC,CAAC,OAAOkC,GAAG,EAAE;MACZ;MACA,IAAIA,GAAG,CAACyC,IAAI,KAAK,cAAc,IAAIzC,GAAG,CAACyC,IAAI,KAAK,UAAU,EAAE;QAC1D,OAAO,KAAK;MACd;MACA,MAAMzC,GAAG;IACX;IAEA,OAAO,IAAI;EACb;;EAIA;AACF;AACA;;EAGE,MAAMiE,YAAYA,CAACnG,UAAkB,EAAiB;IACpD,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,QAAQ;IACvB,MAAM,IAAI,CAAC8C,oBAAoB,CAAC;MAAE9C,MAAM;MAAET;IAAW,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC;IAClE,OAAO,IAAI,CAACf,SAAS,CAACe,UAAU,CAAC;EACnC;;EAEA;AACF;AACA;EACE,MAAMoG,SAASA,CAACpG,UAAkB,EAAEC,UAAkB,EAAEoG,OAAuB,EAA4B;IACzG,IAAI,CAAC7L,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,OAAO,IAAI,CAACsG,gBAAgB,CAACvG,UAAU,EAAEC,UAAU,EAAE,CAAC,EAAE,CAAC,EAAEoG,OAAO,CAAC;EACrE;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACE,MAAME,gBAAgBA,CACpBvG,UAAkB,EAClBC,UAAkB,EAClBuG,MAAc,EACdpD,MAAM,GAAG,CAAC,EACViD,OAAuB,EACG;IAC1B,IAAI,CAAC7L,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAAC9F,QAAQ,CAACqM,MAAM,CAAC,EAAE;MACrB,MAAM,IAAI3G,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA,IAAI,CAAC1F,QAAQ,CAACiJ,MAAM,CAAC,EAAE;MACrB,MAAM,IAAIvD,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IAEA,IAAI4G,KAAK,GAAG,EAAE;IACd,IAAID,MAAM,IAAIpD,MAAM,EAAE;MACpB,IAAIoD,MAAM,EAAE;QACVC,KAAK,GAAI,SAAQ,CAACD,MAAO,GAAE;MAC7B,CAAC,MAAM;QACLC,KAAK,GAAG,UAAU;QAClBD,MAAM,GAAG,CAAC;MACZ;MACA,IAAIpD,MAAM,EAAE;QACVqD,KAAK,IAAK,GAAE,CAACrD,MAAM,GAAGoD,MAAM,GAAG,CAAE,EAAC;MACpC;IACF;IAEA,IAAI7F,KAAK,GAAG,EAAE;IACd,IAAID,OAAuB,GAAG;MAC5B,IAAI+F,KAAK,KAAK,EAAE,IAAI;QAAEA;MAAM,CAAC;IAC/B,CAAC;IAED,IAAIJ,OAAO,EAAE;MACX,MAAMK,UAAkC,GAAG;QACzC,IAAIL,OAAO,CAACM,oBAAoB,IAAI;UAClC,iDAAiD,EAAEN,OAAO,CAACM;QAC7D,CAAC,CAAC;QACF,IAAIN,OAAO,CAACO,cAAc,IAAI;UAAE,2CAA2C,EAAEP,OAAO,CAACO;QAAe,CAAC,CAAC;QACtG,IAAIP,OAAO,CAACQ,iBAAiB,IAAI;UAC/B,+CAA+C,EAAER,OAAO,CAACQ;QAC3D,CAAC;MACH,CAAC;MACDlG,KAAK,GAAGxI,EAAE,CAAC0K,SAAS,CAACwD,OAAO,CAAC;MAC7B3F,OAAO,GAAG;QACR,GAAGzF,eAAe,CAACyL,UAAU,CAAC;QAC9B,GAAGhG;MACL,CAAC;IACH;IAEA,MAAMoG,mBAAmB,GAAG,CAAC,GAAG,CAAC;IACjC,IAAIL,KAAK,EAAE;MACTK,mBAAmB,CAACC,IAAI,CAAC,GAAG,CAAC;IAC/B;IACA,MAAMtG,MAAM,GAAG,KAAK;IAEpB,OAAO,MAAM,IAAI,CAACwC,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAES,OAAO;MAAEC;IAAM,CAAC,EAAE,EAAE,EAAEmG,mBAAmB,CAAC;EACjH;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,MAAME,UAAUA,CAAChH,UAAkB,EAAEC,UAAkB,EAAEgH,QAAgB,EAAEZ,OAAuB,EAAiB;IACjH;IACA,IAAI,CAAC7L,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAAC1F,QAAQ,CAAC0M,QAAQ,CAAC,EAAE;MACvB,MAAM,IAAIpH,SAAS,CAAC,qCAAqC,CAAC;IAC5D;IAEA,MAAMqH,iBAAiB,GAAG,MAAAA,CAAA,KAA6B;MACrD,IAAIC,cAA+B;MACnC,MAAMC,OAAO,GAAG,MAAM,IAAI,CAACC,UAAU,CAACrH,UAAU,EAAEC,UAAU,EAAEoG,OAAO,CAAC;MACtE,MAAMiB,WAAW,GAAG3D,MAAM,CAAC4D,IAAI,CAACH,OAAO,CAACI,IAAI,CAAC,CAAClG,QAAQ,CAAC,QAAQ,CAAC;MAChE,MAAMmG,QAAQ,GAAI,GAAER,QAAS,IAAGK,WAAY,aAAY;MAExD,MAAMnO,GAAG,CAACuO,KAAK,CAAC7P,IAAI,CAAC8P,OAAO,CAACV,QAAQ,CAAC,EAAE;QAAEW,SAAS,EAAE;MAAK,CAAC,CAAC;MAE5D,IAAIpB,MAAM,GAAG,CAAC;MACd,IAAI;QACF,MAAMqB,KAAK,GAAG,MAAM1O,GAAG,CAAC2O,IAAI,CAACL,QAAQ,CAAC;QACtC,IAAIL,OAAO,CAACW,IAAI,KAAKF,KAAK,CAACE,IAAI,EAAE;UAC/B,OAAON,QAAQ;QACjB;QACAjB,MAAM,GAAGqB,KAAK,CAACE,IAAI;QACnBZ,cAAc,GAAGzP,EAAE,CAACsQ,iBAAiB,CAACP,QAAQ,EAAE;UAAEQ,KAAK,EAAE;QAAI,CAAC,CAAC;MACjE,CAAC,CAAC,OAAOnG,CAAC,EAAE;QACV,IAAIA,CAAC,YAAYtE,KAAK,IAAKsE,CAAC,CAAiC6C,IAAI,KAAK,QAAQ,EAAE;UAC9E;UACAwC,cAAc,GAAGzP,EAAE,CAACsQ,iBAAiB,CAACP,QAAQ,EAAE;YAAEQ,KAAK,EAAE;UAAI,CAAC,CAAC;QACjE,CAAC,MAAM;UACL;UACA,MAAMnG,CAAC;QACT;MACF;MAEA,MAAMoG,cAAc,GAAG,MAAM,IAAI,CAAC3B,gBAAgB,CAACvG,UAAU,EAAEC,UAAU,EAAEuG,MAAM,EAAE,CAAC,EAAEH,OAAO,CAAC;MAE9F,MAAMjN,aAAa,CAAC+O,QAAQ,CAACD,cAAc,EAAEf,cAAc,CAAC;MAC5D,MAAMU,KAAK,GAAG,MAAM1O,GAAG,CAAC2O,IAAI,CAACL,QAAQ,CAAC;MACtC,IAAII,KAAK,CAACE,IAAI,KAAKX,OAAO,CAACW,IAAI,EAAE;QAC/B,OAAON,QAAQ;MACjB;MAEA,MAAM,IAAIjK,KAAK,CAAC,sDAAsD,CAAC;IACzE,CAAC;IAED,MAAMiK,QAAQ,GAAG,MAAMP,iBAAiB,CAAC,CAAC;IAC1C,MAAM/N,GAAG,CAACiP,MAAM,CAACX,QAAQ,EAAER,QAAQ,CAAC;EACtC;;EAEA;AACF;AACA;EACE,MAAMI,UAAUA,CAACrH,UAAkB,EAAEC,UAAkB,EAAEoI,QAAyB,EAA2B;IAC3G,MAAMC,UAAU,GAAGD,QAAQ,IAAI,CAAC,CAAC;IACjC,IAAI,CAAC7N,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IAEA,IAAI,CAAC7F,QAAQ,CAACkO,UAAU,CAAC,EAAE;MACzB,MAAM,IAAIhQ,MAAM,CAACuF,oBAAoB,CAAC,qCAAqC,CAAC;IAC9E;IAEA,MAAM8C,KAAK,GAAGxI,EAAE,CAAC0K,SAAS,CAACyF,UAAU,CAAC;IACtC,MAAM7H,MAAM,GAAG,MAAM;IACrB,MAAMgD,GAAG,GAAG,MAAM,IAAI,CAACF,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU;IAAM,CAAC,CAAC;IAEtF,OAAO;MACLoH,IAAI,EAAEQ,QAAQ,CAAC9E,GAAG,CAAC/C,OAAO,CAAC,gBAAgB,CAAW,CAAC;MACvD8H,QAAQ,EAAEhP,eAAe,CAACiK,GAAG,CAAC/C,OAAyB,CAAC;MACxD+H,YAAY,EAAE,IAAI1E,IAAI,CAACN,GAAG,CAAC/C,OAAO,CAAC,eAAe,CAAW,CAAC;MAC9DgI,SAAS,EAAE9O,YAAY,CAAC6J,GAAG,CAAC/C,OAAyB,CAAC;MACtD8G,IAAI,EAAErM,YAAY,CAACsI,GAAG,CAAC/C,OAAO,CAAC8G,IAAI;IACrC,CAAC;EACH;EAEA,MAAMmB,YAAYA,CAAC3I,UAAkB,EAAEC,UAAkB,EAAE2I,UAA0B,EAAiB;IACpG,IAAI,CAACpO,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,wBAAuBrE,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IAEA,IAAI2I,UAAU,IAAI,CAACxO,QAAQ,CAACwO,UAAU,CAAC,EAAE;MACvC,MAAM,IAAItQ,MAAM,CAACuF,oBAAoB,CAAC,uCAAuC,CAAC;IAChF;IAEA,MAAM4C,MAAM,GAAG,QAAQ;IAEvB,MAAMC,OAAuB,GAAG,CAAC,CAAC;IAClC,IAAIkI,UAAU,aAAVA,UAAU,eAAVA,UAAU,CAAEC,gBAAgB,EAAE;MAChCnI,OAAO,CAAC,mCAAmC,CAAC,GAAG,IAAI;IACrD;IACA,IAAIkI,UAAU,aAAVA,UAAU,eAAVA,UAAU,CAAEE,WAAW,EAAE;MAC3BpI,OAAO,CAAC,sBAAsB,CAAC,GAAG,IAAI;IACxC;IAEA,MAAMqI,WAAmC,GAAG,CAAC,CAAC;IAC9C,IAAIH,UAAU,aAAVA,UAAU,eAAVA,UAAU,CAAEF,SAAS,EAAE;MACzBK,WAAW,CAACL,SAAS,GAAI,GAAEE,UAAU,CAACF,SAAU,EAAC;IACnD;IACA,MAAM/H,KAAK,GAAGxI,EAAE,CAAC0K,SAAS,CAACkG,WAAW,CAAC;IAEvC,MAAM,IAAI,CAACxF,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAES,OAAO;MAAEC;IAAM,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;EACrG;;EAEA;;EAEAqI,qBAAqBA,CACnBC,MAAc,EACdC,MAAc,EACdtB,SAAkB,EAC0B;IAC5C,IAAIsB,MAAM,KAAK3L,SAAS,EAAE;MACxB2L,MAAM,GAAG,EAAE;IACb;IACA,IAAItB,SAAS,KAAKrK,SAAS,EAAE;MAC3BqK,SAAS,GAAG,KAAK;IACnB;IACA,IAAI,CAACpN,iBAAiB,CAACyO,MAAM,CAAC,EAAE;MAC9B,MAAM,IAAI3Q,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAG4E,MAAM,CAAC;IAC3E;IACA,IAAI,CAACrO,aAAa,CAACsO,MAAM,CAAC,EAAE;MAC1B,MAAM,IAAI5Q,MAAM,CAAC6Q,kBAAkB,CAAE,oBAAmBD,MAAO,EAAC,CAAC;IACnE;IACA,IAAI,CAAClP,SAAS,CAAC4N,SAAS,CAAC,EAAE;MACzB,MAAM,IAAI/H,SAAS,CAAC,uCAAuC,CAAC;IAC9D;IACA,MAAMuJ,SAAS,GAAGxB,SAAS,GAAG,EAAE,GAAG,GAAG;IACtC,IAAIyB,SAAS,GAAG,EAAE;IAClB,IAAIC,cAAc,GAAG,EAAE;IACvB,MAAMC,OAAkB,GAAG,EAAE;IAC7B,IAAIC,KAAK,GAAG,KAAK;;IAEjB;IACA,MAAMC,UAAU,GAAG,IAAI3R,MAAM,CAAC4R,QAAQ,CAAC;MAAEC,UAAU,EAAE;IAAK,CAAC,CAAC;IAC5DF,UAAU,CAACG,KAAK,GAAG,MAAM;MACvB;MACA,IAAIL,OAAO,CAACnG,MAAM,EAAE;QAClB,OAAOqG,UAAU,CAAC1C,IAAI,CAACwC,OAAO,CAACM,KAAK,CAAC,CAAC,CAAC;MACzC;MACA,IAAIL,KAAK,EAAE;QACT,OAAOC,UAAU,CAAC1C,IAAI,CAAC,IAAI,CAAC;MAC9B;MACA,IAAI,CAAC+C,0BAA0B,CAACb,MAAM,EAAEC,MAAM,EAAEG,SAAS,EAAEC,cAAc,EAAEF,SAAS,CAAC,CAACjE,IAAI,CACvFC,MAAM,IAAK;QACV;QACA;QACAA,MAAM,CAAC2E,QAAQ,CAAC1H,OAAO,CAAE6G,MAAM,IAAKK,OAAO,CAACxC,IAAI,CAACmC,MAAM,CAAC,CAAC;QACzDnR,KAAK,CAACiS,UAAU,CACd5E,MAAM,CAACmE,OAAO,EACd,CAACU,MAAM,EAAEhF,EAAE,KAAK;UACd;UACA;UACA;UACA,IAAI,CAACiF,SAAS,CAACjB,MAAM,EAAEgB,MAAM,CAACE,GAAG,EAAEF,MAAM,CAACG,QAAQ,CAAC,CAACjF,IAAI,CACrDkF,KAAa,IAAK;YACjB;YACA;YACAJ,MAAM,CAAClC,IAAI,GAAGsC,KAAK,CAACC,MAAM,CAAC,CAACC,GAAG,EAAEC,IAAI,KAAKD,GAAG,GAAGC,IAAI,CAACzC,IAAI,EAAE,CAAC,CAAC;YAC7DwB,OAAO,CAACxC,IAAI,CAACkD,MAAM,CAAC;YACpBhF,EAAE,CAAC,CAAC;UACN,CAAC,EACA/C,GAAU,IAAK+C,EAAE,CAAC/C,GAAG,CACxB,CAAC;QACH,CAAC,EACAA,GAAG,IAAK;UACP,IAAIA,GAAG,EAAE;YACPuH,UAAU,CAACgB,IAAI,CAAC,OAAO,EAAEvI,GAAG,CAAC;YAC7B;UACF;UACA,IAAIkD,MAAM,CAACsF,WAAW,EAAE;YACtBrB,SAAS,GAAGjE,MAAM,CAACuF,aAAa;YAChCrB,cAAc,GAAGlE,MAAM,CAACwF,kBAAkB;UAC5C,CAAC,MAAM;YACLpB,KAAK,GAAG,IAAI;UACd;;UAEA;UACA;UACAC,UAAU,CAACG,KAAK,CAAC,CAAC;QACpB,CACF,CAAC;MACH,CAAC,EACA9H,CAAC,IAAK;QACL2H,UAAU,CAACgB,IAAI,CAAC,OAAO,EAAE3I,CAAC,CAAC;MAC7B,CACF,CAAC;IACH,CAAC;IACD,OAAO2H,UAAU;EACnB;;EAEA;AACF;AACA;EACE,MAAMK,0BAA0BA,CAC9B9J,UAAkB,EAClBkJ,MAAc,EACdG,SAAiB,EACjBC,cAAsB,EACtBF,SAAiB,EACa;IAC9B,IAAI,CAAC5O,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACzF,QAAQ,CAAC2O,MAAM,CAAC,EAAE;MACrB,MAAM,IAAIrJ,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA,IAAI,CAACtF,QAAQ,CAAC8O,SAAS,CAAC,EAAE;MACxB,MAAM,IAAIxJ,SAAS,CAAC,sCAAsC,CAAC;IAC7D;IACA,IAAI,CAACtF,QAAQ,CAAC+O,cAAc,CAAC,EAAE;MAC7B,MAAM,IAAIzJ,SAAS,CAAC,2CAA2C,CAAC;IAClE;IACA,IAAI,CAACtF,QAAQ,CAAC6O,SAAS,CAAC,EAAE;MACxB,MAAM,IAAIvJ,SAAS,CAAC,sCAAsC,CAAC;IAC7D;IACA,MAAMgL,OAAO,GAAG,EAAE;IAClBA,OAAO,CAAC9D,IAAI,CAAE,UAASzL,SAAS,CAAC4N,MAAM,CAAE,EAAC,CAAC;IAC3C2B,OAAO,CAAC9D,IAAI,CAAE,aAAYzL,SAAS,CAAC8N,SAAS,CAAE,EAAC,CAAC;IAEjD,IAAIC,SAAS,EAAE;MACbwB,OAAO,CAAC9D,IAAI,CAAE,cAAazL,SAAS,CAAC+N,SAAS,CAAE,EAAC,CAAC;IACpD;IACA,IAAIC,cAAc,EAAE;MAClBuB,OAAO,CAAC9D,IAAI,CAAE,oBAAmBuC,cAAe,EAAC,CAAC;IACpD;IAEA,MAAMwB,UAAU,GAAG,IAAI;IACvBD,OAAO,CAAC9D,IAAI,CAAE,eAAc+D,UAAW,EAAC,CAAC;IACzCD,OAAO,CAACE,IAAI,CAAC,CAAC;IACdF,OAAO,CAACG,OAAO,CAAC,SAAS,CAAC;IAC1B,IAAIrK,KAAK,GAAG,EAAE;IACd,IAAIkK,OAAO,CAACzH,MAAM,GAAG,CAAC,EAAE;MACtBzC,KAAK,GAAI,GAAEkK,OAAO,CAACI,IAAI,CAAC,GAAG,CAAE,EAAC;IAChC;IACA,MAAMxK,MAAM,GAAG,KAAK;IACpB,MAAMgD,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,CAAC;IACtE,MAAM+C,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,OAAOlH,UAAU,CAAC2O,kBAAkB,CAACxH,IAAI,CAAC;EAC5C;;EAEA;AACF;AACA;AACA;EACE,MAAMyH,0BAA0BA,CAACnL,UAAkB,EAAEC,UAAkB,EAAES,OAAuB,EAAmB;IACjH,IAAI,CAAClG,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAAC7F,QAAQ,CAACsG,OAAO,CAAC,EAAE;MACtB,MAAM,IAAIpI,MAAM,CAACgO,sBAAsB,CAAC,wCAAwC,CAAC;IACnF;IACA,MAAM7F,MAAM,GAAG,MAAM;IACrB,MAAME,KAAK,GAAG,SAAS;IACvB,MAAM8C,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU,KAAK;MAAED;IAAQ,CAAC,CAAC;IAC3F,MAAMgD,IAAI,GAAG,MAAM9H,YAAY,CAAC6H,GAAG,CAAC;IACpC,OAAOxH,sBAAsB,CAACyH,IAAI,CAACpC,QAAQ,CAAC,CAAC,CAAC;EAChD;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACE,MAAM8J,oBAAoBA,CAACpL,UAAkB,EAAEC,UAAkB,EAAEmK,QAAgB,EAAiB;IAClG,MAAM3J,MAAM,GAAG,QAAQ;IACvB,MAAME,KAAK,GAAI,YAAWyJ,QAAS,EAAC;IAEpC,MAAMiB,cAAc,GAAG;MAAE5K,MAAM;MAAET,UAAU;MAAEC,UAAU,EAAEA,UAAU;MAAEU;IAAM,CAAC;IAC5E,MAAM,IAAI,CAAC4C,oBAAoB,CAAC8H,cAAc,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC;EAC5D;EAEA,MAAMC,YAAYA,CAACtL,UAAkB,EAAEC,UAAkB,EAA+B;IAAA,IAAAsL,aAAA;IACtF,IAAI,CAAC/Q,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IAEA,IAAIuL,YAAgE;IACpE,IAAInC,SAAS,GAAG,EAAE;IAClB,IAAIC,cAAc,GAAG,EAAE;IACvB,SAAS;MACP,MAAMlE,MAAM,GAAG,MAAM,IAAI,CAAC0E,0BAA0B,CAAC9J,UAAU,EAAEC,UAAU,EAAEoJ,SAAS,EAAEC,cAAc,EAAE,EAAE,CAAC;MAC3G,KAAK,MAAMW,MAAM,IAAI7E,MAAM,CAACmE,OAAO,EAAE;QACnC,IAAIU,MAAM,CAACE,GAAG,KAAKlK,UAAU,EAAE;UAC7B,IAAI,CAACuL,YAAY,IAAIvB,MAAM,CAACwB,SAAS,CAACC,OAAO,CAAC,CAAC,GAAGF,YAAY,CAACC,SAAS,CAACC,OAAO,CAAC,CAAC,EAAE;YAClFF,YAAY,GAAGvB,MAAM;UACvB;QACF;MACF;MACA,IAAI7E,MAAM,CAACsF,WAAW,EAAE;QACtBrB,SAAS,GAAGjE,MAAM,CAACuF,aAAa;QAChCrB,cAAc,GAAGlE,MAAM,CAACwF,kBAAkB;QAC1C;MACF;MAEA;IACF;IACA,QAAAW,aAAA,GAAOC,YAAY,cAAAD,aAAA,uBAAZA,aAAA,CAAcnB,QAAQ;EAC/B;;EAEA;AACF;AACA;EACE,MAAMuB,uBAAuBA,CAC3B3L,UAAkB,EAClBC,UAAkB,EAClBmK,QAAgB,EAChBwB,KAGG,EACkD;IACrD,IAAI,CAACpR,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAAC1F,QAAQ,CAAC6P,QAAQ,CAAC,EAAE;MACvB,MAAM,IAAIvK,SAAS,CAAC,qCAAqC,CAAC;IAC5D;IACA,IAAI,CAACzF,QAAQ,CAACwR,KAAK,CAAC,EAAE;MACpB,MAAM,IAAI/L,SAAS,CAAC,iCAAiC,CAAC;IACxD;IAEA,IAAI,CAACuK,QAAQ,EAAE;MACb,MAAM,IAAI9R,MAAM,CAACuF,oBAAoB,CAAC,0BAA0B,CAAC;IACnE;IAEA,MAAM4C,MAAM,GAAG,MAAM;IACrB,MAAME,KAAK,GAAI,YAAWrF,SAAS,CAAC8O,QAAQ,CAAE,EAAC;IAE/C,MAAMyB,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC,CAAC;IACpC,MAAMyG,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAAC;MAClCoG,uBAAuB,EAAE;QACvBlG,CAAC,EAAE;UACDC,KAAK,EAAE;QACT,CAAC;QACDkG,IAAI,EAAEH,KAAK,CAACI,GAAG,CAAExE,IAAI,IAAK;UACxB,OAAO;YACLyE,UAAU,EAAEzE,IAAI,CAAC0E,IAAI;YACrBC,IAAI,EAAE3E,IAAI,CAACA;UACb,CAAC;QACH,CAAC;MACH;IACF,CAAC,CAAC;IAEF,MAAM/D,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU;IAAM,CAAC,EAAEuC,OAAO,CAAC;IAC3F,MAAMQ,IAAI,GAAG,MAAM9H,YAAY,CAAC6H,GAAG,CAAC;IACpC,MAAM2B,MAAM,GAAGpJ,sBAAsB,CAAC0H,IAAI,CAACpC,QAAQ,CAAC,CAAC,CAAC;IACtD,IAAI,CAAC8D,MAAM,EAAE;MACX,MAAM,IAAI5H,KAAK,CAAC,sCAAsC,CAAC;IACzD;IAEA,IAAI4H,MAAM,CAACV,OAAO,EAAE;MAClB;MACA,MAAM,IAAIpM,MAAM,CAACmM,OAAO,CAACW,MAAM,CAACgH,UAAU,CAAC;IAC7C;IAEA,OAAO;MACL;MACA;MACA5E,IAAI,EAAEpC,MAAM,CAACoC,IAAc;MAC3BkB,SAAS,EAAE9O,YAAY,CAAC6J,GAAG,CAAC/C,OAAyB;IACvD,CAAC;EACH;;EAEA;AACF;AACA;EACE,MAAgBwJ,SAASA,CAAClK,UAAkB,EAAEC,UAAkB,EAAEmK,QAAgB,EAA2B;IAC3G,IAAI,CAAC5P,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAAC1F,QAAQ,CAAC6P,QAAQ,CAAC,EAAE;MACvB,MAAM,IAAIvK,SAAS,CAAC,qCAAqC,CAAC;IAC5D;IACA,IAAI,CAACuK,QAAQ,EAAE;MACb,MAAM,IAAI9R,MAAM,CAACuF,oBAAoB,CAAC,0BAA0B,CAAC;IACnE;IAEA,MAAMwM,KAAqB,GAAG,EAAE;IAChC,IAAIgC,MAAM,GAAG,CAAC;IACd,IAAIjH,MAAM;IACV,GAAG;MACDA,MAAM,GAAG,MAAM,IAAI,CAACkH,cAAc,CAACtM,UAAU,EAAEC,UAAU,EAAEmK,QAAQ,EAAEiC,MAAM,CAAC;MAC5EA,MAAM,GAAGjH,MAAM,CAACiH,MAAM;MACtBhC,KAAK,CAACtD,IAAI,CAAC,GAAG3B,MAAM,CAACiF,KAAK,CAAC;IAC7B,CAAC,QAAQjF,MAAM,CAACsF,WAAW;IAE3B,OAAOL,KAAK;EACd;;EAEA;AACF;AACA;EACE,MAAciC,cAAcA,CAACtM,UAAkB,EAAEC,UAAkB,EAAEmK,QAAgB,EAAEiC,MAAc,EAAE;IACrG,IAAI,CAAC7R,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAAC1F,QAAQ,CAAC6P,QAAQ,CAAC,EAAE;MACvB,MAAM,IAAIvK,SAAS,CAAC,qCAAqC,CAAC;IAC5D;IACA,IAAI,CAAC1F,QAAQ,CAACkS,MAAM,CAAC,EAAE;MACrB,MAAM,IAAIxM,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA,IAAI,CAACuK,QAAQ,EAAE;MACb,MAAM,IAAI9R,MAAM,CAACuF,oBAAoB,CAAC,0BAA0B,CAAC;IACnE;IAEA,IAAI8C,KAAK,GAAI,YAAWrF,SAAS,CAAC8O,QAAQ,CAAE,EAAC;IAC7C,IAAIiC,MAAM,EAAE;MACV1L,KAAK,IAAK,uBAAsB0L,MAAO,EAAC;IAC1C;IAEA,MAAM5L,MAAM,GAAG,KAAK;IACpB,MAAMgD,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU;IAAM,CAAC,CAAC;IAClF,OAAOpE,UAAU,CAACgQ,cAAc,CAAC,MAAM1Q,YAAY,CAAC4H,GAAG,CAAC,CAAC;EAC3D;EAEA,MAAM+I,WAAWA,CAAA,EAAkC;IACjD,MAAM/L,MAAM,GAAG,KAAK;IACpB,MAAMgM,UAAU,GAAG,IAAI,CAAC3O,MAAM,IAAIrF,cAAc;IAChD,MAAMiU,OAAO,GAAG,MAAM,IAAI,CAACzJ,gBAAgB,CAAC;MAAExC;IAAO,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,EAAEgM,UAAU,CAAC;IAC9E,MAAME,SAAS,GAAG,MAAM9Q,YAAY,CAAC6Q,OAAO,CAAC;IAC7C,OAAOnQ,UAAU,CAACqQ,eAAe,CAACD,SAAS,CAAC;EAC9C;;EAEA;AACF;AACA;EACEE,iBAAiBA,CAAC9E,IAAY,EAAE;IAC9B,IAAI,CAAC5N,QAAQ,CAAC4N,IAAI,CAAC,EAAE;MACnB,MAAM,IAAIlI,SAAS,CAAC,iCAAiC,CAAC;IACxD;IACA,IAAIkI,IAAI,GAAG,IAAI,CAAC5K,aAAa,EAAE;MAC7B,MAAM,IAAI0C,SAAS,CAAE,gCAA+B,IAAI,CAAC1C,aAAc,EAAC,CAAC;IAC3E;IACA,IAAI,IAAI,CAAC+B,gBAAgB,EAAE;MACzB,OAAO,IAAI,CAACjC,QAAQ;IACtB;IACA,IAAIA,QAAQ,GAAG,IAAI,CAACA,QAAQ;IAC5B,SAAS;MACP;MACA;MACA,IAAIA,QAAQ,GAAG,KAAK,GAAG8K,IAAI,EAAE;QAC3B,OAAO9K,QAAQ;MACjB;MACA;MACAA,QAAQ,IAAI,EAAE,GAAG,IAAI,GAAG,IAAI;IAC9B;EACF;;EAEA;AACF;AACA;EACE,MAAM6P,UAAUA,CAAC9M,UAAkB,EAAEC,UAAkB,EAAEgH,QAAgB,EAAEuB,QAAyB,EAAE;IACpG,IAAI,CAAChO,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IAEA,IAAI,CAAC1F,QAAQ,CAAC0M,QAAQ,CAAC,EAAE;MACvB,MAAM,IAAIpH,SAAS,CAAC,qCAAqC,CAAC;IAC5D;IACA,IAAI2I,QAAQ,IAAI,CAACpO,QAAQ,CAACoO,QAAQ,CAAC,EAAE;MACnC,MAAM,IAAI3I,SAAS,CAAC,qCAAqC,CAAC;IAC5D;;IAEA;IACA2I,QAAQ,GAAG1O,iBAAiB,CAAC0O,QAAQ,IAAI,CAAC,CAAC,EAAEvB,QAAQ,CAAC;IACtD,MAAMa,IAAI,GAAG,MAAM3O,GAAG,CAAC2O,IAAI,CAACb,QAAQ,CAAC;IACrC,OAAO,MAAM,IAAI,CAAC8F,SAAS,CAAC/M,UAAU,EAAEC,UAAU,EAAEvI,EAAE,CAACsV,gBAAgB,CAAC/F,QAAQ,CAAC,EAAEa,IAAI,CAACC,IAAI,EAAES,QAAQ,CAAC;EACzG;;EAEA;AACF;AACA;AACA;EACE,MAAMuE,SAASA,CACb/M,UAAkB,EAClBC,UAAkB,EAClBnI,MAAyC,EACzCiQ,IAAa,EACbS,QAA6B,EACA;IAC7B,IAAI,CAAChO,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,wBAAuBrE,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;;IAEA;IACA;IACA,IAAI7F,QAAQ,CAAC2N,IAAI,CAAC,EAAE;MAClBS,QAAQ,GAAGT,IAAI;IACjB;IACA;IACA,MAAMrH,OAAO,GAAGzF,eAAe,CAACuN,QAAQ,CAAC;IACzC,IAAI,OAAO1Q,MAAM,KAAK,QAAQ,IAAIA,MAAM,YAAY6L,MAAM,EAAE;MAC1D;MACAoE,IAAI,GAAGjQ,MAAM,CAACsL,MAAM;MACpBtL,MAAM,GAAGoD,cAAc,CAACpD,MAAM,CAAC;IACjC,CAAC,MAAM,IAAI,CAACwC,gBAAgB,CAACxC,MAAM,CAAC,EAAE;MACpC,MAAM,IAAI+H,SAAS,CAAC,4EAA4E,CAAC;IACnG;IAEA,IAAI1F,QAAQ,CAAC4N,IAAI,CAAC,IAAIA,IAAI,GAAG,CAAC,EAAE;MAC9B,MAAM,IAAIzP,MAAM,CAACuF,oBAAoB,CAAE,wCAAuCkK,IAAK,EAAC,CAAC;IACvF;;IAEA;IACA;IACA,IAAI,CAAC5N,QAAQ,CAAC4N,IAAI,CAAC,EAAE;MACnBA,IAAI,GAAG,IAAI,CAAC5K,aAAa;IAC3B;;IAEA;IACA;IACA,IAAI4K,IAAI,KAAKxK,SAAS,EAAE;MACtB,MAAM0P,QAAQ,GAAG,MAAMxT,gBAAgB,CAAC3B,MAAM,CAAC;MAC/C,IAAImV,QAAQ,KAAK,IAAI,EAAE;QACrBlF,IAAI,GAAGkF,QAAQ;MACjB;IACF;IAEA,IAAI,CAAC9S,QAAQ,CAAC4N,IAAI,CAAC,EAAE;MACnB;MACAA,IAAI,GAAG,IAAI,CAAC5K,aAAa;IAC3B;IACA,IAAI4K,IAAI,KAAK,CAAC,EAAE;MACd,OAAO,IAAI,CAACmF,YAAY,CAAClN,UAAU,EAAEC,UAAU,EAAES,OAAO,EAAEiD,MAAM,CAAC4D,IAAI,CAAC,EAAE,CAAC,CAAC;IAC5E;IAEA,MAAMtK,QAAQ,GAAG,IAAI,CAAC4P,iBAAiB,CAAC9E,IAAI,CAAC;IAC7C,IAAI,OAAOjQ,MAAM,KAAK,QAAQ,IAAI6L,MAAM,CAACC,QAAQ,CAAC9L,MAAM,CAAC,IAAIiQ,IAAI,IAAI9K,QAAQ,EAAE;MAC7E,MAAMkQ,GAAG,GAAG7S,gBAAgB,CAACxC,MAAM,CAAC,GAAG,MAAM8D,YAAY,CAAC9D,MAAM,CAAC,GAAG6L,MAAM,CAAC4D,IAAI,CAACzP,MAAM,CAAC;MACvF,OAAO,IAAI,CAACoV,YAAY,CAAClN,UAAU,EAAEC,UAAU,EAAES,OAAO,EAAEyM,GAAG,CAAC;IAChE;IAEA,OAAO,IAAI,CAACC,YAAY,CAACpN,UAAU,EAAEC,UAAU,EAAES,OAAO,EAAE5I,MAAM,EAAEmF,QAAQ,CAAC;EAC7E;;EAEA;AACF;AACA;AACA;EACE,MAAciQ,YAAYA,CACxBlN,UAAkB,EAClBC,UAAkB,EAClBS,OAAuB,EACvByM,GAAW,EACkB;IAC7B,MAAM;MAAEE,MAAM;MAAEhK;IAAU,CAAC,GAAGxJ,UAAU,CAACsT,GAAG,EAAE,IAAI,CAAChO,YAAY,CAAC;IAChEuB,OAAO,CAAC,gBAAgB,CAAC,GAAGyM,GAAG,CAAC/J,MAAM;IACtC,IAAI,CAAC,IAAI,CAACjE,YAAY,EAAE;MACtBuB,OAAO,CAAC,aAAa,CAAC,GAAG2M,MAAM;IACjC;IACA,MAAM5J,GAAG,GAAG,MAAM,IAAI,CAACH,sBAAsB,CAC3C;MACE7C,MAAM,EAAE,KAAK;MACbT,UAAU;MACVC,UAAU;MACVS;IACF,CAAC,EACDyM,GAAG,EACH9J,SAAS,EACT,CAAC,GAAG,CAAC,EACL,EACF,CAAC;IACD,MAAM1H,aAAa,CAAC8H,GAAG,CAAC;IACxB,OAAO;MACL+D,IAAI,EAAErM,YAAY,CAACsI,GAAG,CAAC/C,OAAO,CAAC8G,IAAI,CAAC;MACpCkB,SAAS,EAAE9O,YAAY,CAAC6J,GAAG,CAAC/C,OAAyB;IACvD,CAAC;EACH;;EAEA;AACF;AACA;AACA;EACE,MAAc0M,YAAYA,CACxBpN,UAAkB,EAClBC,UAAkB,EAClBS,OAAuB,EACvBgD,IAAqB,EACrBzG,QAAgB,EACa;IAC7B;IACA;IACA,MAAMqQ,QAA8B,GAAG,CAAC,CAAC;;IAEzC;IACA;IACA,MAAMC,KAAa,GAAG,EAAE;IAExB,MAAMC,gBAAgB,GAAG,MAAM,IAAI,CAAClC,YAAY,CAACtL,UAAU,EAAEC,UAAU,CAAC;IACxE,IAAImK,QAAgB;IACpB,IAAI,CAACoD,gBAAgB,EAAE;MACrBpD,QAAQ,GAAG,MAAM,IAAI,CAACe,0BAA0B,CAACnL,UAAU,EAAEC,UAAU,EAAES,OAAO,CAAC;IACnF,CAAC,MAAM;MACL0J,QAAQ,GAAGoD,gBAAgB;MAC3B,MAAMC,OAAO,GAAG,MAAM,IAAI,CAACvD,SAAS,CAAClK,UAAU,EAAEC,UAAU,EAAEuN,gBAAgB,CAAC;MAC9EC,OAAO,CAACpL,OAAO,CAAEP,CAAC,IAAK;QACrBwL,QAAQ,CAACxL,CAAC,CAACoK,IAAI,CAAC,GAAGpK,CAAC;MACtB,CAAC,CAAC;IACJ;IAEA,MAAM4L,QAAQ,GAAG,IAAI1V,YAAY,CAAC;MAAE+P,IAAI,EAAE9K,QAAQ;MAAE0Q,WAAW,EAAE;IAAM,CAAC,CAAC;;IAEzE;IACA,MAAM,CAACzV,CAAC,EAAE0V,CAAC,CAAC,GAAG,MAAMC,OAAO,CAACC,GAAG,CAAC,CAC/B,IAAID,OAAO,CAAC,CAACE,OAAO,EAAEC,MAAM,KAAK;MAC/BtK,IAAI,CAACuK,IAAI,CAACP,QAAQ,CAAC,CAACQ,EAAE,CAAC,OAAO,EAAEF,MAAM,CAAC;MACvCN,QAAQ,CAACQ,EAAE,CAAC,KAAK,EAAEH,OAAO,CAAC,CAACG,EAAE,CAAC,OAAO,EAAEF,MAAM,CAAC;IACjD,CAAC,CAAC,EACF,CAAC,YAAY;MACX,IAAIG,UAAU,GAAG,CAAC;MAElB,WAAW,MAAMC,KAAK,IAAIV,QAAQ,EAAE;QAClC,MAAMW,GAAG,GAAG5W,MAAM,CAAC6W,UAAU,CAAC,KAAK,CAAC,CAACC,MAAM,CAACH,KAAK,CAAC,CAACI,MAAM,CAAC,CAAC;QAE3D,MAAMC,OAAO,GAAGnB,QAAQ,CAACa,UAAU,CAAC;QACpC,IAAIM,OAAO,EAAE;UACX,IAAIA,OAAO,CAACjH,IAAI,KAAK6G,GAAG,CAAC/M,QAAQ,CAAC,KAAK,CAAC,EAAE;YACxCiM,KAAK,CAACxG,IAAI,CAAC;cAAEmF,IAAI,EAAEiC,UAAU;cAAE3G,IAAI,EAAEiH,OAAO,CAACjH;YAAK,CAAC,CAAC;YACpD2G,UAAU,EAAE;YACZ;UACF;QACF;QAEAA,UAAU,EAAE;;QAEZ;QACA,MAAMvO,OAAsB,GAAG;UAC7Ba,MAAM,EAAE,KAAK;UACbE,KAAK,EAAExI,EAAE,CAAC0K,SAAS,CAAC;YAAEsL,UAAU;YAAE/D;UAAS,CAAC,CAAC;UAC7C1J,OAAO,EAAE;YACP,gBAAgB,EAAE0N,KAAK,CAAChL,MAAM;YAC9B,aAAa,EAAEiL,GAAG,CAAC/M,QAAQ,CAAC,QAAQ;UACtC,CAAC;UACDtB,UAAU;UACVC;QACF,CAAC;QAED,MAAMgC,QAAQ,GAAG,MAAM,IAAI,CAACsB,oBAAoB,CAAC3D,OAAO,EAAEwO,KAAK,CAAC;QAEhE,IAAI5G,IAAI,GAAGvF,QAAQ,CAACvB,OAAO,CAAC8G,IAAI;QAChC,IAAIA,IAAI,EAAE;UACRA,IAAI,GAAGA,IAAI,CAAChF,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAACA,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC;QACjD,CAAC,MAAM;UACLgF,IAAI,GAAG,EAAE;QACX;QAEA+F,KAAK,CAACxG,IAAI,CAAC;UAAEmF,IAAI,EAAEiC,UAAU;UAAE3G;QAAK,CAAC,CAAC;MACxC;MAEA,OAAO,MAAM,IAAI,CAACmE,uBAAuB,CAAC3L,UAAU,EAAEC,UAAU,EAAEmK,QAAQ,EAAEmD,KAAK,CAAC;IACpF,CAAC,EAAE,CAAC,CACL,CAAC;IAEF,OAAOK,CAAC;EACV;EAIA,MAAMc,uBAAuBA,CAAC1O,UAAkB,EAAiB;IAC/D,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,QAAQ;IACvB,MAAME,KAAK,GAAG,aAAa;IAC3B,MAAM,IAAI,CAAC4C,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,EAAE,CAAC;EACpF;EAIA,MAAMgO,oBAAoBA,CAAC3O,UAAkB,EAAE4O,iBAAwC,EAAE;IACvF,IAAI,CAACpU,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAAC5F,QAAQ,CAACwU,iBAAiB,CAAC,EAAE;MAChC,MAAM,IAAItW,MAAM,CAACuF,oBAAoB,CAAC,8CAA8C,CAAC;IACvF,CAAC,MAAM;MACL,IAAI3F,CAAC,CAACgC,OAAO,CAAC0U,iBAAiB,CAACC,IAAI,CAAC,EAAE;QACrC,MAAM,IAAIvW,MAAM,CAACuF,oBAAoB,CAAC,sBAAsB,CAAC;MAC/D,CAAC,MAAM,IAAI+Q,iBAAiB,CAACC,IAAI,IAAI,CAACtU,QAAQ,CAACqU,iBAAiB,CAACC,IAAI,CAAC,EAAE;QACtE,MAAM,IAAIvW,MAAM,CAACuF,oBAAoB,CAAC,wBAAwB,EAAE+Q,iBAAiB,CAACC,IAAI,CAAC;MACzF;MACA,IAAI3W,CAAC,CAACgC,OAAO,CAAC0U,iBAAiB,CAACE,KAAK,CAAC,EAAE;QACtC,MAAM,IAAIxW,MAAM,CAACuF,oBAAoB,CAAC,gDAAgD,CAAC;MACzF;IACF;IACA,MAAM4C,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,aAAa;IAC3B,MAAMD,OAA+B,GAAG,CAAC,CAAC;IAE1C,MAAMqO,uBAAuB,GAAG;MAC9BC,wBAAwB,EAAE;QACxBC,IAAI,EAAEL,iBAAiB,CAACC,IAAI;QAC5BK,IAAI,EAAEN,iBAAiB,CAACE;MAC1B;IACF,CAAC;IAED,MAAMjD,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MAAEC,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM,CAAC;MAAEC,QAAQ,EAAE;IAAK,CAAC,CAAC;IACrF,MAAMsG,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAACqJ,uBAAuB,CAAC;IAC5DrO,OAAO,CAAC,aAAa,CAAC,GAAGtF,KAAK,CAAC8H,OAAO,CAAC;IACvC,MAAM,IAAI,CAACK,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW,KAAK;MAAED;IAAQ,CAAC,EAAEwC,OAAO,CAAC;EAClF;EAIA,MAAMiM,oBAAoBA,CAACnP,UAAkB,EAAE;IAC7C,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,aAAa;IAE3B,MAAM+L,OAAO,GAAG,MAAM,IAAI,CAACzJ,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;IAC1F,MAAMgM,SAAS,GAAG,MAAM9Q,YAAY,CAAC6Q,OAAO,CAAC;IAC7C,OAAOnQ,UAAU,CAAC6S,sBAAsB,CAACzC,SAAS,CAAC;EACrD;EAQA,MAAM0C,kBAAkBA,CACtBrP,UAAkB,EAClBC,UAAkB,EAClBoG,OAAmC,EACP;IAC5B,IAAI,CAAC7L,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IAEA,IAAIoG,OAAO,EAAE;MACX,IAAI,CAACjM,QAAQ,CAACiM,OAAO,CAAC,EAAE;QACtB,MAAM,IAAIxG,SAAS,CAAC,oCAAoC,CAAC;MAC3D,CAAC,MAAM,IAAIoB,MAAM,CAACqO,IAAI,CAACjJ,OAAO,CAAC,CAACjD,MAAM,GAAG,CAAC,IAAIiD,OAAO,CAACqC,SAAS,IAAI,CAACnO,QAAQ,CAAC8L,OAAO,CAACqC,SAAS,CAAC,EAAE;QAC/F,MAAM,IAAI7I,SAAS,CAAC,sCAAsC,EAAEwG,OAAO,CAACqC,SAAS,CAAC;MAChF;IACF;IAEA,MAAMjI,MAAM,GAAG,KAAK;IACpB,IAAIE,KAAK,GAAG,YAAY;IAExB,IAAI0F,OAAO,aAAPA,OAAO,eAAPA,OAAO,CAAEqC,SAAS,EAAE;MACtB/H,KAAK,IAAK,cAAa0F,OAAO,CAACqC,SAAU,EAAC;IAC5C;IAEA,MAAMgE,OAAO,GAAG,MAAM,IAAI,CAACzJ,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU;IAAM,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC;IACjG,MAAM4O,MAAM,GAAG,MAAM1T,YAAY,CAAC6Q,OAAO,CAAC;IAC1C,OAAOtQ,0BAA0B,CAACmT,MAAM,CAAC;EAC3C;EAGA,MAAMC,kBAAkBA,CACtBxP,UAAkB,EAClBC,UAAkB,EAClBwP,OAAO,GAAG;IACRC,MAAM,EAAEhX,iBAAiB,CAACiX;EAC5B,CAA8B,EACf;IACf,IAAI,CAACnV,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IAEA,IAAI,CAAC7F,QAAQ,CAACqV,OAAO,CAAC,EAAE;MACtB,MAAM,IAAI5P,SAAS,CAAC,oCAAoC,CAAC;IAC3D,CAAC,MAAM;MACL,IAAI,CAAC,CAACnH,iBAAiB,CAACiX,OAAO,EAAEjX,iBAAiB,CAACkX,QAAQ,CAAC,CAAC1P,QAAQ,CAACuP,OAAO,aAAPA,OAAO,uBAAPA,OAAO,CAAEC,MAAM,CAAC,EAAE;QACtF,MAAM,IAAI7P,SAAS,CAAC,kBAAkB,GAAG4P,OAAO,CAACC,MAAM,CAAC;MAC1D;MACA,IAAID,OAAO,CAAC/G,SAAS,IAAI,CAAC+G,OAAO,CAAC/G,SAAS,CAACtF,MAAM,EAAE;QAClD,MAAM,IAAIvD,SAAS,CAAC,sCAAsC,GAAG4P,OAAO,CAAC/G,SAAS,CAAC;MACjF;IACF;IAEA,MAAMjI,MAAM,GAAG,KAAK;IACpB,IAAIE,KAAK,GAAG,YAAY;IAExB,IAAI8O,OAAO,CAAC/G,SAAS,EAAE;MACrB/H,KAAK,IAAK,cAAa8O,OAAO,CAAC/G,SAAU,EAAC;IAC5C;IAEA,MAAMmH,MAAM,GAAG;MACbC,MAAM,EAAEL,OAAO,CAACC;IAClB,CAAC;IAED,MAAM7D,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MAAEsT,QAAQ,EAAE,WAAW;MAAErT,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM,CAAC;MAAEC,QAAQ,EAAE;IAAK,CAAC,CAAC;IAC5G,MAAMsG,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAACmK,MAAM,CAAC;IAC3C,MAAMnP,OAA+B,GAAG,CAAC,CAAC;IAC1CA,OAAO,CAAC,aAAa,CAAC,GAAGtF,KAAK,CAAC8H,OAAO,CAAC;IAEvC,MAAM,IAAI,CAACK,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU,KAAK;MAAED;IAAQ,CAAC,EAAEwC,OAAO,CAAC;EAC9F;;EAEA;AACF;AACA;EACE,MAAM8M,gBAAgBA,CAAChQ,UAAkB,EAAkB;IACzD,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,wBAAuBrE,UAAW,EAAC,CAAC;IAC/E;IAEA,MAAMS,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,SAAS;IACvB,MAAM0K,cAAc,GAAG;MAAE5K,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC;IAEpD,MAAMsB,QAAQ,GAAG,MAAM,IAAI,CAACgB,gBAAgB,CAACoI,cAAc,CAAC;IAC5D,MAAM3H,IAAI,GAAG,MAAM7H,YAAY,CAACoG,QAAQ,CAAC;IACzC,OAAO1F,UAAU,CAAC0T,YAAY,CAACvM,IAAI,CAAC;EACtC;;EAEA;AACF;AACA;EACE,MAAMwM,gBAAgBA,CAAClQ,UAAkB,EAAEC,UAAkB,EAAEoG,OAAuB,EAAkB;IACtG,MAAM5F,MAAM,GAAG,KAAK;IACpB,IAAIE,KAAK,GAAG,SAAS;IAErB,IAAI,CAACnG,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGpE,UAAU,CAAC;IAC/E;IACA,IAAIoG,OAAO,IAAI,CAACjM,QAAQ,CAACiM,OAAO,CAAC,EAAE;MACjC,MAAM,IAAI/N,MAAM,CAACuF,oBAAoB,CAAC,oCAAoC,CAAC;IAC7E;IAEA,IAAIwI,OAAO,IAAIA,OAAO,CAACqC,SAAS,EAAE;MAChC/H,KAAK,GAAI,GAAEA,KAAM,cAAa0F,OAAO,CAACqC,SAAU,EAAC;IACnD;IACA,MAAM2C,cAA6B,GAAG;MAAE5K,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC;IACnE,IAAIV,UAAU,EAAE;MACdoL,cAAc,CAAC,YAAY,CAAC,GAAGpL,UAAU;IAC3C;IAEA,MAAMgC,QAAQ,GAAG,MAAM,IAAI,CAACgB,gBAAgB,CAACoI,cAAc,CAAC;IAC5D,MAAM3H,IAAI,GAAG,MAAM7H,YAAY,CAACoG,QAAQ,CAAC;IACzC,OAAO1F,UAAU,CAAC0T,YAAY,CAACvM,IAAI,CAAC;EACtC;;EAEA;AACF;AACA;EACE,MAAMyM,eAAeA,CAACnQ,UAAkB,EAAEoQ,MAAc,EAAiB;IACvE;IACA,IAAI,CAAC5V,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,wBAAuBrE,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAACzF,QAAQ,CAAC6V,MAAM,CAAC,EAAE;MACrB,MAAM,IAAI9X,MAAM,CAAC+X,wBAAwB,CAAE,0BAAyBD,MAAO,qBAAoB,CAAC;IAClG;IAEA,MAAMzP,KAAK,GAAG,QAAQ;IAEtB,IAAIF,MAAM,GAAG,QAAQ;IACrB,IAAI2P,MAAM,EAAE;MACV3P,MAAM,GAAG,KAAK;IAChB;IAEA,MAAM,IAAI,CAAC8C,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,EAAEyP,MAAM,EAAE,CAAC,GAAG,CAAC,EAAE,EAAE,CAAC;EACnF;;EAEA;AACF;AACA;EACE,MAAME,eAAeA,CAACtQ,UAAkB,EAAmB;IACzD;IACA,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,wBAAuBrE,UAAW,EAAC,CAAC;IAC/E;IAEA,MAAMS,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,QAAQ;IACtB,MAAM8C,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,CAAC;IACtE,OAAO,MAAM9E,YAAY,CAAC4H,GAAG,CAAC;EAChC;EAEA,MAAM8M,kBAAkBA,CAACvQ,UAAkB,EAAEC,UAAkB,EAAEuQ,aAAwB,GAAG,CAAC,CAAC,EAAiB;IAC7G,IAAI,CAAChW,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,wBAAuBrE,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAAC7F,QAAQ,CAACoW,aAAa,CAAC,EAAE;MAC5B,MAAM,IAAIlY,MAAM,CAACuF,oBAAoB,CAAC,0CAA0C,CAAC;IACnF,CAAC,MAAM;MACL,IAAI2S,aAAa,CAAC3H,gBAAgB,IAAI,CAAC7O,SAAS,CAACwW,aAAa,CAAC3H,gBAAgB,CAAC,EAAE;QAChF,MAAM,IAAIvQ,MAAM,CAACuF,oBAAoB,CAAE,uCAAsC2S,aAAa,CAAC3H,gBAAiB,EAAC,CAAC;MAChH;MACA,IACE2H,aAAa,CAACC,IAAI,IAClB,CAAC,CAAC7X,eAAe,CAAC8X,UAAU,EAAE9X,eAAe,CAAC+X,UAAU,CAAC,CAACzQ,QAAQ,CAACsQ,aAAa,CAACC,IAAI,CAAC,EACtF;QACA,MAAM,IAAInY,MAAM,CAACuF,oBAAoB,CAAE,kCAAiC2S,aAAa,CAACC,IAAK,EAAC,CAAC;MAC/F;MACA,IAAID,aAAa,CAACI,eAAe,IAAI,CAACrW,QAAQ,CAACiW,aAAa,CAACI,eAAe,CAAC,EAAE;QAC7E,MAAM,IAAItY,MAAM,CAACuF,oBAAoB,CAAE,sCAAqC2S,aAAa,CAACI,eAAgB,EAAC,CAAC;MAC9G;MACA,IAAIJ,aAAa,CAAC9H,SAAS,IAAI,CAACnO,QAAQ,CAACiW,aAAa,CAAC9H,SAAS,CAAC,EAAE;QACjE,MAAM,IAAIpQ,MAAM,CAACuF,oBAAoB,CAAE,gCAA+B2S,aAAa,CAAC9H,SAAU,EAAC,CAAC;MAClG;IACF;IAEA,MAAMjI,MAAM,GAAG,KAAK;IACpB,IAAIE,KAAK,GAAG,WAAW;IAEvB,MAAMD,OAAuB,GAAG,CAAC,CAAC;IAClC,IAAI8P,aAAa,CAAC3H,gBAAgB,EAAE;MAClCnI,OAAO,CAAC,mCAAmC,CAAC,GAAG,IAAI;IACrD;IAEA,MAAMmL,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MAAEsT,QAAQ,EAAE,WAAW;MAAErT,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM,CAAC;MAAEC,QAAQ,EAAE;IAAK,CAAC,CAAC;IAC5G,MAAMS,MAA8B,GAAG,CAAC,CAAC;IAEzC,IAAImT,aAAa,CAACC,IAAI,EAAE;MACtBpT,MAAM,CAACwT,IAAI,GAAGL,aAAa,CAACC,IAAI;IAClC;IACA,IAAID,aAAa,CAACI,eAAe,EAAE;MACjCvT,MAAM,CAACyT,eAAe,GAAGN,aAAa,CAACI,eAAe;IACxD;IACA,IAAIJ,aAAa,CAAC9H,SAAS,EAAE;MAC3B/H,KAAK,IAAK,cAAa6P,aAAa,CAAC9H,SAAU,EAAC;IAClD;IAEA,MAAMxF,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAACrI,MAAM,CAAC;IAE3CqD,OAAO,CAAC,aAAa,CAAC,GAAGtF,KAAK,CAAC8H,OAAO,CAAC;IACvC,MAAM,IAAI,CAACK,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU,KAAK;MAAED;IAAQ,CAAC,EAAEwC,OAAO,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;EAC1G;EAKA,MAAM6N,mBAAmBA,CAAC/Q,UAAkB,EAAE;IAC5C,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,aAAa;IAE3B,MAAM+L,OAAO,GAAG,MAAM,IAAI,CAACzJ,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,CAAC;IAC1E,MAAMgM,SAAS,GAAG,MAAM9Q,YAAY,CAAC6Q,OAAO,CAAC;IAC7C,OAAOnQ,UAAU,CAACyU,qBAAqB,CAACrE,SAAS,CAAC;EACpD;EAOA,MAAMsE,mBAAmBA,CAACjR,UAAkB,EAAEkR,cAAyD,EAAE;IACvG,MAAMC,cAAc,GAAG,CAACvY,eAAe,CAAC8X,UAAU,EAAE9X,eAAe,CAAC+X,UAAU,CAAC;IAC/E,MAAMS,UAAU,GAAG,CAACvY,wBAAwB,CAACwY,IAAI,EAAExY,wBAAwB,CAACyY,KAAK,CAAC;IAElF,IAAI,CAAC9W,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IAEA,IAAIkR,cAAc,CAACT,IAAI,IAAI,CAACU,cAAc,CAACjR,QAAQ,CAACgR,cAAc,CAACT,IAAI,CAAC,EAAE;MACxE,MAAM,IAAI5Q,SAAS,CAAE,wCAAuCsR,cAAe,EAAC,CAAC;IAC/E;IACA,IAAID,cAAc,CAACK,IAAI,IAAI,CAACH,UAAU,CAAClR,QAAQ,CAACgR,cAAc,CAACK,IAAI,CAAC,EAAE;MACpE,MAAM,IAAI1R,SAAS,CAAE,wCAAuCuR,UAAW,EAAC,CAAC;IAC3E;IACA,IAAIF,cAAc,CAACM,QAAQ,IAAI,CAACrX,QAAQ,CAAC+W,cAAc,CAACM,QAAQ,CAAC,EAAE;MACjE,MAAM,IAAI3R,SAAS,CAAE,4CAA2C,CAAC;IACnE;IAEA,MAAMY,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,aAAa;IAE3B,MAAMkP,MAA6B,GAAG;MACpC4B,iBAAiB,EAAE;IACrB,CAAC;IACD,MAAMC,UAAU,GAAGzQ,MAAM,CAACqO,IAAI,CAAC4B,cAAc,CAAC;IAE9C,MAAMS,YAAY,GAAG,CAAC,MAAM,EAAE,MAAM,EAAE,UAAU,CAAC,CAACC,KAAK,CAAEC,GAAG,IAAKH,UAAU,CAACxR,QAAQ,CAAC2R,GAAG,CAAC,CAAC;IAC1F;IACA,IAAIH,UAAU,CAACtO,MAAM,GAAG,CAAC,EAAE;MACzB,IAAI,CAACuO,YAAY,EAAE;QACjB,MAAM,IAAI9R,SAAS,CAChB,yGACH,CAAC;MACH,CAAC,MAAM;QACLgQ,MAAM,CAACX,IAAI,GAAG;UACZ4C,gBAAgB,EAAE,CAAC;QACrB,CAAC;QACD,IAAIZ,cAAc,CAACT,IAAI,EAAE;UACvBZ,MAAM,CAACX,IAAI,CAAC4C,gBAAgB,CAACjB,IAAI,GAAGK,cAAc,CAACT,IAAI;QACzD;QACA,IAAIS,cAAc,CAACK,IAAI,KAAK1Y,wBAAwB,CAACwY,IAAI,EAAE;UACzDxB,MAAM,CAACX,IAAI,CAAC4C,gBAAgB,CAACC,IAAI,GAAGb,cAAc,CAACM,QAAQ;QAC7D,CAAC,MAAM,IAAIN,cAAc,CAACK,IAAI,KAAK1Y,wBAAwB,CAACyY,KAAK,EAAE;UACjEzB,MAAM,CAACX,IAAI,CAAC4C,gBAAgB,CAACE,KAAK,GAAGd,cAAc,CAACM,QAAQ;QAC9D;MACF;IACF;IAEA,MAAM3F,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MACjCsT,QAAQ,EAAE,yBAAyB;MACnCrT,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM,CAAC;MAC7BC,QAAQ,EAAE;IACZ,CAAC,CAAC;IACF,MAAMsG,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAACmK,MAAM,CAAC;IAE3C,MAAMnP,OAAuB,GAAG,CAAC,CAAC;IAClCA,OAAO,CAAC,aAAa,CAAC,GAAGtF,KAAK,CAAC8H,OAAO,CAAC;IAEvC,MAAM,IAAI,CAACK,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW,KAAK;MAAED;IAAQ,CAAC,EAAEwC,OAAO,CAAC;EAClF;EAEA,MAAM+O,mBAAmBA,CAACjS,UAAkB,EAA0C;IACpF,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,YAAY;IAE1B,MAAM+L,OAAO,GAAG,MAAM,IAAI,CAACzJ,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,CAAC;IAC1E,MAAMgM,SAAS,GAAG,MAAM9Q,YAAY,CAAC6Q,OAAO,CAAC;IAC7C,OAAO,MAAMnQ,UAAU,CAAC2V,2BAA2B,CAACvF,SAAS,CAAC;EAChE;EAEA,MAAMwF,mBAAmBA,CAACnS,UAAkB,EAAEoS,aAA4C,EAAiB;IACzG,IAAI,CAAC5X,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACiB,MAAM,CAACqO,IAAI,CAAC8C,aAAa,CAAC,CAAChP,MAAM,EAAE;MACtC,MAAM,IAAI9K,MAAM,CAACuF,oBAAoB,CAAC,0CAA0C,CAAC;IACnF;IAEA,MAAM4C,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,YAAY;IAC1B,MAAMkL,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MACjCsT,QAAQ,EAAE,yBAAyB;MACnCrT,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM,CAAC;MAC7BC,QAAQ,EAAE;IACZ,CAAC,CAAC;IACF,MAAMsG,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAAC0M,aAAa,CAAC;IAElD,MAAM,IAAI,CAAC7O,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,EAAEuC,OAAO,CAAC;EACzE;EAEA,MAAcmP,UAAUA,CAACC,aAA+B,EAAiB;IACvE,MAAM;MAAEtS,UAAU;MAAEC,UAAU;MAAEsS,IAAI;MAAEC;IAAQ,CAAC,GAAGF,aAAa;IAC/D,MAAM7R,MAAM,GAAG,KAAK;IACpB,IAAIE,KAAK,GAAG,SAAS;IAErB,IAAI6R,OAAO,IAAIA,OAAO,aAAPA,OAAO,eAAPA,OAAO,CAAE9J,SAAS,EAAE;MACjC/H,KAAK,GAAI,GAAEA,KAAM,cAAa6R,OAAO,CAAC9J,SAAU,EAAC;IACnD;IACA,MAAM+J,QAAQ,GAAG,EAAE;IACnB,KAAK,MAAM,CAACtI,GAAG,EAAEuI,KAAK,CAAC,IAAIzR,MAAM,CAACC,OAAO,CAACqR,IAAI,CAAC,EAAE;MAC/CE,QAAQ,CAAC1L,IAAI,CAAC;QAAE4L,GAAG,EAAExI,GAAG;QAAEyI,KAAK,EAAEF;MAAM,CAAC,CAAC;IAC3C;IACA,MAAMG,aAAa,GAAG;MACpBC,OAAO,EAAE;QACPC,MAAM,EAAE;UACNC,GAAG,EAAEP;QACP;MACF;IACF,CAAC;IACD,MAAM/R,OAAO,GAAG,CAAC,CAAmB;IACpC,MAAMmL,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MAAEG,QAAQ,EAAE,IAAI;MAAEF,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM;IAAE,CAAC,CAAC;IACrF,MAAMsW,UAAU,GAAGtP,MAAM,CAAC4D,IAAI,CAACsE,OAAO,CAACnG,WAAW,CAACmN,aAAa,CAAC,CAAC;IAClE,MAAMxH,cAAc,GAAG;MACrB5K,MAAM;MACNT,UAAU;MACVW,KAAK;MACLD,OAAO;MAEP,IAAIT,UAAU,IAAI;QAAEA,UAAU,EAAEA;MAAW,CAAC;IAC9C,CAAC;IAEDS,OAAO,CAAC,aAAa,CAAC,GAAGtF,KAAK,CAAC6X,UAAU,CAAC;IAE1C,MAAM,IAAI,CAAC1P,oBAAoB,CAAC8H,cAAc,EAAE4H,UAAU,CAAC;EAC7D;EAEA,MAAcC,aAAaA,CAAC;IAAElT,UAAU;IAAEC,UAAU;IAAE2I;EAAgC,CAAC,EAAiB;IACtG,MAAMnI,MAAM,GAAG,QAAQ;IACvB,IAAIE,KAAK,GAAG,SAAS;IAErB,IAAIiI,UAAU,IAAI3H,MAAM,CAACqO,IAAI,CAAC1G,UAAU,CAAC,CAACxF,MAAM,IAAIwF,UAAU,CAACF,SAAS,EAAE;MACxE/H,KAAK,GAAI,GAAEA,KAAM,cAAaiI,UAAU,CAACF,SAAU,EAAC;IACtD;IACA,MAAM2C,cAAc,GAAG;MAAE5K,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU;IAAM,CAAC;IAEhE,IAAIV,UAAU,EAAE;MACdoL,cAAc,CAAC,YAAY,CAAC,GAAGpL,UAAU;IAC3C;IACA,MAAM,IAAI,CAACgD,gBAAgB,CAACoI,cAAc,EAAE,EAAE,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;EAC7D;EAEA,MAAM8H,gBAAgBA,CAACnT,UAAkB,EAAEuS,IAAU,EAAiB;IACpE,IAAI,CAAC/X,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAAC3F,aAAa,CAACkY,IAAI,CAAC,EAAE;MACxB,MAAM,IAAIja,MAAM,CAACuF,oBAAoB,CAAC,iCAAiC,CAAC;IAC1E;IACA,IAAIoD,MAAM,CAACqO,IAAI,CAACiD,IAAI,CAAC,CAACnP,MAAM,GAAG,EAAE,EAAE;MACjC,MAAM,IAAI9K,MAAM,CAACuF,oBAAoB,CAAC,6BAA6B,CAAC;IACtE;IAEA,MAAM,IAAI,CAACwU,UAAU,CAAC;MAAErS,UAAU;MAAEuS;IAAK,CAAC,CAAC;EAC7C;EAEA,MAAMa,mBAAmBA,CAACpT,UAAkB,EAAE;IAC5C,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAM,IAAI,CAACkT,aAAa,CAAC;MAAElT;IAAW,CAAC,CAAC;EAC1C;EAEA,MAAMqT,gBAAgBA,CAACrT,UAAkB,EAAEC,UAAkB,EAAEsS,IAAU,EAAEC,OAAqB,EAAE;IAChG,IAAI,CAAChY,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGpE,UAAU,CAAC;IAC/E;IAEA,IAAI,CAAC5F,aAAa,CAACkY,IAAI,CAAC,EAAE;MACxB,MAAM,IAAIja,MAAM,CAACuF,oBAAoB,CAAC,iCAAiC,CAAC;IAC1E;IACA,IAAIoD,MAAM,CAACqO,IAAI,CAACiD,IAAI,CAAC,CAACnP,MAAM,GAAG,EAAE,EAAE;MACjC,MAAM,IAAI9K,MAAM,CAACuF,oBAAoB,CAAC,6BAA6B,CAAC;IACtE;IAEA,MAAM,IAAI,CAACwU,UAAU,CAAC;MAAErS,UAAU;MAAEC,UAAU;MAAEsS,IAAI;MAAEC;IAAQ,CAAC,CAAC;EAClE;EAEA,MAAMc,mBAAmBA,CAACtT,UAAkB,EAAEC,UAAkB,EAAE2I,UAAuB,EAAE;IACzF,IAAI,CAACpO,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGpE,UAAU,CAAC;IAC/E;IACA,IAAI2I,UAAU,IAAI3H,MAAM,CAACqO,IAAI,CAAC1G,UAAU,CAAC,CAACxF,MAAM,IAAI,CAAChJ,QAAQ,CAACwO,UAAU,CAAC,EAAE;MACzE,MAAM,IAAItQ,MAAM,CAACuF,oBAAoB,CAAC,uCAAuC,CAAC;IAChF;IAEA,MAAM,IAAI,CAACqV,aAAa,CAAC;MAAElT,UAAU;MAAEC,UAAU;MAAE2I;IAAW,CAAC,CAAC;EAClE;EAEA,MAAM2K,mBAAmBA,CACvBvT,UAAkB,EAClBC,UAAkB,EAClBuT,UAAyB,EACW;IACpC,IAAI,CAAChZ,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,wBAAuBrE,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAAC/H,CAAC,CAACgC,OAAO,CAACsZ,UAAU,CAAC,EAAE;MAC1B,IAAI,CAACjZ,QAAQ,CAACiZ,UAAU,CAACC,UAAU,CAAC,EAAE;QACpC,MAAM,IAAI5T,SAAS,CAAC,0CAA0C,CAAC;MACjE;MACA,IAAI,CAAC3H,CAAC,CAACgC,OAAO,CAACsZ,UAAU,CAACE,kBAAkB,CAAC,EAAE;QAC7C,IAAI,CAACtZ,QAAQ,CAACoZ,UAAU,CAACE,kBAAkB,CAAC,EAAE;UAC5C,MAAM,IAAI7T,SAAS,CAAC,+CAA+C,CAAC;QACtE;MACF,CAAC,MAAM;QACL,MAAM,IAAIA,SAAS,CAAC,gCAAgC,CAAC;MACvD;MACA,IAAI,CAAC3H,CAAC,CAACgC,OAAO,CAACsZ,UAAU,CAACG,mBAAmB,CAAC,EAAE;QAC9C,IAAI,CAACvZ,QAAQ,CAACoZ,UAAU,CAACG,mBAAmB,CAAC,EAAE;UAC7C,MAAM,IAAI9T,SAAS,CAAC,gDAAgD,CAAC;QACvE;MACF,CAAC,MAAM;QACL,MAAM,IAAIA,SAAS,CAAC,iCAAiC,CAAC;MACxD;IACF,CAAC,MAAM;MACL,MAAM,IAAIA,SAAS,CAAC,wCAAwC,CAAC;IAC/D;IAEA,MAAMY,MAAM,GAAG,MAAM;IACrB,MAAME,KAAK,GAAI,sBAAqB;IAEpC,MAAMkP,MAAiC,GAAG,CACxC;MACE+D,UAAU,EAAEJ,UAAU,CAACC;IACzB,CAAC,EACD;MACEI,cAAc,EAAEL,UAAU,CAACM,cAAc,IAAI;IAC/C,CAAC,EACD;MACEC,kBAAkB,EAAE,CAACP,UAAU,CAACE,kBAAkB;IACpD,CAAC,EACD;MACEM,mBAAmB,EAAE,CAACR,UAAU,CAACG,mBAAmB;IACtD,CAAC,CACF;;IAED;IACA,IAAIH,UAAU,CAACS,eAAe,EAAE;MAC9BpE,MAAM,CAAC9I,IAAI,CAAC;QAAEmN,eAAe,EAAEV,UAAU,aAAVA,UAAU,uBAAVA,UAAU,CAAES;MAAgB,CAAC,CAAC;IAC/D;IACA;IACA,IAAIT,UAAU,CAACW,SAAS,EAAE;MACxBtE,MAAM,CAAC9I,IAAI,CAAC;QAAEqN,SAAS,EAAEZ,UAAU,CAACW;MAAU,CAAC,CAAC;IAClD;IAEA,MAAMtI,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MACjCsT,QAAQ,EAAE,4BAA4B;MACtCrT,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM,CAAC;MAC7BC,QAAQ,EAAE;IACZ,CAAC,CAAC;IACF,MAAMsG,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAACmK,MAAM,CAAC;IAE3C,MAAMpM,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU;IAAM,CAAC,EAAEuC,OAAO,CAAC;IAC3F,MAAMQ,IAAI,GAAG,MAAM9H,YAAY,CAAC6H,GAAG,CAAC;IACpC,OAAOpH,gCAAgC,CAACqH,IAAI,CAAC;EAC/C;EAEA,MAAc2Q,oBAAoBA,CAACrU,UAAkB,EAAEsU,YAAkC,EAAiB;IACxG,MAAM7T,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,WAAW;IAEzB,MAAMD,OAAuB,GAAG,CAAC,CAAC;IAClC,MAAMmL,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MACjCsT,QAAQ,EAAE,wBAAwB;MAClCnT,QAAQ,EAAE,IAAI;MACdF,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM;IAC9B,CAAC,CAAC;IACF,MAAMuG,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAAC4O,YAAY,CAAC;IACjD5T,OAAO,CAAC,aAAa,CAAC,GAAGtF,KAAK,CAAC8H,OAAO,CAAC;IAEvC,MAAM,IAAI,CAACK,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW,KAAK;MAAED;IAAQ,CAAC,EAAEwC,OAAO,CAAC;EAClF;EAEA,MAAMqR,qBAAqBA,CAACvU,UAAkB,EAAiB;IAC7D,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,QAAQ;IACvB,MAAME,KAAK,GAAG,WAAW;IACzB,MAAM,IAAI,CAAC4C,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC;EAC3E;EAEA,MAAM6T,kBAAkBA,CAACxU,UAAkB,EAAEyU,eAAqC,EAAiB;IACjG,IAAI,CAACja,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI9H,CAAC,CAACgC,OAAO,CAACua,eAAe,CAAC,EAAE;MAC9B,MAAM,IAAI,CAACF,qBAAqB,CAACvU,UAAU,CAAC;IAC9C,CAAC,MAAM;MACL,MAAM,IAAI,CAACqU,oBAAoB,CAACrU,UAAU,EAAEyU,eAAe,CAAC;IAC9D;EACF;EAEA,MAAMC,kBAAkBA,CAAC1U,UAAkB,EAAmC;IAC5E,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,WAAW;IAEzB,MAAM8C,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,CAAC;IACtE,MAAM+C,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,OAAOlH,UAAU,CAACoY,oBAAoB,CAACjR,IAAI,CAAC;EAC9C;EAEA,MAAMkR,mBAAmBA,CAAC5U,UAAkB,EAAE6U,gBAAmC,EAAiB;IAChG,IAAI,CAACra,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAAC9H,CAAC,CAACgC,OAAO,CAAC2a,gBAAgB,CAAC,IAAIA,gBAAgB,CAAC3F,IAAI,CAAC9L,MAAM,GAAG,CAAC,EAAE;MACpE,MAAM,IAAI9K,MAAM,CAACuF,oBAAoB,CAAC,kDAAkD,GAAGgX,gBAAgB,CAAC3F,IAAI,CAAC;IACnH;IAEA,IAAI4F,aAAa,GAAGD,gBAAgB;IACpC,IAAI3c,CAAC,CAACgC,OAAO,CAAC2a,gBAAgB,CAAC,EAAE;MAC/BC,aAAa,GAAG;QACd;QACA5F,IAAI,EAAE,CACJ;UACE6F,kCAAkC,EAAE;YAClCC,YAAY,EAAE;UAChB;QACF,CAAC;MAEL,CAAC;IACH;IAEA,MAAMvU,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,YAAY;IAC1B,MAAMkL,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MACjCsT,QAAQ,EAAE,mCAAmC;MAC7CrT,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM,CAAC;MAC7BC,QAAQ,EAAE;IACZ,CAAC,CAAC;IACF,MAAMsG,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAACoP,aAAa,CAAC;IAElD,MAAMpU,OAAuB,GAAG,CAAC,CAAC;IAClCA,OAAO,CAAC,aAAa,CAAC,GAAGtF,KAAK,CAAC8H,OAAO,CAAC;IAEvC,MAAM,IAAI,CAACK,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW,KAAK;MAAED;IAAQ,CAAC,EAAEwC,OAAO,CAAC;EAClF;EAEA,MAAM+R,mBAAmBA,CAACjV,UAAkB,EAAE;IAC5C,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,YAAY;IAE1B,MAAM8C,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,CAAC;IACtE,MAAM+C,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,OAAOlH,UAAU,CAAC2Y,2BAA2B,CAACxR,IAAI,CAAC;EACrD;EAEA,MAAMyR,sBAAsBA,CAACnV,UAAkB,EAAE;IAC/C,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,QAAQ;IACvB,MAAME,KAAK,GAAG,YAAY;IAE1B,MAAM,IAAI,CAAC4C,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC;EAC3E;EAEA,MAAMyU,kBAAkBA,CACtBpV,UAAkB,EAClBC,UAAkB,EAClBoG,OAAgC,EACiB;IACjD,IAAI,CAAC7L,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,IAAIoG,OAAO,IAAI,CAACjM,QAAQ,CAACiM,OAAO,CAAC,EAAE;MACjC,MAAM,IAAI/N,MAAM,CAACuF,oBAAoB,CAAC,oCAAoC,CAAC;IAC7E,CAAC,MAAM,IAAIwI,OAAO,aAAPA,OAAO,eAAPA,OAAO,CAAEqC,SAAS,IAAI,CAACnO,QAAQ,CAAC8L,OAAO,CAACqC,SAAS,CAAC,EAAE;MAC7D,MAAM,IAAIpQ,MAAM,CAACuF,oBAAoB,CAAC,sCAAsC,CAAC;IAC/E;IAEA,MAAM4C,MAAM,GAAG,KAAK;IACpB,IAAIE,KAAK,GAAG,WAAW;IACvB,IAAI0F,OAAO,aAAPA,OAAO,eAAPA,OAAO,CAAEqC,SAAS,EAAE;MACtB/H,KAAK,IAAK,cAAa0F,OAAO,CAACqC,SAAU,EAAC;IAC5C;IACA,MAAMjF,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU;IAAM,CAAC,CAAC;IAClF,MAAM+C,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,OAAOlH,UAAU,CAAC8Y,0BAA0B,CAAC3R,IAAI,CAAC;EACpD;EAEA,MAAM4R,aAAaA,CAACtV,UAAkB,EAAEuV,WAA+B,EAAoC;IACzG,IAAI,CAAC/a,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACwV,KAAK,CAACC,OAAO,CAACF,WAAW,CAAC,EAAE;MAC/B,MAAM,IAAIjd,MAAM,CAACuF,oBAAoB,CAAC,8BAA8B,CAAC;IACvE;IAEA,MAAM6X,gBAAgB,GAAG,MAAOC,KAAyB,IAAuC;MAC9F,MAAMC,UAAuC,GAAGD,KAAK,CAAC3J,GAAG,CAAE0G,KAAK,IAAK;QACnE,OAAOtY,QAAQ,CAACsY,KAAK,CAAC,GAAG;UAAEC,GAAG,EAAED,KAAK,CAAC7N,IAAI;UAAEgR,SAAS,EAAEnD,KAAK,CAAChK;QAAU,CAAC,GAAG;UAAEiK,GAAG,EAAED;QAAM,CAAC;MAC3F,CAAC,CAAC;MAEF,MAAMoD,UAAU,GAAG;QAAEC,MAAM,EAAE;UAAEC,KAAK,EAAE,IAAI;UAAE/U,MAAM,EAAE2U;QAAW;MAAE,CAAC;MAClE,MAAM1S,OAAO,GAAGS,MAAM,CAAC4D,IAAI,CAAC,IAAInP,MAAM,CAACqE,OAAO,CAAC;QAAEG,QAAQ,EAAE;MAAK,CAAC,CAAC,CAAC8I,WAAW,CAACoQ,UAAU,CAAC,CAAC;MAC3F,MAAMpV,OAAuB,GAAG;QAAE,aAAa,EAAEtF,KAAK,CAAC8H,OAAO;MAAE,CAAC;MAEjE,MAAMO,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;QAAExC,MAAM,EAAE,MAAM;QAAET,UAAU;QAAEW,KAAK,EAAE,QAAQ;QAAED;MAAQ,CAAC,EAAEwC,OAAO,CAAC;MAC1G,MAAMQ,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;MACpC,OAAOlH,UAAU,CAAC0Z,mBAAmB,CAACvS,IAAI,CAAC;IAC7C,CAAC;IAED,MAAMwS,UAAU,GAAG,IAAI,EAAC;IACxB;IACA,MAAMC,OAAO,GAAG,EAAE;IAClB,KAAK,IAAIC,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAGb,WAAW,CAACnS,MAAM,EAAEgT,CAAC,IAAIF,UAAU,EAAE;MACvDC,OAAO,CAACpP,IAAI,CAACwO,WAAW,CAACc,KAAK,CAACD,CAAC,EAAEA,CAAC,GAAGF,UAAU,CAAC,CAAC;IACpD;IAEA,MAAMI,YAAY,GAAG,MAAMzI,OAAO,CAACC,GAAG,CAACqI,OAAO,CAACnK,GAAG,CAAC0J,gBAAgB,CAAC,CAAC;IACrE,OAAOY,YAAY,CAACC,IAAI,CAAC,CAAC;EAC5B;EAEA,MAAMC,sBAAsBA,CAACxW,UAAkB,EAAEC,UAAkB,EAAiB;IAClF,IAAI,CAACzF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAACme,sBAAsB,CAAC,uBAAuB,GAAGzW,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IACA,MAAMyW,cAAc,GAAG,MAAM,IAAI,CAACpL,YAAY,CAACtL,UAAU,EAAEC,UAAU,CAAC;IACtE,MAAMQ,MAAM,GAAG,QAAQ;IACvB,MAAME,KAAK,GAAI,YAAW+V,cAAe,EAAC;IAC1C,MAAM,IAAI,CAACnT,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAEU;IAAM,CAAC,EAAE,EAAE,EAAE,CAAC,GAAG,CAAC,CAAC;EACvF;EAEA,MAAcgW,YAAYA,CACxBC,gBAAwB,EACxBC,gBAAwB,EACxBC,6BAAqC,EACrCC,UAAkC,EAClC;IACA,IAAI,OAAOA,UAAU,IAAI,UAAU,EAAE;MACnCA,UAAU,GAAG,IAAI;IACnB;IAEA,IAAI,CAACvc,iBAAiB,CAACoc,gBAAgB,CAAC,EAAE;MACxC,MAAM,IAAIte,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGuS,gBAAgB,CAAC;IACrF;IACA,IAAI,CAAClc,iBAAiB,CAACmc,gBAAgB,CAAC,EAAE;MACxC,MAAM,IAAIve,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBuQ,gBAAiB,EAAC,CAAC;IACrF;IACA,IAAI,CAACtc,QAAQ,CAACuc,6BAA6B,CAAC,EAAE;MAC5C,MAAM,IAAIjX,SAAS,CAAC,0DAA0D,CAAC;IACjF;IACA,IAAIiX,6BAA6B,KAAK,EAAE,EAAE;MACxC,MAAM,IAAIxe,MAAM,CAAC6Q,kBAAkB,CAAE,qBAAoB,CAAC;IAC5D;IAEA,IAAI4N,UAAU,IAAI,IAAI,IAAI,EAAEA,UAAU,YAAY1d,cAAc,CAAC,EAAE;MACjE,MAAM,IAAIwG,SAAS,CAAC,+CAA+C,CAAC;IACtE;IAEA,MAAMa,OAAuB,GAAG,CAAC,CAAC;IAClCA,OAAO,CAAC,mBAAmB,CAAC,GAAGnF,iBAAiB,CAACub,6BAA6B,CAAC;IAE/E,IAAIC,UAAU,EAAE;MACd,IAAIA,UAAU,CAACC,QAAQ,KAAK,EAAE,EAAE;QAC9BtW,OAAO,CAAC,qCAAqC,CAAC,GAAGqW,UAAU,CAACC,QAAQ;MACtE;MACA,IAAID,UAAU,CAACE,UAAU,KAAK,EAAE,EAAE;QAChCvW,OAAO,CAAC,uCAAuC,CAAC,GAAGqW,UAAU,CAACE,UAAU;MAC1E;MACA,IAAIF,UAAU,CAACG,SAAS,KAAK,EAAE,EAAE;QAC/BxW,OAAO,CAAC,4BAA4B,CAAC,GAAGqW,UAAU,CAACG,SAAS;MAC9D;MACA,IAAIH,UAAU,CAACI,eAAe,KAAK,EAAE,EAAE;QACrCzW,OAAO,CAAC,iCAAiC,CAAC,GAAGqW,UAAU,CAACI,eAAe;MACzE;IACF;IAEA,MAAM1W,MAAM,GAAG,KAAK;IAEpB,MAAMgD,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MACtCxC,MAAM;MACNT,UAAU,EAAE4W,gBAAgB;MAC5B3W,UAAU,EAAE4W,gBAAgB;MAC5BnW;IACF,CAAC,CAAC;IACF,MAAMgD,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,OAAOlH,UAAU,CAAC6a,eAAe,CAAC1T,IAAI,CAAC;EACzC;EAEA,MAAc2T,YAAYA,CACxBC,YAA+B,EAC/BC,UAAkC,EACL;IAC7B,IAAI,EAAED,YAAY,YAAY9e,iBAAiB,CAAC,EAAE;MAChD,MAAM,IAAIF,MAAM,CAACuF,oBAAoB,CAAC,gDAAgD,CAAC;IACzF;IACA,IAAI,EAAE0Z,UAAU,YAAYhf,sBAAsB,CAAC,EAAE;MACnD,MAAM,IAAID,MAAM,CAACuF,oBAAoB,CAAC,mDAAmD,CAAC;IAC5F;IACA,IAAI,CAAC0Z,UAAU,CAACC,QAAQ,CAAC,CAAC,EAAE;MAC1B,OAAO3J,OAAO,CAACG,MAAM,CAAC,CAAC;IACzB;IACA,IAAI,CAACuJ,UAAU,CAACC,QAAQ,CAAC,CAAC,EAAE;MAC1B,OAAO3J,OAAO,CAACG,MAAM,CAAC,CAAC;IACzB;IAEA,MAAMtN,OAAO,GAAGO,MAAM,CAACE,MAAM,CAAC,CAAC,CAAC,EAAEmW,YAAY,CAACG,UAAU,CAAC,CAAC,EAAEF,UAAU,CAACE,UAAU,CAAC,CAAC,CAAC;IAErF,MAAMzX,UAAU,GAAGuX,UAAU,CAACG,MAAM;IACpC,MAAMzX,UAAU,GAAGsX,UAAU,CAACtW,MAAM;IAEpC,MAAMR,MAAM,GAAG,KAAK;IAEpB,MAAMgD,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEC,UAAU;MAAES;IAAQ,CAAC,CAAC;IACpF,MAAMgD,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,MAAMkU,OAAO,GAAGpb,UAAU,CAAC6a,eAAe,CAAC1T,IAAI,CAAC;IAChD,MAAMkU,UAA+B,GAAGnU,GAAG,CAAC/C,OAAO;IAEnD,MAAMmX,eAAe,GAAGD,UAAU,IAAIA,UAAU,CAAC,gBAAgB,CAAC;IAClE,MAAM7P,IAAI,GAAG,OAAO8P,eAAe,KAAK,QAAQ,GAAGA,eAAe,GAAGta,SAAS;IAE9E,OAAO;MACLma,MAAM,EAAEH,UAAU,CAACG,MAAM;MACzB/E,GAAG,EAAE4E,UAAU,CAACtW,MAAM;MACtB6W,YAAY,EAAEH,OAAO,CAAClP,YAAY;MAClCsP,QAAQ,EAAEve,eAAe,CAACoe,UAA4B,CAAC;MACvD/B,SAAS,EAAEjc,YAAY,CAACge,UAA4B,CAAC;MACrDI,eAAe,EAAEre,kBAAkB,CAACie,UAA4B,CAAC;MACjEK,IAAI,EAAE9c,YAAY,CAACyc,UAAU,CAACpQ,IAAI,CAAC;MACnC0Q,IAAI,EAAEnQ;IACR,CAAC;EACH;EASA,MAAMoQ,UAAUA,CAAC,GAAGC,OAAyB,EAA6B;IACxE,IAAI,OAAOA,OAAO,CAAC,CAAC,CAAC,KAAK,QAAQ,EAAE;MAClC,MAAM,CAACxB,gBAAgB,EAAEC,gBAAgB,EAAEC,6BAA6B,EAAEC,UAAU,CAAC,GAAGqB,OAKvF;MACD,OAAO,MAAM,IAAI,CAACzB,YAAY,CAACC,gBAAgB,EAAEC,gBAAgB,EAAEC,6BAA6B,EAAEC,UAAU,CAAC;IAC/G;IACA,MAAM,CAACsB,MAAM,EAAEC,IAAI,CAAC,GAAGF,OAAsD;IAC7E,OAAO,MAAM,IAAI,CAACf,YAAY,CAACgB,MAAM,EAAEC,IAAI,CAAC;EAC9C;EAEA,MAAMC,UAAUA,CACdC,UAMC,EACDtV,OAAgB,EAChB;IACA,MAAM;MAAElD,UAAU;MAAEC,UAAU;MAAEwY,QAAQ;MAAEtK,UAAU;MAAEzN;IAAQ,CAAC,GAAG8X,UAAU;IAE5E,MAAM/X,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAI,YAAW8X,QAAS,eAActK,UAAW,EAAC;IAC7D,MAAM9C,cAAc,GAAG;MAAE5K,MAAM;MAAET,UAAU;MAAEC,UAAU,EAAEA,UAAU;MAAEU,KAAK;MAAED;IAAQ,CAAC;IACrF,MAAM+C,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAACoI,cAAc,EAAEnI,OAAO,CAAC;IAChE,MAAMQ,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,MAAMiV,OAAO,GAAGpc,gBAAgB,CAACoH,IAAI,CAAC;IACtC,MAAMiV,WAAW,GAAGxd,YAAY,CAACsI,GAAG,CAAC/C,OAAO,CAAC8G,IAAI,CAAC,IAAIrM,YAAY,CAACud,OAAO,CAACvM,IAAI,CAAC;IAChF,OAAO;MACL3E,IAAI,EAAEmR,WAAW;MACjBxO,GAAG,EAAElK,UAAU;MACfiM,IAAI,EAAEiC;IACR,CAAC;EACH;EAEA,MAAMyK,aAAaA,CACjBC,aAAqC,EACrCC,aAAkC,EAClC;IAAEC,cAAc,GAAG;EAAG,CAAC,GAAG,CAAC,CAAC,EACsE;IAClG,MAAMC,iBAAiB,GAAGF,aAAa,CAAC1V,MAAM;IAE9C,IAAI,CAACoS,KAAK,CAACC,OAAO,CAACqD,aAAa,CAAC,EAAE;MACjC,MAAM,IAAIxgB,MAAM,CAACuF,oBAAoB,CAAC,oDAAoD,CAAC;IAC7F;IACA,IAAI,EAAEgb,aAAa,YAAYtgB,sBAAsB,CAAC,EAAE;MACtD,MAAM,IAAID,MAAM,CAACuF,oBAAoB,CAAC,mDAAmD,CAAC;IAC5F;IAEA,IAAImb,iBAAiB,GAAG,CAAC,IAAIA,iBAAiB,GAAGje,gBAAgB,CAACke,eAAe,EAAE;MACjF,MAAM,IAAI3gB,MAAM,CAACuF,oBAAoB,CAClC,yCAAwC9C,gBAAgB,CAACke,eAAgB,kBAC5E,CAAC;IACH;IAEA,KAAK,IAAI7C,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAG4C,iBAAiB,EAAE5C,CAAC,EAAE,EAAE;MAC1C,MAAM8C,IAAI,GAAGJ,aAAa,CAAC1C,CAAC,CAAsB;MAClD,IAAI,CAAC8C,IAAI,CAAC1B,QAAQ,CAAC,CAAC,EAAE;QACpB,OAAO,KAAK;MACd;IACF;IAEA,IAAI,CAAEqB,aAAa,CAA4BrB,QAAQ,CAAC,CAAC,EAAE;MACzD,OAAO,KAAK;IACd;IAEA,MAAM2B,cAAc,GAAIC,SAA4B,IAAK;MACvD,IAAI/Q,QAAQ,GAAG,CAAC,CAAC;MACjB,IAAI,CAACnQ,CAAC,CAACgC,OAAO,CAACkf,SAAS,CAACC,SAAS,CAAC,EAAE;QACnChR,QAAQ,GAAG;UACTK,SAAS,EAAE0Q,SAAS,CAACC;QACvB,CAAC;MACH;MACA,OAAOhR,QAAQ;IACjB,CAAC;IACD,MAAMiR,cAAwB,GAAG,EAAE;IACnC,IAAIC,SAAS,GAAG,CAAC;IACjB,IAAIC,UAAU,GAAG,CAAC;IAElB,MAAMC,cAAc,GAAGX,aAAa,CAAC9M,GAAG,CAAE0N,OAAO,IAC/C,IAAI,CAACrS,UAAU,CAACqS,OAAO,CAAChC,MAAM,EAAEgC,OAAO,CAACzY,MAAM,EAAEkY,cAAc,CAACO,OAAO,CAAC,CACzE,CAAC;IAED,MAAMC,cAAc,GAAG,MAAM9L,OAAO,CAACC,GAAG,CAAC2L,cAAc,CAAC;IAExD,MAAMG,cAAc,GAAGD,cAAc,CAAC3N,GAAG,CAAC,CAAC6N,WAAW,EAAEC,KAAK,KAAK;MAChE,MAAMV,SAAwC,GAAGN,aAAa,CAACgB,KAAK,CAAC;MAErE,IAAIC,WAAW,GAAGF,WAAW,CAAC9R,IAAI;MAClC;MACA;MACA,IAAIqR,SAAS,IAAIA,SAAS,CAACY,UAAU,EAAE;QACrC;QACA;QACA;QACA,MAAMC,QAAQ,GAAGb,SAAS,CAACc,KAAK;QAChC,MAAMC,MAAM,GAAGf,SAAS,CAACgB,GAAG;QAC5B,IAAID,MAAM,IAAIJ,WAAW,IAAIE,QAAQ,GAAG,CAAC,EAAE;UACzC,MAAM,IAAI3hB,MAAM,CAACuF,oBAAoB,CAClC,kBAAiBic,KAAM,iCAAgCG,QAAS,KAAIE,MAAO,cAAaJ,WAAY,GACvG,CAAC;QACH;QACAA,WAAW,GAAGI,MAAM,GAAGF,QAAQ,GAAG,CAAC;MACrC;;MAEA;MACA,IAAIF,WAAW,GAAGhf,gBAAgB,CAACsf,iBAAiB,IAAIP,KAAK,GAAGd,iBAAiB,GAAG,CAAC,EAAE;QACrF,MAAM,IAAI1gB,MAAM,CAACuF,oBAAoB,CAClC,kBAAiBic,KAAM,kBAAiBC,WAAY,gCACvD,CAAC;MACH;;MAEA;MACAR,SAAS,IAAIQ,WAAW;MACxB,IAAIR,SAAS,GAAGxe,gBAAgB,CAACuf,6BAA6B,EAAE;QAC9D,MAAM,IAAIhiB,MAAM,CAACuF,oBAAoB,CAAE,oCAAmC0b,SAAU,WAAU,CAAC;MACjG;;MAEA;MACAD,cAAc,CAACQ,KAAK,CAAC,GAAGC,WAAW;;MAEnC;MACAP,UAAU,IAAIxe,aAAa,CAAC+e,WAAW,CAAC;MACxC;MACA,IAAIP,UAAU,GAAGze,gBAAgB,CAACke,eAAe,EAAE;QACjD,MAAM,IAAI3gB,MAAM,CAACuF,oBAAoB,CAClC,mDAAkD9C,gBAAgB,CAACke,eAAgB,QACtF,CAAC;MACH;MAEA,OAAOY,WAAW;IACpB,CAAC,CAAC;IAEF,IAAKL,UAAU,KAAK,CAAC,IAAID,SAAS,IAAIxe,gBAAgB,CAACwf,aAAa,IAAKhB,SAAS,KAAK,CAAC,EAAE;MACxF,OAAO,MAAM,IAAI,CAACpB,UAAU,CAACW,aAAa,CAAC,CAAC,CAAC,EAAuBD,aAAa,CAAC,EAAC;IACrF;;IAEA;IACA,KAAK,IAAIzC,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAG4C,iBAAiB,EAAE5C,CAAC,EAAE,EAAE;MAC1C;MAAE0C,aAAa,CAAC1C,CAAC,CAAC,CAAuBoE,SAAS,GAAIZ,cAAc,CAACxD,CAAC,CAAC,CAAoB5O,IAAI;IACjG;IAEA,MAAMiT,iBAAiB,GAAGb,cAAc,CAAC5N,GAAG,CAAC,CAAC6N,WAAW,EAAEa,GAAG,KAAK;MACjE,OAAOnhB,mBAAmB,CAAC+f,cAAc,CAACoB,GAAG,CAAC,EAAY5B,aAAa,CAAC4B,GAAG,CAAsB,CAAC;IACpG,CAAC,CAAC;IAEF,MAAMC,uBAAuB,GAAIvQ,QAAgB,IAAK;MACpD,MAAMwQ,oBAAwC,GAAG,EAAE;MAEnDH,iBAAiB,CAACpY,OAAO,CAAC,CAACwY,SAAS,EAAEC,UAAkB,KAAK;QAC3D,IAAID,SAAS,EAAE;UACb,MAAM;YAAEE,UAAU,EAAEC,QAAQ;YAAEC,QAAQ,EAAEC,MAAM;YAAEC,OAAO,EAAEC;UAAU,CAAC,GAAGP,SAAS;UAEhF,MAAMQ,SAAS,GAAGP,UAAU,GAAG,CAAC,EAAC;UACjC,MAAMQ,YAAY,GAAG9F,KAAK,CAACjO,IAAI,CAACyT,QAAQ,CAAC;UAEzC,MAAMta,OAAO,GAAIoY,aAAa,CAACgC,UAAU,CAAC,CAAuBrD,UAAU,CAAC,CAAC;UAE7E6D,YAAY,CAACjZ,OAAO,CAAC,CAACkZ,UAAU,EAAEC,UAAU,KAAK;YAC/C,MAAMC,QAAQ,GAAGP,MAAM,CAACM,UAAU,CAAC;YAEnC,MAAME,SAAS,GAAI,GAAEN,SAAS,CAAC1D,MAAO,IAAG0D,SAAS,CAACna,MAAO,EAAC;YAC3DP,OAAO,CAAC,mBAAmB,CAAC,GAAI,GAAEgb,SAAU,EAAC;YAC7Chb,OAAO,CAAC,yBAAyB,CAAC,GAAI,SAAQ6a,UAAW,IAAGE,QAAS,EAAC;YAEtE,MAAME,gBAAgB,GAAG;cACvB3b,UAAU,EAAE6Y,aAAa,CAACnB,MAAM;cAChCzX,UAAU,EAAE4Y,aAAa,CAAC5X,MAAM;cAChCwX,QAAQ,EAAErO,QAAQ;cAClB+D,UAAU,EAAEkN,SAAS;cACrB3a,OAAO,EAAEA,OAAO;cAChBgb,SAAS,EAAEA;YACb,CAAC;YAEDd,oBAAoB,CAAC7T,IAAI,CAAC4U,gBAAgB,CAAC;UAC7C,CAAC,CAAC;QACJ;MACF,CAAC,CAAC;MAEF,OAAOf,oBAAoB;IAC7B,CAAC;IAED,MAAMgB,cAAc,GAAG,MAAOC,UAA8B,IAAK;MAC/D,MAAMC,WAA0D,GAAG,EAAE;;MAErE;MACA,KAAK,MAAMnG,KAAK,IAAIzd,CAAC,CAACkW,KAAK,CAACyN,UAAU,EAAE9C,cAAc,CAAC,EAAE;QACvD,MAAMzC,YAAY,GAAG,MAAMzI,OAAO,CAACC,GAAG,CAAC6H,KAAK,CAAC3J,GAAG,CAAExB,IAAI,IAAK,IAAI,CAAC+N,UAAU,CAAC/N,IAAI,CAAC,CAAC,CAAC;QAElFsR,WAAW,CAAC/U,IAAI,CAAC,GAAGuP,YAAY,CAAC;MACnC;;MAEA;MACA,OAAOwF,WAAW;IACpB,CAAC;IAED,MAAMC,kBAAkB,GAAG,MAAO3R,QAAgB,IAAK;MACrD,MAAMyR,UAAU,GAAGlB,uBAAuB,CAACvQ,QAAQ,CAAC;MACpD,MAAM4R,QAAQ,GAAG,MAAMJ,cAAc,CAACC,UAAU,CAAC;MACjD,OAAOG,QAAQ,CAAChQ,GAAG,CAAEiQ,QAAQ,KAAM;QAAEzU,IAAI,EAAEyU,QAAQ,CAACzU,IAAI;QAAE0E,IAAI,EAAE+P,QAAQ,CAAC/P;MAAK,CAAC,CAAC,CAAC;IACnF,CAAC;IAED,MAAMgQ,gBAAgB,GAAGrD,aAAa,CAACpB,UAAU,CAAC,CAAC;IAEnD,MAAMrN,QAAQ,GAAG,MAAM,IAAI,CAACe,0BAA0B,CAAC0N,aAAa,CAACnB,MAAM,EAAEmB,aAAa,CAAC5X,MAAM,EAAEib,gBAAgB,CAAC;IACpH,IAAI;MACF,MAAMC,SAAS,GAAG,MAAMJ,kBAAkB,CAAC3R,QAAQ,CAAC;MACpD,OAAO,MAAM,IAAI,CAACuB,uBAAuB,CAACkN,aAAa,CAACnB,MAAM,EAAEmB,aAAa,CAAC5X,MAAM,EAAEmJ,QAAQ,EAAE+R,SAAS,CAAC;IAC5G,CAAC,CAAC,OAAOja,GAAG,EAAE;MACZ,OAAO,MAAM,IAAI,CAACkJ,oBAAoB,CAACyN,aAAa,CAACnB,MAAM,EAAEmB,aAAa,CAAC5X,MAAM,EAAEmJ,QAAQ,CAAC;IAC9F;EACF;EAEA,MAAMgS,YAAYA,CAChB3b,MAAc,EACdT,UAAkB,EAClBC,UAAkB,EAClBoc,OAAmD,EACnDC,SAAuC,EACvCC,WAAkB,EACD;IAAA,IAAAC,YAAA;IACjB,IAAI,IAAI,CAACzd,SAAS,EAAE;MAClB,MAAM,IAAIzG,MAAM,CAACmkB,qBAAqB,CAAE,aAAYhc,MAAO,iDAAgD,CAAC;IAC9G;IAEA,IAAI,CAAC4b,OAAO,EAAE;MACZA,OAAO,GAAG1jB,uBAAuB;IACnC;IACA,IAAI,CAAC2jB,SAAS,EAAE;MACdA,SAAS,GAAG,CAAC,CAAC;IAChB;IACA,IAAI,CAACC,WAAW,EAAE;MAChBA,WAAW,GAAG,IAAIxY,IAAI,CAAC,CAAC;IAC1B;;IAEA;IACA,IAAIsY,OAAO,IAAI,OAAOA,OAAO,KAAK,QAAQ,EAAE;MAC1C,MAAM,IAAIxc,SAAS,CAAC,oCAAoC,CAAC;IAC3D;IACA,IAAIyc,SAAS,IAAI,OAAOA,SAAS,KAAK,QAAQ,EAAE;MAC9C,MAAM,IAAIzc,SAAS,CAAC,sCAAsC,CAAC;IAC7D;IACA,IAAK0c,WAAW,IAAI,EAAEA,WAAW,YAAYxY,IAAI,CAAC,IAAMwY,WAAW,IAAIG,KAAK,EAAAF,YAAA,GAACD,WAAW,cAAAC,YAAA,uBAAXA,YAAA,CAAa9Q,OAAO,CAAC,CAAC,CAAE,EAAE;MACrG,MAAM,IAAI7L,SAAS,CAAC,gDAAgD,CAAC;IACvE;IAEA,MAAMc,KAAK,GAAG2b,SAAS,GAAGnkB,EAAE,CAAC0K,SAAS,CAACyZ,SAAS,CAAC,GAAG/e,SAAS;IAE7D,IAAI;MACF,MAAMO,MAAM,GAAG,MAAM,IAAI,CAAC+F,oBAAoB,CAAC7D,UAAU,CAAC;MAC1D,MAAM,IAAI,CAACwB,oBAAoB,CAAC,CAAC;MACjC,MAAMnC,UAAU,GAAG,IAAI,CAACkB,iBAAiB,CAAC;QAAEE,MAAM;QAAE3C,MAAM;QAAEkC,UAAU;QAAEC,UAAU;QAAEU;MAAM,CAAC,CAAC;MAE5F,OAAO1H,kBAAkB,CACvBoG,UAAU,EACV,IAAI,CAACT,SAAS,EACd,IAAI,CAACC,SAAS,EACd,IAAI,CAACC,YAAY,EACjBhB,MAAM,EACNye,WAAW,EACXF,OACF,CAAC;IACH,CAAC,CAAC,OAAOna,GAAG,EAAE;MACZ,IAAIA,GAAG,YAAY5J,MAAM,CAAC+L,sBAAsB,EAAE;QAChD,MAAM,IAAI/L,MAAM,CAACuF,oBAAoB,CAAE,mCAAkCmC,UAAW,GAAE,CAAC;MACzF;MAEA,MAAMkC,GAAG;IACX;EACF;EAEA,MAAMya,kBAAkBA,CACtB3c,UAAkB,EAClBC,UAAkB,EAClBoc,OAAgB,EAChBO,WAAyC,EACzCL,WAAkB,EACD;IACjB,IAAI,CAAC/hB,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IAEA,MAAM4c,gBAAgB,GAAG,CACvB,uBAAuB,EACvB,2BAA2B,EAC3B,kBAAkB,EAClB,wBAAwB,EACxB,8BAA8B,EAC9B,2BAA2B,CAC5B;IACDA,gBAAgB,CAACxa,OAAO,CAAEya,MAAM,IAAK;MACnC;MACA,IAAIF,WAAW,KAAKrf,SAAS,IAAIqf,WAAW,CAACE,MAAM,CAAC,KAAKvf,SAAS,IAAI,CAAChD,QAAQ,CAACqiB,WAAW,CAACE,MAAM,CAAC,CAAC,EAAE;QACpG,MAAM,IAAIjd,SAAS,CAAE,mBAAkBid,MAAO,6BAA4B,CAAC;MAC7E;IACF,CAAC,CAAC;IACF,OAAO,IAAI,CAACV,YAAY,CAAC,KAAK,EAAEpc,UAAU,EAAEC,UAAU,EAAEoc,OAAO,EAAEO,WAAW,EAAEL,WAAW,CAAC;EAC5F;EAEA,MAAMQ,kBAAkBA,CAAC/c,UAAkB,EAAEC,UAAkB,EAAEoc,OAAgB,EAAmB;IAClG,IAAI,CAAC7hB,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,wBAAuBrE,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAACtF,iBAAiB,CAACuF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI3H,MAAM,CAACgO,sBAAsB,CAAE,wBAAuBrG,UAAW,EAAC,CAAC;IAC/E;IAEA,OAAO,IAAI,CAACmc,YAAY,CAAC,KAAK,EAAEpc,UAAU,EAAEC,UAAU,EAAEoc,OAAO,CAAC;EAClE;EAEAW,aAAaA,CAAA,EAAe;IAC1B,OAAO,IAAIvhB,UAAU,CAAC,CAAC;EACzB;EAEA,MAAMwhB,mBAAmBA,CAACC,UAAsB,EAA6B;IAC3E,IAAI,IAAI,CAACne,SAAS,EAAE;MAClB,MAAM,IAAIzG,MAAM,CAACmkB,qBAAqB,CAAC,kEAAkE,CAAC;IAC5G;IACA,IAAI,CAACriB,QAAQ,CAAC8iB,UAAU,CAAC,EAAE;MACzB,MAAM,IAAIrd,SAAS,CAAC,uCAAuC,CAAC;IAC9D;IACA,MAAMG,UAAU,GAAGkd,UAAU,CAACC,QAAQ,CAAClU,MAAgB;IACvD,IAAI;MACF,MAAMnL,MAAM,GAAG,MAAM,IAAI,CAAC+F,oBAAoB,CAAC7D,UAAU,CAAC;MAE1D,MAAM8D,IAAI,GAAG,IAAIC,IAAI,CAAC,CAAC;MACvB,MAAMqZ,OAAO,GAAGtiB,YAAY,CAACgJ,IAAI,CAAC;MAClC,MAAM,IAAI,CAACtC,oBAAoB,CAAC,CAAC;MAEjC,IAAI,CAAC0b,UAAU,CAAC9M,MAAM,CAACiN,UAAU,EAAE;QACjC;QACA;QACA,MAAMhB,OAAO,GAAG,IAAItY,IAAI,CAAC,CAAC;QAC1BsY,OAAO,CAACiB,UAAU,CAAC3kB,uBAAuB,CAAC;QAC3CukB,UAAU,CAACK,UAAU,CAAClB,OAAO,CAAC;MAChC;MAEAa,UAAU,CAAC9M,MAAM,CAAC2G,UAAU,CAAChQ,IAAI,CAAC,CAAC,IAAI,EAAE,aAAa,EAAEqW,OAAO,CAAC,CAAC;MACjEF,UAAU,CAACC,QAAQ,CAAC,YAAY,CAAC,GAAGC,OAAO;MAE3CF,UAAU,CAAC9M,MAAM,CAAC2G,UAAU,CAAChQ,IAAI,CAAC,CAAC,IAAI,EAAE,kBAAkB,EAAE,kBAAkB,CAAC,CAAC;MACjFmW,UAAU,CAACC,QAAQ,CAAC,iBAAiB,CAAC,GAAG,kBAAkB;MAE3DD,UAAU,CAAC9M,MAAM,CAAC2G,UAAU,CAAChQ,IAAI,CAAC,CAAC,IAAI,EAAE,mBAAmB,EAAE,IAAI,CAACnI,SAAS,GAAG,GAAG,GAAGlF,QAAQ,CAACoE,MAAM,EAAEgG,IAAI,CAAC,CAAC,CAAC;MAC7GoZ,UAAU,CAACC,QAAQ,CAAC,kBAAkB,CAAC,GAAG,IAAI,CAACve,SAAS,GAAG,GAAG,GAAGlF,QAAQ,CAACoE,MAAM,EAAEgG,IAAI,CAAC;MAEvF,IAAI,IAAI,CAAChF,YAAY,EAAE;QACrBoe,UAAU,CAAC9M,MAAM,CAAC2G,UAAU,CAAChQ,IAAI,CAAC,CAAC,IAAI,EAAE,uBAAuB,EAAE,IAAI,CAACjI,YAAY,CAAC,CAAC;QACrFoe,UAAU,CAACC,QAAQ,CAAC,sBAAsB,CAAC,GAAG,IAAI,CAACre,YAAY;MACjE;MAEA,MAAM0e,YAAY,GAAG7Z,MAAM,CAAC4D,IAAI,CAAC3E,IAAI,CAACC,SAAS,CAACqa,UAAU,CAAC9M,MAAM,CAAC,CAAC,CAAC9O,QAAQ,CAAC,QAAQ,CAAC;MAEtF4b,UAAU,CAACC,QAAQ,CAAC/M,MAAM,GAAGoN,YAAY;MAEzCN,UAAU,CAACC,QAAQ,CAAC,iBAAiB,CAAC,GAAGnkB,sBAAsB,CAAC8E,MAAM,EAAEgG,IAAI,EAAE,IAAI,CAACjF,SAAS,EAAE2e,YAAY,CAAC;MAC3G,MAAMhd,IAAI,GAAG;QACX1C,MAAM,EAAEA,MAAM;QACdkC,UAAU,EAAEA,UAAU;QACtBS,MAAM,EAAE;MACV,CAAC;MACD,MAAMpB,UAAU,GAAG,IAAI,CAACkB,iBAAiB,CAACC,IAAI,CAAC;MAC/C,MAAMid,OAAO,GAAG,IAAI,CAAC/f,IAAI,IAAI,EAAE,IAAI,IAAI,CAACA,IAAI,KAAK,GAAG,GAAG,EAAE,GAAI,IAAG,IAAI,CAACA,IAAI,CAAC4D,QAAQ,CAAC,CAAE,EAAC;MACtF,MAAMoc,MAAM,GAAI,GAAEre,UAAU,CAACpB,QAAS,KAAIoB,UAAU,CAACtB,IAAK,GAAE0f,OAAQ,GAAEpe,UAAU,CAACxH,IAAK,EAAC;MACvF,OAAO;QAAE8lB,OAAO,EAAED,MAAM;QAAEP,QAAQ,EAAED,UAAU,CAACC;MAAS,CAAC;IAC3D,CAAC,CAAC,OAAOjb,GAAG,EAAE;MACZ,IAAIA,GAAG,YAAY5J,MAAM,CAAC+L,sBAAsB,EAAE;QAChD,MAAM,IAAI/L,MAAM,CAACuF,oBAAoB,CAAE,mCAAkCmC,UAAW,GAAE,CAAC;MACzF;MAEA,MAAMkC,GAAG;IACX;EACF;EACA;EACA,MAAM0b,gBAAgBA,CAAC5d,UAAkB,EAAEkJ,MAAe,EAAEmD,MAAe,EAAEwR,aAAmC,EAAE;IAChH,IAAI,CAACrjB,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACzF,QAAQ,CAAC2O,MAAM,CAAC,EAAE;MACrB,MAAM,IAAIrJ,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA,IAAIwM,MAAM,IAAI,CAAC9R,QAAQ,CAAC8R,MAAM,CAAC,EAAE;MAC/B,MAAM,IAAIxM,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IAEA,IAAIge,aAAa,IAAI,CAACzjB,QAAQ,CAACyjB,aAAa,CAAC,EAAE;MAC7C,MAAM,IAAIhe,SAAS,CAAC,0CAA0C,CAAC;IACjE;IACA,IAAI;MAAEie,SAAS;MAAEC,OAAO;MAAEC,cAAc;MAAEC,eAAe;MAAE5U;IAAU,CAAC,GAAGwU,aAAoC;IAE7G,IAAI,CAACtjB,QAAQ,CAACujB,SAAS,CAAC,EAAE;MACxB,MAAM,IAAIje,SAAS,CAAC,sCAAsC,CAAC;IAC7D;IACA,IAAI,CAAC1F,QAAQ,CAAC4jB,OAAO,CAAC,EAAE;MACtB,MAAM,IAAIle,SAAS,CAAC,oCAAoC,CAAC;IAC3D;IAEA,MAAMgL,OAAO,GAAG,EAAE;IAClB;IACAA,OAAO,CAAC9D,IAAI,CAAE,UAASzL,SAAS,CAAC4N,MAAM,CAAE,EAAC,CAAC;IAC3C2B,OAAO,CAAC9D,IAAI,CAAE,aAAYzL,SAAS,CAACwiB,SAAS,CAAE,EAAC,CAAC;IACjDjT,OAAO,CAAC9D,IAAI,CAAE,mBAAkB,CAAC;IAEjC,IAAIiX,cAAc,EAAE;MAClBnT,OAAO,CAAC9D,IAAI,CAAE,UAAS,CAAC;IAC1B;IAEA,IAAIiX,cAAc,EAAE;MAClB;MACA,IAAI3U,SAAS,EAAE;QACbwB,OAAO,CAAC9D,IAAI,CAAE,cAAasC,SAAU,EAAC,CAAC;MACzC;MACA,IAAI4U,eAAe,EAAE;QACnBpT,OAAO,CAAC9D,IAAI,CAAE,qBAAoBkX,eAAgB,EAAC,CAAC;MACtD;IACF,CAAC,MAAM,IAAI5R,MAAM,EAAE;MACjBA,MAAM,GAAG/Q,SAAS,CAAC+Q,MAAM,CAAC;MAC1BxB,OAAO,CAAC9D,IAAI,CAAE,UAASsF,MAAO,EAAC,CAAC;IAClC;;IAEA;IACA,IAAI0R,OAAO,EAAE;MACX,IAAIA,OAAO,IAAI,IAAI,EAAE;QACnBA,OAAO,GAAG,IAAI;MAChB;MACAlT,OAAO,CAAC9D,IAAI,CAAE,YAAWgX,OAAQ,EAAC,CAAC;IACrC;IACAlT,OAAO,CAACE,IAAI,CAAC,CAAC;IACd,IAAIpK,KAAK,GAAG,EAAE;IACd,IAAIkK,OAAO,CAACzH,MAAM,GAAG,CAAC,EAAE;MACtBzC,KAAK,GAAI,GAAEkK,OAAO,CAACI,IAAI,CAAC,GAAG,CAAE,EAAC;IAChC;IAEA,MAAMxK,MAAM,GAAG,KAAK;IACpB,MAAMgD,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,CAAC;IACtE,MAAM+C,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,MAAMya,WAAW,GAAGhiB,gBAAgB,CAACwH,IAAI,CAAC;IAC1C,OAAOwa,WAAW;EACpB;EAEAC,WAAWA,CACTne,UAAkB,EAClBkJ,MAAe,EACftB,SAAmB,EACnBwW,QAA0C,EAChB;IAC1B,IAAIlV,MAAM,KAAK3L,SAAS,EAAE;MACxB2L,MAAM,GAAG,EAAE;IACb;IACA,IAAItB,SAAS,KAAKrK,SAAS,EAAE;MAC3BqK,SAAS,GAAG,KAAK;IACnB;IACA,IAAI,CAACpN,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACpF,aAAa,CAACsO,MAAM,CAAC,EAAE;MAC1B,MAAM,IAAI5Q,MAAM,CAAC6Q,kBAAkB,CAAE,oBAAmBD,MAAO,EAAC,CAAC;IACnE;IACA,IAAI,CAAC3O,QAAQ,CAAC2O,MAAM,CAAC,EAAE;MACrB,MAAM,IAAIrJ,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA,IAAI,CAAC7F,SAAS,CAAC4N,SAAS,CAAC,EAAE;MACzB,MAAM,IAAI/H,SAAS,CAAC,uCAAuC,CAAC;IAC9D;IACA,IAAIue,QAAQ,IAAI,CAAChkB,QAAQ,CAACgkB,QAAQ,CAAC,EAAE;MACnC,MAAM,IAAIve,SAAS,CAAC,qCAAqC,CAAC;IAC5D;IACA,IAAIwM,MAA0B,GAAG,EAAE;IACnC,IAAIhD,SAA6B,GAAG,EAAE;IACtC,IAAI4U,eAAmC,GAAG,EAAE;IAC5C,IAAII,OAAqB,GAAG,EAAE;IAC9B,IAAI7U,KAAK,GAAG,KAAK;IACjB,MAAMC,UAA2B,GAAG,IAAI3R,MAAM,CAAC4R,QAAQ,CAAC;MAAEC,UAAU,EAAE;IAAK,CAAC,CAAC;IAC7EF,UAAU,CAACG,KAAK,GAAG,YAAY;MAC7B;MACA,IAAIyU,OAAO,CAACjb,MAAM,EAAE;QAClBqG,UAAU,CAAC1C,IAAI,CAACsX,OAAO,CAACxU,KAAK,CAAC,CAAC,CAAC;QAChC;MACF;MACA,IAAIL,KAAK,EAAE;QACT,OAAOC,UAAU,CAAC1C,IAAI,CAAC,IAAI,CAAC;MAC9B;MAEA,IAAI;QACF,MAAM8W,aAAa,GAAG;UACpBC,SAAS,EAAElW,SAAS,GAAG,EAAE,GAAG,GAAG;UAAE;UACjCmW,OAAO,EAAE,IAAI;UACbC,cAAc,EAAEI,QAAQ,aAARA,QAAQ,uBAARA,QAAQ,CAAEJ,cAAc;UACxC;UACA3U,SAAS,EAAEA,SAAS;UACpB4U,eAAe,EAAEA;QACnB,CAAC;QAED,MAAM7Y,MAA0B,GAAG,MAAM,IAAI,CAACwY,gBAAgB,CAAC5d,UAAU,EAAEkJ,MAAM,EAAEmD,MAAM,EAAEwR,aAAa,CAAC;QACzG,IAAIzY,MAAM,CAACsF,WAAW,EAAE;UACtB2B,MAAM,GAAGjH,MAAM,CAACkZ,UAAU,IAAI/gB,SAAS;UACvC,IAAI6H,MAAM,CAACiE,SAAS,EAAE;YACpBA,SAAS,GAAGjE,MAAM,CAACiE,SAAS;UAC9B;UACA,IAAIjE,MAAM,CAAC6Y,eAAe,EAAE;YAC1BA,eAAe,GAAG7Y,MAAM,CAAC6Y,eAAe;UAC1C;QACF,CAAC,MAAM;UACLzU,KAAK,GAAG,IAAI;QACd;QACA,IAAIpE,MAAM,CAACiZ,OAAO,EAAE;UAClBA,OAAO,GAAGjZ,MAAM,CAACiZ,OAAO;QAC1B;QACA;QACA5U,UAAU,CAACG,KAAK,CAAC,CAAC;MACpB,CAAC,CAAC,OAAO1H,GAAG,EAAE;QACZuH,UAAU,CAACgB,IAAI,CAAC,OAAO,EAAEvI,GAAG,CAAC;MAC/B;IACF,CAAC;IACD,OAAOuH,UAAU;EACnB;EAEA,MAAM8U,kBAAkBA,CACtBve,UAAkB,EAClBkJ,MAAc,EACdsV,iBAAyB,EACzBpV,SAAiB,EACjBqV,OAAe,EACfC,UAAkB,EACQ;IAC1B,IAAI,CAAClkB,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACzF,QAAQ,CAAC2O,MAAM,CAAC,EAAE;MACrB,MAAM,IAAIrJ,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA,IAAI,CAACtF,QAAQ,CAACikB,iBAAiB,CAAC,EAAE;MAChC,MAAM,IAAI3e,SAAS,CAAC,8CAA8C,CAAC;IACrE;IACA,IAAI,CAACtF,QAAQ,CAAC6O,SAAS,CAAC,EAAE;MACxB,MAAM,IAAIvJ,SAAS,CAAC,sCAAsC,CAAC;IAC7D;IACA,IAAI,CAAC1F,QAAQ,CAACskB,OAAO,CAAC,EAAE;MACtB,MAAM,IAAI5e,SAAS,CAAC,oCAAoC,CAAC;IAC3D;IACA,IAAI,CAACtF,QAAQ,CAACmkB,UAAU,CAAC,EAAE;MACzB,MAAM,IAAI7e,SAAS,CAAC,uCAAuC,CAAC;IAC9D;IAEA,MAAMgL,OAAO,GAAG,EAAE;IAClBA,OAAO,CAAC9D,IAAI,CAAE,aAAY,CAAC;IAC3B8D,OAAO,CAAC9D,IAAI,CAAE,mBAAkB,CAAC;IACjC8D,OAAO,CAAC9D,IAAI,CAAE,UAASzL,SAAS,CAAC4N,MAAM,CAAE,EAAC,CAAC;IAC3C2B,OAAO,CAAC9D,IAAI,CAAE,aAAYzL,SAAS,CAAC8N,SAAS,CAAE,EAAC,CAAC;IAEjD,IAAIoV,iBAAiB,EAAE;MACrB3T,OAAO,CAAC9D,IAAI,CAAE,sBAAqBzL,SAAS,CAACkjB,iBAAiB,CAAE,EAAC,CAAC;IACpE;IACA,IAAIE,UAAU,EAAE;MACd7T,OAAO,CAAC9D,IAAI,CAAE,eAAczL,SAAS,CAACojB,UAAU,CAAE,EAAC,CAAC;IACtD;IACA,IAAID,OAAO,EAAE;MACX,IAAIA,OAAO,IAAI,IAAI,EAAE;QACnBA,OAAO,GAAG,IAAI;MAChB;MACA5T,OAAO,CAAC9D,IAAI,CAAE,YAAW0X,OAAQ,EAAC,CAAC;IACrC;IACA5T,OAAO,CAACE,IAAI,CAAC,CAAC;IACd,IAAIpK,KAAK,GAAG,EAAE;IACd,IAAIkK,OAAO,CAACzH,MAAM,GAAG,CAAC,EAAE;MACtBzC,KAAK,GAAI,GAAEkK,OAAO,CAACI,IAAI,CAAC,GAAG,CAAE,EAAC;IAChC;IAEA,MAAMxK,MAAM,GAAG,KAAK;IACpB,MAAMgD,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,CAAC;IACtE,MAAM+C,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,OAAOtH,kBAAkB,CAACuH,IAAI,CAAC;EACjC;EAEAib,aAAaA,CACX3e,UAAkB,EAClBkJ,MAAe,EACftB,SAAmB,EACnB8W,UAAmB,EACO;IAC1B,IAAIxV,MAAM,KAAK3L,SAAS,EAAE;MACxB2L,MAAM,GAAG,EAAE;IACb;IACA,IAAItB,SAAS,KAAKrK,SAAS,EAAE;MAC3BqK,SAAS,GAAG,KAAK;IACnB;IACA,IAAI8W,UAAU,KAAKnhB,SAAS,EAAE;MAC5BmhB,UAAU,GAAG,EAAE;IACjB;IACA,IAAI,CAAClkB,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAACpF,aAAa,CAACsO,MAAM,CAAC,EAAE;MAC1B,MAAM,IAAI5Q,MAAM,CAAC6Q,kBAAkB,CAAE,oBAAmBD,MAAO,EAAC,CAAC;IACnE;IACA,IAAI,CAAC3O,QAAQ,CAAC2O,MAAM,CAAC,EAAE;MACrB,MAAM,IAAIrJ,SAAS,CAAC,mCAAmC,CAAC;IAC1D;IACA,IAAI,CAAC7F,SAAS,CAAC4N,SAAS,CAAC,EAAE;MACzB,MAAM,IAAI/H,SAAS,CAAC,uCAAuC,CAAC;IAC9D;IACA,IAAI,CAACtF,QAAQ,CAACmkB,UAAU,CAAC,EAAE;MACzB,MAAM,IAAI7e,SAAS,CAAC,uCAAuC,CAAC;IAC9D;IAEA,MAAMuJ,SAAS,GAAGxB,SAAS,GAAG,EAAE,GAAG,GAAG;IACtC,MAAMgX,SAAS,GAAG1V,MAAM;IACxB,MAAM2V,aAAa,GAAGH,UAAU;IAChC,IAAIF,iBAAiB,GAAG,EAAE;IAC1B,IAAIH,OAAqB,GAAG,EAAE;IAC9B,IAAI7U,KAAK,GAAG,KAAK;IACjB,MAAMC,UAA2B,GAAG,IAAI3R,MAAM,CAAC4R,QAAQ,CAAC;MAAEC,UAAU,EAAE;IAAK,CAAC,CAAC;IAC7EF,UAAU,CAACG,KAAK,GAAG,YAAY;MAC7B,IAAIyU,OAAO,CAACjb,MAAM,EAAE;QAClBqG,UAAU,CAAC1C,IAAI,CAACsX,OAAO,CAACxU,KAAK,CAAC,CAAC,CAAC;QAChC;MACF;MACA,IAAIL,KAAK,EAAE;QACT,OAAOC,UAAU,CAAC1C,IAAI,CAAC,IAAI,CAAC;MAC9B;MAEA,IAAI;QACF,MAAM3B,MAAM,GAAG,MAAM,IAAI,CAACmZ,kBAAkB,CAC1Cve,UAAU,EACV4e,SAAS,EACTJ,iBAAiB,EACjBpV,SAAS,EACT,IAAI,EACJyV,aACF,CAAC;QACD,IAAIzZ,MAAM,CAACsF,WAAW,EAAE;UACtB8T,iBAAiB,GAAGpZ,MAAM,CAAC0Z,qBAAqB;QAClD,CAAC,MAAM;UACLtV,KAAK,GAAG,IAAI;QACd;QACA6U,OAAO,GAAGjZ,MAAM,CAACiZ,OAAO;QACxB;QACA5U,UAAU,CAACG,KAAK,CAAC,CAAC;MACpB,CAAC,CAAC,OAAO1H,GAAG,EAAE;QACZuH,UAAU,CAACgB,IAAI,CAAC,OAAO,EAAEvI,GAAG,CAAC;MAC/B;IACF,CAAC;IACD,OAAOuH,UAAU;EACnB;EAEA,MAAMsV,qBAAqBA,CAAC/e,UAAkB,EAAE6P,MAA0B,EAAiB;IACzF,IAAI,CAACrV,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,IAAI,CAAC5F,QAAQ,CAACyV,MAAM,CAAC,EAAE;MACrB,MAAM,IAAIhQ,SAAS,CAAC,gDAAgD,CAAC;IACvE;IACA,MAAMY,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,cAAc;IAC5B,MAAMkL,OAAO,GAAG,IAAIzT,MAAM,CAACqE,OAAO,CAAC;MACjCsT,QAAQ,EAAE,2BAA2B;MACrCrT,UAAU,EAAE;QAAEC,MAAM,EAAE;MAAM,CAAC;MAC7BC,QAAQ,EAAE;IACZ,CAAC,CAAC;IACF,MAAMsG,OAAO,GAAG2I,OAAO,CAACnG,WAAW,CAACmK,MAAM,CAAC;IAC3C,MAAM,IAAI,CAACtM,oBAAoB,CAAC;MAAE9C,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,EAAEuC,OAAO,CAAC;EACzE;EAEA,MAAM8b,2BAA2BA,CAAChf,UAAkB,EAAiB;IACnE,MAAM,IAAI,CAAC+e,qBAAqB,CAAC/e,UAAU,EAAE,IAAIlH,kBAAkB,CAAC,CAAC,CAAC;EACxE;EAEA,MAAMmmB,qBAAqBA,CAACjf,UAAkB,EAAqC;IACjF,IAAI,CAACxF,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAC,uBAAuB,GAAGrE,UAAU,CAAC;IAC/E;IACA,MAAMS,MAAM,GAAG,KAAK;IACpB,MAAME,KAAK,GAAG,cAAc;IAC5B,MAAM8C,GAAG,GAAG,MAAM,IAAI,CAACR,gBAAgB,CAAC;MAAExC,MAAM;MAAET,UAAU;MAAEW;IAAM,CAAC,CAAC;IACtE,MAAM+C,IAAI,GAAG,MAAM7H,YAAY,CAAC4H,GAAG,CAAC;IACpC,OAAO1H,uBAAuB,CAAC2H,IAAI,CAAC;EACtC;EAEAwb,wBAAwBA,CACtBlf,UAAkB,EAClBkJ,MAAc,EACdiW,MAAc,EACdC,MAA2B,EACP;IACpB,IAAI,CAAC5kB,iBAAiB,CAACwF,UAAU,CAAC,EAAE;MAClC,MAAM,IAAI1H,MAAM,CAAC+L,sBAAsB,CAAE,wBAAuBrE,UAAW,EAAC,CAAC;IAC/E;IACA,IAAI,CAACzF,QAAQ,CAAC2O,MAAM,CAAC,EAAE;MACrB,MAAM,IAAIrJ,SAAS,CAAC,+BAA+B,CAAC;IACtD;IACA,IAAI,CAACtF,QAAQ,CAAC4kB,MAAM,CAAC,EAAE;MACrB,MAAM,IAAItf,SAAS,CAAC,+BAA+B,CAAC;IACtD;IACA,IAAI,CAAC2V,KAAK,CAACC,OAAO,CAAC2J,MAAM,CAAC,EAAE;MAC1B,MAAM,IAAIvf,SAAS,CAAC,8BAA8B,CAAC;IACrD;IACA,MAAMwf,QAAQ,GAAG,IAAItmB,kBAAkB,CAAC,IAAI,EAAEiH,UAAU,EAAEkJ,MAAM,EAAEiW,MAAM,EAAEC,MAAM,CAAC;IACjFC,QAAQ,CAACC,KAAK,CAAC,CAAC;IAChB,OAAOD,QAAQ;EACjB;AACF"}