@intlayer/cli 8.2.4 → 8.3.0-canary.0

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.
Files changed (151) hide show
  1. package/dist/cjs/IntlayerEventListener.cjs +1 -1
  2. package/dist/cjs/IntlayerEventListener.cjs.map +1 -1
  3. package/dist/cjs/_virtual/_utils_asset.cjs +1 -1
  4. package/dist/cjs/auth/login.cjs +2 -2
  5. package/dist/cjs/auth/login.cjs.map +1 -1
  6. package/dist/cjs/build.cjs +1 -1
  7. package/dist/cjs/ci.cjs +1 -1
  8. package/dist/cjs/cli.cjs +1 -1
  9. package/dist/cjs/cli.cjs.map +1 -1
  10. package/dist/cjs/config.cjs +1 -1
  11. package/dist/cjs/editor.cjs +1 -1
  12. package/dist/cjs/extract.cjs +1 -1
  13. package/dist/cjs/extract.cjs.map +1 -1
  14. package/dist/cjs/fill/fill.cjs +1 -1
  15. package/dist/cjs/fill/fill.cjs.map +1 -1
  16. package/dist/cjs/fill/formatAutoFilledFilePath.cjs +1 -1
  17. package/dist/cjs/fill/formatAutoFilledFilePath.cjs.map +1 -1
  18. package/dist/cjs/fill/formatFillData.cjs +1 -1
  19. package/dist/cjs/fill/formatFillData.cjs.map +1 -1
  20. package/dist/cjs/fill/getAvailableLocalesInDictionary.cjs.map +1 -1
  21. package/dist/cjs/fill/getFilterMissingContentPerLocale.cjs.map +1 -1
  22. package/dist/cjs/fill/listTranslationsTasks.cjs +1 -1
  23. package/dist/cjs/fill/listTranslationsTasks.cjs.map +1 -1
  24. package/dist/cjs/fill/translateDictionary.cjs +1 -1
  25. package/dist/cjs/fill/translateDictionary.cjs.map +1 -1
  26. package/dist/cjs/fill/writeFill.cjs +1 -1
  27. package/dist/cjs/fill/writeFill.cjs.map +1 -1
  28. package/dist/cjs/getTargetDictionary.cjs +1 -1
  29. package/dist/cjs/getTargetDictionary.cjs.map +1 -1
  30. package/dist/cjs/index.cjs +1 -1
  31. package/dist/cjs/init.cjs +1 -1
  32. package/dist/cjs/initMCP.cjs +1 -1
  33. package/dist/cjs/initSkills.cjs +1 -1
  34. package/dist/cjs/listContentDeclaration.cjs +1 -1
  35. package/dist/cjs/listContentDeclaration.cjs.map +1 -1
  36. package/dist/cjs/listProjects.cjs +1 -1
  37. package/dist/cjs/liveSync.cjs +4 -4
  38. package/dist/cjs/liveSync.cjs.map +1 -1
  39. package/dist/cjs/pull.cjs +1 -1
  40. package/dist/cjs/pull.cjs.map +1 -1
  41. package/dist/cjs/push/push.cjs +1 -1
  42. package/dist/cjs/push/push.cjs.map +1 -1
  43. package/dist/cjs/pushConfig.cjs +1 -1
  44. package/dist/cjs/reviewDoc/reviewDoc.cjs +1 -1
  45. package/dist/cjs/reviewDoc/reviewDoc.cjs.map +1 -1
  46. package/dist/cjs/reviewDoc/reviewDocBlockAware.cjs +1 -1
  47. package/dist/cjs/reviewDoc/reviewDocBlockAware.cjs.map +1 -1
  48. package/dist/cjs/searchDoc.cjs +1 -1
  49. package/dist/cjs/test/listMissingTranslations.cjs.map +1 -1
  50. package/dist/cjs/test/test.cjs +1 -1
  51. package/dist/cjs/translateDoc/translateDoc.cjs +1 -1
  52. package/dist/cjs/translateDoc/translateDoc.cjs.map +1 -1
  53. package/dist/cjs/translateDoc/translateFile.cjs +2 -2
  54. package/dist/cjs/translateDoc/translateFile.cjs.map +1 -1
  55. package/dist/cjs/utils/checkAccess.cjs +1 -1
  56. package/dist/cjs/utils/checkAccess.cjs.map +1 -1
  57. package/dist/cjs/utils/chunkInference.cjs.map +1 -1
  58. package/dist/cjs/utils/getOutputFilePath.cjs.map +1 -1
  59. package/dist/cjs/utils/getParentPackageJSON.cjs +1 -1
  60. package/dist/cjs/utils/setupAI.cjs.map +1 -1
  61. package/dist/cjs/watch.cjs +1 -1
  62. package/dist/esm/IntlayerEventListener.mjs +1 -1
  63. package/dist/esm/IntlayerEventListener.mjs.map +1 -1
  64. package/dist/esm/_virtual/_utils_asset.mjs +1 -1
  65. package/dist/esm/auth/login.mjs +2 -2
  66. package/dist/esm/build.mjs +1 -1
  67. package/dist/esm/ci.mjs +1 -1
  68. package/dist/esm/cli.mjs +1 -1
  69. package/dist/esm/cli.mjs.map +1 -1
  70. package/dist/esm/config.mjs +1 -1
  71. package/dist/esm/editor.mjs +1 -1
  72. package/dist/esm/extract.mjs +1 -1
  73. package/dist/esm/extract.mjs.map +1 -1
  74. package/dist/esm/fill/fill.mjs +1 -1
  75. package/dist/esm/fill/fill.mjs.map +1 -1
  76. package/dist/esm/fill/formatAutoFilledFilePath.mjs +1 -1
  77. package/dist/esm/fill/formatAutoFilledFilePath.mjs.map +1 -1
  78. package/dist/esm/fill/formatFillData.mjs +1 -1
  79. package/dist/esm/fill/formatFillData.mjs.map +1 -1
  80. package/dist/esm/fill/getAvailableLocalesInDictionary.mjs.map +1 -1
  81. package/dist/esm/fill/getFilterMissingContentPerLocale.mjs.map +1 -1
  82. package/dist/esm/fill/listTranslationsTasks.mjs +1 -1
  83. package/dist/esm/fill/listTranslationsTasks.mjs.map +1 -1
  84. package/dist/esm/fill/translateDictionary.mjs +1 -1
  85. package/dist/esm/fill/translateDictionary.mjs.map +1 -1
  86. package/dist/esm/fill/writeFill.mjs +1 -1
  87. package/dist/esm/fill/writeFill.mjs.map +1 -1
  88. package/dist/esm/getTargetDictionary.mjs +1 -1
  89. package/dist/esm/getTargetDictionary.mjs.map +1 -1
  90. package/dist/esm/index.mjs +1 -1
  91. package/dist/esm/init.mjs +1 -1
  92. package/dist/esm/initMCP.mjs +1 -1
  93. package/dist/esm/initSkills.mjs +1 -1
  94. package/dist/esm/listContentDeclaration.mjs +1 -1
  95. package/dist/esm/listContentDeclaration.mjs.map +1 -1
  96. package/dist/esm/listProjects.mjs +1 -1
  97. package/dist/esm/liveSync.mjs +3 -3
  98. package/dist/esm/liveSync.mjs.map +1 -1
  99. package/dist/esm/pull.mjs +1 -1
  100. package/dist/esm/pull.mjs.map +1 -1
  101. package/dist/esm/push/push.mjs +1 -1
  102. package/dist/esm/push/push.mjs.map +1 -1
  103. package/dist/esm/pushConfig.mjs +1 -1
  104. package/dist/esm/reviewDoc/reviewDoc.mjs +1 -1
  105. package/dist/esm/reviewDoc/reviewDoc.mjs.map +1 -1
  106. package/dist/esm/reviewDoc/reviewDocBlockAware.mjs +1 -1
  107. package/dist/esm/reviewDoc/reviewDocBlockAware.mjs.map +1 -1
  108. package/dist/esm/searchDoc.mjs +1 -1
  109. package/dist/esm/test/listMissingTranslations.mjs.map +1 -1
  110. package/dist/esm/test/test.mjs +1 -1
  111. package/dist/esm/translateDoc/translateDoc.mjs +1 -1
  112. package/dist/esm/translateDoc/translateDoc.mjs.map +1 -1
  113. package/dist/esm/translateDoc/translateFile.mjs +2 -2
  114. package/dist/esm/translateDoc/translateFile.mjs.map +1 -1
  115. package/dist/esm/utils/checkAccess.mjs +1 -1
  116. package/dist/esm/utils/checkAccess.mjs.map +1 -1
  117. package/dist/esm/utils/chunkInference.mjs.map +1 -1
  118. package/dist/esm/utils/getOutputFilePath.mjs.map +1 -1
  119. package/dist/esm/utils/getParentPackageJSON.mjs +1 -1
  120. package/dist/esm/utils/setupAI.mjs.map +1 -1
  121. package/dist/esm/watch.mjs +1 -1
  122. package/dist/types/IntlayerEventListener.d.ts +1 -1
  123. package/dist/types/extract.d.ts +2 -1
  124. package/dist/types/extract.d.ts.map +1 -1
  125. package/dist/types/fill/fill.d.ts +1 -1
  126. package/dist/types/fill/formatAutoFilledFilePath.d.ts +3 -2
  127. package/dist/types/fill/formatAutoFilledFilePath.d.ts.map +1 -1
  128. package/dist/types/fill/formatFillData.d.ts +5 -3
  129. package/dist/types/fill/formatFillData.d.ts.map +1 -1
  130. package/dist/types/fill/getAvailableLocalesInDictionary.d.ts +2 -1
  131. package/dist/types/fill/getAvailableLocalesInDictionary.d.ts.map +1 -1
  132. package/dist/types/fill/getFilterMissingContentPerLocale.d.ts +1 -1
  133. package/dist/types/fill/listTranslationsTasks.d.ts +3 -1
  134. package/dist/types/fill/listTranslationsTasks.d.ts.map +1 -1
  135. package/dist/types/fill/translateDictionary.d.ts +2 -1
  136. package/dist/types/fill/translateDictionary.d.ts.map +1 -1
  137. package/dist/types/fill/writeFill.d.ts +3 -1
  138. package/dist/types/fill/writeFill.d.ts.map +1 -1
  139. package/dist/types/getTargetDictionary.d.ts +1 -1
  140. package/dist/types/reviewDoc/reviewDoc.d.ts +1 -1
  141. package/dist/types/reviewDoc/reviewDocBlockAware.d.ts +1 -1
  142. package/dist/types/reviewDoc/reviewDocBlockAware.d.ts.map +1 -1
  143. package/dist/types/test/listMissingTranslations.d.ts +2 -1
  144. package/dist/types/test/listMissingTranslations.d.ts.map +1 -1
  145. package/dist/types/translateDoc/types.d.ts +2 -1
  146. package/dist/types/translateDoc/types.d.ts.map +1 -1
  147. package/dist/types/utils/checkAccess.d.ts +1 -1
  148. package/dist/types/utils/chunkInference.d.ts +1 -1
  149. package/dist/types/utils/getOutputFilePath.d.ts +1 -1
  150. package/dist/types/utils/setupAI.d.ts +1 -1
  151. package/package.json +23 -23
