@formatjs/cli-lib 8.6.0 → 8.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"file":"gts_extractor-Dc-C_f4R.js","names":[],"sources":["../gts_extractor.ts"],"sourcesContent":["import {Preprocessor} from 'content-tag'\nimport {parseFile as parseHbsFile} from '#packages/cli-lib/hbs_extractor.js'\nimport {parseScript} from '#packages/cli-lib/parse_script.js'\nlet p = new Preprocessor()\n\nexport function parseFile(\n source: string,\n fileName: string,\n options: any\n): void {\n const scriptParseFn = parseScript(options, fileName)\n const transformedSource = p.process(source, {filename: fileName})\n\n scriptParseFn(transformedSource.code)\n\n // extract template from transformed source to then run through hbs processor\n const parseResult = p.parse(source, {filename: fileName})\n\n for (let parsed of parseResult) {\n parseHbsFile(parsed.contents, fileName, options)\n }\n}\n"],"mappings":";;;;AAGA,IAAI,IAAI,IAAI,cAAc;AAE1B,SAAgB,UACd,QACA,UACA,SACM;AACgB,aAAY,SAAS,SAAS,CAC1B,EAAE,QAAQ,QAAQ,EAAC,UAAU,UAAS,CAAC,CAEjC,KAAK;CAGrC,MAAM,cAAc,EAAE,MAAM,QAAQ,EAAC,UAAU,UAAS,CAAC;AAEzD,MAAK,IAAI,UAAU,YACjB,aAAa,OAAO,UAAU,UAAU,QAAQ"}
1
+ {"version":3,"file":"gts_extractor-Dc-C_f4R.js","names":[],"sources":["../gts_extractor.ts"],"sourcesContent":["import {Preprocessor} from 'content-tag'\nimport {parseFile as parseHbsFile} from '#packages/cli-lib/hbs_extractor.js'\nimport {parseScript} from '#packages/cli-lib/parse_script.js'\nlet p = new Preprocessor()\n\nexport function parseFile(\n source: string,\n fileName: string,\n options: any\n): void {\n const scriptParseFn = parseScript(options, fileName)\n const transformedSource = p.process(source, {filename: fileName})\n\n scriptParseFn(transformedSource.code)\n\n // extract template from transformed source to then run through hbs processor\n const parseResult = p.parse(source, {filename: fileName})\n\n for (let parsed of parseResult) {\n parseHbsFile(parsed.contents, fileName, options)\n }\n}\n"],"mappings":";;;;AAGA,IAAI,IAAI,IAAI,cAAc;AAE1B,SAAgB,UACd,QACA,UACA,SACM;CAIN,YAHkC,SAAS,SAG9B,CAFa,EAAE,QAAQ,QAAQ,EAAC,UAAU,UAAS,CAEjC,CAAC,KAAK;CAGrC,MAAM,cAAc,EAAE,MAAM,QAAQ,EAAC,UAAU,UAAS,CAAC;CAEzD,KAAK,IAAI,UAAU,aACjB,YAAa,OAAO,UAAU,UAAU,QAAQ"}
@@ -1 +1 @@
1
- {"version":3,"file":"hbs_extractor-Eiqkz4d7.js","names":[],"sources":["../hbs_extractor.ts"],"sourcesContent":["import {type Opts} from '@formatjs/ts-transformer'\nimport {preprocess, traverse, type AST, type ASTv1} from '@glimmer/syntax'\n\nfunction extractText(\n node: AST.MustacheStatement | AST.SubExpression,\n fileName: string,\n options: Opts\n) {\n if (!options.onMsgExtracted) return\n if (!options.overrideIdFn) return\n\n if (node.path.type !== 'PathExpression') return\n\n if (['format-message', 'formatMessage'].includes(node.path.original)) {\n let [first, second] = node.params\n\n if (first.type !== 'StringLiteral') return\n\n let message = first?.value\n\n let desc: string | undefined\n if (second?.type === 'StringLiteral') {\n desc = second.value?.trim().replace(/\\s+/gm, ' ')\n }\n\n let defaultMessage = message?.trim().replace(/\\s+/gm, ' ')\n\n let id =\n typeof options.overrideIdFn === 'string'\n ? options.overrideIdFn\n : options.overrideIdFn(undefined, defaultMessage, desc, fileName)\n\n options.onMsgExtracted(fileName, [\n {\n id: id,\n defaultMessage: defaultMessage,\n description: desc,\n },\n ])\n }\n}\n\nexport function parseFile(\n source: string,\n fileName: string,\n options: any\n): void {\n const ast = preprocess(source)\n traverse(ast, {\n MustacheStatement(node: ASTv1.MustacheStatement) {\n extractText(node, fileName, options)\n },\n SubExpression(node: ASTv1.SubExpression) {\n extractText(node, fileName, options)\n },\n })\n}\n"],"mappings":";;AAGA,SAAS,YACP,MACA,UACA,SACA;AACA,KAAI,CAAC,QAAQ,eAAgB;AAC7B,KAAI,CAAC,QAAQ,aAAc;AAE3B,KAAI,KAAK,KAAK,SAAS,iBAAkB;AAEzC,KAAI,CAAC,kBAAkB,gBAAgB,CAAC,SAAS,KAAK,KAAK,SAAS,EAAE;EACpE,IAAI,CAAC,OAAO,UAAU,KAAK;AAE3B,MAAI,MAAM,SAAS,gBAAiB;EAEpC,IAAI,UAAU,OAAO;EAErB,IAAI;AACJ,MAAI,QAAQ,SAAS,gBACnB,QAAO,OAAO,OAAO,MAAM,CAAC,QAAQ,SAAS,IAAI;EAGnD,IAAI,iBAAiB,SAAS,MAAM,CAAC,QAAQ,SAAS,IAAI;EAE1D,IAAI,KACF,OAAO,QAAQ,iBAAiB,WAC5B,QAAQ,eACR,QAAQ,aAAa,KAAA,GAAW,gBAAgB,MAAM,SAAS;AAErE,UAAQ,eAAe,UAAU,CAC/B;GACM;GACY;GAChB,aAAa;GACd,CACF,CAAC;;;AAIN,SAAgB,UACd,QACA,UACA,SACM;AAEN,UADY,WAAW,OAAO,EAChB;EACZ,kBAAkB,MAA+B;AAC/C,eAAY,MAAM,UAAU,QAAQ;;EAEtC,cAAc,MAA2B;AACvC,eAAY,MAAM,UAAU,QAAQ;;EAEvC,CAAC"}
1
+ {"version":3,"file":"hbs_extractor-Eiqkz4d7.js","names":[],"sources":["../hbs_extractor.ts"],"sourcesContent":["import {type Opts} from '@formatjs/ts-transformer'\nimport {preprocess, traverse, type AST, type ASTv1} from '@glimmer/syntax'\n\nfunction extractText(\n node: AST.MustacheStatement | AST.SubExpression,\n fileName: string,\n options: Opts\n) {\n if (!options.onMsgExtracted) return\n if (!options.overrideIdFn) return\n\n if (node.path.type !== 'PathExpression') return\n\n if (['format-message', 'formatMessage'].includes(node.path.original)) {\n let [first, second] = node.params\n\n if (first.type !== 'StringLiteral') return\n\n let message = first?.value\n\n let desc: string | undefined\n if (second?.type === 'StringLiteral') {\n desc = second.value?.trim().replace(/\\s+/gm, ' ')\n }\n\n let defaultMessage = message?.trim().replace(/\\s+/gm, ' ')\n\n let id =\n typeof options.overrideIdFn === 'string'\n ? options.overrideIdFn\n : options.overrideIdFn(undefined, defaultMessage, desc, fileName)\n\n options.onMsgExtracted(fileName, [\n {\n id: id,\n defaultMessage: defaultMessage,\n description: desc,\n },\n ])\n }\n}\n\nexport function parseFile(\n source: string,\n fileName: string,\n options: any\n): void {\n const ast = preprocess(source)\n traverse(ast, {\n MustacheStatement(node: ASTv1.MustacheStatement) {\n extractText(node, fileName, options)\n },\n SubExpression(node: ASTv1.SubExpression) {\n extractText(node, fileName, options)\n },\n })\n}\n"],"mappings":";;AAGA,SAAS,YACP,MACA,UACA,SACA;CACA,IAAI,CAAC,QAAQ,gBAAgB;CAC7B,IAAI,CAAC,QAAQ,cAAc;CAE3B,IAAI,KAAK,KAAK,SAAS,kBAAkB;CAEzC,IAAI,CAAC,kBAAkB,gBAAgB,CAAC,SAAS,KAAK,KAAK,SAAS,EAAE;EACpE,IAAI,CAAC,OAAO,UAAU,KAAK;EAE3B,IAAI,MAAM,SAAS,iBAAiB;EAEpC,IAAI,UAAU,OAAO;EAErB,IAAI;EACJ,IAAI,QAAQ,SAAS,iBACnB,OAAO,OAAO,OAAO,MAAM,CAAC,QAAQ,SAAS,IAAI;EAGnD,IAAI,iBAAiB,SAAS,MAAM,CAAC,QAAQ,SAAS,IAAI;EAE1D,IAAI,KACF,OAAO,QAAQ,iBAAiB,WAC5B,QAAQ,eACR,QAAQ,aAAa,KAAA,GAAW,gBAAgB,MAAM,SAAS;EAErE,QAAQ,eAAe,UAAU,CAC/B;GACM;GACY;GAChB,aAAa;GACd,CACF,CAAC;;;AAIN,SAAgB,UACd,QACA,UACA,SACM;CAEN,SADY,WAAW,OACX,EAAE;EACZ,kBAAkB,MAA+B;GAC/C,YAAY,MAAM,UAAU,QAAQ;;EAEtC,cAAc,MAA2B;GACvC,YAAY,MAAM,UAAU,QAAQ;;EAEvC,CAAC"}
package/index.js CHANGED
@@ -326,7 +326,9 @@ async function extractAndWrite(files, extractOpts) {
326
326
  const require$1 = createRequire(import.meta.url);
327
327
  const NATIVE_PACKAGES = {
328
328
  "darwin-arm64": "@formatjs/cli-native-darwin-arm64",
329
- "linux-x64": "@formatjs/cli-native-linux-x64"
329
+ "linux-arm64": "@formatjs/cli-native-linux-arm64",
330
+ "linux-x64": "@formatjs/cli-native-linux-x64",
331
+ "win32-x64": "@formatjs/cli-native-win32-x64"
330
332
  };
331
333
  let nativeBinding;
332
334
  function loadNative() {
package/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":["format","compile","format","compile","format","compile","format","compile","format","compile","compile","defaultFormatter","transifex","smartling","simple","lokalise","crowdin","stringify","require"],"sources":["../formatters/crowdin.ts","../formatters/default.ts","../formatters/lokalise.ts","../formatters/simple.ts","../formatters/smartling.ts","../formatters/transifex.ts","../formatters/index.ts","../extract.ts","../native.ts","../compile.ts"],"sourcesContent":["import {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport type CrowdinJson = Record<\n string,\n {\n message: string\n description?: string\n }\n>\n\nexport const format: FormatFn<CrowdinJson> = msgs => {\n const results: CrowdinJson = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = {\n message: msg.defaultMessage!,\n description:\n typeof msg.description === 'string'\n ? msg.description\n : JSON.stringify(msg.description),\n }\n }\n return results\n}\n\nexport const compile: CompileFn<CrowdinJson> = msgs => {\n const results: Record<string, string> = {}\n for (const [id, msg] of Object.entries(msgs)) {\n if (id === 'smartling') {\n continue\n }\n results[id] = msg.message\n }\n return results\n}\n","import {type MessageDescriptor} from '@formatjs/ts-transformer'\nexport type FormatFn<T = Record<string, MessageDescriptor>> = (\n msgs: Record<string, MessageDescriptor>\n) => T\n\nexport type CompileFn<T = Record<string, MessageDescriptor>> = (\n msgs: T\n) => Record<string, string>\n\nexport type SerializeFn<T = Record<string, MessageDescriptor>> = (\n msgs: T\n) => string\n\nexport const format: FormatFn = msgs => msgs\n\nexport const compile: CompileFn = msgs => {\n const results: Record<string, string> = {}\n for (const k in msgs) {\n results[k] = msgs[k].defaultMessage!\n }\n return results\n}\n","import {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport type StructuredJson = Record<\n string,\n {\n translation: string\n notes?: string\n context?: string\n limit?: string\n }\n>\n\nexport const format: FormatFn<StructuredJson> = msgs => {\n const results: StructuredJson = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = {\n translation: msg.defaultMessage!,\n notes:\n typeof msg.description === 'string'\n ? msg.description\n : JSON.stringify(msg.description),\n }\n }\n return results\n}\n\nexport const compile: CompileFn<StructuredJson> = msgs => {\n const results: Record<string, string> = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = msg.translation\n }\n return results\n}\n","import {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport type PhraseJson = Record<string, string>\n\nexport const format: FormatFn<PhraseJson> = msgs => {\n return Object.keys(msgs).reduce((all: PhraseJson, k) => {\n all[k] = msgs[k].defaultMessage!\n return all\n }, {})\n}\n\nexport const compile: CompileFn<PhraseJson> = msgs => msgs\n","import type {Comparator} from 'json-stable-stringify'\nimport {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport interface SmartlingDirectives {\n translate_paths: [\n {\n path: string\n key: string\n instruction: string\n },\n ]\n variants_enabled: boolean\n string_format: string\n [k: string]: any\n}\n\nexport type SmartlingJson = {\n smartling: SmartlingDirectives\n} & Record<\n string,\n {\n message: string\n description?: string\n }\n>\n\nexport const format: FormatFn<SmartlingJson> = msgs => {\n const results: SmartlingJson = {\n smartling: {\n translate_paths: [\n {\n path: '*/message',\n key: '{*}/message',\n instruction: '*/description',\n },\n ],\n variants_enabled: true,\n string_format: 'icu',\n },\n } as any\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = {\n message: msg.defaultMessage!,\n description:\n typeof msg.description === 'string'\n ? msg.description\n : JSON.stringify(msg.description),\n }\n }\n return results\n}\n\nexport const compareMessages: Comparator = (el1, el2) => {\n // `smartling` has to be the 1st key\n if (el1.key === 'smartling') {\n return -1\n }\n if (el2.key === 'smartling') {\n return 1\n }\n return el1.key < el2.key ? -1 : el1.key === el2.key ? 0 : 1\n}\n\nexport const compile: CompileFn<SmartlingJson> = msgs => {\n const results: Record<string, string> = {}\n for (const [id, msg] of Object.entries(msgs)) {\n if (id === 'smartling') {\n continue\n }\n results[id] = msg.message\n }\n return results\n}\n","import {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport type StructuredJson = Record<\n string,\n {\n string: string\n developer_comment?: string\n context?: string\n character_limit?: string\n }\n>\n\nexport const format: FormatFn<StructuredJson> = msgs => {\n const results: StructuredJson = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = {\n string: msg.defaultMessage!,\n developer_comment:\n typeof msg.description === 'string'\n ? msg.description\n : JSON.stringify(msg.description),\n }\n }\n return results\n}\n\nexport const compile: CompileFn<StructuredJson> = msgs => {\n const results: Record<string, string> = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = msg.string\n }\n return results\n}\n","import type {Comparator} from 'json-stable-stringify'\nimport {resolve} from 'path'\nimport {pathToFileURL} from 'url'\nimport * as crowdin from '#packages/cli-lib/formatters/crowdin.js'\nimport * as defaultFormatter from '#packages/cli-lib/formatters/default.js'\nimport {\n type CompileFn,\n type FormatFn,\n type SerializeFn,\n} from '#packages/cli-lib/formatters/default.js'\nimport * as lokalise from '#packages/cli-lib/formatters/lokalise.js'\nimport * as simple from '#packages/cli-lib/formatters/simple.js'\nimport * as smartling from '#packages/cli-lib/formatters/smartling.js'\nimport * as transifex from '#packages/cli-lib/formatters/transifex.js'\n\nexport interface Formatter<T> {\n serialize?: SerializeFn<T>\n format: FormatFn<T>\n compile: CompileFn<T>\n compareMessages?: Comparator\n}\n\nexport async function resolveBuiltinFormatter(\n format?: string | Formatter<unknown>\n): Promise<any> {\n if (!format) {\n return defaultFormatter\n }\n if (typeof format !== 'string') {\n return format\n }\n switch (format) {\n case 'transifex':\n return transifex\n case 'smartling':\n return smartling\n case 'simple':\n return simple\n case 'lokalise':\n return lokalise\n case 'crowdin':\n return crowdin\n }\n try {\n // eslint-disable-next-line import/dynamic-import-chunkname\n return import(pathToFileURL(resolve(process.cwd(), format)).href)\n } catch (e) {\n console.error(`Cannot resolve formatter ${format}`)\n throw e\n }\n}\n","import {\n type MessageDescriptor,\n type Opts,\n interpolateName,\n} from '@formatjs/ts-transformer'\nimport {outputFile} from 'fs-extra/esm'\nimport {\n debug,\n getStdinAsString,\n warn,\n writeStdout,\n} from '#packages/cli-lib/console_utils.js'\nimport * as stringifyNs from 'json-stable-stringify'\n\nimport {\n type Formatter,\n resolveBuiltinFormatter,\n} from '#packages/cli-lib/formatters/index.js'\nimport {parseScript} from '#packages/cli-lib/parse_script.js'\nimport {readFile} from 'fs/promises'\n\nconst stringify = (stringifyNs as any).default || stringifyNs\nexport interface ExtractionResult<M = Record<string, string>> {\n /**\n * List of extracted messages\n */\n messages: MessageDescriptor[]\n /**\n * Metadata extracted w/ `pragma`\n */\n meta?: M\n}\n\nexport interface ExtractedMessageDescriptor extends MessageDescriptor {\n /**\n * Line number\n */\n line?: number\n /**\n * Column number\n */\n col?: number\n /**\n * Metadata extracted from pragma\n */\n meta?: Record<string, string>\n}\n\nexport type ExtractCLIOptions = Omit<\n ExtractOpts,\n 'overrideIdFn' | 'onMsgExtracted' | 'onMetaExtracted'\n> & {\n /**\n * Output File\n */\n outFile?: string\n /**\n * Input File\n */\n inFile?: string\n /**\n * Ignore file glob pattern\n */\n ignore?: string[]\n /**\n * Whether to follow symbolic links when traversing directories.\n * Defaults to true for compatibility with pnpm symlinked node_modules.\n */\n followLinks?: boolean\n}\n\nexport type ExtractOpts = Opts & {\n /**\n * Whether to throw an error if we had any issues with\n * 1 of the source files\n */\n throws?: boolean\n /**\n * Message ID interpolation pattern\n */\n idInterpolationPattern?: string\n /**\n * Whether we read from stdin instead of a file\n */\n readFromStdin?: boolean\n /**\n * Either path to a formatter file that controls the shape of JSON file from `outFile` or {@link Formatter} object.\n */\n format?: string | Formatter<any>\n /**\n * Whether to hoist selectors & flatten sentences\n */\n flatten?: boolean\n /**\n * An AbortSignal to cancel the extraction\n */\n signal?: AbortSignal\n} & Pick<Opts, 'onMsgExtracted' | 'onMetaExtracted'>\n\nfunction calculateLineColFromOffset(\n text: string,\n start?: number\n): Pick<ExtractedMessageDescriptor, 'line' | 'col'> {\n if (!start) {\n return {line: 1, col: 1}\n }\n const chunk = text.slice(0, start)\n const lines = chunk.split('\\n')\n const lastLine = lines[lines.length - 1]\n return {line: lines.length, col: lastLine.length}\n}\n\nasync function processFile(\n source: string,\n fn: string,\n {idInterpolationPattern, ...opts}: Opts & {idInterpolationPattern?: string}\n) {\n let messages: ExtractedMessageDescriptor[] = []\n let meta: Record<string, string> | undefined\n\n const onMsgExtracted = opts.onMsgExtracted\n const onMetaExtracted = opts.onMetaExtracted\n\n opts = {\n ...opts,\n additionalComponentNames: [\n '$formatMessage',\n ...(opts.additionalComponentNames || []),\n ],\n onMsgExtracted(filePath, msgs) {\n if (opts.extractSourceLocation) {\n msgs = msgs.map(msg => ({\n ...msg,\n ...calculateLineColFromOffset(source, msg.start),\n }))\n }\n messages = messages.concat(msgs)\n\n if (onMsgExtracted) {\n onMsgExtracted(filePath, msgs)\n }\n },\n onMetaExtracted(filePath, m) {\n meta = m\n\n if (onMetaExtracted) {\n onMetaExtracted(filePath, m)\n }\n },\n }\n\n if (!opts.overrideIdFn && idInterpolationPattern) {\n opts = {\n ...opts,\n overrideIdFn: (id, defaultMessage, description, fileName) =>\n id ||\n interpolateName(\n {\n resourcePath: fileName,\n } as any,\n idInterpolationPattern,\n {\n content: description\n ? `${defaultMessage}#${\n typeof description === 'string'\n ? description\n : stringify(description)\n }`\n : defaultMessage,\n }\n ),\n }\n }\n\n debug('Processing opts for %s: %s', fn, opts)\n\n const scriptParseFn = parseScript(opts, fn)\n if (fn.endsWith('.vue')) {\n debug('Processing %s using vue extractor', fn)\n const {parseFile} = await import('./vue_extractor.js')\n parseFile(source, fn, scriptParseFn)\n } else if (fn.endsWith('.svelte')) {\n debug('Processing %s using svelte extractor', fn)\n const {parseFile} = await import('./svelte_extractor.js')\n parseFile(source, fn, scriptParseFn)\n } else if (fn.endsWith('.hbs')) {\n debug('Processing %s using hbs extractor', fn)\n const {parseFile} = await import('./hbs_extractor.js')\n parseFile(source, fn, opts)\n } else if (fn.endsWith('.gts') || fn.endsWith('.gjs')) {\n debug('Processing %s as gts/gjs file', fn)\n const {parseFile} = await import('./gts_extractor.js')\n parseFile(source, fn, opts)\n } else {\n debug('Processing %s using typescript extractor', fn)\n scriptParseFn(source)\n }\n debug('Done extracting %s messages: %s', fn, messages)\n if (meta) {\n debug('Extracted meta:', meta)\n messages.forEach(m => (m.meta = meta))\n }\n return {messages, meta}\n}\n\n/**\n * Extract strings from source files\n * @param files list of files\n * @param extractOpts extract options\n * @returns messages serialized as JSON string since key order\n * matters for some `format`\n */\nexport async function extract(\n files: readonly string[],\n extractOpts: ExtractOpts\n): Promise<string> {\n const {throws, readFromStdin, signal, ...opts} = extractOpts\n // When throws is not explicitly true, we want to collect partial results\n const shouldThrow = throws === true\n // Pass throws option to transformer for per-message error handling\n const optsWithThrows = {\n ...opts,\n throws: shouldThrow,\n onMsgError: !shouldThrow\n ? (_: string, e: Error) => warn(e.message)\n : undefined,\n }\n let rawResults: Array<ExtractionResult | undefined> = []\n try {\n if (readFromStdin) {\n debug(`Reading input from stdin`)\n // Read from stdin\n if (process.stdin.isTTY) {\n warn('Reading source file from TTY.')\n }\n const stdinSource = await getStdinAsString()\n rawResults = [await processFile(stdinSource, 'dummy', optsWithThrows)]\n } else {\n // Use Promise.allSettled when throws is not explicitly true to collect partial results\n if (!shouldThrow) {\n const settledResults = await Promise.allSettled(\n files.map(async fn => {\n debug('Extracting file:', fn)\n const source = await readFile(fn, {encoding: 'utf8', signal})\n return processFile(source, fn, optsWithThrows)\n })\n )\n rawResults = settledResults.map(result => {\n if (result.status === 'fulfilled') {\n return result.value\n } else {\n warn(String(result.reason))\n return undefined\n }\n })\n } else {\n rawResults = await Promise.all(\n files.map(async fn => {\n debug('Extracting file:', fn)\n const source = await readFile(fn, {encoding: 'utf8', signal})\n return processFile(source, fn, optsWithThrows)\n })\n )\n }\n }\n } catch (e) {\n if (shouldThrow) {\n throw e\n } else {\n warn(String(e))\n }\n }\n\n const formatter: Formatter<unknown> = await resolveBuiltinFormatter(\n opts.format\n )\n const extractionResults = rawResults.filter((r): r is ExtractionResult => !!r)\n\n const extractedMessages = new Map<string, MessageDescriptor>()\n\n for (const {messages} of extractionResults) {\n for (const message of messages) {\n const {id, description, defaultMessage} = message\n if (!id) {\n const error = new Error(\n `[FormatJS CLI] Missing message id for message:\n${JSON.stringify(message, undefined, 2)}`\n )\n if (throws) {\n throw error\n } else {\n warn(error.message)\n }\n continue\n }\n\n if (extractedMessages.has(id)) {\n const existing = extractedMessages.get(id)!\n if (\n stringify(description) !== stringify(existing.description) ||\n defaultMessage !== existing.defaultMessage\n ) {\n const error = new Error(\n `[FormatJS CLI] Duplicate message id: \"${id}\", ` +\n 'but the `description` and/or `defaultMessage` are different.'\n )\n if (throws) {\n throw error\n } else {\n warn(error.message)\n }\n }\n }\n extractedMessages.set(id, message)\n }\n }\n const results: Record<string, Omit<MessageDescriptor, 'id'>> = {}\n const messages = Array.from(extractedMessages.values())\n for (const {id, ...msg} of messages) {\n // GH #3537: flatten is now applied during extraction in the babel plugin,\n // so we don't need to apply it again here. The messages are already flattened.\n results[id] = msg\n }\n if (typeof formatter.serialize === 'function') {\n return formatter.serialize(formatter.format(results as any))\n }\n return (\n stringify(formatter.format(results as any), {\n space: 2,\n cmp: formatter.compareMessages || undefined,\n }) ?? ''\n )\n}\n\n/**\n * Extract strings from source files, also writes to a file.\n * @param files list of files\n * @param extractOpts extract options\n * @returns A Promise that resolves if output file was written successfully\n */\nexport default async function extractAndWrite(\n files: readonly string[],\n extractOpts: ExtractCLIOptions\n): Promise<void> {\n const {outFile, ...opts} = extractOpts\n const serializedResult = (await extract(files, opts)) + '\\n'\n if (outFile) {\n debug('Writing output file:', outFile)\n return outputFile(outFile, serializedResult)\n }\n await writeStdout(serializedResult)\n}\n","import {createRequire} from 'module'\n\nconst require = createRequire(import.meta.url)\n\nconst NATIVE_PACKAGES: Record<string, string> = {\n 'darwin-arm64': '@formatjs/cli-native-darwin-arm64',\n 'linux-x64': '@formatjs/cli-native-linux-x64',\n}\n\nexport interface NativeCompileOptions {\n ast?: boolean\n format?: string\n ignoreTag?: boolean\n pseudoLocale?: string\n skipErrors?: boolean\n followLinks?: boolean\n}\n\nexport interface NativeCompileMessage {\n id: string\n message: string\n sourceFile: string\n}\n\nexport interface NativeBinding {\n compile(inputFiles: string[], opts?: NativeCompileOptions): string\n compileMessages(\n messages: NativeCompileMessage[],\n opts?: NativeCompileOptions\n ): string\n supportedBuiltinFormatters(): string[]\n}\n\nlet nativeBinding: NativeBinding | null | undefined\n\nexport function loadNative(): NativeBinding {\n if (nativeBinding !== undefined) {\n if (nativeBinding) {\n return nativeBinding\n }\n throw new Error('Native @formatjs/cli-lib binding is unavailable')\n }\n\n const overridePath = process.env.FORMATJS_CLI_LIB_NATIVE_PATH\n const platformPackage = NATIVE_PACKAGES[`${process.platform}-${process.arch}`]\n const candidates = [overridePath, platformPackage].filter(Boolean) as string[]\n const loadErrors: string[] = []\n\n for (const candidate of candidates) {\n try {\n nativeBinding = require(candidate) as NativeBinding\n return nativeBinding\n } catch (e) {\n loadErrors.push(`${candidate}: ${(e as Error).message}`)\n }\n }\n\n nativeBinding = null\n throw new Error(\n `Native @formatjs/cli-lib binding is unavailable.\\n${loadErrors.join('\\n')}`\n )\n}\n\nexport function compileWithNative(\n inputFiles: string[],\n opts: NativeCompileOptions = {}\n): string {\n const native = loadNative()\n return native.compile(inputFiles, {\n ast: opts.ast,\n format: opts.format,\n followLinks: opts.followLinks,\n ignoreTag: opts.ignoreTag,\n pseudoLocale: opts.pseudoLocale,\n skipErrors: opts.skipErrors,\n })\n}\n\nexport function compileMessagesWithNative(\n messages: NativeCompileMessage[],\n opts: NativeCompileOptions = {}\n): string {\n const native = loadNative()\n return native.compileMessages(messages, {\n ast: opts.ast,\n ignoreTag: opts.ignoreTag,\n pseudoLocale: opts.pseudoLocale,\n skipErrors: opts.skipErrors,\n })\n}\n","import {outputFile} from 'fs-extra/esm'\nimport {readFile} from 'fs/promises'\nimport * as glob from 'fast-glob'\nimport * as stringifyNs from 'json-stable-stringify'\nimport {resolve} from 'path'\nimport {pathToFileURL} from 'url'\nimport {debug, warn, writeStdout} from '#packages/cli-lib/console_utils.js'\nimport {type Formatter} from '#packages/cli-lib/formatters/index.js'\nimport {\n type NativeCompileMessage,\n compileMessagesWithNative,\n compileWithNative,\n} from '#packages/cli-lib/native.js'\n\nconst dynamicImport = new Function('specifier', 'return import(specifier)') as (\n specifier: string\n) => Promise<Formatter<unknown>>\nconst globSync = glob.sync\nconst stringify = (stringifyNs as any).default || stringifyNs\nconst CUSTOM_FORMATTER_FILE_DEPRECATION_WARNING =\n 'Passing a custom formatter file to --format during compilation is deprecated and will be removed in a future release. Prefer a built-in formatter name or pre-process translation files before running formatjs compile.'\n\nconst BUILTIN_FORMATTERS = new Set([\n 'default',\n 'simple',\n 'transifex',\n 'smartling',\n 'lokalise',\n 'crowdin',\n])\n\nexport type CompileFn = (msgs: any) => Record<string, string>\n\nexport type PseudoLocale = 'xx-LS' | 'xx-AC' | 'xx-HA' | 'en-XA' | 'en-XB'\n\nexport interface CompileCLIOpts extends Opts {\n /**\n * The target file that contains compiled messages.\n */\n outFile?: string\n}\n\nexport interface Opts {\n /**\n * Whether to compile message into AST instead of just string\n */\n ast?: boolean\n /**\n * Whether to continue compiling messages after encountering an error.\n * Any keys with errors will not be included in the output file.\n */\n skipErrors?: boolean\n /**\n * Built-in formatter name or path to a formatter file that converts\n * <translation_files> to `Record<string, string>` so we can compile.\n * Formatter file paths are deprecated for compilation.\n */\n format?: string | Formatter<unknown>\n /**\n * Whether to compile to pseudo locale\n */\n pseudoLocale?: PseudoLocale\n /**\n * Whether the parser to treat HTML/XML tags as string literal\n * instead of parsing them as tag token.\n * When this is false we only allow simple tags without\n * any attributes\n */\n ignoreTag?: boolean\n /**\n * An AbortSignal to cancel the compilation before invoking native code.\n */\n signal?: AbortSignal\n /**\n * Whether to follow symbolic links when traversing directories.\n * Defaults to true for compatibility with pnpm symlinked node_modules.\n */\n followLinks?: boolean\n}\n\n/**\n * Compile extracted translation files with the native formatjs CLI binding.\n * @param inputFiles Input files or glob patterns\n * @param opts Options\n * @returns serialized result in string format\n */\nexport async function compile(\n inputFiles: string[],\n opts: Opts = {}\n): Promise<string> {\n debug('Compiling files:', inputFiles)\n const {\n ast,\n format,\n pseudoLocale,\n skipErrors,\n ignoreTag,\n signal,\n followLinks,\n } = opts\n signal?.throwIfAborted()\n\n if (format && !isBuiltinFormatter(format)) {\n return compileWithCustomFormatter(inputFiles, opts)\n }\n\n return compileWithNative(inputFiles, {\n ast,\n format: typeof format === 'string' ? format : undefined,\n followLinks,\n ignoreTag,\n pseudoLocale,\n skipErrors,\n })\n}\n\nfunction isBuiltinFormatter(\n format: string | Formatter<unknown>\n): format is string {\n return typeof format === 'string' && BUILTIN_FORMATTERS.has(format)\n}\n\nasync function compileWithCustomFormatter(\n inputFiles: string[],\n opts: Opts\n): Promise<string> {\n if (typeof opts.format === 'string') {\n await warn(CUSTOM_FORMATTER_FILE_DEPRECATION_WARNING)\n }\n\n const formatter = await resolveCustomFormatter(opts.format)\n const files = globSync(inputFiles, {\n followSymbolicLinks: opts.followLinks ?? true,\n })\n\n if (!files.length) {\n throw new Error('No translation files found matching the patterns')\n }\n\n const messages: NativeCompileMessage[] = []\n await Promise.all(\n files.map(async file => {\n opts.signal?.throwIfAborted()\n const content = await readFile(file, {\n encoding: 'utf8',\n signal: opts.signal,\n })\n const compiled = formatter.compile(JSON.parse(content))\n for (const id of Object.keys(compiled)) {\n messages.push({\n id,\n message: compiled[id],\n sourceFile: file,\n })\n }\n })\n )\n\n const serialized = compileMessagesWithNative(messages, {\n ast: opts.ast,\n ignoreTag: opts.ignoreTag,\n pseudoLocale: opts.pseudoLocale,\n skipErrors: opts.skipErrors,\n })\n\n if (formatter.compareMessages) {\n return (\n stringify(JSON.parse(serialized), {\n space: 2,\n cmp: formatter.compareMessages,\n }) ?? ''\n )\n }\n\n return serialized\n}\n\nasync function resolveCustomFormatter(\n format: string | Formatter<unknown> | undefined\n): Promise<Formatter<unknown>> {\n if (!format) {\n throw new Error('A custom formatter is required')\n }\n if (typeof format !== 'string') {\n return format\n }\n try {\n return dynamicImport(pathToFileURL(resolve(process.cwd(), format)).href)\n } catch (e) {\n console.error(`Cannot resolve formatter ${format}`)\n throw e\n }\n}\n\n/**\n * Compile extracted translation files with the native formatjs CLI binding and\n * write output to `outFile` when provided.\n * @param inputFiles Input files or glob patterns\n * @param compileOpts options\n * @returns A `Promise` that resolves if file was written successfully\n */\nexport default async function compileAndWrite(\n inputFiles: string[],\n compileOpts: CompileCLIOpts = {}\n): Promise<void> {\n const {outFile, ...opts} = compileOpts\n const serializedResult = (await compile(inputFiles, opts)) + '\\n'\n if (outFile) {\n debug('Writing output file:', outFile)\n return outputFile(outFile, serializedResult)\n }\n await writeStdout(serializedResult)\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":["format","compile","format","compile","format","compile","format","compile","format","compile","compile","defaultFormatter","transifex","smartling","simple","lokalise","crowdin","stringify","require"],"sources":["../formatters/crowdin.ts","../formatters/default.ts","../formatters/lokalise.ts","../formatters/simple.ts","../formatters/smartling.ts","../formatters/transifex.ts","../formatters/index.ts","../extract.ts","../native.ts","../compile.ts"],"sourcesContent":["import {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport type CrowdinJson = Record<\n string,\n {\n message: string\n description?: string\n }\n>\n\nexport const format: FormatFn<CrowdinJson> = msgs => {\n const results: CrowdinJson = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = {\n message: msg.defaultMessage!,\n description:\n typeof msg.description === 'string'\n ? msg.description\n : JSON.stringify(msg.description),\n }\n }\n return results\n}\n\nexport const compile: CompileFn<CrowdinJson> = msgs => {\n const results: Record<string, string> = {}\n for (const [id, msg] of Object.entries(msgs)) {\n if (id === 'smartling') {\n continue\n }\n results[id] = msg.message\n }\n return results\n}\n","import {type MessageDescriptor} from '@formatjs/ts-transformer'\nexport type FormatFn<T = Record<string, MessageDescriptor>> = (\n msgs: Record<string, MessageDescriptor>\n) => T\n\nexport type CompileFn<T = Record<string, MessageDescriptor>> = (\n msgs: T\n) => Record<string, string>\n\nexport type SerializeFn<T = Record<string, MessageDescriptor>> = (\n msgs: T\n) => string\n\nexport const format: FormatFn = msgs => msgs\n\nexport const compile: CompileFn = msgs => {\n const results: Record<string, string> = {}\n for (const k in msgs) {\n results[k] = msgs[k].defaultMessage!\n }\n return results\n}\n","import {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport type StructuredJson = Record<\n string,\n {\n translation: string\n notes?: string\n context?: string\n limit?: string\n }\n>\n\nexport const format: FormatFn<StructuredJson> = msgs => {\n const results: StructuredJson = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = {\n translation: msg.defaultMessage!,\n notes:\n typeof msg.description === 'string'\n ? msg.description\n : JSON.stringify(msg.description),\n }\n }\n return results\n}\n\nexport const compile: CompileFn<StructuredJson> = msgs => {\n const results: Record<string, string> = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = msg.translation\n }\n return results\n}\n","import {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport type PhraseJson = Record<string, string>\n\nexport const format: FormatFn<PhraseJson> = msgs => {\n return Object.keys(msgs).reduce((all: PhraseJson, k) => {\n all[k] = msgs[k].defaultMessage!\n return all\n }, {})\n}\n\nexport const compile: CompileFn<PhraseJson> = msgs => msgs\n","import type {Comparator} from 'json-stable-stringify'\nimport {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport interface SmartlingDirectives {\n translate_paths: [\n {\n path: string\n key: string\n instruction: string\n },\n ]\n variants_enabled: boolean\n string_format: string\n [k: string]: any\n}\n\nexport type SmartlingJson = {\n smartling: SmartlingDirectives\n} & Record<\n string,\n {\n message: string\n description?: string\n }\n>\n\nexport const format: FormatFn<SmartlingJson> = msgs => {\n const results: SmartlingJson = {\n smartling: {\n translate_paths: [\n {\n path: '*/message',\n key: '{*}/message',\n instruction: '*/description',\n },\n ],\n variants_enabled: true,\n string_format: 'icu',\n },\n } as any\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = {\n message: msg.defaultMessage!,\n description:\n typeof msg.description === 'string'\n ? msg.description\n : JSON.stringify(msg.description),\n }\n }\n return results\n}\n\nexport const compareMessages: Comparator = (el1, el2) => {\n // `smartling` has to be the 1st key\n if (el1.key === 'smartling') {\n return -1\n }\n if (el2.key === 'smartling') {\n return 1\n }\n return el1.key < el2.key ? -1 : el1.key === el2.key ? 0 : 1\n}\n\nexport const compile: CompileFn<SmartlingJson> = msgs => {\n const results: Record<string, string> = {}\n for (const [id, msg] of Object.entries(msgs)) {\n if (id === 'smartling') {\n continue\n }\n results[id] = msg.message\n }\n return results\n}\n","import {\n type CompileFn,\n type FormatFn,\n} from '#packages/cli-lib/formatters/default.js'\n\nexport type StructuredJson = Record<\n string,\n {\n string: string\n developer_comment?: string\n context?: string\n character_limit?: string\n }\n>\n\nexport const format: FormatFn<StructuredJson> = msgs => {\n const results: StructuredJson = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = {\n string: msg.defaultMessage!,\n developer_comment:\n typeof msg.description === 'string'\n ? msg.description\n : JSON.stringify(msg.description),\n }\n }\n return results\n}\n\nexport const compile: CompileFn<StructuredJson> = msgs => {\n const results: Record<string, string> = {}\n for (const [id, msg] of Object.entries(msgs)) {\n results[id] = msg.string\n }\n return results\n}\n","import type {Comparator} from 'json-stable-stringify'\nimport {resolve} from 'path'\nimport {pathToFileURL} from 'url'\nimport * as crowdin from '#packages/cli-lib/formatters/crowdin.js'\nimport * as defaultFormatter from '#packages/cli-lib/formatters/default.js'\nimport {\n type CompileFn,\n type FormatFn,\n type SerializeFn,\n} from '#packages/cli-lib/formatters/default.js'\nimport * as lokalise from '#packages/cli-lib/formatters/lokalise.js'\nimport * as simple from '#packages/cli-lib/formatters/simple.js'\nimport * as smartling from '#packages/cli-lib/formatters/smartling.js'\nimport * as transifex from '#packages/cli-lib/formatters/transifex.js'\n\nexport interface Formatter<T> {\n serialize?: SerializeFn<T>\n format: FormatFn<T>\n compile: CompileFn<T>\n compareMessages?: Comparator\n}\n\nexport async function resolveBuiltinFormatter(\n format?: string | Formatter<unknown>\n): Promise<any> {\n if (!format) {\n return defaultFormatter\n }\n if (typeof format !== 'string') {\n return format\n }\n switch (format) {\n case 'transifex':\n return transifex\n case 'smartling':\n return smartling\n case 'simple':\n return simple\n case 'lokalise':\n return lokalise\n case 'crowdin':\n return crowdin\n }\n try {\n // eslint-disable-next-line import/dynamic-import-chunkname\n return import(pathToFileURL(resolve(process.cwd(), format)).href)\n } catch (e) {\n console.error(`Cannot resolve formatter ${format}`)\n throw e\n }\n}\n","import {\n type MessageDescriptor,\n type Opts,\n interpolateName,\n} from '@formatjs/ts-transformer'\nimport {outputFile} from 'fs-extra/esm'\nimport {\n debug,\n getStdinAsString,\n warn,\n writeStdout,\n} from '#packages/cli-lib/console_utils.js'\nimport * as stringifyNs from 'json-stable-stringify'\n\nimport {\n type Formatter,\n resolveBuiltinFormatter,\n} from '#packages/cli-lib/formatters/index.js'\nimport {parseScript} from '#packages/cli-lib/parse_script.js'\nimport {readFile} from 'fs/promises'\n\nconst stringify = (stringifyNs as any).default || stringifyNs\nexport interface ExtractionResult<M = Record<string, string>> {\n /**\n * List of extracted messages\n */\n messages: MessageDescriptor[]\n /**\n * Metadata extracted w/ `pragma`\n */\n meta?: M\n}\n\nexport interface ExtractedMessageDescriptor extends MessageDescriptor {\n /**\n * Line number\n */\n line?: number\n /**\n * Column number\n */\n col?: number\n /**\n * Metadata extracted from pragma\n */\n meta?: Record<string, string>\n}\n\nexport type ExtractCLIOptions = Omit<\n ExtractOpts,\n 'overrideIdFn' | 'onMsgExtracted' | 'onMetaExtracted'\n> & {\n /**\n * Output File\n */\n outFile?: string\n /**\n * Input File\n */\n inFile?: string\n /**\n * Ignore file glob pattern\n */\n ignore?: string[]\n /**\n * Whether to follow symbolic links when traversing directories.\n * Defaults to true for compatibility with pnpm symlinked node_modules.\n */\n followLinks?: boolean\n}\n\nexport type ExtractOpts = Opts & {\n /**\n * Whether to throw an error if we had any issues with\n * 1 of the source files\n */\n throws?: boolean\n /**\n * Message ID interpolation pattern\n */\n idInterpolationPattern?: string\n /**\n * Whether we read from stdin instead of a file\n */\n readFromStdin?: boolean\n /**\n * Either path to a formatter file that controls the shape of JSON file from `outFile` or {@link Formatter} object.\n */\n format?: string | Formatter<any>\n /**\n * Whether to hoist selectors & flatten sentences\n */\n flatten?: boolean\n /**\n * An AbortSignal to cancel the extraction\n */\n signal?: AbortSignal\n} & Pick<Opts, 'onMsgExtracted' | 'onMetaExtracted'>\n\nfunction calculateLineColFromOffset(\n text: string,\n start?: number\n): Pick<ExtractedMessageDescriptor, 'line' | 'col'> {\n if (!start) {\n return {line: 1, col: 1}\n }\n const chunk = text.slice(0, start)\n const lines = chunk.split('\\n')\n const lastLine = lines[lines.length - 1]\n return {line: lines.length, col: lastLine.length}\n}\n\nasync function processFile(\n source: string,\n fn: string,\n {idInterpolationPattern, ...opts}: Opts & {idInterpolationPattern?: string}\n) {\n let messages: ExtractedMessageDescriptor[] = []\n let meta: Record<string, string> | undefined\n\n const onMsgExtracted = opts.onMsgExtracted\n const onMetaExtracted = opts.onMetaExtracted\n\n opts = {\n ...opts,\n additionalComponentNames: [\n '$formatMessage',\n ...(opts.additionalComponentNames || []),\n ],\n onMsgExtracted(filePath, msgs) {\n if (opts.extractSourceLocation) {\n msgs = msgs.map(msg => ({\n ...msg,\n ...calculateLineColFromOffset(source, msg.start),\n }))\n }\n messages = messages.concat(msgs)\n\n if (onMsgExtracted) {\n onMsgExtracted(filePath, msgs)\n }\n },\n onMetaExtracted(filePath, m) {\n meta = m\n\n if (onMetaExtracted) {\n onMetaExtracted(filePath, m)\n }\n },\n }\n\n if (!opts.overrideIdFn && idInterpolationPattern) {\n opts = {\n ...opts,\n overrideIdFn: (id, defaultMessage, description, fileName) =>\n id ||\n interpolateName(\n {\n resourcePath: fileName,\n } as any,\n idInterpolationPattern,\n {\n content: description\n ? `${defaultMessage}#${\n typeof description === 'string'\n ? description\n : stringify(description)\n }`\n : defaultMessage,\n }\n ),\n }\n }\n\n debug('Processing opts for %s: %s', fn, opts)\n\n const scriptParseFn = parseScript(opts, fn)\n if (fn.endsWith('.vue')) {\n debug('Processing %s using vue extractor', fn)\n const {parseFile} = await import('./vue_extractor.js')\n parseFile(source, fn, scriptParseFn)\n } else if (fn.endsWith('.svelte')) {\n debug('Processing %s using svelte extractor', fn)\n const {parseFile} = await import('./svelte_extractor.js')\n parseFile(source, fn, scriptParseFn)\n } else if (fn.endsWith('.hbs')) {\n debug('Processing %s using hbs extractor', fn)\n const {parseFile} = await import('./hbs_extractor.js')\n parseFile(source, fn, opts)\n } else if (fn.endsWith('.gts') || fn.endsWith('.gjs')) {\n debug('Processing %s as gts/gjs file', fn)\n const {parseFile} = await import('./gts_extractor.js')\n parseFile(source, fn, opts)\n } else {\n debug('Processing %s using typescript extractor', fn)\n scriptParseFn(source)\n }\n debug('Done extracting %s messages: %s', fn, messages)\n if (meta) {\n debug('Extracted meta:', meta)\n messages.forEach(m => (m.meta = meta))\n }\n return {messages, meta}\n}\n\n/**\n * Extract strings from source files\n * @param files list of files\n * @param extractOpts extract options\n * @returns messages serialized as JSON string since key order\n * matters for some `format`\n */\nexport async function extract(\n files: readonly string[],\n extractOpts: ExtractOpts\n): Promise<string> {\n const {throws, readFromStdin, signal, ...opts} = extractOpts\n // When throws is not explicitly true, we want to collect partial results\n const shouldThrow = throws === true\n // Pass throws option to transformer for per-message error handling\n const optsWithThrows = {\n ...opts,\n throws: shouldThrow,\n onMsgError: !shouldThrow\n ? (_: string, e: Error) => warn(e.message)\n : undefined,\n }\n let rawResults: Array<ExtractionResult | undefined> = []\n try {\n if (readFromStdin) {\n debug(`Reading input from stdin`)\n // Read from stdin\n if (process.stdin.isTTY) {\n warn('Reading source file from TTY.')\n }\n const stdinSource = await getStdinAsString()\n rawResults = [await processFile(stdinSource, 'dummy', optsWithThrows)]\n } else {\n // Use Promise.allSettled when throws is not explicitly true to collect partial results\n if (!shouldThrow) {\n const settledResults = await Promise.allSettled(\n files.map(async fn => {\n debug('Extracting file:', fn)\n const source = await readFile(fn, {encoding: 'utf8', signal})\n return processFile(source, fn, optsWithThrows)\n })\n )\n rawResults = settledResults.map(result => {\n if (result.status === 'fulfilled') {\n return result.value\n } else {\n warn(String(result.reason))\n return undefined\n }\n })\n } else {\n rawResults = await Promise.all(\n files.map(async fn => {\n debug('Extracting file:', fn)\n const source = await readFile(fn, {encoding: 'utf8', signal})\n return processFile(source, fn, optsWithThrows)\n })\n )\n }\n }\n } catch (e) {\n if (shouldThrow) {\n throw e\n } else {\n warn(String(e))\n }\n }\n\n const formatter: Formatter<unknown> = await resolveBuiltinFormatter(\n opts.format\n )\n const extractionResults = rawResults.filter((r): r is ExtractionResult => !!r)\n\n const extractedMessages = new Map<string, MessageDescriptor>()\n\n for (const {messages} of extractionResults) {\n for (const message of messages) {\n const {id, description, defaultMessage} = message\n if (!id) {\n const error = new Error(\n `[FormatJS CLI] Missing message id for message:\n${JSON.stringify(message, undefined, 2)}`\n )\n if (throws) {\n throw error\n } else {\n warn(error.message)\n }\n continue\n }\n\n if (extractedMessages.has(id)) {\n const existing = extractedMessages.get(id)!\n if (\n stringify(description) !== stringify(existing.description) ||\n defaultMessage !== existing.defaultMessage\n ) {\n const error = new Error(\n `[FormatJS CLI] Duplicate message id: \"${id}\", ` +\n 'but the `description` and/or `defaultMessage` are different.'\n )\n if (throws) {\n throw error\n } else {\n warn(error.message)\n }\n }\n }\n extractedMessages.set(id, message)\n }\n }\n const results: Record<string, Omit<MessageDescriptor, 'id'>> = {}\n const messages = Array.from(extractedMessages.values())\n for (const {id, ...msg} of messages) {\n // GH #3537: flatten is now applied during extraction in the babel plugin,\n // so we don't need to apply it again here. The messages are already flattened.\n results[id] = msg\n }\n if (typeof formatter.serialize === 'function') {\n return formatter.serialize(formatter.format(results as any))\n }\n return (\n stringify(formatter.format(results as any), {\n space: 2,\n cmp: formatter.compareMessages || undefined,\n }) ?? ''\n )\n}\n\n/**\n * Extract strings from source files, also writes to a file.\n * @param files list of files\n * @param extractOpts extract options\n * @returns A Promise that resolves if output file was written successfully\n */\nexport default async function extractAndWrite(\n files: readonly string[],\n extractOpts: ExtractCLIOptions\n): Promise<void> {\n const {outFile, ...opts} = extractOpts\n const serializedResult = (await extract(files, opts)) + '\\n'\n if (outFile) {\n debug('Writing output file:', outFile)\n return outputFile(outFile, serializedResult)\n }\n await writeStdout(serializedResult)\n}\n","import {createRequire} from 'module'\n\nconst require = createRequire(import.meta.url)\n\nconst NATIVE_PACKAGES: Record<string, string> = {\n 'darwin-arm64': '@formatjs/cli-native-darwin-arm64',\n 'linux-arm64': '@formatjs/cli-native-linux-arm64',\n 'linux-x64': '@formatjs/cli-native-linux-x64',\n 'win32-x64': '@formatjs/cli-native-win32-x64',\n}\n\nexport interface NativeCompileOptions {\n ast?: boolean\n format?: string\n ignoreTag?: boolean\n pseudoLocale?: string\n skipErrors?: boolean\n followLinks?: boolean\n}\n\nexport interface NativeCompileMessage {\n id: string\n message: string\n sourceFile: string\n}\n\nexport interface NativeBinding {\n compile(inputFiles: string[], opts?: NativeCompileOptions): string\n compileMessages(\n messages: NativeCompileMessage[],\n opts?: NativeCompileOptions\n ): string\n supportedBuiltinFormatters(): string[]\n}\n\nlet nativeBinding: NativeBinding | null | undefined\n\nexport function loadNative(): NativeBinding {\n if (nativeBinding !== undefined) {\n if (nativeBinding) {\n return nativeBinding\n }\n throw new Error('Native @formatjs/cli-lib binding is unavailable')\n }\n\n const overridePath = process.env.FORMATJS_CLI_LIB_NATIVE_PATH\n const platformPackage = NATIVE_PACKAGES[`${process.platform}-${process.arch}`]\n const candidates = [overridePath, platformPackage].filter(Boolean) as string[]\n const loadErrors: string[] = []\n\n for (const candidate of candidates) {\n try {\n nativeBinding = require(candidate) as NativeBinding\n return nativeBinding\n } catch (e) {\n loadErrors.push(`${candidate}: ${(e as Error).message}`)\n }\n }\n\n nativeBinding = null\n throw new Error(\n `Native @formatjs/cli-lib binding is unavailable.\\n${loadErrors.join('\\n')}`\n )\n}\n\nexport function compileWithNative(\n inputFiles: string[],\n opts: NativeCompileOptions = {}\n): string {\n const native = loadNative()\n return native.compile(inputFiles, {\n ast: opts.ast,\n format: opts.format,\n followLinks: opts.followLinks,\n ignoreTag: opts.ignoreTag,\n pseudoLocale: opts.pseudoLocale,\n skipErrors: opts.skipErrors,\n })\n}\n\nexport function compileMessagesWithNative(\n messages: NativeCompileMessage[],\n opts: NativeCompileOptions = {}\n): string {\n const native = loadNative()\n return native.compileMessages(messages, {\n ast: opts.ast,\n ignoreTag: opts.ignoreTag,\n pseudoLocale: opts.pseudoLocale,\n skipErrors: opts.skipErrors,\n })\n}\n","import {outputFile} from 'fs-extra/esm'\nimport {readFile} from 'fs/promises'\nimport * as glob from 'fast-glob'\nimport * as stringifyNs from 'json-stable-stringify'\nimport {resolve} from 'path'\nimport {pathToFileURL} from 'url'\nimport {debug, warn, writeStdout} from '#packages/cli-lib/console_utils.js'\nimport {type Formatter} from '#packages/cli-lib/formatters/index.js'\nimport {\n type NativeCompileMessage,\n compileMessagesWithNative,\n compileWithNative,\n} from '#packages/cli-lib/native.js'\n\nconst dynamicImport = new Function('specifier', 'return import(specifier)') as (\n specifier: string\n) => Promise<Formatter<unknown>>\nconst globSync = glob.sync\nconst stringify = (stringifyNs as any).default || stringifyNs\nconst CUSTOM_FORMATTER_FILE_DEPRECATION_WARNING =\n 'Passing a custom formatter file to --format during compilation is deprecated and will be removed in a future release. Prefer a built-in formatter name or pre-process translation files before running formatjs compile.'\n\nconst BUILTIN_FORMATTERS = new Set([\n 'default',\n 'simple',\n 'transifex',\n 'smartling',\n 'lokalise',\n 'crowdin',\n])\n\nexport type CompileFn = (msgs: any) => Record<string, string>\n\nexport type PseudoLocale = 'xx-LS' | 'xx-AC' | 'xx-HA' | 'en-XA' | 'en-XB'\n\nexport interface CompileCLIOpts extends Opts {\n /**\n * The target file that contains compiled messages.\n */\n outFile?: string\n}\n\nexport interface Opts {\n /**\n * Whether to compile message into AST instead of just string\n */\n ast?: boolean\n /**\n * Whether to continue compiling messages after encountering an error.\n * Any keys with errors will not be included in the output file.\n */\n skipErrors?: boolean\n /**\n * Built-in formatter name or path to a formatter file that converts\n * <translation_files> to `Record<string, string>` so we can compile.\n * Formatter file paths are deprecated for compilation.\n */\n format?: string | Formatter<unknown>\n /**\n * Whether to compile to pseudo locale\n */\n pseudoLocale?: PseudoLocale\n /**\n * Whether the parser to treat HTML/XML tags as string literal\n * instead of parsing them as tag token.\n * When this is false we only allow simple tags without\n * any attributes\n */\n ignoreTag?: boolean\n /**\n * An AbortSignal to cancel the compilation before invoking native code.\n */\n signal?: AbortSignal\n /**\n * Whether to follow symbolic links when traversing directories.\n * Defaults to true for compatibility with pnpm symlinked node_modules.\n */\n followLinks?: boolean\n}\n\n/**\n * Compile extracted translation files with the native formatjs CLI binding.\n * @param inputFiles Input files or glob patterns\n * @param opts Options\n * @returns serialized result in string format\n */\nexport async function compile(\n inputFiles: string[],\n opts: Opts = {}\n): Promise<string> {\n debug('Compiling files:', inputFiles)\n const {\n ast,\n format,\n pseudoLocale,\n skipErrors,\n ignoreTag,\n signal,\n followLinks,\n } = opts\n signal?.throwIfAborted()\n\n if (format && !isBuiltinFormatter(format)) {\n return compileWithCustomFormatter(inputFiles, opts)\n }\n\n return compileWithNative(inputFiles, {\n ast,\n format: typeof format === 'string' ? format : undefined,\n followLinks,\n ignoreTag,\n pseudoLocale,\n skipErrors,\n })\n}\n\nfunction isBuiltinFormatter(\n format: string | Formatter<unknown>\n): format is string {\n return typeof format === 'string' && BUILTIN_FORMATTERS.has(format)\n}\n\nasync function compileWithCustomFormatter(\n inputFiles: string[],\n opts: Opts\n): Promise<string> {\n if (typeof opts.format === 'string') {\n await warn(CUSTOM_FORMATTER_FILE_DEPRECATION_WARNING)\n }\n\n const formatter = await resolveCustomFormatter(opts.format)\n const files = globSync(inputFiles, {\n followSymbolicLinks: opts.followLinks ?? true,\n })\n\n if (!files.length) {\n throw new Error('No translation files found matching the patterns')\n }\n\n const messages: NativeCompileMessage[] = []\n await Promise.all(\n files.map(async file => {\n opts.signal?.throwIfAborted()\n const content = await readFile(file, {\n encoding: 'utf8',\n signal: opts.signal,\n })\n const compiled = formatter.compile(JSON.parse(content))\n for (const id of Object.keys(compiled)) {\n messages.push({\n id,\n message: compiled[id],\n sourceFile: file,\n })\n }\n })\n )\n\n const serialized = compileMessagesWithNative(messages, {\n ast: opts.ast,\n ignoreTag: opts.ignoreTag,\n pseudoLocale: opts.pseudoLocale,\n skipErrors: opts.skipErrors,\n })\n\n if (formatter.compareMessages) {\n return (\n stringify(JSON.parse(serialized), {\n space: 2,\n cmp: formatter.compareMessages,\n }) ?? ''\n )\n }\n\n return serialized\n}\n\nasync function resolveCustomFormatter(\n format: string | Formatter<unknown> | undefined\n): Promise<Formatter<unknown>> {\n if (!format) {\n throw new Error('A custom formatter is required')\n }\n if (typeof format !== 'string') {\n return format\n }\n try {\n return dynamicImport(pathToFileURL(resolve(process.cwd(), format)).href)\n } catch (e) {\n console.error(`Cannot resolve formatter ${format}`)\n throw e\n }\n}\n\n/**\n * Compile extracted translation files with the native formatjs CLI binding and\n * write output to `outFile` when provided.\n * @param inputFiles Input files or glob patterns\n * @param compileOpts options\n * @returns A `Promise` that resolves if file was written successfully\n */\nexport default async function compileAndWrite(\n inputFiles: string[],\n compileOpts: CompileCLIOpts = {}\n): Promise<void> {\n const {outFile, ...opts} = compileOpts\n const serializedResult = (await compile(inputFiles, opts)) + '\\n'\n if (outFile) {\n debug('Writing output file:', outFile)\n return outputFile(outFile, serializedResult)\n }\n await writeStdout(serializedResult)\n}\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@formatjs/cli-lib",
3
3
  "description": "Lib for CLI for formatjs.",
4
- "version": "8.6.0",
4
+ "version": "8.7.1",
5
5
  "license": "MIT",
6
6
  "author": "Linjie Ding <linjie@airtable.com>",
7
7
  "type": "module",
@@ -22,9 +22,9 @@
22
22
  "json-stable-stringify": "^1.3.0",
23
23
  "loud-rejection": "^2",
24
24
  "typescript": "^5.6 || 6",
25
- "@formatjs/icu-messageformat-parser": "3.5.8",
26
- "@formatjs/icu-skeleton-parser": "2.1.8",
27
- "@formatjs/ts-transformer": "4.4.8"
25
+ "@formatjs/icu-skeleton-parser": "2.1.9",
26
+ "@formatjs/ts-transformer": "4.4.9",
27
+ "@formatjs/icu-messageformat-parser": "3.5.9"
28
28
  },
29
29
  "peerDependencies": {
30
30
  "@glimmer/syntax": "^0.84.3 || ^0.95.0",
@@ -55,6 +55,8 @@
55
55
  "main": "index.js",
56
56
  "optionalDependencies": {
57
57
  "@formatjs/cli-native-darwin-arm64": "1.1.0",
58
+ "@formatjs/cli-native-linux-arm64": "1.2.0",
59
+ "@formatjs/cli-native-win32-x64": "1.1.1",
58
60
  "@formatjs/cli-native-linux-x64": "1.1.0"
59
61
  },
60
62
  "peerDependenciesMeta": {
@@ -1 +1 @@
1
- {"version":3,"file":"parse_script-EspfGgzZ.js","names":["clearLine"],"sources":["../console_utils.ts","../parse_script.ts"],"sourcesContent":["import {\n clearLine as nativeClearLine,\n cursorTo as nativeCursorTo,\n} from 'readline'\nimport {format, promisify, styleText} from 'node:util'\n\nconst CLEAR_WHOLE_LINE = 0\n\nexport const writeStderr: (arg1: string | Uint8Array) => Promise<void> =\n promisify(process.stderr.write).bind(process.stderr)\nexport const writeStdout: (arg1: string | Uint8Array) => Promise<void> =\n promisify(process.stdout.write).bind(process.stdout)\n\n// From:\n// https://github.com/yarnpkg/yarn/blob/53d8004229f543f342833310d5af63a4b6e59c8a/src/reporters/console/util.js\nexport async function clearLine(\n terminal: (typeof process)['stderr']\n): Promise<void> {\n if (!terminal.hasColors?.()) {\n if (terminal.isTTY) {\n // terminal\n if (terminal.columns > 0) {\n await writeStderr(`\\r${' '.repeat(terminal.columns - 1)}`)\n }\n await writeStderr(`\\r`)\n }\n // ignore piping to file\n } else {\n nativeClearLine(terminal, CLEAR_WHOLE_LINE)\n nativeCursorTo(terminal, 0, undefined)\n }\n}\n\ntype LogLevel = 'debug' | 'warn' | 'error'\n\nconst LEVEL_COLORS: Record<LogLevel, 'green' | 'yellow' | 'red'> = {\n debug: 'green',\n warn: 'yellow',\n error: 'red',\n}\n\nfunction label(level: LogLevel, message: string) {\n return `[@formatjs/cli] [${styleText(\n LEVEL_COLORS[level],\n level.toUpperCase()\n )}] ${message}`\n}\n\nexport async function debug(message: string, ...args: any[]): Promise<void> {\n if (process.env.LOG_LEVEL !== 'debug') {\n return\n }\n await clearLine(process.stderr)\n await writeStderr(format(label('debug', message), ...args))\n await writeStderr('\\n')\n}\n\nexport async function warn(message: string, ...args: any[]): Promise<void> {\n await clearLine(process.stderr)\n await writeStderr(format(label('warn', message), ...args))\n await writeStderr('\\n')\n}\n\nexport async function error(message: string, ...args: any[]): Promise<void> {\n await clearLine(process.stderr)\n await writeStderr(format(label('error', message), ...args))\n await writeStderr('\\n')\n}\n\nexport function getStdinAsString(): Promise<string> {\n let result = ''\n return new Promise(resolve => {\n process.stdin.setEncoding('utf-8')\n process.stdin.on('readable', () => {\n let chunk\n while ((chunk = process.stdin.read())) {\n result += chunk\n }\n })\n process.stdin.on('end', () => {\n resolve(result)\n })\n })\n}\n","import {type Opts, transformWithTs} from '@formatjs/ts-transformer'\nimport * as ts from 'typescript'\nimport {debug} from '#packages/cli-lib/console_utils.js'\n/**\n * Invoid TypeScript module transpilation with our TS transformer\n * @param opts Formatjs TS Transformer opt\n * @param fn filename\n */\nexport function parseScript(opts: Opts, fn?: string) {\n return (source: string): void => {\n let output\n try {\n debug('Using TS compiler to process file', fn)\n output = ts.transpileModule(source, {\n compilerOptions: {\n allowJs: true,\n target: ts.ScriptTarget.ESNext,\n noEmit: true,\n experimentalDecorators: true,\n },\n reportDiagnostics: true,\n fileName: fn,\n transformers: {\n before: [transformWithTs(ts, opts)],\n },\n })\n } catch (e) {\n if (e instanceof Error) {\n e.message = `Error processing file ${fn} \n${e.message || ''}`\n }\n throw e\n }\n if (output.diagnostics) {\n const errs = output.diagnostics.filter(\n d => d.category === ts.DiagnosticCategory.Error\n )\n if (errs.length) {\n throw new Error(\n ts.formatDiagnosticsWithColorAndContext(errs, {\n getCanonicalFileName: fileName => fileName,\n getCurrentDirectory: () => process.cwd(),\n getNewLine: () => ts.sys.newLine,\n })\n )\n }\n }\n }\n}\n"],"mappings":";;;;;AAMA,MAAM,mBAAmB;AAEzB,MAAa,cACX,UAAU,QAAQ,OAAO,MAAM,CAAC,KAAK,QAAQ,OAAO;AACtD,MAAa,cACX,UAAU,QAAQ,OAAO,MAAM,CAAC,KAAK,QAAQ,OAAO;AAItD,eAAsBA,YACpB,UACe;AACf,KAAI,CAAC,SAAS,aAAa;MACrB,SAAS,OAAO;AAElB,OAAI,SAAS,UAAU,EACrB,OAAM,YAAY,KAAK,IAAI,OAAO,SAAS,UAAU,EAAE,GAAG;AAE5D,SAAM,YAAY,KAAK;;QAGpB;AACL,YAAgB,UAAU,iBAAiB;AAC3C,WAAe,UAAU,GAAG,KAAA,EAAU;;;AAM1C,MAAM,eAA6D;CACjE,OAAO;CACP,MAAM;CACN,OAAO;CACR;AAED,SAAS,MAAM,OAAiB,SAAiB;AAC/C,QAAO,oBAAoB,UACzB,aAAa,QACb,MAAM,aAAa,CACpB,CAAC,IAAI;;AAGR,eAAsB,MAAM,SAAiB,GAAG,MAA4B;AAC1E,KAAI,QAAQ,IAAI,cAAc,QAC5B;AAEF,OAAMA,YAAU,QAAQ,OAAO;AAC/B,OAAM,YAAY,OAAO,MAAM,SAAS,QAAQ,EAAE,GAAG,KAAK,CAAC;AAC3D,OAAM,YAAY,KAAK;;AAGzB,eAAsB,KAAK,SAAiB,GAAG,MAA4B;AACzE,OAAMA,YAAU,QAAQ,OAAO;AAC/B,OAAM,YAAY,OAAO,MAAM,QAAQ,QAAQ,EAAE,GAAG,KAAK,CAAC;AAC1D,OAAM,YAAY,KAAK;;AASzB,SAAgB,mBAAoC;CAClD,IAAI,SAAS;AACb,QAAO,IAAI,SAAQ,YAAW;AAC5B,UAAQ,MAAM,YAAY,QAAQ;AAClC,UAAQ,MAAM,GAAG,kBAAkB;GACjC,IAAI;AACJ,UAAQ,QAAQ,QAAQ,MAAM,MAAM,CAClC,WAAU;IAEZ;AACF,UAAQ,MAAM,GAAG,aAAa;AAC5B,WAAQ,OAAO;IACf;GACF;;;;;;;;;AC1EJ,SAAgB,YAAY,MAAY,IAAa;AACnD,SAAQ,WAAyB;EAC/B,IAAI;AACJ,MAAI;AACF,SAAM,qCAAqC,GAAG;AAC9C,YAAS,GAAG,gBAAgB,QAAQ;IAClC,iBAAiB;KACf,SAAS;KACT,QAAQ,GAAG,aAAa;KACxB,QAAQ;KACR,wBAAwB;KACzB;IACD,mBAAmB;IACnB,UAAU;IACV,cAAc,EACZ,QAAQ,CAAC,gBAAgB,IAAI,KAAK,CAAC,EACpC;IACF,CAAC;WACK,GAAG;AACV,OAAI,aAAa,MACf,GAAE,UAAU,yBAAyB,GAAG;EAC9C,EAAE,WAAW;AAET,SAAM;;AAER,MAAI,OAAO,aAAa;GACtB,MAAM,OAAO,OAAO,YAAY,QAC9B,MAAK,EAAE,aAAa,GAAG,mBAAmB,MAC3C;AACD,OAAI,KAAK,OACP,OAAM,IAAI,MACR,GAAG,qCAAqC,MAAM;IAC5C,uBAAsB,aAAY;IAClC,2BAA2B,QAAQ,KAAK;IACxC,kBAAkB,GAAG,IAAI;IAC1B,CAAC,CACH"}
1
+ {"version":3,"file":"parse_script-EspfGgzZ.js","names":["clearLine"],"sources":["../console_utils.ts","../parse_script.ts"],"sourcesContent":["import {\n clearLine as nativeClearLine,\n cursorTo as nativeCursorTo,\n} from 'readline'\nimport {format, promisify, styleText} from 'node:util'\n\nconst CLEAR_WHOLE_LINE = 0\n\nexport const writeStderr: (arg1: string | Uint8Array) => Promise<void> =\n promisify(process.stderr.write).bind(process.stderr)\nexport const writeStdout: (arg1: string | Uint8Array) => Promise<void> =\n promisify(process.stdout.write).bind(process.stdout)\n\n// From:\n// https://github.com/yarnpkg/yarn/blob/53d8004229f543f342833310d5af63a4b6e59c8a/src/reporters/console/util.js\nexport async function clearLine(\n terminal: (typeof process)['stderr']\n): Promise<void> {\n if (!terminal.hasColors?.()) {\n if (terminal.isTTY) {\n // terminal\n if (terminal.columns > 0) {\n await writeStderr(`\\r${' '.repeat(terminal.columns - 1)}`)\n }\n await writeStderr(`\\r`)\n }\n // ignore piping to file\n } else {\n nativeClearLine(terminal, CLEAR_WHOLE_LINE)\n nativeCursorTo(terminal, 0, undefined)\n }\n}\n\ntype LogLevel = 'debug' | 'warn' | 'error'\n\nconst LEVEL_COLORS: Record<LogLevel, 'green' | 'yellow' | 'red'> = {\n debug: 'green',\n warn: 'yellow',\n error: 'red',\n}\n\nfunction label(level: LogLevel, message: string) {\n return `[@formatjs/cli] [${styleText(\n LEVEL_COLORS[level],\n level.toUpperCase()\n )}] ${message}`\n}\n\nexport async function debug(message: string, ...args: any[]): Promise<void> {\n if (process.env.LOG_LEVEL !== 'debug') {\n return\n }\n await clearLine(process.stderr)\n await writeStderr(format(label('debug', message), ...args))\n await writeStderr('\\n')\n}\n\nexport async function warn(message: string, ...args: any[]): Promise<void> {\n await clearLine(process.stderr)\n await writeStderr(format(label('warn', message), ...args))\n await writeStderr('\\n')\n}\n\nexport async function error(message: string, ...args: any[]): Promise<void> {\n await clearLine(process.stderr)\n await writeStderr(format(label('error', message), ...args))\n await writeStderr('\\n')\n}\n\nexport function getStdinAsString(): Promise<string> {\n let result = ''\n return new Promise(resolve => {\n process.stdin.setEncoding('utf-8')\n process.stdin.on('readable', () => {\n let chunk\n while ((chunk = process.stdin.read())) {\n result += chunk\n }\n })\n process.stdin.on('end', () => {\n resolve(result)\n })\n })\n}\n","import {type Opts, transformWithTs} from '@formatjs/ts-transformer'\nimport * as ts from 'typescript'\nimport {debug} from '#packages/cli-lib/console_utils.js'\n/**\n * Invoid TypeScript module transpilation with our TS transformer\n * @param opts Formatjs TS Transformer opt\n * @param fn filename\n */\nexport function parseScript(opts: Opts, fn?: string) {\n return (source: string): void => {\n let output\n try {\n debug('Using TS compiler to process file', fn)\n output = ts.transpileModule(source, {\n compilerOptions: {\n allowJs: true,\n target: ts.ScriptTarget.ESNext,\n noEmit: true,\n experimentalDecorators: true,\n },\n reportDiagnostics: true,\n fileName: fn,\n transformers: {\n before: [transformWithTs(ts, opts)],\n },\n })\n } catch (e) {\n if (e instanceof Error) {\n e.message = `Error processing file ${fn} \n${e.message || ''}`\n }\n throw e\n }\n if (output.diagnostics) {\n const errs = output.diagnostics.filter(\n d => d.category === ts.DiagnosticCategory.Error\n )\n if (errs.length) {\n throw new Error(\n ts.formatDiagnosticsWithColorAndContext(errs, {\n getCanonicalFileName: fileName => fileName,\n getCurrentDirectory: () => process.cwd(),\n getNewLine: () => ts.sys.newLine,\n })\n )\n }\n }\n }\n}\n"],"mappings":";;;;;AAMA,MAAM,mBAAmB;AAEzB,MAAa,cACX,UAAU,QAAQ,OAAO,MAAM,CAAC,KAAK,QAAQ,OAAO;AACtD,MAAa,cACX,UAAU,QAAQ,OAAO,MAAM,CAAC,KAAK,QAAQ,OAAO;AAItD,eAAsBA,YACpB,UACe;CACf,IAAI,CAAC,SAAS,aAAa;MACrB,SAAS,OAAO;GAElB,IAAI,SAAS,UAAU,GACrB,MAAM,YAAY,KAAK,IAAI,OAAO,SAAS,UAAU,EAAE,GAAG;GAE5D,MAAM,YAAY,KAAK;;QAGpB;EACL,UAAgB,UAAU,iBAAiB;EAC3C,SAAe,UAAU,GAAG,KAAA,EAAU;;;AAM1C,MAAM,eAA6D;CACjE,OAAO;CACP,MAAM;CACN,OAAO;CACR;AAED,SAAS,MAAM,OAAiB,SAAiB;CAC/C,OAAO,oBAAoB,UACzB,aAAa,QACb,MAAM,aAAa,CACpB,CAAC,IAAI;;AAGR,eAAsB,MAAM,SAAiB,GAAG,MAA4B;CAC1E,IAAI,QAAQ,IAAI,cAAc,SAC5B;CAEF,MAAMA,YAAU,QAAQ,OAAO;CAC/B,MAAM,YAAY,OAAO,MAAM,SAAS,QAAQ,EAAE,GAAG,KAAK,CAAC;CAC3D,MAAM,YAAY,KAAK;;AAGzB,eAAsB,KAAK,SAAiB,GAAG,MAA4B;CACzE,MAAMA,YAAU,QAAQ,OAAO;CAC/B,MAAM,YAAY,OAAO,MAAM,QAAQ,QAAQ,EAAE,GAAG,KAAK,CAAC;CAC1D,MAAM,YAAY,KAAK;;AASzB,SAAgB,mBAAoC;CAClD,IAAI,SAAS;CACb,OAAO,IAAI,SAAQ,YAAW;EAC5B,QAAQ,MAAM,YAAY,QAAQ;EAClC,QAAQ,MAAM,GAAG,kBAAkB;GACjC,IAAI;GACJ,OAAQ,QAAQ,QAAQ,MAAM,MAAM,EAClC,UAAU;IAEZ;EACF,QAAQ,MAAM,GAAG,aAAa;GAC5B,QAAQ,OAAO;IACf;GACF;;;;;;;;;AC1EJ,SAAgB,YAAY,MAAY,IAAa;CACnD,QAAQ,WAAyB;EAC/B,IAAI;EACJ,IAAI;GACF,MAAM,qCAAqC,GAAG;GAC9C,SAAS,GAAG,gBAAgB,QAAQ;IAClC,iBAAiB;KACf,SAAS;KACT,QAAQ,GAAG,aAAa;KACxB,QAAQ;KACR,wBAAwB;KACzB;IACD,mBAAmB;IACnB,UAAU;IACV,cAAc,EACZ,QAAQ,CAAC,gBAAgB,IAAI,KAAK,CAAC,EACpC;IACF,CAAC;WACK,GAAG;GACV,IAAI,aAAa,OACf,EAAE,UAAU,yBAAyB,GAAG;EAC9C,EAAE,WAAW;GAET,MAAM;;EAER,IAAI,OAAO,aAAa;GACtB,MAAM,OAAO,OAAO,YAAY,QAC9B,MAAK,EAAE,aAAa,GAAG,mBAAmB,MAC3C;GACD,IAAI,KAAK,QACP,MAAM,IAAI,MACR,GAAG,qCAAqC,MAAM;IAC5C,uBAAsB,aAAY;IAClC,2BAA2B,QAAQ,KAAK;IACxC,kBAAkB,GAAG,IAAI;IAC1B,CAAC,CACH"}
@@ -1 +1 @@
1
- {"version":3,"file":"svelte_extractor-Fr1Z3JHX.js","names":[],"sources":["../svelte_extractor.ts"],"sourcesContent":["import {parse} from 'svelte/compiler'\n\nexport type ScriptParseFn = (source: string) => void\n\ninterface SvelteNode {\n type: string\n start: number\n end: number\n [key: string]: any\n}\n\nfunction walkFragment(\n fragment: SvelteNode | null | undefined,\n source: string,\n parseScriptFn: ScriptParseFn\n): void {\n if (!fragment || !fragment.nodes) {\n return\n }\n for (const node of fragment.nodes) {\n walkNode(node, source, parseScriptFn)\n }\n}\n\nfunction extractExpression(\n expression: SvelteNode | null | undefined,\n source: string,\n parseScriptFn: ScriptParseFn\n): void {\n if (!expression || expression.start == null || expression.end == null) {\n return\n }\n const content = source.slice(expression.start, expression.end)\n try {\n parseScriptFn(`(${content})`)\n } catch (e) {\n console.warn(\n `Failed to parse \"${content}\". Ignore this if content has no extractable message`,\n e\n )\n }\n}\n\nfunction walkNode(\n node: SvelteNode,\n source: string,\n parseScriptFn: ScriptParseFn\n): void {\n switch (node.type) {\n case 'ExpressionTag':\n extractExpression(node.expression, source, parseScriptFn)\n break\n\n case 'ConstTag':\n // {@const x = expr} — the declaration contains the expression\n if (node.declaration) {\n const content = source.slice(\n node.declaration.start,\n node.declaration.end\n )\n try {\n parseScriptFn(content)\n } catch (e) {\n console.warn(\n `Failed to parse const declaration. Ignore this if content has no extractable message`,\n e\n )\n }\n }\n break\n\n case 'IfBlock':\n extractExpression(node.test, source, parseScriptFn)\n walkFragment(node.consequent, source, parseScriptFn)\n walkFragment(node.alternate, source, parseScriptFn)\n break\n\n case 'EachBlock':\n extractExpression(node.expression, source, parseScriptFn)\n if (node.key) {\n extractExpression(node.key, source, parseScriptFn)\n }\n walkFragment(node.body, source, parseScriptFn)\n walkFragment(node.fallback, source, parseScriptFn)\n break\n\n case 'AwaitBlock':\n extractExpression(node.expression, source, parseScriptFn)\n walkFragment(node.pending, source, parseScriptFn)\n walkFragment(node.then, source, parseScriptFn)\n walkFragment(node.catch, source, parseScriptFn)\n break\n\n case 'KeyBlock':\n extractExpression(node.expression, source, parseScriptFn)\n walkFragment(node.fragment, source, parseScriptFn)\n break\n\n case 'SnippetBlock':\n walkFragment(node.body, source, parseScriptFn)\n break\n\n case 'RegularElement':\n case 'Component':\n case 'SvelteComponent':\n case 'SvelteElement':\n case 'SvelteHead':\n case 'SvelteBody':\n case 'SvelteWindow':\n case 'SvelteDocument':\n case 'SlotElement':\n case 'SvelteSelf':\n case 'SvelteFragment':\n // Process attributes\n if (node.attributes) {\n for (const attr of node.attributes) {\n if (attr.type === 'Attribute') {\n if (Array.isArray(attr.value)) {\n for (const v of attr.value) {\n if (v.type === 'ExpressionTag') {\n extractExpression(v.expression, source, parseScriptFn)\n }\n }\n } else if (\n typeof attr.value === 'object' &&\n attr.value !== null &&\n attr.value.type === 'ExpressionTag'\n ) {\n extractExpression(attr.value.expression, source, parseScriptFn)\n }\n } else if (attr.type === 'SpreadAttribute') {\n extractExpression(attr.expression, source, parseScriptFn)\n } else if (attr.expression) {\n // Directive nodes (OnDirective, BindDirective, etc.)\n extractExpression(attr.expression, source, parseScriptFn)\n }\n }\n }\n // Process children\n walkFragment(node.fragment, source, parseScriptFn)\n break\n }\n}\n\nexport function parseFile(\n source: string,\n filename: string,\n parseScriptFn: ScriptParseFn\n): void {\n const ast = parse(source, {modern: true, filename})\n\n // Walk the template fragment\n walkFragment(ast.fragment as unknown as SvelteNode, source, parseScriptFn)\n\n // Process <script> block\n if (ast.instance) {\n const program = ast.instance.content as unknown as SvelteNode\n parseScriptFn(source.slice(program.start, program.end))\n }\n\n // Process <script context=\"module\"> / <script module> block\n if (ast.module) {\n const program = ast.module.content as unknown as SvelteNode\n parseScriptFn(source.slice(program.start, program.end))\n }\n}\n"],"mappings":";;AAWA,SAAS,aACP,UACA,QACA,eACM;AACN,KAAI,CAAC,YAAY,CAAC,SAAS,MACzB;AAEF,MAAK,MAAM,QAAQ,SAAS,MAC1B,UAAS,MAAM,QAAQ,cAAc;;AAIzC,SAAS,kBACP,YACA,QACA,eACM;AACN,KAAI,CAAC,cAAc,WAAW,SAAS,QAAQ,WAAW,OAAO,KAC/D;CAEF,MAAM,UAAU,OAAO,MAAM,WAAW,OAAO,WAAW,IAAI;AAC9D,KAAI;AACF,gBAAc,IAAI,QAAQ,GAAG;UACtB,GAAG;AACV,UAAQ,KACN,oBAAoB,QAAQ,uDAC5B,EACD;;;AAIL,SAAS,SACP,MACA,QACA,eACM;AACN,SAAQ,KAAK,MAAb;EACE,KAAK;AACH,qBAAkB,KAAK,YAAY,QAAQ,cAAc;AACzD;EAEF,KAAK;AAEH,OAAI,KAAK,aAAa;IACpB,MAAM,UAAU,OAAO,MACrB,KAAK,YAAY,OACjB,KAAK,YAAY,IAClB;AACD,QAAI;AACF,mBAAc,QAAQ;aACf,GAAG;AACV,aAAQ,KACN,wFACA,EACD;;;AAGL;EAEF,KAAK;AACH,qBAAkB,KAAK,MAAM,QAAQ,cAAc;AACnD,gBAAa,KAAK,YAAY,QAAQ,cAAc;AACpD,gBAAa,KAAK,WAAW,QAAQ,cAAc;AACnD;EAEF,KAAK;AACH,qBAAkB,KAAK,YAAY,QAAQ,cAAc;AACzD,OAAI,KAAK,IACP,mBAAkB,KAAK,KAAK,QAAQ,cAAc;AAEpD,gBAAa,KAAK,MAAM,QAAQ,cAAc;AAC9C,gBAAa,KAAK,UAAU,QAAQ,cAAc;AAClD;EAEF,KAAK;AACH,qBAAkB,KAAK,YAAY,QAAQ,cAAc;AACzD,gBAAa,KAAK,SAAS,QAAQ,cAAc;AACjD,gBAAa,KAAK,MAAM,QAAQ,cAAc;AAC9C,gBAAa,KAAK,OAAO,QAAQ,cAAc;AAC/C;EAEF,KAAK;AACH,qBAAkB,KAAK,YAAY,QAAQ,cAAc;AACzD,gBAAa,KAAK,UAAU,QAAQ,cAAc;AAClD;EAEF,KAAK;AACH,gBAAa,KAAK,MAAM,QAAQ,cAAc;AAC9C;EAEF,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;AAEH,OAAI,KAAK;SACF,MAAM,QAAQ,KAAK,WACtB,KAAI,KAAK,SAAS;SACZ,MAAM,QAAQ,KAAK,MAAM;WACtB,MAAM,KAAK,KAAK,MACnB,KAAI,EAAE,SAAS,gBACb,mBAAkB,EAAE,YAAY,QAAQ,cAAc;gBAI1D,OAAO,KAAK,UAAU,YACtB,KAAK,UAAU,QACf,KAAK,MAAM,SAAS,gBAEpB,mBAAkB,KAAK,MAAM,YAAY,QAAQ,cAAc;eAExD,KAAK,SAAS,kBACvB,mBAAkB,KAAK,YAAY,QAAQ,cAAc;aAChD,KAAK,WAEd,mBAAkB,KAAK,YAAY,QAAQ,cAAc;;AAK/D,gBAAa,KAAK,UAAU,QAAQ,cAAc;AAClD;;;AAIN,SAAgB,UACd,QACA,UACA,eACM;CACN,MAAM,MAAM,MAAM,QAAQ;EAAC,QAAQ;EAAM;EAAS,CAAC;AAGnD,cAAa,IAAI,UAAmC,QAAQ,cAAc;AAG1E,KAAI,IAAI,UAAU;EAChB,MAAM,UAAU,IAAI,SAAS;AAC7B,gBAAc,OAAO,MAAM,QAAQ,OAAO,QAAQ,IAAI,CAAC;;AAIzD,KAAI,IAAI,QAAQ;EACd,MAAM,UAAU,IAAI,OAAO;AAC3B,gBAAc,OAAO,MAAM,QAAQ,OAAO,QAAQ,IAAI,CAAC"}
1
+ {"version":3,"file":"svelte_extractor-Fr1Z3JHX.js","names":[],"sources":["../svelte_extractor.ts"],"sourcesContent":["import {parse} from 'svelte/compiler'\n\nexport type ScriptParseFn = (source: string) => void\n\ninterface SvelteNode {\n type: string\n start: number\n end: number\n [key: string]: any\n}\n\nfunction walkFragment(\n fragment: SvelteNode | null | undefined,\n source: string,\n parseScriptFn: ScriptParseFn\n): void {\n if (!fragment || !fragment.nodes) {\n return\n }\n for (const node of fragment.nodes) {\n walkNode(node, source, parseScriptFn)\n }\n}\n\nfunction extractExpression(\n expression: SvelteNode | null | undefined,\n source: string,\n parseScriptFn: ScriptParseFn\n): void {\n if (!expression || expression.start == null || expression.end == null) {\n return\n }\n const content = source.slice(expression.start, expression.end)\n try {\n parseScriptFn(`(${content})`)\n } catch (e) {\n console.warn(\n `Failed to parse \"${content}\". Ignore this if content has no extractable message`,\n e\n )\n }\n}\n\nfunction walkNode(\n node: SvelteNode,\n source: string,\n parseScriptFn: ScriptParseFn\n): void {\n switch (node.type) {\n case 'ExpressionTag':\n extractExpression(node.expression, source, parseScriptFn)\n break\n\n case 'ConstTag':\n // {@const x = expr} — the declaration contains the expression\n if (node.declaration) {\n const content = source.slice(\n node.declaration.start,\n node.declaration.end\n )\n try {\n parseScriptFn(content)\n } catch (e) {\n console.warn(\n `Failed to parse const declaration. Ignore this if content has no extractable message`,\n e\n )\n }\n }\n break\n\n case 'IfBlock':\n extractExpression(node.test, source, parseScriptFn)\n walkFragment(node.consequent, source, parseScriptFn)\n walkFragment(node.alternate, source, parseScriptFn)\n break\n\n case 'EachBlock':\n extractExpression(node.expression, source, parseScriptFn)\n if (node.key) {\n extractExpression(node.key, source, parseScriptFn)\n }\n walkFragment(node.body, source, parseScriptFn)\n walkFragment(node.fallback, source, parseScriptFn)\n break\n\n case 'AwaitBlock':\n extractExpression(node.expression, source, parseScriptFn)\n walkFragment(node.pending, source, parseScriptFn)\n walkFragment(node.then, source, parseScriptFn)\n walkFragment(node.catch, source, parseScriptFn)\n break\n\n case 'KeyBlock':\n extractExpression(node.expression, source, parseScriptFn)\n walkFragment(node.fragment, source, parseScriptFn)\n break\n\n case 'SnippetBlock':\n walkFragment(node.body, source, parseScriptFn)\n break\n\n case 'RegularElement':\n case 'Component':\n case 'SvelteComponent':\n case 'SvelteElement':\n case 'SvelteHead':\n case 'SvelteBody':\n case 'SvelteWindow':\n case 'SvelteDocument':\n case 'SlotElement':\n case 'SvelteSelf':\n case 'SvelteFragment':\n // Process attributes\n if (node.attributes) {\n for (const attr of node.attributes) {\n if (attr.type === 'Attribute') {\n if (Array.isArray(attr.value)) {\n for (const v of attr.value) {\n if (v.type === 'ExpressionTag') {\n extractExpression(v.expression, source, parseScriptFn)\n }\n }\n } else if (\n typeof attr.value === 'object' &&\n attr.value !== null &&\n attr.value.type === 'ExpressionTag'\n ) {\n extractExpression(attr.value.expression, source, parseScriptFn)\n }\n } else if (attr.type === 'SpreadAttribute') {\n extractExpression(attr.expression, source, parseScriptFn)\n } else if (attr.expression) {\n // Directive nodes (OnDirective, BindDirective, etc.)\n extractExpression(attr.expression, source, parseScriptFn)\n }\n }\n }\n // Process children\n walkFragment(node.fragment, source, parseScriptFn)\n break\n }\n}\n\nexport function parseFile(\n source: string,\n filename: string,\n parseScriptFn: ScriptParseFn\n): void {\n const ast = parse(source, {modern: true, filename})\n\n // Walk the template fragment\n walkFragment(ast.fragment as unknown as SvelteNode, source, parseScriptFn)\n\n // Process <script> block\n if (ast.instance) {\n const program = ast.instance.content as unknown as SvelteNode\n parseScriptFn(source.slice(program.start, program.end))\n }\n\n // Process <script context=\"module\"> / <script module> block\n if (ast.module) {\n const program = ast.module.content as unknown as SvelteNode\n parseScriptFn(source.slice(program.start, program.end))\n }\n}\n"],"mappings":";;AAWA,SAAS,aACP,UACA,QACA,eACM;CACN,IAAI,CAAC,YAAY,CAAC,SAAS,OACzB;CAEF,KAAK,MAAM,QAAQ,SAAS,OAC1B,SAAS,MAAM,QAAQ,cAAc;;AAIzC,SAAS,kBACP,YACA,QACA,eACM;CACN,IAAI,CAAC,cAAc,WAAW,SAAS,QAAQ,WAAW,OAAO,MAC/D;CAEF,MAAM,UAAU,OAAO,MAAM,WAAW,OAAO,WAAW,IAAI;CAC9D,IAAI;EACF,cAAc,IAAI,QAAQ,GAAG;UACtB,GAAG;EACV,QAAQ,KACN,oBAAoB,QAAQ,uDAC5B,EACD;;;AAIL,SAAS,SACP,MACA,QACA,eACM;CACN,QAAQ,KAAK,MAAb;EACE,KAAK;GACH,kBAAkB,KAAK,YAAY,QAAQ,cAAc;GACzD;EAEF,KAAK;GAEH,IAAI,KAAK,aAAa;IACpB,MAAM,UAAU,OAAO,MACrB,KAAK,YAAY,OACjB,KAAK,YAAY,IAClB;IACD,IAAI;KACF,cAAc,QAAQ;aACf,GAAG;KACV,QAAQ,KACN,wFACA,EACD;;;GAGL;EAEF,KAAK;GACH,kBAAkB,KAAK,MAAM,QAAQ,cAAc;GACnD,aAAa,KAAK,YAAY,QAAQ,cAAc;GACpD,aAAa,KAAK,WAAW,QAAQ,cAAc;GACnD;EAEF,KAAK;GACH,kBAAkB,KAAK,YAAY,QAAQ,cAAc;GACzD,IAAI,KAAK,KACP,kBAAkB,KAAK,KAAK,QAAQ,cAAc;GAEpD,aAAa,KAAK,MAAM,QAAQ,cAAc;GAC9C,aAAa,KAAK,UAAU,QAAQ,cAAc;GAClD;EAEF,KAAK;GACH,kBAAkB,KAAK,YAAY,QAAQ,cAAc;GACzD,aAAa,KAAK,SAAS,QAAQ,cAAc;GACjD,aAAa,KAAK,MAAM,QAAQ,cAAc;GAC9C,aAAa,KAAK,OAAO,QAAQ,cAAc;GAC/C;EAEF,KAAK;GACH,kBAAkB,KAAK,YAAY,QAAQ,cAAc;GACzD,aAAa,KAAK,UAAU,QAAQ,cAAc;GAClD;EAEF,KAAK;GACH,aAAa,KAAK,MAAM,QAAQ,cAAc;GAC9C;EAEF,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;EACL,KAAK;GAEH,IAAI,KAAK;SACF,MAAM,QAAQ,KAAK,YACtB,IAAI,KAAK,SAAS;SACZ,MAAM,QAAQ,KAAK,MAAM;WACtB,MAAM,KAAK,KAAK,OACnB,IAAI,EAAE,SAAS,iBACb,kBAAkB,EAAE,YAAY,QAAQ,cAAc;YAGrD,IACL,OAAO,KAAK,UAAU,YACtB,KAAK,UAAU,QACf,KAAK,MAAM,SAAS,iBAEpB,kBAAkB,KAAK,MAAM,YAAY,QAAQ,cAAc;WAE5D,IAAI,KAAK,SAAS,mBACvB,kBAAkB,KAAK,YAAY,QAAQ,cAAc;SACpD,IAAI,KAAK,YAEd,kBAAkB,KAAK,YAAY,QAAQ,cAAc;;GAK/D,aAAa,KAAK,UAAU,QAAQ,cAAc;GAClD;;;AAIN,SAAgB,UACd,QACA,UACA,eACM;CACN,MAAM,MAAM,MAAM,QAAQ;EAAC,QAAQ;EAAM;EAAS,CAAC;CAGnD,aAAa,IAAI,UAAmC,QAAQ,cAAc;CAG1E,IAAI,IAAI,UAAU;EAChB,MAAM,UAAU,IAAI,SAAS;EAC7B,cAAc,OAAO,MAAM,QAAQ,OAAO,QAAQ,IAAI,CAAC;;CAIzD,IAAI,IAAI,QAAQ;EACd,MAAM,UAAU,IAAI,OAAO;EAC3B,cAAc,OAAO,MAAM,QAAQ,OAAO,QAAQ,IAAI,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"vue_extractor-Btw05cfG.js","names":[],"sources":["../vue_extractor.ts"],"sourcesContent":["import {\n NodeTypes,\n type TextNode,\n type CompoundExpressionNode,\n type DirectiveNode,\n type ElementNode,\n type InterpolationNode,\n type RootNode,\n type SimpleExpressionNode,\n type TemplateChildNode,\n} from '@vue/compiler-core'\nimport {parse} from 'vue/compiler-sfc'\n\nexport type ScriptParseFn = (source: string) => void\n\nfunction walk(\n node:\n | RootNode\n | TemplateChildNode\n | SimpleExpressionNode\n | CompoundExpressionNode\n | InterpolationNode\n | TextNode\n | string\n | symbol\n | undefined,\n visitor: (\n node:\n | ElementNode\n | InterpolationNode\n | CompoundExpressionNode\n | SimpleExpressionNode\n ) => void\n) {\n if (typeof node !== 'object' || node == null) {\n return\n }\n if (node.type === NodeTypes.ROOT) {\n node.children.forEach(n => walk(n, visitor))\n return\n }\n if (\n node.type !== NodeTypes.ELEMENT &&\n node.type !== NodeTypes.COMPOUND_EXPRESSION &&\n node.type !== NodeTypes.INTERPOLATION\n ) {\n return\n }\n visitor(node)\n if (node.type === NodeTypes.INTERPOLATION) {\n visitor(node.content)\n } else if (node.type === NodeTypes.ELEMENT) {\n node.children.forEach(n => walk(n, visitor))\n node.props\n .filter(\n (prop): prop is DirectiveNode => prop.type === NodeTypes.DIRECTIVE\n )\n .filter(prop => !!prop.exp)\n .forEach(prop => visitor(prop.exp!))\n } else {\n node.children.forEach(n => walk(n, visitor))\n }\n}\n\nfunction templateSimpleExpressionNodeVisitor(parseScriptFn: ScriptParseFn) {\n return (\n n:\n | ElementNode\n | CompoundExpressionNode\n | CompoundExpressionNode['children'][0]\n ) => {\n if (typeof n !== 'object') {\n return\n }\n if (n.type !== NodeTypes.SIMPLE_EXPRESSION) {\n return\n }\n\n const {content} = n as SimpleExpressionNode\n // Wrap this in () since a vue comp node attribute can just be\n // an object literal which, by itself is invalid TS\n // but with () it becomes an ExpressionStatement\n try {\n parseScriptFn(`(${content})`)\n } catch (e) {\n console.warn(\n `Failed to parse \"${content}\". Ignore this if content has no extractable message`,\n e\n )\n }\n }\n}\n\nexport function parseFile(\n source: string,\n filename: string,\n parseScriptFn: ScriptParseFn\n): void {\n const {descriptor, errors} = parse(source, {\n filename,\n })\n if (errors.length) {\n throw errors[0]\n }\n const {script, scriptSetup, template} = descriptor\n\n if (template) {\n walk(template.ast, templateSimpleExpressionNodeVisitor(parseScriptFn))\n }\n\n if (script) {\n parseScriptFn(script.content)\n }\n\n if (scriptSetup) {\n parseScriptFn(scriptSetup.content)\n }\n}\n"],"mappings":";;;AAeA,SAAS,KACP,MAUA,SAOA;AACA,KAAI,OAAO,SAAS,YAAY,QAAQ,KACtC;AAEF,KAAI,KAAK,SAAS,UAAU,MAAM;AAChC,OAAK,SAAS,SAAQ,MAAK,KAAK,GAAG,QAAQ,CAAC;AAC5C;;AAEF,KACE,KAAK,SAAS,UAAU,WACxB,KAAK,SAAS,UAAU,uBACxB,KAAK,SAAS,UAAU,cAExB;AAEF,SAAQ,KAAK;AACb,KAAI,KAAK,SAAS,UAAU,cAC1B,SAAQ,KAAK,QAAQ;UACZ,KAAK,SAAS,UAAU,SAAS;AAC1C,OAAK,SAAS,SAAQ,MAAK,KAAK,GAAG,QAAQ,CAAC;AAC5C,OAAK,MACF,QACE,SAAgC,KAAK,SAAS,UAAU,UAC1D,CACA,QAAO,SAAQ,CAAC,CAAC,KAAK,IAAI,CAC1B,SAAQ,SAAQ,QAAQ,KAAK,IAAK,CAAC;OAEtC,MAAK,SAAS,SAAQ,MAAK,KAAK,GAAG,QAAQ,CAAC;;AAIhD,SAAS,oCAAoC,eAA8B;AACzE,SACE,MAIG;AACH,MAAI,OAAO,MAAM,SACf;AAEF,MAAI,EAAE,SAAS,UAAU,kBACvB;EAGF,MAAM,EAAC,YAAW;AAIlB,MAAI;AACF,iBAAc,IAAI,QAAQ,GAAG;WACtB,GAAG;AACV,WAAQ,KACN,oBAAoB,QAAQ,uDAC5B,EACD;;;;AAKP,SAAgB,UACd,QACA,UACA,eACM;CACN,MAAM,EAAC,YAAY,WAAU,MAAM,QAAQ,EACzC,UACD,CAAC;AACF,KAAI,OAAO,OACT,OAAM,OAAO;CAEf,MAAM,EAAC,QAAQ,aAAa,aAAY;AAExC,KAAI,SACF,MAAK,SAAS,KAAK,oCAAoC,cAAc,CAAC;AAGxE,KAAI,OACF,eAAc,OAAO,QAAQ;AAG/B,KAAI,YACF,eAAc,YAAY,QAAQ"}
1
+ {"version":3,"file":"vue_extractor-Btw05cfG.js","names":[],"sources":["../vue_extractor.ts"],"sourcesContent":["import {\n NodeTypes,\n type TextNode,\n type CompoundExpressionNode,\n type DirectiveNode,\n type ElementNode,\n type InterpolationNode,\n type RootNode,\n type SimpleExpressionNode,\n type TemplateChildNode,\n} from '@vue/compiler-core'\nimport {parse} from 'vue/compiler-sfc'\n\nexport type ScriptParseFn = (source: string) => void\n\nfunction walk(\n node:\n | RootNode\n | TemplateChildNode\n | SimpleExpressionNode\n | CompoundExpressionNode\n | InterpolationNode\n | TextNode\n | string\n | symbol\n | undefined,\n visitor: (\n node:\n | ElementNode\n | InterpolationNode\n | CompoundExpressionNode\n | SimpleExpressionNode\n ) => void\n) {\n if (typeof node !== 'object' || node == null) {\n return\n }\n if (node.type === NodeTypes.ROOT) {\n node.children.forEach(n => walk(n, visitor))\n return\n }\n if (\n node.type !== NodeTypes.ELEMENT &&\n node.type !== NodeTypes.COMPOUND_EXPRESSION &&\n node.type !== NodeTypes.INTERPOLATION\n ) {\n return\n }\n visitor(node)\n if (node.type === NodeTypes.INTERPOLATION) {\n visitor(node.content)\n } else if (node.type === NodeTypes.ELEMENT) {\n node.children.forEach(n => walk(n, visitor))\n node.props\n .filter(\n (prop): prop is DirectiveNode => prop.type === NodeTypes.DIRECTIVE\n )\n .filter(prop => !!prop.exp)\n .forEach(prop => visitor(prop.exp!))\n } else {\n node.children.forEach(n => walk(n, visitor))\n }\n}\n\nfunction templateSimpleExpressionNodeVisitor(parseScriptFn: ScriptParseFn) {\n return (\n n:\n | ElementNode\n | CompoundExpressionNode\n | CompoundExpressionNode['children'][0]\n ) => {\n if (typeof n !== 'object') {\n return\n }\n if (n.type !== NodeTypes.SIMPLE_EXPRESSION) {\n return\n }\n\n const {content} = n as SimpleExpressionNode\n // Wrap this in () since a vue comp node attribute can just be\n // an object literal which, by itself is invalid TS\n // but with () it becomes an ExpressionStatement\n try {\n parseScriptFn(`(${content})`)\n } catch (e) {\n console.warn(\n `Failed to parse \"${content}\". Ignore this if content has no extractable message`,\n e\n )\n }\n }\n}\n\nexport function parseFile(\n source: string,\n filename: string,\n parseScriptFn: ScriptParseFn\n): void {\n const {descriptor, errors} = parse(source, {\n filename,\n })\n if (errors.length) {\n throw errors[0]\n }\n const {script, scriptSetup, template} = descriptor\n\n if (template) {\n walk(template.ast, templateSimpleExpressionNodeVisitor(parseScriptFn))\n }\n\n if (script) {\n parseScriptFn(script.content)\n }\n\n if (scriptSetup) {\n parseScriptFn(scriptSetup.content)\n }\n}\n"],"mappings":";;;AAeA,SAAS,KACP,MAUA,SAOA;CACA,IAAI,OAAO,SAAS,YAAY,QAAQ,MACtC;CAEF,IAAI,KAAK,SAAS,UAAU,MAAM;EAChC,KAAK,SAAS,SAAQ,MAAK,KAAK,GAAG,QAAQ,CAAC;EAC5C;;CAEF,IACE,KAAK,SAAS,UAAU,WACxB,KAAK,SAAS,UAAU,uBACxB,KAAK,SAAS,UAAU,eAExB;CAEF,QAAQ,KAAK;CACb,IAAI,KAAK,SAAS,UAAU,eAC1B,QAAQ,KAAK,QAAQ;MAChB,IAAI,KAAK,SAAS,UAAU,SAAS;EAC1C,KAAK,SAAS,SAAQ,MAAK,KAAK,GAAG,QAAQ,CAAC;EAC5C,KAAK,MACF,QACE,SAAgC,KAAK,SAAS,UAAU,UAC1D,CACA,QAAO,SAAQ,CAAC,CAAC,KAAK,IAAI,CAC1B,SAAQ,SAAQ,QAAQ,KAAK,IAAK,CAAC;QAEtC,KAAK,SAAS,SAAQ,MAAK,KAAK,GAAG,QAAQ,CAAC;;AAIhD,SAAS,oCAAoC,eAA8B;CACzE,QACE,MAIG;EACH,IAAI,OAAO,MAAM,UACf;EAEF,IAAI,EAAE,SAAS,UAAU,mBACvB;EAGF,MAAM,EAAC,YAAW;EAIlB,IAAI;GACF,cAAc,IAAI,QAAQ,GAAG;WACtB,GAAG;GACV,QAAQ,KACN,oBAAoB,QAAQ,uDAC5B,EACD;;;;AAKP,SAAgB,UACd,QACA,UACA,eACM;CACN,MAAM,EAAC,YAAY,WAAU,MAAM,QAAQ,EACzC,UACD,CAAC;CACF,IAAI,OAAO,QACT,MAAM,OAAO;CAEf,MAAM,EAAC,QAAQ,aAAa,aAAY;CAExC,IAAI,UACF,KAAK,SAAS,KAAK,oCAAoC,cAAc,CAAC;CAGxE,IAAI,QACF,cAAc,OAAO,QAAQ;CAG/B,IAAI,aACF,cAAc,YAAY,QAAQ"}