string-catalog-mcp 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.prettierrc +8 -0
- package/README.md +127 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +23 -0
- package/dist/mcp/prompts/batch-translate.d.ts +2 -0
- package/dist/mcp/prompts/batch-translate.js +88 -0
- package/dist/mcp/prompts/index.d.ts +2 -0
- package/dist/mcp/prompts/index.js +12 -0
- package/dist/mcp/prompts/review-translations.d.ts +2 -0
- package/dist/mcp/prompts/review-translations.js +75 -0
- package/dist/mcp/prompts/translate-strings.d.ts +2 -0
- package/dist/mcp/prompts/translate-strings.js +81 -0
- package/dist/mcp/tools/get-catalog-statistics.d.ts +2 -0
- package/dist/mcp/tools/get-catalog-statistics.js +25 -0
- package/dist/mcp/tools/get-translations-for-key.d.ts +2 -0
- package/dist/mcp/tools/get-translations-for-key.js +36 -0
- package/dist/mcp/tools/index.d.ts +2 -0
- package/dist/mcp/tools/index.js +18 -0
- package/dist/mcp/tools/list-all-keys.d.ts +2 -0
- package/dist/mcp/tools/list-all-keys.js +44 -0
- package/dist/mcp/tools/list-supported-languages.d.ts +2 -0
- package/dist/mcp/tools/list-supported-languages.js +30 -0
- package/dist/mcp/tools/search-keys.d.ts +2 -0
- package/dist/mcp/tools/search-keys.js +32 -0
- package/dist/mcp/tools/update-translations.d.ts +2 -0
- package/dist/mcp/tools/update-translations.js +78 -0
- package/dist/string-catalog.d.ts +53 -0
- package/dist/string-catalog.js +220 -0
- package/dist/tools/get-catalog-statistics.d.ts +2 -0
- package/dist/tools/get-catalog-statistics.js +25 -0
- package/dist/tools/get-translations-for-key.d.ts +2 -0
- package/dist/tools/get-translations-for-key.js +36 -0
- package/dist/tools/index.d.ts +2 -0
- package/dist/tools/index.js +18 -0
- package/dist/tools/list-all-keys.d.ts +2 -0
- package/dist/tools/list-all-keys.js +44 -0
- package/dist/tools/list-supported-languages.d.ts +2 -0
- package/dist/tools/list-supported-languages.js +30 -0
- package/dist/tools/search-keys.d.ts +2 -0
- package/dist/tools/search-keys.js +32 -0
- package/dist/tools/update-translations.d.ts +2 -0
- package/dist/tools/update-translations.js +78 -0
- package/dist/types.d.ts +86 -0
- package/dist/types.js +6 -0
- package/eslint.config.js +23 -0
- package/images/mcp.jpeg +0 -0
- package/package.json +49 -0
- package/src/index.ts +25 -0
- package/src/mcp/prompts/batch-translate.ts +91 -0
- package/src/mcp/prompts/index.ts +10 -0
- package/src/mcp/prompts/review-translations.ts +79 -0
- package/src/mcp/prompts/translate-strings.ts +85 -0
- package/src/mcp/tools/get-catalog-statistics.ts +29 -0
- package/src/mcp/tools/get-translations-for-key.ts +45 -0
- package/src/mcp/tools/index.ts +16 -0
- package/src/mcp/tools/list-all-keys.ts +52 -0
- package/src/mcp/tools/list-supported-languages.ts +38 -0
- package/src/mcp/tools/search-keys.ts +40 -0
- package/src/mcp/tools/update-translations.ts +89 -0
- package/src/string-catalog.ts +232 -0
- package/src/types.ts +82 -0
- package/tsconfig.json +28 -0
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerListSupportedLanguages = registerListSupportedLanguages;
|
|
4
|
+
const zod_1 = require("zod");
|
|
5
|
+
const string_catalog_1 = require("../../string-catalog");
|
|
6
|
+
function registerListSupportedLanguages(server) {
|
|
7
|
+
server.registerTool('list_supported_languages', {
|
|
8
|
+
description: 'List all supported languages in a given Xcode String Catalog (.xcstrings) file. Returns the source language and all languages that have translations.',
|
|
9
|
+
inputSchema: {
|
|
10
|
+
filePath: zod_1.z.string().describe('Absolute path to the .xcstrings file'),
|
|
11
|
+
},
|
|
12
|
+
}, async ({ filePath }) => {
|
|
13
|
+
const catalog = new string_catalog_1.StringCatalog(filePath);
|
|
14
|
+
const languages = catalog.getSupportedLanguages();
|
|
15
|
+
const sourceLanguage = catalog.getSourceLanguage();
|
|
16
|
+
return {
|
|
17
|
+
content: [
|
|
18
|
+
{
|
|
19
|
+
type: 'text',
|
|
20
|
+
text: JSON.stringify({
|
|
21
|
+
sourceLanguage,
|
|
22
|
+
supportedLanguages: languages,
|
|
23
|
+
count: languages.length,
|
|
24
|
+
}, null, 2),
|
|
25
|
+
},
|
|
26
|
+
],
|
|
27
|
+
};
|
|
28
|
+
});
|
|
29
|
+
}
|
|
30
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlzdC1zdXBwb3J0ZWQtbGFuZ3VhZ2VzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL21jcC90b29scy9saXN0LXN1cHBvcnRlZC1sYW5ndWFnZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFJQSx3RUFpQ0M7QUFwQ0QsNkJBQXdCO0FBQ3hCLHlEQUFxRDtBQUVyRCxTQUFnQiw4QkFBOEIsQ0FBQyxNQUFpQjtJQUM1RCxNQUFNLENBQUMsWUFBWSxDQUNmLDBCQUEwQixFQUMxQjtRQUNJLFdBQVcsRUFDUCx1SkFBdUo7UUFDM0osV0FBVyxFQUFFO1lBQ1QsUUFBUSxFQUFFLE9BQUMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxRQUFRLENBQUMsc0NBQXNDLENBQUM7U0FDeEU7S0FDSixFQUNELEtBQUssRUFBRSxFQUFFLFFBQVEsRUFBRSxFQUFFLEVBQUU7UUFDbkIsTUFBTSxPQUFPLEdBQUcsSUFBSSw4QkFBYSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQzVDLE1BQU0sU0FBUyxHQUFHLE9BQU8sQ0FBQyxxQkFBcUIsRUFBRSxDQUFDO1FBQ2xELE1BQU0sY0FBYyxHQUFHLE9BQU8sQ0FBQyxpQkFBaUIsRUFBRSxDQUFDO1FBRW5ELE9BQU87WUFDSCxPQUFPLEVBQUU7Z0JBQ0w7b0JBQ0ksSUFBSSxFQUFFLE1BQWU7b0JBQ3JCLElBQUksRUFBRSxJQUFJLENBQUMsU0FBUyxDQUNoQjt3QkFDSSxjQUFjO3dCQUNkLGtCQUFrQixFQUFFLFNBQVM7d0JBQzdCLEtBQUssRUFBRSxTQUFTLENBQUMsTUFBTTtxQkFDMUIsRUFDRCxJQUFJLEVBQ0osQ0FBQyxDQUNKO2lCQUNKO2FBQ0o7U0FDSixDQUFDO0lBQ04sQ0FBQyxDQUNKLENBQUM7QUFDTixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTWNwU2VydmVyIH0gZnJvbSAnQG1vZGVsY29udGV4dHByb3RvY29sL3Nkay9zZXJ2ZXIvbWNwLmpzJztcbmltcG9ydCB7IHogfSBmcm9tICd6b2QnO1xuaW1wb3J0IHsgU3RyaW5nQ2F0YWxvZyB9IGZyb20gJy4uLy4uL3N0cmluZy1jYXRhbG9nJztcblxuZXhwb3J0IGZ1bmN0aW9uIHJlZ2lzdGVyTGlzdFN1cHBvcnRlZExhbmd1YWdlcyhzZXJ2ZXI6IE1jcFNlcnZlcikge1xuICAgIHNlcnZlci5yZWdpc3RlclRvb2woXG4gICAgICAgICdsaXN0X3N1cHBvcnRlZF9sYW5ndWFnZXMnLFxuICAgICAgICB7XG4gICAgICAgICAgICBkZXNjcmlwdGlvbjpcbiAgICAgICAgICAgICAgICAnTGlzdCBhbGwgc3VwcG9ydGVkIGxhbmd1YWdlcyBpbiBhIGdpdmVuIFhjb2RlIFN0cmluZyBDYXRhbG9nICgueGNzdHJpbmdzKSBmaWxlLiBSZXR1cm5zIHRoZSBzb3VyY2UgbGFuZ3VhZ2UgYW5kIGFsbCBsYW5ndWFnZXMgdGhhdCBoYXZlIHRyYW5zbGF0aW9ucy4nLFxuICAgICAgICAgICAgaW5wdXRTY2hlbWE6IHtcbiAgICAgICAgICAgICAgICBmaWxlUGF0aDogei5zdHJpbmcoKS5kZXNjcmliZSgnQWJzb2x1dGUgcGF0aCB0byB0aGUgLnhjc3RyaW5ncyBmaWxlJyksXG4gICAgICAgICAgICB9LFxuICAgICAgICB9LFxuICAgICAgICBhc3luYyAoeyBmaWxlUGF0aCB9KSA9PiB7XG4gICAgICAgICAgICBjb25zdCBjYXRhbG9nID0gbmV3IFN0cmluZ0NhdGFsb2coZmlsZVBhdGgpO1xuICAgICAgICAgICAgY29uc3QgbGFuZ3VhZ2VzID0gY2F0YWxvZy5nZXRTdXBwb3J0ZWRMYW5ndWFnZXMoKTtcbiAgICAgICAgICAgIGNvbnN0IHNvdXJjZUxhbmd1YWdlID0gY2F0YWxvZy5nZXRTb3VyY2VMYW5ndWFnZSgpO1xuXG4gICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAgIGNvbnRlbnQ6IFtcbiAgICAgICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICAgICAgdHlwZTogJ3RleHQnIGFzIGNvbnN0LFxuICAgICAgICAgICAgICAgICAgICAgICAgdGV4dDogSlNPTi5zdHJpbmdpZnkoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzb3VyY2VMYW5ndWFnZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3VwcG9ydGVkTGFuZ3VhZ2VzOiBsYW5ndWFnZXMsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvdW50OiBsYW5ndWFnZXMubGVuZ3RoLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgbnVsbCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAyXG4gICAgICAgICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgIF0sXG4gICAgICAgICAgICB9O1xuICAgICAgICB9XG4gICAgKTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerSearchKeys = registerSearchKeys;
|
|
4
|
+
const zod_1 = require("zod");
|
|
5
|
+
const string_catalog_1 = require("../../string-catalog");
|
|
6
|
+
function registerSearchKeys(server) {
|
|
7
|
+
server.registerTool('search_keys', {
|
|
8
|
+
description: 'Search for localization keys containing a specific substring. Useful for finding keys when you only know part of the key name.',
|
|
9
|
+
inputSchema: {
|
|
10
|
+
filePath: zod_1.z.string().describe('Absolute path to the .xcstrings file'),
|
|
11
|
+
query: zod_1.z
|
|
12
|
+
.string()
|
|
13
|
+
.describe('Substring to search for in key names (case-insensitive)'),
|
|
14
|
+
},
|
|
15
|
+
}, async ({ filePath, query }) => {
|
|
16
|
+
const catalog = new string_catalog_1.StringCatalog(filePath);
|
|
17
|
+
const keys = catalog.searchKeys(query);
|
|
18
|
+
return {
|
|
19
|
+
content: [
|
|
20
|
+
{
|
|
21
|
+
type: 'text',
|
|
22
|
+
text: JSON.stringify({
|
|
23
|
+
query,
|
|
24
|
+
matchingKeys: keys,
|
|
25
|
+
count: keys.length,
|
|
26
|
+
}, null, 2),
|
|
27
|
+
},
|
|
28
|
+
],
|
|
29
|
+
};
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VhcmNoLWtleXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbWNwL3Rvb2xzL3NlYXJjaC1rZXlzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBSUEsZ0RBbUNDO0FBdENELDZCQUF3QjtBQUN4Qix5REFBcUQ7QUFFckQsU0FBZ0Isa0JBQWtCLENBQUMsTUFBaUI7SUFDaEQsTUFBTSxDQUFDLFlBQVksQ0FDZixhQUFhLEVBQ2I7UUFDSSxXQUFXLEVBQ1AsZ0lBQWdJO1FBQ3BJLFdBQVcsRUFBRTtZQUNULFFBQVEsRUFBRSxPQUFDLENBQUMsTUFBTSxFQUFFLENBQUMsUUFBUSxDQUFDLHNDQUFzQyxDQUFDO1lBQ3JFLEtBQUssRUFBRSxPQUFDO2lCQUNILE1BQU0sRUFBRTtpQkFDUixRQUFRLENBQUMseURBQXlELENBQUM7U0FDM0U7S0FDSixFQUNELEtBQUssRUFBRSxFQUFFLFFBQVEsRUFBRSxLQUFLLEVBQUUsRUFBRSxFQUFFO1FBQzFCLE1BQU0sT0FBTyxHQUFHLElBQUksOEJBQWEsQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUM1QyxNQUFNLElBQUksR0FBRyxPQUFPLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBRXZDLE9BQU87WUFDSCxPQUFPLEVBQUU7Z0JBQ0w7b0JBQ0ksSUFBSSxFQUFFLE1BQWU7b0JBQ3JCLElBQUksRUFBRSxJQUFJLENBQUMsU0FBUyxDQUNoQjt3QkFDSSxLQUFLO3dCQUNMLFlBQVksRUFBRSxJQUFJO3dCQUNsQixLQUFLLEVBQUUsSUFBSSxDQUFDLE1BQU07cUJBQ3JCLEVBQ0QsSUFBSSxFQUNKLENBQUMsQ0FDSjtpQkFDSjthQUNKO1NBQ0osQ0FBQztJQUNOLENBQUMsQ0FDSixDQUFDO0FBQ04sQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE1jcFNlcnZlciB9IGZyb20gJ0Btb2RlbGNvbnRleHRwcm90b2NvbC9zZGsvc2VydmVyL21jcC5qcyc7XG5pbXBvcnQgeyB6IH0gZnJvbSAnem9kJztcbmltcG9ydCB7IFN0cmluZ0NhdGFsb2cgfSBmcm9tICcuLi8uLi9zdHJpbmctY2F0YWxvZyc7XG5cbmV4cG9ydCBmdW5jdGlvbiByZWdpc3RlclNlYXJjaEtleXMoc2VydmVyOiBNY3BTZXJ2ZXIpIHtcbiAgICBzZXJ2ZXIucmVnaXN0ZXJUb29sKFxuICAgICAgICAnc2VhcmNoX2tleXMnLFxuICAgICAgICB7XG4gICAgICAgICAgICBkZXNjcmlwdGlvbjpcbiAgICAgICAgICAgICAgICAnU2VhcmNoIGZvciBsb2NhbGl6YXRpb24ga2V5cyBjb250YWluaW5nIGEgc3BlY2lmaWMgc3Vic3RyaW5nLiBVc2VmdWwgZm9yIGZpbmRpbmcga2V5cyB3aGVuIHlvdSBvbmx5IGtub3cgcGFydCBvZiB0aGUga2V5IG5hbWUuJyxcbiAgICAgICAgICAgIGlucHV0U2NoZW1hOiB7XG4gICAgICAgICAgICAgICAgZmlsZVBhdGg6IHouc3RyaW5nKCkuZGVzY3JpYmUoJ0Fic29sdXRlIHBhdGggdG8gdGhlIC54Y3N0cmluZ3MgZmlsZScpLFxuICAgICAgICAgICAgICAgIHF1ZXJ5OiB6XG4gICAgICAgICAgICAgICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAgICAgICAgICAgICAuZGVzY3JpYmUoJ1N1YnN0cmluZyB0byBzZWFyY2ggZm9yIGluIGtleSBuYW1lcyAoY2FzZS1pbnNlbnNpdGl2ZSknKSxcbiAgICAgICAgICAgIH0sXG4gICAgICAgIH0sXG4gICAgICAgIGFzeW5jICh7IGZpbGVQYXRoLCBxdWVyeSB9KSA9PiB7XG4gICAgICAgICAgICBjb25zdCBjYXRhbG9nID0gbmV3IFN0cmluZ0NhdGFsb2coZmlsZVBhdGgpO1xuICAgICAgICAgICAgY29uc3Qga2V5cyA9IGNhdGFsb2cuc2VhcmNoS2V5cyhxdWVyeSk7XG5cbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgY29udGVudDogW1xuICAgICAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgICAgICB0eXBlOiAndGV4dCcgYXMgY29uc3QsXG4gICAgICAgICAgICAgICAgICAgICAgICB0ZXh0OiBKU09OLnN0cmluZ2lmeShcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHF1ZXJ5LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBtYXRjaGluZ0tleXM6IGtleXMsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvdW50OiBrZXlzLmxlbmd0aCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIG51bGwsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgMlxuICAgICAgICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgICBdLFxuICAgICAgICAgICAgfTtcbiAgICAgICAgfVxuICAgICk7XG59XG4iXX0=
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerUpdateTranslations = registerUpdateTranslations;
|
|
4
|
+
const zod_1 = require("zod");
|
|
5
|
+
const string_catalog_1 = require("../../string-catalog");
|
|
6
|
+
const toolDescription = `Update or add translations to a String Catalog. Accepts an array of translation entries.
|
|
7
|
+
|
|
8
|
+
IMPORTANT: iOS strings support format placeholders that must be preserved in translations:
|
|
9
|
+
- %@ for strings (objects)
|
|
10
|
+
- %d or %lld for integers
|
|
11
|
+
- %f for floating point numbers
|
|
12
|
+
- %1$@, %2$@ etc. for positional arguments (order can be changed in translations)
|
|
13
|
+
|
|
14
|
+
Example input:
|
|
15
|
+
{
|
|
16
|
+
"data": [
|
|
17
|
+
{
|
|
18
|
+
"key": "hello_world",
|
|
19
|
+
"translations": [
|
|
20
|
+
{ "language": "en", "value": "Hello World" },
|
|
21
|
+
{ "language": "de", "value": "Hallo Welt" },
|
|
22
|
+
{ "language": "no", "value": "Hei Verden" }
|
|
23
|
+
],
|
|
24
|
+
"comment": "Greeting message shown on home screen"
|
|
25
|
+
},
|
|
26
|
+
{
|
|
27
|
+
"key": "items_count",
|
|
28
|
+
"translations": [
|
|
29
|
+
{ "language": "en", "value": "%lld items" },
|
|
30
|
+
{ "language": "de", "value": "%lld Elemente" }
|
|
31
|
+
]
|
|
32
|
+
}
|
|
33
|
+
]
|
|
34
|
+
}`;
|
|
35
|
+
const translationSchema = zod_1.z.object({
|
|
36
|
+
language: zod_1.z.string().describe('Language code (e.g., "en", "de", "no", "vi")'),
|
|
37
|
+
value: zod_1.z
|
|
38
|
+
.string()
|
|
39
|
+
.describe('The translated text. Preserve any format placeholders like %@, %lld, %d'),
|
|
40
|
+
state: zod_1.z
|
|
41
|
+
.enum(['new', 'translated', 'needs_review', 'stale'])
|
|
42
|
+
.optional()
|
|
43
|
+
.describe('Translation state (defaults to "translated")'),
|
|
44
|
+
});
|
|
45
|
+
const translationsSchema = zod_1.z.object({
|
|
46
|
+
key: zod_1.z.string().describe('The localization key'),
|
|
47
|
+
translations: zod_1.z.array(translationSchema).describe('Array of language translations'),
|
|
48
|
+
comment: zod_1.z.string().optional().describe('Optional comment describing the string context'),
|
|
49
|
+
});
|
|
50
|
+
const inputSchema = zod_1.z.object({
|
|
51
|
+
filePath: zod_1.z.string().describe('Absolute path to the .xcstrings file'),
|
|
52
|
+
data: zod_1.z.array(translationsSchema).describe('Array of translation entries to add or update'),
|
|
53
|
+
});
|
|
54
|
+
function registerUpdateTranslations(server) {
|
|
55
|
+
server.registerTool('update_translations', {
|
|
56
|
+
description: toolDescription,
|
|
57
|
+
inputSchema,
|
|
58
|
+
}, async ({ filePath, data }) => {
|
|
59
|
+
const catalog = new string_catalog_1.StringCatalog(filePath);
|
|
60
|
+
const result = catalog.updateTranslations(data);
|
|
61
|
+
catalog.save();
|
|
62
|
+
return {
|
|
63
|
+
content: [
|
|
64
|
+
{
|
|
65
|
+
type: 'text',
|
|
66
|
+
text: JSON.stringify({
|
|
67
|
+
success: true,
|
|
68
|
+
updatedKeys: result.updated,
|
|
69
|
+
createdKeys: result.created,
|
|
70
|
+
totalUpdated: result.updated.length,
|
|
71
|
+
totalCreated: result.created.length,
|
|
72
|
+
}, null, 2),
|
|
73
|
+
},
|
|
74
|
+
],
|
|
75
|
+
};
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBkYXRlLXRyYW5zbGF0aW9ucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9tY3AvdG9vbHMvdXBkYXRlLXRyYW5zbGF0aW9ucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQXdEQSxnRUFnQ0M7QUF2RkQsNkJBQXdCO0FBQ3hCLHlEQUFxRDtBQUVyRCxNQUFNLGVBQWUsR0FBRzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztFQTRCdEIsQ0FBQztBQUVILE1BQU0saUJBQWlCLEdBQUcsT0FBQyxDQUFDLE1BQU0sQ0FBQztJQUMvQixRQUFRLEVBQUUsT0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDLFFBQVEsQ0FBQyw4Q0FBOEMsQ0FBQztJQUM3RSxLQUFLLEVBQUUsT0FBQztTQUNILE1BQU0sRUFBRTtTQUNSLFFBQVEsQ0FBQyx5RUFBeUUsQ0FBQztJQUN4RixLQUFLLEVBQUUsT0FBQztTQUNILElBQUksQ0FBQyxDQUFDLEtBQUssRUFBRSxZQUFZLEVBQUUsY0FBYyxFQUFFLE9BQU8sQ0FBQyxDQUFDO1NBQ3BELFFBQVEsRUFBRTtTQUNWLFFBQVEsQ0FBQyw4Q0FBOEMsQ0FBQztDQUNoRSxDQUFDLENBQUM7QUFFSCxNQUFNLGtCQUFrQixHQUFHLE9BQUMsQ0FBQyxNQUFNLENBQUM7SUFDaEMsR0FBRyxFQUFFLE9BQUMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxRQUFRLENBQUMsc0JBQXNCLENBQUM7SUFDaEQsWUFBWSxFQUFFLE9BQUMsQ0FBQyxLQUFLLENBQUMsaUJBQWlCLENBQUMsQ0FBQyxRQUFRLENBQUMsZ0NBQWdDLENBQUM7SUFDbkYsT0FBTyxFQUFFLE9BQUMsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxRQUFRLEVBQUUsQ0FBQyxRQUFRLENBQUMsZ0RBQWdELENBQUM7Q0FDNUYsQ0FBQyxDQUFDO0FBRUgsTUFBTSxXQUFXLEdBQUcsT0FBQyxDQUFDLE1BQU0sQ0FBQztJQUN6QixRQUFRLEVBQUUsT0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDLFFBQVEsQ0FBQyxzQ0FBc0MsQ0FBQztJQUNyRSxJQUFJLEVBQUUsT0FBQyxDQUFDLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxDQUFDLFFBQVEsQ0FBQywrQ0FBK0MsQ0FBQztDQUM5RixDQUFDLENBQUM7QUFFSCxTQUFnQiwwQkFBMEIsQ0FBQyxNQUFpQjtJQUN4RCxNQUFNLENBQUMsWUFBWSxDQUNmLHFCQUFxQixFQUNyQjtRQUNJLFdBQVcsRUFBRSxlQUFlO1FBQzVCLFdBQVc7S0FDZCxFQUNELEtBQUssRUFBRSxFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUUsRUFBRSxFQUFFO1FBQ3pCLE1BQU0sT0FBTyxHQUFHLElBQUksOEJBQWEsQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUM1QyxNQUFNLE1BQU0sR0FBRyxPQUFPLENBQUMsa0JBQWtCLENBQUMsSUFBSSxDQUFDLENBQUM7UUFDaEQsT0FBTyxDQUFDLElBQUksRUFBRSxDQUFDO1FBRWYsT0FBTztZQUNILE9BQU8sRUFBRTtnQkFDTDtvQkFDSSxJQUFJLEVBQUUsTUFBZTtvQkFDckIsSUFBSSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQ2hCO3dCQUNJLE9BQU8sRUFBRSxJQUFJO3dCQUNiLFdBQVcsRUFBRSxNQUFNLENBQUMsT0FBTzt3QkFDM0IsV0FBVyxFQUFFLE1BQU0sQ0FBQyxPQUFPO3dCQUMzQixZQUFZLEVBQUUsTUFBTSxDQUFDLE9BQU8sQ0FBQyxNQUFNO3dCQUNuQyxZQUFZLEVBQUUsTUFBTSxDQUFDLE9BQU8sQ0FBQyxNQUFNO3FCQUN0QyxFQUNELElBQUksRUFDSixDQUFDLENBQ0o7aUJBQ0o7YUFDSjtTQUNKLENBQUM7SUFDTixDQUFDLENBQ0osQ0FBQztBQUNOLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBNY3BTZXJ2ZXIgfSBmcm9tICdAbW9kZWxjb250ZXh0cHJvdG9jb2wvc2RrL3NlcnZlci9tY3AuanMnO1xuaW1wb3J0IHsgeiB9IGZyb20gJ3pvZCc7XG5pbXBvcnQgeyBTdHJpbmdDYXRhbG9nIH0gZnJvbSAnLi4vLi4vc3RyaW5nLWNhdGFsb2cnO1xuXG5jb25zdCB0b29sRGVzY3JpcHRpb24gPSBgVXBkYXRlIG9yIGFkZCB0cmFuc2xhdGlvbnMgdG8gYSBTdHJpbmcgQ2F0YWxvZy4gQWNjZXB0cyBhbiBhcnJheSBvZiB0cmFuc2xhdGlvbiBlbnRyaWVzLlxuXG5JTVBPUlRBTlQ6IGlPUyBzdHJpbmdzIHN1cHBvcnQgZm9ybWF0IHBsYWNlaG9sZGVycyB0aGF0IG11c3QgYmUgcHJlc2VydmVkIGluIHRyYW5zbGF0aW9uczpcbi0gJUAgZm9yIHN0cmluZ3MgKG9iamVjdHMpXG4tICVkIG9yICVsbGQgZm9yIGludGVnZXJzXG4tICVmIGZvciBmbG9hdGluZyBwb2ludCBudW1iZXJzXG4tICUxJEAsICUyJEAgZXRjLiBmb3IgcG9zaXRpb25hbCBhcmd1bWVudHMgKG9yZGVyIGNhbiBiZSBjaGFuZ2VkIGluIHRyYW5zbGF0aW9ucylcblxuRXhhbXBsZSBpbnB1dDpcbntcbiAgXCJkYXRhXCI6IFtcbiAgICB7XG4gICAgICBcImtleVwiOiBcImhlbGxvX3dvcmxkXCIsXG4gICAgICBcInRyYW5zbGF0aW9uc1wiOiBbXG4gICAgICAgIHsgXCJsYW5ndWFnZVwiOiBcImVuXCIsIFwidmFsdWVcIjogXCJIZWxsbyBXb3JsZFwiIH0sXG4gICAgICAgIHsgXCJsYW5ndWFnZVwiOiBcImRlXCIsIFwidmFsdWVcIjogXCJIYWxsbyBXZWx0XCIgfSxcbiAgICAgICAgeyBcImxhbmd1YWdlXCI6IFwibm9cIiwgXCJ2YWx1ZVwiOiBcIkhlaSBWZXJkZW5cIiB9XG4gICAgICBdLFxuICAgICAgXCJjb21tZW50XCI6IFwiR3JlZXRpbmcgbWVzc2FnZSBzaG93biBvbiBob21lIHNjcmVlblwiXG4gICAgfSxcbiAgICB7XG4gICAgICBcImtleVwiOiBcIml0ZW1zX2NvdW50XCIsXG4gICAgICBcInRyYW5zbGF0aW9uc1wiOiBbXG4gICAgICAgIHsgXCJsYW5ndWFnZVwiOiBcImVuXCIsIFwidmFsdWVcIjogXCIlbGxkIGl0ZW1zXCIgfSxcbiAgICAgICAgeyBcImxhbmd1YWdlXCI6IFwiZGVcIiwgXCJ2YWx1ZVwiOiBcIiVsbGQgRWxlbWVudGVcIiB9XG4gICAgICBdXG4gICAgfVxuICBdXG59YDtcblxuY29uc3QgdHJhbnNsYXRpb25TY2hlbWEgPSB6Lm9iamVjdCh7XG4gICAgbGFuZ3VhZ2U6IHouc3RyaW5nKCkuZGVzY3JpYmUoJ0xhbmd1YWdlIGNvZGUgKGUuZy4sIFwiZW5cIiwgXCJkZVwiLCBcIm5vXCIsIFwidmlcIiknKSxcbiAgICB2YWx1ZTogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLmRlc2NyaWJlKCdUaGUgdHJhbnNsYXRlZCB0ZXh0LiBQcmVzZXJ2ZSBhbnkgZm9ybWF0IHBsYWNlaG9sZGVycyBsaWtlICVALCAlbGxkLCAlZCcpLFxuICAgIHN0YXRlOiB6XG4gICAgICAgIC5lbnVtKFsnbmV3JywgJ3RyYW5zbGF0ZWQnLCAnbmVlZHNfcmV2aWV3JywgJ3N0YWxlJ10pXG4gICAgICAgIC5vcHRpb25hbCgpXG4gICAgICAgIC5kZXNjcmliZSgnVHJhbnNsYXRpb24gc3RhdGUgKGRlZmF1bHRzIHRvIFwidHJhbnNsYXRlZFwiKScpLFxufSk7XG5cbmNvbnN0IHRyYW5zbGF0aW9uc1NjaGVtYSA9IHoub2JqZWN0KHtcbiAgICBrZXk6IHouc3RyaW5nKCkuZGVzY3JpYmUoJ1RoZSBsb2NhbGl6YXRpb24ga2V5JyksXG4gICAgdHJhbnNsYXRpb25zOiB6LmFycmF5KHRyYW5zbGF0aW9uU2NoZW1hKS5kZXNjcmliZSgnQXJyYXkgb2YgbGFuZ3VhZ2UgdHJhbnNsYXRpb25zJyksXG4gICAgY29tbWVudDogei5zdHJpbmcoKS5vcHRpb25hbCgpLmRlc2NyaWJlKCdPcHRpb25hbCBjb21tZW50IGRlc2NyaWJpbmcgdGhlIHN0cmluZyBjb250ZXh0JyksXG59KTtcblxuY29uc3QgaW5wdXRTY2hlbWEgPSB6Lm9iamVjdCh7XG4gICAgZmlsZVBhdGg6IHouc3RyaW5nKCkuZGVzY3JpYmUoJ0Fic29sdXRlIHBhdGggdG8gdGhlIC54Y3N0cmluZ3MgZmlsZScpLFxuICAgIGRhdGE6IHouYXJyYXkodHJhbnNsYXRpb25zU2NoZW1hKS5kZXNjcmliZSgnQXJyYXkgb2YgdHJhbnNsYXRpb24gZW50cmllcyB0byBhZGQgb3IgdXBkYXRlJyksXG59KTtcblxuZXhwb3J0IGZ1bmN0aW9uIHJlZ2lzdGVyVXBkYXRlVHJhbnNsYXRpb25zKHNlcnZlcjogTWNwU2VydmVyKSB7XG4gICAgc2VydmVyLnJlZ2lzdGVyVG9vbChcbiAgICAgICAgJ3VwZGF0ZV90cmFuc2xhdGlvbnMnLFxuICAgICAgICB7XG4gICAgICAgICAgICBkZXNjcmlwdGlvbjogdG9vbERlc2NyaXB0aW9uLFxuICAgICAgICAgICAgaW5wdXRTY2hlbWEsXG4gICAgICAgIH0sXG4gICAgICAgIGFzeW5jICh7IGZpbGVQYXRoLCBkYXRhIH0pID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGNhdGFsb2cgPSBuZXcgU3RyaW5nQ2F0YWxvZyhmaWxlUGF0aCk7XG4gICAgICAgICAgICBjb25zdCByZXN1bHQgPSBjYXRhbG9nLnVwZGF0ZVRyYW5zbGF0aW9ucyhkYXRhKTtcbiAgICAgICAgICAgIGNhdGFsb2cuc2F2ZSgpO1xuXG4gICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAgIGNvbnRlbnQ6IFtcbiAgICAgICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICAgICAgdHlwZTogJ3RleHQnIGFzIGNvbnN0LFxuICAgICAgICAgICAgICAgICAgICAgICAgdGV4dDogSlNPTi5zdHJpbmdpZnkoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzdWNjZXNzOiB0cnVlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1cGRhdGVkS2V5czogcmVzdWx0LnVwZGF0ZWQsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZWRLZXlzOiByZXN1bHQuY3JlYXRlZCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdG90YWxVcGRhdGVkOiByZXN1bHQudXBkYXRlZC5sZW5ndGgsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRvdGFsQ3JlYXRlZDogcmVzdWx0LmNyZWF0ZWQubGVuZ3RoLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgbnVsbCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAyXG4gICAgICAgICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgIF0sXG4gICAgICAgICAgICB9O1xuICAgICAgICB9XG4gICAgKTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { TranslationInput, KeyTranslationsResult } from './types';
|
|
2
|
+
/**
|
|
3
|
+
* StringCatalog class for reading and manipulating .xcstrings files
|
|
4
|
+
*/
|
|
5
|
+
export declare class StringCatalog {
|
|
6
|
+
private filePath;
|
|
7
|
+
private data;
|
|
8
|
+
constructor(filePath: string);
|
|
9
|
+
private load;
|
|
10
|
+
/**
|
|
11
|
+
* Get the source language of the catalog
|
|
12
|
+
*/
|
|
13
|
+
getSourceLanguage(): string;
|
|
14
|
+
/**
|
|
15
|
+
* Get all supported languages in the catalog
|
|
16
|
+
*/
|
|
17
|
+
getSupportedLanguages(): string[];
|
|
18
|
+
/**
|
|
19
|
+
* Get all keys in the catalog
|
|
20
|
+
*/
|
|
21
|
+
getAllKeys(): string[];
|
|
22
|
+
/**
|
|
23
|
+
* Get translations for a specific key
|
|
24
|
+
*/
|
|
25
|
+
getTranslationsForKey(key: string): KeyTranslationsResult | null;
|
|
26
|
+
/**
|
|
27
|
+
* Search for keys containing a specific substring
|
|
28
|
+
*/
|
|
29
|
+
searchKeys(query: string): string[];
|
|
30
|
+
/**
|
|
31
|
+
* Update translations for one or more keys
|
|
32
|
+
*/
|
|
33
|
+
updateTranslations(translations: TranslationInput[]): {
|
|
34
|
+
updated: string[];
|
|
35
|
+
created: string[];
|
|
36
|
+
};
|
|
37
|
+
/**
|
|
38
|
+
* Save changes to the file
|
|
39
|
+
*/
|
|
40
|
+
save(): void;
|
|
41
|
+
/**
|
|
42
|
+
* Get statistics about the catalog
|
|
43
|
+
*/
|
|
44
|
+
getStatistics(): {
|
|
45
|
+
totalKeys: number;
|
|
46
|
+
languages: string[];
|
|
47
|
+
translationCoverage: Record<string, {
|
|
48
|
+
translated: number;
|
|
49
|
+
total: number;
|
|
50
|
+
percentage: number;
|
|
51
|
+
}>;
|
|
52
|
+
};
|
|
53
|
+
}
|
|
@@ -0,0 +1,220 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
36
|
+
exports.StringCatalog = void 0;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const path = __importStar(require("path"));
|
|
39
|
+
/**
|
|
40
|
+
* StringCatalog class for reading and manipulating .xcstrings files
|
|
41
|
+
*/
|
|
42
|
+
class StringCatalog {
|
|
43
|
+
filePath;
|
|
44
|
+
data;
|
|
45
|
+
constructor(filePath) {
|
|
46
|
+
this.filePath = path.resolve(filePath);
|
|
47
|
+
this.data = this.load();
|
|
48
|
+
}
|
|
49
|
+
load() {
|
|
50
|
+
if (!fs.existsSync(this.filePath)) {
|
|
51
|
+
throw new Error(`String catalog file not found: ${this.filePath}`);
|
|
52
|
+
}
|
|
53
|
+
const content = fs.readFileSync(this.filePath, 'utf-8');
|
|
54
|
+
return JSON.parse(content);
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Get the source language of the catalog
|
|
58
|
+
*/
|
|
59
|
+
getSourceLanguage() {
|
|
60
|
+
return this.data.sourceLanguage;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Get all supported languages in the catalog
|
|
64
|
+
*/
|
|
65
|
+
getSupportedLanguages() {
|
|
66
|
+
const languages = new Set();
|
|
67
|
+
languages.add(this.data.sourceLanguage);
|
|
68
|
+
for (const key in this.data.strings) {
|
|
69
|
+
const entry = this.data.strings[key];
|
|
70
|
+
if (entry.localizations) {
|
|
71
|
+
for (const lang of Object.keys(entry.localizations)) {
|
|
72
|
+
languages.add(lang);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
return Array.from(languages).sort();
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Get all keys in the catalog
|
|
80
|
+
*/
|
|
81
|
+
getAllKeys() {
|
|
82
|
+
return Object.keys(this.data.strings).sort();
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Get translations for a specific key
|
|
86
|
+
*/
|
|
87
|
+
getTranslationsForKey(key) {
|
|
88
|
+
const entry = this.data.strings[key];
|
|
89
|
+
if (!entry) {
|
|
90
|
+
return null;
|
|
91
|
+
}
|
|
92
|
+
const translations = [];
|
|
93
|
+
if (entry.localizations) {
|
|
94
|
+
for (const [language, localization] of Object.entries(entry.localizations)) {
|
|
95
|
+
if (localization.stringUnit) {
|
|
96
|
+
translations.push({
|
|
97
|
+
language,
|
|
98
|
+
value: localization.stringUnit.value,
|
|
99
|
+
state: localization.stringUnit.state,
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
else if (localization.variations?.plural) {
|
|
103
|
+
// For plural strings, show the "other" form as the primary value
|
|
104
|
+
const plural = localization.variations.plural;
|
|
105
|
+
const primaryForm = plural.other || plural.one || plural.zero;
|
|
106
|
+
if (primaryForm) {
|
|
107
|
+
translations.push({
|
|
108
|
+
language,
|
|
109
|
+
value: `[plural] ${primaryForm.stringUnit.value}`,
|
|
110
|
+
state: primaryForm.stringUnit.state,
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
return {
|
|
117
|
+
key,
|
|
118
|
+
sourceLanguage: this.data.sourceLanguage,
|
|
119
|
+
translations: translations.sort((a, b) => a.language.localeCompare(b.language)),
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Search for keys containing a specific substring
|
|
124
|
+
*/
|
|
125
|
+
searchKeys(query) {
|
|
126
|
+
const lowerQuery = query.toLowerCase();
|
|
127
|
+
return this.getAllKeys().filter((key) => key.toLowerCase().includes(lowerQuery));
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* Update translations for one or more keys
|
|
131
|
+
*/
|
|
132
|
+
updateTranslations(translations) {
|
|
133
|
+
const updated = [];
|
|
134
|
+
const created = [];
|
|
135
|
+
for (const translation of translations) {
|
|
136
|
+
const { key, translations: langTranslations, comment } = translation;
|
|
137
|
+
const isNew = !this.data.strings[key];
|
|
138
|
+
if (isNew) {
|
|
139
|
+
this.data.strings[key] = {};
|
|
140
|
+
created.push(key);
|
|
141
|
+
}
|
|
142
|
+
else {
|
|
143
|
+
updated.push(key);
|
|
144
|
+
}
|
|
145
|
+
const entry = this.data.strings[key];
|
|
146
|
+
if (comment) {
|
|
147
|
+
entry.comment = comment;
|
|
148
|
+
}
|
|
149
|
+
if (!entry.localizations) {
|
|
150
|
+
entry.localizations = {};
|
|
151
|
+
}
|
|
152
|
+
for (const langTrans of langTranslations) {
|
|
153
|
+
entry.localizations[langTrans.language] = {
|
|
154
|
+
stringUnit: {
|
|
155
|
+
state: langTrans.state || 'translated',
|
|
156
|
+
value: langTrans.value,
|
|
157
|
+
},
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
return { updated, created };
|
|
162
|
+
}
|
|
163
|
+
/**
|
|
164
|
+
* Save changes to the file
|
|
165
|
+
*/
|
|
166
|
+
save() {
|
|
167
|
+
// Sort keys alphabetically for consistent output
|
|
168
|
+
const sortedStrings = {};
|
|
169
|
+
const keys = Object.keys(this.data.strings).sort();
|
|
170
|
+
for (const key of keys) {
|
|
171
|
+
sortedStrings[key] = this.data.strings[key];
|
|
172
|
+
}
|
|
173
|
+
const output = {
|
|
174
|
+
sourceLanguage: this.data.sourceLanguage,
|
|
175
|
+
strings: sortedStrings,
|
|
176
|
+
};
|
|
177
|
+
if (this.data.version) {
|
|
178
|
+
output.version = this.data.version;
|
|
179
|
+
}
|
|
180
|
+
// Write with 2-space indentation to match Xcode format
|
|
181
|
+
fs.writeFileSync(this.filePath, JSON.stringify(output, null, 2) + '\n', 'utf-8');
|
|
182
|
+
}
|
|
183
|
+
/**
|
|
184
|
+
* Get statistics about the catalog
|
|
185
|
+
*/
|
|
186
|
+
getStatistics() {
|
|
187
|
+
const languages = this.getSupportedLanguages();
|
|
188
|
+
const allKeys = this.getAllKeys();
|
|
189
|
+
const totalKeys = allKeys.length;
|
|
190
|
+
const translationCoverage = {};
|
|
191
|
+
for (const lang of languages) {
|
|
192
|
+
let translated = 0;
|
|
193
|
+
for (const key of allKeys) {
|
|
194
|
+
const entry = this.data.strings[key];
|
|
195
|
+
const localization = entry.localizations?.[lang];
|
|
196
|
+
if (localization?.stringUnit?.state === 'translated') {
|
|
197
|
+
translated++;
|
|
198
|
+
}
|
|
199
|
+
else if (localization?.variations?.plural) {
|
|
200
|
+
// Count plural as translated if it has an "other" form
|
|
201
|
+
if (localization.variations.plural.other?.stringUnit?.state === 'translated') {
|
|
202
|
+
translated++;
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
translationCoverage[lang] = {
|
|
207
|
+
translated,
|
|
208
|
+
total: totalKeys,
|
|
209
|
+
percentage: totalKeys > 0 ? Math.round((translated / totalKeys) * 100) : 0,
|
|
210
|
+
};
|
|
211
|
+
}
|
|
212
|
+
return {
|
|
213
|
+
totalKeys,
|
|
214
|
+
languages,
|
|
215
|
+
translationCoverage,
|
|
216
|
+
};
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
exports.StringCatalog = StringCatalog;
|
|
220
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"string-catalog.js","sourceRoot":"","sources":["../src/string-catalog.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,uCAAyB;AACzB,2CAA6B;AAU7B;;GAEG;AACH,MAAa,aAAa;IACd,QAAQ,CAAS;IACjB,IAAI,CAAY;IAExB,YAAY,QAAgB;QACxB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC;IAC5B,CAAC;IAEO,IAAI;QACR,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;YAChC,MAAM,IAAI,KAAK,CAAC,kCAAkC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;QACvE,CAAC;QAED,MAAM,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QACxD,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAc,CAAC;IAC5C,CAAC;IAED;;OAEG;IACH,iBAAiB;QACb,OAAO,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC;IACpC,CAAC;IAED;;OAEG;IACH,qBAAqB;QACjB,MAAM,SAAS,GAAG,IAAI,GAAG,EAAU,CAAC;QACpC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QAExC,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;YAClC,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;YACrC,IAAI,KAAK,CAAC,aAAa,EAAE,CAAC;gBACtB,KAAK,MAAM,IAAI,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,EAAE,CAAC;oBAClD,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;gBACxB,CAAC;YACL,CAAC;QACL,CAAC;QAED,OAAO,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,IAAI,EAAE,CAAC;IACxC,CAAC;IAED;;OAEG;IACH,UAAU;QACN,OAAO,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,CAAC;IACjD,CAAC;IAED;;OAEG;IACH,qBAAqB,CAAC,GAAW;QAC7B,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QACrC,IAAI,CAAC,KAAK,EAAE,CAAC;YACT,OAAO,IAAI,CAAC;QAChB,CAAC;QAED,MAAM,YAAY,GAAqB,EAAE,CAAC;QAE1C,IAAI,KAAK,CAAC,aAAa,EAAE,CAAC;YACtB,KAAK,MAAM,CAAC,QAAQ,EAAE,YAAY,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,aAAa,CAAC,EAAE,CAAC;gBACzE,IAAI,YAAY,CAAC,UAAU,EAAE,CAAC;oBAC1B,YAAY,CAAC,IAAI,CAAC;wBACd,QAAQ;wBACR,KAAK,EAAE,YAAY,CAAC,UAAU,CAAC,KAAK;wBACpC,KAAK,EAAE,YAAY,CAAC,UAAU,CAAC,KAAK;qBACvC,CAAC,CAAC;gBACP,CAAC;qBAAM,IAAI,YAAY,CAAC,UAAU,EAAE,MAAM,EAAE,CAAC;oBACzC,iEAAiE;oBACjE,MAAM,MAAM,GAAG,YAAY,CAAC,UAAU,CAAC,MAAM,CAAC;oBAC9C,MAAM,WAAW,GAAG,MAAM,CAAC,KAAK,IAAI,MAAM,CAAC,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC;oBAC9D,IAAI,WAAW,EAAE,CAAC;wBACd,YAAY,CAAC,IAAI,CAAC;4BACd,QAAQ;4BACR,KAAK,EAAE,YAAY,WAAW,CAAC,UAAU,CAAC,KAAK,EAAE;4BACjD,KAAK,EAAE,WAAW,CAAC,UAAU,CAAC,KAAK;yBACtC,CAAC,CAAC;oBACP,CAAC;gBACL,CAAC;YACL,CAAC;QACL,CAAC;QAED,OAAO;YACH,GAAG;YACH,cAAc,EAAE,IAAI,CAAC,IAAI,CAAC,cAAc;YACxC,YAAY,EAAE,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;SAClF,CAAC;IACN,CAAC;IAED;;OAEG;IACH,UAAU,CAAC,KAAa;QACpB,MAAM,UAAU,GAAG,KAAK,CAAC,WAAW,EAAE,CAAC;QACvC,OAAO,IAAI,CAAC,UAAU,EAAE,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC;IACrF,CAAC;IAED;;OAEG;IACH,kBAAkB,CAAC,YAAgC;QAC/C,MAAM,OAAO,GAAa,EAAE,CAAC;QAC7B,MAAM,OAAO,GAAa,EAAE,CAAC;QAE7B,KAAK,MAAM,WAAW,IAAI,YAAY,EAAE,CAAC;YACrC,MAAM,EAAE,GAAG,EAAE,YAAY,EAAE,gBAAgB,EAAE,OAAO,EAAE,GAAG,WAAW,CAAC;YAErE,MAAM,KAAK,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;YAEtC,IAAI,KAAK,EAAE,CAAC;gBACR,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC;gBAC5B,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACtB,CAAC;iBAAM,CAAC;gBACJ,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YACtB,CAAC;YAED,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;YAErC,IAAI,OAAO,EAAE,CAAC;gBACV,KAAK,CAAC,OAAO,GAAG,OAAO,CAAC;YAC5B,CAAC;YAED,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;gBACvB,KAAK,CAAC,aAAa,GAAG,EAAE,CAAC;YAC7B,CAAC;YAED,KAAK,MAAM,SAAS,IAAI,gBAAgB,EAAE,CAAC;gBACvC,KAAK,CAAC,aAAa,CAAC,SAAS,CAAC,QAAQ,CAAC,GAAG;oBACtC,UAAU,EAAE;wBACR,KAAK,EAAE,SAAS,CAAC,KAAK,IAAI,YAAY;wBACtC,KAAK,EAAE,SAAS,CAAC,KAAK;qBACzB;iBACJ,CAAC;YACN,CAAC;QACL,CAAC;QAED,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC;IAChC,CAAC;IAED;;OAEG;IACH,IAAI;QACA,iDAAiD;QACjD,MAAM,aAAa,GAAmC,EAAE,CAAC;QACzD,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,CAAC;QAEnD,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACrB,aAAa,CAAC,GAAG,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAChD,CAAC;QAED,MAAM,MAAM,GAAc;YACtB,cAAc,EAAE,IAAI,CAAC,IAAI,CAAC,cAAc;YACxC,OAAO,EAAE,aAAa;SACzB,CAAC;QAEF,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;YACpB,MAAM,CAAC,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC;QACvC,CAAC;QAED,uDAAuD;QACvD,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,GAAG,IAAI,EAAE,OAAO,CAAC,CAAC;IACrF,CAAC;IAED;;OAEG;IACH,aAAa;QAQT,MAAM,SAAS,GAAG,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC/C,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QAClC,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC;QAEjC,MAAM,mBAAmB,GAGrB,EAAE,CAAC;QAEP,KAAK,MAAM,IAAI,IAAI,SAAS,EAAE,CAAC;YAC3B,IAAI,UAAU,GAAG,CAAC,CAAC;YAEnB,KAAK,MAAM,GAAG,IAAI,OAAO,EAAE,CAAC;gBACxB,MAAM,KAAK,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gBACrC,MAAM,YAAY,GAAG,KAAK,CAAC,aAAa,EAAE,CAAC,IAAI,CAAC,CAAC;gBAEjD,IAAI,YAAY,EAAE,UAAU,EAAE,KAAK,KAAK,YAAY,EAAE,CAAC;oBACnD,UAAU,EAAE,CAAC;gBACjB,CAAC;qBAAM,IAAI,YAAY,EAAE,UAAU,EAAE,MAAM,EAAE,CAAC;oBAC1C,uDAAuD;oBACvD,IAAI,YAAY,CAAC,UAAU,CAAC,MAAM,CAAC,KAAK,EAAE,UAAU,EAAE,KAAK,KAAK,YAAY,EAAE,CAAC;wBAC3E,UAAU,EAAE,CAAC;oBACjB,CAAC;gBACL,CAAC;YACL,CAAC;YAED,mBAAmB,CAAC,IAAI,CAAC,GAAG;gBACxB,UAAU;gBACV,KAAK,EAAE,SAAS;gBAChB,UAAU,EAAE,SAAS,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,UAAU,GAAG,SAAS,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;aAC7E,CAAC;QACN,CAAC;QAED,OAAO;YACH,SAAS;YACT,SAAS;YACT,mBAAmB;SACtB,CAAC;IACN,CAAC;CACJ;AAzND,sCAyNC","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport {\n    XCStrings,\n    StringEntry,\n    TranslationInput,\n    KeyTranslationsResult,\n    KeyTranslation,\n    LocalizationState,\n} from './types';\n\n/**\n * StringCatalog class for reading and manipulating .xcstrings files\n */\nexport class StringCatalog {\n    private filePath: string;\n    private data: XCStrings;\n\n    constructor(filePath: string) {\n        this.filePath = path.resolve(filePath);\n        this.data = this.load();\n    }\n\n    private load(): XCStrings {\n        if (!fs.existsSync(this.filePath)) {\n            throw new Error(`String catalog file not found: ${this.filePath}`);\n        }\n\n        const content = fs.readFileSync(this.filePath, 'utf-8');\n        return JSON.parse(content) as XCStrings;\n    }\n\n    /**\n     * Get the source language of the catalog\n     */\n    getSourceLanguage(): string {\n        return this.data.sourceLanguage;\n    }\n\n    /**\n     * Get all supported languages in the catalog\n     */\n    getSupportedLanguages(): string[] {\n        const languages = new Set<string>();\n        languages.add(this.data.sourceLanguage);\n\n        for (const key in this.data.strings) {\n            const entry = this.data.strings[key];\n            if (entry.localizations) {\n                for (const lang of Object.keys(entry.localizations)) {\n                    languages.add(lang);\n                }\n            }\n        }\n\n        return Array.from(languages).sort();\n    }\n\n    /**\n     * Get all keys in the catalog\n     */\n    getAllKeys(): string[] {\n        return Object.keys(this.data.strings).sort();\n    }\n\n    /**\n     * Get translations for a specific key\n     */\n    getTranslationsForKey(key: string): KeyTranslationsResult | null {\n        const entry = this.data.strings[key];\n        if (!entry) {\n            return null;\n        }\n\n        const translations: KeyTranslation[] = [];\n\n        if (entry.localizations) {\n            for (const [language, localization] of Object.entries(entry.localizations)) {\n                if (localization.stringUnit) {\n                    translations.push({\n                        language,\n                        value: localization.stringUnit.value,\n                        state: localization.stringUnit.state,\n                    });\n                } else if (localization.variations?.plural) {\n                    // For plural strings, show the \"other\" form as the primary value\n                    const plural = localization.variations.plural;\n                    const primaryForm = plural.other || plural.one || plural.zero;\n                    if (primaryForm) {\n                        translations.push({\n                            language,\n                            value: `[plural] ${primaryForm.stringUnit.value}`,\n                            state: primaryForm.stringUnit.state,\n                        });\n                    }\n                }\n            }\n        }\n\n        return {\n            key,\n            sourceLanguage: this.data.sourceLanguage,\n            translations: translations.sort((a, b) => a.language.localeCompare(b.language)),\n        };\n    }\n\n    /**\n     * Search for keys containing a specific substring\n     */\n    searchKeys(query: string): string[] {\n        const lowerQuery = query.toLowerCase();\n        return this.getAllKeys().filter((key) => key.toLowerCase().includes(lowerQuery));\n    }\n\n    /**\n     * Update translations for one or more keys\n     */\n    updateTranslations(translations: TranslationInput[]): { updated: string[]; created: string[] } {\n        const updated: string[] = [];\n        const created: string[] = [];\n\n        for (const translation of translations) {\n            const { key, translations: langTranslations, comment } = translation;\n\n            const isNew = !this.data.strings[key];\n\n            if (isNew) {\n                this.data.strings[key] = {};\n                created.push(key);\n            } else {\n                updated.push(key);\n            }\n\n            const entry = this.data.strings[key];\n\n            if (comment) {\n                entry.comment = comment;\n            }\n\n            if (!entry.localizations) {\n                entry.localizations = {};\n            }\n\n            for (const langTrans of langTranslations) {\n                entry.localizations[langTrans.language] = {\n                    stringUnit: {\n                        state: langTrans.state || 'translated',\n                        value: langTrans.value,\n                    },\n                };\n            }\n        }\n\n        return { updated, created };\n    }\n\n    /**\n     * Save changes to the file\n     */\n    save(): void {\n        // Sort keys alphabetically for consistent output\n        const sortedStrings: { [key: string]: StringEntry } = {};\n        const keys = Object.keys(this.data.strings).sort();\n\n        for (const key of keys) {\n            sortedStrings[key] = this.data.strings[key];\n        }\n\n        const output: XCStrings = {\n            sourceLanguage: this.data.sourceLanguage,\n            strings: sortedStrings,\n        };\n\n        if (this.data.version) {\n            output.version = this.data.version;\n        }\n\n        // Write with 2-space indentation to match Xcode format\n        fs.writeFileSync(this.filePath, JSON.stringify(output, null, 2) + '\\n', 'utf-8');\n    }\n\n    /**\n     * Get statistics about the catalog\n     */\n    getStatistics(): {\n        totalKeys: number;\n        languages: string[];\n        translationCoverage: Record<\n            string,\n            { translated: number; total: number; percentage: number }\n        >;\n    } {\n        const languages = this.getSupportedLanguages();\n        const allKeys = this.getAllKeys();\n        const totalKeys = allKeys.length;\n\n        const translationCoverage: Record<\n            string,\n            { translated: number; total: number; percentage: number }\n        > = {};\n\n        for (const lang of languages) {\n            let translated = 0;\n\n            for (const key of allKeys) {\n                const entry = this.data.strings[key];\n                const localization = entry.localizations?.[lang];\n\n                if (localization?.stringUnit?.state === 'translated') {\n                    translated++;\n                } else if (localization?.variations?.plural) {\n                    // Count plural as translated if it has an \"other\" form\n                    if (localization.variations.plural.other?.stringUnit?.state === 'translated') {\n                        translated++;\n                    }\n                }\n            }\n\n            translationCoverage[lang] = {\n                translated,\n                total: totalKeys,\n                percentage: totalKeys > 0 ? Math.round((translated / totalKeys) * 100) : 0,\n            };\n        }\n\n        return {\n            totalKeys,\n            languages,\n            translationCoverage,\n        };\n    }\n}\n"]}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerGetCatalogStatistics = registerGetCatalogStatistics;
|
|
4
|
+
const zod_1 = require("zod");
|
|
5
|
+
const string_catalog_1 = require("../string-catalog");
|
|
6
|
+
function registerGetCatalogStatistics(server) {
|
|
7
|
+
server.registerTool('get_catalog_statistics', {
|
|
8
|
+
description: 'Get statistics about a String Catalog including total keys, supported languages, and translation coverage percentage for each language.',
|
|
9
|
+
inputSchema: {
|
|
10
|
+
filePath: zod_1.z.string().describe('Absolute path to the .xcstrings file'),
|
|
11
|
+
},
|
|
12
|
+
}, async ({ filePath }) => {
|
|
13
|
+
const catalog = new string_catalog_1.StringCatalog(filePath);
|
|
14
|
+
const stats = catalog.getStatistics();
|
|
15
|
+
return {
|
|
16
|
+
content: [
|
|
17
|
+
{
|
|
18
|
+
type: 'text',
|
|
19
|
+
text: JSON.stringify(stats, null, 2),
|
|
20
|
+
},
|
|
21
|
+
],
|
|
22
|
+
};
|
|
23
|
+
});
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2V0LWNhdGFsb2ctc3RhdGlzdGljcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90b29scy9nZXQtY2F0YWxvZy1zdGF0aXN0aWNzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBSUEsb0VBd0JDO0FBM0JELDZCQUF3QjtBQUN4QixzREFBa0Q7QUFFbEQsU0FBZ0IsNEJBQTRCLENBQUMsTUFBaUI7SUFDMUQsTUFBTSxDQUFDLFlBQVksQ0FDZix3QkFBd0IsRUFDeEI7UUFDSSxXQUFXLEVBQ1AseUlBQXlJO1FBQzdJLFdBQVcsRUFBRTtZQUNULFFBQVEsRUFBRSxPQUFDLENBQUMsTUFBTSxFQUFFLENBQUMsUUFBUSxDQUFDLHNDQUFzQyxDQUFDO1NBQ3hFO0tBQ0osRUFDRCxLQUFLLEVBQUUsRUFBRSxRQUFRLEVBQUUsRUFBRSxFQUFFO1FBQ25CLE1BQU0sT0FBTyxHQUFHLElBQUksOEJBQWEsQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUM1QyxNQUFNLEtBQUssR0FBRyxPQUFPLENBQUMsYUFBYSxFQUFFLENBQUM7UUFFdEMsT0FBTztZQUNILE9BQU8sRUFBRTtnQkFDTDtvQkFDSSxJQUFJLEVBQUUsTUFBZTtvQkFDckIsSUFBSSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsS0FBSyxFQUFFLElBQUksRUFBRSxDQUFDLENBQUM7aUJBQ3ZDO2FBQ0o7U0FDSixDQUFDO0lBQ04sQ0FBQyxDQUNKLENBQUM7QUFDTixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTWNwU2VydmVyIH0gZnJvbSAnQG1vZGVsY29udGV4dHByb3RvY29sL3Nkay9zZXJ2ZXIvbWNwLmpzJztcbmltcG9ydCB7IHogfSBmcm9tICd6b2QnO1xuaW1wb3J0IHsgU3RyaW5nQ2F0YWxvZyB9IGZyb20gJy4uL3N0cmluZy1jYXRhbG9nJztcblxuZXhwb3J0IGZ1bmN0aW9uIHJlZ2lzdGVyR2V0Q2F0YWxvZ1N0YXRpc3RpY3Moc2VydmVyOiBNY3BTZXJ2ZXIpIHtcbiAgICBzZXJ2ZXIucmVnaXN0ZXJUb29sKFxuICAgICAgICAnZ2V0X2NhdGFsb2dfc3RhdGlzdGljcycsXG4gICAgICAgIHtcbiAgICAgICAgICAgIGRlc2NyaXB0aW9uOlxuICAgICAgICAgICAgICAgICdHZXQgc3RhdGlzdGljcyBhYm91dCBhIFN0cmluZyBDYXRhbG9nIGluY2x1ZGluZyB0b3RhbCBrZXlzLCBzdXBwb3J0ZWQgbGFuZ3VhZ2VzLCBhbmQgdHJhbnNsYXRpb24gY292ZXJhZ2UgcGVyY2VudGFnZSBmb3IgZWFjaCBsYW5ndWFnZS4nLFxuICAgICAgICAgICAgaW5wdXRTY2hlbWE6IHtcbiAgICAgICAgICAgICAgICBmaWxlUGF0aDogei5zdHJpbmcoKS5kZXNjcmliZSgnQWJzb2x1dGUgcGF0aCB0byB0aGUgLnhjc3RyaW5ncyBmaWxlJyksXG4gICAgICAgICAgICB9LFxuICAgICAgICB9LFxuICAgICAgICBhc3luYyAoeyBmaWxlUGF0aCB9KSA9PiB7XG4gICAgICAgICAgICBjb25zdCBjYXRhbG9nID0gbmV3IFN0cmluZ0NhdGFsb2coZmlsZVBhdGgpO1xuICAgICAgICAgICAgY29uc3Qgc3RhdHMgPSBjYXRhbG9nLmdldFN0YXRpc3RpY3MoKTtcblxuICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICBjb250ZW50OiBbXG4gICAgICAgICAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIHR5cGU6ICd0ZXh0JyBhcyBjb25zdCxcbiAgICAgICAgICAgICAgICAgICAgICAgIHRleHQ6IEpTT04uc3RyaW5naWZ5KHN0YXRzLCBudWxsLCAyKSxcbiAgICAgICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgICBdLFxuICAgICAgICAgICAgfTtcbiAgICAgICAgfVxuICAgICk7XG59XG4iXX0=
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerGetTranslationsForKey = registerGetTranslationsForKey;
|
|
4
|
+
const zod_1 = require("zod");
|
|
5
|
+
const string_catalog_1 = require("../string-catalog");
|
|
6
|
+
function registerGetTranslationsForKey(server) {
|
|
7
|
+
server.registerTool('get_translations_for_key', {
|
|
8
|
+
description: 'Get all translations for a specific key in a String Catalog. Shows the translated text in each supported language along with the translation state.',
|
|
9
|
+
inputSchema: {
|
|
10
|
+
filePath: zod_1.z.string().describe('Absolute path to the .xcstrings file'),
|
|
11
|
+
key: zod_1.z.string().describe('The localization key to look up'),
|
|
12
|
+
},
|
|
13
|
+
}, async ({ filePath, key }) => {
|
|
14
|
+
const catalog = new string_catalog_1.StringCatalog(filePath);
|
|
15
|
+
const result = catalog.getTranslationsForKey(key);
|
|
16
|
+
if (!result) {
|
|
17
|
+
return {
|
|
18
|
+
content: [
|
|
19
|
+
{
|
|
20
|
+
type: 'text',
|
|
21
|
+
text: JSON.stringify({ error: `Key "${key}" not found in catalog` }, null, 2),
|
|
22
|
+
},
|
|
23
|
+
],
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
return {
|
|
27
|
+
content: [
|
|
28
|
+
{
|
|
29
|
+
type: 'text',
|
|
30
|
+
text: JSON.stringify(result, null, 2),
|
|
31
|
+
},
|
|
32
|
+
],
|
|
33
|
+
};
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2V0LXRyYW5zbGF0aW9ucy1mb3Ita2V5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3Rvb2xzL2dldC10cmFuc2xhdGlvbnMtZm9yLWtleS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUlBLHNFQXdDQztBQTNDRCw2QkFBd0I7QUFDeEIsc0RBQWtEO0FBRWxELFNBQWdCLDZCQUE2QixDQUFDLE1BQWlCO0lBQzNELE1BQU0sQ0FBQyxZQUFZLENBQ2YsMEJBQTBCLEVBQzFCO1FBQ0ksV0FBVyxFQUNQLHFKQUFxSjtRQUN6SixXQUFXLEVBQUU7WUFDVCxRQUFRLEVBQUUsT0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDLFFBQVEsQ0FBQyxzQ0FBc0MsQ0FBQztZQUNyRSxHQUFHLEVBQUUsT0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDLFFBQVEsQ0FBQyxpQ0FBaUMsQ0FBQztTQUM5RDtLQUNKLEVBQ0QsS0FBSyxFQUFFLEVBQUUsUUFBUSxFQUFFLEdBQUcsRUFBRSxFQUFFLEVBQUU7UUFDeEIsTUFBTSxPQUFPLEdBQUcsSUFBSSw4QkFBYSxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQzVDLE1BQU0sTUFBTSxHQUFHLE9BQU8sQ0FBQyxxQkFBcUIsQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUVsRCxJQUFJLENBQUMsTUFBTSxFQUFFLENBQUM7WUFDVixPQUFPO2dCQUNILE9BQU8sRUFBRTtvQkFDTDt3QkFDSSxJQUFJLEVBQUUsTUFBZTt3QkFDckIsSUFBSSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQ2hCLEVBQUUsS0FBSyxFQUFFLFFBQVEsR0FBRyx3QkFBd0IsRUFBRSxFQUM5QyxJQUFJLEVBQ0osQ0FBQyxDQUNKO3FCQUNKO2lCQUNKO2FBQ0osQ0FBQztRQUNOLENBQUM7UUFFRCxPQUFPO1lBQ0gsT0FBTyxFQUFFO2dCQUNMO29CQUNJLElBQUksRUFBRSxNQUFlO29CQUNyQixJQUFJLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxNQUFNLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQztpQkFDeEM7YUFDSjtTQUNKLENBQUM7SUFDTixDQUFDLENBQ0osQ0FBQztBQUNOLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBNY3BTZXJ2ZXIgfSBmcm9tICdAbW9kZWxjb250ZXh0cHJvdG9jb2wvc2RrL3NlcnZlci9tY3AuanMnO1xuaW1wb3J0IHsgeiB9IGZyb20gJ3pvZCc7XG5pbXBvcnQgeyBTdHJpbmdDYXRhbG9nIH0gZnJvbSAnLi4vc3RyaW5nLWNhdGFsb2cnO1xuXG5leHBvcnQgZnVuY3Rpb24gcmVnaXN0ZXJHZXRUcmFuc2xhdGlvbnNGb3JLZXkoc2VydmVyOiBNY3BTZXJ2ZXIpIHtcbiAgICBzZXJ2ZXIucmVnaXN0ZXJUb29sKFxuICAgICAgICAnZ2V0X3RyYW5zbGF0aW9uc19mb3Jfa2V5JyxcbiAgICAgICAge1xuICAgICAgICAgICAgZGVzY3JpcHRpb246XG4gICAgICAgICAgICAgICAgJ0dldCBhbGwgdHJhbnNsYXRpb25zIGZvciBhIHNwZWNpZmljIGtleSBpbiBhIFN0cmluZyBDYXRhbG9nLiBTaG93cyB0aGUgdHJhbnNsYXRlZCB0ZXh0IGluIGVhY2ggc3VwcG9ydGVkIGxhbmd1YWdlIGFsb25nIHdpdGggdGhlIHRyYW5zbGF0aW9uIHN0YXRlLicsXG4gICAgICAgICAgICBpbnB1dFNjaGVtYToge1xuICAgICAgICAgICAgICAgIGZpbGVQYXRoOiB6LnN0cmluZygpLmRlc2NyaWJlKCdBYnNvbHV0ZSBwYXRoIHRvIHRoZSAueGNzdHJpbmdzIGZpbGUnKSxcbiAgICAgICAgICAgICAgICBrZXk6IHouc3RyaW5nKCkuZGVzY3JpYmUoJ1RoZSBsb2NhbGl6YXRpb24ga2V5IHRvIGxvb2sgdXAnKSxcbiAgICAgICAgICAgIH0sXG4gICAgICAgIH0sXG4gICAgICAgIGFzeW5jICh7IGZpbGVQYXRoLCBrZXkgfSkgPT4ge1xuICAgICAgICAgICAgY29uc3QgY2F0YWxvZyA9IG5ldyBTdHJpbmdDYXRhbG9nKGZpbGVQYXRoKTtcbiAgICAgICAgICAgIGNvbnN0IHJlc3VsdCA9IGNhdGFsb2cuZ2V0VHJhbnNsYXRpb25zRm9yS2V5KGtleSk7XG5cbiAgICAgICAgICAgIGlmICghcmVzdWx0KSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICAgICAgY29udGVudDogW1xuICAgICAgICAgICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHR5cGU6ICd0ZXh0JyBhcyBjb25zdCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB0ZXh0OiBKU09OLnN0cmluZ2lmeShcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgeyBlcnJvcjogYEtleSBcIiR7a2V5fVwiIG5vdCBmb3VuZCBpbiBjYXRhbG9nYCB9LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBudWxsLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAyXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgICAgIF0sXG4gICAgICAgICAgICAgICAgfTtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICBjb250ZW50OiBbXG4gICAgICAgICAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIHR5cGU6ICd0ZXh0JyBhcyBjb25zdCxcbiAgICAgICAgICAgICAgICAgICAgICAgIHRleHQ6IEpTT04uc3RyaW5naWZ5KHJlc3VsdCwgbnVsbCwgMiksXG4gICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgXSxcbiAgICAgICAgICAgIH07XG4gICAgICAgIH1cbiAgICApO1xufVxuIl19
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerAllTools = registerAllTools;
|
|
4
|
+
const list_supported_languages_1 = require("./list-supported-languages");
|
|
5
|
+
const get_translations_for_key_1 = require("./get-translations-for-key");
|
|
6
|
+
const search_keys_1 = require("./search-keys");
|
|
7
|
+
const update_translations_1 = require("./update-translations");
|
|
8
|
+
const get_catalog_statistics_1 = require("./get-catalog-statistics");
|
|
9
|
+
const list_all_keys_1 = require("./list-all-keys");
|
|
10
|
+
function registerAllTools(server) {
|
|
11
|
+
(0, list_supported_languages_1.registerListSupportedLanguages)(server);
|
|
12
|
+
(0, get_translations_for_key_1.registerGetTranslationsForKey)(server);
|
|
13
|
+
(0, search_keys_1.registerSearchKeys)(server);
|
|
14
|
+
(0, update_translations_1.registerUpdateTranslations)(server);
|
|
15
|
+
(0, get_catalog_statistics_1.registerGetCatalogStatistics)(server);
|
|
16
|
+
(0, list_all_keys_1.registerListAllKeys)(server);
|
|
17
|
+
}
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdG9vbHMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFRQSw0Q0FPQztBQWRELHlFQUE0RTtBQUM1RSx5RUFBMkU7QUFDM0UsK0NBQW1EO0FBQ25ELCtEQUFtRTtBQUNuRSxxRUFBd0U7QUFDeEUsbURBQXNEO0FBRXRELFNBQWdCLGdCQUFnQixDQUFDLE1BQWlCO0lBQzlDLElBQUEseURBQThCLEVBQUMsTUFBTSxDQUFDLENBQUM7SUFDdkMsSUFBQSx3REFBNkIsRUFBQyxNQUFNLENBQUMsQ0FBQztJQUN0QyxJQUFBLGdDQUFrQixFQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQzNCLElBQUEsZ0RBQTBCLEVBQUMsTUFBTSxDQUFDLENBQUM7SUFDbkMsSUFBQSxxREFBNEIsRUFBQyxNQUFNLENBQUMsQ0FBQztJQUNyQyxJQUFBLG1DQUFtQixFQUFDLE1BQU0sQ0FBQyxDQUFDO0FBQ2hDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBNY3BTZXJ2ZXIgfSBmcm9tICdAbW9kZWxjb250ZXh0cHJvdG9jb2wvc2RrL3NlcnZlci9tY3AuanMnO1xuaW1wb3J0IHsgcmVnaXN0ZXJMaXN0U3VwcG9ydGVkTGFuZ3VhZ2VzIH0gZnJvbSAnLi9saXN0LXN1cHBvcnRlZC1sYW5ndWFnZXMnO1xuaW1wb3J0IHsgcmVnaXN0ZXJHZXRUcmFuc2xhdGlvbnNGb3JLZXkgfSBmcm9tICcuL2dldC10cmFuc2xhdGlvbnMtZm9yLWtleSc7XG5pbXBvcnQgeyByZWdpc3RlclNlYXJjaEtleXMgfSBmcm9tICcuL3NlYXJjaC1rZXlzJztcbmltcG9ydCB7IHJlZ2lzdGVyVXBkYXRlVHJhbnNsYXRpb25zIH0gZnJvbSAnLi91cGRhdGUtdHJhbnNsYXRpb25zJztcbmltcG9ydCB7IHJlZ2lzdGVyR2V0Q2F0YWxvZ1N0YXRpc3RpY3MgfSBmcm9tICcuL2dldC1jYXRhbG9nLXN0YXRpc3RpY3MnO1xuaW1wb3J0IHsgcmVnaXN0ZXJMaXN0QWxsS2V5cyB9IGZyb20gJy4vbGlzdC1hbGwta2V5cyc7XG5cbmV4cG9ydCBmdW5jdGlvbiByZWdpc3RlckFsbFRvb2xzKHNlcnZlcjogTWNwU2VydmVyKSB7XG4gICAgcmVnaXN0ZXJMaXN0U3VwcG9ydGVkTGFuZ3VhZ2VzKHNlcnZlcik7XG4gICAgcmVnaXN0ZXJHZXRUcmFuc2xhdGlvbnNGb3JLZXkoc2VydmVyKTtcbiAgICByZWdpc3RlclNlYXJjaEtleXMoc2VydmVyKTtcbiAgICByZWdpc3RlclVwZGF0ZVRyYW5zbGF0aW9ucyhzZXJ2ZXIpO1xuICAgIHJlZ2lzdGVyR2V0Q2F0YWxvZ1N0YXRpc3RpY3Moc2VydmVyKTtcbiAgICByZWdpc3Rlckxpc3RBbGxLZXlzKHNlcnZlcik7XG59XG4iXX0=
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.registerListAllKeys = registerListAllKeys;
|
|
4
|
+
const zod_1 = require("zod");
|
|
5
|
+
const string_catalog_1 = require("../string-catalog");
|
|
6
|
+
function registerListAllKeys(server) {
|
|
7
|
+
server.registerTool('list_all_keys', {
|
|
8
|
+
description: 'List all localization keys in a String Catalog. Returns keys sorted alphabetically.',
|
|
9
|
+
inputSchema: {
|
|
10
|
+
filePath: zod_1.z.string().describe('Absolute path to the .xcstrings file'),
|
|
11
|
+
limit: zod_1.z
|
|
12
|
+
.number()
|
|
13
|
+
.optional()
|
|
14
|
+
.default(100)
|
|
15
|
+
.describe('Maximum number of keys to return (default: 100)'),
|
|
16
|
+
offset: zod_1.z
|
|
17
|
+
.number()
|
|
18
|
+
.optional()
|
|
19
|
+
.default(0)
|
|
20
|
+
.describe('Number of keys to skip (for pagination, default: 0)'),
|
|
21
|
+
},
|
|
22
|
+
}, async ({ filePath, limit: limitArg, offset: offsetArg }) => {
|
|
23
|
+
const limit = limitArg ?? 100;
|
|
24
|
+
const offset = offsetArg ?? 0;
|
|
25
|
+
const catalog = new string_catalog_1.StringCatalog(filePath);
|
|
26
|
+
const allKeys = catalog.getAllKeys();
|
|
27
|
+
const paginatedKeys = allKeys.slice(offset, offset + limit);
|
|
28
|
+
return {
|
|
29
|
+
content: [
|
|
30
|
+
{
|
|
31
|
+
type: 'text',
|
|
32
|
+
text: JSON.stringify({
|
|
33
|
+
keys: paginatedKeys,
|
|
34
|
+
total: allKeys.length,
|
|
35
|
+
offset,
|
|
36
|
+
limit,
|
|
37
|
+
hasMore: offset + limit < allKeys.length,
|
|
38
|
+
}, null, 2),
|
|
39
|
+
},
|
|
40
|
+
],
|
|
41
|
+
};
|
|
42
|
+
});
|
|
43
|
+
}
|
|
44
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibGlzdC1hbGwta2V5cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy90b29scy9saXN0LWFsbC1rZXlzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBSUEsa0RBK0NDO0FBbERELDZCQUF3QjtBQUN4QixzREFBa0Q7QUFFbEQsU0FBZ0IsbUJBQW1CLENBQUMsTUFBaUI7SUFDakQsTUFBTSxDQUFDLFlBQVksQ0FDZixlQUFlLEVBQ2Y7UUFDSSxXQUFXLEVBQ1AscUZBQXFGO1FBQ3pGLFdBQVcsRUFBRTtZQUNULFFBQVEsRUFBRSxPQUFDLENBQUMsTUFBTSxFQUFFLENBQUMsUUFBUSxDQUFDLHNDQUFzQyxDQUFDO1lBQ3JFLEtBQUssRUFBRSxPQUFDO2lCQUNILE1BQU0sRUFBRTtpQkFDUixRQUFRLEVBQUU7aUJBQ1YsT0FBTyxDQUFDLEdBQUcsQ0FBQztpQkFDWixRQUFRLENBQUMsaURBQWlELENBQUM7WUFDaEUsTUFBTSxFQUFFLE9BQUM7aUJBQ0osTUFBTSxFQUFFO2lCQUNSLFFBQVEsRUFBRTtpQkFDVixPQUFPLENBQUMsQ0FBQyxDQUFDO2lCQUNWLFFBQVEsQ0FBQyxxREFBcUQsQ0FBQztTQUN2RTtLQUNKLEVBQ0QsS0FBSyxFQUFFLEVBQUUsUUFBUSxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUUsTUFBTSxFQUFFLFNBQVMsRUFBRSxFQUFFLEVBQUU7UUFDdkQsTUFBTSxLQUFLLEdBQUcsUUFBUSxJQUFJLEdBQUcsQ0FBQztRQUM5QixNQUFNLE1BQU0sR0FBRyxTQUFTLElBQUksQ0FBQyxDQUFDO1FBQzlCLE1BQU0sT0FBTyxHQUFHLElBQUksOEJBQWEsQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUM1QyxNQUFNLE9BQU8sR0FBRyxPQUFPLENBQUMsVUFBVSxFQUFFLENBQUM7UUFDckMsTUFBTSxhQUFhLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQyxNQUFNLEVBQUUsTUFBTSxHQUFHLEtBQUssQ0FBQyxDQUFDO1FBRTVELE9BQU87WUFDSCxPQUFPLEVBQUU7Z0JBQ0w7b0JBQ0ksSUFBSSxFQUFFLE1BQWU7b0JBQ3JCLElBQUksRUFBRSxJQUFJLENBQUMsU0FBUyxDQUNoQjt3QkFDSSxJQUFJLEVBQUUsYUFBYTt3QkFDbkIsS0FBSyxFQUFFLE9BQU8sQ0FBQyxNQUFNO3dCQUNyQixNQUFNO3dCQUNOLEtBQUs7d0JBQ0wsT0FBTyxFQUFFLE1BQU0sR0FBRyxLQUFLLEdBQUcsT0FBTyxDQUFDLE1BQU07cUJBQzNDLEVBQ0QsSUFBSSxFQUNKLENBQUMsQ0FDSjtpQkFDSjthQUNKO1NBQ0osQ0FBQztJQUNOLENBQUMsQ0FDSixDQUFDO0FBQ04sQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE1jcFNlcnZlciB9IGZyb20gJ0Btb2RlbGNvbnRleHRwcm90b2NvbC9zZGsvc2VydmVyL21jcC5qcyc7XG5pbXBvcnQgeyB6IH0gZnJvbSAnem9kJztcbmltcG9ydCB7IFN0cmluZ0NhdGFsb2cgfSBmcm9tICcuLi9zdHJpbmctY2F0YWxvZyc7XG5cbmV4cG9ydCBmdW5jdGlvbiByZWdpc3Rlckxpc3RBbGxLZXlzKHNlcnZlcjogTWNwU2VydmVyKSB7XG4gICAgc2VydmVyLnJlZ2lzdGVyVG9vbChcbiAgICAgICAgJ2xpc3RfYWxsX2tleXMnLFxuICAgICAgICB7XG4gICAgICAgICAgICBkZXNjcmlwdGlvbjpcbiAgICAgICAgICAgICAgICAnTGlzdCBhbGwgbG9jYWxpemF0aW9uIGtleXMgaW4gYSBTdHJpbmcgQ2F0YWxvZy4gUmV0dXJucyBrZXlzIHNvcnRlZCBhbHBoYWJldGljYWxseS4nLFxuICAgICAgICAgICAgaW5wdXRTY2hlbWE6IHtcbiAgICAgICAgICAgICAgICBmaWxlUGF0aDogei5zdHJpbmcoKS5kZXNjcmliZSgnQWJzb2x1dGUgcGF0aCB0byB0aGUgLnhjc3RyaW5ncyBmaWxlJyksXG4gICAgICAgICAgICAgICAgbGltaXQ6IHpcbiAgICAgICAgICAgICAgICAgICAgLm51bWJlcigpXG4gICAgICAgICAgICAgICAgICAgIC5vcHRpb25hbCgpXG4gICAgICAgICAgICAgICAgICAgIC5kZWZhdWx0KDEwMClcbiAgICAgICAgICAgICAgICAgICAgLmRlc2NyaWJlKCdNYXhpbXVtIG51bWJlciBvZiBrZXlzIHRvIHJldHVybiAoZGVmYXVsdDogMTAwKScpLFxuICAgICAgICAgICAgICAgIG9mZnNldDogelxuICAgICAgICAgICAgICAgICAgICAubnVtYmVyKClcbiAgICAgICAgICAgICAgICAgICAgLm9wdGlvbmFsKClcbiAgICAgICAgICAgICAgICAgICAgLmRlZmF1bHQoMClcbiAgICAgICAgICAgICAgICAgICAgLmRlc2NyaWJlKCdOdW1iZXIgb2Yga2V5cyB0byBza2lwIChmb3IgcGFnaW5hdGlvbiwgZGVmYXVsdDogMCknKSxcbiAgICAgICAgICAgIH0sXG4gICAgICAgIH0sXG4gICAgICAgIGFzeW5jICh7IGZpbGVQYXRoLCBsaW1pdDogbGltaXRBcmcsIG9mZnNldDogb2Zmc2V0QXJnIH0pID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGxpbWl0ID0gbGltaXRBcmcgPz8gMTAwO1xuICAgICAgICAgICAgY29uc3Qgb2Zmc2V0ID0gb2Zmc2V0QXJnID8/IDA7XG4gICAgICAgICAgICBjb25zdCBjYXRhbG9nID0gbmV3IFN0cmluZ0NhdGFsb2coZmlsZVBhdGgpO1xuICAgICAgICAgICAgY29uc3QgYWxsS2V5cyA9IGNhdGFsb2cuZ2V0QWxsS2V5cygpO1xuICAgICAgICAgICAgY29uc3QgcGFnaW5hdGVkS2V5cyA9IGFsbEtleXMuc2xpY2Uob2Zmc2V0LCBvZmZzZXQgKyBsaW1pdCk7XG5cbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgY29udGVudDogW1xuICAgICAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgICAgICB0eXBlOiAndGV4dCcgYXMgY29uc3QsXG4gICAgICAgICAgICAgICAgICAgICAgICB0ZXh0OiBKU09OLnN0cmluZ2lmeShcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGtleXM6IHBhZ2luYXRlZEtleXMsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRvdGFsOiBhbGxLZXlzLmxlbmd0aCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgb2Zmc2V0LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBsaW1pdCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaGFzTW9yZTogb2Zmc2V0ICsgbGltaXQgPCBhbGxLZXlzLmxlbmd0aCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIG51bGwsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgMlxuICAgICAgICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgICBdLFxuICAgICAgICAgICAgfTtcbiAgICAgICAgfVxuICAgICk7XG59XG4iXX0=
|