@@ -1 +1 @@
1
- {"version":3,"file":"cli.mjs","names":["pathDirname"],"sources":["../../src/cli.ts"],"sourcesContent":["import { dirname as pathDirname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport type { AIOptions as BaseAIOptions } from '@intlayer/api';\nimport type { DiffMode, ListGitFilesOptions } from '@intlayer/chokidar/cli';\nimport { setPrefix } from '@intlayer/config/logger';\nimport {\n type GetConfigurationOptions,\n getConfiguration,\n} from '@intlayer/config/node';\nimport type { CustomIntlayerConfig } from '@intlayer/types';\nimport { Command } from 'commander';\nimport { login } from './auth/login';\nimport { build } from './build';\nimport { runCI } from './ci';\nimport { getConfig } from './config';\nimport { startEditor } from './editor';\nimport { extract } from './extract';\nimport { type FillOptions, fill } from './fill/fill';\nimport { init } from './init';\nimport { initMCP } from './initMCP';\nimport { initSkills } from './initSkills';\nimport { listContentDeclaration } from './listContentDeclaration';\nimport { listProjectsCommand } from './listProjects';\nimport { liveSync } from './liveSync';\nimport { pull } from './pull';\nimport { push } from './push/push';\nimport { pushConfig } from './pushConfig';\nimport { reviewDoc } from './reviewDoc/reviewDoc';\nimport { searchDoc } from './searchDoc';\nimport { testMissingTranslations } from './test';\nimport { translateDoc } from './translateDoc/translateDoc';\nimport { getParentPackageJSON } from './utils/getParentPackageJSON';\nimport { watchContentDeclaration } from './watch';\n\n// Extended AI options to include customPrompt\ntype AIOptions = BaseAIOptions & {\n customPrompt?: string;\n};\n\nconst isESModule = typeof import.meta.url === 'string';\n\nexport const dirname = isESModule\n ? pathDirname(fileURLToPath(import.meta.url))\n : __dirname;\n\nconst packageJson = getParentPackageJSON(dirname);\n\nconst logOptions = [\n ['--verbose', 'Verbose (default to true using CLI)'],\n ['--prefix [prefix]', 'Prefix'],\n];\n\nconst configurationOptions = [\n ['--env-file [envFile]', 'Environment file'],\n ['-e, --env [env]', 'Environment'],\n ['--base-dir [baseDir]', 'Base directory'],\n ['--no-cache [noCache]', 'No cache'],\n ...logOptions,\n];\n\nconst aiOptions = [\n ['--provider [provider]', 'Provider'],\n ['--temperature [temperature]', 'Temperature'],\n ['--model [model]', 'Model'],\n ['--api-key [apiKey]', 'Provider API key'],\n ['--custom-prompt [prompt]', 'Custom prompt'],\n ['--application-context [applicationContext]', 'Application context'],\n ['--data-serialization [dataSerialization]', 'Data serialization'],\n];\n\nconst gitOptions = [\n ['--git-diff [gitDiff]', 'Git diff mode - Check git diff between two refs'],\n ['--git-diff-base [gitDiffBase]', 'Git diff base ref'],\n ['--git-diff-current [gitDiffCurrent]', 'Git diff current ref'],\n ['--uncommitted [uncommitted]', 'Uncommitted'],\n ['--unpushed [unpushed]', 'Unpushed'],\n ['--untracked [untracked]', 'Untracked'],\n];\n\nconst extractKeysFromOptions = (options: object, keys: string[]) =>\n keys.filter((key) => options[key as keyof typeof options] !== undefined);\n\n/**\n * Helper functions to apply common options to commands\n */\nconst applyOptions = (command: Command, options: string[][]) => {\n options.forEach(([flag, description]) => {\n command.option(flag, description);\n });\n return command;\n};\n\nconst removeUndefined = <T extends Record<string, any>>(obj: T): T =>\n Object.fromEntries(\n Object.entries(obj).filter(([_, value]) => value !== undefined)\n ) as T;\n\nconst applyConfigOptions = (command: Command) =>\n applyOptions(command, configurationOptions);\nconst applyAIOptions = (command: Command) => applyOptions(command, aiOptions);\nconst applyGitOptions = (command: Command) => applyOptions(command, gitOptions);\n\nconst extractAiOptions = (options: AIOptions): AIOptions | undefined => {\n const {\n apiKey,\n provider,\n model,\n temperature,\n applicationContext,\n customPrompt,\n dataSerialization,\n } = options;\n\n const configuration = getConfiguration();\n\n const { ai } = configuration;\n\n return removeUndefined({\n ...ai,\n apiKey: apiKey ?? configuration.ai?.apiKey,\n provider: provider ?? (configuration.ai?.provider as AIOptions['provider']),\n model: model ?? configuration.ai?.model,\n temperature: temperature ?? configuration.ai?.temperature,\n applicationContext:\n applicationContext ?? configuration.ai?.applicationContext,\n customPrompt: customPrompt ?? (configuration.ai as any)?.customPrompt,\n dataSerialization: dataSerialization ?? configuration.ai?.dataSerialization,\n });\n};\n\ntype GitOptions = {\n gitDiff?: boolean;\n gitDiffBase?: string;\n gitDiffCurrent?: string;\n uncommitted?: boolean;\n unpushed?: boolean;\n untracked?: boolean;\n};\n\nconst gitOptionKeys: (keyof GitOptions)[] = [\n 'gitDiff',\n 'gitDiffBase',\n 'gitDiffCurrent',\n 'uncommitted',\n 'unpushed',\n 'untracked',\n];\n\nconst extractGitOptions = (\n options: GitOptions\n): ListGitFilesOptions | undefined => {\n const filteredOptions = extractKeysFromOptions(options, gitOptionKeys);\n\n const isOptionEmpty = filteredOptions.length === 0;\n\n if (isOptionEmpty) return undefined;\n\n const {\n gitDiff,\n gitDiffBase,\n gitDiffCurrent,\n uncommitted,\n unpushed,\n untracked,\n } = options;\n\n const mode = [\n gitDiff && 'gitDiff',\n uncommitted && 'uncommitted',\n unpushed && 'unpushed',\n untracked && 'untracked',\n ].filter(Boolean) as DiffMode[];\n\n return removeUndefined({\n mode,\n baseRef: gitDiffBase,\n currentRef: gitDiffCurrent,\n absolute: true,\n });\n};\n\ntype LogOptions = {\n prefix?: string;\n verbose?: boolean;\n};\n\nexport type ConfigurationOptions = {\n baseDir?: string;\n env?: string;\n envFile?: string;\n noCache?: boolean;\n checkTypes?: boolean;\n} & LogOptions;\n\nconst configurationOptionKeys: (keyof ConfigurationOptions)[] = [\n 'baseDir',\n 'env',\n 'envFile',\n 'verbose',\n 'prefix',\n 'checkTypes',\n];\n\nconst extractConfigOptions = (\n options: ConfigurationOptions & { with?: string }\n): GetConfigurationOptions | undefined => {\n const filteredOptions = extractKeysFromOptions(\n options,\n configurationOptionKeys\n );\n\n const isOptionEmpty = filteredOptions.length === 0;\n\n if (isOptionEmpty) {\n return undefined;\n }\n\n const { baseDir, env, envFile, verbose, noCache, checkTypes } = options;\n\n const log = removeUndefined({\n mode:\n typeof verbose !== 'undefined'\n ? verbose\n ? 'verbose'\n : 'default'\n : undefined,\n });\n\n const build = removeUndefined({\n checkTypes,\n });\n\n const override: CustomIntlayerConfig = removeUndefined({\n log:\n Object.keys(log).length > 0\n ? (log as CustomIntlayerConfig['log'])\n : undefined,\n build: Object.keys(build).length > 0 ? build : undefined,\n });\n\n return removeUndefined({\n baseDir,\n env,\n envFile,\n override: Object.keys(override).length > 0 ? override : undefined,\n cache: typeof noCache !== 'undefined' ? !noCache : undefined,\n });\n};\n\n/**\n * Set the API for the CLI\n *\n * Example of commands:\n *\n * npm run intlayer build --watch\n * npm run intlayer push --dictionaries id1 id2 id3 --deleteLocaleDir\n */\nexport const setAPI = (): Command => {\n const isWithCommand = process.argv.includes('--with');\n\n if (!isWithCommand) {\n setPrefix('');\n }\n\n const program = new Command();\n\n program.version(packageJson.version!).description('Intlayer CLI');\n\n // Explicit version subcommand for convenience: `npx intlayer version`\n program\n .command('version')\n .description('Print the Intlayer CLI version')\n .action(() => {\n // Prefer the resolved package.json version; fallback to unknown\n // Keeping output minimal to align with common CLI behavior\n console.log(packageJson.version ?? 'unknown');\n });\n\n /**\n * AUTH\n */\n const loginCmd = program\n .command('login')\n .description('Login to Intlayer')\n .option('--cms-url [cmsUrl]', 'CMS URL');\n\n applyConfigOptions(loginCmd);\n\n loginCmd.action((options) => {\n const configOptions = extractConfigOptions(options) ?? {\n override: {\n log: {\n prefix: '',\n mode: 'verbose',\n },\n },\n };\n\n return login({\n cmsUrl: options.cmsUrl,\n configOptions,\n });\n });\n\n /**\n * INIT\n */\n const initCmd = program\n .command('init')\n .description('Initialize Intlayer in the project')\n .option('--project-root [projectRoot]', 'Project root directory')\n .action((options) => init(options.projectRoot));\n\n initCmd\n .command('skills')\n .description('Initialize Intlayer skills in the project')\n .option('--project-root [projectRoot]', 'Project root directory')\n .action((options) => initSkills(options.projectRoot));\n\n initCmd\n .command('mcp')\n .description('Initialize Intlayer MCP server in the project')\n .option('--project-root [projectRoot]', 'Project root directory')\n .action((options) => initMCP(options.projectRoot));\n\n /**\n * DICTIONARIES\n */\n\n const dictionariesProgram = program\n .command('dictionary')\n .alias('dictionaries')\n .alias('dic')\n .description('Dictionaries operations');\n\n // Dictionary build command\n const buildOptions = {\n description: 'Build the dictionaries',\n options: [\n ['-w, --watch', 'Watch for changes'],\n ['--skip-prepare', 'Skip the prepare step'],\n ['--with [with...]', 'Start command in parallel with the build'],\n ['--check-types', 'Check TypeScript type and log errors'],\n ],\n };\n\n // Add build command to dictionaries program\n const dictionariesBuildCmd = dictionariesProgram\n .command('build')\n .description(buildOptions.description);\n\n applyOptions(dictionariesBuildCmd, buildOptions.options);\n applyConfigOptions(dictionariesBuildCmd);\n dictionariesBuildCmd.action((options) => {\n build({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Add build command to root program as well\n const rootBuildCmd = program\n .command('build')\n .description(buildOptions.description);\n\n applyOptions(rootBuildCmd, buildOptions.options);\n applyConfigOptions(rootBuildCmd);\n rootBuildCmd.action((options) => {\n build({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n const watchOptions = {\n description: 'Watch the dictionaries changes',\n options: [['--with [with...]', 'Start command in parallel with the build']],\n };\n\n // Add build command to dictionaries program\n const dictionariesWatchCmd = dictionariesProgram\n .command('watch')\n .description(buildOptions.description);\n\n applyOptions(dictionariesWatchCmd, watchOptions.options);\n applyConfigOptions(dictionariesWatchCmd);\n dictionariesWatchCmd.action((options) => {\n watchContentDeclaration({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Add build command to root program as well\n const rootWatchCmd = program\n .command('watch')\n .description(buildOptions.description);\n\n applyOptions(rootWatchCmd, watchOptions.options);\n applyConfigOptions(rootWatchCmd);\n rootWatchCmd.action((options) => {\n watchContentDeclaration({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Dictionary pull command\n const pullOptions = {\n description: 'Pull dictionaries from the server',\n options: [\n ['-d, --dictionaries [ids...]', 'List of dictionary IDs to pull'],\n [\n '--dictionary [ids...]',\n 'List of dictionary IDs to pull (alias for --dictionaries)',\n ],\n ['--new-dictionaries-path [path]', 'Path to save the new dictionaries'],\n // Backward-compatibility for older tests/flags (camelCase)\n [\n '--newDictionariesPath [path]',\n '[alias] Path to save the new dictionaries',\n ],\n ],\n };\n\n // Add pull command to dictionaries program\n const dictionariesPullCmd = dictionariesProgram\n .command('pull')\n .description(pullOptions.description);\n\n applyOptions(dictionariesPullCmd, pullOptions.options);\n applyConfigOptions(dictionariesPullCmd);\n dictionariesPullCmd.action((options) => {\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries ?? []),\n ...(options.dictionary ?? []),\n ];\n\n pull({\n ...options,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n configOptions: {\n ...options.configOptions,\n baseDir: options.baseDir,\n },\n });\n });\n\n // Add pull command to root program as well\n const rootPullCmd = program\n .command('pull')\n .description(pullOptions.description);\n\n applyOptions(rootPullCmd, pullOptions.options);\n applyConfigOptions(rootPullCmd);\n rootPullCmd.action((options) => {\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries ?? []),\n ...(options.dictionary ?? []),\n ];\n\n pull({\n ...options,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Dictionary push command\n const pushOptions = {\n description:\n 'Push all dictionaries. Create or update the pushed dictionaries',\n options: [\n ['-d, --dictionaries [ids...]', 'List of dictionary IDs to push'],\n [\n '--dictionary [ids...]',\n 'List of dictionary IDs to push (alias for --dictionaries)',\n ],\n [\n '-r, --delete-locale-dictionary',\n 'Delete the local dictionaries after pushing',\n ],\n [\n '-k, --keep-locale-dictionary',\n 'Keep the local dictionaries after pushing',\n ],\n // Backward-compatibility for older tests/flags (camelCase)\n [\n '--deleteLocaleDictionary',\n '[alias] Delete the local dictionaries after pushing',\n ],\n [\n '--keepLocaleDictionary',\n '[alias] Keep the local dictionaries after pushing',\n ],\n [\n '--build [build]',\n 'Build the dictionaries before pushing to ensure the content is up to date. True will force the build, false will skip the build, undefined will allow using the cache of the build',\n ],\n ],\n };\n\n // Add push command to dictionaries program\n const dictionariesPushCmd = dictionariesProgram\n .command('push')\n .description(pushOptions.description);\n\n applyOptions(dictionariesPushCmd, pushOptions.options);\n applyConfigOptions(dictionariesPushCmd);\n applyGitOptions(dictionariesPushCmd);\n\n dictionariesPushCmd.action((options) => {\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries || []),\n ...(options.dictionary || []),\n ];\n\n return push({\n ...options,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n gitOptions: extractGitOptions(options),\n configOptions: extractConfigOptions(options),\n } as FillOptions);\n });\n\n // Add push command to root program as well\n const rootPushCmd = program\n .command('push')\n .description(pushOptions.description);\n\n applyOptions(rootPushCmd, pushOptions.options);\n applyConfigOptions(rootPushCmd);\n applyGitOptions(rootPushCmd);\n\n rootPushCmd.action((options) => {\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries || []),\n ...(options.dictionary || []),\n ];\n\n return push({\n ...options,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n gitOptions: extractGitOptions(options),\n configOptions: extractConfigOptions(options),\n } as FillOptions);\n });\n\n /**\n * CONFIGURATION\n */\n\n // Define the parent command\n const configurationProgram = program\n .command('configuration')\n .alias('config')\n .alias('conf')\n .description('Configuration operations');\n\n const configGetCmd = configurationProgram\n .command('get')\n .description('Get the configuration');\n\n applyConfigOptions(configGetCmd);\n configGetCmd.action((options) => {\n getConfig({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Define the `push config` subcommand and add it to the `push` command\n const configPushCmd = configurationProgram\n .command('push')\n .description('Push the configuration');\n\n applyConfigOptions(configPushCmd);\n configPushCmd.action((options) => {\n pushConfig({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n /**\n * PROJECTS\n */\n\n const projectsProgram = program\n .command('projects')\n .alias('project')\n .description('List Intlayer projects');\n\n const projectsListCmd = projectsProgram\n .command('list')\n .description('List all Intlayer projects in the directory')\n .option('--base-dir [baseDir]', 'Base directory to search from')\n .option(\n '--git-root',\n 'Search from the git root directory instead of the base directory'\n )\n .option('--json', 'Output the results as JSON');\n\n projectsListCmd.action((options) => {\n listProjectsCommand({\n baseDir: options.baseDir,\n gitRoot: options.gitRoot,\n json: options.json,\n });\n });\n\n // Add alias for projects list command at root level\n const rootProjectsListCmd = program\n .command('projects-list')\n .alias('pl')\n .description('List all Intlayer projects in the directory')\n .option('--base-dir [baseDir]', 'Base directory to search from')\n .option(\n '--git-root',\n 'Search from the git root directory instead of the base directory'\n )\n .option('--absolute', 'Output the results as absolute paths')\n .option('--json', 'Output the results as JSON');\n\n rootProjectsListCmd.action((options) => {\n listProjectsCommand({\n baseDir: options.baseDir,\n gitRoot: options.gitRoot,\n json: options.json,\n absolute: options.absolute,\n });\n });\n\n /**\n * CONTENT DECLARATION\n */\n\n const contentProgram = program\n .command('content')\n .description('Content declaration operations');\n\n contentProgram\n .command('list')\n .description('List the content declaration files')\n .option('--json', 'Output the results as JSON')\n .option('--absolute', 'Output the results as absolute paths')\n .action((options) => {\n listContentDeclaration({\n json: options.json,\n absolute: options.absolute,\n });\n });\n\n // Add alias for content list command\n program\n .command('list')\n .description('List the content declaration files')\n .option('--json', 'Output the results as JSON')\n .option('--absolute', 'Output the results as absolute paths')\n .action((options) => {\n listContentDeclaration({\n json: options.json,\n absolute: options.absolute,\n });\n });\n\n const testProgram = contentProgram\n .command('test')\n .description('Test if there are missing translations')\n .option(\n '--build [build]',\n 'Build the dictionaries before testing to ensure the content is up to date. True will force the build, false will skip the build, undefined will allow using the cache of the build'\n );\n\n applyConfigOptions(testProgram);\n testProgram.action((options) => {\n testMissingTranslations({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Add alias for content test command\n const rootTestCmd = program\n .command('test')\n .description('Test if there are missing translations')\n .option(\n '--build [build]',\n 'Build the dictionaries before testing to ensure the content is up to date. True will force the build, false will skip the build, undefined will allow using the cache of the build'\n );\n\n applyConfigOptions(rootTestCmd);\n rootTestCmd.action((options) => {\n testMissingTranslations({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n const fillProgram = program\n .command('fill')\n .description('Fill the dictionaries')\n .option('-f, --file [files...]', 'List of Dictionary files to fill')\n .option('--source-locale [sourceLocale]', 'Source locale to translate from')\n .option(\n '--output-locales [outputLocales...]',\n 'Target locales to translate to'\n )\n .option(\n '--mode [mode]',\n 'Fill mode: complete, review. Complete will fill all missing content, review will fill missing content and review existing keys',\n 'complete'\n )\n .option('-k, --keys [keys...]', 'Filter dictionaries based on keys')\n .option(\n '--key [keys...]',\n 'Filter dictionaries based on keys (alias for --keys)'\n )\n .option(\n '--excluded-keys [excludedKeys...]',\n 'Filter out dictionaries based on keys'\n )\n .option(\n '--excluded-key [excludedKeys...]',\n 'Filter out dictionaries based on keys (alias for --excluded-keys)'\n )\n .option(\n '--path-filter [pathFilters...]',\n 'Filter dictionaries based on glob pattern'\n )\n .option(\n '--build [build]',\n 'Build the dictionaries before filling to ensure the content is up to date. True will force the build, false will skip the build, undefined will allow using the cache of the build'\n )\n .option(\n '--skip-metadata',\n 'Skip filling missing metadata (description, title, tags) for dictionaries'\n );\n\n applyConfigOptions(fillProgram);\n applyAIOptions(fillProgram);\n applyGitOptions(fillProgram);\n\n fillProgram.action((options) => {\n // Merge key aliases\n const keys = [...(options.keys ?? []), ...(options.key ?? [])];\n const excludedKeys = [\n ...(options.excludedKeys ?? []),\n ...(options.excludedKey ?? []),\n ];\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries ?? []),\n ...(options.dictionary ?? []),\n ];\n\n return fill({\n ...options,\n keys: keys.length > 0 ? keys : undefined,\n excludedKeys: excludedKeys.length > 0 ? excludedKeys : undefined,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n aiOptions: extractAiOptions(options),\n gitOptions: extractGitOptions(options),\n configOptions: extractConfigOptions(options),\n } as FillOptions);\n });\n\n /**\n * DOCS\n */\n\n const docParams = [\n ['--doc-pattern [docPattern...]', 'Documentation pattern'],\n [\n '--excluded-glob-pattern [excludedGlobPattern...]',\n 'Excluded glob pattern',\n ],\n [\n '--nb-simultaneous-file-processed [nbSimultaneousFileProcessed]',\n 'Number of simultaneous file processed',\n ],\n ['--locales [locales...]', 'Locales'],\n ['--base-locale [baseLocale]', 'Base locale'],\n [\n '--custom-instructions [customInstructions]',\n 'Custom instructions added to the prompt. Usefull to apply specific rules regarding formatting, urls translation, etc.',\n ],\n [\n '--skip-if-modified-before [skipIfModifiedBefore]',\n 'Skip the file if it has been modified before the given time. Can be an absolute time as \"2025-12-05\" (string or Date) or a relative time in ms `1 * 60 * 60 * 1000` (1 hour). This option check update time of the file using the `fs.stat` method. So it could be impacted by Git or other tools that modify the file.',\n ],\n [\n '--skip-if-modified-after [skipIfModifiedAfter]',\n 'Skip the file if it has been modified within the given time. Can be an absolute time as \"2025-12-05\" (string or Date) or a relative time in ms `1 * 60 * 60 * 1000` (1 hour). This option check update time of the file using the `fs.stat` method. So it could be impacted by Git or other tools that modify the file.',\n ],\n ['--skip-if-exists', 'Skip the file if it already exists'],\n ];\n\n const docProgram = program\n .command('doc')\n .description('Documentation operations');\n\n const translateProgram = docProgram\n .command('translate')\n .description('Translate the documentation');\n\n applyConfigOptions(translateProgram);\n applyAIOptions(translateProgram);\n applyGitOptions(translateProgram);\n applyOptions(translateProgram, docParams);\n\n translateProgram.action((options) =>\n translateDoc({\n docPattern: options.docPattern,\n excludedGlobPattern: options.excludedGlobPattern,\n locales: options.locales,\n baseLocale: options.baseLocale,\n aiOptions: extractAiOptions(options),\n gitOptions: extractGitOptions(options),\n nbSimultaneousFileProcessed: options.nbSimultaneousFileProcessed,\n configOptions: extractConfigOptions(options),\n customInstructions: options.customInstructions,\n skipIfModifiedBefore: options.skipIfModifiedBefore,\n skipIfModifiedAfter: options.skipIfModifiedAfter,\n skipIfExists: options.skipIfExists,\n })\n );\n\n const reviewProgram = docProgram\n .command('review')\n .description('Review the documentation');\n\n applyConfigOptions(reviewProgram);\n applyAIOptions(reviewProgram);\n applyGitOptions(reviewProgram);\n applyOptions(reviewProgram, docParams);\n\n reviewProgram.action((options) =>\n reviewDoc({\n docPattern: options.docPattern,\n excludedGlobPattern: options.excludedGlobPattern,\n locales: options.locales,\n baseLocale: options.baseLocale,\n aiOptions: extractAiOptions(options),\n gitOptions: extractGitOptions(options),\n nbSimultaneousFileProcessed: options.nbSimultaneousFileProcessed,\n configOptions: extractConfigOptions(options),\n customInstructions: options.customInstructions,\n skipIfModifiedBefore: options.skipIfModifiedBefore,\n skipIfModifiedAfter: options.skipIfModifiedAfter,\n skipIfExists: options.skipIfExists,\n })\n );\n\n const searchProgram = docProgram\n .command('search')\n .description('Search the documentation')\n .argument('<query>', 'Search query')\n .option('--limit [limit]', 'Limit the number of results', '10');\n\n applyConfigOptions(searchProgram);\n\n searchProgram.action((query, options) =>\n searchDoc({\n query,\n limit: options.limit ? parseInt(options.limit, 10) : 10,\n configOptions: extractConfigOptions(options),\n })\n );\n\n /**\n * LIVE SYNC\n */\n\n const liveOptions = [\n ['--with [with...]', 'Start command in parallel with the live sync'],\n ];\n\n const liveCmd = program\n .command('live')\n .description(\n 'Live sync - Watch for changes made on the CMS and update the application content accordingly'\n );\n\n applyOptions(liveCmd, liveOptions);\n applyConfigOptions(liveCmd);\n\n liveCmd.action((options) => liveSync(options));\n\n /**\n * EDITOR\n */\n\n const editorProgram = program\n .command('editor')\n .description('Visual editor operations');\n\n const editorStartCmd = editorProgram\n .command('start')\n .description('Start the Intlayer visual editor');\n\n applyConfigOptions(editorStartCmd);\n\n editorStartCmd.action((options) => {\n startEditor({\n env: options.env,\n envFile: options.envFile,\n });\n });\n\n /**\n * EXTRACT\n */\n const extractProgram = program\n .command('extract')\n .alias('ext')\n .description(\n 'Extract strings from components to be placed in a .content file close to the component'\n );\n\n extractProgram\n .option('-f, --file [files...]', 'List of files to extract')\n .option(\n '-o, --output-content-declarations [outputContentDeclarations]',\n 'Path to output content declaration files'\n )\n .option('--code-only', 'Only extract the component code', false)\n .option('--declaration-only', 'Only generate content declaration', false)\n .action((options) => {\n extract({\n files: options.file,\n outputContentDeclarations: options.outputContentDeclarations,\n configOptions: extractConfigOptions(options),\n codeOnly: options.codeOnly,\n declarationOnly: options.declarationOnly,\n });\n });\n\n applyConfigOptions(extractProgram);\n\n program.parse(process.argv);\n\n /**\n * CI / AUTOMATION\n *\n * Used to iterate over all projects in a monorepo, and help to parse secrets\n */\n program\n .command('ci')\n .description(\n 'Run Intlayer commands with auto-injected credentials from INTLAYER_PROJECT_CREDENTIALS. Detects current project or iterates over all projects.'\n )\n .argument(\n '<command...>',\n 'The intlayer command to execute (e.g., \"fill\", \"push\")'\n )\n .allowUnknownOption() // Allows passing flags like --verbose to the subcommand\n .action((args) => {\n runCI(args);\n });\n\n return program;\n};\n"],"mappings":"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"}
1
+ {"version":3,"file":"cli.mjs","names":["pathDirname"],"sources":["../../src/cli.ts"],"sourcesContent":["import { dirname as pathDirname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport type { AIOptions as BaseAIOptions } from '@intlayer/api';\nimport type { DiffMode, ListGitFilesOptions } from '@intlayer/chokidar/cli';\nimport { setPrefix } from '@intlayer/config/logger';\nimport {\n type GetConfigurationOptions,\n getConfiguration,\n} from '@intlayer/config/node';\nimport type { CustomIntlayerConfig } from '@intlayer/types/config';\nimport { Command } from 'commander';\nimport { login } from './auth/login';\nimport { build } from './build';\nimport { runCI } from './ci';\nimport { getConfig } from './config';\nimport { startEditor } from './editor';\nimport { extract } from './extract';\nimport { type FillOptions, fill } from './fill/fill';\nimport { init } from './init';\nimport { initMCP } from './initMCP';\nimport { initSkills } from './initSkills';\nimport { listContentDeclaration } from './listContentDeclaration';\nimport { listProjectsCommand } from './listProjects';\nimport { liveSync } from './liveSync';\nimport { pull } from './pull';\nimport { push } from './push/push';\nimport { pushConfig } from './pushConfig';\nimport { reviewDoc } from './reviewDoc/reviewDoc';\nimport { searchDoc } from './searchDoc';\nimport { testMissingTranslations } from './test';\nimport { translateDoc } from './translateDoc/translateDoc';\nimport { getParentPackageJSON } from './utils/getParentPackageJSON';\nimport { watchContentDeclaration } from './watch';\n\n// Extended AI options to include customPrompt\ntype AIOptions = BaseAIOptions & {\n customPrompt?: string;\n};\n\nconst isESModule = typeof import.meta.url === 'string';\n\nexport const dirname = isESModule\n ? pathDirname(fileURLToPath(import.meta.url))\n : __dirname;\n\nconst packageJson = getParentPackageJSON(dirname);\n\nconst logOptions = [\n ['--verbose', 'Verbose (default to true using CLI)'],\n ['--prefix [prefix]', 'Prefix'],\n];\n\nconst configurationOptions = [\n ['--env-file [envFile]', 'Environment file'],\n ['-e, --env [env]', 'Environment'],\n ['--base-dir [baseDir]', 'Base directory'],\n ['--no-cache [noCache]', 'No cache'],\n ...logOptions,\n];\n\nconst aiOptions = [\n ['--provider [provider]', 'Provider'],\n ['--temperature [temperature]', 'Temperature'],\n ['--model [model]', 'Model'],\n ['--api-key [apiKey]', 'Provider API key'],\n ['--custom-prompt [prompt]', 'Custom prompt'],\n ['--application-context [applicationContext]', 'Application context'],\n ['--data-serialization [dataSerialization]', 'Data serialization'],\n];\n\nconst gitOptions = [\n ['--git-diff [gitDiff]', 'Git diff mode - Check git diff between two refs'],\n ['--git-diff-base [gitDiffBase]', 'Git diff base ref'],\n ['--git-diff-current [gitDiffCurrent]', 'Git diff current ref'],\n ['--uncommitted [uncommitted]', 'Uncommitted'],\n ['--unpushed [unpushed]', 'Unpushed'],\n ['--untracked [untracked]', 'Untracked'],\n];\n\nconst extractKeysFromOptions = (options: object, keys: string[]) =>\n keys.filter((key) => options[key as keyof typeof options] !== undefined);\n\n/**\n * Helper functions to apply common options to commands\n */\nconst applyOptions = (command: Command, options: string[][]) => {\n options.forEach(([flag, description]) => {\n command.option(flag, description);\n });\n return command;\n};\n\nconst removeUndefined = <T extends Record<string, any>>(obj: T): T =>\n Object.fromEntries(\n Object.entries(obj).filter(([_, value]) => value !== undefined)\n ) as T;\n\nconst applyConfigOptions = (command: Command) =>\n applyOptions(command, configurationOptions);\nconst applyAIOptions = (command: Command) => applyOptions(command, aiOptions);\nconst applyGitOptions = (command: Command) => applyOptions(command, gitOptions);\n\nconst extractAiOptions = (options: AIOptions): AIOptions | undefined => {\n const {\n apiKey,\n provider,\n model,\n temperature,\n applicationContext,\n customPrompt,\n dataSerialization,\n } = options;\n\n const configuration = getConfiguration();\n\n const { ai } = configuration;\n\n return removeUndefined({\n ...ai,\n apiKey: apiKey ?? configuration.ai?.apiKey,\n provider: provider ?? (configuration.ai?.provider as AIOptions['provider']),\n model: model ?? configuration.ai?.model,\n temperature: temperature ?? configuration.ai?.temperature,\n applicationContext:\n applicationContext ?? configuration.ai?.applicationContext,\n customPrompt: customPrompt ?? (configuration.ai as any)?.customPrompt,\n dataSerialization: dataSerialization ?? configuration.ai?.dataSerialization,\n });\n};\n\ntype GitOptions = {\n gitDiff?: boolean;\n gitDiffBase?: string;\n gitDiffCurrent?: string;\n uncommitted?: boolean;\n unpushed?: boolean;\n untracked?: boolean;\n};\n\nconst gitOptionKeys: (keyof GitOptions)[] = [\n 'gitDiff',\n 'gitDiffBase',\n 'gitDiffCurrent',\n 'uncommitted',\n 'unpushed',\n 'untracked',\n];\n\nconst extractGitOptions = (\n options: GitOptions\n): ListGitFilesOptions | undefined => {\n const filteredOptions = extractKeysFromOptions(options, gitOptionKeys);\n\n const isOptionEmpty = filteredOptions.length === 0;\n\n if (isOptionEmpty) return undefined;\n\n const {\n gitDiff,\n gitDiffBase,\n gitDiffCurrent,\n uncommitted,\n unpushed,\n untracked,\n } = options;\n\n const mode = [\n gitDiff && 'gitDiff',\n uncommitted && 'uncommitted',\n unpushed && 'unpushed',\n untracked && 'untracked',\n ].filter(Boolean) as DiffMode[];\n\n return removeUndefined({\n mode,\n baseRef: gitDiffBase,\n currentRef: gitDiffCurrent,\n absolute: true,\n });\n};\n\ntype LogOptions = {\n prefix?: string;\n verbose?: boolean;\n};\n\nexport type ConfigurationOptions = {\n baseDir?: string;\n env?: string;\n envFile?: string;\n noCache?: boolean;\n checkTypes?: boolean;\n} & LogOptions;\n\nconst configurationOptionKeys: (keyof ConfigurationOptions)[] = [\n 'baseDir',\n 'env',\n 'envFile',\n 'verbose',\n 'prefix',\n 'checkTypes',\n];\n\nconst extractConfigOptions = (\n options: ConfigurationOptions & { with?: string }\n): GetConfigurationOptions | undefined => {\n const filteredOptions = extractKeysFromOptions(\n options,\n configurationOptionKeys\n );\n\n const isOptionEmpty = filteredOptions.length === 0;\n\n if (isOptionEmpty) {\n return undefined;\n }\n\n const { baseDir, env, envFile, verbose, noCache, checkTypes } = options;\n\n const log = removeUndefined({\n mode:\n typeof verbose !== 'undefined'\n ? verbose\n ? 'verbose'\n : 'default'\n : undefined,\n });\n\n const build = removeUndefined({\n checkTypes,\n });\n\n const override: CustomIntlayerConfig = removeUndefined({\n log:\n Object.keys(log).length > 0\n ? (log as CustomIntlayerConfig['log'])\n : undefined,\n build: Object.keys(build).length > 0 ? build : undefined,\n });\n\n return removeUndefined({\n baseDir,\n env,\n envFile,\n override: Object.keys(override).length > 0 ? override : undefined,\n cache: typeof noCache !== 'undefined' ? !noCache : undefined,\n });\n};\n\n/**\n * Set the API for the CLI\n *\n * Example of commands:\n *\n * npm run intlayer build --watch\n * npm run intlayer push --dictionaries id1 id2 id3 --deleteLocaleDir\n */\nexport const setAPI = (): Command => {\n const isWithCommand = process.argv.includes('--with');\n\n if (!isWithCommand) {\n setPrefix('');\n }\n\n const program = new Command();\n\n program.version(packageJson.version!).description('Intlayer CLI');\n\n // Explicit version subcommand for convenience: `npx intlayer version`\n program\n .command('version')\n .description('Print the Intlayer CLI version')\n .action(() => {\n // Prefer the resolved package.json version; fallback to unknown\n // Keeping output minimal to align with common CLI behavior\n console.log(packageJson.version ?? 'unknown');\n });\n\n /**\n * AUTH\n */\n const loginCmd = program\n .command('login')\n .description('Login to Intlayer')\n .option('--cms-url [cmsUrl]', 'CMS URL');\n\n applyConfigOptions(loginCmd);\n\n loginCmd.action((options) => {\n const configOptions = extractConfigOptions(options) ?? {\n override: {\n log: {\n prefix: '',\n mode: 'verbose',\n },\n },\n };\n\n return login({\n cmsUrl: options.cmsUrl,\n configOptions,\n });\n });\n\n /**\n * INIT\n */\n const initCmd = program\n .command('init')\n .description('Initialize Intlayer in the project')\n .option('--project-root [projectRoot]', 'Project root directory')\n .action((options) => init(options.projectRoot));\n\n initCmd\n .command('skills')\n .description('Initialize Intlayer skills in the project')\n .option('--project-root [projectRoot]', 'Project root directory')\n .action((options) => initSkills(options.projectRoot));\n\n initCmd\n .command('mcp')\n .description('Initialize Intlayer MCP server in the project')\n .option('--project-root [projectRoot]', 'Project root directory')\n .action((options) => initMCP(options.projectRoot));\n\n /**\n * DICTIONARIES\n */\n\n const dictionariesProgram = program\n .command('dictionary')\n .alias('dictionaries')\n .alias('dic')\n .description('Dictionaries operations');\n\n // Dictionary build command\n const buildOptions = {\n description: 'Build the dictionaries',\n options: [\n ['-w, --watch', 'Watch for changes'],\n ['--skip-prepare', 'Skip the prepare step'],\n ['--with [with...]', 'Start command in parallel with the build'],\n ['--check-types', 'Check TypeScript type and log errors'],\n ],\n };\n\n // Add build command to dictionaries program\n const dictionariesBuildCmd = dictionariesProgram\n .command('build')\n .description(buildOptions.description);\n\n applyOptions(dictionariesBuildCmd, buildOptions.options);\n applyConfigOptions(dictionariesBuildCmd);\n dictionariesBuildCmd.action((options) => {\n build({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Add build command to root program as well\n const rootBuildCmd = program\n .command('build')\n .description(buildOptions.description);\n\n applyOptions(rootBuildCmd, buildOptions.options);\n applyConfigOptions(rootBuildCmd);\n rootBuildCmd.action((options) => {\n build({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n const watchOptions = {\n description: 'Watch the dictionaries changes',\n options: [['--with [with...]', 'Start command in parallel with the build']],\n };\n\n // Add build command to dictionaries program\n const dictionariesWatchCmd = dictionariesProgram\n .command('watch')\n .description(buildOptions.description);\n\n applyOptions(dictionariesWatchCmd, watchOptions.options);\n applyConfigOptions(dictionariesWatchCmd);\n dictionariesWatchCmd.action((options) => {\n watchContentDeclaration({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Add build command to root program as well\n const rootWatchCmd = program\n .command('watch')\n .description(buildOptions.description);\n\n applyOptions(rootWatchCmd, watchOptions.options);\n applyConfigOptions(rootWatchCmd);\n rootWatchCmd.action((options) => {\n watchContentDeclaration({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Dictionary pull command\n const pullOptions = {\n description: 'Pull dictionaries from the server',\n options: [\n ['-d, --dictionaries [ids...]', 'List of dictionary IDs to pull'],\n [\n '--dictionary [ids...]',\n 'List of dictionary IDs to pull (alias for --dictionaries)',\n ],\n ['--new-dictionaries-path [path]', 'Path to save the new dictionaries'],\n // Backward-compatibility for older tests/flags (camelCase)\n [\n '--newDictionariesPath [path]',\n '[alias] Path to save the new dictionaries',\n ],\n ],\n };\n\n // Add pull command to dictionaries program\n const dictionariesPullCmd = dictionariesProgram\n .command('pull')\n .description(pullOptions.description);\n\n applyOptions(dictionariesPullCmd, pullOptions.options);\n applyConfigOptions(dictionariesPullCmd);\n dictionariesPullCmd.action((options) => {\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries ?? []),\n ...(options.dictionary ?? []),\n ];\n\n pull({\n ...options,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n configOptions: {\n ...options.configOptions,\n baseDir: options.baseDir,\n },\n });\n });\n\n // Add pull command to root program as well\n const rootPullCmd = program\n .command('pull')\n .description(pullOptions.description);\n\n applyOptions(rootPullCmd, pullOptions.options);\n applyConfigOptions(rootPullCmd);\n rootPullCmd.action((options) => {\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries ?? []),\n ...(options.dictionary ?? []),\n ];\n\n pull({\n ...options,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Dictionary push command\n const pushOptions = {\n description:\n 'Push all dictionaries. Create or update the pushed dictionaries',\n options: [\n ['-d, --dictionaries [ids...]', 'List of dictionary IDs to push'],\n [\n '--dictionary [ids...]',\n 'List of dictionary IDs to push (alias for --dictionaries)',\n ],\n [\n '-r, --delete-locale-dictionary',\n 'Delete the local dictionaries after pushing',\n ],\n [\n '-k, --keep-locale-dictionary',\n 'Keep the local dictionaries after pushing',\n ],\n // Backward-compatibility for older tests/flags (camelCase)\n [\n '--deleteLocaleDictionary',\n '[alias] Delete the local dictionaries after pushing',\n ],\n [\n '--keepLocaleDictionary',\n '[alias] Keep the local dictionaries after pushing',\n ],\n [\n '--build [build]',\n 'Build the dictionaries before pushing to ensure the content is up to date. True will force the build, false will skip the build, undefined will allow using the cache of the build',\n ],\n ],\n };\n\n // Add push command to dictionaries program\n const dictionariesPushCmd = dictionariesProgram\n .command('push')\n .description(pushOptions.description);\n\n applyOptions(dictionariesPushCmd, pushOptions.options);\n applyConfigOptions(dictionariesPushCmd);\n applyGitOptions(dictionariesPushCmd);\n\n dictionariesPushCmd.action((options) => {\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries || []),\n ...(options.dictionary || []),\n ];\n\n return push({\n ...options,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n gitOptions: extractGitOptions(options),\n configOptions: extractConfigOptions(options),\n } as FillOptions);\n });\n\n // Add push command to root program as well\n const rootPushCmd = program\n .command('push')\n .description(pushOptions.description);\n\n applyOptions(rootPushCmd, pushOptions.options);\n applyConfigOptions(rootPushCmd);\n applyGitOptions(rootPushCmd);\n\n rootPushCmd.action((options) => {\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries || []),\n ...(options.dictionary || []),\n ];\n\n return push({\n ...options,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n gitOptions: extractGitOptions(options),\n configOptions: extractConfigOptions(options),\n } as FillOptions);\n });\n\n /**\n * CONFIGURATION\n */\n\n // Define the parent command\n const configurationProgram = program\n .command('configuration')\n .alias('config')\n .alias('conf')\n .description('Configuration operations');\n\n const configGetCmd = configurationProgram\n .command('get')\n .description('Get the configuration');\n\n applyConfigOptions(configGetCmd);\n configGetCmd.action((options) => {\n getConfig({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Define the `push config` subcommand and add it to the `push` command\n const configPushCmd = configurationProgram\n .command('push')\n .description('Push the configuration');\n\n applyConfigOptions(configPushCmd);\n configPushCmd.action((options) => {\n pushConfig({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n /**\n * PROJECTS\n */\n\n const projectsProgram = program\n .command('projects')\n .alias('project')\n .description('List Intlayer projects');\n\n const projectsListCmd = projectsProgram\n .command('list')\n .description('List all Intlayer projects in the directory')\n .option('--base-dir [baseDir]', 'Base directory to search from')\n .option(\n '--git-root',\n 'Search from the git root directory instead of the base directory'\n )\n .option('--json', 'Output the results as JSON');\n\n projectsListCmd.action((options) => {\n listProjectsCommand({\n baseDir: options.baseDir,\n gitRoot: options.gitRoot,\n json: options.json,\n });\n });\n\n // Add alias for projects list command at root level\n const rootProjectsListCmd = program\n .command('projects-list')\n .alias('pl')\n .description('List all Intlayer projects in the directory')\n .option('--base-dir [baseDir]', 'Base directory to search from')\n .option(\n '--git-root',\n 'Search from the git root directory instead of the base directory'\n )\n .option('--absolute', 'Output the results as absolute paths')\n .option('--json', 'Output the results as JSON');\n\n rootProjectsListCmd.action((options) => {\n listProjectsCommand({\n baseDir: options.baseDir,\n gitRoot: options.gitRoot,\n json: options.json,\n absolute: options.absolute,\n });\n });\n\n /**\n * CONTENT DECLARATION\n */\n\n const contentProgram = program\n .command('content')\n .description('Content declaration operations');\n\n contentProgram\n .command('list')\n .description('List the content declaration files')\n .option('--json', 'Output the results as JSON')\n .option('--absolute', 'Output the results as absolute paths')\n .action((options) => {\n listContentDeclaration({\n json: options.json,\n absolute: options.absolute,\n });\n });\n\n // Add alias for content list command\n program\n .command('list')\n .description('List the content declaration files')\n .option('--json', 'Output the results as JSON')\n .option('--absolute', 'Output the results as absolute paths')\n .action((options) => {\n listContentDeclaration({\n json: options.json,\n absolute: options.absolute,\n });\n });\n\n const testProgram = contentProgram\n .command('test')\n .description('Test if there are missing translations')\n .option(\n '--build [build]',\n 'Build the dictionaries before testing to ensure the content is up to date. True will force the build, false will skip the build, undefined will allow using the cache of the build'\n );\n\n applyConfigOptions(testProgram);\n testProgram.action((options) => {\n testMissingTranslations({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n // Add alias for content test command\n const rootTestCmd = program\n .command('test')\n .description('Test if there are missing translations')\n .option(\n '--build [build]',\n 'Build the dictionaries before testing to ensure the content is up to date. True will force the build, false will skip the build, undefined will allow using the cache of the build'\n );\n\n applyConfigOptions(rootTestCmd);\n rootTestCmd.action((options) => {\n testMissingTranslations({\n ...options,\n configOptions: extractConfigOptions(options),\n });\n });\n\n const fillProgram = program\n .command('fill')\n .description('Fill the dictionaries')\n .option('-f, --file [files...]', 'List of Dictionary files to fill')\n .option('--source-locale [sourceLocale]', 'Source locale to translate from')\n .option(\n '--output-locales [outputLocales...]',\n 'Target locales to translate to'\n )\n .option(\n '--mode [mode]',\n 'Fill mode: complete, review. Complete will fill all missing content, review will fill missing content and review existing keys',\n 'complete'\n )\n .option('-k, --keys [keys...]', 'Filter dictionaries based on keys')\n .option(\n '--key [keys...]',\n 'Filter dictionaries based on keys (alias for --keys)'\n )\n .option(\n '--excluded-keys [excludedKeys...]',\n 'Filter out dictionaries based on keys'\n )\n .option(\n '--excluded-key [excludedKeys...]',\n 'Filter out dictionaries based on keys (alias for --excluded-keys)'\n )\n .option(\n '--path-filter [pathFilters...]',\n 'Filter dictionaries based on glob pattern'\n )\n .option(\n '--build [build]',\n 'Build the dictionaries before filling to ensure the content is up to date. True will force the build, false will skip the build, undefined will allow using the cache of the build'\n )\n .option(\n '--skip-metadata',\n 'Skip filling missing metadata (description, title, tags) for dictionaries'\n );\n\n applyConfigOptions(fillProgram);\n applyAIOptions(fillProgram);\n applyGitOptions(fillProgram);\n\n fillProgram.action((options) => {\n // Merge key aliases\n const keys = [...(options.keys ?? []), ...(options.key ?? [])];\n const excludedKeys = [\n ...(options.excludedKeys ?? []),\n ...(options.excludedKey ?? []),\n ];\n // Merge dictionary aliases\n const dictionaries = [\n ...(options.dictionaries ?? []),\n ...(options.dictionary ?? []),\n ];\n\n return fill({\n ...options,\n keys: keys.length > 0 ? keys : undefined,\n excludedKeys: excludedKeys.length > 0 ? excludedKeys : undefined,\n dictionaries: dictionaries.length > 0 ? dictionaries : undefined,\n aiOptions: extractAiOptions(options),\n gitOptions: extractGitOptions(options),\n configOptions: extractConfigOptions(options),\n } as FillOptions);\n });\n\n /**\n * DOCS\n */\n\n const docParams = [\n ['--doc-pattern [docPattern...]', 'Documentation pattern'],\n [\n '--excluded-glob-pattern [excludedGlobPattern...]',\n 'Excluded glob pattern',\n ],\n [\n '--nb-simultaneous-file-processed [nbSimultaneousFileProcessed]',\n 'Number of simultaneous file processed',\n ],\n ['--locales [locales...]', 'Locales'],\n ['--base-locale [baseLocale]', 'Base locale'],\n [\n '--custom-instructions [customInstructions]',\n 'Custom instructions added to the prompt. Usefull to apply specific rules regarding formatting, urls translation, etc.',\n ],\n [\n '--skip-if-modified-before [skipIfModifiedBefore]',\n 'Skip the file if it has been modified before the given time. Can be an absolute time as \"2025-12-05\" (string or Date) or a relative time in ms `1 * 60 * 60 * 1000` (1 hour). This option check update time of the file using the `fs.stat` method. So it could be impacted by Git or other tools that modify the file.',\n ],\n [\n '--skip-if-modified-after [skipIfModifiedAfter]',\n 'Skip the file if it has been modified within the given time. Can be an absolute time as \"2025-12-05\" (string or Date) or a relative time in ms `1 * 60 * 60 * 1000` (1 hour). This option check update time of the file using the `fs.stat` method. So it could be impacted by Git or other tools that modify the file.',\n ],\n ['--skip-if-exists', 'Skip the file if it already exists'],\n ];\n\n const docProgram = program\n .command('doc')\n .description('Documentation operations');\n\n const translateProgram = docProgram\n .command('translate')\n .description('Translate the documentation');\n\n applyConfigOptions(translateProgram);\n applyAIOptions(translateProgram);\n applyGitOptions(translateProgram);\n applyOptions(translateProgram, docParams);\n\n translateProgram.action((options) =>\n translateDoc({\n docPattern: options.docPattern,\n excludedGlobPattern: options.excludedGlobPattern,\n locales: options.locales,\n baseLocale: options.baseLocale,\n aiOptions: extractAiOptions(options),\n gitOptions: extractGitOptions(options),\n nbSimultaneousFileProcessed: options.nbSimultaneousFileProcessed,\n configOptions: extractConfigOptions(options),\n customInstructions: options.customInstructions,\n skipIfModifiedBefore: options.skipIfModifiedBefore,\n skipIfModifiedAfter: options.skipIfModifiedAfter,\n skipIfExists: options.skipIfExists,\n })\n );\n\n const reviewProgram = docProgram\n .command('review')\n .description('Review the documentation');\n\n applyConfigOptions(reviewProgram);\n applyAIOptions(reviewProgram);\n applyGitOptions(reviewProgram);\n applyOptions(reviewProgram, docParams);\n\n reviewProgram.action((options) =>\n reviewDoc({\n docPattern: options.docPattern,\n excludedGlobPattern: options.excludedGlobPattern,\n locales: options.locales,\n baseLocale: options.baseLocale,\n aiOptions: extractAiOptions(options),\n gitOptions: extractGitOptions(options),\n nbSimultaneousFileProcessed: options.nbSimultaneousFileProcessed,\n configOptions: extractConfigOptions(options),\n customInstructions: options.customInstructions,\n skipIfModifiedBefore: options.skipIfModifiedBefore,\n skipIfModifiedAfter: options.skipIfModifiedAfter,\n skipIfExists: options.skipIfExists,\n })\n );\n\n const searchProgram = docProgram\n .command('search')\n .description('Search the documentation')\n .argument('<query>', 'Search query')\n .option('--limit [limit]', 'Limit the number of results', '10');\n\n applyConfigOptions(searchProgram);\n\n searchProgram.action((query, options) =>\n searchDoc({\n query,\n limit: options.limit ? parseInt(options.limit, 10) : 10,\n configOptions: extractConfigOptions(options),\n })\n );\n\n /**\n * LIVE SYNC\n */\n\n const liveOptions = [\n ['--with [with...]', 'Start command in parallel with the live sync'],\n ];\n\n const liveCmd = program\n .command('live')\n .description(\n 'Live sync - Watch for changes made on the CMS and update the application content accordingly'\n );\n\n applyOptions(liveCmd, liveOptions);\n applyConfigOptions(liveCmd);\n\n liveCmd.action((options) => liveSync(options));\n\n /**\n * EDITOR\n */\n\n const editorProgram = program\n .command('editor')\n .description('Visual editor operations');\n\n const editorStartCmd = editorProgram\n .command('start')\n .description('Start the Intlayer visual editor');\n\n applyConfigOptions(editorStartCmd);\n\n editorStartCmd.action((options) => {\n startEditor({\n env: options.env,\n envFile: options.envFile,\n });\n });\n\n /**\n * EXTRACT\n */\n const extractProgram = program\n .command('extract')\n .alias('ext')\n .description(\n 'Extract strings from components to be placed in a .content file close to the component'\n );\n\n extractProgram\n .option('-f, --file [files...]', 'List of files to extract')\n .option(\n '-o, --output-content-declarations [outputContentDeclarations]',\n 'Path to output content declaration files'\n )\n .option('--code-only', 'Only extract the component code', false)\n .option('--declaration-only', 'Only generate content declaration', false)\n .action((options) => {\n extract({\n files: options.file,\n outputContentDeclarations: options.outputContentDeclarations,\n configOptions: extractConfigOptions(options),\n codeOnly: options.codeOnly,\n declarationOnly: options.declarationOnly,\n });\n });\n\n applyConfigOptions(extractProgram);\n\n program.parse(process.argv);\n\n /**\n * CI / AUTOMATION\n *\n * Used to iterate over all projects in a monorepo, and help to parse secrets\n */\n program\n .command('ci')\n .description(\n 'Run Intlayer commands with auto-injected credentials from INTLAYER_PROJECT_CREDENTIALS. Detects current project or iterates over all projects.'\n )\n .argument(\n '<command...>',\n 'The intlayer command to execute (e.g., \"fill\", \"push\")'\n )\n .allowUnknownOption() // Allows passing flags like --verbose to the subcommand\n .action((args) => {\n runCI(args);\n });\n\n return program;\n};\n"],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- import{getAppLogger as e}from"@intlayer/config/logger";import{logConfigDetails as t}from"@intlayer/chokidar/cli";import{getConfiguration as n}from"@intlayer/config/node";const r=r=>{let i=n(r?.configOptions);t(r?.configOptions),e(i)(JSON.stringify(i,null,2))};export{r as getConfig};
1
+ import{logConfigDetails as e}from"@intlayer/chokidar/cli";import{getAppLogger as t}from"@intlayer/config/logger";import{getConfiguration as n}from"@intlayer/config/node";const r=r=>{let i=n(r?.configOptions);e(r?.configOptions),t(i)(JSON.stringify(i,null,2))};export{r as getConfig};
2
2
  //# sourceMappingURL=config.mjs.map
@@ -1,2 +1,2 @@
1
- import{__require as e}from"./_virtual/_rolldown/runtime.mjs";import{spawn as t}from"node:child_process";import{dirname as n,resolve as r}from"node:path";import{createRequire as i}from"node:module";import{isESModule as a}from"@intlayer/config/utils";const o=o=>{let s=[`start`];o.env&&s.push(`--env`,o.env),o.envFile&&s.push(`--env-file`,o.envFile);let c=e=>t(process.execPath,[e,...s],{stdio:`inherit`,env:{...process.env}}),l=(e,n)=>t(e,n,{stdio:`inherit`,env:{...process.env}});try{let t=c(r(n((a?i(import.meta.url):e).resolve(`intlayer-editor/package.json`)),`bin`,`intlayer-editor.mjs`));t.on(`error`,()=>u()),t.on(`exit`,e=>{e===255&&process.exit(e??0)})}catch{u()}function u(){l(`bun`,[`dlx`,`intlayer-editor`,...s]).on(`error`,()=>{l(`npx`,[`-y`,`intlayer-editor`,...s]).on(`error`,e=>{console.error(`Unable to execute intlayer-editor via bun or npx.`),console.error(String(e?.message??e)),process.exit(1)})})}};export{o as startEditor};
1
+ import{__require as e}from"./_virtual/_rolldown/runtime.mjs";import{dirname as t,resolve as n}from"node:path";import{spawn as r}from"node:child_process";import{createRequire as i}from"node:module";import{isESModule as a}from"@intlayer/config/utils";const o=o=>{let s=[`start`];o.env&&s.push(`--env`,o.env),o.envFile&&s.push(`--env-file`,o.envFile);let c=e=>r(process.execPath,[e,...s],{stdio:`inherit`,env:{...process.env}}),l=(e,t)=>r(e,t,{stdio:`inherit`,env:{...process.env}});try{let r=c(n(t((a?i(import.meta.url):e).resolve(`intlayer-editor/package.json`)),`bin`,`intlayer-editor.mjs`));r.on(`error`,()=>u()),r.on(`exit`,e=>{e===255&&process.exit(e??0)})}catch{u()}function u(){l(`bun`,[`dlx`,`intlayer-editor`,...s]).on(`error`,()=>{l(`npx`,[`-y`,`intlayer-editor`,...s]).on(`error`,e=>{console.error(`Unable to execute intlayer-editor via bun or npx.`),console.error(String(e?.message??e)),process.exit(1)})})}};export{o as startEditor};
2
2
  //# sourceMappingURL=editor.mjs.map
@@ -1,2 +1,2 @@
1
- import{colorizePath as e,getAppLogger as t}from"@intlayer/config/logger";import{logConfigDetails as n}from"@intlayer/chokidar/cli";import{getConfiguration as r}from"@intlayer/config/node";import{relative as i,resolve as a}from"node:path";import{existsSync as o}from"node:fs";import{readFile as s}from"node:fs/promises";import{extractContent as c}from"@intlayer/babel";import{getUnmergedDictionaries as l}from"@intlayer/unmerged-dictionaries-entry";import u from"enquirer";import d from"fast-glob";const f=async e=>{try{let t=a(e,`package.json`);if(!o(t))return{};let n=await s(t,`utf8`);return JSON.parse(n).dependencies}catch{return{}}},p=async s=>{let p=r(s.configOptions);n(s?.configOptions);let m=t(p),{baseDir:h,codeDir:g,excludedPath:_}=p.content,{traversePattern:v}=p.build,y=t=>e(i(h,t)),b=await f(h),x=`react-intlayer`;b[`next-intlayer`]?x=`next-intlayer`:b[`vue-intlayer`]?x=`vue-intlayer`:b[`svelte-intlayer`]?x=`svelte-intlayer`:b[`react-intlayer`]?x=`react-intlayer`:b[`preact-intlayer`]?x=`preact-intlayer`:b[`solid-intlayer`]?x=`solid-intlayer`:b[`angular-intlayer`]?x=`angular-intlayer`:b[`express-intlayer`]&&(x=`express-intlayer`);let S=s.files??[];if(S.length===0){let e=e=>(_??[]).some(t=>{let n=t.split(`/`).filter(e=>!e.includes(`*`)&&e.length>0),r=e.split(`/`);return n.some(e=>r.includes(e))}),t=(await Promise.all(g.filter(t=>!e(t)).map(e=>d(v,{cwd:e,ignore:_,absolute:!0})))).flat(),n=[...new Set(t)].filter(e=>o(e)).map(e=>({value:e,label:i(h,e)}));if(n.length===0){m(`No extractable files found in the project.`);return}let r=`__select_all__`,a;try{let e=Math.max((process.stdout.columns||80)-15,20),t=t=>t.length>e?`...${t.slice(-(e-3))}`:t,{files:i}=await u.prompt({type:`autocomplete`,name:`files`,message:`Select files to extract (Type to search):`,multiple:!0,limit:40,choices:[{name:r,message:`────── Select all ──────`},...n.map(e=>({name:e.value,message:t(e.label)}))],async toggle(e,t){if(!(!e||e.disabled))return e.enabled=t??!e.enabled,e.name===r&&this.choices.filter(e=>e.name!==r).forEach(t=>{t.enabled=e.enabled}),this.render()},format(){return this.state?.submitted&&this.options?.multiple?`${this.selected.filter(e=>e.name!==r).length} file(s) selected`:this.input??``}});a=i.filter(e=>e!==r)}catch{a=Symbol(`cancel`)}typeof a==`symbol`&&process.exit(0),S=a}if(S.length===0){m(`No files selected for extraction.`);return}let C=S.map(e=>a(h,e)).filter(e=>o(e)?!0:(m(`File not found: ${y(e)}`),!1));if(C.length===0)return;let w=l(p);await Promise.all(C.map(async e=>{try{await c(e,x,{unmergedDictionaries:w,configuration:p,outputDir:s.outputContentDeclarations,codeOnly:s.codeOnly,declarationOnly:s.declarationOnly})}catch(t){m(`Failed to transform ${e}: ${t.message}`)}}))};export{p as extract};
1
+ import{existsSync as e}from"node:fs";import{readFile as t}from"node:fs/promises";import{relative as n,resolve as r}from"node:path";import{extractContent as i}from"@intlayer/babel";import{logConfigDetails as a}from"@intlayer/chokidar/cli";import{colorizePath as o,getAppLogger as s}from"@intlayer/config/logger";import{getConfiguration as c}from"@intlayer/config/node";import{getUnmergedDictionaries as l}from"@intlayer/unmerged-dictionaries-entry";import u from"enquirer";import d from"fast-glob";const f=async n=>{try{let i=r(n,`package.json`);if(!e(i))return{};let a=await t(i,`utf8`);return JSON.parse(a).dependencies}catch{return{}}},p=async t=>{let p=c(t.configOptions);a(t?.configOptions);let m=s(p),{baseDir:h,codeDir:g,excludedPath:_}=p.content,{traversePattern:v}=p.build,y=e=>o(n(h,e)),b=await f(h),x=`react-intlayer`;b[`next-intlayer`]?x=`next-intlayer`:b[`vue-intlayer`]?x=`vue-intlayer`:b[`svelte-intlayer`]?x=`svelte-intlayer`:b[`react-intlayer`]?x=`react-intlayer`:b[`preact-intlayer`]?x=`preact-intlayer`:b[`solid-intlayer`]?x=`solid-intlayer`:b[`angular-intlayer`]?x=`angular-intlayer`:b[`express-intlayer`]&&(x=`express-intlayer`);let S=t.files??[];if(S.length===0){let t=e=>(_??[]).some(t=>{let n=t.split(`/`).filter(e=>!e.includes(`*`)&&e.length>0),r=e.split(`/`);return n.some(e=>r.includes(e))}),r=(await Promise.all(g.filter(e=>!t(e)).map(e=>d(v,{cwd:e,ignore:_,absolute:!0})))).flat(),i=[...new Set(r)].filter(t=>e(t)).map(e=>({value:e,label:n(h,e)}));if(i.length===0){m(`No extractable files found in the project.`);return}let a=`__select_all__`,o;try{let e=Math.max((process.stdout.columns||80)-15,20),t=t=>t.length>e?`...${t.slice(-(e-3))}`:t,{files:n}=await u.prompt({type:`autocomplete`,name:`files`,message:`Select files to extract (Type to search):`,multiple:!0,limit:40,choices:[{name:a,message:`────── Select all ──────`},...i.map(e=>({name:e.value,message:t(e.label)}))],async toggle(e,t){if(!(!e||e.disabled))return e.enabled=t??!e.enabled,e.name===a&&this.choices.filter(e=>e.name!==a).forEach(t=>{t.enabled=e.enabled}),this.render()},format(){return this.state?.submitted&&this.options?.multiple?`${this.selected.filter(e=>e.name!==a).length} file(s) selected`:this.input??``}});o=n.filter(e=>e!==a)}catch{o=Symbol(`cancel`)}typeof o==`symbol`&&process.exit(0),S=o}if(S.length===0){m(`No files selected for extraction.`);return}let C=S.map(e=>r(h,e)).filter(t=>e(t)?!0:(m(`File not found: ${y(t)}`),!1));if(C.length===0)return;let w=l(p);await Promise.all(C.map(async e=>{try{await i(e,x,{unmergedDictionaries:w,configuration:p,output:t.output,codeOnly:t.codeOnly,declarationOnly:t.declarationOnly})}catch(t){m(`Failed to transform ${e}: ${t.message}`)}}))};export{p as extract};
2
2
  //# sourceMappingURL=extract.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"extract.mjs","names":[],"sources":["../../src/extract.ts"],"sourcesContent":["import { existsSync } from 'node:fs';\nimport { readFile } from 'node:fs/promises';\nimport { relative, resolve } from 'node:path';\nimport { extractContent, type PackageName } from '@intlayer/babel';\nimport { logConfigDetails } from '@intlayer/chokidar/cli';\nimport { colorizePath, getAppLogger } from '@intlayer/config/logger';\nimport {\n type GetConfigurationOptions,\n getConfiguration,\n} from '@intlayer/config/node';\nimport { getUnmergedDictionaries } from '@intlayer/unmerged-dictionaries-entry';\nimport enquirer from 'enquirer';\nimport fg from 'fast-glob';\n\ntype ExtractOptions = {\n files?: string[];\n outputContentDeclarations?: string;\n configOptions?: GetConfigurationOptions;\n codeOnly?: boolean;\n declarationOnly?: boolean;\n};\n\n// Helper to read package.json dependencies\nconst getDependencies = async (baseDir: string) => {\n try {\n const packageJsonPath = resolve(baseDir, 'package.json');\n if (!existsSync(packageJsonPath)) {\n // Try parent directory if not found in baseDir\n return {};\n }\n const file = await readFile(packageJsonPath, 'utf8');\n const packageJSON = JSON.parse(file);\n\n return packageJSON.dependencies;\n } catch {\n return {};\n }\n};\n\nexport const extract = async (options: ExtractOptions) => {\n const configuration = getConfiguration(options.configOptions);\n logConfigDetails(options?.configOptions);\n\n const appLogger = getAppLogger(configuration);\n const { baseDir, codeDir, excludedPath } = configuration.content;\n const { traversePattern } = configuration.build;\n\n const formatPath = (path: string) => {\n const relativePath = relative(baseDir, path);\n return colorizePath(relativePath);\n };\n\n // Detect package\n const dependencies = await getDependencies(baseDir);\n let packageName: PackageName = 'react-intlayer';\n\n if (dependencies['next-intlayer']) {\n packageName = 'next-intlayer';\n } else if (dependencies['vue-intlayer']) {\n packageName = 'vue-intlayer';\n } else if (dependencies['svelte-intlayer']) {\n packageName = 'svelte-intlayer';\n } else if (dependencies['react-intlayer']) {\n packageName = 'react-intlayer';\n } else if (dependencies['preact-intlayer']) {\n packageName = 'preact-intlayer';\n } else if (dependencies['solid-intlayer']) {\n packageName = 'solid-intlayer';\n } else if (dependencies['angular-intlayer']) {\n packageName = 'angular-intlayer';\n } else if (dependencies['express-intlayer']) {\n packageName = 'express-intlayer';\n }\n\n let filesToExtract = options.files ?? [];\n\n if (filesToExtract.length === 0) {\n // Filter out codeDirs that are themselves inside an excluded path (e.g. dist, node_modules)\n const isDirExcluded = (dirPath: string): boolean =>\n (excludedPath ?? []).some((pattern) => {\n const segments = pattern\n .split('/')\n .filter((s) => !s.includes('*') && s.length > 0);\n const parts = dirPath.split('/');\n return segments.some((seg) => parts.includes(seg));\n });\n\n // Await all promises simultaneously\n const resultsArray = await Promise.all(\n codeDir\n .filter((dir) => !isDirExcluded(dir))\n .map((dir) =>\n fg(traversePattern, {\n cwd: dir,\n ignore: excludedPath,\n absolute: true,\n })\n )\n );\n\n // Flatten the nested arrays and remove duplicates\n const allFiles = resultsArray.flat();\n\n const uniqueFiles = [...new Set(allFiles)].filter((file) =>\n existsSync(file)\n );\n\n // Relative paths for selection\n const choices = uniqueFiles.map((file) => {\n const relPath = relative(baseDir, file);\n return {\n value: file,\n label: relPath,\n };\n });\n\n if (choices.length === 0) {\n appLogger('No extractable files found in the project.');\n return;\n }\n\n const SELECT_ALL = '__select_all__';\n\n type PromptChoice = {\n name: string;\n enabled: boolean;\n disabled?: boolean | string;\n };\n\n type PromptContext = {\n choices: PromptChoice[];\n render(): void | Promise<void>;\n state: { submitted: boolean };\n selected: PromptChoice[];\n input: string;\n options: { multiple?: boolean };\n };\n\n let selectedFiles: string[] | symbol;\n try {\n const maxLen = Math.max((process.stdout.columns || 80) - 15, 20);\n const truncatePath = (path: string) =>\n path.length > maxLen ? `...${path.slice(-(maxLen - 3))}` : path;\n\n const { files: enquirerSelectedFiles } = await enquirer.prompt<{\n files: string[];\n }>({\n type: 'autocomplete',\n name: 'files',\n message: 'Select files to extract (Type to search):',\n multiple: true,\n // @ts-ignore limit exist but is not typed\n limit: 40,\n choices: [\n { name: SELECT_ALL, message: '────── Select all ──────' },\n ...choices.map((choice) => ({\n name: choice.value,\n message: truncatePath(choice.label),\n })),\n ],\n async toggle(\n this: PromptContext,\n choice: PromptChoice,\n enabled?: boolean\n ) {\n if (!choice || choice.disabled) return;\n choice.enabled = enabled == null ? !choice.enabled : enabled;\n\n if (choice.name === SELECT_ALL) {\n this.choices\n .filter((choiceEl) => choiceEl.name !== SELECT_ALL)\n .forEach((choiceEl) => {\n choiceEl.enabled = choice.enabled;\n });\n }\n\n return this.render();\n },\n format(this: PromptContext) {\n if (this.state?.submitted && this.options?.multiple) {\n return `${this.selected.filter((s) => s.name !== SELECT_ALL).length} file(s) selected`;\n }\n return this.input ?? '';\n },\n });\n\n selectedFiles = enquirerSelectedFiles.filter((f) => f !== SELECT_ALL);\n } catch {\n selectedFiles = Symbol('cancel');\n }\n\n if (typeof selectedFiles === 'symbol') {\n // User cancelled\n process.exit(0);\n }\n\n filesToExtract = selectedFiles as string[];\n }\n\n if (filesToExtract.length === 0) {\n appLogger('No files selected for extraction.');\n return;\n }\n\n const absoluteFiles = filesToExtract\n .map((file) => resolve(baseDir, file))\n .filter((file) => {\n if (!existsSync(file)) {\n appLogger(`File not found: ${formatPath(file)}`);\n return false;\n }\n return true;\n });\n\n if (absoluteFiles.length === 0) {\n return;\n }\n\n const unmergedDictionaries = getUnmergedDictionaries(configuration);\n\n await Promise.all(\n absoluteFiles.map(async (filePath) => {\n try {\n await extractContent(filePath, packageName, {\n unmergedDictionaries,\n configuration,\n outputDir: options.outputContentDeclarations,\n codeOnly: options.codeOnly,\n declarationOnly: options.declarationOnly,\n });\n } catch (error) {\n appLogger(\n `Failed to transform ${filePath}: ${(error as Error).message}`\n );\n }\n })\n );\n};\n"],"mappings":"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"}
1
+ {"version":3,"file":"extract.mjs","names":[],"sources":["../../src/extract.ts"],"sourcesContent":["import { existsSync } from 'node:fs';\nimport { readFile } from 'node:fs/promises';\nimport { relative, resolve } from 'node:path';\nimport { extractContent, type PackageName } from '@intlayer/babel';\nimport { logConfigDetails } from '@intlayer/chokidar/cli';\nimport { colorizePath, getAppLogger } from '@intlayer/config/logger';\nimport {\n type GetConfigurationOptions,\n getConfiguration,\n} from '@intlayer/config/node';\nimport type { FilePathPattern } from '@intlayer/types/filePathPattern';\nimport { getUnmergedDictionaries } from '@intlayer/unmerged-dictionaries-entry';\nimport enquirer from 'enquirer';\nimport fg from 'fast-glob';\n\ntype ExtractOptions = {\n files?: string[];\n output?: FilePathPattern;\n configOptions?: GetConfigurationOptions;\n codeOnly?: boolean;\n declarationOnly?: boolean;\n};\n\n// Helper to read package.json dependencies\nconst getDependencies = async (baseDir: string) => {\n try {\n const packageJsonPath = resolve(baseDir, 'package.json');\n if (!existsSync(packageJsonPath)) {\n // Try parent directory if not found in baseDir\n return {};\n }\n const file = await readFile(packageJsonPath, 'utf8');\n const packageJSON = JSON.parse(file);\n\n return packageJSON.dependencies;\n } catch {\n return {};\n }\n};\n\nexport const extract = async (options: ExtractOptions) => {\n const configuration = getConfiguration(options.configOptions);\n logConfigDetails(options?.configOptions);\n\n const appLogger = getAppLogger(configuration);\n const { baseDir, codeDir, excludedPath } = configuration.content;\n const { traversePattern } = configuration.build;\n\n const formatPath = (path: string) => {\n const relativePath = relative(baseDir, path);\n return colorizePath(relativePath);\n };\n\n // Detect package\n const dependencies = await getDependencies(baseDir);\n let packageName: PackageName = 'react-intlayer';\n\n if (dependencies['next-intlayer']) {\n packageName = 'next-intlayer';\n } else if (dependencies['vue-intlayer']) {\n packageName = 'vue-intlayer';\n } else if (dependencies['svelte-intlayer']) {\n packageName = 'svelte-intlayer';\n } else if (dependencies['react-intlayer']) {\n packageName = 'react-intlayer';\n } else if (dependencies['preact-intlayer']) {\n packageName = 'preact-intlayer';\n } else if (dependencies['solid-intlayer']) {\n packageName = 'solid-intlayer';\n } else if (dependencies['angular-intlayer']) {\n packageName = 'angular-intlayer';\n } else if (dependencies['express-intlayer']) {\n packageName = 'express-intlayer';\n }\n\n let filesToExtract = options.files ?? [];\n\n if (filesToExtract.length === 0) {\n // Filter out codeDirs that are themselves inside an excluded path (e.g. dist, node_modules)\n const isDirExcluded = (dirPath: string): boolean =>\n (excludedPath ?? []).some((pattern) => {\n const segments = pattern\n .split('/')\n .filter((s) => !s.includes('*') && s.length > 0);\n const parts = dirPath.split('/');\n return segments.some((seg) => parts.includes(seg));\n });\n\n // Await all promises simultaneously\n const resultsArray = await Promise.all(\n codeDir\n .filter((dir) => !isDirExcluded(dir))\n .map((dir) =>\n fg(traversePattern, {\n cwd: dir,\n ignore: excludedPath,\n absolute: true,\n })\n )\n );\n\n // Flatten the nested arrays and remove duplicates\n const allFiles = resultsArray.flat();\n\n const uniqueFiles = [...new Set(allFiles)].filter((file) =>\n existsSync(file)\n );\n\n // Relative paths for selection\n const choices = uniqueFiles.map((file) => {\n const relPath = relative(baseDir, file);\n return {\n value: file,\n label: relPath,\n };\n });\n\n if (choices.length === 0) {\n appLogger('No extractable files found in the project.');\n return;\n }\n\n const SELECT_ALL = '__select_all__';\n\n type PromptChoice = {\n name: string;\n enabled: boolean;\n disabled?: boolean | string;\n };\n\n type PromptContext = {\n choices: PromptChoice[];\n render(): void | Promise<void>;\n state: { submitted: boolean };\n selected: PromptChoice[];\n input: string;\n options: { multiple?: boolean };\n };\n\n let selectedFiles: string[] | symbol;\n try {\n const maxLen = Math.max((process.stdout.columns || 80) - 15, 20);\n const truncatePath = (path: string) =>\n path.length > maxLen ? `...${path.slice(-(maxLen - 3))}` : path;\n\n const { files: enquirerSelectedFiles } = await enquirer.prompt<{\n files: string[];\n }>({\n type: 'autocomplete',\n name: 'files',\n message: 'Select files to extract (Type to search):',\n multiple: true,\n // @ts-ignore limit exist but is not typed\n limit: 40,\n choices: [\n { name: SELECT_ALL, message: '────── Select all ──────' },\n ...choices.map((choice) => ({\n name: choice.value,\n message: truncatePath(choice.label),\n })),\n ],\n async toggle(\n this: PromptContext,\n choice: PromptChoice,\n enabled?: boolean\n ) {\n if (!choice || choice.disabled) return;\n choice.enabled = enabled == null ? !choice.enabled : enabled;\n\n if (choice.name === SELECT_ALL) {\n this.choices\n .filter((choiceEl) => choiceEl.name !== SELECT_ALL)\n .forEach((choiceEl) => {\n choiceEl.enabled = choice.enabled;\n });\n }\n\n return this.render();\n },\n format(this: PromptContext) {\n if (this.state?.submitted && this.options?.multiple) {\n return `${this.selected.filter((s) => s.name !== SELECT_ALL).length} file(s) selected`;\n }\n return this.input ?? '';\n },\n });\n\n selectedFiles = enquirerSelectedFiles.filter((f) => f !== SELECT_ALL);\n } catch {\n selectedFiles = Symbol('cancel');\n }\n\n if (typeof selectedFiles === 'symbol') {\n // User cancelled\n process.exit(0);\n }\n\n filesToExtract = selectedFiles as string[];\n }\n\n if (filesToExtract.length === 0) {\n appLogger('No files selected for extraction.');\n return;\n }\n\n const absoluteFiles = filesToExtract\n .map((file) => resolve(baseDir, file))\n .filter((file) => {\n if (!existsSync(file)) {\n appLogger(`File not found: ${formatPath(file)}`);\n return false;\n }\n return true;\n });\n\n if (absoluteFiles.length === 0) {\n return;\n }\n\n const unmergedDictionaries = getUnmergedDictionaries(configuration);\n\n await Promise.all(\n absoluteFiles.map(async (filePath) => {\n try {\n await extractContent(filePath, packageName, {\n unmergedDictionaries,\n configuration,\n output: options.output,\n codeOnly: options.codeOnly,\n declarationOnly: options.declarationOnly,\n });\n } catch (error) {\n appLogger(\n `Failed to transform ${filePath}: ${(error as Error).message}`\n );\n }\n })\n );\n};\n"],"mappings":"ifAwBA,MAAM,EAAkB,KAAO,IAAoB,CACjD,GAAI,CACF,IAAM,EAAkB,EAAQ,EAAS,eAAe,CACxD,GAAI,CAAC,EAAW,EAAgB,CAE9B,MAAO,EAAE,CAEX,IAAM,EAAO,MAAM,EAAS,EAAiB,OAAO,CAGpD,OAFoB,KAAK,MAAM,EAAK,CAEjB,kBACb,CACN,MAAO,EAAE,GAIA,EAAU,KAAO,IAA4B,CACxD,IAAM,EAAgB,EAAiB,EAAQ,cAAc,CAC7D,EAAiB,GAAS,cAAc,CAExC,IAAM,EAAY,EAAa,EAAc,CACvC,CAAE,UAAS,UAAS,gBAAiB,EAAc,QACnD,CAAE,mBAAoB,EAAc,MAEpC,EAAc,GAEX,EADc,EAAS,EAAS,EAAK,CACX,CAI7B,EAAe,MAAM,EAAgB,EAAQ,CAC/C,EAA2B,iBAE3B,EAAa,iBACf,EAAc,gBACL,EAAa,gBACtB,EAAc,eACL,EAAa,mBACtB,EAAc,kBACL,EAAa,kBACtB,EAAc,iBACL,EAAa,mBACtB,EAAc,kBACL,EAAa,kBACtB,EAAc,iBACL,EAAa,oBACtB,EAAc,mBACL,EAAa,sBACtB,EAAc,oBAGhB,IAAI,EAAiB,EAAQ,OAAS,EAAE,CAExC,GAAI,EAAe,SAAW,EAAG,CAE/B,IAAM,EAAiB,IACpB,GAAgB,EAAE,EAAE,KAAM,GAAY,CACrC,IAAM,EAAW,EACd,MAAM,IAAI,CACV,OAAQ,GAAM,CAAC,EAAE,SAAS,IAAI,EAAI,EAAE,OAAS,EAAE,CAC5C,EAAQ,EAAQ,MAAM,IAAI,CAChC,OAAO,EAAS,KAAM,GAAQ,EAAM,SAAS,EAAI,CAAC,EAClD,CAgBE,GAbe,MAAM,QAAQ,IACjC,EACG,OAAQ,GAAQ,CAAC,EAAc,EAAI,CAAC,CACpC,IAAK,GACJ,EAAG,EAAiB,CAClB,IAAK,EACL,OAAQ,EACR,SAAU,GACX,CAAC,CACH,CACJ,EAG6B,MAAM,CAO9B,EALc,CAAC,GAAG,IAAI,IAAI,EAAS,CAAC,CAAC,OAAQ,GACjD,EAAW,EAAK,CACjB,CAG2B,IAAK,IAExB,CACL,MAAO,EACP,MAHc,EAAS,EAAS,EAAK,CAItC,EACD,CAEF,GAAI,EAAQ,SAAW,EAAG,CACxB,EAAU,6CAA6C,CACvD,OAGF,IAAM,EAAa,iBAiBf,EACJ,GAAI,CACF,IAAM,EAAS,KAAK,KAAK,QAAQ,OAAO,SAAW,IAAM,GAAI,GAAG,CAC1D,EAAgB,GACpB,EAAK,OAAS,EAAS,MAAM,EAAK,MAAM,EAAE,EAAS,GAAG,GAAK,EAEvD,CAAE,MAAO,GAA0B,MAAM,EAAS,OAErD,CACD,KAAM,eACN,KAAM,QACN,QAAS,4CACT,SAAU,GAEV,MAAO,GACP,QAAS,CACP,CAAE,KAAM,EAAY,QAAS,2BAA4B,CACzD,GAAG,EAAQ,IAAK,IAAY,CAC1B,KAAM,EAAO,MACb,QAAS,EAAa,EAAO,MAAM,CACpC,EAAE,CACJ,CACD,MAAM,OAEJ,EACA,EACA,CACI,MAAC,GAAU,EAAO,UAWtB,MAVA,GAAO,QAAU,GAAkB,CAAC,EAAO,QAEvC,EAAO,OAAS,GAClB,KAAK,QACF,OAAQ,GAAa,EAAS,OAAS,EAAW,CAClD,QAAS,GAAa,CACrB,EAAS,QAAU,EAAO,SAC1B,CAGC,KAAK,QAAQ,EAEtB,QAA4B,CAI1B,OAHI,KAAK,OAAO,WAAa,KAAK,SAAS,SAClC,GAAG,KAAK,SAAS,OAAQ,GAAM,EAAE,OAAS,EAAW,CAAC,OAAO,mBAE/D,KAAK,OAAS,IAExB,CAAC,CAEF,EAAgB,EAAsB,OAAQ,GAAM,IAAM,EAAW,MAC/D,CACN,EAAgB,OAAO,SAAS,CAG9B,OAAO,GAAkB,UAE3B,QAAQ,KAAK,EAAE,CAGjB,EAAiB,EAGnB,GAAI,EAAe,SAAW,EAAG,CAC/B,EAAU,oCAAoC,CAC9C,OAGF,IAAM,EAAgB,EACnB,IAAK,GAAS,EAAQ,EAAS,EAAK,CAAC,CACrC,OAAQ,GACF,EAAW,EAAK,CAId,IAHL,EAAU,mBAAmB,EAAW,EAAK,GAAG,CACzC,IAGT,CAEJ,GAAI,EAAc,SAAW,EAC3B,OAGF,IAAM,EAAuB,EAAwB,EAAc,CAEnE,MAAM,QAAQ,IACZ,EAAc,IAAI,KAAO,IAAa,CACpC,GAAI,CACF,MAAM,EAAe,EAAU,EAAa,CAC1C,uBACA,gBACA,OAAQ,EAAQ,OAChB,SAAU,EAAQ,SAClB,gBAAiB,EAAQ,gBAC1B,CAAC,OACK,EAAO,CACd,EACE,uBAAuB,EAAS,IAAK,EAAgB,UACtD,GAEH,CACH"}
@@ -1,2 +1,2 @@
1
- import{ensureArray as e,getTargetUnmergedDictionaries as t}from"../getTargetDictionary.mjs";import{setupAI as n}from"../utils/setupAI.mjs";import{listTranslationsTasks as r}from"./listTranslationsTasks.mjs";import{translateDictionary as i}from"./translateDictionary.mjs";import{writeFill as a}from"./writeFill.mjs";import{ANSIColors as o,colorize as s,colorizeKey as c,colorizePath as l,getAppLogger as u}from"@intlayer/config/logger";import{logConfigDetails as d}from"@intlayer/chokidar/cli";import{formatPath as f,getGlobalLimiter as p,getTaskLimiter as m}from"@intlayer/chokidar/utils";import{getConfiguration as h}from"@intlayer/config/node";import{basename as g,relative as _}from"node:path";import{prepareIntlayer as v,writeContentDeclaration as y}from"@intlayer/chokidar/build";const b=async b=>{let x=h(b?.configOptions);d(b?.configOptions);let S=u(x);b?.build===!0?await v(x,{forceRun:!0}):b?.build===void 0&&await v(x);let{defaultLocale:C,locales:w}=x.internationalization,T=b?.mode??`complete`,E=b?.sourceLocale??C,D=b?.outputLocales?e(b.outputLocales):w,O=await n(x,b?.aiOptions);if(!O?.hasAIAccess)return;let{aiClient:k,aiConfig:A}=O,j=await t(b),M=new Set;j.forEach(e=>{M.add(e.key)});let N=Array.from(M);if(S([`Affected dictionary keys for processing:`,N.length>0?N.map(e=>c(e)).join(`, `):s(`No keys found`,o.YELLOW)]),N.length===0)return;let P=r(j.map(e=>e.localId),D,T,E,x),F=b?.nbConcurrentTranslations??7,I=p(F),L=m(Math.max(1,Math.min(b?.nbConcurrentTasks??F,P.length))),R=P.map(e=>L(async()=>{let t=_(x?.content?.baseDir??process.cwd(),e?.dictionaryFilePath??``);S(`${e.dictionaryPreset} Processing ${l(g(t))}`,{level:`info`});let n=await i(e,x,{mode:T,aiOptions:b?.aiOptions,fillMetadata:!b?.skipMetadata,onHandle:I,aiClient:k,aiConfig:A});if(!n?.dictionaryOutput)return;let{dictionaryOutput:r,sourceLocale:o}=n,s=typeof r.fill==`string`||typeof r.fill==`object`,c=typeof r.locale==`string`,u=s?r.fill:c?x.dictionary?.fill??!0:!1;typeof u==`string`||typeof u==`object`?await a({...r,fill:u},D,[o],x):(await y(r,x),r.filePath&&S(`${e.dictionaryPreset} Content declaration written to ${f(g(r.filePath))}`,{level:`info`}))}));await Promise.all(R),await I.onIdle()};export{b as fill};
1
+ import{ensureArray as e,getTargetUnmergedDictionaries as t}from"../getTargetDictionary.mjs";import{setupAI as n}from"../utils/setupAI.mjs";import{listTranslationsTasks as r}from"./listTranslationsTasks.mjs";import{translateDictionary as i}from"./translateDictionary.mjs";import{writeFill as a}from"./writeFill.mjs";import{basename as o,relative as s}from"node:path";import{logConfigDetails as c}from"@intlayer/chokidar/cli";import{ANSIColors as l,colorize as u,colorizeKey as d,colorizePath as f,getAppLogger as p}from"@intlayer/config/logger";import{getConfiguration as m}from"@intlayer/config/node";import{prepareIntlayer as h,writeContentDeclaration as g}from"@intlayer/chokidar/build";import{formatPath as _,getGlobalLimiter as v,getTaskLimiter as y}from"@intlayer/chokidar/utils";const b=async b=>{let x=m(b?.configOptions);c(b?.configOptions);let S=p(x);b?.build===!0?await h(x,{forceRun:!0}):b?.build===void 0&&await h(x);let{defaultLocale:C,locales:w}=x.internationalization,T=b?.mode??`complete`,E=b?.sourceLocale??C,D=b?.outputLocales?e(b.outputLocales):w,O=await n(x,b?.aiOptions);if(!O?.hasAIAccess)return;let{aiClient:k,aiConfig:A}=O,j=await t(b),M=new Set;j.forEach(e=>{M.add(e.key)});let N=Array.from(M);if(S([`Affected dictionary keys for processing:`,N.length>0?N.map(e=>d(e)).join(`, `):u(`No keys found`,l.YELLOW)]),N.length===0)return;let P=r(j.map(e=>e.localId),D,T,E,x),F=b?.nbConcurrentTranslations??7,I=v(F),L=y(Math.max(1,Math.min(b?.nbConcurrentTasks??F,P.length))),R=P.map(e=>L(async()=>{let t=s(x?.content?.baseDir??process.cwd(),e?.dictionaryFilePath??``);S(`${e.dictionaryPreset} Processing ${f(o(t))}`,{level:`info`});let n=await i(e,x,{mode:T,aiOptions:b?.aiOptions,fillMetadata:!b?.skipMetadata,onHandle:I,aiClient:k,aiConfig:A});if(!n?.dictionaryOutput)return;let{dictionaryOutput:r,sourceLocale:c}=n,l=typeof r.fill==`string`||typeof r.fill==`object`,u=typeof r.locale==`string`,d=l?r.fill:u?x.dictionary?.fill??!0:!1;typeof d==`string`||typeof d==`object`?await a({...r,fill:d},D,[c],x):(await g(r,x),r.filePath&&S(`${e.dictionaryPreset} Content declaration written to ${_(o(r.filePath))}`,{level:`info`}))}));await Promise.all(R),await I.onIdle()};export{b as fill};
2
2
  //# sourceMappingURL=fill.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"fill.mjs","names":[],"sources":["../../../src/fill/fill.ts"],"sourcesContent":["import { basename, relative } from 'node:path';\nimport type { AIOptions } from '@intlayer/api';\nimport {\n prepareIntlayer,\n writeContentDeclaration,\n} from '@intlayer/chokidar/build';\nimport {\n type ListGitFilesOptions,\n logConfigDetails,\n} from '@intlayer/chokidar/cli';\nimport {\n formatPath,\n getGlobalLimiter,\n getTaskLimiter,\n} from '@intlayer/chokidar/utils';\nimport {\n ANSIColors,\n colorize,\n colorizeKey,\n colorizePath,\n getAppLogger,\n} from '@intlayer/config/logger';\nimport { getConfiguration } from '@intlayer/config/node';\nimport type { Locale } from '@intlayer/types';\nimport {\n ensureArray,\n type GetTargetDictionaryOptions,\n getTargetUnmergedDictionaries,\n} from '../getTargetDictionary';\nimport { setupAI } from '../utils/setupAI';\nimport {\n listTranslationsTasks,\n type TranslationTask,\n} from './listTranslationsTasks';\nimport { translateDictionary } from './translateDictionary';\nimport { writeFill } from './writeFill';\n\nconst NB_CONCURRENT_TRANSLATIONS = 7;\n\n// Arguments for the fill function\nexport type FillOptions = {\n sourceLocale?: Locale;\n outputLocales?: Locale | Locale[];\n mode?: 'complete' | 'review';\n gitOptions?: ListGitFilesOptions;\n aiOptions?: AIOptions; // Added aiOptions to be passed to translateJSON\n verbose?: boolean;\n nbConcurrentTranslations?: number;\n nbConcurrentTasks?: number; // NEW: number of tasks that may run at once\n build?: boolean;\n skipMetadata?: boolean;\n} & GetTargetDictionaryOptions;\n\n/**\n * Fill translations based on the provided options.\n */\nexport const fill = async (options?: FillOptions): Promise<void> => {\n const configuration = getConfiguration(options?.configOptions);\n logConfigDetails(options?.configOptions);\n\n const appLogger = getAppLogger(configuration);\n\n if (options?.build === true) {\n await prepareIntlayer(configuration, { forceRun: true });\n } else if (typeof options?.build === 'undefined') {\n await prepareIntlayer(configuration);\n }\n\n const { defaultLocale, locales } = configuration.internationalization;\n const mode = options?.mode ?? 'complete';\n const baseLocale = options?.sourceLocale ?? defaultLocale;\n\n const outputLocales = options?.outputLocales\n ? ensureArray(options.outputLocales)\n : locales;\n\n const aiResult = await setupAI(configuration, options?.aiOptions);\n\n if (!aiResult?.hasAIAccess) return;\n\n const { aiClient, aiConfig } = aiResult;\n\n const targetUnmergedDictionaries =\n await getTargetUnmergedDictionaries(options);\n\n const affectedDictionaryKeys = new Set<string>();\n targetUnmergedDictionaries.forEach((dict) => {\n affectedDictionaryKeys.add(dict.key);\n });\n\n const keysToProcess = Array.from(affectedDictionaryKeys);\n\n appLogger([\n 'Affected dictionary keys for processing:',\n keysToProcess.length > 0\n ? keysToProcess.map((key) => colorizeKey(key)).join(', ')\n : colorize('No keys found', ANSIColors.YELLOW),\n ]);\n\n if (keysToProcess.length === 0) return;\n\n /**\n * List the translations tasks\n *\n * Create a list of per-locale dictionaries to translate\n *\n * In 'complete' mode, filter only the missing locales to translate\n */\n const translationTasks: TranslationTask[] = listTranslationsTasks(\n targetUnmergedDictionaries.map((dictionary) => dictionary.localId!),\n outputLocales,\n mode,\n baseLocale,\n configuration\n );\n\n // AI calls in flight at once (translateJSON + metadata audit)\n const nbConcurrentTranslations =\n options?.nbConcurrentTranslations ?? NB_CONCURRENT_TRANSLATIONS;\n const globalLimiter = getGlobalLimiter(nbConcurrentTranslations);\n\n // NEW: number of *tasks* that may run at once (start/prepare/log/write)\n const nbConcurrentTasks = Math.max(\n 1,\n Math.min(\n options?.nbConcurrentTasks ?? nbConcurrentTranslations,\n translationTasks.length\n )\n );\n\n const taskLimiter = getTaskLimiter(nbConcurrentTasks);\n\n const runners = translationTasks.map((task) =>\n taskLimiter(async () => {\n const relativePath = relative(\n configuration?.content?.baseDir ?? process.cwd(),\n task?.dictionaryFilePath ?? ''\n );\n\n // log AFTER acquiring a task slot\n appLogger(\n `${task.dictionaryPreset} Processing ${colorizePath(basename(relativePath))}`,\n { level: 'info' }\n );\n\n const translationTaskResult = await translateDictionary(\n task,\n configuration,\n {\n mode,\n aiOptions: options?.aiOptions,\n fillMetadata: !options?.skipMetadata,\n onHandle: globalLimiter, // <= AI calls go through here\n aiClient,\n aiConfig,\n }\n );\n\n if (!translationTaskResult?.dictionaryOutput) return;\n\n const { dictionaryOutput, sourceLocale } = translationTaskResult;\n\n // Determine if we should write to separate files\n // - If dictionary has explicit fill setting (string or object), use it\n // - If dictionary is per-locale AND has no explicit fill=false, use global fill config\n // - If dictionary is multilingual (no locale property), always write to same file\n const hasDictionaryLevelFill =\n typeof dictionaryOutput.fill === 'string' ||\n typeof dictionaryOutput.fill === 'object';\n\n const isPerLocale = typeof dictionaryOutput.locale === 'string';\n\n const effectiveFill = hasDictionaryLevelFill\n ? dictionaryOutput.fill\n : isPerLocale\n ? (configuration.dictionary?.fill ?? true)\n : false; // Multilingual dictionaries don't use fill by default\n\n const isFillOtherFile =\n typeof effectiveFill === 'string' || typeof effectiveFill === 'object';\n\n if (isFillOtherFile) {\n await writeFill(\n {\n ...dictionaryOutput,\n // Ensure fill is set on the dictionary for writeFill to use\n fill: effectiveFill,\n },\n outputLocales,\n [sourceLocale],\n configuration\n );\n } else {\n await writeContentDeclaration(dictionaryOutput, configuration);\n\n if (dictionaryOutput.filePath) {\n appLogger(\n `${task.dictionaryPreset} Content declaration written to ${formatPath(basename(dictionaryOutput.filePath))}`,\n { level: 'info' }\n );\n }\n }\n })\n );\n\n await Promise.all(runners);\n await (globalLimiter as any).onIdle();\n};\n"],"mappings":"ixBAqCA,MAmBa,EAAO,KAAO,IAAyC,CAClE,IAAM,EAAgB,EAAiB,GAAS,cAAc,CAC9D,EAAiB,GAAS,cAAc,CAExC,IAAM,EAAY,EAAa,EAAc,CAEzC,GAAS,QAAU,GACrB,MAAM,EAAgB,EAAe,CAAE,SAAU,GAAM,CAAC,CACxC,GAAS,QAAU,QACnC,MAAM,EAAgB,EAAc,CAGtC,GAAM,CAAE,gBAAe,WAAY,EAAc,qBAC3C,EAAO,GAAS,MAAQ,WACxB,EAAa,GAAS,cAAgB,EAEtC,EAAgB,GAAS,cAC3B,EAAY,EAAQ,cAAc,CAClC,EAEE,EAAW,MAAM,EAAQ,EAAe,GAAS,UAAU,CAEjE,GAAI,CAAC,GAAU,YAAa,OAE5B,GAAM,CAAE,WAAU,YAAa,EAEzB,EACJ,MAAM,EAA8B,EAAQ,CAExC,EAAyB,IAAI,IACnC,EAA2B,QAAS,GAAS,CAC3C,EAAuB,IAAI,EAAK,IAAI,EACpC,CAEF,IAAM,EAAgB,MAAM,KAAK,EAAuB,CASxD,GAPA,EAAU,CACR,2CACA,EAAc,OAAS,EACnB,EAAc,IAAK,GAAQ,EAAY,EAAI,CAAC,CAAC,KAAK,KAAK,CACvD,EAAS,gBAAiB,EAAW,OAAO,CACjD,CAAC,CAEE,EAAc,SAAW,EAAG,OAShC,IAAM,EAAsC,EAC1C,EAA2B,IAAK,GAAe,EAAW,QAAS,CACnE,EACA,EACA,EACA,EACD,CAGK,EACJ,GAAS,0BAA4B,EACjC,EAAgB,EAAiB,EAAyB,CAW1D,EAAc,EARM,KAAK,IAC7B,EACA,KAAK,IACH,GAAS,mBAAqB,EAC9B,EAAiB,OAClB,CACF,CAEoD,CAE/C,EAAU,EAAiB,IAAK,GACpC,EAAY,SAAY,CACtB,IAAM,EAAe,EACnB,GAAe,SAAS,SAAW,QAAQ,KAAK,CAChD,GAAM,oBAAsB,GAC7B,CAGD,EACE,GAAG,EAAK,iBAAiB,cAAc,EAAa,EAAS,EAAa,CAAC,GAC3E,CAAE,MAAO,OAAQ,CAClB,CAED,IAAM,EAAwB,MAAM,EAClC,EACA,EACA,CACE,OACA,UAAW,GAAS,UACpB,aAAc,CAAC,GAAS,aACxB,SAAU,EACV,WACA,WACD,CACF,CAED,GAAI,CAAC,GAAuB,iBAAkB,OAE9C,GAAM,CAAE,mBAAkB,gBAAiB,EAMrC,EACJ,OAAO,EAAiB,MAAS,UACjC,OAAO,EAAiB,MAAS,SAE7B,EAAc,OAAO,EAAiB,QAAW,SAEjD,EAAgB,EAClB,EAAiB,KACjB,EACG,EAAc,YAAY,MAAQ,GACnC,GAGJ,OAAO,GAAkB,UAAY,OAAO,GAAkB,SAG9D,MAAM,EACJ,CACE,GAAG,EAEH,KAAM,EACP,CACD,EACA,CAAC,EAAa,CACd,EACD,EAED,MAAM,EAAwB,EAAkB,EAAc,CAE1D,EAAiB,UACnB,EACE,GAAG,EAAK,iBAAiB,kCAAkC,EAAW,EAAS,EAAiB,SAAS,CAAC,GAC1G,CAAE,MAAO,OAAQ,CAClB,GAGL,CACH,CAED,MAAM,QAAQ,IAAI,EAAQ,CAC1B,MAAO,EAAsB,QAAQ"}
1
+ {"version":3,"file":"fill.mjs","names":[],"sources":["../../../src/fill/fill.ts"],"sourcesContent":["import { basename, relative } from 'node:path';\nimport type { AIOptions } from '@intlayer/api';\nimport {\n prepareIntlayer,\n writeContentDeclaration,\n} from '@intlayer/chokidar/build';\nimport {\n type ListGitFilesOptions,\n logConfigDetails,\n} from '@intlayer/chokidar/cli';\nimport {\n formatPath,\n getGlobalLimiter,\n getTaskLimiter,\n} from '@intlayer/chokidar/utils';\nimport {\n ANSIColors,\n colorize,\n colorizeKey,\n colorizePath,\n getAppLogger,\n} from '@intlayer/config/logger';\nimport { getConfiguration } from '@intlayer/config/node';\nimport type { Locale } from '@intlayer/types/allLocales';\nimport {\n ensureArray,\n type GetTargetDictionaryOptions,\n getTargetUnmergedDictionaries,\n} from '../getTargetDictionary';\nimport { setupAI } from '../utils/setupAI';\nimport {\n listTranslationsTasks,\n type TranslationTask,\n} from './listTranslationsTasks';\nimport { translateDictionary } from './translateDictionary';\nimport { writeFill } from './writeFill';\n\nconst NB_CONCURRENT_TRANSLATIONS = 7;\n\n// Arguments for the fill function\nexport type FillOptions = {\n sourceLocale?: Locale;\n outputLocales?: Locale | Locale[];\n mode?: 'complete' | 'review';\n gitOptions?: ListGitFilesOptions;\n aiOptions?: AIOptions; // Added aiOptions to be passed to translateJSON\n verbose?: boolean;\n nbConcurrentTranslations?: number;\n nbConcurrentTasks?: number; // NEW: number of tasks that may run at once\n build?: boolean;\n skipMetadata?: boolean;\n} & GetTargetDictionaryOptions;\n\n/**\n * Fill translations based on the provided options.\n */\nexport const fill = async (options?: FillOptions): Promise<void> => {\n const configuration = getConfiguration(options?.configOptions);\n logConfigDetails(options?.configOptions);\n\n const appLogger = getAppLogger(configuration);\n\n if (options?.build === true) {\n await prepareIntlayer(configuration, { forceRun: true });\n } else if (typeof options?.build === 'undefined') {\n await prepareIntlayer(configuration);\n }\n\n const { defaultLocale, locales } = configuration.internationalization;\n const mode = options?.mode ?? 'complete';\n const baseLocale = options?.sourceLocale ?? defaultLocale;\n\n const outputLocales = options?.outputLocales\n ? ensureArray(options.outputLocales)\n : locales;\n\n const aiResult = await setupAI(configuration, options?.aiOptions);\n\n if (!aiResult?.hasAIAccess) return;\n\n const { aiClient, aiConfig } = aiResult;\n\n const targetUnmergedDictionaries =\n await getTargetUnmergedDictionaries(options);\n\n const affectedDictionaryKeys = new Set<string>();\n targetUnmergedDictionaries.forEach((dict) => {\n affectedDictionaryKeys.add(dict.key);\n });\n\n const keysToProcess = Array.from(affectedDictionaryKeys);\n\n appLogger([\n 'Affected dictionary keys for processing:',\n keysToProcess.length > 0\n ? keysToProcess.map((key) => colorizeKey(key)).join(', ')\n : colorize('No keys found', ANSIColors.YELLOW),\n ]);\n\n if (keysToProcess.length === 0) return;\n\n /**\n * List the translations tasks\n *\n * Create a list of per-locale dictionaries to translate\n *\n * In 'complete' mode, filter only the missing locales to translate\n */\n const translationTasks: TranslationTask[] = listTranslationsTasks(\n targetUnmergedDictionaries.map((dictionary) => dictionary.localId!),\n outputLocales,\n mode,\n baseLocale,\n configuration\n );\n\n // AI calls in flight at once (translateJSON + metadata audit)\n const nbConcurrentTranslations =\n options?.nbConcurrentTranslations ?? NB_CONCURRENT_TRANSLATIONS;\n const globalLimiter = getGlobalLimiter(nbConcurrentTranslations);\n\n // NEW: number of *tasks* that may run at once (start/prepare/log/write)\n const nbConcurrentTasks = Math.max(\n 1,\n Math.min(\n options?.nbConcurrentTasks ?? nbConcurrentTranslations,\n translationTasks.length\n )\n );\n\n const taskLimiter = getTaskLimiter(nbConcurrentTasks);\n\n const runners = translationTasks.map((task) =>\n taskLimiter(async () => {\n const relativePath = relative(\n configuration?.content?.baseDir ?? process.cwd(),\n task?.dictionaryFilePath ?? ''\n );\n\n // log AFTER acquiring a task slot\n appLogger(\n `${task.dictionaryPreset} Processing ${colorizePath(basename(relativePath))}`,\n { level: 'info' }\n );\n\n const translationTaskResult = await translateDictionary(\n task,\n configuration,\n {\n mode,\n aiOptions: options?.aiOptions,\n fillMetadata: !options?.skipMetadata,\n onHandle: globalLimiter, // <= AI calls go through here\n aiClient,\n aiConfig,\n }\n );\n\n if (!translationTaskResult?.dictionaryOutput) return;\n\n const { dictionaryOutput, sourceLocale } = translationTaskResult;\n\n // Determine if we should write to separate files\n // - If dictionary has explicit fill setting (string or object), use it\n // - If dictionary is per-locale AND has no explicit fill=false, use global fill config\n // - If dictionary is multilingual (no locale property), always write to same file\n const hasDictionaryLevelFill =\n typeof dictionaryOutput.fill === 'string' ||\n typeof dictionaryOutput.fill === 'object';\n\n const isPerLocale = typeof dictionaryOutput.locale === 'string';\n\n const effectiveFill = hasDictionaryLevelFill\n ? dictionaryOutput.fill\n : isPerLocale\n ? (configuration.dictionary?.fill ?? true)\n : false; // Multilingual dictionaries don't use fill by default\n\n const isFillOtherFile =\n typeof effectiveFill === 'string' || typeof effectiveFill === 'object';\n\n if (isFillOtherFile) {\n await writeFill(\n {\n ...dictionaryOutput,\n // Ensure fill is set on the dictionary for writeFill to use\n fill: effectiveFill,\n },\n outputLocales,\n [sourceLocale],\n configuration\n );\n } else {\n await writeContentDeclaration(dictionaryOutput, configuration);\n\n if (dictionaryOutput.filePath) {\n appLogger(\n `${task.dictionaryPreset} Content declaration written to ${formatPath(basename(dictionaryOutput.filePath))}`,\n { level: 'info' }\n );\n }\n }\n })\n );\n\n await Promise.all(runners);\n await (globalLimiter as any).onIdle();\n};\n"],"mappings":"ixBAqCA,MAmBa,EAAO,KAAO,IAAyC,CAClE,IAAM,EAAgB,EAAiB,GAAS,cAAc,CAC9D,EAAiB,GAAS,cAAc,CAExC,IAAM,EAAY,EAAa,EAAc,CAEzC,GAAS,QAAU,GACrB,MAAM,EAAgB,EAAe,CAAE,SAAU,GAAM,CAAC,CACxC,GAAS,QAAU,QACnC,MAAM,EAAgB,EAAc,CAGtC,GAAM,CAAE,gBAAe,WAAY,EAAc,qBAC3C,EAAO,GAAS,MAAQ,WACxB,EAAa,GAAS,cAAgB,EAEtC,EAAgB,GAAS,cAC3B,EAAY,EAAQ,cAAc,CAClC,EAEE,EAAW,MAAM,EAAQ,EAAe,GAAS,UAAU,CAEjE,GAAI,CAAC,GAAU,YAAa,OAE5B,GAAM,CAAE,WAAU,YAAa,EAEzB,EACJ,MAAM,EAA8B,EAAQ,CAExC,EAAyB,IAAI,IACnC,EAA2B,QAAS,GAAS,CAC3C,EAAuB,IAAI,EAAK,IAAI,EACpC,CAEF,IAAM,EAAgB,MAAM,KAAK,EAAuB,CASxD,GAPA,EAAU,CACR,2CACA,EAAc,OAAS,EACnB,EAAc,IAAK,GAAQ,EAAY,EAAI,CAAC,CAAC,KAAK,KAAK,CACvD,EAAS,gBAAiB,EAAW,OAAO,CACjD,CAAC,CAEE,EAAc,SAAW,EAAG,OAShC,IAAM,EAAsC,EAC1C,EAA2B,IAAK,GAAe,EAAW,QAAS,CACnE,EACA,EACA,EACA,EACD,CAGK,EACJ,GAAS,0BAA4B,EACjC,EAAgB,EAAiB,EAAyB,CAW1D,EAAc,EARM,KAAK,IAC7B,EACA,KAAK,IACH,GAAS,mBAAqB,EAC9B,EAAiB,OAClB,CACF,CAEoD,CAE/C,EAAU,EAAiB,IAAK,GACpC,EAAY,SAAY,CACtB,IAAM,EAAe,EACnB,GAAe,SAAS,SAAW,QAAQ,KAAK,CAChD,GAAM,oBAAsB,GAC7B,CAGD,EACE,GAAG,EAAK,iBAAiB,cAAc,EAAa,EAAS,EAAa,CAAC,GAC3E,CAAE,MAAO,OAAQ,CAClB,CAED,IAAM,EAAwB,MAAM,EAClC,EACA,EACA,CACE,OACA,UAAW,GAAS,UACpB,aAAc,CAAC,GAAS,aACxB,SAAU,EACV,WACA,WACD,CACF,CAED,GAAI,CAAC,GAAuB,iBAAkB,OAE9C,GAAM,CAAE,mBAAkB,gBAAiB,EAMrC,EACJ,OAAO,EAAiB,MAAS,UACjC,OAAO,EAAiB,MAAS,SAE7B,EAAc,OAAO,EAAiB,QAAW,SAEjD,EAAgB,EAClB,EAAiB,KACjB,EACG,EAAc,YAAY,MAAQ,GACnC,GAGJ,OAAO,GAAkB,UAAY,OAAO,GAAkB,SAG9D,MAAM,EACJ,CACE,GAAG,EAEH,KAAM,EACP,CACD,EACA,CAAC,EAAa,CACd,EACD,EAED,MAAM,EAAwB,EAAkB,EAAc,CAE1D,EAAiB,UACnB,EACE,GAAG,EAAK,iBAAiB,kCAAkC,EAAW,EAAS,EAAiB,SAAS,CAAC,GAC1G,CAAE,MAAO,OAAQ,CAClB,GAGL,CACH,CAED,MAAM,QAAQ,IAAI,EAAQ,CAC1B,MAAO,EAAsB,QAAQ"}
@@ -1,2 +1,2 @@
1
- import{basename as e,dirname as t,isAbsolute as n,normalize as r,resolve as i}from"node:path";const a=(a,o,s,c,l)=>{if(!a||typeof a!=`string`)throw Error(`autoFillField must be a non-empty string`);if(!o||typeof o!=`string`)throw Error(`dictionaryKey must be a non-empty string`);if(!s||typeof s!=`string`)throw Error(`dictionaryFilePath must be a non-empty string`);if(!c||typeof c!=`string`)throw Error(`baseDir must be a non-empty string`);let u=e(s).split(`.`).slice(0,-2).join(`.`),d=a.replace(/\{\{key\}\}/g,o).replace(/\{\{fileName\}\}/g,u);l&&(d=d.replace(/\{\{locale\}\}/g,l));let f=n(s)?s:i(c,s);if(d.startsWith(`./`)||d.startsWith(`../`))return i(t(f),d);if(n(d)){let e=r(d),t=r(c);if(d.startsWith(`/`)&&!e.startsWith(t)){let e=i(c,d.substring(1));if(!d.startsWith(`/usr/`)&&!d.startsWith(`/etc/`)&&!d.startsWith(`/var/`)&&!d.startsWith(`/home/`)&&!d.startsWith(`/Users/`))return e}return e}return r(d)};export{a as formatAutoFilledFilePath};
1
+ import{basename as e,dirname as t,isAbsolute as n,normalize as r,resolve as i}from"node:path";import{parseFilePathPattern as a}from"@intlayer/config/utils";const o=async(o,s,c,l,u)=>{if(!o)throw Error(`autoFillField must be provided`);if(!s||typeof s!=`string`)throw Error(`dictionaryKey must be a non-empty string`);if(!c||typeof c!=`string`)throw Error(`dictionaryFilePath must be a non-empty string`);if(!l||typeof l!=`string`)throw Error(`baseDir must be a non-empty string`);let d=await a(o,{key:s,fileName:e(c).split(`.`).slice(0,-2).join(`.`),locale:u}),f=n(c)?c:i(l,c);if(d.startsWith(`./`)||d.startsWith(`../`))return i(t(f),d);if(n(d)){let e=r(d),t=r(l);if(d.startsWith(`/`)&&!e.startsWith(t)){let e=i(l,d.substring(1));if(!d.startsWith(`/usr/`)&&!d.startsWith(`/etc/`)&&!d.startsWith(`/var/`)&&!d.startsWith(`/home/`)&&!d.startsWith(`/Users/`))return e}return e}return r(d)};export{o as formatAutoFilledFilePath};
2
2
  //# sourceMappingURL=formatAutoFilledFilePath.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"formatAutoFilledFilePath.mjs","names":[],"sources":["../../../src/fill/formatAutoFilledFilePath.ts"],"sourcesContent":["import { basename, dirname, isAbsolute, normalize, resolve } from 'node:path';\nimport type { LocalesValues } from '@intlayer/types';\n\nexport const formatAutoFilledFilePath = (\n autoFillField: string,\n dictionaryKey: string,\n dictionaryFilePath: string,\n baseDir: string,\n locale?: LocalesValues\n): string => {\n // Validate inputs\n if (!autoFillField || typeof autoFillField !== 'string') {\n throw new Error('autoFillField must be a non-empty string');\n }\n if (!dictionaryKey || typeof dictionaryKey !== 'string') {\n throw new Error('dictionaryKey must be a non-empty string');\n }\n if (!dictionaryFilePath || typeof dictionaryFilePath !== 'string') {\n throw new Error('dictionaryFilePath must be a non-empty string');\n }\n if (!baseDir || typeof baseDir !== 'string') {\n throw new Error('baseDir must be a non-empty string');\n }\n\n // Extract the original filename without extensions (.content.ts -> dictionaryFieldEditor)\n const originalFileName = basename(dictionaryFilePath)\n .split('.')\n .slice(0, -2) // Remove last 2 extensions (.content.tsx)\n .join('.');\n\n // Replace placeholders in autoFillField\n let result: string = autoFillField\n .replace(/\\{\\{key\\}\\}/g, dictionaryKey) // Use original filename, not dictionaryKey\n .replace(/\\{\\{fileName\\}\\}/g, originalFileName);\n\n if (locale) {\n result = result.replace(/\\{\\{locale\\}\\}/g, locale);\n }\n\n // Normalize the dictionary file path - if it's relative, make it absolute relative to baseDir\n const absoluteDictionaryPath = isAbsolute(dictionaryFilePath)\n ? dictionaryFilePath\n : resolve(baseDir, dictionaryFilePath);\n\n // Handle relative paths (starting with ./ or ../)\n if (result.startsWith('./') || result.startsWith('../')) {\n const fileDir = dirname(absoluteDictionaryPath);\n const resolvedPath = resolve(fileDir, result);\n\n return resolvedPath;\n }\n\n // Handle absolute paths\n if (isAbsolute(result)) {\n const normalizedResult = normalize(result);\n const normalizedBaseDir = normalize(baseDir);\n\n // Check if it's relative to baseDir (starts with /)\n // but not a system path (like /usr/local)\n if (\n result.startsWith('/') &&\n !normalizedResult.startsWith(normalizedBaseDir)\n ) {\n // Try to resolve it relative to baseDir first\n const relativeToBase = resolve(baseDir, result.substring(1));\n\n // If the path doesn't exist in common system directories, treat as relative to baseDir\n if (\n !result.startsWith('/usr/') &&\n !result.startsWith('/etc/') &&\n !result.startsWith('/var/') &&\n !result.startsWith('/home/') &&\n !result.startsWith('/Users/')\n ) {\n return relativeToBase;\n }\n }\n\n // It's a true system absolute path\n return normalizedResult;\n }\n\n // Default case: treat as relative to baseDir\n return normalize(result);\n};\n"],"mappings":"8FAGA,MAAa,GACX,EACA,EACA,EACA,EACA,IACW,CAEX,GAAI,CAAC,GAAiB,OAAO,GAAkB,SAC7C,MAAU,MAAM,2CAA2C,CAE7D,GAAI,CAAC,GAAiB,OAAO,GAAkB,SAC7C,MAAU,MAAM,2CAA2C,CAE7D,GAAI,CAAC,GAAsB,OAAO,GAAuB,SACvD,MAAU,MAAM,gDAAgD,CAElE,GAAI,CAAC,GAAW,OAAO,GAAY,SACjC,MAAU,MAAM,qCAAqC,CAIvD,IAAM,EAAmB,EAAS,EAAmB,CAClD,MAAM,IAAI,CACV,MAAM,EAAG,GAAG,CACZ,KAAK,IAAI,CAGR,EAAiB,EAClB,QAAQ,eAAgB,EAAc,CACtC,QAAQ,oBAAqB,EAAiB,CAE7C,IACF,EAAS,EAAO,QAAQ,kBAAmB,EAAO,EAIpD,IAAM,EAAyB,EAAW,EAAmB,CACzD,EACA,EAAQ,EAAS,EAAmB,CAGxC,GAAI,EAAO,WAAW,KAAK,EAAI,EAAO,WAAW,MAAM,CAIrD,OAFqB,EADL,EAAQ,EAAuB,CACT,EAAO,CAM/C,GAAI,EAAW,EAAO,CAAE,CACtB,IAAM,EAAmB,EAAU,EAAO,CACpC,EAAoB,EAAU,EAAQ,CAI5C,GACE,EAAO,WAAW,IAAI,EACtB,CAAC,EAAiB,WAAW,EAAkB,CAC/C,CAEA,IAAM,EAAiB,EAAQ,EAAS,EAAO,UAAU,EAAE,CAAC,CAG5D,GACE,CAAC,EAAO,WAAW,QAAQ,EAC3B,CAAC,EAAO,WAAW,QAAQ,EAC3B,CAAC,EAAO,WAAW,QAAQ,EAC3B,CAAC,EAAO,WAAW,SAAS,EAC5B,CAAC,EAAO,WAAW,UAAU,CAE7B,OAAO,EAKX,OAAO,EAIT,OAAO,EAAU,EAAO"}
1
+ {"version":3,"file":"formatAutoFilledFilePath.mjs","names":[],"sources":["../../../src/fill/formatAutoFilledFilePath.ts"],"sourcesContent":["import { basename, dirname, isAbsolute, normalize, resolve } from 'node:path';\nimport { parseFilePathPattern } from '@intlayer/config/utils';\nimport type { Locale } from '@intlayer/types/allLocales';\nimport type { FilePathPattern } from '@intlayer/types/filePathPattern';\n\nexport const formatAutoFilledFilePath = async (\n autoFillField: FilePathPattern,\n dictionaryKey: string,\n dictionaryFilePath: string,\n baseDir: string,\n locale?: Locale\n): Promise<string> => {\n // Validate inputs\n if (!autoFillField) {\n throw new Error('autoFillField must be provided');\n }\n if (!dictionaryKey || typeof dictionaryKey !== 'string') {\n throw new Error('dictionaryKey must be a non-empty string');\n }\n if (!dictionaryFilePath || typeof dictionaryFilePath !== 'string') {\n throw new Error('dictionaryFilePath must be a non-empty string');\n }\n if (!baseDir || typeof baseDir !== 'string') {\n throw new Error('baseDir must be a non-empty string');\n }\n\n // Extract the original filename without extensions (.content.ts -> dictionaryFieldEditor)\n const originalFileName = basename(dictionaryFilePath)\n .split('.')\n .slice(0, -2) // Remove last 2 extensions (.content.tsx)\n .join('.');\n\n // Replace placeholders in autoFillField\n const result: string = await parseFilePathPattern(autoFillField, {\n key: dictionaryKey,\n fileName: originalFileName,\n locale,\n });\n\n // Normalize the dictionary file path - if it's relative, make it absolute relative to baseDir\n const absoluteDictionaryPath = isAbsolute(dictionaryFilePath)\n ? dictionaryFilePath\n : resolve(baseDir, dictionaryFilePath);\n\n // Handle relative paths (starting with ./ or ../)\n if (result.startsWith('./') || result.startsWith('../')) {\n const fileDir = dirname(absoluteDictionaryPath);\n const resolvedPath = resolve(fileDir, result);\n\n return resolvedPath;\n }\n\n // Handle absolute paths\n if (isAbsolute(result)) {\n const normalizedResult = normalize(result);\n const normalizedBaseDir = normalize(baseDir);\n\n // Check if it's relative to baseDir (starts with /)\n // but not a system path (like /usr/local)\n if (\n result.startsWith('/') &&\n !normalizedResult.startsWith(normalizedBaseDir)\n ) {\n // Try to resolve it relative to baseDir first\n const relativeToBase = resolve(baseDir, result.substring(1));\n\n // If the path doesn't exist in common system directories, treat as relative to baseDir\n if (\n !result.startsWith('/usr/') &&\n !result.startsWith('/etc/') &&\n !result.startsWith('/var/') &&\n !result.startsWith('/home/') &&\n !result.startsWith('/Users/')\n ) {\n return relativeToBase;\n }\n }\n\n // It's a true system absolute path\n return normalizedResult;\n }\n\n // Default case: treat as relative to baseDir\n return normalize(result);\n};\n"],"mappings":"4JAKA,MAAa,EAA2B,MACtC,EACA,EACA,EACA,EACA,IACoB,CAEpB,GAAI,CAAC,EACH,MAAU,MAAM,iCAAiC,CAEnD,GAAI,CAAC,GAAiB,OAAO,GAAkB,SAC7C,MAAU,MAAM,2CAA2C,CAE7D,GAAI,CAAC,GAAsB,OAAO,GAAuB,SACvD,MAAU,MAAM,gDAAgD,CAElE,GAAI,CAAC,GAAW,OAAO,GAAY,SACjC,MAAU,MAAM,qCAAqC,CAUvD,IAAM,EAAiB,MAAM,EAAqB,EAAe,CAC/D,IAAK,EACL,SARuB,EAAS,EAAmB,CAClD,MAAM,IAAI,CACV,MAAM,EAAG,GAAG,CACZ,KAAK,IAAI,CAMV,SACD,CAAC,CAGI,EAAyB,EAAW,EAAmB,CACzD,EACA,EAAQ,EAAS,EAAmB,CAGxC,GAAI,EAAO,WAAW,KAAK,EAAI,EAAO,WAAW,MAAM,CAIrD,OAFqB,EADL,EAAQ,EAAuB,CACT,EAAO,CAM/C,GAAI,EAAW,EAAO,CAAE,CACtB,IAAM,EAAmB,EAAU,EAAO,CACpC,EAAoB,EAAU,EAAQ,CAI5C,GACE,EAAO,WAAW,IAAI,EACtB,CAAC,EAAiB,WAAW,EAAkB,CAC/C,CAEA,IAAM,EAAiB,EAAQ,EAAS,EAAO,UAAU,EAAE,CAAC,CAG5D,GACE,CAAC,EAAO,WAAW,QAAQ,EAC3B,CAAC,EAAO,WAAW,QAAQ,EAC3B,CAAC,EAAO,WAAW,QAAQ,EAC3B,CAAC,EAAO,WAAW,SAAS,EAC5B,CAAC,EAAO,WAAW,UAAU,CAE7B,OAAO,EAKX,OAAO,EAIT,OAAO,EAAU,EAAO"}
@@ -1,2 +1,2 @@
1
- import{formatAutoFilledFilePath as e}from"./formatAutoFilledFilePath.mjs";const t=(t,n,r,i,a)=>{let o=[],s=a.content.baseDir;if(!t)return o;if(typeof t==`string`){if(t.includes(`{{locale}}`)){let a=n.map(n=>{let a=e(t,i,r,s,n);return{localeList:[n],filePath:a,isPerLocale:!0}});o.push(...a)}else{let a=e(t,i,r,s);o.push({localeList:n,filePath:a,isPerLocale:!1})}return o}if(typeof t==`object`){let n=Object.keys(t).filter(e=>typeof t[e]==`string`).filter(e=>!!t[e]).map(n=>{let a=e(t[n],i,r,s,n);return{localeList:[n],filePath:a,isPerLocale:!0}}).reduce((e,t)=>{let n=e.find(e=>e.filePath===t.filePath);return n?n.localeList.push(...t.localeList):e.push(t),e},[]);o.push(...n)}return o};export{t as formatFillData};
1
+ import{basename as e,dirname as t,extname as n,relative as r}from"node:path";import{getFormatFromExtension as i,resolveRelativePath as a}from"@intlayer/chokidar/utils";import{parseStringPattern as o}from"@intlayer/config/utils";const s=async(s,c,l,u,d)=>{if(!s||typeof s==`boolean`)return[];let{baseDir:f}=d.system,{defaultLocale:p}=d.internationalization,m=n(l),h=e(l),g=h.includes(`.content.`)?h.split(`.content.`)[0]:h.split(`.`)[0],_=g.charAt(0).toLowerCase()+g.slice(1),v=i(m),y=(e,a)=>{let o=`json`;if(typeof a==`string`){let e=i(n(a));e&&(o=e)}return{key:u,componentDirPath:r(f,t(l)),componentFileName:g,fileName:_,componentFormat:v,componentExtension:m,format:o,locale:e,extension:d.content.fileExtensions[0]}},b=async(e,t)=>{let n=`###########locale###########`,r=!1;if(typeof e==`string`?r=e.includes(`{{locale}}`):typeof e==`function`&&(r=(await e(y(n,e))).includes(n)),r){let n=[];for(let r of t){let t=a(typeof e==`string`?o(e,y(r,e)):await e(y(r,e)),l,f);n.push({filePath:t,localeList:[r],isPerLocale:!0})}return n.reduce((e,t)=>{let n=e.find(e=>e.filePath===t.filePath);return n?(n.localeList.includes(t.localeList[0])||n.localeList.push(...t.localeList),n.isPerLocale=!1):e.push(t),e},[])}else return[{filePath:a(typeof e==`string`?o(e,y(p,e)):await e(y(p,e)),l,f),localeList:t,isPerLocale:!1}]};if(typeof s==`object`&&s){let e=[];for(let t of c){let n=s[t];if(n&&typeof n!=`boolean`){let r=await b(n,[t]);e.push(...r)}}return e.reduce((e,t)=>{let n=e.find(e=>e.filePath===t.filePath);if(n){for(let e of t.localeList)n.localeList.includes(e)||n.localeList.push(e);n.isPerLocale=!1}else e.push(t);return e},[])}return typeof s==`string`||typeof s==`function`?b(s,c):[]};export{s as formatFillData};
2
2
  //# sourceMappingURL=formatFillData.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"formatFillData.mjs","names":[],"sources":["../../../src/fill/formatFillData.ts"],"sourcesContent":["import type { Fill, IntlayerConfig, LocalesValues } from '@intlayer/types';\nimport { formatAutoFilledFilePath } from './formatAutoFilledFilePath';\n\nexport type FillData = {\n localeList: LocalesValues[];\n filePath: string;\n isPerLocale: boolean;\n};\n\nexport const formatFillData = (\n fillField: Fill,\n localeList: LocalesValues[],\n filePath: string,\n dictionaryKey: string,\n configuration: IntlayerConfig\n): FillData[] => {\n const outputContentDeclarationFile: FillData[] = [];\n\n const baseDir = configuration.content.baseDir;\n\n if (!fillField) return outputContentDeclarationFile;\n\n if (typeof fillField === 'string') {\n if (fillField.includes('{{locale}}')) {\n const output = localeList.map((locale) => {\n const formattedFilePath = formatAutoFilledFilePath(\n fillField,\n dictionaryKey,\n filePath,\n baseDir,\n locale\n );\n\n return {\n localeList: [locale],\n filePath: formattedFilePath,\n isPerLocale: true,\n };\n });\n\n outputContentDeclarationFile.push(...output);\n } else {\n const formattedFilePath = formatAutoFilledFilePath(\n fillField,\n dictionaryKey,\n filePath,\n baseDir\n );\n\n outputContentDeclarationFile.push({\n localeList,\n filePath: formattedFilePath,\n isPerLocale: false,\n });\n }\n\n return outputContentDeclarationFile;\n }\n\n if (typeof fillField === 'object') {\n const localeList = Object.keys(fillField).filter(\n (locale) =>\n typeof fillField[locale as keyof typeof fillField] === 'string'\n ) as LocalesValues[];\n\n const output: FillData[] = localeList\n .filter((locale) => Boolean(fillField[locale as keyof typeof fillField]))\n .map((locale) => {\n const formattedFilePath = formatAutoFilledFilePath(\n fillField[locale as keyof typeof fillField] as string,\n dictionaryKey,\n filePath,\n baseDir,\n locale\n );\n\n return {\n localeList: [locale],\n filePath: formattedFilePath,\n isPerLocale: true,\n };\n });\n\n // Group by filePath and merge localeList\n const groupedByFilePath = output.reduce((acc, curr) => {\n const existing = acc.find((item) => item.filePath === curr.filePath);\n if (existing) {\n existing.localeList.push(...curr.localeList);\n } else {\n acc.push(curr);\n }\n return acc;\n }, [] as FillData[]);\n\n outputContentDeclarationFile.push(...groupedByFilePath);\n }\n\n return outputContentDeclarationFile;\n};\n"],"mappings":"0EASA,MAAa,GACX,EACA,EACA,EACA,EACA,IACe,CACf,IAAM,EAA2C,EAAE,CAE7C,EAAU,EAAc,QAAQ,QAEtC,GAAI,CAAC,EAAW,OAAO,EAEvB,GAAI,OAAO,GAAc,SAAU,CACjC,GAAI,EAAU,SAAS,aAAa,CAAE,CACpC,IAAM,EAAS,EAAW,IAAK,GAAW,CACxC,IAAM,EAAoB,EACxB,EACA,EACA,EACA,EACA,EACD,CAED,MAAO,CACL,WAAY,CAAC,EAAO,CACpB,SAAU,EACV,YAAa,GACd,EACD,CAEF,EAA6B,KAAK,GAAG,EAAO,KACvC,CACL,IAAM,EAAoB,EACxB,EACA,EACA,EACA,EACD,CAED,EAA6B,KAAK,CAChC,aACA,SAAU,EACV,YAAa,GACd,CAAC,CAGJ,OAAO,EAGT,GAAI,OAAO,GAAc,SAAU,CAyBjC,IAAM,EAxBa,OAAO,KAAK,EAAU,CAAC,OACvC,GACC,OAAO,EAAU,IAAsC,SAC1D,CAGE,OAAQ,GAAW,EAAQ,EAAU,GAAmC,CACxE,IAAK,GAAW,CACf,IAAM,EAAoB,EACxB,EAAU,GACV,EACA,EACA,EACA,EACD,CAED,MAAO,CACL,WAAY,CAAC,EAAO,CACpB,SAAU,EACV,YAAa,GACd,EACD,CAG6B,QAAQ,EAAK,IAAS,CACrD,IAAM,EAAW,EAAI,KAAM,GAAS,EAAK,WAAa,EAAK,SAAS,CAMpE,OALI,EACF,EAAS,WAAW,KAAK,GAAG,EAAK,WAAW,CAE5C,EAAI,KAAK,EAAK,CAET,GACN,EAAE,CAAe,CAEpB,EAA6B,KAAK,GAAG,EAAkB,CAGzD,OAAO"}
1
+ {"version":3,"file":"formatFillData.mjs","names":[],"sources":["../../../src/fill/formatFillData.ts"],"sourcesContent":["import { basename, dirname, extname, relative } from 'node:path';\nimport {\n getFormatFromExtension,\n resolveRelativePath,\n} from '@intlayer/chokidar/utils';\nimport { parseStringPattern } from '@intlayer/config/utils';\nimport type { Locale } from '@intlayer/types/allLocales';\nimport type { IntlayerConfig } from '@intlayer/types/config';\nimport type { DictionaryKey, Fill } from '@intlayer/types/dictionary';\nimport type {\n FilePathPattern,\n FilePathPatternContext,\n} from '@intlayer/types/filePathPattern';\n\nexport type FillData = {\n localeList: Locale[];\n filePath: string;\n isPerLocale: boolean;\n};\n\nexport const formatFillData = async (\n fillField: Fill,\n localeList: Locale[],\n filePath: string,\n dictionaryKey: DictionaryKey,\n configuration: IntlayerConfig\n): Promise<FillData[]> => {\n if (!fillField || typeof fillField === 'boolean') return [];\n\n const { baseDir } = configuration.system;\n const { defaultLocale } = configuration.internationalization;\n\n const extension = extname(filePath);\n const base = basename(filePath);\n\n // Extract the original filename without extensions\n const cleanComponentFileName = base.includes('.content.')\n ? base.split('.content.')[0]\n : base.split('.')[0];\n\n const uncapitalizedName =\n cleanComponentFileName.charAt(0).toLowerCase() +\n cleanComponentFileName.slice(1);\n\n const componentFormat = getFormatFromExtension(extension as any);\n\n const getContext = (\n locale: Locale,\n patternType?: FilePathPattern\n ): FilePathPatternContext => {\n let format: FilePathPatternContext['format'] = 'json';\n if (typeof patternType === 'string') {\n const extFormat = getFormatFromExtension(extname(patternType) as any);\n if (extFormat) {\n format = extFormat as any;\n }\n }\n\n return {\n key: dictionaryKey,\n componentDirPath: relative(baseDir, dirname(filePath)),\n componentFileName: cleanComponentFileName,\n fileName: uncapitalizedName,\n componentFormat:\n componentFormat as FilePathPatternContext['componentFormat'],\n componentExtension:\n extension as FilePathPatternContext['componentExtension'],\n format,\n locale,\n extension: configuration.content.fileExtensions[0] as any,\n };\n };\n\n const processPattern = async (\n pattern: FilePathPattern,\n locales: Locale[]\n ): Promise<FillData[]> => {\n const dummyLocale = '###########locale###########' as Locale;\n let isPatternPerLocale = false;\n\n if (typeof pattern === 'string') {\n isPatternPerLocale = pattern.includes('{{locale}}');\n } else if (typeof pattern === 'function') {\n const pathWithDummy = await pattern(getContext(dummyLocale, pattern));\n isPatternPerLocale = pathWithDummy.includes(dummyLocale);\n }\n\n if (isPatternPerLocale) {\n const resolvedPaths: FillData[] = [];\n\n // Generate one path per locale\n for (const locale of locales) {\n const rawPath =\n typeof pattern === 'string'\n ? parseStringPattern(pattern, getContext(locale, pattern))\n : await pattern(getContext(locale, pattern));\n\n const absolutePath = resolveRelativePath(rawPath, filePath, baseDir);\n\n resolvedPaths.push({\n filePath: absolutePath,\n localeList: [locale],\n isPerLocale: true,\n });\n }\n\n // Group by filePath in case multiple locales resolve to the same path\n const groupedByFilePath = resolvedPaths.reduce((acc, curr) => {\n const existing = acc.find((item) => item.filePath === curr.filePath);\n if (existing) {\n if (!existing.localeList.includes(curr.localeList[0])) {\n existing.localeList.push(...curr.localeList);\n }\n // If multiple locales share a path, it's no longer strictly per-locale\n existing.isPerLocale = false;\n } else {\n acc.push(curr);\n }\n return acc;\n }, [] as FillData[]);\n\n return groupedByFilePath;\n } else {\n // Single multi-lingual path using default locale for pattern resolution\n const rawPath =\n typeof pattern === 'string'\n ? parseStringPattern(pattern, getContext(defaultLocale, pattern))\n : await pattern(getContext(defaultLocale, pattern));\n\n const absolutePath = resolveRelativePath(rawPath, filePath, baseDir);\n\n return [\n {\n filePath: absolutePath,\n localeList: locales,\n isPerLocale: false,\n },\n ];\n }\n };\n\n // 1. Handle Record of Locales\n if (typeof fillField === 'object' && fillField !== null) {\n const results: FillData[] = [];\n\n for (const locale of localeList) {\n const pattern = (fillField as Record<string, any>)[locale];\n if (pattern && typeof pattern !== 'boolean') {\n const res = await processPattern(pattern as FilePathPattern, [locale]);\n results.push(...res);\n }\n }\n\n // Merge identical file paths if they stem from different locales having the same output path\n const grouped = results.reduce((acc, curr) => {\n const existing = acc.find((item) => item.filePath === curr.filePath);\n if (existing) {\n for (const loc of curr.localeList) {\n if (!existing.localeList.includes(loc)) {\n existing.localeList.push(loc);\n }\n }\n existing.isPerLocale = false;\n } else {\n acc.push(curr);\n }\n return acc;\n }, [] as FillData[]);\n\n return grouped;\n }\n\n // 2. Handle static string or function patterns\n if (typeof fillField === 'string' || typeof fillField === 'function') {\n return processPattern(fillField as FilePathPattern, localeList);\n }\n\n return [];\n};\n"],"mappings":"oOAoBA,MAAa,EAAiB,MAC5B,EACA,EACA,EACA,EACA,IACwB,CACxB,GAAI,CAAC,GAAa,OAAO,GAAc,UAAW,MAAO,EAAE,CAE3D,GAAM,CAAE,WAAY,EAAc,OAC5B,CAAE,iBAAkB,EAAc,qBAElC,EAAY,EAAQ,EAAS,CAC7B,EAAO,EAAS,EAAS,CAGzB,EAAyB,EAAK,SAAS,YAAY,CACrD,EAAK,MAAM,YAAY,CAAC,GACxB,EAAK,MAAM,IAAI,CAAC,GAEd,EACJ,EAAuB,OAAO,EAAE,CAAC,aAAa,CAC9C,EAAuB,MAAM,EAAE,CAE3B,EAAkB,EAAuB,EAAiB,CAE1D,GACJ,EACA,IAC2B,CAC3B,IAAI,EAA2C,OAC/C,GAAI,OAAO,GAAgB,SAAU,CACnC,IAAM,EAAY,EAAuB,EAAQ,EAAY,CAAQ,CACjE,IACF,EAAS,GAIb,MAAO,CACL,IAAK,EACL,iBAAkB,EAAS,EAAS,EAAQ,EAAS,CAAC,CACtD,kBAAmB,EACnB,SAAU,EAER,kBACF,mBACE,EACF,SACA,SACA,UAAW,EAAc,QAAQ,eAAe,GACjD,EAGG,EAAiB,MACrB,EACA,IACwB,CACxB,IAAM,EAAc,+BAChB,EAAqB,GASzB,GAPI,OAAO,GAAY,SACrB,EAAqB,EAAQ,SAAS,aAAa,CAC1C,OAAO,GAAY,aAE5B,GADsB,MAAM,EAAQ,EAAW,EAAa,EAAQ,CAAC,EAClC,SAAS,EAAY,EAGtD,EAAoB,CACtB,IAAM,EAA4B,EAAE,CAGpC,IAAK,IAAM,KAAU,EAAS,CAM5B,IAAM,EAAe,EAJnB,OAAO,GAAY,SACf,EAAmB,EAAS,EAAW,EAAQ,EAAQ,CAAC,CACxD,MAAM,EAAQ,EAAW,EAAQ,EAAQ,CAAC,CAEE,EAAU,EAAQ,CAEpE,EAAc,KAAK,CACjB,SAAU,EACV,WAAY,CAAC,EAAO,CACpB,YAAa,GACd,CAAC,CAkBJ,OAd0B,EAAc,QAAQ,EAAK,IAAS,CAC5D,IAAM,EAAW,EAAI,KAAM,GAAS,EAAK,WAAa,EAAK,SAAS,CAUpE,OATI,GACG,EAAS,WAAW,SAAS,EAAK,WAAW,GAAG,EACnD,EAAS,WAAW,KAAK,GAAG,EAAK,WAAW,CAG9C,EAAS,YAAc,IAEvB,EAAI,KAAK,EAAK,CAET,GACN,EAAE,CAAe,MAYpB,MAAO,CACL,CACE,SAJiB,EAJnB,OAAO,GAAY,SACf,EAAmB,EAAS,EAAW,EAAe,EAAQ,CAAC,CAC/D,MAAM,EAAQ,EAAW,EAAe,EAAQ,CAAC,CAEL,EAAU,EAAQ,CAKhE,WAAY,EACZ,YAAa,GACd,CACF,EAKL,GAAI,OAAO,GAAc,UAAY,EAAoB,CACvD,IAAM,EAAsB,EAAE,CAE9B,IAAK,IAAM,KAAU,EAAY,CAC/B,IAAM,EAAW,EAAkC,GACnD,GAAI,GAAW,OAAO,GAAY,UAAW,CAC3C,IAAM,EAAM,MAAM,EAAe,EAA4B,CAAC,EAAO,CAAC,CACtE,EAAQ,KAAK,GAAG,EAAI,EAoBxB,OAfgB,EAAQ,QAAQ,EAAK,IAAS,CAC5C,IAAM,EAAW,EAAI,KAAM,GAAS,EAAK,WAAa,EAAK,SAAS,CACpE,GAAI,EAAU,CACZ,IAAK,IAAM,KAAO,EAAK,WAChB,EAAS,WAAW,SAAS,EAAI,EACpC,EAAS,WAAW,KAAK,EAAI,CAGjC,EAAS,YAAc,QAEvB,EAAI,KAAK,EAAK,CAEhB,OAAO,GACN,EAAE,CAAe,CAUtB,OAJI,OAAO,GAAc,UAAY,OAAO,GAAc,WACjD,EAAe,EAA8B,EAAW,CAG1D,EAAE"}
@@ -1 +1 @@
1
- {"version":3,"file":"getAvailableLocalesInDictionary.mjs","names":[],"sources":["../../../src/fill/getAvailableLocalesInDictionary.ts"],"sourcesContent":["import type { Dictionary, Locale } from '@intlayer/types';\n\n/**\n * Recursively traverses dictionary content to find all locales that have actual translations.\n * Returns a Set of locale strings that are present in the translation nodes.\n */\nconst extractLocalesFromContent = (\n content: any,\n locales: Set<Locale> = new Set()\n): Set<Locale> => {\n if (!content || typeof content !== 'object') {\n return locales;\n }\n\n // Check if this is a translation node\n if (content.nodeType === 'translation' && content.translation) {\n // Add all locale keys from the translation map\n Object.keys(content.translation).forEach((locale) => {\n locales.add(locale as Locale);\n });\n }\n\n // Recursively check nested objects\n for (const value of Object.values(content)) {\n if (value && typeof value === 'object') {\n extractLocalesFromContent(value, locales);\n }\n }\n\n return locales;\n};\n\n/**\n * Gets all locales that have actual translations in the dictionary content.\n * Only returns locales that are present in at least one translation node.\n */\nexport const getAvailableLocalesInDictionary = (\n dictionary: Dictionary\n): Locale[] => {\n const localesSet = extractLocalesFromContent(dictionary.content);\n return Array.from(localesSet);\n};\n"],"mappings":"AAMA,MAAM,GACJ,EACA,EAAuB,IAAI,MACX,CAChB,GAAI,CAAC,GAAW,OAAO,GAAY,SACjC,OAAO,EAIL,EAAQ,WAAa,eAAiB,EAAQ,aAEhD,OAAO,KAAK,EAAQ,YAAY,CAAC,QAAS,GAAW,CACnD,EAAQ,IAAI,EAAiB,EAC7B,CAIJ,IAAK,IAAM,KAAS,OAAO,OAAO,EAAQ,CACpC,GAAS,OAAO,GAAU,UAC5B,EAA0B,EAAO,EAAQ,CAI7C,OAAO,GAOI,EACX,GACa,CACb,IAAM,EAAa,EAA0B,EAAW,QAAQ,CAChE,OAAO,MAAM,KAAK,EAAW"}
1
+ {"version":3,"file":"getAvailableLocalesInDictionary.mjs","names":[],"sources":["../../../src/fill/getAvailableLocalesInDictionary.ts"],"sourcesContent":["import type { Dictionary } from '@intlayer/types/dictionary';\nimport type { Locale } from '@intlayer/types/allLocales';\n\n/**\n * Recursively traverses dictionary content to find all locales that have actual translations.\n * Returns a Set of locale strings that are present in the translation nodes.\n */\nconst extractLocalesFromContent = (\n content: any,\n locales: Set<Locale> = new Set()\n): Set<Locale> => {\n if (!content || typeof content !== 'object') {\n return locales;\n }\n\n // Check if this is a translation node\n if (content.nodeType === 'translation' && content.translation) {\n // Add all locale keys from the translation map\n Object.keys(content.translation).forEach((locale) => {\n locales.add(locale as Locale);\n });\n }\n\n // Recursively check nested objects\n for (const value of Object.values(content)) {\n if (value && typeof value === 'object') {\n extractLocalesFromContent(value, locales);\n }\n }\n\n return locales;\n};\n\n/**\n * Gets all locales that have actual translations in the dictionary content.\n * Only returns locales that are present in at least one translation node.\n */\nexport const getAvailableLocalesInDictionary = (\n dictionary: Dictionary\n): Locale[] => {\n const localesSet = extractLocalesFromContent(dictionary.content);\n return Array.from(localesSet);\n};\n"],"mappings":"AAOA,MAAM,GACJ,EACA,EAAuB,IAAI,MACX,CAChB,GAAI,CAAC,GAAW,OAAO,GAAY,SACjC,OAAO,EAIL,EAAQ,WAAa,eAAiB,EAAQ,aAEhD,OAAO,KAAK,EAAQ,YAAY,CAAC,QAAS,GAAW,CACnD,EAAQ,IAAI,EAAiB,EAC7B,CAIJ,IAAK,IAAM,KAAS,OAAO,OAAO,EAAQ,CACpC,GAAS,OAAO,GAAU,UAC5B,EAA0B,EAAO,EAAQ,CAI7C,OAAO,GAOI,EACX,GACa,CACb,IAAM,EAAa,EAA0B,EAAW,QAAQ,CAChE,OAAO,MAAM,KAAK,EAAW"}
@@ -1 +1 @@
1
- {"version":3,"file":"getFilterMissingContentPerLocale.mjs","names":[],"sources":["../../../src/fill/getFilterMissingContentPerLocale.ts"],"sourcesContent":["import type { Dictionary } from '@intlayer/types';\n\n/**\n * Recursively compares source content with target content and returns only the missing parts.\n * For per-locale files where content is simple JSON (not translation nodes).\n *\n * @param sourceContent - The source content to check\n * @param targetContent - The existing target content\n * @returns Only the content that's missing in the target\n */\nconst filterMissingContent = (sourceContent: any, targetContent: any): any => {\n // Source is null - only missing if target doesn't already have null\n if (sourceContent === null) {\n return targetContent === null ? undefined : null;\n }\n\n // If target doesn't exist or is null, all source content is missing\n if (targetContent === undefined || targetContent === null) {\n return sourceContent;\n }\n\n // Primitive values: if target exists (even if empty string), consider it translated\n if (typeof sourceContent !== 'object') {\n return undefined;\n }\n\n // Handle arrays\n if (Array.isArray(sourceContent)) {\n if (!Array.isArray(targetContent)) {\n return sourceContent;\n }\n\n const missingItems = sourceContent\n .map((item, index) => filterMissingContent(item, targetContent[index]))\n .filter((item) => item !== undefined);\n\n return missingItems.length > 0 ? missingItems : undefined;\n }\n\n // Handle objects\n const result: any = {};\n let hasMissingContent = false;\n\n for (const [key, value] of Object.entries(sourceContent)) {\n const targetValue = targetContent?.[key];\n const missingValue = filterMissingContent(value, targetValue);\n\n if (missingValue !== undefined) {\n result[key] = missingValue;\n hasMissingContent = true;\n }\n }\n\n return hasMissingContent ? result : undefined;\n};\n\n/**\n * Filters a dictionary to only include content that's missing in the target dictionary.\n * Used for per-locale content declarations in 'complete' mode.\n *\n * @param sourceDictionary - The source dictionary with content to translate\n * @param targetDictionary - The existing target dictionary\n * @returns A dictionary with only the missing content\n */\nexport const getFilterMissingContentPerLocale = (\n sourceDictionary: Dictionary,\n targetDictionary: Dictionary | undefined\n): Dictionary => {\n if (!targetDictionary || !targetDictionary.content) {\n // If no target exists, all source content is missing\n return sourceDictionary;\n }\n\n const missingContent = filterMissingContent(\n sourceDictionary.content,\n targetDictionary.content\n );\n\n return {\n ...sourceDictionary,\n content: missingContent ?? {},\n };\n};\n"],"mappings":"AAUA,MAAM,GAAwB,EAAoB,IAA4B,CAE5E,GAAI,IAAkB,KACpB,OAAO,IAAkB,KAAO,IAAA,GAAY,KAI9C,GAAI,GAAiD,KACnD,OAAO,EAIT,GAAI,OAAO,GAAkB,SAC3B,OAIF,GAAI,MAAM,QAAQ,EAAc,CAAE,CAChC,GAAI,CAAC,MAAM,QAAQ,EAAc,CAC/B,OAAO,EAGT,IAAM,EAAe,EAClB,KAAK,EAAM,IAAU,EAAqB,EAAM,EAAc,GAAO,CAAC,CACtE,OAAQ,GAAS,IAAS,IAAA,GAAU,CAEvC,OAAO,EAAa,OAAS,EAAI,EAAe,IAAA,GAIlD,IAAM,EAAc,EAAE,CAClB,EAAoB,GAExB,IAAK,GAAM,CAAC,EAAK,KAAU,OAAO,QAAQ,EAAc,CAAE,CACxD,IAAM,EAAc,IAAgB,GAC9B,EAAe,EAAqB,EAAO,EAAY,CAEzD,IAAiB,IAAA,KACnB,EAAO,GAAO,EACd,EAAoB,IAIxB,OAAO,EAAoB,EAAS,IAAA,IAWzB,GACX,EACA,IACe,CACf,GAAI,CAAC,GAAoB,CAAC,EAAiB,QAEzC,OAAO,EAGT,IAAM,EAAiB,EACrB,EAAiB,QACjB,EAAiB,QAClB,CAED,MAAO,CACL,GAAG,EACH,QAAS,GAAkB,EAAE,CAC9B"}
1
+ {"version":3,"file":"getFilterMissingContentPerLocale.mjs","names":[],"sources":["../../../src/fill/getFilterMissingContentPerLocale.ts"],"sourcesContent":["import type { Dictionary } from '@intlayer/types/dictionary';\n\n/**\n * Recursively compares source content with target content and returns only the missing parts.\n * For per-locale files where content is simple JSON (not translation nodes).\n *\n * @param sourceContent - The source content to check\n * @param targetContent - The existing target content\n * @returns Only the content that's missing in the target\n */\nconst filterMissingContent = (sourceContent: any, targetContent: any): any => {\n // Source is null - only missing if target doesn't already have null\n if (sourceContent === null) {\n return targetContent === null ? undefined : null;\n }\n\n // If target doesn't exist or is null, all source content is missing\n if (targetContent === undefined || targetContent === null) {\n return sourceContent;\n }\n\n // Primitive values: if target exists (even if empty string), consider it translated\n if (typeof sourceContent !== 'object') {\n return undefined;\n }\n\n // Handle arrays\n if (Array.isArray(sourceContent)) {\n if (!Array.isArray(targetContent)) {\n return sourceContent;\n }\n\n const missingItems = sourceContent\n .map((item, index) => filterMissingContent(item, targetContent[index]))\n .filter((item) => item !== undefined);\n\n return missingItems.length > 0 ? missingItems : undefined;\n }\n\n // Handle objects\n const result: any = {};\n let hasMissingContent = false;\n\n for (const [key, value] of Object.entries(sourceContent)) {\n const targetValue = targetContent?.[key];\n const missingValue = filterMissingContent(value, targetValue);\n\n if (missingValue !== undefined) {\n result[key] = missingValue;\n hasMissingContent = true;\n }\n }\n\n return hasMissingContent ? result : undefined;\n};\n\n/**\n * Filters a dictionary to only include content that's missing in the target dictionary.\n * Used for per-locale content declarations in 'complete' mode.\n *\n * @param sourceDictionary - The source dictionary with content to translate\n * @param targetDictionary - The existing target dictionary\n * @returns A dictionary with only the missing content\n */\nexport const getFilterMissingContentPerLocale = (\n sourceDictionary: Dictionary,\n targetDictionary: Dictionary | undefined\n): Dictionary => {\n if (!targetDictionary || !targetDictionary.content) {\n // If no target exists, all source content is missing\n return sourceDictionary;\n }\n\n const missingContent = filterMissingContent(\n sourceDictionary.content,\n targetDictionary.content\n );\n\n return {\n ...sourceDictionary,\n content: missingContent ?? {},\n };\n};\n"],"mappings":"AAUA,MAAM,GAAwB,EAAoB,IAA4B,CAE5E,GAAI,IAAkB,KACpB,OAAO,IAAkB,KAAO,IAAA,GAAY,KAI9C,GAAI,GAAiD,KACnD,OAAO,EAIT,GAAI,OAAO,GAAkB,SAC3B,OAIF,GAAI,MAAM,QAAQ,EAAc,CAAE,CAChC,GAAI,CAAC,MAAM,QAAQ,EAAc,CAC/B,OAAO,EAGT,IAAM,EAAe,EAClB,KAAK,EAAM,IAAU,EAAqB,EAAM,EAAc,GAAO,CAAC,CACtE,OAAQ,GAAS,IAAS,IAAA,GAAU,CAEvC,OAAO,EAAa,OAAS,EAAI,EAAe,IAAA,GAIlD,IAAM,EAAc,EAAE,CAClB,EAAoB,GAExB,IAAK,GAAM,CAAC,EAAK,KAAU,OAAO,QAAQ,EAAc,CAAE,CACxD,IAAM,EAAc,IAAgB,GAC9B,EAAe,EAAqB,EAAO,EAAY,CAEzD,IAAiB,IAAA,KACnB,EAAO,GAAO,EACd,EAAoB,IAIxB,OAAO,EAAoB,EAAS,IAAA,IAWzB,GACX,EACA,IACe,CACf,GAAI,CAAC,GAAoB,CAAC,EAAiB,QAEzC,OAAO,EAGT,IAAM,EAAiB,EACrB,EAAiB,QACjB,EAAiB,QAClB,CAED,MAAO,CACL,GAAG,EACH,QAAS,GAAkB,EAAE,CAC9B"}
@@ -1,2 +1,2 @@
1
- import{listMissingTranslationsWithConfig as e}from"../test/listMissingTranslations.mjs";import{ANSIColors as t,colon as n,colorize as r,colorizeKey as i,colorizePath as a,getAppLogger as o}from"@intlayer/config/logger";import{formatLocale as s}from"@intlayer/chokidar/utils";import{basename as c}from"node:path";import{getUnmergedDictionaries as l}from"@intlayer/unmerged-dictionaries-entry";import{getFilterTranslationsOnlyDictionary as u}from"@intlayer/core/plugins";import{getDictionaries as d}from"@intlayer/dictionaries-entry";const f=(f,p,m,h,g)=>{let _=o(g),v=d(g),y=l(g),b=Object.values(y).flat().filter(e=>f.includes(e.localId)),{missingTranslations:x}=e(g),S=Math.max(...b.map(e=>e.key.length)),C=[];for(let e of b){let o=n([` - `,r(`[`,t.GREY_DARK),i(e.key),r(`]`,t.GREY_DARK)].join(``),{colSize:S+6}),l=e.key,d=e.localId,f=v[l];if((e.filled??!1)===!0||(e.fill??g.dictionary?.fill??!1)===!1)continue;let y=e.locale??h;if(!f){_(`${o} Dictionary not found in dictionariesRecord. Skipping.`,{level:`warn`});continue}if(!e.filePath){_(`${o} Dictionary has no file path. Skipping.`,{level:`warn`});continue}let b=u(f,y);if(Object.keys(b).length===0){_(`${o} No content found for dictionary in source locale ${s(y)}. Skipping translation for this dictionary.`,{level:`warn`});continue}let w=p;if(m===`complete`&&(w=x.find(e=>e.key===l)?.locales.filter(e=>p.includes(e))??[]),w.length===0){_(`${o} ${r(`No locales to fill, Skipping`,t.GREY_DARK)} ${a(c(e.filePath))}`,{level:`warn`});continue}C.push({dictionaryKey:l,dictionaryLocalId:d,sourceLocale:y,targetLocales:w,dictionaryPreset:o,dictionaryFilePath:e.filePath})}return C};export{f as listTranslationsTasks};
1
+ import{listMissingTranslationsWithConfig as e}from"../test/listMissingTranslations.mjs";import{basename as t}from"node:path";import{ANSIColors as n,colon as r,colorize as i,colorizeKey as a,colorizePath as o,getAppLogger as s}from"@intlayer/config/logger";import{getUnmergedDictionaries as c}from"@intlayer/unmerged-dictionaries-entry";import{formatLocale as l}from"@intlayer/chokidar/utils";import{getFilterTranslationsOnlyDictionary as u}from"@intlayer/core/plugins";import{getDictionaries as d}from"@intlayer/dictionaries-entry";const f=(f,p,m,h,g)=>{let _=s(g),v=d(g),y=c(g),b=Object.values(y).flat().filter(e=>f.includes(e.localId)),{missingTranslations:x}=e(g),S=Math.max(...b.map(e=>e.key.length)),C=[];for(let e of b){let s=r([` - `,i(`[`,n.GREY_DARK),a(e.key),i(`]`,n.GREY_DARK)].join(``),{colSize:S+6}),c=e.key,d=e.localId,f=v[c];if((e.filled??!1)===!0||(e.fill??g.dictionary?.fill??!1)===!1)continue;let y=e.locale??h;if(!f){_(`${s} Dictionary not found in dictionariesRecord. Skipping.`,{level:`warn`});continue}if(!e.filePath){_(`${s} Dictionary has no file path. Skipping.`,{level:`warn`});continue}let b=u(f,y);if(Object.keys(b).length===0){_(`${s} No content found for dictionary in source locale ${l(y)}. Skipping translation for this dictionary.`,{level:`warn`});continue}let w=p;if(m===`complete`&&(w=x.find(e=>e.key===c)?.locales.filter(e=>p.includes(e))??[]),w.length===0){_(`${s} ${i(`No locales to fill, Skipping`,n.GREY_DARK)} ${o(t(e.filePath))}`,{level:`warn`});continue}C.push({dictionaryKey:c,dictionaryLocalId:d,sourceLocale:y,targetLocales:w,dictionaryPreset:s,dictionaryFilePath:e.filePath})}return C};export{f as listTranslationsTasks};
2
2
  //# sourceMappingURL=listTranslationsTasks.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"listTranslationsTasks.mjs","names":[],"sources":["../../../src/fill/listTranslationsTasks.ts"],"sourcesContent":["import { basename } from 'node:path';\nimport { formatLocale } from '@intlayer/chokidar/utils';\nimport {\n ANSIColors,\n colon,\n colorize,\n colorizeKey,\n colorizePath,\n getAppLogger,\n} from '@intlayer/config/logger';\nimport { getFilterTranslationsOnlyDictionary } from '@intlayer/core/plugins';\nimport { getDictionaries } from '@intlayer/dictionaries-entry';\nimport type {\n Dictionary,\n IntlayerConfig,\n LocalDictionaryId,\n Locale,\n} from '@intlayer/types';\nimport { getUnmergedDictionaries } from '@intlayer/unmerged-dictionaries-entry';\nimport { listMissingTranslationsWithConfig } from '../test';\n\nexport type TranslationTask = {\n dictionaryKey: string;\n dictionaryLocalId: LocalDictionaryId;\n sourceLocale: Locale;\n targetLocales: Locale[];\n dictionaryPreset: string;\n dictionaryFilePath: string;\n};\n\nexport const listTranslationsTasks = (\n localIds: LocalDictionaryId[],\n outputLocales: Locale[],\n mode: 'complete' | 'review',\n baseLocale: Locale,\n configuration: IntlayerConfig\n): TranslationTask[] => {\n const appLogger = getAppLogger(configuration);\n\n const mergedDictionariesRecord = getDictionaries(configuration);\n const unmergedDictionariesRecord = getUnmergedDictionaries(configuration);\n\n const allFlatDictionaries = Object.values(unmergedDictionariesRecord).flat();\n const dictionariesToProcess = allFlatDictionaries.filter((dictionary) =>\n localIds.includes(dictionary.localId!)\n );\n\n const { missingTranslations } =\n listMissingTranslationsWithConfig(configuration);\n\n const maxKeyLength = Math.max(\n ...dictionariesToProcess.map((dictionary) => dictionary.key.length)\n );\n\n const translationTasks: TranslationTask[] = [];\n\n for (const targetUnmergedDictionary of dictionariesToProcess) {\n const dictionaryPreset = colon(\n [\n ' - ',\n colorize('[', ANSIColors.GREY_DARK),\n colorizeKey(targetUnmergedDictionary.key),\n colorize(']', ANSIColors.GREY_DARK),\n ].join(''),\n { colSize: maxKeyLength + 6 }\n );\n\n const dictionaryKey = targetUnmergedDictionary.key;\n const dictionaryLocalId = targetUnmergedDictionary.localId!;\n const mainDictionaryToProcess: Dictionary =\n mergedDictionariesRecord[dictionaryKey];\n const dictionaryFilled = targetUnmergedDictionary.filled ?? false;\n\n if (dictionaryFilled === true) {\n continue;\n }\n\n const dictionaryFill =\n targetUnmergedDictionary.fill ?? configuration.dictionary?.fill ?? false;\n\n if (dictionaryFill === false) continue;\n\n const sourceLocale: Locale = (targetUnmergedDictionary.locale ??\n baseLocale) as Locale;\n\n if (!mainDictionaryToProcess) {\n appLogger(\n `${dictionaryPreset} Dictionary not found in dictionariesRecord. Skipping.`,\n {\n level: 'warn',\n }\n );\n continue;\n }\n\n if (!targetUnmergedDictionary.filePath) {\n appLogger(`${dictionaryPreset} Dictionary has no file path. Skipping.`, {\n level: 'warn',\n });\n continue;\n }\n\n const sourceLocaleContent = getFilterTranslationsOnlyDictionary(\n mainDictionaryToProcess,\n sourceLocale\n );\n\n if (\n Object.keys(sourceLocaleContent as Record<string, unknown>).length === 0\n ) {\n appLogger(\n `${dictionaryPreset} No content found for dictionary in source locale ${formatLocale(sourceLocale)}. Skipping translation for this dictionary.`,\n {\n level: 'warn',\n }\n );\n continue;\n }\n\n /**\n * In 'complete' mode, filter only the missing locales to translate\n *\n * Skip the dictionary if there are no missing locales to translate\n */\n let outputLocalesList: Locale[] = outputLocales as Locale[];\n\n if (mode === 'complete') {\n outputLocalesList =\n missingTranslations\n .find(\n (missingTranslation) => missingTranslation.key === dictionaryKey\n )\n ?.locales.filter((locale) => outputLocales.includes(locale)) ?? [];\n }\n\n if (outputLocalesList.length === 0) {\n appLogger(\n `${dictionaryPreset} ${colorize('No locales to fill, Skipping', ANSIColors.GREY_DARK)} ${colorizePath(basename(targetUnmergedDictionary.filePath))}`,\n {\n level: 'warn',\n }\n );\n continue;\n }\n\n translationTasks.push({\n dictionaryKey,\n dictionaryLocalId,\n sourceLocale,\n targetLocales: outputLocalesList,\n dictionaryPreset,\n dictionaryFilePath: targetUnmergedDictionary.filePath,\n });\n }\n\n // Return the list of tasks to execute\n return translationTasks;\n};\n"],"mappings":"ohBA8BA,MAAa,GACX,EACA,EACA,EACA,EACA,IACsB,CACtB,IAAM,EAAY,EAAa,EAAc,CAEvC,EAA2B,EAAgB,EAAc,CACzD,EAA6B,EAAwB,EAAc,CAGnE,EADsB,OAAO,OAAO,EAA2B,CAAC,MAAM,CAC1B,OAAQ,GACxD,EAAS,SAAS,EAAW,QAAS,CACvC,CAEK,CAAE,uBACN,EAAkC,EAAc,CAE5C,EAAe,KAAK,IACxB,GAAG,EAAsB,IAAK,GAAe,EAAW,IAAI,OAAO,CACpE,CAEK,EAAsC,EAAE,CAE9C,IAAK,IAAM,KAA4B,EAAuB,CAC5D,IAAM,EAAmB,EACvB,CACE,MACA,EAAS,IAAK,EAAW,UAAU,CACnC,EAAY,EAAyB,IAAI,CACzC,EAAS,IAAK,EAAW,UAAU,CACpC,CAAC,KAAK,GAAG,CACV,CAAE,QAAS,EAAe,EAAG,CAC9B,CAEK,EAAgB,EAAyB,IACzC,EAAoB,EAAyB,QAC7C,EACJ,EAAyB,GAU3B,IATyB,EAAyB,QAAU,MAEnC,KAKvB,EAAyB,MAAQ,EAAc,YAAY,MAAQ,MAE9C,GAAO,SAE9B,IAAM,EAAwB,EAAyB,QACrD,EAEF,GAAI,CAAC,EAAyB,CAC5B,EACE,GAAG,EAAiB,wDACpB,CACE,MAAO,OACR,CACF,CACD,SAGF,GAAI,CAAC,EAAyB,SAAU,CACtC,EAAU,GAAG,EAAiB,yCAA0C,CACtE,MAAO,OACR,CAAC,CACF,SAGF,IAAM,EAAsB,EAC1B,EACA,EACD,CAED,GACE,OAAO,KAAK,EAA+C,CAAC,SAAW,EACvE,CACA,EACE,GAAG,EAAiB,oDAAoD,EAAa,EAAa,CAAC,6CACnG,CACE,MAAO,OACR,CACF,CACD,SAQF,IAAI,EAA8B,EAWlC,GATI,IAAS,aACX,EACE,EACG,KACE,GAAuB,EAAmB,MAAQ,EACpD,EACC,QAAQ,OAAQ,GAAW,EAAc,SAAS,EAAO,CAAC,EAAI,EAAE,EAGpE,EAAkB,SAAW,EAAG,CAClC,EACE,GAAG,EAAiB,GAAG,EAAS,+BAAgC,EAAW,UAAU,CAAC,GAAG,EAAa,EAAS,EAAyB,SAAS,CAAC,GAClJ,CACE,MAAO,OACR,CACF,CACD,SAGF,EAAiB,KAAK,CACpB,gBACA,oBACA,eACA,cAAe,EACf,mBACA,mBAAoB,EAAyB,SAC9C,CAAC,CAIJ,OAAO"}
1
+ {"version":3,"file":"listTranslationsTasks.mjs","names":[],"sources":["../../../src/fill/listTranslationsTasks.ts"],"sourcesContent":["import { basename } from 'node:path';\nimport { formatLocale } from '@intlayer/chokidar/utils';\nimport {\n ANSIColors,\n colon,\n colorize,\n colorizeKey,\n colorizePath,\n getAppLogger,\n} from '@intlayer/config/logger';\nimport { getFilterTranslationsOnlyDictionary } from '@intlayer/core/plugins';\nimport { getDictionaries } from '@intlayer/dictionaries-entry';\nimport type { Dictionary, LocalDictionaryId } from '@intlayer/types/dictionary';\nimport type { IntlayerConfig } from '@intlayer/types/config';\nimport type { Locale } from '@intlayer/types/allLocales';\nimport { getUnmergedDictionaries } from '@intlayer/unmerged-dictionaries-entry';\nimport { listMissingTranslationsWithConfig } from '../test';\n\nexport type TranslationTask = {\n dictionaryKey: string;\n dictionaryLocalId: LocalDictionaryId;\n sourceLocale: Locale;\n targetLocales: Locale[];\n dictionaryPreset: string;\n dictionaryFilePath: string;\n};\n\nexport const listTranslationsTasks = (\n localIds: LocalDictionaryId[],\n outputLocales: Locale[],\n mode: 'complete' | 'review',\n baseLocale: Locale,\n configuration: IntlayerConfig\n): TranslationTask[] => {\n const appLogger = getAppLogger(configuration);\n\n const mergedDictionariesRecord = getDictionaries(configuration);\n const unmergedDictionariesRecord = getUnmergedDictionaries(configuration);\n\n const allFlatDictionaries = Object.values(unmergedDictionariesRecord).flat();\n const dictionariesToProcess = allFlatDictionaries.filter((dictionary) =>\n localIds.includes(dictionary.localId!)\n );\n\n const { missingTranslations } =\n listMissingTranslationsWithConfig(configuration);\n\n const maxKeyLength = Math.max(\n ...dictionariesToProcess.map((dictionary) => dictionary.key.length)\n );\n\n const translationTasks: TranslationTask[] = [];\n\n for (const targetUnmergedDictionary of dictionariesToProcess) {\n const dictionaryPreset = colon(\n [\n ' - ',\n colorize('[', ANSIColors.GREY_DARK),\n colorizeKey(targetUnmergedDictionary.key),\n colorize(']', ANSIColors.GREY_DARK),\n ].join(''),\n { colSize: maxKeyLength + 6 }\n );\n\n const dictionaryKey = targetUnmergedDictionary.key;\n const dictionaryLocalId = targetUnmergedDictionary.localId!;\n const mainDictionaryToProcess: Dictionary =\n mergedDictionariesRecord[dictionaryKey];\n const dictionaryFilled = targetUnmergedDictionary.filled ?? false;\n\n if (dictionaryFilled === true) {\n continue;\n }\n\n const dictionaryFill =\n targetUnmergedDictionary.fill ?? configuration.dictionary?.fill ?? false;\n\n if (dictionaryFill === false) continue;\n\n const sourceLocale: Locale = (targetUnmergedDictionary.locale ??\n baseLocale) as Locale;\n\n if (!mainDictionaryToProcess) {\n appLogger(\n `${dictionaryPreset} Dictionary not found in dictionariesRecord. Skipping.`,\n {\n level: 'warn',\n }\n );\n continue;\n }\n\n if (!targetUnmergedDictionary.filePath) {\n appLogger(`${dictionaryPreset} Dictionary has no file path. Skipping.`, {\n level: 'warn',\n });\n continue;\n }\n\n const sourceLocaleContent = getFilterTranslationsOnlyDictionary(\n mainDictionaryToProcess,\n sourceLocale\n );\n\n if (\n Object.keys(sourceLocaleContent as Record<string, unknown>).length === 0\n ) {\n appLogger(\n `${dictionaryPreset} No content found for dictionary in source locale ${formatLocale(sourceLocale)}. Skipping translation for this dictionary.`,\n {\n level: 'warn',\n }\n );\n continue;\n }\n\n /**\n * In 'complete' mode, filter only the missing locales to translate\n *\n * Skip the dictionary if there are no missing locales to translate\n */\n let outputLocalesList: Locale[] = outputLocales as Locale[];\n\n if (mode === 'complete') {\n outputLocalesList =\n missingTranslations\n .find(\n (missingTranslation) => missingTranslation.key === dictionaryKey\n )\n ?.locales.filter((locale) => outputLocales.includes(locale)) ?? [];\n }\n\n if (outputLocalesList.length === 0) {\n appLogger(\n `${dictionaryPreset} ${colorize('No locales to fill, Skipping', ANSIColors.GREY_DARK)} ${colorizePath(basename(targetUnmergedDictionary.filePath))}`,\n {\n level: 'warn',\n }\n );\n continue;\n }\n\n translationTasks.push({\n dictionaryKey,\n dictionaryLocalId,\n sourceLocale,\n targetLocales: outputLocalesList,\n dictionaryPreset,\n dictionaryFilePath: targetUnmergedDictionary.filePath,\n });\n }\n\n // Return the list of tasks to execute\n return translationTasks;\n};\n"],"mappings":"ohBA2BA,MAAa,GACX,EACA,EACA,EACA,EACA,IACsB,CACtB,IAAM,EAAY,EAAa,EAAc,CAEvC,EAA2B,EAAgB,EAAc,CACzD,EAA6B,EAAwB,EAAc,CAGnE,EADsB,OAAO,OAAO,EAA2B,CAAC,MAAM,CAC1B,OAAQ,GACxD,EAAS,SAAS,EAAW,QAAS,CACvC,CAEK,CAAE,uBACN,EAAkC,EAAc,CAE5C,EAAe,KAAK,IACxB,GAAG,EAAsB,IAAK,GAAe,EAAW,IAAI,OAAO,CACpE,CAEK,EAAsC,EAAE,CAE9C,IAAK,IAAM,KAA4B,EAAuB,CAC5D,IAAM,EAAmB,EACvB,CACE,MACA,EAAS,IAAK,EAAW,UAAU,CACnC,EAAY,EAAyB,IAAI,CACzC,EAAS,IAAK,EAAW,UAAU,CACpC,CAAC,KAAK,GAAG,CACV,CAAE,QAAS,EAAe,EAAG,CAC9B,CAEK,EAAgB,EAAyB,IACzC,EAAoB,EAAyB,QAC7C,EACJ,EAAyB,GAU3B,IATyB,EAAyB,QAAU,MAEnC,KAKvB,EAAyB,MAAQ,EAAc,YAAY,MAAQ,MAE9C,GAAO,SAE9B,IAAM,EAAwB,EAAyB,QACrD,EAEF,GAAI,CAAC,EAAyB,CAC5B,EACE,GAAG,EAAiB,wDACpB,CACE,MAAO,OACR,CACF,CACD,SAGF,GAAI,CAAC,EAAyB,SAAU,CACtC,EAAU,GAAG,EAAiB,yCAA0C,CACtE,MAAO,OACR,CAAC,CACF,SAGF,IAAM,EAAsB,EAC1B,EACA,EACD,CAED,GACE,OAAO,KAAK,EAA+C,CAAC,SAAW,EACvE,CACA,EACE,GAAG,EAAiB,oDAAoD,EAAa,EAAa,CAAC,6CACnG,CACE,MAAO,OACR,CACF,CACD,SAQF,IAAI,EAA8B,EAWlC,GATI,IAAS,aACX,EACE,EACG,KACE,GAAuB,EAAmB,MAAQ,EACpD,EACC,QAAQ,OAAQ,GAAW,EAAc,SAAS,EAAO,CAAC,EAAI,EAAE,EAGpE,EAAkB,SAAW,EAAG,CAClC,EACE,GAAG,EAAiB,GAAG,EAAS,+BAAgC,EAAW,UAAU,CAAC,GAAG,EAAa,EAAS,EAAyB,SAAS,CAAC,GAClJ,CACE,MAAO,OACR,CACF,CACD,SAGF,EAAiB,KAAK,CACpB,gBACA,oBACA,eACA,cAAe,EACf,mBACA,mBAAoB,EAAyB,SAC9C,CAAC,CAIJ,OAAO"}
@@ -1,2 +1,2 @@
1
- import{deepMergeContent as e}from"./deepMergeContent.mjs";import{getFilterMissingContentPerLocale as t}from"./getFilterMissingContentPerLocale.mjs";import{getIntlayerAPIProxy as n}from"@intlayer/api";import{ANSIColors as r,colon as i,colorize as a,colorizeNumber as o,colorizePath as s,getAppLogger as c}from"@intlayer/config/logger";import{chunkJSON as l,formatLocale as u,mergeChunks as d,reconstructFromSingleChunk as f,reduceObjectFormat as p,verifyIdenticObjectFormat as m}from"@intlayer/chokidar/utils";import{basename as h}from"node:path";import{retryManager as g}from"@intlayer/config/utils";import{getUnmergedDictionaries as _}from"@intlayer/unmerged-dictionaries-entry";import{getFilterMissingTranslationsDictionary as v,getMultilingualDictionary as y,getPerLocaleDictionary as b,insertContentInDictionary as x}from"@intlayer/core/plugins";const S=e=>!e.description||!e.title||!e.tags,C=e=>{if(typeof e!=`object`||!e||Array.isArray(e))return{content:e,nulls:void 0,hasNulls:!1};let t={},n={},r=!1;for(let[i,a]of Object.entries(e))if(a===null)n[i]=null,r=!0;else{let e=C(a);t[i]=e.content,e.hasNulls&&(n[i]=e.nulls,r=!0)}return{content:t,nulls:r?n:void 0,hasNulls:r}},w=1e3*10;let T=0;const E=async(E,D,O)=>{let k=c(D),A=n(void 0,D),{mode:j,aiOptions:M,fillMetadata:N,aiClient:P,aiConfig:F}={mode:`complete`,fillMetadata:!0,...O},I=()=>{T=0,O?.onSuccess?.()};return await g(async()=>{let n=_(D),c=n[E.dictionaryKey].find(e=>e.localId===E.dictionaryLocalId);if(!c)return k(`${E.dictionaryPreset}Dictionary not found in unmergedDictionariesRecord. Skipping.`,{level:`warn`}),{...E,dictionaryOutput:null};let L;if(N&&(S(c)||j===`review`)){let e=b(c,D.internationalization.defaultLocale);k(`${E.dictionaryPreset} Filling missing metadata for ${s(h(c.filePath))}`,{level:`info`});let t=async()=>P&&F?{data:await P.auditDictionaryMetadata({fileContent:JSON.stringify(e),aiConfig:F})}:await A.ai.auditContentDeclarationMetadata({fileContent:JSON.stringify(e),aiOptions:M});L=(O?.onHandle?await O.onHandle(t):await t()).data?.fileContent}let R=await Promise.all(E.targetLocales.map(async _=>{let y=structuredClone(c),x;if(typeof c.locale==`string`){let e=c.filePath?.replace(RegExp(`/${E.sourceLocale}/`,`g`),`/${_}/`),r=e?n[E.dictionaryKey]?.find(t=>t.filePath===e&&t.locale===_):void 0;x=r??{key:c.key,content:{},filePath:e,locale:_},j===`complete`&&(y=t(y,r))}else j===`complete`&&(y=v(y,_)),y=b(y,E.sourceLocale),x=b(c,_);let S=i([a(`[`,r.GREY_DARK),u(_),a(`]`,r.GREY_DARK)].join(``),{colSize:18}),D=(e,t)=>t<=1?``:i([a(`[`,r.GREY_DARK),o(e+1),a(`/${t}`,r.GREY_DARK),a(`]`,r.GREY_DARK)].join(``),{colSize:5});k(`${E.dictionaryPreset}${S} Preparing ${s(h(x.filePath))}`,{level:`info`});let N=typeof y.content==`object`&&y.content!==null||Array.isArray(y.content),{content:R,nulls:z}=C(N?y.content:{__INTLAYER_ROOT_PRIMITIVE_CONTENT__:y.content}),B=l(R,7e3),V=B.length;V>1&&k(`${E.dictionaryPreset}${S} Split into ${o(V)} chunks for translation`,{level:`info`});let H=[],U=B.map(e=>{let t=D(e.index,e.total);V>1&&k(`${E.dictionaryPreset}${S}${t} Translating chunk`,{level:`info`});let n=f(e),i=p(N?x.content:{__INTLAYER_ROOT_PRIMITIVE_CONTENT__:x.content},n),s=async()=>await g(async()=>{let e;if(e=P&&F?await P.translateJSON({entryFileContent:n,presetOutputContent:i,dictionaryDescription:y.description??L?.description??``,entryLocale:E.sourceLocale,outputLocale:_,mode:j,aiConfig:F}):await A.ai.translateJSON({entryFileContent:n,presetOutputContent:i,dictionaryDescription:y.description??L?.description??``,entryLocale:E.sourceLocale,outputLocale:_,mode:j,aiOptions:M}).then(e=>e.data),!e?.fileContent)throw Error(`No content result`);let{isIdentic:t}=m(e.fileContent,n);if(!t)throw Error(`Translation result does not match expected format`);return I(),e.fileContent},{maxRetry:3,delay:w,onError:({error:t,attempt:n,maxRetry:i})=>{let s=D(e.index,e.total);k(`${E.dictionaryPreset}${S}${s} ${a(`Error filling:`,r.RED)} ${a(typeof t==`string`?t:JSON.stringify(t),r.GREY_DARK)} - Attempt ${o(n+1)} of ${o(i)}`,{level:`error`}),T+=1,T>=10&&(k(`There is something wrong.`,{level:`error`}),process.exit(1))}})();return(O?.onHandle?O.onHandle(s):s()).then(t=>({chunk:e,result:t}))});(await Promise.all(U)).sort((e,t)=>e.chunk.index-t.chunk.index).forEach(({result:e})=>{H.push(e)});let W=d(H);z&&(W=e(W,z));let G={...y,content:W}.content;return N||(G=G?.__INTLAYER_ROOT_PRIMITIVE_CONTENT__),typeof c.locale==`string`&&(G=e(x.content??{},G)),[_,G]})),z=Object.fromEntries(R),B={...y(c.locale?{...c,key:c.key,content:{}}:c),locale:void 0,...L};for(let e of E.targetLocales)z[e]&&(B=x(B,z[e],e));if(k(`${E.dictionaryPreset} ${a(`Translation completed successfully`,r.GREEN)} for ${s(h(B.filePath))}`,{level:`info`}),c.locale&&(c.fill===!0||c.fill===void 0)&&c.location===`local`){let e=c.filePath.split(`.`).slice(0,-1),t=e[e.length-1];return JSON.parse(JSON.stringify({...E,dictionaryOutput:{...B,fill:void 0,filled:!0}}).replaceAll(RegExp(`\\.${t}\\.[a-zA-Z0-9]+`,`g`),`.filled.${t}.json`))}return{...E,dictionaryOutput:B}},{maxRetry:2,delay:w,onError:({error:e,attempt:t,maxRetry:n})=>k(`${E.dictionaryPreset} ${a(`Error:`,r.RED)} ${a(typeof e==`string`?e:JSON.stringify(e),r.GREY_DARK)} - Attempt ${o(t+1)} of ${o(n)}`,{level:`error`}),onMaxTryReached:({error:e})=>k(`${E.dictionaryPreset} ${a(`Maximum number of retries reached:`,r.RED)} ${a(typeof e==`string`?e:JSON.stringify(e),r.GREY_DARK)}`,{level:`error`})})()};export{E as translateDictionary};
1
+ import{deepMergeContent as e}from"./deepMergeContent.mjs";import{getFilterMissingContentPerLocale as t}from"./getFilterMissingContentPerLocale.mjs";import{basename as n}from"node:path";import{ANSIColors as r,colon as i,colorize as a,colorizeNumber as o,colorizePath as s,getAppLogger as c}from"@intlayer/config/logger";import{getUnmergedDictionaries as l}from"@intlayer/unmerged-dictionaries-entry";import{chunkJSON as u,formatLocale as d,mergeChunks as f,reconstructFromSingleChunk as p,reduceObjectFormat as m,verifyIdenticObjectFormat as h}from"@intlayer/chokidar/utils";import{getFilterMissingTranslationsDictionary as g,getMultilingualDictionary as _,getPerLocaleDictionary as v,insertContentInDictionary as y}from"@intlayer/core/plugins";import{getIntlayerAPIProxy as b}from"@intlayer/api";import{retryManager as x}from"@intlayer/config/utils";const S=e=>!e.description||!e.title||!e.tags,C=e=>{if(typeof e!=`object`||!e||Array.isArray(e))return{content:e,nulls:void 0,hasNulls:!1};let t={},n={},r=!1;for(let[i,a]of Object.entries(e))if(a===null)n[i]=null,r=!0;else{let e=C(a);t[i]=e.content,e.hasNulls&&(n[i]=e.nulls,r=!0)}return{content:t,nulls:r?n:void 0,hasNulls:r}},w=1e3*10;let T=0;const E=async(E,D,O)=>{let k=c(D),A=b(void 0,D),{mode:j,aiOptions:M,fillMetadata:N,aiClient:P,aiConfig:F}={mode:`complete`,fillMetadata:!0,...O},I=()=>{T=0,O?.onSuccess?.()};return await x(async()=>{let c=l(D),b=c[E.dictionaryKey].find(e=>e.localId===E.dictionaryLocalId);if(!b)return k(`${E.dictionaryPreset}Dictionary not found in unmergedDictionariesRecord. Skipping.`,{level:`warn`}),{...E,dictionaryOutput:null};let L;if(N&&(S(b)||j===`review`)){let e=v(b,D.internationalization.defaultLocale);k(`${E.dictionaryPreset} Filling missing metadata for ${s(n(b.filePath))}`,{level:`info`});let t=async()=>P&&F?{data:await P.auditDictionaryMetadata({fileContent:JSON.stringify(e),aiConfig:F})}:await A.ai.auditContentDeclarationMetadata({fileContent:JSON.stringify(e),aiOptions:M});L=(O?.onHandle?await O.onHandle(t):await t()).data?.fileContent}let R=await Promise.all(E.targetLocales.map(async l=>{let _=structuredClone(b),y;if(typeof b.locale==`string`){let e=b.filePath?.replace(RegExp(`/${E.sourceLocale}/`,`g`),`/${l}/`),n=e?c[E.dictionaryKey]?.find(t=>t.filePath===e&&t.locale===l):void 0;y=n??{key:b.key,content:{},filePath:e,locale:l},j===`complete`&&(_=t(_,n))}else j===`complete`&&(_=g(_,l)),_=v(_,E.sourceLocale),y=v(b,l);let S=i([a(`[`,r.GREY_DARK),d(l),a(`]`,r.GREY_DARK)].join(``),{colSize:18}),D=(e,t)=>t<=1?``:i([a(`[`,r.GREY_DARK),o(e+1),a(`/${t}`,r.GREY_DARK),a(`]`,r.GREY_DARK)].join(``),{colSize:5});k(`${E.dictionaryPreset}${S} Preparing ${s(n(y.filePath))}`,{level:`info`});let N=typeof _.content==`object`&&_.content!==null||Array.isArray(_.content),{content:R,nulls:z}=C(N?_.content:{__INTLAYER_ROOT_PRIMITIVE_CONTENT__:_.content}),B=u(R,7e3),V=B.length;V>1&&k(`${E.dictionaryPreset}${S} Split into ${o(V)} chunks for translation`,{level:`info`});let H=[],U=B.map(e=>{let t=D(e.index,e.total);V>1&&k(`${E.dictionaryPreset}${S}${t} Translating chunk`,{level:`info`});let n=p(e),i=m(N?y.content:{__INTLAYER_ROOT_PRIMITIVE_CONTENT__:y.content},n),s=async()=>await x(async()=>{let e;if(e=P&&F?await P.translateJSON({entryFileContent:n,presetOutputContent:i,dictionaryDescription:_.description??L?.description??``,entryLocale:E.sourceLocale,outputLocale:l,mode:j,aiConfig:F}):await A.ai.translateJSON({entryFileContent:n,presetOutputContent:i,dictionaryDescription:_.description??L?.description??``,entryLocale:E.sourceLocale,outputLocale:l,mode:j,aiOptions:M}).then(e=>e.data),!e?.fileContent)throw Error(`No content result`);let{isIdentic:t}=h(e.fileContent,n);if(!t)throw Error(`Translation result does not match expected format`);return I(),e.fileContent},{maxRetry:3,delay:w,onError:({error:t,attempt:n,maxRetry:i})=>{let s=D(e.index,e.total);k(`${E.dictionaryPreset}${S}${s} ${a(`Error filling:`,r.RED)} ${a(typeof t==`string`?t:JSON.stringify(t),r.GREY_DARK)} - Attempt ${o(n+1)} of ${o(i)}`,{level:`error`}),T+=1,T>=10&&(k(`There is something wrong.`,{level:`error`}),process.exit(1))}})();return(O?.onHandle?O.onHandle(s):s()).then(t=>({chunk:e,result:t}))});(await Promise.all(U)).sort((e,t)=>e.chunk.index-t.chunk.index).forEach(({result:e})=>{H.push(e)});let W=f(H);z&&(W=e(W,z));let G={..._,content:W}.content;return N||(G=G?.__INTLAYER_ROOT_PRIMITIVE_CONTENT__),typeof b.locale==`string`&&(G=e(y.content??{},G)),[l,G]})),z=Object.fromEntries(R),B={..._(b.locale?{...b,key:b.key,content:{}}:b),locale:void 0,...L};for(let e of E.targetLocales)z[e]&&(B=y(B,z[e],e));if(k(`${E.dictionaryPreset} ${a(`Translation completed successfully`,r.GREEN)} for ${s(n(B.filePath))}`,{level:`info`}),b.locale&&(b.fill===!0||b.fill===void 0)&&b.location===`local`){let e=b.filePath.split(`.`).slice(0,-1),t=e[e.length-1];return JSON.parse(JSON.stringify({...E,dictionaryOutput:{...B,fill:void 0,filled:!0}}).replaceAll(RegExp(`\\.${t}\\.[a-zA-Z0-9]+`,`g`),`.filled.${t}.json`))}return{...E,dictionaryOutput:B}},{maxRetry:2,delay:w,onError:({error:e,attempt:t,maxRetry:n})=>k(`${E.dictionaryPreset} ${a(`Error:`,r.RED)} ${a(typeof e==`string`?e:JSON.stringify(e),r.GREY_DARK)} - Attempt ${o(t+1)} of ${o(n)}`,{level:`error`}),onMaxTryReached:({error:e})=>k(`${E.dictionaryPreset} ${a(`Maximum number of retries reached:`,r.RED)} ${a(typeof e==`string`?e:JSON.stringify(e),r.GREY_DARK)}`,{level:`error`})})()};export{E as translateDictionary};
2
2
  //# sourceMappingURL=translateDictionary.mjs.map