@intlayer/cli 8.12.5-canary.0 → 9.0.0-canary.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (100) hide show
  1. package/dist/cjs/cli.cjs +18 -3
  2. package/dist/cjs/cli.cjs.map +1 -1
  3. package/dist/cjs/index.cjs +2 -0
  4. package/dist/cjs/reviewDoc/reviewDoc.cjs +38 -11
  5. package/dist/cjs/reviewDoc/reviewDoc.cjs.map +1 -1
  6. package/dist/cjs/reviewDoc/reviewDocBlockAware.cjs +54 -37
  7. package/dist/cjs/reviewDoc/reviewDocBlockAware.cjs.map +1 -1
  8. package/dist/cjs/reviewDoc/reviewDocLog.cjs +48 -0
  9. package/dist/cjs/reviewDoc/reviewDocLog.cjs.map +1 -0
  10. package/dist/cjs/scan.cjs +75 -0
  11. package/dist/cjs/scan.cjs.map +1 -0
  12. package/dist/cjs/utils/formatLineRanges.cjs +44 -0
  13. package/dist/cjs/utils/formatLineRanges.cjs.map +1 -0
  14. package/dist/esm/cli.mjs +18 -3
  15. package/dist/esm/cli.mjs.map +1 -1
  16. package/dist/esm/index.mjs +2 -1
  17. package/dist/esm/reviewDoc/reviewDoc.mjs +38 -11
  18. package/dist/esm/reviewDoc/reviewDoc.mjs.map +1 -1
  19. package/dist/esm/reviewDoc/reviewDocBlockAware.mjs +54 -37
  20. package/dist/esm/reviewDoc/reviewDocBlockAware.mjs.map +1 -1
  21. package/dist/esm/reviewDoc/reviewDocLog.mjs +46 -0
  22. package/dist/esm/reviewDoc/reviewDocLog.mjs.map +1 -0
  23. package/dist/esm/scan.mjs +72 -0
  24. package/dist/esm/scan.mjs.map +1 -0
  25. package/dist/esm/utils/formatLineRanges.mjs +42 -0
  26. package/dist/esm/utils/formatLineRanges.mjs.map +1 -0
  27. package/dist/types/cli.d.ts.map +1 -1
  28. package/dist/types/index.d.ts +2 -1
  29. package/dist/types/reviewDoc/reviewDoc.d.ts +8 -1
  30. package/dist/types/reviewDoc/reviewDoc.d.ts.map +1 -1
  31. package/dist/types/reviewDoc/reviewDocBlockAware.d.ts +8 -6
  32. package/dist/types/reviewDoc/reviewDocBlockAware.d.ts.map +1 -1
  33. package/dist/types/reviewDoc/reviewDocLog.d.ts +25 -0
  34. package/dist/types/reviewDoc/reviewDocLog.d.ts.map +1 -0
  35. package/dist/types/scan.d.ts +22 -0
  36. package/dist/types/scan.d.ts.map +1 -0
  37. package/dist/types/utils/formatLineRanges.d.ts +21 -0
  38. package/dist/types/utils/formatLineRanges.d.ts.map +1 -0
  39. package/package.json +14 -14
  40. package/dist/cjs/translation-alignment/alignBlocks.cjs +0 -68
  41. package/dist/cjs/translation-alignment/alignBlocks.cjs.map +0 -1
  42. package/dist/cjs/translation-alignment/computeSimilarity.cjs +0 -26
  43. package/dist/cjs/translation-alignment/computeSimilarity.cjs.map +0 -1
  44. package/dist/cjs/translation-alignment/fingerprintBlock.cjs +0 -24
  45. package/dist/cjs/translation-alignment/fingerprintBlock.cjs.map +0 -1
  46. package/dist/cjs/translation-alignment/index.cjs +0 -22
  47. package/dist/cjs/translation-alignment/mapChangedLinesToBlocks.cjs +0 -19
  48. package/dist/cjs/translation-alignment/mapChangedLinesToBlocks.cjs.map +0 -1
  49. package/dist/cjs/translation-alignment/normalizeBlock.cjs +0 -23
  50. package/dist/cjs/translation-alignment/normalizeBlock.cjs.map +0 -1
  51. package/dist/cjs/translation-alignment/pipeline.cjs +0 -38
  52. package/dist/cjs/translation-alignment/pipeline.cjs.map +0 -1
  53. package/dist/cjs/translation-alignment/planActions.cjs +0 -47
  54. package/dist/cjs/translation-alignment/planActions.cjs.map +0 -1
  55. package/dist/cjs/translation-alignment/rebuildDocument.cjs +0 -50
  56. package/dist/cjs/translation-alignment/rebuildDocument.cjs.map +0 -1
  57. package/dist/cjs/translation-alignment/segmentDocument.cjs +0 -67
  58. package/dist/cjs/translation-alignment/segmentDocument.cjs.map +0 -1
  59. package/dist/cjs/translation-alignment/types.cjs +0 -0
  60. package/dist/esm/translation-alignment/alignBlocks.mjs +0 -67
  61. package/dist/esm/translation-alignment/alignBlocks.mjs.map +0 -1
  62. package/dist/esm/translation-alignment/computeSimilarity.mjs +0 -23
  63. package/dist/esm/translation-alignment/computeSimilarity.mjs.map +0 -1
  64. package/dist/esm/translation-alignment/fingerprintBlock.mjs +0 -21
  65. package/dist/esm/translation-alignment/fingerprintBlock.mjs.map +0 -1
  66. package/dist/esm/translation-alignment/index.mjs +0 -11
  67. package/dist/esm/translation-alignment/mapChangedLinesToBlocks.mjs +0 -17
  68. package/dist/esm/translation-alignment/mapChangedLinesToBlocks.mjs.map +0 -1
  69. package/dist/esm/translation-alignment/normalizeBlock.mjs +0 -21
  70. package/dist/esm/translation-alignment/normalizeBlock.mjs.map +0 -1
  71. package/dist/esm/translation-alignment/pipeline.mjs +0 -36
  72. package/dist/esm/translation-alignment/pipeline.mjs.map +0 -1
  73. package/dist/esm/translation-alignment/planActions.mjs +0 -45
  74. package/dist/esm/translation-alignment/planActions.mjs.map +0 -1
  75. package/dist/esm/translation-alignment/rebuildDocument.mjs +0 -47
  76. package/dist/esm/translation-alignment/rebuildDocument.mjs.map +0 -1
  77. package/dist/esm/translation-alignment/segmentDocument.mjs +0 -65
  78. package/dist/esm/translation-alignment/segmentDocument.mjs.map +0 -1
  79. package/dist/esm/translation-alignment/types.mjs +0 -0
  80. package/dist/types/translation-alignment/alignBlocks.d.ts +0 -7
  81. package/dist/types/translation-alignment/alignBlocks.d.ts.map +0 -1
  82. package/dist/types/translation-alignment/computeSimilarity.d.ts +0 -6
  83. package/dist/types/translation-alignment/computeSimilarity.d.ts.map +0 -1
  84. package/dist/types/translation-alignment/fingerprintBlock.d.ts +0 -7
  85. package/dist/types/translation-alignment/fingerprintBlock.d.ts.map +0 -1
  86. package/dist/types/translation-alignment/index.d.ts +0 -11
  87. package/dist/types/translation-alignment/mapChangedLinesToBlocks.d.ts +0 -7
  88. package/dist/types/translation-alignment/mapChangedLinesToBlocks.d.ts.map +0 -1
  89. package/dist/types/translation-alignment/normalizeBlock.d.ts +0 -7
  90. package/dist/types/translation-alignment/normalizeBlock.d.ts.map +0 -1
  91. package/dist/types/translation-alignment/pipeline.d.ts +0 -25
  92. package/dist/types/translation-alignment/pipeline.d.ts.map +0 -1
  93. package/dist/types/translation-alignment/planActions.d.ts +0 -7
  94. package/dist/types/translation-alignment/planActions.d.ts.map +0 -1
  95. package/dist/types/translation-alignment/rebuildDocument.d.ts +0 -32
  96. package/dist/types/translation-alignment/rebuildDocument.d.ts.map +0 -1
  97. package/dist/types/translation-alignment/segmentDocument.d.ts +0 -7
  98. package/dist/types/translation-alignment/segmentDocument.d.ts.map +0 -1
  99. package/dist/types/translation-alignment/types.d.ts +0 -49
  100. package/dist/types/translation-alignment/types.d.ts.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"reviewDocBlockAware.cjs","names":["readAsset","ANSIColors","buildAlignmentPlan","mergeReviewedSegments","chunkInference","ENGLISH","sanitizeChunk","fixChunkStartEndChars","validateTranslation"],"sources":["../../../src/reviewDoc/reviewDocBlockAware.ts"],"sourcesContent":["import { mkdirSync, writeFileSync } from 'node:fs';\nimport { readFile } from 'node:fs/promises';\nimport { dirname } from 'node:path';\nimport { readAsset } from 'utils:asset';\nimport type { AIConfig } from '@intlayer/ai';\nimport type { AIOptions } from '@intlayer/api';\nimport { formatLocale, formatPath } from '@intlayer/chokidar/utils';\nimport * as ANSIColors from '@intlayer/config/colors';\nimport {\n colon,\n colorize,\n colorizeNumber,\n getAppLogger,\n} from '@intlayer/config/logger';\nimport {\n type GetConfigurationOptions,\n getConfiguration,\n} from '@intlayer/config/node';\nimport { retryManager } from '@intlayer/config/utils';\nimport { getLocaleName } from '@intlayer/core/localization';\nimport type { Locale } from '@intlayer/types/allLocales';\nimport { ENGLISH } from '@intlayer/types/locales';\nimport { sanitizeChunk, validateTranslation } from '../translateDoc/validation';\nimport {\n buildAlignmentPlan,\n mergeReviewedSegments,\n} from '../translation-alignment/pipeline';\nimport { chunkInference } from '../utils/chunkInference';\nimport { fixChunkStartEndChars } from '../utils/fixChunkStartEndChars';\nimport type { AIClient } from '../utils/setupAI';\n\n/**\n * Review a file using block-aware alignment.\n * This approach:\n * 1. Segments both English and French documents into semantic blocks\n * 2. Aligns blocks using structure (special chars, numbers) and context\n * 3. Detects which blocks changed, were added, or deleted\n * 4. Only sends changed/new blocks to AI for translation\n * 5. Handles reordering automatically\n */\nexport const reviewFileBlockAware = async (\n baseFilePath: string,\n outputFilePath: string,\n locale: Locale,\n baseLocale: Locale,\n aiOptions?: AIOptions,\n configOptions?: GetConfigurationOptions,\n customInstructions?: string,\n changedLines?: number[],\n aiClient?: AIClient,\n aiConfig?: AIConfig\n) => {\n const configuration = getConfiguration(configOptions);\n const applicationLogger = getAppLogger(configuration);\n\n const englishText = await readFile(baseFilePath, 'utf-8');\n const frenchText = await readFile(outputFilePath, 'utf-8').catch(() => '');\n\n const basePrompt = readAsset('./prompts/REVIEW_PROMPT.md', 'utf-8')\n .replaceAll('{{localeName}}', `${formatLocale(locale, false)}`)\n .replaceAll('{{baseLocaleName}}', `${formatLocale(baseLocale, false)}`)\n .replace('{{applicationContext}}', aiOptions?.applicationContext ?? '-')\n .replace('{{customInstructions}}', customInstructions ?? '-');\n\n const filePrefixText = `${ANSIColors.GREY_DARK}[${formatPath(baseFilePath)}${ANSIColors.GREY_DARK}] `;\n const filePrefix = [\n colon(filePrefixText, { colSize: 40 }),\n `→ ${ANSIColors.RESET}`,\n ].join('');\n const prefixText = `${ANSIColors.GREY_DARK}[${formatPath(baseFilePath)}${ANSIColors.GREY_DARK}][${formatLocale(locale)}${ANSIColors.GREY_DARK}] `;\n const prefix = [\n colon(prefixText, { colSize: 40 }),\n `→ ${ANSIColors.RESET}`,\n ].join('');\n\n // Build block-aware alignment and plan\n const { englishBlocks, frenchBlocks, plan, segmentsToReview } =\n buildAlignmentPlan({\n englishText,\n frenchText,\n changedLines,\n });\n\n applicationLogger(\n `${filePrefix}Block-aware alignment complete. Total blocks: EN=${colorizeNumber(englishBlocks.length)}, FR=${colorizeNumber(frenchBlocks.length)}`\n );\n applicationLogger(\n `${filePrefix}Actions: reuse=${colorizeNumber(plan.actions.filter((a) => a.kind === 'reuse').length)}, review=${colorizeNumber(plan.actions.filter((a) => a.kind === 'review').length)}, new=${colorizeNumber(plan.actions.filter((a) => a.kind === 'insert_new').length)}, delete=${colorizeNumber(plan.actions.filter((a) => a.kind === 'delete').length)}`\n );\n\n if (segmentsToReview.length === 0) {\n applicationLogger(\n `${filePrefix}No segments need review, reusing existing translation`\n );\n mkdirSync(dirname(outputFilePath), { recursive: true });\n writeFileSync(\n outputFilePath,\n mergeReviewedSegments(plan, frenchBlocks, new Map())\n );\n applicationLogger(\n `${colorize('✔', ANSIColors.GREEN)} File ${formatPath(outputFilePath)} updated successfully (no changes needed).`\n );\n return;\n }\n\n applicationLogger(\n `${filePrefix}Segments to review: ${colorizeNumber(segmentsToReview.length)}`\n );\n\n // Review segments that need AI translation\n const reviewedSegmentsMap = new Map<number, string>();\n\n for (const segment of segmentsToReview) {\n const segmentNumber = segmentsToReview.indexOf(segment) + 1;\n const englishBlock = segment.englishBlock;\n\n const getBaseChunkContextPrompt = () =>\n `**BLOCK ${segmentNumber} of ${segmentsToReview.length}** is the base block in ${formatLocale(baseLocale, false)} as reference.\\n` +\n `///chunksStart///\\n` +\n englishBlock.content +\n `///chunksEnd///`;\n\n const getFrenchChunkPrompt = () =>\n `**BLOCK ${segmentNumber} of ${segmentsToReview.length}** is the current block to review in ${formatLocale(locale, false)}.\\n` +\n `///chunksStart///\\n` +\n (segment.frenchBlockText ?? '') +\n `///chunksEnd///`;\n\n const reviewedChunkResult = await retryManager(async () => {\n const result = await chunkInference(\n [\n { role: 'system', content: basePrompt },\n { role: 'system', content: getBaseChunkContextPrompt() },\n { role: 'system', content: getFrenchChunkPrompt() },\n {\n role: 'system',\n content: `The next user message will be the **BLOCK ${colorizeNumber(segmentNumber)} of ${colorizeNumber(segmentsToReview.length)}** that should be translated in ${getLocaleName(locale, ENGLISH)} (${locale}).`,\n },\n ],\n [{ role: 'user', content: englishBlock.content }],\n aiOptions,\n configuration,\n aiClient,\n aiConfig\n );\n\n applicationLogger(\n `${prefix}${colorizeNumber(result.tokenUsed)} tokens used - Block ${colorizeNumber(segmentNumber)} of ${colorizeNumber(segmentsToReview.length)}`\n );\n\n // Sanitize artifacts (e.g. Markdown code block wrappers)\n let processedChunk = sanitizeChunk(\n result?.fileContent,\n englishBlock.content\n );\n\n // Fix start/end characters\n processedChunk = fixChunkStartEndChars(\n processedChunk,\n englishBlock.content\n );\n\n // Validate Translation (YAML, Code fences, Length ratio)\n const isValid = validateTranslation(\n englishBlock.content,\n processedChunk,\n applicationLogger\n );\n\n if (!isValid) {\n throw new Error(\n 'Validation failed for chunk (structure or length mismatch). Retrying...'\n );\n }\n\n return processedChunk;\n })();\n\n reviewedSegmentsMap.set(segment.actionIndex, reviewedChunkResult);\n }\n\n // Merge reviewed segments back into final document\n const finalFrenchOutput = mergeReviewedSegments(\n plan,\n frenchBlocks,\n reviewedSegmentsMap\n );\n\n mkdirSync(dirname(outputFilePath), { recursive: true });\n writeFileSync(outputFilePath, finalFrenchOutput);\n\n applicationLogger(\n `${colorize('✔', ANSIColors.GREEN)} File ${formatPath(outputFilePath)} created/updated successfully.`\n );\n};\n"],"mappings":"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"}
