lezu 0.0.11
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.
- package/README.md +454 -0
- package/dist/api-CS5JMU72.js +8 -0
- package/dist/api-CS5JMU72.js.map +1 -0
- package/dist/chunk-2MDUFSO7.js +249 -0
- package/dist/chunk-2MDUFSO7.js.map +1 -0
- package/dist/chunk-LYGGTR5I.js +218 -0
- package/dist/chunk-LYGGTR5I.js.map +1 -0
- package/dist/chunk-UN7HQJOX.js +111 -0
- package/dist/chunk-UN7HQJOX.js.map +1 -0
- package/dist/chunk-VGQICGPH.js +98 -0
- package/dist/chunk-VGQICGPH.js.map +1 -0
- package/dist/get-EDBOC5W6.js +10 -0
- package/dist/get-EDBOC5W6.js.map +1 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +501 -0
- package/dist/index.js.map +1 -0
- package/dist/load-3MKWVORU.js +12 -0
- package/dist/load-3MKWVORU.js.map +1 -0
- package/package.json +49 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/commands/load.ts","../src/writer.ts"],"sourcesContent":["import { Command } from 'commander'\nimport chalk from 'chalk'\nimport ora from 'ora'\nimport { loadConfig, validateConfig } from '../config.js'\nimport { LezuApiClient } from '../api.js'\nimport { TranslationWriter } from '../writer.js'\nimport type { Config } from '../types.js'\n\nexport function createLoadCommand() {\n const command = new Command('load')\n .description('Download translation files from Lezu')\n .option('-p, --project <id>', 'Project ID')\n .option('-k, --api-key <key>', 'API key')\n .option('-d, --dest <path>', 'Destination folder', './src/i18n')\n .option('-f, --format <format>', 'Output format (json, js, ts, yaml, po)', 'json')\n .option('-l, --languages <languages>', 'Comma-separated list of languages to download')\n .option('-r, --release <id>', 'Specific release ID')\n .option('-e, --environment <env>', 'Environment', 'production')\n .option('--api-url <url>', 'API URL', 'https://api.lezu.app')\n .option('--flatten', 'Flatten nested keys')\n .option('--namespace', 'Use namespace folder structure')\n .option('--include-empty', 'Include empty translations')\n .option('-w, --watch', 'Watch for changes')\n .option('--watch-interval <ms>', 'Watch interval in milliseconds', '5000')\n .action(async (options) => {\n try {\n // Parse languages from comma-separated string\n if (options.languages) {\n options.languages = options.languages.split(',').map((l: string) => l.trim())\n }\n\n // Convert watch interval to number\n if (options.watchInterval) {\n options.watchInterval = parseInt(options.watchInterval, 10)\n }\n\n // Load configuration\n const config = loadConfig(options)\n\n // Validate configuration\n const errors = validateConfig(config)\n if (errors.length > 0) {\n console.error(chalk.red('Configuration errors:'))\n errors.forEach(error => console.error(chalk.red(` ⢠${error}`)))\n process.exit(1)\n }\n\n // Run the sync\n await syncTranslations(config)\n\n // Watch mode\n if (config.watch) {\n console.log(chalk.cyan(`\\nš Watching for changes every ${config.watchInterval}ms...`))\n console.log(chalk.gray('Press Ctrl+C to stop'))\n \n setInterval(async () => {\n console.log(chalk.gray('\\nš Checking for updates...'))\n await syncTranslations(config)\n }, config.watchInterval)\n }\n } catch (error) {\n console.error(chalk.red('\\nā Error:'), error instanceof Error ? error.message : error)\n process.exit(1)\n }\n })\n\n return command\n}\n\nexport async function syncTranslations(config: Config) {\n const spinner = ora('Connecting to Lezu API...').start()\n\n try {\n // Create API client\n const client = new LezuApiClient(config)\n\n // If no release specified and not using latest, try to get latest release\n if (!config.release && config.environment) {\n spinner.text = 'Fetching latest release...'\n const latestRelease = await client.getLatestRelease()\n if (latestRelease) {\n config.release = latestRelease\n spinner.succeed(`Using latest release: ${latestRelease}`)\n spinner.start('Fetching translations...')\n } else {\n spinner.info('No releases found, fetching current translations...')\n spinner.start()\n }\n }\n\n // Fetch translations\n spinner.text = 'Fetching translations...'\n const response = await client.getTranslations()\n\n if (!response.translations || Object.keys(response.translations).length === 0) {\n spinner.fail('No translations found')\n return\n }\n\n spinner.succeed(`Fetched translations for ${Object.keys(response.translations).length} languages`)\n\n // Write translations\n const writer = new TranslationWriter(config)\n writer.write(response.translations)\n\n console.log(chalk.green('\\n⨠Translations synced successfully!'))\n\n // Show summary\n console.log(chalk.gray('\\nSummary:'))\n console.log(chalk.gray(` ⢠Project: ${config.projectId}`))\n console.log(chalk.gray(` ⢠Languages: ${Object.keys(response.translations).join(', ')}`))\n console.log(chalk.gray(` ⢠Destination: ${config.dest}`))\n console.log(chalk.gray(` ⢠Format: ${config.format}`))\n \n if (response.meta?.version) {\n console.log(chalk.gray(` ⢠Version: ${response.meta.version}`))\n }\n\n } catch (error) {\n spinner.fail('Failed to sync translations')\n throw error\n }\n}","import { writeFileSync, mkdirSync } from 'fs'\nimport { resolve, dirname, join } from 'path'\nimport type { Config, TranslationBundle } from './types.js'\n\nexport class TranslationWriter {\n private config: Config\n\n constructor(config: Config) {\n this.config = config\n }\n\n write(translations: Record<string, any>): void {\n // Ensure destination directory exists\n mkdirSync(this.config.dest, { recursive: true })\n\n // Write each language to its own file\n for (const [language, content] of Object.entries(translations)) {\n const processedContent = this.processContent(content)\n const filePath = this.getFilePath(language)\n \n mkdirSync(dirname(filePath), { recursive: true })\n \n const fileContent = this.formatContent(processedContent, language)\n writeFileSync(filePath, fileContent, 'utf-8')\n \n console.log(`ā
Written ${language} translations to ${filePath}`)\n }\n }\n\n private processContent(content: any): any {\n if (this.config.flatten) {\n return this.flattenObject(content)\n }\n \n if (!this.config.includeEmpty) {\n return this.removeEmptyValues(content)\n }\n \n return content\n }\n\n private flattenObject(obj: any, prefix = ''): Record<string, string> {\n const result: Record<string, string> = {}\n \n for (const [key, value] of Object.entries(obj)) {\n const newKey = prefix ? `${prefix}.${key}` : key\n \n if (typeof value === 'object' && value !== null && !Array.isArray(value)) {\n Object.assign(result, this.flattenObject(value, newKey))\n } else {\n result[newKey] = value as string\n }\n }\n \n return result\n }\n\n private removeEmptyValues(obj: any): any {\n if (typeof obj !== 'object' || obj === null) {\n return obj\n }\n \n const result: any = {}\n \n for (const [key, value] of Object.entries(obj)) {\n if (typeof value === 'object' && value !== null) {\n const cleaned = this.removeEmptyValues(value)\n if (Object.keys(cleaned).length > 0) {\n result[key] = cleaned\n }\n } else if (value !== '' && value !== null && value !== undefined) {\n result[key] = value\n }\n }\n \n return result\n }\n\n private getFilePath(language: string): string {\n const extension = this.getFileExtension()\n \n if (this.config.namespace) {\n // Use namespace structure: dest/locales/en.json\n return join(this.config.dest, 'locales', `${language}.${extension}`)\n } else {\n // Use flat structure: dest/en.json\n return join(this.config.dest, `${language}.${extension}`)\n }\n }\n\n private getFileExtension(): string {\n switch (this.config.format) {\n case 'json':\n return 'json'\n case 'js':\n return 'js'\n case 'ts':\n return 'ts'\n case 'yaml':\n return 'yaml'\n case 'po':\n return 'po'\n default:\n return 'json'\n }\n }\n\n private formatContent(content: any, language: string): string {\n switch (this.config.format) {\n case 'json':\n return JSON.stringify(content, null, 2)\n \n case 'js':\n return `export default ${JSON.stringify(content, null, 2)}`\n \n case 'ts':\n return `const translations = ${JSON.stringify(content, null, 2)} as const\\n\\nexport default translations`\n \n case 'yaml':\n return this.toYaml(content)\n \n case 'po':\n return this.toPo(content, language)\n \n default:\n return JSON.stringify(content, null, 2)\n }\n }\n\n private toYaml(obj: any, indent = 0): string {\n let result = ''\n const spaces = ' '.repeat(indent)\n \n for (const [key, value] of Object.entries(obj)) {\n if (typeof value === 'object' && value !== null) {\n result += `${spaces}${key}:\\n${this.toYaml(value, indent + 2)}`\n } else {\n const escapedValue = String(value).includes(':') || String(value).includes('#') \n ? `\"${String(value).replace(/\"/g, '\\\\\"')}\"` \n : value\n result += `${spaces}${key}: ${escapedValue}\\n`\n }\n }\n \n return result\n }\n\n private toPo(obj: any, language: string): string {\n const now = new Date().toISOString()\n let result = `# Translation file for ${language}\n# Generated by Lezu CLI on ${now}\nmsgid \"\"\nmsgstr \"\"\n\"Language: ${language}\\\\n\"\n\"MIME-Version: 1.0\\\\n\"\n\"Content-Type: text/plain; charset=UTF-8\\\\n\"\n\"Content-Transfer-Encoding: 8bit\\\\n\"\n\"Generated-By: Lezu CLI\\\\n\"\n\n`\n\n const flatEntries = this.flattenForPo(obj)\n \n for (const [key, value] of Object.entries(flatEntries)) {\n const escapedKey = this.escapePoString(key)\n const escapedValue = this.escapePoString(String(value))\n \n result += `msgid \"${escapedKey}\"\\n`\n result += `msgstr \"${escapedValue}\"\\n\\n`\n }\n \n return result\n }\n\n private flattenForPo(obj: any, prefix = ''): Record<string, string> {\n const result: Record<string, string> = {}\n \n for (const [key, value] of Object.entries(obj)) {\n const fullKey = prefix ? `${prefix}.${key}` : key\n \n if (typeof value === 'object' && value !== null && !Array.isArray(value)) {\n Object.assign(result, this.flattenForPo(value, fullKey))\n } else {\n result[fullKey] = String(value)\n }\n }\n \n return result\n }\n\n private escapePoString(str: string): string {\n return str\n .replace(/\\\\/g, '\\\\\\\\')\n .replace(/\"/g, '\\\\\"')\n .replace(/\\n/g, '\\\\n')\n .replace(/\\r/g, '\\\\r')\n .replace(/\\t/g, '\\\\t')\n }\n}"],"mappings":"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","names":[]}
|
|
@@ -0,0 +1,218 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import {
|
|
3
|
+
loadConfig,
|
|
4
|
+
validateConfig
|
|
5
|
+
} from "./chunk-VGQICGPH.js";
|
|
6
|
+
import {
|
|
7
|
+
LezuApiClient
|
|
8
|
+
} from "./chunk-UN7HQJOX.js";
|
|
9
|
+
|
|
10
|
+
// src/commands/get.ts
|
|
11
|
+
import { Command } from "commander";
|
|
12
|
+
import chalk from "chalk";
|
|
13
|
+
import ora from "ora";
|
|
14
|
+
function createGetCommand() {
|
|
15
|
+
const command = new Command("get").description("Get information about your project");
|
|
16
|
+
command.command("languages").alias("langs").description("List all languages in your project").option("-p, --project <id>", "Project ID").option("-a, --api-key <key>", "API key").option("--api-url <url>", "API URL", "https://api.lezu.app").option("--include-disabled", "Include disabled languages").option("--json", "Output as JSON").action(async (options) => {
|
|
17
|
+
try {
|
|
18
|
+
const config = loadConfig(options);
|
|
19
|
+
const errors = validateConfig(config);
|
|
20
|
+
if (errors.length > 0) {
|
|
21
|
+
console.error(chalk.red("Configuration errors:"));
|
|
22
|
+
errors.forEach((error) => console.error(chalk.red(` \u2022 ${error}`)));
|
|
23
|
+
process.exit(1);
|
|
24
|
+
}
|
|
25
|
+
await getLanguages(config, options);
|
|
26
|
+
} catch (error) {
|
|
27
|
+
console.error(chalk.red("\n\u274C Error:"), error instanceof Error ? error.message : error);
|
|
28
|
+
process.exit(1);
|
|
29
|
+
}
|
|
30
|
+
});
|
|
31
|
+
command.command("releases").description("List all releases in your project").option("-p, --project <id>", "Project ID").option("-a, --api-key <key>", "API key").option("--api-url <url>", "API URL", "https://api.lezu.app").option("-e, --environment <env>", "Environment", "production").option("--json", "Output as JSON").option("--limit <n>", "Limit number of results", "10").action(async (options) => {
|
|
32
|
+
try {
|
|
33
|
+
const config = loadConfig(options);
|
|
34
|
+
const errors = validateConfig(config);
|
|
35
|
+
if (errors.length > 0) {
|
|
36
|
+
console.error(chalk.red("Configuration errors:"));
|
|
37
|
+
errors.forEach((error) => console.error(chalk.red(` \u2022 ${error}`)));
|
|
38
|
+
process.exit(1);
|
|
39
|
+
}
|
|
40
|
+
await getReleases(config, options);
|
|
41
|
+
} catch (error) {
|
|
42
|
+
console.error(chalk.red("\n\u274C Error:"), error instanceof Error ? error.message : error);
|
|
43
|
+
process.exit(1);
|
|
44
|
+
}
|
|
45
|
+
});
|
|
46
|
+
command.command("project").alias("info").description("Get project information and statistics").option("-p, --project <id>", "Project ID").option("-a, --api-key <key>", "API key").option("--api-url <url>", "API URL", "https://api.lezu.app").option("--json", "Output as JSON").option("--stats", "Include detailed statistics").action(async (options) => {
|
|
47
|
+
try {
|
|
48
|
+
const config = loadConfig(options);
|
|
49
|
+
const errors = validateConfig(config);
|
|
50
|
+
if (errors.length > 0) {
|
|
51
|
+
console.error(chalk.red("Configuration errors:"));
|
|
52
|
+
errors.forEach((error) => console.error(chalk.red(` \u2022 ${error}`)));
|
|
53
|
+
process.exit(1);
|
|
54
|
+
}
|
|
55
|
+
await getProject(config, options);
|
|
56
|
+
} catch (error) {
|
|
57
|
+
console.error(chalk.red("\n\u274C Error:"), error instanceof Error ? error.message : error);
|
|
58
|
+
process.exit(1);
|
|
59
|
+
}
|
|
60
|
+
});
|
|
61
|
+
return command;
|
|
62
|
+
}
|
|
63
|
+
async function getLanguages(config, options) {
|
|
64
|
+
const spinner = ora("Fetching languages...").start();
|
|
65
|
+
try {
|
|
66
|
+
const client = new LezuApiClient(config);
|
|
67
|
+
const response = await client.getLanguages();
|
|
68
|
+
spinner.succeed("Languages fetched successfully!");
|
|
69
|
+
if (options.json) {
|
|
70
|
+
console.log(JSON.stringify(response, null, 2));
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
console.log(chalk.green("\n\u{1F4DA} Project Languages:"));
|
|
74
|
+
if (!response.languages || response.languages.length === 0) {
|
|
75
|
+
console.log(chalk.gray(" No languages found"));
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
response.languages.forEach((lang) => {
|
|
79
|
+
const status = lang.enabled ? chalk.green("\u2713") : chalk.gray("\u2717");
|
|
80
|
+
const progress = lang.progress ? ` (${lang.progress.percentage}% complete)` : "";
|
|
81
|
+
const source = lang.is_source ? chalk.yellow(" [SOURCE]") : "";
|
|
82
|
+
console.log(` ${status} ${lang.language?.name || lang.language_code} (${lang.language_code})${source}${progress}`);
|
|
83
|
+
if (lang.progress && lang.progress.total > 0) {
|
|
84
|
+
console.log(chalk.gray(` ${lang.progress.translated}/${lang.progress.total} keys translated`));
|
|
85
|
+
}
|
|
86
|
+
});
|
|
87
|
+
console.log(chalk.gray(`
|
|
88
|
+
Total: ${response.total} languages`));
|
|
89
|
+
} catch (error) {
|
|
90
|
+
spinner.fail("Failed to fetch languages");
|
|
91
|
+
throw error;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
async function getReleases(config, options) {
|
|
95
|
+
const spinner = ora("Fetching releases...").start();
|
|
96
|
+
try {
|
|
97
|
+
const client = new LezuApiClient(config);
|
|
98
|
+
const response = await client.getReleases();
|
|
99
|
+
spinner.succeed("Releases fetched successfully!");
|
|
100
|
+
if (options.json) {
|
|
101
|
+
console.log(JSON.stringify(response, null, 2));
|
|
102
|
+
return;
|
|
103
|
+
}
|
|
104
|
+
console.log(chalk.green(`
|
|
105
|
+
\u{1F680} Releases (${options.environment} environment):`));
|
|
106
|
+
if (!response.releases || response.releases.length === 0) {
|
|
107
|
+
console.log(chalk.gray(" No releases found"));
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
const limit = parseInt(options.limit, 10);
|
|
111
|
+
const releases = response.releases.slice(0, limit);
|
|
112
|
+
releases.forEach((release, index) => {
|
|
113
|
+
const current = release.is_current ? chalk.yellow(" [CURRENT]") : "";
|
|
114
|
+
const date = new Date(release.created_at).toLocaleDateString();
|
|
115
|
+
const reviewedTag = release.reviewed_only ? chalk.blue(" [REVIEWED ONLY]") : "";
|
|
116
|
+
console.log(` ${index + 1}. ${release.version}${current}${reviewedTag}`);
|
|
117
|
+
if (release.name && release.name !== release.version) {
|
|
118
|
+
console.log(chalk.gray(` ${release.name}`));
|
|
119
|
+
}
|
|
120
|
+
if (release.description) {
|
|
121
|
+
console.log(chalk.gray(` ${release.description}`));
|
|
122
|
+
}
|
|
123
|
+
console.log(chalk.gray(` Created: ${date}`));
|
|
124
|
+
if (release.created_by_user?.email) {
|
|
125
|
+
console.log(chalk.gray(` By: ${release.created_by_user.email}`));
|
|
126
|
+
}
|
|
127
|
+
console.log();
|
|
128
|
+
});
|
|
129
|
+
if (response.releases.length > limit) {
|
|
130
|
+
console.log(chalk.gray(`... and ${response.releases.length - limit} more releases`));
|
|
131
|
+
console.log(chalk.gray(`Use --limit ${response.releases.length} to see all releases`));
|
|
132
|
+
}
|
|
133
|
+
} catch (error) {
|
|
134
|
+
spinner.fail("Failed to fetch releases");
|
|
135
|
+
throw error;
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
async function getProject(config, options) {
|
|
139
|
+
const spinner = ora("Fetching project information...").start();
|
|
140
|
+
try {
|
|
141
|
+
const client = new LezuApiClient(config);
|
|
142
|
+
let projectData, statsData;
|
|
143
|
+
if (options.stats) {
|
|
144
|
+
const [projectResponse, statsResponse] = await Promise.all([
|
|
145
|
+
client.getProject(),
|
|
146
|
+
fetch(`${config.apiUrl}/v1/projects/${config.projectId}/stats`, {
|
|
147
|
+
headers: {
|
|
148
|
+
"Authorization": `Bearer ${config.apiKey}`
|
|
149
|
+
}
|
|
150
|
+
}).then((r) => r.json())
|
|
151
|
+
]);
|
|
152
|
+
projectData = projectResponse.project;
|
|
153
|
+
statsData = statsResponse.stats;
|
|
154
|
+
} else {
|
|
155
|
+
const response = await client.getProject();
|
|
156
|
+
projectData = response.project;
|
|
157
|
+
}
|
|
158
|
+
spinner.succeed("Project information fetched successfully!");
|
|
159
|
+
if (options.json) {
|
|
160
|
+
console.log(JSON.stringify({ project: projectData, stats: statsData }, null, 2));
|
|
161
|
+
return;
|
|
162
|
+
}
|
|
163
|
+
console.log(chalk.green("\n\u{1F4CB} Project Information:"));
|
|
164
|
+
console.log(` Name: ${projectData.name}`);
|
|
165
|
+
console.log(` Key: ${projectData.key}`);
|
|
166
|
+
console.log(` ID: ${projectData.id}`);
|
|
167
|
+
if (projectData.description) {
|
|
168
|
+
console.log(` Description: ${projectData.description}`);
|
|
169
|
+
}
|
|
170
|
+
console.log(` Source Language: ${projectData.source_language}`);
|
|
171
|
+
console.log(` Created: ${new Date(projectData.created_at).toLocaleDateString()}`);
|
|
172
|
+
if (projectData.stats) {
|
|
173
|
+
console.log(chalk.green("\n\u{1F4CA} Basic Statistics:"));
|
|
174
|
+
console.log(` Translation Keys: ${projectData.stats.keys}`);
|
|
175
|
+
console.log(` Languages: ${projectData.stats.languages}`);
|
|
176
|
+
console.log(` Translations: ${projectData.stats.translations}`);
|
|
177
|
+
console.log(` Completeness: ${projectData.stats.completeness}%`);
|
|
178
|
+
}
|
|
179
|
+
if (statsData) {
|
|
180
|
+
console.log(chalk.green("\n\u{1F4C8} Detailed Statistics:"));
|
|
181
|
+
console.log(` Total Keys: ${statsData.overview.totalKeys}`);
|
|
182
|
+
console.log(` Total Languages: ${statsData.overview.totalLanguages}`);
|
|
183
|
+
console.log(` Total Releases: ${statsData.overview.totalReleases}`);
|
|
184
|
+
console.log(` Keys Added Last Month: ${statsData.overview.keysAddedLastMonth}`);
|
|
185
|
+
if (statsData.languages && statsData.languages.length > 0) {
|
|
186
|
+
console.log(chalk.green("\n\u{1F30D} Language Progress:"));
|
|
187
|
+
statsData.languages.forEach((lang) => {
|
|
188
|
+
const percentage = lang.percentage;
|
|
189
|
+
const bar = "\u2588".repeat(Math.floor(percentage / 5)) + "\u2591".repeat(20 - Math.floor(percentage / 5));
|
|
190
|
+
console.log(` ${lang.language_code}: [${bar}] ${percentage}%`);
|
|
191
|
+
console.log(chalk.gray(` ${lang.translated}/${lang.total} translated, ${lang.reviewed} reviewed, ${lang.missing} missing`));
|
|
192
|
+
});
|
|
193
|
+
}
|
|
194
|
+
if (statsData.recentReleases && statsData.recentReleases.length > 0) {
|
|
195
|
+
console.log(chalk.green("\n\u{1F680} Recent Releases:"));
|
|
196
|
+
statsData.recentReleases.forEach((release) => {
|
|
197
|
+
const date = new Date(release.created_at).toLocaleDateString();
|
|
198
|
+
console.log(` \u2022 ${release.version} (${date})`);
|
|
199
|
+
});
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
if (projectData.environments && projectData.environments.length > 0) {
|
|
203
|
+
console.log(chalk.green("\n\u{1F3D7}\uFE0F Environments:"));
|
|
204
|
+
projectData.environments.forEach((env) => {
|
|
205
|
+
const current = env.current_release_id ? ` (current: ${env.current_release_id.substring(0, 8)}...)` : "";
|
|
206
|
+
console.log(` \u2022 ${env.name}${current}`);
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
} catch (error) {
|
|
210
|
+
spinner.fail("Failed to fetch project information");
|
|
211
|
+
throw error;
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
export {
|
|
216
|
+
createGetCommand
|
|
217
|
+
};
|
|
218
|
+
//# sourceMappingURL=chunk-LYGGTR5I.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/commands/get.ts"],"sourcesContent":["import { Command } from 'commander'\nimport chalk from 'chalk'\nimport ora from 'ora'\nimport { loadConfig, validateConfig } from '../config.js'\nimport { LezuApiClient } from '../api.js'\nimport type { Config } from '../types.js'\n\nexport function createGetCommand() {\n const command = new Command('get')\n .description('Get information about your project')\n \n // Languages subcommand\n command\n .command('languages')\n .alias('langs')\n .description('List all languages in your project')\n .option('-p, --project <id>', 'Project ID')\n .option('-a, --api-key <key>', 'API key')\n .option('--api-url <url>', 'API URL', 'https://api.lezu.app')\n .option('--include-disabled', 'Include disabled languages')\n .option('--json', 'Output as JSON')\n .action(async (options) => {\n try {\n const config = loadConfig(options)\n const errors = validateConfig(config)\n if (errors.length > 0) {\n console.error(chalk.red('Configuration errors:'))\n errors.forEach(error => console.error(chalk.red(` ⢠${error}`)))\n process.exit(1)\n }\n \n await getLanguages(config, options)\n } catch (error) {\n console.error(chalk.red('\\nā Error:'), error instanceof Error ? error.message : error)\n process.exit(1)\n }\n })\n \n // Releases subcommand\n command\n .command('releases')\n .description('List all releases in your project')\n .option('-p, --project <id>', 'Project ID')\n .option('-a, --api-key <key>', 'API key')\n .option('--api-url <url>', 'API URL', 'https://api.lezu.app')\n .option('-e, --environment <env>', 'Environment', 'production')\n .option('--json', 'Output as JSON')\n .option('--limit <n>', 'Limit number of results', '10')\n .action(async (options) => {\n try {\n const config = loadConfig(options)\n const errors = validateConfig(config)\n if (errors.length > 0) {\n console.error(chalk.red('Configuration errors:'))\n errors.forEach(error => console.error(chalk.red(` ⢠${error}`)))\n process.exit(1)\n }\n \n await getReleases(config, options)\n } catch (error) {\n console.error(chalk.red('\\nā Error:'), error instanceof Error ? error.message : error)\n process.exit(1)\n }\n })\n \n // Project info subcommand\n command\n .command('project')\n .alias('info')\n .description('Get project information and statistics')\n .option('-p, --project <id>', 'Project ID')\n .option('-a, --api-key <key>', 'API key')\n .option('--api-url <url>', 'API URL', 'https://api.lezu.app')\n .option('--json', 'Output as JSON')\n .option('--stats', 'Include detailed statistics')\n .action(async (options) => {\n try {\n const config = loadConfig(options)\n const errors = validateConfig(config)\n if (errors.length > 0) {\n console.error(chalk.red('Configuration errors:'))\n errors.forEach(error => console.error(chalk.red(` ⢠${error}`)))\n process.exit(1)\n }\n \n await getProject(config, options)\n } catch (error) {\n console.error(chalk.red('\\nā Error:'), error instanceof Error ? error.message : error)\n process.exit(1)\n }\n })\n \n return command\n}\n\nasync function getLanguages(config: Config, options: any) {\n const spinner = ora('Fetching languages...').start()\n \n try {\n const client = new LezuApiClient(config)\n const response = await client.getLanguages()\n \n spinner.succeed('Languages fetched successfully!')\n \n if (options.json) {\n console.log(JSON.stringify(response, null, 2))\n return\n }\n \n console.log(chalk.green('\\nš Project Languages:'))\n \n if (!response.languages || response.languages.length === 0) {\n console.log(chalk.gray(' No languages found'))\n return\n }\n \n response.languages.forEach((lang: any) => {\n const status = lang.enabled ? chalk.green('ā') : chalk.gray('ā')\n const progress = lang.progress ? ` (${lang.progress.percentage}% complete)` : ''\n const source = lang.is_source ? chalk.yellow(' [SOURCE]') : ''\n \n console.log(` ${status} ${lang.language?.name || lang.language_code} (${lang.language_code})${source}${progress}`)\n \n if (lang.progress && lang.progress.total > 0) {\n console.log(chalk.gray(` ${lang.progress.translated}/${lang.progress.total} keys translated`))\n }\n })\n \n console.log(chalk.gray(`\\nTotal: ${response.total} languages`))\n \n } catch (error) {\n spinner.fail('Failed to fetch languages')\n throw error\n }\n}\n\nasync function getReleases(config: Config, options: any) {\n const spinner = ora('Fetching releases...').start()\n \n try {\n const client = new LezuApiClient(config)\n const response = await client.getReleases()\n \n spinner.succeed('Releases fetched successfully!')\n \n if (options.json) {\n console.log(JSON.stringify(response, null, 2))\n return\n }\n \n console.log(chalk.green(`\\nš Releases (${options.environment} environment):`))\n \n if (!response.releases || response.releases.length === 0) {\n console.log(chalk.gray(' No releases found'))\n return\n }\n \n const limit = parseInt(options.limit, 10)\n const releases = response.releases.slice(0, limit)\n \n releases.forEach((release: any, index: number) => {\n const current = release.is_current ? chalk.yellow(' [CURRENT]') : ''\n const date = new Date(release.created_at).toLocaleDateString()\n const reviewedTag = release.reviewed_only ? chalk.blue(' [REVIEWED ONLY]') : ''\n \n console.log(` ${index + 1}. ${release.version}${current}${reviewedTag}`)\n if (release.name && release.name !== release.version) {\n console.log(chalk.gray(` ${release.name}`))\n }\n if (release.description) {\n console.log(chalk.gray(` ${release.description}`))\n }\n console.log(chalk.gray(` Created: ${date}`))\n if (release.created_by_user?.email) {\n console.log(chalk.gray(` By: ${release.created_by_user.email}`))\n }\n console.log() // empty line\n })\n \n if (response.releases.length > limit) {\n console.log(chalk.gray(`... and ${response.releases.length - limit} more releases`))\n console.log(chalk.gray(`Use --limit ${response.releases.length} to see all releases`))\n }\n \n } catch (error) {\n spinner.fail('Failed to fetch releases')\n throw error\n }\n}\n\nasync function getProject(config: Config, options: any) {\n const spinner = ora('Fetching project information...').start()\n \n try {\n const client = new LezuApiClient(config)\n \n let projectData, statsData\n \n if (options.stats) {\n // Fetch both project info and detailed stats\n const [projectResponse, statsResponse] = await Promise.all([\n client.getProject(),\n fetch(`${config.apiUrl}/v1/projects/${config.projectId}/stats`, {\n headers: {\n 'Authorization': `Bearer ${config.apiKey}`,\n },\n }).then(r => r.json())\n ])\n projectData = projectResponse.project\n statsData = statsResponse.stats\n } else {\n const response = await client.getProject()\n projectData = response.project\n }\n \n spinner.succeed('Project information fetched successfully!')\n \n if (options.json) {\n console.log(JSON.stringify({ project: projectData, stats: statsData }, null, 2))\n return\n }\n \n console.log(chalk.green('\\nš Project Information:'))\n console.log(` Name: ${projectData.name}`)\n console.log(` Key: ${projectData.key}`)\n console.log(` ID: ${projectData.id}`)\n if (projectData.description) {\n console.log(` Description: ${projectData.description}`)\n }\n console.log(` Source Language: ${projectData.source_language}`)\n console.log(` Created: ${new Date(projectData.created_at).toLocaleDateString()}`)\n \n // Basic stats (always available)\n if (projectData.stats) {\n console.log(chalk.green('\\nš Basic Statistics:'))\n console.log(` Translation Keys: ${projectData.stats.keys}`)\n console.log(` Languages: ${projectData.stats.languages}`)\n console.log(` Translations: ${projectData.stats.translations}`)\n console.log(` Completeness: ${projectData.stats.completeness}%`)\n }\n \n // Detailed stats (when --stats is used)\n if (statsData) {\n console.log(chalk.green('\\nš Detailed Statistics:'))\n console.log(` Total Keys: ${statsData.overview.totalKeys}`)\n console.log(` Total Languages: ${statsData.overview.totalLanguages}`)\n console.log(` Total Releases: ${statsData.overview.totalReleases}`)\n console.log(` Keys Added Last Month: ${statsData.overview.keysAddedLastMonth}`)\n \n if (statsData.languages && statsData.languages.length > 0) {\n console.log(chalk.green('\\nš Language Progress:'))\n statsData.languages.forEach((lang: any) => {\n const percentage = lang.percentage\n const bar = 'ā'.repeat(Math.floor(percentage / 5)) + 'ā'.repeat(20 - Math.floor(percentage / 5))\n console.log(` ${lang.language_code}: [${bar}] ${percentage}%`)\n console.log(chalk.gray(` ${lang.translated}/${lang.total} translated, ${lang.reviewed} reviewed, ${lang.missing} missing`))\n })\n }\n \n if (statsData.recentReleases && statsData.recentReleases.length > 0) {\n console.log(chalk.green('\\nš Recent Releases:'))\n statsData.recentReleases.forEach((release: any) => {\n const date = new Date(release.created_at).toLocaleDateString()\n console.log(` ⢠${release.version} (${date})`)\n })\n }\n }\n \n // Show environments\n if (projectData.environments && projectData.environments.length > 0) {\n console.log(chalk.green('\\nšļø Environments:'))\n projectData.environments.forEach((env: any) => {\n const current = env.current_release_id ? ` (current: ${env.current_release_id.substring(0, 8)}...)` : ''\n console.log(` ⢠${env.name}${current}`)\n })\n }\n \n } catch (error) {\n spinner.fail('Failed to fetch project information')\n throw error\n }\n}"],"mappings":"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","names":[]}
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
// src/api.ts
|
|
4
|
+
import fetch from "node-fetch";
|
|
5
|
+
var LezuApiClient = class {
|
|
6
|
+
config;
|
|
7
|
+
constructor(config) {
|
|
8
|
+
this.config = config;
|
|
9
|
+
}
|
|
10
|
+
async request(endpoint, options = {}) {
|
|
11
|
+
const url = `${this.config.apiUrl}${endpoint}`;
|
|
12
|
+
const response = await fetch(url, {
|
|
13
|
+
...options,
|
|
14
|
+
headers: {
|
|
15
|
+
"Authorization": `Bearer ${this.config.apiKey}`,
|
|
16
|
+
"Content-Type": "application/json",
|
|
17
|
+
...options.headers
|
|
18
|
+
}
|
|
19
|
+
});
|
|
20
|
+
if (!response.ok) {
|
|
21
|
+
const error = await response.text();
|
|
22
|
+
throw new Error(`API request failed: ${response.status} ${response.statusText}
|
|
23
|
+
${error}`);
|
|
24
|
+
}
|
|
25
|
+
return response.json();
|
|
26
|
+
}
|
|
27
|
+
async getTranslations() {
|
|
28
|
+
let endpoint = `/v1/translations/bundle?environment=${this.config.environment}`;
|
|
29
|
+
if (this.config.release) {
|
|
30
|
+
endpoint = `/v1/releases/${this.config.release}/bundle`;
|
|
31
|
+
}
|
|
32
|
+
if (this.config.languages && this.config.languages.length > 0) {
|
|
33
|
+
const params = new URLSearchParams();
|
|
34
|
+
this.config.languages.forEach((lang) => params.append("languages", lang));
|
|
35
|
+
endpoint += `?${params.toString()}`;
|
|
36
|
+
}
|
|
37
|
+
const response = await this.request(endpoint);
|
|
38
|
+
return {
|
|
39
|
+
translations: response.translations || response,
|
|
40
|
+
meta: {
|
|
41
|
+
version: response.version,
|
|
42
|
+
timestamp: response.timestamp || (/* @__PURE__ */ new Date()).toISOString(),
|
|
43
|
+
languages: response.languages || Object.keys(response.translations || response)
|
|
44
|
+
}
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
async getReleases() {
|
|
48
|
+
return this.request(
|
|
49
|
+
`/v1/releases?environment=${this.config.environment}`
|
|
50
|
+
);
|
|
51
|
+
}
|
|
52
|
+
async getLatestRelease() {
|
|
53
|
+
try {
|
|
54
|
+
const releases = await this.getReleases();
|
|
55
|
+
if (releases.releases && releases.releases.length > 0) {
|
|
56
|
+
return releases.releases[0].id;
|
|
57
|
+
}
|
|
58
|
+
return null;
|
|
59
|
+
} catch (error) {
|
|
60
|
+
console.warn("Failed to fetch latest release:", error);
|
|
61
|
+
return null;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
async getLanguages() {
|
|
65
|
+
return this.request(`/v1/languages`);
|
|
66
|
+
}
|
|
67
|
+
async addTranslationKey(key, translations) {
|
|
68
|
+
return this.request(
|
|
69
|
+
`/v1/keys`,
|
|
70
|
+
{
|
|
71
|
+
method: "POST",
|
|
72
|
+
body: JSON.stringify({ key, translations })
|
|
73
|
+
}
|
|
74
|
+
);
|
|
75
|
+
}
|
|
76
|
+
async translateKey(text, sourceLanguage, targetLanguages) {
|
|
77
|
+
const response = await this.request("/v1/translations/translate/batch", {
|
|
78
|
+
method: "POST",
|
|
79
|
+
body: JSON.stringify({
|
|
80
|
+
texts: [text],
|
|
81
|
+
sourceLanguage,
|
|
82
|
+
targetLanguages
|
|
83
|
+
})
|
|
84
|
+
});
|
|
85
|
+
return response.translations[text] || {};
|
|
86
|
+
}
|
|
87
|
+
async createRelease(data = {}) {
|
|
88
|
+
return this.request(
|
|
89
|
+
`/v1/releases`,
|
|
90
|
+
{
|
|
91
|
+
method: "POST",
|
|
92
|
+
body: JSON.stringify({
|
|
93
|
+
environment: this.config.environment || "production",
|
|
94
|
+
version: data.version || (/* @__PURE__ */ new Date()).toISOString().split("T")[0],
|
|
95
|
+
name: data.name,
|
|
96
|
+
description: data.description,
|
|
97
|
+
reviewedOnly: data.reviewedOnly || false,
|
|
98
|
+
setAsCurrent: true
|
|
99
|
+
})
|
|
100
|
+
}
|
|
101
|
+
);
|
|
102
|
+
}
|
|
103
|
+
async getProject() {
|
|
104
|
+
return this.request(`/v1/projects/${this.config.projectId}`);
|
|
105
|
+
}
|
|
106
|
+
};
|
|
107
|
+
|
|
108
|
+
export {
|
|
109
|
+
LezuApiClient
|
|
110
|
+
};
|
|
111
|
+
//# sourceMappingURL=chunk-UN7HQJOX.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/api.ts"],"sourcesContent":["import fetch from 'node-fetch'\nimport type { Config, ApiResponse, ReleaseResponse } from './types.js'\n\nexport class LezuApiClient {\n private config: Config\n\n constructor(config: Config) {\n this.config = config\n }\n\n private async request<T>(endpoint: string, options: any = {}): Promise<T> {\n const url = `${this.config.apiUrl}${endpoint}`\n \n const response = await fetch(url, {\n ...options,\n headers: {\n 'Authorization': `Bearer ${this.config.apiKey}`,\n 'Content-Type': 'application/json',\n ...options.headers\n }\n })\n\n if (!response.ok) {\n const error = await response.text()\n throw new Error(`API request failed: ${response.status} ${response.statusText}\\n${error}`)\n }\n\n return response.json() as Promise<T>\n }\n\n async getTranslations(): Promise<ApiResponse> {\n let endpoint = `/v1/translations/bundle?environment=${this.config.environment}`\n \n if (this.config.release) {\n endpoint = `/v1/releases/${this.config.release}/bundle`\n }\n\n // Add language filter if specified\n if (this.config.languages && this.config.languages.length > 0) {\n const params = new URLSearchParams()\n this.config.languages.forEach(lang => params.append('languages', lang))\n endpoint += `?${params.toString()}`\n }\n\n const response = await this.request<any>(endpoint)\n \n // Transform the response to match our expected format\n return {\n translations: response.translations || response,\n meta: {\n version: response.version,\n timestamp: response.timestamp || new Date().toISOString(),\n languages: response.languages || Object.keys(response.translations || response)\n }\n }\n }\n\n async getReleases(): Promise<ReleaseResponse> {\n return this.request<ReleaseResponse>(\n `/v1/releases?environment=${this.config.environment}`\n )\n }\n\n async getLatestRelease(): Promise<string | null> {\n try {\n const releases = await this.getReleases()\n if (releases.releases && releases.releases.length > 0) {\n // Releases are ordered by created_at desc, so first one is latest\n return releases.releases[0].id\n }\n return null\n } catch (error) {\n console.warn('Failed to fetch latest release:', error)\n return null\n }\n }\n\n async getLanguages(): Promise<any> {\n return this.request<any>(`/v1/languages`)\n }\n\n async addTranslationKey(key: string, translations: Record<string, string>): Promise<any> {\n return this.request<any>(\n `/v1/keys`,\n {\n method: 'POST',\n body: JSON.stringify({ key, translations })\n }\n )\n }\n\n async translateKey(text: string, sourceLanguage: string, targetLanguages: string[]): Promise<Record<string, string>> {\n // Use batch translate for efficiency\n const response = await this.request<any>('/v1/translations/translate/batch', {\n method: 'POST',\n body: JSON.stringify({\n texts: [text],\n sourceLanguage,\n targetLanguages\n })\n })\n \n return response.translations[text] || {}\n }\n\n async createRelease(data: {\n version?: string\n name?: string\n description?: string\n reviewedOnly?: boolean\n } = {}): Promise<any> {\n return this.request<any>(\n `/v1/releases`,\n {\n method: 'POST',\n body: JSON.stringify({\n environment: this.config.environment || 'production',\n version: data.version || new Date().toISOString().split('T')[0],\n name: data.name,\n description: data.description,\n reviewedOnly: data.reviewedOnly || false,\n setAsCurrent: true\n })\n }\n )\n }\n\n async getProject(): Promise<any> {\n return this.request<any>(`/v1/projects/${this.config.projectId}`)\n }\n}"],"mappings":"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","names":[]}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
// src/config.ts
|
|
4
|
+
import { readFileSync, existsSync } from "fs";
|
|
5
|
+
import { resolve } from "path";
|
|
6
|
+
import dotenv from "dotenv";
|
|
7
|
+
|
|
8
|
+
// src/types.ts
|
|
9
|
+
import { z } from "zod";
|
|
10
|
+
var ConfigSchema = z.object({
|
|
11
|
+
projectId: z.string().optional(),
|
|
12
|
+
apiKey: z.string().optional(),
|
|
13
|
+
apiUrl: z.string().default("https://api.lezu.app"),
|
|
14
|
+
dest: z.string().default("./src/i18n"),
|
|
15
|
+
format: z.enum(["json", "js", "ts", "yaml", "po"]).default("json"),
|
|
16
|
+
languages: z.array(z.string()).optional(),
|
|
17
|
+
release: z.string().optional(),
|
|
18
|
+
environment: z.string().default("production"),
|
|
19
|
+
flatten: z.boolean().default(false),
|
|
20
|
+
namespace: z.boolean().default(false),
|
|
21
|
+
includeEmpty: z.boolean().default(false),
|
|
22
|
+
watch: z.boolean().default(false),
|
|
23
|
+
watchInterval: z.number().default(5e3)
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
// src/config.ts
|
|
27
|
+
dotenv.config();
|
|
28
|
+
function loadConfig(options = {}) {
|
|
29
|
+
let config = {
|
|
30
|
+
apiUrl: "https://api.lezu.app",
|
|
31
|
+
dest: "./src/i18n",
|
|
32
|
+
format: "json",
|
|
33
|
+
environment: "production",
|
|
34
|
+
flatten: false,
|
|
35
|
+
namespace: false,
|
|
36
|
+
includeEmpty: false,
|
|
37
|
+
watch: false,
|
|
38
|
+
watchInterval: 5e3
|
|
39
|
+
};
|
|
40
|
+
const configFiles = [".lezurc", ".lezurc.json"];
|
|
41
|
+
for (const file of configFiles) {
|
|
42
|
+
const configPath = resolve(process.cwd(), file);
|
|
43
|
+
if (existsSync(configPath)) {
|
|
44
|
+
try {
|
|
45
|
+
const content = readFileSync(configPath, "utf-8");
|
|
46
|
+
const fileConfig = JSON.parse(content);
|
|
47
|
+
config = { ...config, ...fileConfig };
|
|
48
|
+
console.log(`\u{1F4C1} Loaded config from ${file}`);
|
|
49
|
+
break;
|
|
50
|
+
} catch (error) {
|
|
51
|
+
console.warn(`\u26A0\uFE0F Failed to load config from ${file}:`, error);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
if (process.env.LEZU_PROJECT_ID) {
|
|
56
|
+
config.projectId = process.env.LEZU_PROJECT_ID;
|
|
57
|
+
}
|
|
58
|
+
if (process.env.LEZU_API_KEY) {
|
|
59
|
+
config.apiKey = process.env.LEZU_API_KEY;
|
|
60
|
+
}
|
|
61
|
+
if (process.env.LEZU_API_URL) {
|
|
62
|
+
config.apiUrl = process.env.LEZU_API_URL;
|
|
63
|
+
}
|
|
64
|
+
if (process.env.LEZU_DEST) {
|
|
65
|
+
config.dest = process.env.LEZU_DEST;
|
|
66
|
+
}
|
|
67
|
+
if (process.env.LEZU_FORMAT) {
|
|
68
|
+
config.format = process.env.LEZU_FORMAT;
|
|
69
|
+
}
|
|
70
|
+
if (process.env.LEZU_ENVIRONMENT) {
|
|
71
|
+
config.environment = process.env.LEZU_ENVIRONMENT;
|
|
72
|
+
}
|
|
73
|
+
config = { ...config, ...options };
|
|
74
|
+
try {
|
|
75
|
+
return ConfigSchema.parse(config);
|
|
76
|
+
} catch (error) {
|
|
77
|
+
if (error instanceof Error) {
|
|
78
|
+
throw new Error(`Invalid configuration: ${error.message}`);
|
|
79
|
+
}
|
|
80
|
+
throw error;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
function validateConfig(config) {
|
|
84
|
+
const errors = [];
|
|
85
|
+
if (!config.projectId) {
|
|
86
|
+
errors.push("Project ID is required (use --project, LEZU_PROJECT_ID env var, or projectId in config file)");
|
|
87
|
+
}
|
|
88
|
+
if (!config.apiKey) {
|
|
89
|
+
errors.push("API Key is required (use --api-key, LEZU_API_KEY env var, or apiKey in config file)");
|
|
90
|
+
}
|
|
91
|
+
return errors;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
export {
|
|
95
|
+
loadConfig,
|
|
96
|
+
validateConfig
|
|
97
|
+
};
|
|
98
|
+
//# sourceMappingURL=chunk-VGQICGPH.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/config.ts","../src/types.ts"],"sourcesContent":["import { readFileSync, existsSync } from 'fs'\nimport { resolve } from 'path'\nimport dotenv from 'dotenv'\nimport { Config, ConfigSchema } from './types.js'\n\n// Load environment variables\ndotenv.config()\n\nexport function loadConfig(options: Partial<Config> = {}): Config {\n // 1. Start with defaults\n let config: Partial<Config> = {\n apiUrl: 'https://api.lezu.app',\n dest: './src/i18n',\n format: 'json',\n environment: 'production',\n flatten: false,\n namespace: false,\n includeEmpty: false,\n watch: false,\n watchInterval: 5000\n }\n\n // 2. Load from config file (only JSON files to avoid JS module complexity)\n const configFiles = ['.lezurc', '.lezurc.json']\n for (const file of configFiles) {\n const configPath = resolve(process.cwd(), file)\n if (existsSync(configPath)) {\n try {\n // JSON config files only\n const content = readFileSync(configPath, 'utf-8')\n const fileConfig = JSON.parse(content)\n config = { ...config, ...fileConfig }\n console.log(`š Loaded config from ${file}`)\n break\n } catch (error) {\n console.warn(`ā ļø Failed to load config from ${file}:`, error)\n }\n }\n }\n\n // 3. Load from environment variables\n if (process.env.LEZU_PROJECT_ID) {\n config.projectId = process.env.LEZU_PROJECT_ID\n }\n if (process.env.LEZU_API_KEY) {\n config.apiKey = process.env.LEZU_API_KEY\n }\n if (process.env.LEZU_API_URL) {\n config.apiUrl = process.env.LEZU_API_URL\n }\n if (process.env.LEZU_DEST) {\n config.dest = process.env.LEZU_DEST\n }\n if (process.env.LEZU_FORMAT) {\n config.format = process.env.LEZU_FORMAT as any\n }\n if (process.env.LEZU_ENVIRONMENT) {\n config.environment = process.env.LEZU_ENVIRONMENT\n }\n\n // 4. Apply CLI options (highest priority)\n config = { ...config, ...options }\n\n // 5. Validate and return\n try {\n return ConfigSchema.parse(config)\n } catch (error) {\n if (error instanceof Error) {\n throw new Error(`Invalid configuration: ${error.message}`)\n }\n throw error\n }\n}\n\nexport function validateConfig(config: Config): string[] {\n const errors: string[] = []\n\n if (!config.projectId) {\n errors.push('Project ID is required (use --project, LEZU_PROJECT_ID env var, or projectId in config file)')\n }\n\n if (!config.apiKey) {\n errors.push('API Key is required (use --api-key, LEZU_API_KEY env var, or apiKey in config file)')\n }\n\n return errors\n}","import { z } from 'zod'\n\nexport const ConfigSchema = z.object({\n projectId: z.string().optional(),\n apiKey: z.string().optional(),\n apiUrl: z.string().default('https://api.lezu.app'),\n dest: z.string().default('./src/i18n'),\n format: z.enum(['json', 'js', 'ts', 'yaml', 'po']).default('json'),\n languages: z.array(z.string()).optional(),\n release: z.string().optional(),\n environment: z.string().default('production'),\n flatten: z.boolean().default(false),\n namespace: z.boolean().default(false),\n includeEmpty: z.boolean().default(false),\n watch: z.boolean().default(false),\n watchInterval: z.number().default(5000)\n})\n\nexport type Config = z.infer<typeof ConfigSchema>\n\nexport interface TranslationBundle {\n [key: string]: string | TranslationBundle\n}\n\nexport interface ApiResponse {\n translations: Record<string, any>\n meta?: {\n version?: string\n timestamp?: string\n languages?: string[]\n }\n}\n\nexport interface ReleaseResponse {\n releases: Array<{\n id: string\n version: string\n name?: string\n created_at: string\n }>\n}"],"mappings":";;;AAAA,SAAS,cAAc,kBAAkB;AACzC,SAAS,eAAe;AACxB,OAAO,YAAY;;;ACFnB,SAAS,SAAS;AAEX,IAAM,eAAe,EAAE,OAAO;AAAA,EACnC,WAAW,EAAE,OAAO,EAAE,SAAS;AAAA,EAC/B,QAAQ,EAAE,OAAO,EAAE,SAAS;AAAA,EAC5B,QAAQ,EAAE,OAAO,EAAE,QAAQ,sBAAsB;AAAA,EACjD,MAAM,EAAE,OAAO,EAAE,QAAQ,YAAY;AAAA,EACrC,QAAQ,EAAE,KAAK,CAAC,QAAQ,MAAM,MAAM,QAAQ,IAAI,CAAC,EAAE,QAAQ,MAAM;AAAA,EACjE,WAAW,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,SAAS;AAAA,EACxC,SAAS,EAAE,OAAO,EAAE,SAAS;AAAA,EAC7B,aAAa,EAAE,OAAO,EAAE,QAAQ,YAAY;AAAA,EAC5C,SAAS,EAAE,QAAQ,EAAE,QAAQ,KAAK;AAAA,EAClC,WAAW,EAAE,QAAQ,EAAE,QAAQ,KAAK;AAAA,EACpC,cAAc,EAAE,QAAQ,EAAE,QAAQ,KAAK;AAAA,EACvC,OAAO,EAAE,QAAQ,EAAE,QAAQ,KAAK;AAAA,EAChC,eAAe,EAAE,OAAO,EAAE,QAAQ,GAAI;AACxC,CAAC;;;ADVD,OAAO,OAAO;AAEP,SAAS,WAAW,UAA2B,CAAC,GAAW;AAEhE,MAAI,SAA0B;AAAA,IAC5B,QAAQ;AAAA,IACR,MAAM;AAAA,IACN,QAAQ;AAAA,IACR,aAAa;AAAA,IACb,SAAS;AAAA,IACT,WAAW;AAAA,IACX,cAAc;AAAA,IACd,OAAO;AAAA,IACP,eAAe;AAAA,EACjB;AAGA,QAAM,cAAc,CAAC,WAAW,cAAc;AAC9C,aAAW,QAAQ,aAAa;AAC9B,UAAM,aAAa,QAAQ,QAAQ,IAAI,GAAG,IAAI;AAC9C,QAAI,WAAW,UAAU,GAAG;AAC1B,UAAI;AAEF,cAAM,UAAU,aAAa,YAAY,OAAO;AAChD,cAAM,aAAa,KAAK,MAAM,OAAO;AACrC,iBAAS,EAAE,GAAG,QAAQ,GAAG,WAAW;AACpC,gBAAQ,IAAI,gCAAyB,IAAI,EAAE;AAC3C;AAAA,MACF,SAAS,OAAO;AACd,gBAAQ,KAAK,4CAAkC,IAAI,KAAK,KAAK;AAAA,MAC/D;AAAA,IACF;AAAA,EACF;AAGA,MAAI,QAAQ,IAAI,iBAAiB;AAC/B,WAAO,YAAY,QAAQ,IAAI;AAAA,EACjC;AACA,MAAI,QAAQ,IAAI,cAAc;AAC5B,WAAO,SAAS,QAAQ,IAAI;AAAA,EAC9B;AACA,MAAI,QAAQ,IAAI,cAAc;AAC5B,WAAO,SAAS,QAAQ,IAAI;AAAA,EAC9B;AACA,MAAI,QAAQ,IAAI,WAAW;AACzB,WAAO,OAAO,QAAQ,IAAI;AAAA,EAC5B;AACA,MAAI,QAAQ,IAAI,aAAa;AAC3B,WAAO,SAAS,QAAQ,IAAI;AAAA,EAC9B;AACA,MAAI,QAAQ,IAAI,kBAAkB;AAChC,WAAO,cAAc,QAAQ,IAAI;AAAA,EACnC;AAGA,WAAS,EAAE,GAAG,QAAQ,GAAG,QAAQ;AAGjC,MAAI;AACF,WAAO,aAAa,MAAM,MAAM;AAAA,EAClC,SAAS,OAAO;AACd,QAAI,iBAAiB,OAAO;AAC1B,YAAM,IAAI,MAAM,0BAA0B,MAAM,OAAO,EAAE;AAAA,IAC3D;AACA,UAAM;AAAA,EACR;AACF;AAEO,SAAS,eAAe,QAA0B;AACvD,QAAM,SAAmB,CAAC;AAE1B,MAAI,CAAC,OAAO,WAAW;AACrB,WAAO,KAAK,8FAA8F;AAAA,EAC5G;AAEA,MAAI,CAAC,OAAO,QAAQ;AAClB,WAAO,KAAK,qFAAqF;AAAA,EACnG;AAEA,SAAO;AACT;","names":[]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
#!/usr/bin/env node
|