astx
Version:
super powerful structural search and replace for JavaScript and TypeScript to automate your refactoring
365 lines (334 loc) • 36.5 kB
JavaScript
import path from 'path'
import chalk from 'chalk'
import formatDiff from '../util/formatDiff.mjs'
import { isEmpty, once } from 'lodash-es'
import inquirer from 'inquirer'
import fs from 'fs-extra'
import dedent from 'dedent-js'
import CodeFrameError from '../util/CodeFrameError.mjs'
import { formatIpcMatches } from '../util/formatMatches.mjs'
import {
AstxWorkerPool,
astxCosmiconfig,
runTransform,
} from '../node/index.mjs'
import { invertIpcError, makeIpcTransformResult } from '../node/ipc.mjs'
import ansiEscapes from 'ansi-escapes'
import { spinner } from './spinner.mjs'
import '../node/registerTsNode.mjs'
import isInteractive from '../util/isInteractive.mjs'
/* eslint-disable no-console */
const transform = {
command: '$0 [filesAndDirectories..]',
describe: 'apply a transform to the given files and directories',
builder: (yargs) =>
yargs
.positional('filesAndDirectories', {
type: 'string',
array: true,
})
.option('transform', {
alias: 't',
describe: `path to the transform file. Can be either a local path or url. Defaults to ./astx.ts or ./astx.js if --find isn't given`,
})
.options('parser', {
describe:
'parser to use (options: babel, babel/auto, recast/babel, recast/babel/auto)',
type: 'string',
})
.options('parserOptions', {
describe: 'options for parser',
type: 'string',
})
.option('find', {
alias: 'f',
describe: 'search pattern',
type: 'string',
})
.option('replace', {
alias: 'r',
describe: 'replace pattern',
type: 'string',
})
.option('yes', {
alias: 'y',
describe: `don't ask for confirmation before writing changes`,
type: 'boolean',
})
.option('gitignore', {
type: 'boolean',
describe: `ignore gitignored files`,
default: true,
})
.option('workers', {
type: 'number',
describe: 'number of worker threads to use',
})
.option('debugConfig', {
type: 'boolean',
describe: 'print found config and location',
}),
handler: async (argv) => {
var _ref, _argv$workers
const startTime = Date.now()
const configResult = await astxCosmiconfig.search()
if (argv.debugConfig) {
console.log(JSON.stringify(configResult, null, 2))
process.exit(0)
}
const config =
configResult === null || configResult === void 0
? void 0
: configResult.config
const paths = (argv.filesAndDirectories || []).filter(
(x) => typeof x === 'string'
)
const { transform, transformFile } = await (async () => {
if (argv.transform) {
const transformFile = path.resolve(argv.transform)
return {
transformFile,
transform: await import(transformFile),
}
} else if (argv.find) {
const getOpt = (regex) => {
const index = process.argv.findIndex((a) => regex.test(a))
return index >= 0 ? process.argv[index + 1] : undefined
} // yargs Eats quotes, not cool...
const find = getOpt(/^(-f|--find)$/)
const replace = getOpt(/^(-r|--replace)$/)
return {
transform: {
find,
replace,
},
}
} else {
const files = [path.resolve('astx.ts'), path.resolve('astx.js')]
for (const transformFile of files) {
if (await fs.pathExists(transformFile)) {
return {
transformFile,
transform: await import(transformFile),
}
}
}
throw new Error(`missing transform file: ${files.join(' or ')}`)
}
})()
const { parser, parserOptions, gitignore } = argv
const results = {}
let errorCount = 0
let changedCount = 0
let unchangedCount = 0
let progress = {
type: 'progress',
completed: 0,
total: 0,
globDone: false,
}
let progressDisplayed = false
function clearProgress() {
if (progressDisplayed) {
process.stderr.write(ansiEscapes.cursorLeft + ansiEscapes.eraseLine)
progressDisplayed = false
}
}
function showProgress() {
clearProgress()
progressDisplayed = true
const { completed, total, globDone } = progress
process.stderr.write(
chalk.magenta(
`${spinner()} Running... ${completed}/${total}${
globDone && total
? ` (${((completed * 100) / total).toFixed(1)}%)`
: ''
} ${((Date.now() - startTime) / 1000).toFixed(2)}s`
)
)
}
let spinnerInterval
const interactive = isInteractive()
const workers =
(_ref =
(_argv$workers = argv.workers) !== null && _argv$workers !== void 0
? _argv$workers
: process.env.ASTX_WORKERS
? parseInt(process.env.ASTX_WORKERS)
: undefined) !== null && _ref !== void 0
? _ref
: config === null || config === void 0
? void 0
: config.workers
const pool =
workers === 0
? null
: new AstxWorkerPool({
capacity: workers,
})
try {
if (interactive) {
spinnerInterval = setInterval(showProgress, 30)
}
const runTransformOptions = {
gitignore: gitignore ? undefined : null,
transform,
transformFile,
paths,
config: {
parser: parser,
parserOptions: parserOptions ? JSON.parse(parserOptions) : undefined,
},
}
for await (const _event of pool
? pool.runTransform(runTransformOptions)
: runTransform(runTransformOptions)) {
const event =
!pool && _event.type === 'result'
? {
type: 'result',
result: makeIpcTransformResult(_event),
}
: _event
if (event.type === 'progress') {
progress = event
if (interactive) showProgress()
continue
}
clearProgress()
const {
file,
source,
transformed,
reports,
matches,
error: _error,
} = event.result
const error = _error ? invertIpcError(_error) : undefined
const relpath = path.relative(process.cwd(), file)
const logHeader = once((logFn) =>
logFn(
chalk.blue(dedent`
${'='.repeat(relpath.length)}
${chalk.bold(relpath)}
${'='.repeat(relpath.length)}
`)
)
)
if (error) {
errorCount++
logHeader(console.error)
if (error instanceof CodeFrameError) {
console.error(
error.format({
highlightCode: true,
forceColor: true,
stack: true,
})
)
} else {
console.error(chalk.red(error.stack))
}
} else if (source && transformed && source !== transformed) {
changedCount++
results[file] = transformed
if (!argv.yes) {
logHeader(console.log)
console.log(formatDiff(source, transformed))
}
} else if (
matches !== null &&
matches !== void 0 &&
matches.length &&
source &&
transform.find &&
!transform.replace &&
!transform.astx
) {
logHeader(console.log)
console.log(formatIpcMatches(source, matches))
} else {
unchangedCount++
}
if (
reports !== null &&
reports !== void 0 &&
reports.length &&
!transform.onReport
) {
logHeader(console.error)
console.error(
chalk.blue(dedent`
Reports
-------
`)
)
reports === null || reports === void 0
? void 0
: reports.forEach((r) => console.error(r))
}
}
} catch (error) {
console.error(
chalk.red(error instanceof Error ? error.stack : String(error))
)
process.exit(1)
} finally {
if (spinnerInterval != null) clearInterval(spinnerInterval)
clearProgress()
}
if (transform.replace || transform.astx) {
console.error(
chalk.yellow(
`${changedCount} file${changedCount === 1 ? '' : 's'} changed`
)
)
console.error(
chalk.green(
`${unchangedCount} file${unchangedCount === 1 ? '' : 's'} unchanged`
)
)
if (errorCount > 0) {
console.error(
chalk.red(`${errorCount} file${errorCount === 1 ? '' : 's'} errored`)
)
}
} else if (transform.find) {
console.error(
chalk.yellow(
`\n${unchangedCount} file${
changedCount === 1 ? '' : 's'
} had no matches`
)
)
}
if (!isEmpty(results)) {
const apply = argv.yes
? true
: (
await inquirer.prompt([
{
type: 'confirm',
name: 'apply',
message: 'Apply changes',
default: false,
},
])
).apply
if (apply) {
for (const file in results) {
await fs.writeFile(file, results[file], 'utf8')
console.error(`Wrote ${file}`)
}
}
if (process.send)
process.send({
exit: 0,
})
}
await (pool === null || pool === void 0 ? void 0 : pool.end())
process.exit(0)
},
}
export default transform //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["path","chalk","formatDiff","isEmpty","once","inquirer","fs","dedent","CodeFrameError","formatIpcMatches","AstxWorkerPool","astxCosmiconfig","runTransform","invertIpcError","makeIpcTransformResult","ansiEscapes","spinner","isInteractive","transform","command","describe","builder","yargs","positional","type","array","option","alias","options","default","handler","argv","startTime","Date","now","configResult","search","debugConfig","console","log","JSON","stringify","process","exit","config","paths","filesAndDirectories","filter","x","transformFile","resolve","find","getOpt","regex","index","findIndex","a","test","undefined","replace","files","pathExists","Error","join","parser","parserOptions","gitignore","results","errorCount","changedCount","unchangedCount","progress","completed","total","globDone","progressDisplayed","clearProgress","stderr","write","cursorLeft","eraseLine","showProgress","magenta","toFixed","spinnerInterval","interactive","workers","env","ASTX_WORKERS","parseInt","pool","capacity","setInterval","runTransformOptions","parse","_event","event","result","file","source","transformed","reports","matches","error","_error","relpath","relative","cwd","logHeader","logFn","blue","repeat","length","bold","format","highlightCode","forceColor","stack","red","yes","astx","onReport","forEach","r","String","clearInterval","yellow","green","apply","prompt","name","message","writeFile","send","end"],"sources":["../../src/cli/transform.ts"],"sourcesContent":["import { Arguments, Argv, CommandModule } from 'yargs'\nimport path from 'path'\nimport chalk from 'chalk'\nimport formatDiff from '../util/formatDiff'\nimport { isEmpty, once } from 'lodash'\nimport inquirer from 'inquirer'\nimport fs from 'fs-extra'\nimport dedent from 'dedent-js'\nimport CodeFrameError from '../util/CodeFrameError'\nimport { formatIpcMatches } from '../util/formatMatches'\nimport {\n  AstxWorkerPool,\n  astxCosmiconfig,\n  runTransform,\n  Progress,\n} from '../node'\nimport {\n  invertIpcError,\n  IpcTransformResult,\n  makeIpcTransformResult,\n} from '../node/ipc'\nimport { Transform } from '../Astx'\nimport ansiEscapes from 'ansi-escapes'\nimport { spinner } from './spinner'\nimport '../node/registerTsNode'\nimport isInteractive from '../util/isInteractive'\n\n/* eslint-disable no-console */\n\ntype Options = {\n  transform?: string\n  parser?: string\n  parserOptions?: string\n  find?: string\n  replace?: string\n  filesAndDirectories?: string[]\n  yes?: boolean\n  gitignore?: boolean\n  threads?: number\n  debugConfig?: boolean\n}\n\nconst transform: CommandModule<Options> = {\n  command: '$0 [filesAndDirectories..]',\n  describe: 'apply a transform to the given files and directories',\n  builder: (yargs: Argv<Options>) =>\n    yargs\n      .positional('filesAndDirectories', {\n        type: 'string',\n        array: true,\n      })\n      .option('transform', {\n        alias: 't',\n        describe: `path to the transform file. Can be either a local path or url. Defaults to ./astx.ts or ./astx.js if --find isn't given`,\n      })\n      .options('parser', {\n        describe:\n          'parser to use (options: babel, babel/auto, recast/babel, recast/babel/auto)',\n        type: 'string',\n      })\n      .options('parserOptions', {\n        describe: 'options for parser',\n        type: 'string',\n      })\n      .option('find', {\n        alias: 'f',\n        describe: 'search pattern',\n        type: 'string',\n      })\n      .option('replace', {\n        alias: 'r',\n        describe: 'replace pattern',\n        type: 'string',\n      })\n      .option('yes', {\n        alias: 'y',\n        describe: `don't ask for confirmation before writing changes`,\n        type: 'boolean',\n      })\n      .option('gitignore', {\n        type: 'boolean',\n        describe: `ignore gitignored files`,\n        default: true,\n      })\n      .option('workers', {\n        type: 'number',\n        describe: 'number of worker threads to use',\n      })\n      .option('debugConfig', {\n        type: 'boolean',\n        describe: 'print found config and location',\n      }),\n\n  handler: async (argv: Arguments<Options>) => {\n    const startTime = Date.now()\n\n    const configResult = await astxCosmiconfig.search()\n    if (argv.debugConfig) {\n      console.log(JSON.stringify(configResult, null, 2))\n      process.exit(0)\n    }\n    const config = configResult?.config\n\n    const paths = (argv.filesAndDirectories || []).filter(\n      (x) => typeof x === 'string'\n    ) as string[]\n\n    const { transform, transformFile } = await (async (): Promise<{\n      transform: Transform\n      transformFile?: string\n    }> => {\n      if (argv.transform) {\n        const transformFile = path.resolve(argv.transform)\n        return {\n          transformFile,\n          transform: await import(transformFile),\n        }\n      } else if (argv.find) {\n        const getOpt = (regex: RegExp): string | undefined => {\n          const index = process.argv.findIndex((a) => regex.test(a))\n          return index >= 0 ? process.argv[index + 1] : undefined\n        }\n        // yargs Eats quotes, not cool...\n        const find = getOpt(/^(-f|--find)$/)\n        const replace = getOpt(/^(-r|--replace)$/)\n        return { transform: { find, replace } }\n      } else {\n        const files = [path.resolve('astx.ts'), path.resolve('astx.js')]\n        for (const transformFile of files) {\n          if (await fs.pathExists(transformFile)) {\n            return { transformFile, transform: await import(transformFile) }\n          }\n        }\n        throw new Error(`missing transform file: ${files.join(' or ')}`)\n      }\n    })()\n    const { parser, parserOptions, gitignore } = argv\n\n    const results: Record<string, string> = {}\n    let errorCount = 0\n    let changedCount = 0\n    let unchangedCount = 0\n\n    let progress: Progress = {\n      type: 'progress',\n      completed: 0,\n      total: 0,\n      globDone: false,\n    }\n\n    let progressDisplayed = false\n    function clearProgress() {\n      if (progressDisplayed) {\n        process.stderr.write(ansiEscapes.cursorLeft + ansiEscapes.eraseLine)\n        progressDisplayed = false\n      }\n    }\n    function showProgress() {\n      clearProgress()\n      progressDisplayed = true\n      const { completed, total, globDone } = progress\n      process.stderr.write(\n        chalk.magenta(\n          `${spinner()} Running... ${completed}/${total}${\n            globDone && total\n              ? ` (${((completed * 100) / total).toFixed(1)}%)`\n              : ''\n          } ${((Date.now() - startTime) / 1000).toFixed(2)}s`\n        )\n      )\n    }\n    let spinnerInterval\n\n    const interactive = isInteractive()\n    const workers =\n      argv.workers ??\n      (process.env.ASTX_WORKERS\n        ? parseInt(process.env.ASTX_WORKERS)\n        : undefined) ??\n      config?.workers\n    const pool =\n      workers === 0 ? null : new AstxWorkerPool({ capacity: workers })\n    try {\n      if (interactive) {\n        spinnerInterval = setInterval(showProgress, 30)\n      }\n      const runTransformOptions = {\n        gitignore: gitignore ? undefined : null,\n        transform,\n        transformFile,\n        paths,\n        config: {\n          parser: parser as any,\n          parserOptions: parserOptions ? JSON.parse(parserOptions) : undefined,\n        },\n      }\n      for await (const _event of pool\n        ? pool.runTransform(runTransformOptions)\n        : runTransform(runTransformOptions)) {\n        const event: { type: 'result'; result: IpcTransformResult } | Progress =\n          !pool && _event.type === 'result'\n            ? { type: 'result', result: makeIpcTransformResult(_event as any) }\n            : (_event as any)\n        if (event.type === 'progress') {\n          progress = event\n          if (interactive) showProgress()\n          continue\n        }\n        clearProgress()\n        const {\n          file,\n          source,\n          transformed,\n          reports,\n          matches,\n          error: _error,\n        } = event.result\n        const error = _error ? invertIpcError(_error) : undefined\n        const relpath = path.relative(process.cwd(), file)\n        const logHeader = once((logFn: (value: string) => any) =>\n          logFn(\n            chalk.blue(dedent`\n            ${'='.repeat(relpath.length)}\n            ${chalk.bold(relpath)}\n            ${'='.repeat(relpath.length)}\n          `)\n          )\n        )\n\n        if (error) {\n          errorCount++\n          logHeader(console.error)\n          if (error instanceof CodeFrameError) {\n            console.error(\n              error.format({\n                highlightCode: true,\n                forceColor: true,\n                stack: true,\n              })\n            )\n          } else {\n            console.error(chalk.red(error.stack))\n          }\n        } else if (source && transformed && source !== transformed) {\n          changedCount++\n          results[file] = transformed\n          if (!argv.yes) {\n            logHeader(console.log)\n            console.log(formatDiff(source, transformed))\n          }\n        } else if (\n          matches?.length &&\n          source &&\n          transform.find &&\n          !transform.replace &&\n          !transform.astx\n        ) {\n          logHeader(console.log)\n          console.log(formatIpcMatches(source, matches))\n        } else {\n          unchangedCount++\n        }\n\n        if (reports?.length && !transform.onReport) {\n          logHeader(console.error)\n          console.error(\n            chalk.blue(dedent`\n            Reports\n            -------\n          `)\n          )\n          reports?.forEach((r: any) => console.error(r))\n        }\n      }\n    } catch (error) {\n      console.error(\n        chalk.red(error instanceof Error ? error.stack : String(error))\n      )\n      process.exit(1)\n    } finally {\n      if (spinnerInterval != null) clearInterval(spinnerInterval)\n      clearProgress()\n    }\n\n    if (transform.replace || transform.astx) {\n      console.error(\n        chalk.yellow(\n          `${changedCount} file${changedCount === 1 ? '' : 's'} changed`\n        )\n      )\n      console.error(\n        chalk.green(\n          `${unchangedCount} file${unchangedCount === 1 ? '' : 's'} unchanged`\n        )\n      )\n      if (errorCount > 0) {\n        console.error(\n          chalk.red(`${errorCount} file${errorCount === 1 ? '' : 's'} errored`)\n        )\n      }\n    } else if (transform.find) {\n      console.error(\n        chalk.yellow(\n          `\\n${unchangedCount} file${\n            changedCount === 1 ? '' : 's'\n          } had no matches`\n        )\n      )\n    }\n\n    if (!isEmpty(results)) {\n      const apply = argv.yes\n        ? true\n        : (\n            await inquirer.prompt([\n              {\n                type: 'confirm',\n                name: 'apply',\n                message: 'Apply changes',\n                default: false,\n              },\n            ])\n          ).apply\n      if (apply) {\n        for (const file in results) {\n          await fs.writeFile(file, results[file], 'utf8')\n          console.error(`Wrote ${file}`)\n        }\n      }\n      if (process.send) process.send({ exit: 0 })\n    }\n    await pool?.end()\n    process.exit(0)\n  },\n}\n\nexport default transform\n"],"mappings":";AACA,OAAOA,IAAP,MAAiB,MAAjB;AACA,OAAOC,KAAP,MAAkB,OAAlB;AACA,OAAOC,UAAP,MAAuB,oBAAvB;AACA,SAASC,OAAT,EAAkBC,IAAlB,QAA8B,QAA9B;AACA,OAAOC,QAAP,MAAqB,UAArB;AACA,OAAOC,EAAP,MAAe,UAAf;AACA,OAAOC,MAAP,MAAmB,WAAnB;AACA,OAAOC,cAAP,MAA2B,wBAA3B;AACA,SAASC,gBAAT,QAAiC,uBAAjC;AACA;AACEC,cADF;AAEEC,eAFF;AAGEC,YAHF;;AAKO,SALP;AAMA;AACEC,cADF;;AAGEC,sBAHF;AAIO,aAJP;;AAMA,OAAOC,WAAP,MAAwB,cAAxB;AACA,SAASC,OAAT,QAAwB,WAAxB;AACA,OAAO,wBAAP;AACA,OAAOC,aAAP,MAA0B,uBAA1B;;AAEA;;;;;;;;;;;;;;;AAeA,MAAMC,SAAiC,GAAG;EACxCC,OAAO,EAAE,4BAD+B;EAExCC,QAAQ,EAAE,sDAF8B;EAGxCC,OAAO,EAAE,CAACC,KAAD;EACPA,KAAK;EACFC,UADH,CACc,qBADd,EACqC;IACjCC,IAAI,EAAE,QAD2B;IAEjCC,KAAK,EAAE,IAF0B,EADrC;;EAKGC,MALH,CAKU,WALV,EAKuB;IACnBC,KAAK,EAAE,GADY;IAEnBP,QAAQ,EAAG,yHAFQ,EALvB;;EASGQ,OATH,CASW,QATX,EASqB;IACjBR,QAAQ;IACN,6EAFe;IAGjBI,IAAI,EAAE,QAHW,EATrB;;EAcGI,OAdH,CAcW,eAdX,EAc4B;IACxBR,QAAQ,EAAE,oBADc;IAExBI,IAAI,EAAE,QAFkB,EAd5B;;EAkBGE,MAlBH,CAkBU,MAlBV,EAkBkB;IACdC,KAAK,EAAE,GADO;IAEdP,QAAQ,EAAE,gBAFI;IAGdI,IAAI,EAAE,QAHQ,EAlBlB;;EAuBGE,MAvBH,CAuBU,SAvBV,EAuBqB;IACjBC,KAAK,EAAE,GADU;IAEjBP,QAAQ,EAAE,iBAFO;IAGjBI,IAAI,EAAE,QAHW,EAvBrB;;EA4BGE,MA5BH,CA4BU,KA5BV,EA4BiB;IACbC,KAAK,EAAE,GADM;IAEbP,QAAQ,EAAG,mDAFE;IAGbI,IAAI,EAAE,SAHO,EA5BjB;;EAiCGE,MAjCH,CAiCU,WAjCV,EAiCuB;IACnBF,IAAI,EAAE,SADa;IAEnBJ,QAAQ,EAAG,yBAFQ;IAGnBS,OAAO,EAAE,IAHU,EAjCvB;;EAsCGH,MAtCH,CAsCU,SAtCV,EAsCqB;IACjBF,IAAI,EAAE,QADW;IAEjBJ,QAAQ,EAAE,iCAFO,EAtCrB;;EA0CGM,MA1CH,CA0CU,aA1CV,EA0CyB;IACrBF,IAAI,EAAE,SADe;IAErBJ,QAAQ,EAAE,iCAFW,EA1CzB,CAJsC;;;EAmDxCU,OAAO,EAAE,OAAOC,IAAP,KAAoC;IAC3C,MAAMC,SAAS,GAAGC,IAAI,CAACC,GAAL,EAAlB;;IAEA,MAAMC,YAAY,GAAG,MAAMxB,eAAe,CAACyB,MAAhB,EAA3B;IACA,IAAIL,IAAI,CAACM,WAAT,EAAsB;MACpBC,OAAO,CAACC,GAAR,CAAYC,IAAI,CAACC,SAAL,CAAeN,YAAf,EAA6B,IAA7B,EAAmC,CAAnC,CAAZ;MACAO,OAAO,CAACC,IAAR,CAAa,CAAb;IACD;IACD,MAAMC,MAAM,GAAGT,YAAH,aAAGA,YAAH,uBAAGA,YAAY,CAAES,MAA7B;;IAEA,MAAMC,KAAK,GAAG,CAACd,IAAI,CAACe,mBAAL,IAA4B,EAA7B,EAAiCC,MAAjC;IACZ,CAACC,CAAD,KAAO,OAAOA,CAAP,KAAa,QADR,CAAd;;;IAIA,MAAM,EAAE9B,SAAF,EAAa+B,aAAb,KAA+B,MAAM,CAAC;;;IAGtC;MACJ,IAAIlB,IAAI,CAACb,SAAT,EAAoB;QAClB,MAAM+B,aAAa,GAAGjD,IAAI,CAACkD,OAAL,CAAanB,IAAI,CAACb,SAAlB,CAAtB;QACA,OAAO;UACL+B,aADK;UAEL/B,SAAS,EAAE,MAAM,OAAO+B,aAAP,CAFZ,EAAP;;MAID,CAND,MAMO,IAAIlB,IAAI,CAACoB,IAAT,EAAe;QACpB,MAAMC,MAAM,GAAG,CAACC,KAAD,KAAuC;UACpD,MAAMC,KAAK,GAAGZ,OAAO,CAACX,IAAR,CAAawB,SAAb,CAAuB,CAACC,CAAD,KAAOH,KAAK,CAACI,IAAN,CAAWD,CAAX,CAA9B,CAAd;UACA,OAAOF,KAAK,IAAI,CAAT,GAAaZ,OAAO,CAACX,IAAR,CAAauB,KAAK,GAAG,CAArB,CAAb,GAAuCI,SAA9C;QACD,CAHD;QAIA;QACA,MAAMP,IAAI,GAAGC,MAAM,CAAC,eAAD,CAAnB;QACA,MAAMO,OAAO,GAAGP,MAAM,CAAC,kBAAD,CAAtB;QACA,OAAO,EAAElC,SAAS,EAAE,EAAEiC,IAAF,EAAQQ,OAAR,EAAb,EAAP;MACD,CATM,MASA;QACL,MAAMC,KAAK,GAAG,CAAC5D,IAAI,CAACkD,OAAL,CAAa,SAAb,CAAD,EAA0BlD,IAAI,CAACkD,OAAL,CAAa,SAAb,CAA1B,CAAd;QACA,KAAK,MAAMD,aAAX,IAA4BW,KAA5B,EAAmC;UACjC,IAAI,MAAMtD,EAAE,CAACuD,UAAH,CAAcZ,aAAd,CAAV,EAAwC;YACtC,OAAO,EAAEA,aAAF,EAAiB/B,SAAS,EAAE,MAAM,OAAO+B,aAAP,CAAlC,EAAP;UACD;QACF;QACD,MAAM,IAAIa,KAAJ,CAAW,2BAA0BF,KAAK,CAACG,IAAN,CAAW,MAAX,CAAmB,EAAxD,CAAN;MACD;IACF,CA5B0C,GAA3C;IA6BA,MAAM,EAAEC,MAAF,EAAUC,aAAV,EAAyBC,SAAzB,KAAuCnC,IAA7C;;IAEA,MAAMoC,OAA+B,GAAG,EAAxC;IACA,IAAIC,UAAU,GAAG,CAAjB;IACA,IAAIC,YAAY,GAAG,CAAnB;IACA,IAAIC,cAAc,GAAG,CAArB;;IAEA,IAAIC,QAAkB,GAAG;MACvB/C,IAAI,EAAE,UADiB;MAEvBgD,SAAS,EAAE,CAFY;MAGvBC,KAAK,EAAE,CAHgB;MAIvBC,QAAQ,EAAE,KAJa,EAAzB;;;IAOA,IAAIC,iBAAiB,GAAG,KAAxB;IACA,SAASC,aAAT,GAAyB;MACvB,IAAID,iBAAJ,EAAuB;QACrBjC,OAAO,CAACmC,MAAR,CAAeC,KAAf,CAAqB/D,WAAW,CAACgE,UAAZ,GAAyBhE,WAAW,CAACiE,SAA1D;QACAL,iBAAiB,GAAG,KAApB;MACD;IACF;IACD,SAASM,YAAT,GAAwB;MACtBL,aAAa;MACbD,iBAAiB,GAAG,IAApB;MACA,MAAM,EAAEH,SAAF,EAAaC,KAAb,EAAoBC,QAApB,KAAiCH,QAAvC;MACA7B,OAAO,CAACmC,MAAR,CAAeC,KAAf;MACE7E,KAAK,CAACiF,OAAN;MACG,GAAElE,OAAO,EAAG,eAAcwD,SAAU,IAAGC,KAAM;MAC5CC,QAAQ,IAAID,KAAZ;MACK,KAAI,CAAED,SAAS,GAAG,GAAb,GAAoBC,KAArB,EAA4BU,OAA5B,CAAoC,CAApC,CAAuC,IADhD;MAEI;MACL,IAAG,CAAC,CAAClD,IAAI,CAACC,GAAL,KAAaF,SAAd,IAA2B,IAA5B,EAAkCmD,OAAlC,CAA0C,CAA1C,CAA6C,GALnD,CADF;;;IASD;IACD,IAAIC,eAAJ;;IAEA,MAAMC,WAAW,GAAGpE,aAAa,EAAjC;IACA,MAAMqE,OAAO;IACXvD,IAAI,CAACuD,OADM;IAEV5C,OAAO,CAAC6C,GAAR,CAAYC,YAAZ;IACGC,QAAQ,CAAC/C,OAAO,CAAC6C,GAAR,CAAYC,YAAb,CADX;IAEG9B,SAJO;IAKXd,MALW,aAKXA,MALW,uBAKXA,MAAM,CAAE0C,OALV;IAMA,MAAMI,IAAI;IACRJ,OAAO,KAAK,CAAZ,GAAgB,IAAhB,GAAuB,IAAI5E,cAAJ,CAAmB,EAAEiF,QAAQ,EAAEL,OAAZ,EAAnB,CADzB;IAEA,IAAI;MACF,IAAID,WAAJ,EAAiB;QACfD,eAAe,GAAGQ,WAAW,CAACX,YAAD,EAAe,EAAf,CAA7B;MACD;MACD,MAAMY,mBAAmB,GAAG;QAC1B3B,SAAS,EAAEA,SAAS,GAAGR,SAAH,GAAe,IADT;QAE1BxC,SAF0B;QAG1B+B,aAH0B;QAI1BJ,KAJ0B;QAK1BD,MAAM,EAAE;UACNoB,MAAM,EAAEA,MADF;UAENC,aAAa,EAAEA,aAAa,GAAGzB,IAAI,CAACsD,KAAL,CAAW7B,aAAX,CAAH,GAA+BP,SAFrD,EALkB,EAA5B;;;MAUA,WAAW,MAAMqC,MAAjB,IAA2BL,IAAI;MAC3BA,IAAI,CAAC9E,YAAL,CAAkBiF,mBAAlB,CAD2B;MAE3BjF,YAAY,CAACiF,mBAAD,CAFhB,EAEuC;QACrC,MAAMG,KAAgE;QACpE,CAACN,IAAD,IAASK,MAAM,CAACvE,IAAP,KAAgB,QAAzB;QACI,EAAEA,IAAI,EAAE,QAAR,EAAkByE,MAAM,EAAEnF,sBAAsB,CAACiF,MAAD,CAAhD,EADJ;QAEKA,MAHP;QAIA,IAAIC,KAAK,CAACxE,IAAN,KAAe,UAAnB,EAA+B;UAC7B+C,QAAQ,GAAGyB,KAAX;UACA,IAAIX,WAAJ,EAAiBJ,YAAY;UAC7B;QACD;QACDL,aAAa;QACb,MAAM;UACJsB,IADI;UAEJC,MAFI;UAGJC,WAHI;UAIJC,OAJI;UAKJC,OALI;UAMJC,KAAK,EAAEC,MANH;QAOFR,KAAK,CAACC,MAPV;QAQA,MAAMM,KAAK,GAAGC,MAAM,GAAG3F,cAAc,CAAC2F,MAAD,CAAjB,GAA4B9C,SAAhD;QACA,MAAM+C,OAAO,GAAGzG,IAAI,CAAC0G,QAAL,CAAchE,OAAO,CAACiE,GAAR,EAAd,EAA6BT,IAA7B,CAAhB;QACA,MAAMU,SAAS,GAAGxG,IAAI,CAAC,CAACyG,KAAD;QACrBA,KAAK;QACH5G,KAAK,CAAC6G,IAAN,CAAWvG,MAAO;AAC9B,cAAc,IAAIwG,MAAJ,CAAWN,OAAO,CAACO,MAAnB,CAA2B;AACzC,cAAc/G,KAAK,CAACgH,IAAN,CAAWR,OAAX,CAAoB;AAClC,cAAc,IAAIM,MAAJ,CAAWN,OAAO,CAACO,MAAnB,CAA2B;AACzC,WAJY,CADG,CADe,CAAtB;;;;QAUA,IAAIT,KAAJ,EAAW;UACTnC,UAAU;UACVwC,SAAS,CAACtE,OAAO,CAACiE,KAAT,CAAT;UACA,IAAIA,KAAK,YAAY/F,cAArB,EAAqC;YACnC8B,OAAO,CAACiE,KAAR;YACEA,KAAK,CAACW,MAAN,CAAa;cACXC,aAAa,EAAE,IADJ;cAEXC,UAAU,EAAE,IAFD;cAGXC,KAAK,EAAE,IAHI,EAAb,CADF;;;UAOD,CARD,MAQO;YACL/E,OAAO,CAACiE,KAAR,CAActG,KAAK,CAACqH,GAAN,CAAUf,KAAK,CAACc,KAAhB,CAAd;UACD;QACF,CAdD,MAcO,IAAIlB,MAAM,IAAIC,WAAV,IAAyBD,MAAM,KAAKC,WAAxC,EAAqD;UAC1D/B,YAAY;UACZF,OAAO,CAAC+B,IAAD,CAAP,GAAgBE,WAAhB;UACA,IAAI,CAACrE,IAAI,CAACwF,GAAV,EAAe;YACbX,SAAS,CAACtE,OAAO,CAACC,GAAT,CAAT;YACAD,OAAO,CAACC,GAAR,CAAYrC,UAAU,CAACiG,MAAD,EAASC,WAAT,CAAtB;UACD;QACF,CAPM,MAOA;QACLE,OAAO,SAAP,IAAAA,OAAO,WAAP,IAAAA,OAAO,CAAEU,MAAT;QACAb,MADA;QAEAjF,SAAS,CAACiC,IAFV;QAGA,CAACjC,SAAS,CAACyC,OAHX;QAIA,CAACzC,SAAS,CAACsG,IALN;QAML;UACAZ,SAAS,CAACtE,OAAO,CAACC,GAAT,CAAT;UACAD,OAAO,CAACC,GAAR,CAAY9B,gBAAgB,CAAC0F,MAAD,EAASG,OAAT,CAA5B;QACD,CATM,MASA;UACLhC,cAAc;QACf;;QAED,IAAI+B,OAAO,SAAP,IAAAA,OAAO,WAAP,IAAAA,OAAO,CAAEW,MAAT,IAAmB,CAAC9F,SAAS,CAACuG,QAAlC,EAA4C;UAC1Cb,SAAS,CAACtE,OAAO,CAACiE,KAAT,CAAT;UACAjE,OAAO,CAACiE,KAAR;UACEtG,KAAK,CAAC6G,IAAN,CAAWvG,MAAO;AAC9B;AACA;AACA,WAHY,CADF;;UAMA8F,OAAO,SAAP,IAAAA,OAAO,WAAP,YAAAA,OAAO,CAAEqB,OAAT,CAAiB,CAACC,CAAD,KAAYrF,OAAO,CAACiE,KAAR,CAAcoB,CAAd,CAA7B;QACD;MACF;IACF,CA5FD,CA4FE,OAAOpB,KAAP,EAAc;MACdjE,OAAO,CAACiE,KAAR;MACEtG,KAAK,CAACqH,GAAN,CAAUf,KAAK,YAAYzC,KAAjB,GAAyByC,KAAK,CAACc,KAA/B,GAAuCO,MAAM,CAACrB,KAAD,CAAvD,CADF;;MAGA7D,OAAO,CAACC,IAAR,CAAa,CAAb;IACD,CAjGD,SAiGU;MACR,IAAIyC,eAAe,IAAI,IAAvB,EAA6ByC,aAAa,CAACzC,eAAD,CAAb;MAC7BR,aAAa;IACd;;IAED,IAAI1D,SAAS,CAACyC,OAAV,IAAqBzC,SAAS,CAACsG,IAAnC,EAAyC;MACvClF,OAAO,CAACiE,KAAR;MACEtG,KAAK,CAAC6H,MAAN;MACG,GAAEzD,YAAa,QAAOA,YAAY,KAAK,CAAjB,GAAqB,EAArB,GAA0B,GAAI,UADvD,CADF;;;MAKA/B,OAAO,CAACiE,KAAR;MACEtG,KAAK,CAAC8H,KAAN;MACG,GAAEzD,cAAe,QAAOA,cAAc,KAAK,CAAnB,GAAuB,EAAvB,GAA4B,GAAI,YAD3D,CADF;;;MAKA,IAAIF,UAAU,GAAG,CAAjB,EAAoB;QAClB9B,OAAO,CAACiE,KAAR;QACEtG,KAAK,CAACqH,GAAN,CAAW,GAAElD,UAAW,QAAOA,UAAU,KAAK,CAAf,GAAmB,EAAnB,GAAwB,GAAI,UAA3D,CADF;;MAGD;IACF,CAhBD,MAgBO,IAAIlD,SAAS,CAACiC,IAAd,EAAoB;MACzBb,OAAO,CAACiE,KAAR;MACEtG,KAAK,CAAC6H,MAAN;MACG,KAAIxD,cAAe;MAClBD,YAAY,KAAK,CAAjB,GAAqB,EAArB,GAA0B;MAC3B,iBAHH,CADF;;;IAOD;;IAED,IAAI,CAAClE,OAAO,CAACgE,OAAD,CAAZ,EAAuB;MACrB,MAAM6D,KAAK,GAAGjG,IAAI,CAACwF,GAAL;MACV,IADU;MAEV;MACE,MAAMlH,QAAQ,CAAC4H,MAAT,CAAgB;MACpB;QACEzG,IAAI,EAAE,SADR;QAEE0G,IAAI,EAAE,OAFR;QAGEC,OAAO,EAAE,eAHX;QAIEtG,OAAO,EAAE,KAJX,EADoB,CAAhB,CADR;;;MASEmG,KAXN;MAYA,IAAIA,KAAJ,EAAW;QACT,KAAK,MAAM9B,IAAX,IAAmB/B,OAAnB,EAA4B;UAC1B,MAAM7D,EAAE,CAAC8H,SAAH,CAAalC,IAAb,EAAmB/B,OAAO,CAAC+B,IAAD,CAA1B,EAAkC,MAAlC,CAAN;UACA5D,OAAO,CAACiE,KAAR,CAAe,SAAQL,IAAK,EAA5B;QACD;MACF;MACD,IAAIxD,OAAO,CAAC2F,IAAZ,EAAkB3F,OAAO,CAAC2F,IAAR,CAAa,EAAE1F,IAAI,EAAE,CAAR,EAAb;IACnB;IACD,OAAM+C,IAAN,aAAMA,IAAN,uBAAMA,IAAI,CAAE4C,GAAN,EAAN;IACA5F,OAAO,CAACC,IAAR,CAAa,CAAb;EACD,CAnSuC,EAA1C;;;AAsSA,eAAezB,SAAf"}