1
+ {"version":3,"file":"reviewDocBlockAware.cjs","names":["readAsset","ANSIColors","chunkInference","ENGLISH","sanitizeChunk","fixChunkStartEndChars","validateTranslation"],"sources":["../../../src/reviewDoc/reviewDocBlockAware.ts"],"sourcesContent":["import { mkdirSync, writeFileSync } from 'node:fs';\nimport { readFile } from 'node:fs/promises';\nimport { dirname } from 'node:path';\nimport { readAsset } from 'utils:asset';\nimport type { AIConfig } from '@intlayer/ai';\nimport type { AIOptions } from '@intlayer/api';\nimport {\n buildAlignmentPlan,\n mergeReviewedSegments,\n} from '@intlayer/chokidar/docReview';\nimport { formatLocale, formatPath } from '@intlayer/chokidar/utils';\nimport * as ANSIColors from '@intlayer/config/colors';\nimport {\n colon,\n colorize,\n colorizeNumber,\n getAppLogger,\n} from '@intlayer/config/logger';\nimport {\n type GetConfigurationOptions,\n getConfiguration,\n} from '@intlayer/config/node';\nimport { retryManager } from '@intlayer/config/utils';\nimport { getLocaleName } from '@intlayer/core/localization';\nimport type { Locale } from '@intlayer/types/allLocales';\nimport { ENGLISH } from '@intlayer/types/locales';\nimport { sanitizeChunk, validateTranslation } from '../translateDoc/validation';\nimport { chunkInference } from '../utils/chunkInference';\nimport { fixChunkStartEndChars } from '../utils/fixChunkStartEndChars';\nimport type { AIClient } from '../utils/setupAI';\n\n/**\n * Review a file using block-aware alignment.\n *\n * 1. Segments both base and target documents into semantic blocks.\n * 2. Aligns blocks using structure (special chars, numbers) and context.\n * 3. Detects which blocks changed, were added, or deleted.\n * 4. Applies deletions immediately without AI.\n * 5. Sends changed/new blocks to AI in bottom-up order (last block first), so\n * line numbers of earlier blocks are not shifted by edits below them.\n * 6. Rewrites the file after each block so progress is persisted incrementally.\n */\nexport const reviewFileBlockAware = async (\n baseFilePath: string,\n outputFilePath: string,\n locale: Locale,\n baseLocale: Locale,\n aiOptions?: AIOptions,\n configOptions?: GetConfigurationOptions,\n customInstructions?: string,\n changedLines?: number[],\n aiClient?: AIClient,\n aiConfig?: AIConfig\n) => {\n const configuration = getConfiguration(configOptions);\n const applicationLogger = getAppLogger({\n log: { ...configuration.log, prefix: '' },\n });\n\n const baseText = await readFile(baseFilePath, 'utf-8');\n const targetText = await readFile(outputFilePath, 'utf-8').catch(() => '');\n\n const basePrompt = readAsset('./prompts/REVIEW_PROMPT.md', 'utf-8')\n .replaceAll('{{localeName}}', `${formatLocale(locale, false)}`)\n .replaceAll('{{baseLocaleName}}', `${formatLocale(baseLocale, false)}`)\n .replace('{{applicationContext}}', aiOptions?.applicationContext ?? '-')\n .replace('{{customInstructions}}', customInstructions ?? '-');\n\n const filePrefixText = `${ANSIColors.GREY_DARK}[${formatPath(baseFilePath)}${ANSIColors.GREY_DARK}] `;\n const filePrefix = [\n colon(filePrefixText, { colSize: 40 }),\n `→ ${ANSIColors.RESET}`,\n ].join('');\n const prefixText = `${ANSIColors.GREY_DARK}[${formatPath(baseFilePath)}${ANSIColors.GREY_DARK}][${formatLocale(locale)}${ANSIColors.GREY_DARK}] `;\n const prefix = [\n colon(prefixText, { colSize: 40 }),\n `→ ${ANSIColors.RESET}`,\n ].join('');\n\n // Build block-aware alignment and plan\n const { baseBlocks, targetBlocks, plan, segmentsToReview } =\n buildAlignmentPlan({\n baseText,\n targetText,\n changedLines,\n });\n\n const deleteCount = plan.actions.filter((a) => a.kind === 'delete').length;\n\n applicationLogger(\n `${filePrefix}Block-aware alignment complete. Total blocks: base=${colorizeNumber(baseBlocks.length)}, target=${colorizeNumber(targetBlocks.length)}`\n );\n applicationLogger(\n `${filePrefix}Actions: reuse=${colorizeNumber(plan.actions.filter((a) => a.kind === 'reuse').length)}, review=${colorizeNumber(plan.actions.filter((a) => a.kind === 'review').length)}, new=${colorizeNumber(plan.actions.filter((a) => a.kind === 'insert_new').length)}, delete=${colorizeNumber(deleteCount)}`\n );\n\n // Map shared across the entire run: each entry overrides the default behavior\n // of mergeReviewedSegments for that action index.\n const reviewedSegmentsMap = new Map<number, string>();\n\n // --- Step 1: apply deletions immediately (no AI needed) ---\n for (const [actionIndex, action] of plan.actions.entries()) {\n if (action.kind === 'delete') {\n reviewedSegmentsMap.set(actionIndex, '');\n }\n }\n\n const writeCurrentState = (): void => {\n const output = mergeReviewedSegments(\n plan,\n targetBlocks,\n reviewedSegmentsMap\n );\n mkdirSync(dirname(outputFilePath), { recursive: true });\n writeFileSync(outputFilePath, output);\n };\n\n if (deleteCount > 0) {\n writeCurrentState();\n applicationLogger(\n `${filePrefix}${colorizeNumber(deleteCount)} block(s) deleted without AI.`\n );\n }\n\n if (segmentsToReview.length === 0) {\n if (deleteCount === 0) {\n applicationLogger(\n `${filePrefix}No segments need review, reusing existing translation`\n );\n writeCurrentState();\n }\n applicationLogger(\n `${colorize('✔', ANSIColors.GREEN)} File ${formatPath(outputFilePath)} updated successfully (no AI changes needed).`\n );\n return;\n }\n\n applicationLogger(\n `${filePrefix}Segments to review: ${colorizeNumber(segmentsToReview.length)} (processing bottom-up)`\n );\n\n // --- Step 2: process AI segments in bottom-up order ---\n // Reversing ensures edits near the end of the file don't shift line numbers\n // that matter for blocks higher up, and each intermediate file write is valid.\n const segmentsBottomUp = segmentsToReview\n .map((segment, originalIndex) => ({\n segment,\n displayNumber: originalIndex + 1,\n }))\n .reverse();\n\n for (const { segment, displayNumber } of segmentsBottomUp) {\n const baseBlock = segment.baseBlock;\n\n const getBaseChunkContextPrompt = () =>\n `**BLOCK ${displayNumber} of ${segmentsToReview.length}** is the base block in ${formatLocale(baseLocale, false)} as reference.\\n` +\n `///chunksStart///\\n` +\n baseBlock.content +\n `///chunksEnd///`;\n\n const getTargetChunkPrompt = () =>\n `**BLOCK ${displayNumber} of ${segmentsToReview.length}** is the current block to review in ${formatLocale(locale, false)}.\\n` +\n `///chunksStart///\\n` +\n (segment.targetBlockText ?? '') +\n `///chunksEnd///`;\n\n const reviewedChunkResult = await retryManager(async () => {\n const result = await chunkInference(\n [\n { role: 'system', content: basePrompt },\n { role: 'system', content: getBaseChunkContextPrompt() },\n { role: 'system', content: getTargetChunkPrompt() },\n {\n role: 'system',\n content: `The next user message will be the **BLOCK ${colorizeNumber(displayNumber)} of ${colorizeNumber(segmentsToReview.length)}** that should be translated in ${getLocaleName(locale, ENGLISH)} (${locale}).`,\n },\n ],\n [{ role: 'user', content: baseBlock.content }],\n aiOptions,\n configuration,\n aiClient,\n aiConfig\n );\n\n applicationLogger(\n `${prefix}${colorizeNumber(result.tokenUsed)} tokens used - Block ${colorizeNumber(displayNumber)} of ${colorizeNumber(segmentsToReview.length)}`\n );\n\n let processedChunk = sanitizeChunk(\n result?.fileContent,\n baseBlock.content\n );\n processedChunk = fixChunkStartEndChars(processedChunk, baseBlock.content);\n\n const isValid = validateTranslation(\n baseBlock.content,\n processedChunk,\n applicationLogger\n );\n\n if (!isValid) {\n throw new Error(\n 'Validation failed for chunk (structure or length mismatch). Retrying...'\n );\n }\n\n return processedChunk;\n })();\n\n reviewedSegmentsMap.set(segment.actionIndex, reviewedChunkResult);\n\n // Rewrite the file after every block so progress is never lost.\n writeCurrentState();\n }\n\n applicationLogger(\n `${colorize('✔', ANSIColors.GREEN)} File ${formatPath(outputFilePath)} created/updated successfully.`\n );\n};\n"],"mappings":"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"}
@@ -0,0 +1,48 @@
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2
+ const require_runtime = require('../_virtual/_rolldown/runtime.cjs');
3
+ let node_fs = require("node:fs");
4
+ let _intlayer_chokidar_utils = require("@intlayer/chokidar/utils");
5
+ let _intlayer_config_logger = require("@intlayer/config/logger");
6
+ let _intlayer_config_node = require("@intlayer/config/node");
7
+ let node_fs_promises = require("node:fs/promises");
8
+ let _intlayer_chokidar_docReview = require("@intlayer/chokidar/docReview");
9
+
10
+ //#region src/reviewDoc/reviewDocLog.ts
11
+ /**
12
+ * Log-only review of a single file/locale pair.
13
+ *
14
+ * Instead of calling an AI to translate the changed blocks, this compares the
15
+ * base document with its translation and logs the blocks that need attention
16
+ * (with their line ranges and content) so another agent or a human can generate
17
+ * the missing translations.
18
+ *
19
+ * @param baseFilePath - Absolute path of the base (source) document.
20
+ * @param outputFilePath - Absolute path of the target (translated) document.
21
+ * @param locale - The target locale being reviewed.
22
+ * @param baseLocale - The base locale used as reference.
23
+ * @param configOptions - Optional Intlayer configuration overrides.
24
+ * @param changedLines - 1-based base line numbers that changed (from git), if any.
25
+ * @returns The structured review report.
26
+ */
27
+ const logReviewFileBlocks = async (baseFilePath, outputFilePath, locale, baseLocale, configOptions, changedLines) => {
28
+ const appLogger = (0, _intlayer_config_logger.getAppLogger)({ log: {
29
+ ...(0, _intlayer_config_node.getConfiguration)(configOptions).log,
30
+ prefix: ""
31
+ } });
32
+ const report = (0, _intlayer_chokidar_docReview.buildReviewReport)({
33
+ baseText: await (0, node_fs_promises.readFile)(baseFilePath, "utf-8"),
34
+ targetText: (0, node_fs.existsSync)(outputFilePath) ? await (0, node_fs_promises.readFile)(outputFilePath, "utf-8").catch(() => "") : "",
35
+ changedLines
36
+ });
37
+ const formatted = (0, _intlayer_chokidar_docReview.formatReviewReport)(report, {
38
+ baseLabel: (0, _intlayer_chokidar_utils.formatLocale)(baseLocale),
39
+ targetLabel: (0, _intlayer_chokidar_utils.formatLocale)(locale)
40
+ });
41
+ appLogger(`${(0, _intlayer_chokidar_utils.formatPath)(baseFilePath)} → ${(0, _intlayer_chokidar_utils.formatLocale)(locale)}`);
42
+ for (const line of formatted.split("\n")) appLogger(line);
43
+ return report;
44
+ };
45
+
46
+ //#endregion
47
+ exports.logReviewFileBlocks = logReviewFileBlocks;
48
+ //# sourceMappingURL=reviewDocLog.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"reviewDocLog.cjs","names":[],"sources":["../../../src/reviewDoc/reviewDocLog.ts"],"sourcesContent":["import { existsSync } from 'node:fs';\nimport { readFile } from 'node:fs/promises';\nimport {\n buildReviewReport,\n formatReviewReport,\n type ReviewReport,\n} from '@intlayer/chokidar/docReview';\nimport { formatLocale, formatPath } from '@intlayer/chokidar/utils';\nimport { getAppLogger } from '@intlayer/config/logger';\nimport {\n type GetConfigurationOptions,\n getConfiguration,\n} from '@intlayer/config/node';\nimport type { Locale } from '@intlayer/types/allLocales';\n\n/**\n * Log-only review of a single file/locale pair.\n *\n * Instead of calling an AI to translate the changed blocks, this compares the\n * base document with its translation and logs the blocks that need attention\n * (with their line ranges and content) so another agent or a human can generate\n * the missing translations.\n *\n * @param baseFilePath - Absolute path of the base (source) document.\n * @param outputFilePath - Absolute path of the target (translated) document.\n * @param locale - The target locale being reviewed.\n * @param baseLocale - The base locale used as reference.\n * @param configOptions - Optional Intlayer configuration overrides.\n * @param changedLines - 1-based base line numbers that changed (from git), if any.\n * @returns The structured review report.\n */\nexport const logReviewFileBlocks = async (\n baseFilePath: string,\n outputFilePath: string,\n locale: Locale,\n baseLocale: Locale,\n configOptions?: GetConfigurationOptions,\n changedLines?: number[]\n): Promise<ReviewReport> => {\n const configuration = getConfiguration(configOptions);\n const appLogger = getAppLogger({ log: { ...configuration.log, prefix: '' } });\n\n const baseText = await readFile(baseFilePath, 'utf-8');\n const targetText = existsSync(outputFilePath)\n ? await readFile(outputFilePath, 'utf-8').catch(() => '')\n : '';\n\n const report = buildReviewReport({ baseText, targetText, changedLines });\n\n const formatted = formatReviewReport(report, {\n baseLabel: formatLocale(baseLocale),\n targetLabel: formatLocale(locale),\n });\n\n appLogger(`${formatPath(baseFilePath)} → ${formatLocale(locale)}`);\n for (const line of formatted.split('\\n')) {\n appLogger(line);\n }\n\n return report;\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;AA+BA,MAAa,sBAAsB,OACjC,cACA,gBACA,QACA,YACA,eACA,iBAC0B;CAE1B,MAAM,sDAAyB,EAAE,KAAK;EAAE,+CADD,cACiB,CAAC;EAAK,QAAQ;EAAI,EAAE,CAAC;CAO7E,MAAM,6DAA2B;EAAE,+CALH,cAAc,QAAQ;EAKT,oCAJf,eAAe,GACzC,qCAAe,gBAAgB,QAAQ,CAAC,YAAY,GAAG,GACvD;EAEqD;EAAc,CAAC;CAExE,MAAM,iEAA+B,QAAQ;EAC3C,sDAAwB,WAAW;EACnC,wDAA0B,OAAO;EAClC,CAAC;AAEF,WAAU,4CAAc,aAAa,CAAC,gDAAkB,OAAO,GAAG;AAClE,MAAK,MAAM,QAAQ,UAAU,MAAM,KAAK,CACtC,WAAU,KAAK;AAGjB,QAAO"}
@@ -0,0 +1,75 @@
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2
+ const require_runtime = require('./_virtual/_rolldown/runtime.cjs');
3
+ let _intlayer_config_colors = require("@intlayer/config/colors");
4
+ _intlayer_config_colors = require_runtime.__toESM(_intlayer_config_colors);
5
+ let _intlayer_config_logger = require("@intlayer/config/logger");
6
+ let _intlayer_config_node = require("@intlayer/config/node");
7
+ let _intlayer_chokidar_scan = require("@intlayer/chokidar/scan");
8
+
9
+ //#region src/scan.ts
10
+ /** Human-readable labels for each scorable check type. */
11
+ const checkLabels = {
12
+ robots_robotsPresent: "robots.txt present",
13
+ robots_noLocalizedUrlsForgotten: "robots.txt keeps locale paths crawlable",
14
+ sitemap_sitemapPresent: "sitemap.xml present",
15
+ sitemap_noLocalizedUrlsForgotten: "sitemap lists every locale",
16
+ sitemap_hasAlternates: "sitemap has alternate links",
17
+ sitemap_hasXDefault: "sitemap has x-default",
18
+ url_htmlLang: "html lang attribute",
19
+ url_htmlDir: "html dir attribute",
20
+ url_hasCanonical: "canonical link",
21
+ url_hreflang: "hreflang tags",
22
+ url_hasLocalizedLinks: "localized internal links",
23
+ url_hasXDefault: "x-default hreflang",
24
+ url_allAnchorsLocalized: "all internal links localized",
25
+ url_currentLocale: "current locale detected",
26
+ url_unusedBundleContent: "unused bundle locale content"
27
+ };
28
+ const statusIcon = (status) => {
29
+ if (status === "success") return (0, _intlayer_config_logger.colorize)("✓", _intlayer_config_colors.GREEN);
30
+ if (status === "warning") return (0, _intlayer_config_logger.colorize)("⚠", _intlayer_config_colors.YELLOW);
31
+ return (0, _intlayer_config_logger.colorize)("✗", _intlayer_config_colors.RED);
32
+ };
33
+ const scoreColor = (score) => {
34
+ if (score >= 80) return _intlayer_config_colors.GREEN;
35
+ if (score >= 50) return _intlayer_config_colors.YELLOW;
36
+ return _intlayer_config_colors.RED;
37
+ };
38
+ /**
39
+ * Scan a website for i18n/SEO health and bundle weight, printing a formatted
40
+ * report (or JSON with `--json`).
41
+ *
42
+ * @example
43
+ * ```sh
44
+ * npx intlayer scan https://intlayer.org
45
+ * ```
46
+ */
47
+ const scan = async (url, options = {}) => {
48
+ const appLogger = (0, _intlayer_config_logger.getAppLogger)((0, _intlayer_config_node.getConfiguration)(options.configOptions));
49
+ const result = await (0, _intlayer_chokidar_scan.scanWebsite)(url, { deep: options.deep });
50
+ if (options.json) {
51
+ appLogger(JSON.stringify(result, null, 2));
52
+ return;
53
+ }
54
+ appLogger(`\n🔍 Scanned ${(0, _intlayer_config_logger.colorize)(result.url, _intlayer_config_colors.GREY_LIGHT)} ${(0, _intlayer_config_logger.colorize)(`(${result.mode} mode)`, _intlayer_config_colors.GREY)}\n`);
55
+ appLogger(`Score: ${(0, _intlayer_config_logger.colorize)(`${result.score}/100`, scoreColor(result.score))}`);
56
+ appLogger(`Page size: ${(0, _intlayer_config_logger.colorize)((0, _intlayer_chokidar_scan.formatSize)(result.totalPageSize), _intlayer_config_colors.BLUE)} ${(0, _intlayer_config_logger.colorize)(`(HTML ${(0, _intlayer_chokidar_scan.formatSize)(result.htmlSize)})`, _intlayer_config_colors.GREY)}`);
57
+ if (result.locales.length > 0) appLogger(`Locales: ${(0, _intlayer_config_logger.colorize)(result.locales.join(", "), _intlayer_config_colors.GREEN)}`);
58
+ appLogger("\nChecks:");
59
+ for (const event of result.events) {
60
+ const type = event.type.split("\\")[0];
61
+ if (!type) continue;
62
+ const label = checkLabels[type] ?? type;
63
+ appLogger(` ${statusIcon(event.status)} ${label}`);
64
+ }
65
+ if (result.bundle && result.bundle.totalLocaleSize > 0) {
66
+ const { bundle } = result;
67
+ appLogger("\nBundle locale weight:");
68
+ appLogger(` Translations shipped: ${(0, _intlayer_config_logger.colorize)((0, _intlayer_chokidar_scan.formatSize)(bundle.totalLocaleSize), _intlayer_config_colors.BLUE)}`);
69
+ appLogger(` Unused (other locales): ${(0, _intlayer_config_logger.colorize)(`${(0, _intlayer_chokidar_scan.formatSize)(bundle.totalUnusedLocaleSize)} (${bundle.unusedPercentOfLocale}%)`, bundle.unusedPercentOfLocale > 30 ? _intlayer_config_colors.RED : bundle.unusedPercentOfLocale > 0 ? _intlayer_config_colors.YELLOW : _intlayer_config_colors.GREEN)}`);
70
+ }
71
+ };
72
+
73
+ //#endregion
74
+ exports.scan = scan;
75
+ //# sourceMappingURL=scan.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scan.cjs","names":["ANSIColors"],"sources":["../../src/scan.ts"],"sourcesContent":["import {\n formatSize,\n type ScanEvent,\n scanWebsite,\n} from '@intlayer/chokidar/scan';\nimport * as ANSIColors from '@intlayer/config/colors';\nimport { colorize, getAppLogger } from '@intlayer/config/logger';\nimport { getConfiguration } from '@intlayer/config/node';\nimport type { ConfigurationOptions } from './cli';\n\n/** Options accepted by the {@link scan} command. */\nexport type ScanCommandOptions = {\n /** Disable the deeper puppeteer-based render scan. */\n deep?: boolean;\n /** Output the raw {@link ScanResult} as JSON instead of formatted text. */\n json?: boolean;\n configOptions?: ConfigurationOptions;\n};\n\n/** Human-readable labels for each scorable check type. */\nconst checkLabels: Record<string, string> = {\n robots_robotsPresent: 'robots.txt present',\n robots_noLocalizedUrlsForgotten: 'robots.txt keeps locale paths crawlable',\n sitemap_sitemapPresent: 'sitemap.xml present',\n sitemap_noLocalizedUrlsForgotten: 'sitemap lists every locale',\n sitemap_hasAlternates: 'sitemap has alternate links',\n sitemap_hasXDefault: 'sitemap has x-default',\n url_htmlLang: 'html lang attribute',\n url_htmlDir: 'html dir attribute',\n url_hasCanonical: 'canonical link',\n url_hreflang: 'hreflang tags',\n url_hasLocalizedLinks: 'localized internal links',\n url_hasXDefault: 'x-default hreflang',\n url_allAnchorsLocalized: 'all internal links localized',\n url_currentLocale: 'current locale detected',\n url_unusedBundleContent: 'unused bundle locale content',\n};\n\nconst statusIcon = (status: ScanEvent['status']): string => {\n if (status === 'success') return colorize('✓', ANSIColors.GREEN);\n if (status === 'warning') return colorize('⚠', ANSIColors.YELLOW);\n return colorize('✗', ANSIColors.RED);\n};\n\nconst scoreColor = (score: number) => {\n if (score >= 80) return ANSIColors.GREEN;\n if (score >= 50) return ANSIColors.YELLOW;\n return ANSIColors.RED;\n};\n\n/**\n * Scan a website for i18n/SEO health and bundle weight, printing a formatted\n * report (or JSON with `--json`).\n *\n * @example\n * ```sh\n * npx intlayer scan https://intlayer.org\n * ```\n */\nexport const scan = async (\n url: string,\n options: ScanCommandOptions = {}\n): Promise<void> => {\n const configuration = getConfiguration(options.configOptions);\n const appLogger = getAppLogger(configuration);\n\n const result = await scanWebsite(url, { deep: options.deep });\n\n if (options.json) {\n appLogger(JSON.stringify(result, null, 2));\n return;\n }\n\n appLogger(\n `\\n🔍 Scanned ${colorize(result.url, ANSIColors.GREY_LIGHT)} ${colorize(\n `(${result.mode} mode)`,\n ANSIColors.GREY\n )}\\n`\n );\n\n appLogger(\n `Score: ${colorize(`${result.score}/100`, scoreColor(result.score))}`\n );\n appLogger(\n `Page size: ${colorize(formatSize(result.totalPageSize), ANSIColors.BLUE)} ${colorize(\n `(HTML ${formatSize(result.htmlSize)})`,\n ANSIColors.GREY\n )}`\n );\n if (result.locales.length > 0) {\n appLogger(\n `Locales: ${colorize(result.locales.join(', '), ANSIColors.GREEN)}`\n );\n }\n\n appLogger('\\nChecks:');\n for (const event of result.events) {\n const type = event.type.split('\\\\')[0];\n\n if (!type) continue;\n\n const label = checkLabels[type] ?? type;\n appLogger(` ${statusIcon(event.status)} ${label}`);\n }\n\n if (result.bundle && result.bundle.totalLocaleSize > 0) {\n const { bundle } = result;\n appLogger('\\nBundle locale weight:');\n appLogger(\n ` Translations shipped: ${colorize(formatSize(bundle.totalLocaleSize), ANSIColors.BLUE)}`\n );\n appLogger(\n ` Unused (other locales): ${colorize(\n `${formatSize(bundle.totalUnusedLocaleSize)} (${bundle.unusedPercentOfLocale}%)`,\n bundle.unusedPercentOfLocale > 30\n ? ANSIColors.RED\n : bundle.unusedPercentOfLocale > 0\n ? ANSIColors.YELLOW\n : ANSIColors.GREEN\n )}`\n );\n }\n};\n"],"mappings":";;;;;;;;;;AAoBA,MAAM,cAAsC;CAC1C,sBAAsB;CACtB,iCAAiC;CACjC,wBAAwB;CACxB,kCAAkC;CAClC,uBAAuB;CACvB,qBAAqB;CACrB,cAAc;CACd,aAAa;CACb,kBAAkB;CAClB,cAAc;CACd,uBAAuB;CACvB,iBAAiB;CACjB,yBAAyB;CACzB,mBAAmB;CACnB,yBAAyB;CAC1B;AAED,MAAM,cAAc,WAAwC;AAC1D,KAAI,WAAW,UAAW,8CAAgB,KAAKA,wBAAW,MAAM;AAChE,KAAI,WAAW,UAAW,8CAAgB,KAAKA,wBAAW,OAAO;AACjE,8CAAgB,KAAKA,wBAAW,IAAI;;AAGtC,MAAM,cAAc,UAAkB;AACpC,KAAI,SAAS,GAAI,QAAOA,wBAAW;AACnC,KAAI,SAAS,GAAI,QAAOA,wBAAW;AACnC,QAAOA,wBAAW;;;;;;;;;;;AAYpB,MAAa,OAAO,OAClB,KACA,UAA8B,EAAE,KACd;CAElB,MAAM,kGADiC,QAAQ,cACH,CAAC;CAE7C,MAAM,SAAS,+CAAkB,KAAK,EAAE,MAAM,QAAQ,MAAM,CAAC;AAE7D,KAAI,QAAQ,MAAM;AAChB,YAAU,KAAK,UAAU,QAAQ,MAAM,EAAE,CAAC;AAC1C;;AAGF,WACE,sDAAyB,OAAO,KAAKA,wBAAW,WAAW,CAAC,yCAC1D,IAAI,OAAO,KAAK,SAChBA,wBAAW,KACZ,CAAC,IACH;AAED,WACE,gDAAmB,GAAG,OAAO,MAAM,OAAO,WAAW,OAAO,MAAM,CAAC,GACpE;AACD,WACE,4FAAkC,OAAO,cAAc,EAAEA,wBAAW,KAAK,CAAC,yCACxE,iDAAoB,OAAO,SAAS,CAAC,IACrCA,wBAAW,KACZ,GACF;AACD,KAAI,OAAO,QAAQ,SAAS,EAC1B,WACE,kDAAqB,OAAO,QAAQ,KAAK,KAAK,EAAEA,wBAAW,MAAM,GAClE;AAGH,WAAU,YAAY;AACtB,MAAK,MAAM,SAAS,OAAO,QAAQ;EACjC,MAAM,OAAO,MAAM,KAAK,MAAM,KAAK,CAAC;AAEpC,MAAI,CAAC,KAAM;EAEX,MAAM,QAAQ,YAAY,SAAS;AACnC,YAAU,KAAK,WAAW,MAAM,OAAO,CAAC,GAAG,QAAQ;;AAGrD,KAAI,OAAO,UAAU,OAAO,OAAO,kBAAkB,GAAG;EACtD,MAAM,EAAE,WAAW;AACnB,YAAU,0BAA0B;AACpC,YACE,yGAA+C,OAAO,gBAAgB,EAAEA,wBAAW,KAAK,GACzF;AACD,YACE,mEACE,2CAAc,OAAO,sBAAsB,CAAC,IAAI,OAAO,sBAAsB,KAC7E,OAAO,wBAAwB,KAC3BA,wBAAW,MACX,OAAO,wBAAwB,IAC7BA,wBAAW,SACXA,wBAAW,MAClB,GACF"}
@@ -0,0 +1,44 @@
1
+ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
2
+
3
+ //#region src/utils/formatLineRanges.ts
4
+ /**
5
+ * Formats a list of line numbers into a compact, human-readable string where
6
+ * runs of consecutive lines are collapsed into ranges.
7
+ *
8
+ * The input is sorted and de-duplicated first, so callers don't need to
9
+ * pre-process it. A run of a single line is printed as the bare number; a run
10
+ * of two or more consecutive lines is printed as `start-end`.
11
+ *
12
+ * @example
13
+ * formatLineRanges([2, 3, 4, 5, 333, 412, 413, 414]);
14
+ * // → '2-5, 333, 412-414'
15
+ *
16
+ * @param lineNumbers - The (possibly unsorted, possibly duplicated) line numbers.
17
+ * @param separator - String inserted between groups. Defaults to `', '`.
18
+ * @returns The grouped string, or an empty string when no lines are provided.
19
+ */
20
+ const formatLineRanges = (lineNumbers, separator = ", ") => {
21
+ const sortedUniqueLines = [...new Set(lineNumbers)].sort((a, b) => a - b);
22
+ if (sortedUniqueLines.length === 0) return "";
23
+ const groups = [];
24
+ let rangeStart = sortedUniqueLines[0];
25
+ let rangeEnd = rangeStart;
26
+ const pushGroup = () => {
27
+ groups.push(rangeStart === rangeEnd ? `${rangeStart}` : `${rangeStart}-${rangeEnd}`);
28
+ };
29
+ for (const lineNumber of sortedUniqueLines.slice(1)) {
30
+ if (lineNumber === rangeEnd + 1) {
31
+ rangeEnd = lineNumber;
32
+ continue;
33
+ }
34
+ pushGroup();
35
+ rangeStart = lineNumber;
36
+ rangeEnd = lineNumber;
37
+ }
38
+ pushGroup();
39
+ return groups.join(separator);
40
+ };
41
+
42
+ //#endregion
43
+ exports.formatLineRanges = formatLineRanges;
44
+ //# sourceMappingURL=formatLineRanges.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"formatLineRanges.cjs","names":[],"sources":["../../../src/utils/formatLineRanges.ts"],"sourcesContent":["/**\n * Formats a list of line numbers into a compact, human-readable string where\n * runs of consecutive lines are collapsed into ranges.\n *\n * The input is sorted and de-duplicated first, so callers don't need to\n * pre-process it. A run of a single line is printed as the bare number; a run\n * of two or more consecutive lines is printed as `start-end`.\n *\n * @example\n * formatLineRanges([2, 3, 4, 5, 333, 412, 413, 414]);\n * // → '2-5, 333, 412-414'\n *\n * @param lineNumbers - The (possibly unsorted, possibly duplicated) line numbers.\n * @param separator - String inserted between groups. Defaults to `', '`.\n * @returns The grouped string, or an empty string when no lines are provided.\n */\nexport const formatLineRanges = (\n lineNumbers: number[],\n separator = ', '\n): string => {\n const sortedUniqueLines = [...new Set(lineNumbers)].sort((a, b) => a - b);\n\n if (sortedUniqueLines.length === 0) return '';\n\n const groups: string[] = [];\n let rangeStart = sortedUniqueLines[0]!;\n let rangeEnd = rangeStart;\n\n const pushGroup = (): void => {\n groups.push(\n rangeStart === rangeEnd ? `${rangeStart}` : `${rangeStart}-${rangeEnd}`\n );\n };\n\n for (const lineNumber of sortedUniqueLines.slice(1)) {\n if (lineNumber === rangeEnd + 1) {\n // Still inside the current consecutive run.\n rangeEnd = lineNumber;\n continue;\n }\n\n // Gap detected: close the current run and start a new one.\n pushGroup();\n rangeStart = lineNumber;\n rangeEnd = lineNumber;\n }\n\n pushGroup();\n\n return groups.join(separator);\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;AAgBA,MAAa,oBACX,aACA,YAAY,SACD;CACX,MAAM,oBAAoB,CAAC,GAAG,IAAI,IAAI,YAAY,CAAC,CAAC,MAAM,GAAG,MAAM,IAAI,EAAE;AAEzE,KAAI,kBAAkB,WAAW,EAAG,QAAO;CAE3C,MAAM,SAAmB,EAAE;CAC3B,IAAI,aAAa,kBAAkB;CACnC,IAAI,WAAW;CAEf,MAAM,kBAAwB;AAC5B,SAAO,KACL,eAAe,WAAW,GAAG,eAAe,GAAG,WAAW,GAAG,WAC9D;;AAGH,MAAK,MAAM,cAAc,kBAAkB,MAAM,EAAE,EAAE;AACnD,MAAI,eAAe,WAAW,GAAG;AAE/B,cAAW;AACX;;AAIF,aAAW;AACX,eAAa;AACb,aAAW;;AAGb,YAAW;AAEX,QAAO,OAAO,KAAK,UAAU"}
package/dist/esm/cli.mjs CHANGED
@@ -5,6 +5,7 @@ import { listContentDeclaration } from "./listContentDeclaration.mjs";
5
5
  import { init } from "./init.mjs";
