@mxml3gend/gloss 0.1.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.
Binary file
Binary file
Binary file
Binary file
@@ -0,0 +1,3 @@
1
+ <svg xmlns="http://www.w3.org/2000/svg" version="1.1" xmlns:xlink="http://www.w3.org/1999/xlink" width="273" height="273"><svg xmlns="http://www.w3.org/2000/svg" version="1.1" xmlns:xlink="http://www.w3.org/1999/xlink" width="273" height="273" viewBox="0 0 273 273"><image width="273" height="273" xlink:href=""></image></svg><style>@media (prefers-color-scheme: light) { :root { filter: none; } }
2
+ @media (prefers-color-scheme: dark) { :root { filter: none; } }
3
+ </style></svg>
@@ -0,0 +1,20 @@
1
+ <!doctype html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <link rel="icon" type="image/svg+xml" href="/vite.svg" />
6
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
7
+ <title>ui</title>
8
+ <link rel="icon" type="image/png" href="/favicon-96x96.png" sizes="96x96" />
9
+ <link rel="icon" type="image/svg+xml" href="/favicon.svg" />
10
+ <link rel="shortcut icon" href="/favicon.ico" />
11
+ <link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png" />
12
+ <meta name="apple-mobile-web-app-title" content="Gloss" />
13
+ <link rel="manifest" href="/site.webmanifest" />
14
+ <script type="module" crossorigin src="/assets/index-Dhb2pVPI.js"></script>
15
+ <link rel="stylesheet" crossorigin href="/assets/index-CREq9Gop.css">
16
+ </head>
17
+ <body>
18
+ <div id="root"></div>
19
+ </body>
20
+ </html>
@@ -0,0 +1,21 @@
1
+ {
2
+ "name": "Gloss",
3
+ "short_name": "Gloss",
4
+ "icons": [
5
+ {
6
+ "src": "/web-app-manifest-192x192.png",
7
+ "sizes": "192x192",
8
+ "type": "image/png",
9
+ "purpose": "maskable"
10
+ },
11
+ {
12
+ "src": "/web-app-manifest-512x512.png",
13
+ "sizes": "512x512",
14
+ "type": "image/png",
15
+ "purpose": "maskable"
16
+ }
17
+ ],
18
+ "theme_color": "#ffffff",
19
+ "background_color": "#ffffff",
20
+ "display": "standalone"
21
+ }
@@ -0,0 +1 @@
1
+ <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="31.88" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 257"><defs><linearGradient id="IconifyId1813088fe1fbc01fb466" x1="-.828%" x2="57.636%" y1="7.652%" y2="78.411%"><stop offset="0%" stop-color="#41D1FF"></stop><stop offset="100%" stop-color="#BD34FE"></stop></linearGradient><linearGradient id="IconifyId1813088fe1fbc01fb467" x1="43.376%" x2="50.316%" y1="2.242%" y2="89.03%"><stop offset="0%" stop-color="#FFEA83"></stop><stop offset="8.333%" stop-color="#FFDD35"></stop><stop offset="100%" stop-color="#FFA800"></stop></linearGradient></defs><path fill="url(#IconifyId1813088fe1fbc01fb466)" d="M255.153 37.938L134.897 252.976c-2.483 4.44-8.862 4.466-11.382.048L.875 37.958c-2.746-4.814 1.371-10.646 6.827-9.67l120.385 21.517a6.537 6.537 0 0 0 2.322-.004l117.867-21.483c5.438-.991 9.574 4.796 6.877 9.62Z"></path><path fill="url(#IconifyId1813088fe1fbc01fb467)" d="M185.432.063L96.44 17.501a3.268 3.268 0 0 0-2.634 3.014l-5.474 92.456a3.268 3.268 0 0 0 3.997 3.378l24.777-5.718c2.318-.535 4.413 1.507 3.936 3.838l-7.361 36.047c-.495 2.426 1.782 4.5 4.151 3.78l15.304-4.649c2.372-.72 4.652 1.36 4.15 3.788l-11.698 56.621c-.732 3.542 3.979 5.473 5.943 2.437l1.313-2.028l72.516-144.72c1.215-2.423-.88-5.186-3.54-4.672l-25.505 4.922c-2.396.462-4.435-1.77-3.759-4.114l16.646-57.705c.677-2.35-1.37-4.583-3.769-4.113Z"></path></svg>
package/dist/usage.js ADDED
@@ -0,0 +1,241 @@
1
+ import fs from "node:fs/promises";
2
+ import path from "node:path";
3
+ import { createScanMatcher } from "./scanFilters.js";
4
+ import { isLikelyTranslationKey } from "./translationKeys.js";
5
+ const SUPPORTED_EXTENSIONS = [
6
+ ".ts",
7
+ ".tsx",
8
+ ".js",
9
+ ".jsx",
10
+ ".mjs",
11
+ ".cjs",
12
+ ".vue",
13
+ ".svelte",
14
+ ];
15
+ const SKIP_DIRECTORIES = new Set([
16
+ "node_modules",
17
+ ".git",
18
+ "dist",
19
+ "build",
20
+ ".next",
21
+ ".nuxt",
22
+ "coverage",
23
+ ]);
24
+ const projectRoot = () => process.env.INIT_CWD || process.cwd();
25
+ const translationsDir = (cfg) => {
26
+ if (path.isAbsolute(cfg.path)) {
27
+ return cfg.path;
28
+ }
29
+ return path.join(projectRoot(), cfg.path);
30
+ };
31
+ const normalizePath = (filePath) => filePath.split(path.sep).join("/");
32
+ const hasSkippedPathSegment = (relativePath) => normalizePath(relativePath)
33
+ .split("/")
34
+ .some((segment) => SKIP_DIRECTORIES.has(segment));
35
+ const isSupportedFile = (name) => SUPPORTED_EXTENSIONS.some((extension) => name.endsWith(extension));
36
+ const extractTranslationKeys = (content) => {
37
+ const keys = new Set();
38
+ const regexes = [
39
+ /\b(?:t|i18n\.t|translate)\(\s*["'`]([^"'`]+)["'`]\s*[\),]/g,
40
+ /\bi18nKey\s*=\s*["'`]([^"'`]+)["'`]/g,
41
+ ];
42
+ for (const regex of regexes) {
43
+ let match = regex.exec(content);
44
+ while (match) {
45
+ const key = match[1]?.trim();
46
+ if (key && isLikelyTranslationKey(key)) {
47
+ keys.add(key);
48
+ }
49
+ match = regex.exec(content);
50
+ }
51
+ }
52
+ return keys;
53
+ };
54
+ const extractRelativeImports = (content) => {
55
+ const imports = new Set();
56
+ const importRegexes = [
57
+ /import\s+[\s\S]*?\s+from\s+["']([^"']+)["']/g,
58
+ /import\(\s*["']([^"']+)["']\s*\)/g,
59
+ ];
60
+ for (const regex of importRegexes) {
61
+ let match = regex.exec(content);
62
+ while (match) {
63
+ const specifier = match[1];
64
+ if (specifier.startsWith(".")) {
65
+ imports.add(specifier);
66
+ }
67
+ match = regex.exec(content);
68
+ }
69
+ }
70
+ return Array.from(imports);
71
+ };
72
+ const resolveImport = async (fromFile, specifier) => {
73
+ const basePath = path.resolve(path.dirname(fromFile), specifier);
74
+ const candidates = [];
75
+ if (path.extname(basePath)) {
76
+ candidates.push(basePath);
77
+ }
78
+ else {
79
+ for (const extension of SUPPORTED_EXTENSIONS) {
80
+ candidates.push(`${basePath}${extension}`);
81
+ }
82
+ for (const extension of SUPPORTED_EXTENSIONS) {
83
+ candidates.push(path.join(basePath, `index${extension}`));
84
+ }
85
+ }
86
+ for (const candidate of candidates) {
87
+ try {
88
+ const stat = await awaitStat(candidate);
89
+ if (stat?.isFile()) {
90
+ return candidate;
91
+ }
92
+ }
93
+ catch {
94
+ continue;
95
+ }
96
+ }
97
+ return null;
98
+ };
99
+ const awaitStat = async (filePath) => {
100
+ try {
101
+ return await fs.stat(filePath);
102
+ }
103
+ catch {
104
+ return null;
105
+ }
106
+ };
107
+ const isPageFile = (relativePath) => {
108
+ const normalized = normalizePath(relativePath);
109
+ const fileName = path.basename(normalized);
110
+ const hasPagesSegment = normalized.includes("/pages/");
111
+ const hasRoutesSegment = normalized.includes("/routes/");
112
+ const isAppEntry = /^App\.(tsx?|jsx?)$/.test(fileName);
113
+ const isNextAppPage = normalized.includes("/app/") &&
114
+ /(\/|^)(page|layout|route)\.(tsx?|jsx?|js|ts|vue|svelte)$/.test(normalized);
115
+ const isSvelteKitPage = normalized.includes("/routes/") &&
116
+ /(\/|^)\+page(\.server)?\.(ts|js|svelte)$/.test(normalized);
117
+ return (hasPagesSegment ||
118
+ hasRoutesSegment ||
119
+ isAppEntry ||
120
+ isNextAppPage ||
121
+ isSvelteKitPage);
122
+ };
123
+ const collectFiles = async (directory, projectDir, shouldScanFile, files) => {
124
+ const entries = await fs.readdir(directory, { withFileTypes: true });
125
+ for (const entry of entries) {
126
+ if (entry.name.startsWith(".")) {
127
+ continue;
128
+ }
129
+ const fullPath = path.join(directory, entry.name);
130
+ if (entry.isDirectory()) {
131
+ if (SKIP_DIRECTORIES.has(entry.name)) {
132
+ continue;
133
+ }
134
+ await collectFiles(fullPath, projectDir, shouldScanFile, files);
135
+ continue;
136
+ }
137
+ if (!entry.isFile() || !isSupportedFile(entry.name)) {
138
+ continue;
139
+ }
140
+ const relativePath = normalizePath(path.relative(projectDir, fullPath));
141
+ if (hasSkippedPathSegment(relativePath)) {
142
+ continue;
143
+ }
144
+ if (!shouldScanFile(relativePath)) {
145
+ continue;
146
+ }
147
+ const content = await fs.readFile(fullPath, "utf8");
148
+ files.push({
149
+ filePath: fullPath,
150
+ relativePath,
151
+ keys: extractTranslationKeys(content),
152
+ imports: extractRelativeImports(content),
153
+ });
154
+ }
155
+ };
156
+ export async function buildKeyUsageMap(cfg) {
157
+ const root = projectRoot();
158
+ const i18nDirectory = translationsDir(cfg);
159
+ const candidateRoots = [
160
+ path.dirname(i18nDirectory),
161
+ path.join(root, "src"),
162
+ path.join(root, "app"),
163
+ path.join(root, "pages"),
164
+ path.join(root, "routes"),
165
+ ];
166
+ const sourceRoots = Array.from(new Set(candidateRoots.filter((candidate) => path.resolve(candidate) !== root)));
167
+ const files = [];
168
+ const shouldScanFile = createScanMatcher(cfg.scan);
169
+ for (const sourceRoot of sourceRoots) {
170
+ const stat = await awaitStat(sourceRoot);
171
+ if (!stat?.isDirectory()) {
172
+ continue;
173
+ }
174
+ await collectFiles(sourceRoot, root, shouldScanFile, files);
175
+ }
176
+ const fileByPath = new Map(files.map((file) => [path.resolve(file.filePath), file]));
177
+ const adjacency = new Map();
178
+ for (const file of files) {
179
+ const imports = [];
180
+ for (const specifier of file.imports) {
181
+ const resolvedImport = await resolveImport(file.filePath, specifier);
182
+ if (resolvedImport) {
183
+ const normalizedImport = path.resolve(resolvedImport);
184
+ if (fileByPath.has(normalizedImport)) {
185
+ imports.push(normalizedImport);
186
+ }
187
+ }
188
+ }
189
+ adjacency.set(path.resolve(file.filePath), imports);
190
+ }
191
+ const pages = files.filter((file) => isPageFile(file.relativePath));
192
+ const resultFiles = files
193
+ .filter((file) => file.keys.size > 0)
194
+ .map((file) => {
195
+ const normalizedRelativePath = normalizePath(file.relativePath);
196
+ return {
197
+ id: normalizedRelativePath,
198
+ file: normalizedRelativePath,
199
+ keys: Array.from(file.keys).sort(),
200
+ };
201
+ })
202
+ .sort((left, right) => left.file.localeCompare(right.file));
203
+ const resultPages = pages
204
+ .map((pageFile) => {
205
+ const visited = new Set();
206
+ const queue = [path.resolve(pageFile.filePath)];
207
+ const keys = new Set();
208
+ while (queue.length > 0) {
209
+ const current = queue.pop();
210
+ if (!current || visited.has(current)) {
211
+ continue;
212
+ }
213
+ visited.add(current);
214
+ const currentFile = fileByPath.get(current);
215
+ if (!currentFile) {
216
+ continue;
217
+ }
218
+ for (const key of currentFile.keys) {
219
+ keys.add(key);
220
+ }
221
+ for (const next of adjacency.get(current) ?? []) {
222
+ if (!visited.has(next)) {
223
+ queue.push(next);
224
+ }
225
+ }
226
+ }
227
+ const normalizedRelativePath = normalizePath(pageFile.relativePath);
228
+ const id = normalizedRelativePath.replace(/\.[^.]+$/, "");
229
+ return {
230
+ id,
231
+ file: normalizedRelativePath,
232
+ keys: Array.from(keys).sort(),
233
+ };
234
+ })
235
+ .sort((left, right) => left.file.localeCompare(right.file));
236
+ return {
237
+ pages: resultPages,
238
+ files: resultFiles,
239
+ generatedAt: new Date().toISOString(),
240
+ };
241
+ }
@@ -0,0 +1,102 @@
1
+ import fs from "node:fs/promises";
2
+ import path from "node:path";
3
+ import { createScanMatcher } from "./scanFilters.js";
4
+ import { isLikelyTranslationKey } from "./translationKeys.js";
5
+ const IGNORED_DIRECTORIES = new Set([
6
+ "node_modules",
7
+ "dist",
8
+ "build",
9
+ "out",
10
+ ".git",
11
+ ".next",
12
+ ".nuxt",
13
+ ".turbo",
14
+ "coverage",
15
+ "storybook-static",
16
+ ]);
17
+ const SCANNED_EXTENSIONS = new Set([".ts", ".tsx", ".js", ".jsx"]);
18
+ const USAGE_REGEXES = [
19
+ /\b(?:t|i18n\.t|translate)\(\s*["'`]([^"'`]+)["'`]\s*[\),]/g,
20
+ /\bi18nKey\s*=\s*["'`]([^"'`]+)["'`]/g,
21
+ ];
22
+ const projectRoot = () => process.env.INIT_CWD || process.cwd();
23
+ const normalizePath = (filePath) => filePath.split(path.sep).join("/");
24
+ const isScannableFile = (fileName) => SCANNED_EXTENSIONS.has(path.extname(fileName));
25
+ const hasIgnoredPathSegment = (relativePath) => normalizePath(relativePath)
26
+ .split("/")
27
+ .some((segment) => IGNORED_DIRECTORIES.has(segment));
28
+ const resolveTranslationsDir = (cfg, cwd) => {
29
+ if (path.isAbsolute(cfg.path)) {
30
+ return cfg.path;
31
+ }
32
+ return path.resolve(cwd, cfg.path);
33
+ };
34
+ export const inferUsageRoot = (cfg) => {
35
+ const cwd = projectRoot();
36
+ const translationsDirectory = resolveTranslationsDir(cfg, cwd);
37
+ const relativeToCwd = path.relative(cwd, translationsDirectory);
38
+ const isInsideProject = relativeToCwd.length === 0 ||
39
+ (!relativeToCwd.startsWith("..") && !path.isAbsolute(relativeToCwd));
40
+ if (!isInsideProject) {
41
+ return cwd;
42
+ }
43
+ const parentDirectory = path.dirname(translationsDirectory);
44
+ if (path.basename(parentDirectory) === "src") {
45
+ return path.dirname(parentDirectory);
46
+ }
47
+ return parentDirectory;
48
+ };
49
+ export async function scanUsage(rootDir = projectRoot(), scan) {
50
+ const usage = {};
51
+ const seenFilesByKey = new Map();
52
+ const shouldScanFile = createScanMatcher(scan);
53
+ const scanDirectory = async (directory) => {
54
+ const entries = await fs.readdir(directory, { withFileTypes: true });
55
+ for (const entry of entries) {
56
+ const fullPath = path.join(directory, entry.name);
57
+ if (entry.isDirectory()) {
58
+ if (IGNORED_DIRECTORIES.has(entry.name)) {
59
+ continue;
60
+ }
61
+ await scanDirectory(fullPath);
62
+ continue;
63
+ }
64
+ if (!entry.isFile() || !isScannableFile(entry.name)) {
65
+ continue;
66
+ }
67
+ const relativePath = normalizePath(path.relative(rootDir, fullPath));
68
+ if (hasIgnoredPathSegment(relativePath)) {
69
+ continue;
70
+ }
71
+ if (!shouldScanFile(relativePath)) {
72
+ continue;
73
+ }
74
+ const source = await fs.readFile(fullPath, "utf8");
75
+ for (const usageRegex of USAGE_REGEXES) {
76
+ let match = usageRegex.exec(source);
77
+ while (match) {
78
+ const key = match[1]?.trim();
79
+ if (key && isLikelyTranslationKey(key)) {
80
+ if (!usage[key]) {
81
+ usage[key] = { count: 0, files: [] };
82
+ seenFilesByKey.set(key, new Set());
83
+ }
84
+ usage[key].count += 1;
85
+ const fileSet = seenFilesByKey.get(key);
86
+ if (fileSet && !fileSet.has(relativePath)) {
87
+ fileSet.add(relativePath);
88
+ usage[key].files.push(relativePath);
89
+ }
90
+ }
91
+ match = usageRegex.exec(source);
92
+ }
93
+ usageRegex.lastIndex = 0;
94
+ }
95
+ }
96
+ };
97
+ await scanDirectory(rootDir);
98
+ for (const value of Object.values(usage)) {
99
+ value.files.sort((left, right) => left.localeCompare(right));
100
+ }
101
+ return usage;
102
+ }
package/package.json ADDED
@@ -0,0 +1,43 @@
1
+ {
2
+ "name": "@mxml3gend/gloss",
3
+ "version": "0.1.0",
4
+ "description": "Local-first CLI + web app for managing i18n translation files",
5
+ "license": "MIT",
6
+ "type": "module",
7
+ "main": "./dist/index.js",
8
+ "exports": {
9
+ ".": "./dist/index.js"
10
+ },
11
+ "bin": {
12
+ "gloss": "dist/index.js"
13
+ },
14
+ "files": [
15
+ "dist",
16
+ "README.md",
17
+ "LICENSE"
18
+ ],
19
+ "engines": {
20
+ "node": ">=18"
21
+ },
22
+ "publishConfig": {
23
+ "access": "public"
24
+ },
25
+ "scripts": {
26
+ "dev": "tsx src/index.ts",
27
+ "dev:server": "tsx src/devServer.ts",
28
+ "build": "tsc -p tsconfig.json && node ./scripts/copy-ui-dist.mjs",
29
+ "test": "npm run build && node --test tests/**/*.test.mjs"
30
+ },
31
+ "devDependencies": {
32
+ "@types/cors": "^2.8.19",
33
+ "@types/express": "^5.0.6",
34
+ "@types/node": "^25.2.3",
35
+ "@types/open": "^6.1.0",
36
+ "tsx": "^4.21.0"
37
+ },
38
+ "dependencies": {
39
+ "cors": "^2.8.6",
40
+ "express": "^5.2.1",
41
+ "open": "^11.0.0"
42
+ }
43
+ }