6
6
  import { startEditor } from "./editor.mjs";
7
7
  import { watchContentDeclaration } from "./watch.mjs";
8
+ import { scan } from "./scan.mjs";
8
9
  import { login } from "./auth/login.mjs";
9
10
  import { build } from "./build.mjs";
10
11
  import { bundle } from "./bundle.mjs";
@@ -165,7 +166,10 @@ const setAPI = () => {
165
166
  /**
166
167
  * INIT
167
168
  */
168
- const initCmd = program.command("init").description("Initialize Intlayer in the project").option("--project-root [projectRoot]", "Project root directory").option("--no-gitignore", "Do not add .intlayer to .gitignore").action((options) => init(options.projectRoot, { noGitignore: options.gitignore === false }));
169
+ const initCmd = program.command("init").description("Initialize Intlayer in the project").option("--project-root [projectRoot]", "Project root directory").option("--no-gitignore", "Do not add .intlayer to .gitignore").option("--no-github-actions", "Do not scaffold the fill and test GitHub Actions workflows").action((options) => init(options.projectRoot, {
170
+ noGitignore: options.gitignore === false,
171
+ noGithubActions: options.githubActions === false
172
+ }));
169
173
  initCmd.command("skills").description("Initialize Intlayer skills in the project").option("--project-root [projectRoot]", "Project root directory").action((options) => initSkills(options.projectRoot));
170
174
  initCmd.command("mcp").description("Initialize Intlayer MCP server in the project").option("--project-root [projectRoot]", "Project root directory").action((options) => initMCP(options.projectRoot));
171
175
  /**
@@ -412,7 +416,7 @@ const setAPI = () => {
412
416
  skipIfModifiedAfter: options.skipIfModifiedAfter,
413
417
  skipIfExists: options.skipIfExists
414
418
  }));
415
- const reviewProgram = docProgram.command("review").description("Review the documentation");
419
+ const reviewProgram = docProgram.command("review").description("Review the documentation").option("--log", "Log-only mode. Do not translate with AI; instead log the blocks that need attention (with line numbers and content) for the base and target locales, to help another agent generate the translations.");
416
420
  applyConfigOptions(reviewProgram);
417
421
  applyAIOptions(reviewProgram);
418
422
  applyGitOptions(reviewProgram);
@@ -429,7 +433,8 @@ const setAPI = () => {
429
433
  customInstructions: options.customInstructions,
430
434
  skipIfModifiedBefore: options.skipIfModifiedBefore,
431
435
  skipIfModifiedAfter: options.skipIfModifiedAfter,
432
- skipIfExists: options.skipIfExists
436
+ skipIfExists: options.skipIfExists,
437
+ log: options.log
433
438
  }));
434
439
  const searchProgram = docProgram.command("search").description("Search the documentation").argument("<query>", "Search query").option("--limit [limit]", "Limit the number of results", "10");
435
440
  applyConfigOptions(searchProgram);
@@ -473,6 +478,16 @@ const setAPI = () => {
473
478
  configOptions: extractConfigOptions(options)
474
479
  });
475
480
  }));
481
+ /**
482
+ * SCAN
483
+ */
484
+ const scanCmd = program.command("scan").description("Scan a website to measure its page size and audit its i18n / SEO health").argument("<url>", "URL of the website to scan").option("--no-deep", "Disable the deeper puppeteer-based render scan").option("--json", "Output the results as JSON");
485
+ applyConfigOptions(scanCmd);
486
+ scanCmd.action((url, options) => scan(url, {
487
+ deep: options.deep,
488
+ json: options.json,
489
+ configOptions: extractConfigOptions(options)
490
+ }));
476
491
  program.parse(process.argv);
477
492
  /**
478
493
  * CI / AUTOMATION
@@ -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/config';\nimport { Command } from 'commander';\nimport { login } from './auth/login';\nimport { build } from './build';\nimport { bundle } from './bundle';\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 .option('--no-gitignore', 'Do not add .intlayer to .gitignore')\n .action((options) =>\n init(options.projectRoot, {\n noGitignore: options.gitignore === false,\n })\n );\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('--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 configOptions: extractConfigOptions(options),\n codeOnly: options.codeOnly,\n declarationOnly: options.declarationOnly,\n });\n });\n\n /**\n * STANDALONE\n */\n const bundleCmd = program\n .command('standalone')\n .description('Create a standalone bundle of the application content')\n .option(\n '-o, --outfile [outfile]',\n 'Output file for the bundle',\n 'intlayer-bundle.js'\n )\n .option('--packages [packages...]', 'List of packages to bundle')\n .option('--version [version]', 'Version of the packages to bundle')\n .option('--minify', 'Minify the output')\n .option('--platform [platform]', 'Target platform', 'browser')\n .option('--format [format]', 'Output format', 'esm')\n .action((options) => {\n bundle({\n outfile: options.outfile,\n bundlePackages: options.packages,\n version: options.version,\n minify: options.minify,\n platform: options.platform,\n format: options.format,\n configOptions: extractConfigOptions(options),\n });\n });\n\n applyConfigOptions(bundleCmd);\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":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAwCA,MAAM,aAAa,OAAO,OAAO,KAAK,QAAQ;AAE9C,MAAa,UAAU,aACnBA,UAAY,cAAc,OAAO,KAAK,IAAI,CAAC,GAC3C;AAEJ,MAAM,cAAc,qBAAqB,QAAQ;AAOjD,MAAM,uBAAuB;CAC3B,CAAC,wBAAwB,mBAAmB;CAC5C,CAAC,mBAAmB,cAAc;CAClC,CAAC,wBAAwB,iBAAiB;CAC1C,CAAC,wBAAwB,WAAW;CACpC,GAAG,CATH,CAAC,aAAa,sCAAsC,EACpD,CAAC,qBAAqB,SAAS,CAQlB;CACd;AAED,MAAM,YAAY;CAChB,CAAC,yBAAyB,WAAW;CACrC,CAAC,+BAA+B,cAAc;CAC9C,CAAC,mBAAmB,QAAQ;CAC5B,CAAC,sBAAsB,mBAAmB;CAC1C,CAAC,4BAA4B,gBAAgB;CAC7C,CAAC,8CAA8C,sBAAsB;CACrE,CAAC,4CAA4C,qBAAqB;CACnE;AAED,MAAM,aAAa;CACjB,CAAC,wBAAwB,kDAAkD;CAC3E,CAAC,iCAAiC,oBAAoB;CACtD,CAAC,uCAAuC,uBAAuB;CAC/D,CAAC,+BAA+B,cAAc;CAC9C,CAAC,yBAAyB,WAAW;CACrC,CAAC,2BAA2B,YAAY;CACzC;AAED,MAAM,0BAA0B,SAAiB,SAC/C,KAAK,QAAQ,QAAQ,QAAQ,SAAiC,OAAU;;;;AAK1E,MAAM,gBAAgB,SAAkB,YAAwB;AAC9D,SAAQ,SAAS,CAAC,MAAM,iBAAiB;AACvC,UAAQ,OAAO,MAAM,YAAY;GACjC;AACF,QAAO;;AAGT,MAAM,mBAAkD,QACtD,OAAO,YACL,OAAO,QAAQ,IAAI,CAAC,QAAQ,CAAC,GAAG,WAAW,UAAU,OAAU,CAChE;AAEH,MAAM,sBAAsB,YAC1B,aAAa,SAAS,qBAAqB;AAC7C,MAAM,kBAAkB,YAAqB,aAAa,SAAS,UAAU;AAC7E,MAAM,mBAAmB,YAAqB,aAAa,SAAS,WAAW;AAE/E,MAAM,oBAAoB,YAA8C;CACtE,MAAM,EACJ,QACA,UACA,OACA,aACA,oBACA,cACA,sBACE;CAEJ,MAAM,gBAAgB,kBAAkB;CAExC,MAAM,EAAE,OAAO;AAEf,QAAO,gBAAgB;EACrB,GAAG;EACH,QAAQ,UAAU,cAAc,IAAI;EACpC,UAAU,YAAa,cAAc,IAAI;EACzC,OAAO,SAAS,cAAc,IAAI;EAClC,aAAa,eAAe,cAAc,IAAI;EAC9C,oBACE,sBAAsB,cAAc,IAAI;EAC1C,cAAc,gBAAiB,cAAc,IAAY;EACzD,mBAAmB,qBAAqB,cAAc,IAAI;EAC3D,CAAC;;AAYJ,MAAM,gBAAsC;CAC1C;CACA;CACA;CACA;CACA;CACA;CACD;AAED,MAAM,qBACJ,YACoC;AAKpC,KAJwB,uBAAuB,SAAS,cAEnB,CAAC,WAAW,EAE9B,QAAO;CAE1B,MAAM,EACJ,SACA,aACA,gBACA,aACA,UACA,cACE;AASJ,QAAO,gBAAgB;EACrB,MARW;GACX,WAAW;GACX,eAAe;GACf,YAAY;GACZ,aAAa;GACd,CAAC,OAAO,QAGH;EACJ,SAAS;EACT,YAAY;EACZ,UAAU;EACX,CAAC;;AAgBJ,MAAM,0BAA0D;CAC9D;CACA;CACA;CACA;CACA;CACA;CACD;AAED,MAAM,wBACJ,YACwC;AAQxC,KAPwB,uBACtB,SACA,wBAGmC,CAAC,WAAW,EAG/C;CAGF,MAAM,EAAE,SAAS,KAAK,SAAS,SAAS,SAAS,eAAe;CAEhE,MAAM,MAAM,gBAAgB,EAC1B,MACE,OAAO,YAAY,cACf,UACE,YACA,YACF,QACP,CAAC;CAEF,MAAM,QAAQ,gBAAgB,EAC5B,YACD,CAAC;CAEF,MAAM,WAAiC,gBAAgB;EACrD,KACE,OAAO,KAAK,IAAI,CAAC,SAAS,IACrB,MACD;EACN,OAAO,OAAO,KAAK,MAAM,CAAC,SAAS,IAAI,QAAQ;EAChD,CAAC;AAEF,QAAO,gBAAgB;EACrB;EACA;EACA;EACA,UAAU,OAAO,KAAK,SAAS,CAAC,SAAS,IAAI,WAAW;EACxD,OAAO,OAAO,YAAY,cAAc,CAAC,UAAU;EACpD,CAAC;;;;;;;;;;AAWJ,MAAa,eAAwB;AAGnC,KAAI,CAFkB,QAAQ,KAAK,SAAS,SAE1B,CAChB,WAAU,GAAG;CAGf,MAAM,UAAU,IAAI,SAAS;AAE7B,SAAQ,QAAQ,YAAY,QAAS,CAAC,YAAY,eAAe;AAGjE,SACG,QAAQ,UAAU,CAClB,YAAY,iCAAiC,CAC7C,aAAa;AAGZ,UAAQ,IAAI,YAAY,WAAW,UAAU;GAC7C;;;;CAKJ,MAAM,WAAW,QACd,QAAQ,QAAQ,CAChB,YAAY,oBAAoB,CAChC,OAAO,sBAAsB,UAAU;AAE1C,oBAAmB,SAAS;AAE5B,UAAS,QAAQ,YAAY;EAC3B,MAAM,gBAAgB,qBAAqB,QAAQ,IAAI,EACrD,UAAU,EACR,KAAK;GACH,QAAQ;GACR,MAAM;GACP,EACF,EACF;AAED,SAAO,MAAM;GACX,QAAQ,QAAQ;GAChB;GACD,CAAC;GACF;;;;CAKF,MAAM,UAAU,QACb,QAAQ,OAAO,CACf,YAAY,qCAAqC,CACjD,OAAO,gCAAgC,yBAAyB,CAChE,OAAO,kBAAkB,qCAAqC,CAC9D,QAAQ,YACP,KAAK,QAAQ,aAAa,EACxB,aAAa,QAAQ,cAAc,OACpC,CAAC,CACH;AAEH,SACG,QAAQ,SAAS,CACjB,YAAY,4CAA4C,CACxD,OAAO,gCAAgC,yBAAyB,CAChE,QAAQ,YAAY,WAAW,QAAQ,YAAY,CAAC;AAEvD,SACG,QAAQ,MAAM,CACd,YAAY,gDAAgD,CAC5D,OAAO,gCAAgC,yBAAyB,CAChE,QAAQ,YAAY,QAAQ,QAAQ,YAAY,CAAC;;;;CAMpD,MAAM,sBAAsB,QACzB,QAAQ,aAAa,CACrB,MAAM,eAAe,CACrB,MAAM,MAAM,CACZ,YAAY,0BAA0B;CAGzC,MAAM,eAAe;EACnB,aAAa;EACb,SAAS;GACP,CAAC,eAAe,oBAAoB;GACpC,CAAC,kBAAkB,wBAAwB;GAC3C,CAAC,oBAAoB,2CAA2C;GAChE,CAAC,iBAAiB,uCAAuC;GAC1D;EACF;CAGD,MAAM,uBAAuB,oBAC1B,QAAQ,QAAQ,CAChB,YAAY,aAAa,YAAY;AAExC,cAAa,sBAAsB,aAAa,QAAQ;AACxD,oBAAmB,qBAAqB;AACxC,sBAAqB,QAAQ,YAAY;AACvC,QAAM;GACJ,GAAG;GACH,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GACF;CAGF,MAAM,eAAe,QAClB,QAAQ,QAAQ,CAChB,YAAY,aAAa,YAAY;AAExC,cAAa,cAAc,aAAa,QAAQ;AAChD,oBAAmB,aAAa;AAChC,cAAa,QAAQ,YAAY;AAC/B,QAAM;GACJ,GAAG;GACH,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GACF;CAEF,MAAM,eAAe;EACnB,aAAa;EACb,SAAS,CAAC,CAAC,oBAAoB,2CAA2C,CAAC;EAC5E;CAGD,MAAM,uBAAuB,oBAC1B,QAAQ,QAAQ,CAChB,YAAY,aAAa,YAAY;AAExC,cAAa,sBAAsB,aAAa,QAAQ;AACxD,oBAAmB,qBAAqB;AACxC,sBAAqB,QAAQ,YAAY;AACvC,0BAAwB;GACtB,GAAG;GACH,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GACF;CAGF,MAAM,eAAe,QAClB,QAAQ,QAAQ,CAChB,YAAY,aAAa,YAAY;AAExC,cAAa,cAAc,aAAa,QAAQ;AAChD,oBAAmB,aAAa;AAChC,cAAa,QAAQ,YAAY;AAC/B,0BAAwB;GACtB,GAAG;GACH,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GACF;CAGF,MAAM,cAAc;EAClB,aAAa;EACb,SAAS;GACP,CAAC,+BAA+B,iCAAiC;GACjE,CACE,yBACA,4DACD;GACD,CAAC,kCAAkC,oCAAoC;GAEvE,CACE,gCACA,4CACD;GACF;EACF;CAGD,MAAM,sBAAsB,oBACzB,QAAQ,OAAO,CACf,YAAY,YAAY,YAAY;AAEvC,cAAa,qBAAqB,YAAY,QAAQ;AACtD,oBAAmB,oBAAoB;AACvC,qBAAoB,QAAQ,YAAY;EAEtC,MAAM,eAAe,CACnB,GAAI,QAAQ,gBAAgB,EAAE,EAC9B,GAAI,QAAQ,cAAc,EAAE,CAC7B;AAED,OAAK;GACH,GAAG;GACH,cAAc,aAAa,SAAS,IAAI,eAAe;GACvD,eAAe;IACb,GAAG,QAAQ;IACX,SAAS,QAAQ;IAClB;GACF,CAAC;GACF;CAGF,MAAM,cAAc,QACjB,QAAQ,OAAO,CACf,YAAY,YAAY,YAAY;AAEvC,cAAa,aAAa,YAAY,QAAQ;AAC9C,oBAAmB,YAAY;AAC/B,aAAY,QAAQ,YAAY;EAE9B,MAAM,eAAe,CACnB,GAAI,QAAQ,gBAAgB,EAAE,EAC9B,GAAI,QAAQ,cAAc,EAAE,CAC7B;AAED,OAAK;GACH,GAAG;GACH,cAAc,aAAa,SAAS,IAAI,eAAe;GACvD,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GACF;CAGF,MAAM,cAAc;EAClB,aACE;EACF,SAAS;GACP,CAAC,+BAA+B,iCAAiC;GACjE,CACE,yBACA,4DACD;GACD,CACE,kCACA,8CACD;GACD,CACE,gCACA,4CACD;GAED,CACE,4BACA,sDACD;GACD,CACE,0BACA,oDACD;GACD,CACE,mBACA,qLACD;GACF;EACF;CAGD,MAAM,sBAAsB,oBACzB,QAAQ,OAAO,CACf,YAAY,YAAY,YAAY;AAEvC,cAAa,qBAAqB,YAAY,QAAQ;AACtD,oBAAmB,oBAAoB;AACvC,iBAAgB,oBAAoB;AAEpC,qBAAoB,QAAQ,YAAY;EAEtC,MAAM,eAAe,CACnB,GAAI,QAAQ,gBAAgB,EAAE,EAC9B,GAAI,QAAQ,cAAc,EAAE,CAC7B;AAED,SAAO,KAAK;GACV,GAAG;GACH,cAAc,aAAa,SAAS,IAAI,eAAe;GACvD,YAAY,kBAAkB,QAAQ;GACtC,eAAe,qBAAqB,QAAQ;GAC7C,CAAgB;GACjB;CAGF,MAAM,cAAc,QACjB,QAAQ,OAAO,CACf,YAAY,YAAY,YAAY;AAEvC,cAAa,aAAa,YAAY,QAAQ;AAC9C,oBAAmB,YAAY;AAC/B,iBAAgB,YAAY;AAE5B,aAAY,QAAQ,YAAY;EAE9B,MAAM,eAAe,CACnB,GAAI,QAAQ,gBAAgB,EAAE,EAC9B,GAAI,QAAQ,cAAc,EAAE,CAC7B;AAED,SAAO,KAAK;GACV,GAAG;GACH,cAAc,aAAa,SAAS,IAAI,eAAe;GACvD,YAAY,kBAAkB,QAAQ;GACtC,eAAe,qBAAqB,QAAQ;GAC7C,CAAgB;GACjB;;;;CAOF,MAAM,uBAAuB,QAC1B,QAAQ,gBAAgB,CACxB,MAAM,SAAS,CACf,MAAM,OAAO,CACb,YAAY,2BAA2B;CAE1C,MAAM,eAAe,qBAClB,QAAQ,MAAM,CACd,YAAY,wBAAwB;AAEvC,oBAAmB,aAAa;AAChC,cAAa,QAAQ,YAAY;AAC/B,YAAU;GACR,GAAG;GACH,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GACF;CAGF,MAAM,gBAAgB,qBACnB,QAAQ,OAAO,CACf,YAAY,yBAAyB;AAExC,oBAAmB,cAAc;AACjC,eAAc,QAAQ,YAAY;AAChC,aAAW;GACT,GAAG;GACH,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GACF;AAqBF,CAfwB,QACrB,QAAQ,WAAW,CACnB,MAAM,UAAU,CAChB,YAAY,yBAEwB,CACpC,QAAQ,OAAO,CACf,YAAY,8CAA8C,CAC1D,OAAO,wBAAwB,gCAAgC,CAC/D,OACC,cACA,mEACD,CACA,OAAO,UAAU,6BAEL,CAAC,QAAQ,YAAY;AAClC,sBAAoB;GAClB,SAAS,QAAQ;GACjB,SAAS,QAAQ;GACjB,MAAM,QAAQ;GACf,CAAC;GACF;AAeF,CAZ4B,QACzB,QAAQ,gBAAgB,CACxB,MAAM,KAAK,CACX,YAAY,8CAA8C,CAC1D,OAAO,wBAAwB,gCAAgC,CAC/D,OACC,cACA,mEACD,CACA,OAAO,cAAc,uCAAuC,CAC5D,OAAO,UAAU,6BAED,CAAC,QAAQ,YAAY;AACtC,sBAAoB;GAClB,SAAS,QAAQ;GACjB,SAAS,QAAQ;GACjB,MAAM,QAAQ;GACd,UAAU,QAAQ;GACnB,CAAC;GACF;;;;CAMF,MAAM,iBAAiB,QACpB,QAAQ,UAAU,CAClB,YAAY,iCAAiC;AAEhD,gBACG,QAAQ,OAAO,CACf,YAAY,qCAAqC,CACjD,OAAO,UAAU,6BAA6B,CAC9C,OAAO,cAAc,uCAAuC,CAC5D,QAAQ,YAAY;AACnB,yBAAuB;GACrB,MAAM,QAAQ;GACd,UAAU,QAAQ;GACnB,CAAC;GACF;AAGJ,SACG,QAAQ,OAAO,CACf,YAAY,qCAAqC,CACjD,OAAO,UAAU,6BAA6B,CAC9C,OAAO,cAAc,uCAAuC,CAC5D,QAAQ,YAAY;AACnB,yBAAuB;GACrB,MAAM,QAAQ;GACd,UAAU,QAAQ;GACnB,CAAC;GACF;CAEJ,MAAM,cAAc,eACjB,QAAQ,OAAO,CACf,YAAY,yCAAyC,CACrD,OACC,mBACA,qLACD;AAEH,oBAAmB,YAAY;AAC/B,aAAY,QAAQ,YAAY;AAC9B,0BAAwB;GACtB,GAAG;GACH,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GACF;CAGF,MAAM,cAAc,QACjB,QAAQ,OAAO,CACf,YAAY,yCAAyC,CACrD,OACC,mBACA,qLACD;AAEH,oBAAmB,YAAY;AAC/B,aAAY,QAAQ,YAAY;AAC9B,0BAAwB;GACtB,GAAG;GACH,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GACF;CAEF,MAAM,cAAc,QACjB,QAAQ,OAAO,CACf,YAAY,wBAAwB,CACpC,OAAO,yBAAyB,mCAAmC,CACnE,OAAO,kCAAkC,kCAAkC,CAC3E,OACC,uCACA,iCACD,CACA,OACC,iBACA,kIACA,WACD,CACA,OAAO,wBAAwB,oCAAoC,CACnE,OACC,mBACA,uDACD,CACA,OACC,qCACA,wCACD,CACA,OACC,oCACA,oEACD,CACA,OACC,kCACA,4CACD,CACA,OACC,mBACA,qLACD,CACA,OACC,mBACA,4EACD;AAEH,oBAAmB,YAAY;AAC/B,gBAAe,YAAY;AAC3B,iBAAgB,YAAY;AAE5B,aAAY,QAAQ,YAAY;EAE9B,MAAM,OAAO,CAAC,GAAI,QAAQ,QAAQ,EAAE,EAAG,GAAI,QAAQ,OAAO,EAAE,CAAE;EAC9D,MAAM,eAAe,CACnB,GAAI,QAAQ,gBAAgB,EAAE,EAC9B,GAAI,QAAQ,eAAe,EAAE,CAC9B;EAED,MAAM,eAAe,CACnB,GAAI,QAAQ,gBAAgB,EAAE,EAC9B,GAAI,QAAQ,cAAc,EAAE,CAC7B;AAED,SAAO,KAAK;GACV,GAAG;GACH,MAAM,KAAK,SAAS,IAAI,OAAO;GAC/B,cAAc,aAAa,SAAS,IAAI,eAAe;GACvD,cAAc,aAAa,SAAS,IAAI,eAAe;GACvD,WAAW,iBAAiB,QAAQ;GACpC,YAAY,kBAAkB,QAAQ;GACtC,eAAe,qBAAqB,QAAQ;GAC7C,CAAgB;GACjB;;;;CAMF,MAAM,YAAY;EAChB,CAAC,iCAAiC,wBAAwB;EAC1D,CACE,oDACA,wBACD;EACD,CACE,kEACA,wCACD;EACD,CAAC,0BAA0B,UAAU;EACrC,CAAC,8BAA8B,cAAc;EAC7C,CACE,8CACA,wHACD;EACD,CACE,oDACA,4TACD;EACD,CACE,kDACA,4TACD;EACD,CAAC,oBAAoB,qCAAqC;EAC3D;CAED,MAAM,aAAa,QAChB,QAAQ,MAAM,CACd,YAAY,2BAA2B;CAE1C,MAAM,mBAAmB,WACtB,QAAQ,YAAY,CACpB,YAAY,8BAA8B;AAE7C,oBAAmB,iBAAiB;AACpC,gBAAe,iBAAiB;AAChC,iBAAgB,iBAAiB;AACjC,cAAa,kBAAkB,UAAU;AAEzC,kBAAiB,QAAQ,YACvB,aAAa;EACX,YAAY,QAAQ;EACpB,qBAAqB,QAAQ;EAC7B,SAAS,QAAQ;EACjB,YAAY,QAAQ;EACpB,WAAW,iBAAiB,QAAQ;EACpC,YAAY,kBAAkB,QAAQ;EACtC,6BAA6B,QAAQ;EACrC,eAAe,qBAAqB,QAAQ;EAC5C,oBAAoB,QAAQ;EAC5B,sBAAsB,QAAQ;EAC9B,qBAAqB,QAAQ;EAC7B,cAAc,QAAQ;EACvB,CAAC,CACH;CAED,MAAM,gBAAgB,WACnB,QAAQ,SAAS,CACjB,YAAY,2BAA2B;AAE1C,oBAAmB,cAAc;AACjC,gBAAe,cAAc;AAC7B,iBAAgB,cAAc;AAC9B,cAAa,eAAe,UAAU;AAEtC,eAAc,QAAQ,YACpB,UAAU;EACR,YAAY,QAAQ;EACpB,qBAAqB,QAAQ;EAC7B,SAAS,QAAQ;EACjB,YAAY,QAAQ;EACpB,WAAW,iBAAiB,QAAQ;EACpC,YAAY,kBAAkB,QAAQ;EACtC,6BAA6B,QAAQ;EACrC,eAAe,qBAAqB,QAAQ;EAC5C,oBAAoB,QAAQ;EAC5B,sBAAsB,QAAQ;EAC9B,qBAAqB,QAAQ;EAC7B,cAAc,QAAQ;EACvB,CAAC,CACH;CAED,MAAM,gBAAgB,WACnB,QAAQ,SAAS,CACjB,YAAY,2BAA2B,CACvC,SAAS,WAAW,eAAe,CACnC,OAAO,mBAAmB,+BAA+B,KAAK;AAEjE,oBAAmB,cAAc;AAEjC,eAAc,QAAQ,OAAO,YAC3B,UAAU;EACR;EACA,OAAO,QAAQ,QAAQ,SAAS,QAAQ,OAAO,GAAG,GAAG;EACrD,eAAe,qBAAqB,QAAQ;EAC7C,CAAC,CACH;;;;CAMD,MAAM,cAAc,CAClB,CAAC,oBAAoB,+CAA+C,CACrE;CAED,MAAM,UAAU,QACb,QAAQ,OAAO,CACf,YACC,+FACD;AAEH,cAAa,SAAS,YAAY;AAClC,oBAAmB,QAAQ;AAE3B,SAAQ,QAAQ,YAAY,SAAS,QAAQ,CAAC;CAU9C,MAAM,iBAJgB,QACnB,QAAQ,SAAS,CACjB,YAAY,2BAEqB,CACjC,QAAQ,QAAQ,CAChB,YAAY,mCAAmC;AAElD,oBAAmB,eAAe;AAElC,gBAAe,QAAQ,YAAY;AACjC,cAAY;GACV,KAAK,QAAQ;GACb,SAAS,QAAQ;GAClB,CAAC;GACF;AAYF,CAPuB,QACpB,QAAQ,UAAU,CAClB,MAAM,MAAM,CACZ,YACC,yFAGU,CACX,OAAO,yBAAyB,2BAA2B,CAC3D,OAAO,eAAe,mCAAmC,MAAM,CAC/D,OAAO,sBAAsB,qCAAqC,MAAM,CACxE,QAAQ,YAAY;AACnB,UAAQ;GACN,OAAO,QAAQ;GACf,eAAe,qBAAqB,QAAQ;GAC5C,UAAU,QAAQ;GAClB,iBAAiB,QAAQ;GAC1B,CAAC;GACF;AA8BJ,oBAzBkB,QACf,QAAQ,aAAa,CACrB,YAAY,wDAAwD,CACpE,OACC,2BACA,8BACA,qBACD,CACA,OAAO,4BAA4B,6BAA6B,CAChE,OAAO,uBAAuB,oCAAoC,CAClE,OAAO,YAAY,oBAAoB,CACvC,OAAO,yBAAyB,mBAAmB,UAAU,CAC7D,OAAO,qBAAqB,iBAAiB,MAAM,CACnD,QAAQ,YAAY;AACnB,SAAO;GACL,SAAS,QAAQ;GACjB,gBAAgB,QAAQ;GACxB,SAAS,QAAQ;GACjB,QAAQ,QAAQ;GAChB,UAAU,QAAQ;GAClB,QAAQ,QAAQ;GAChB,eAAe,qBAAqB,QAAQ;GAC7C,CAAC;GAGsB,CAAC;AAE7B,SAAQ,MAAM,QAAQ,KAAK;;;;;;AAO3B,SACG,QAAQ,KAAK,CACb,YACC,iJACD,CACA,SACC,gBACA,6DACD,CACA,oBAAoB,CACpB,QAAQ,SAAS;AAChB,QAAM,KAAK;GACX;AAEJ,QAAO"}
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 { bundle } from './bundle';\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 { scan } from './scan';\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 .option('--no-gitignore', 'Do not add .intlayer to .gitignore')\n .option(\n '--no-github-actions',\n 'Do not scaffold the fill and test GitHub Actions workflows'\n )\n .action((options) =>\n init(options.projectRoot, {\n noGitignore: options.gitignore === false,\n noGithubActions: options.githubActions === false,\n })\n );\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 .option(\n '--log',\n 'Log-only mode. Do not translate with AI; instead log the blocks that need attention (with line numbers and content) for the base and target locales, to help another agent generate the translations.'\n );\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 log: options.log,\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('--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 configOptions: extractConfigOptions(options),\n codeOnly: options.codeOnly,\n declarationOnly: options.declarationOnly,\n });\n });\n\n /**\n * STANDALONE\n */\n const bundleCmd = program\n .command('standalone')\n .description('Create a standalone bundle of the application content')\n .option(\n '-o, --outfile [outfile]',\n 'Output file for the bundle',\n 'intlayer-bundle.js'\n )\n .option('--packages [packages...]', 'List of packages to bundle')\n .option('--version [version]', 'Version of the packages to bundle')\n .option('--minify', 'Minify the output')\n .option('--platform [platform]', 'Target platform', 'browser')\n .option('--format [format]', 'Output format', 'esm')\n .action((options) => {\n bundle({\n outfile: options.outfile,\n bundlePackages: options.packages,\n version: options.version,\n minify: options.minify,\n platform: options.platform,\n format: options.format,\n configOptions: extractConfigOptions(options),\n });\n });\n\n applyConfigOptions(bundleCmd);\n\n /**\n * SCAN\n */\n const scanCmd = program\n .command('scan')\n .description(\n 'Scan a website to measure its page size and audit its i18n / SEO health'\n )\n .argument('<url>', 'URL of the website to scan')\n .option('--no-deep', 'Disable the deeper puppeteer-based render scan')\n .option('--json', 'Output the results as JSON');\n\n applyConfigOptions(scanCmd);\n\n scanCmd.action((url, options) =>\n scan(url, {\n deep: options.deep,\n json: options.json,\n configOptions: extractConfigOptions(options),\n })\n );\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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"}
@@ -4,6 +4,7 @@ import { liveSync } from "./liveSync.mjs";
4
4
  import { listContentDeclaration, listContentDeclarationRows } from "./listContentDeclaration.mjs";
5
5
  import { findProjectRoot, init } from "./init.mjs";
6
6
  import { startEditor } from "./editor.mjs";
7
+ import { scan } from "./scan.mjs";
7
8
  import { build } from "./build.mjs";
8
9
  import { bundle } from "./bundle.mjs";
9
10
  import { listMissingTranslations, listMissingTranslationsWithConfig } from "./test/listMissingTranslations.mjs";
@@ -18,4 +19,4 @@ import { searchDoc } from "./searchDoc.mjs";
18
19
  import { translateDoc } from "./translateDoc/translateDoc.mjs";
19
20
  import { dirname, setAPI } from "./cli.mjs";
20
21
 
21
- export { PLATFORM_OPTIONS, build, bundle, dirname, extract, fill, findProjectRoot, getDetectedPlatform, init, initSkills, listContentDeclaration, listContentDeclarationRows, listMissingTranslations, listMissingTranslationsWithConfig, listProjectsCommand, liveSync, pull, push, pushConfig, reviewDoc, searchDoc, setAPI, startEditor, testMissingTranslations, translateDoc };
22
+ export { PLATFORM_OPTIONS, build, bundle, dirname, extract, fill, findProjectRoot, getDetectedPlatform, init, initSkills, listContentDeclaration, listContentDeclarationRows, listMissingTranslations, listMissingTranslationsWithConfig, listProjectsCommand, liveSync, pull, push, pushConfig, reviewDoc, scan, searchDoc, setAPI, startEditor, testMissingTranslations, translateDoc };