@dxos/eslint-plugin-rules 0.8.4-main.ae835ea → 0.8.4-main.bbf232bc24

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.
@@ -0,0 +1,443 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ 'use strict';
6
+
7
+ import { readFileSync, existsSync } from 'node:fs';
8
+ import { join, dirname } from 'node:path';
9
+
10
+ /**
11
+ * Valid type suffixes for translation keys.
12
+ */
13
+ const VALID_SUFFIXES = [
14
+ 'label',
15
+ 'message',
16
+ 'placeholder',
17
+ 'title',
18
+ 'description',
19
+ 'heading',
20
+ 'alt',
21
+ 'button',
22
+ 'name',
23
+ 'value',
24
+ 'icon',
25
+ 'menu',
26
+ ];
27
+
28
+ /**
29
+ * Plural suffixes appended by i18next.
30
+ */
31
+ const PLURAL_SUFFIXES = ['_zero', '_one', '_other'];
32
+
33
+ /**
34
+ * Strip i18next plural suffix from a key.
35
+ */
36
+ const stripPluralSuffix = (key) => {
37
+ for (const suffix of PLURAL_SUFFIXES) {
38
+ if (key.endsWith(suffix)) {
39
+ return key.slice(0, -suffix.length);
40
+ }
41
+ }
42
+ return key;
43
+ };
44
+
45
+ /**
46
+ * Get the plural suffix if present.
47
+ */
48
+ const getPluralSuffix = (key) => {
49
+ for (const suffix of PLURAL_SUFFIXES) {
50
+ if (key.endsWith(suffix)) {
51
+ return suffix;
52
+ }
53
+ }
54
+ return '';
55
+ };
56
+
57
+ /**
58
+ * Check if a key has a valid type suffix (after stripping plural suffix).
59
+ * Only dot-separated suffixes are considered valid (e.g., 'foo.label').
60
+ */
61
+ const hasValidSuffix = (key) => {
62
+ const base = stripPluralSuffix(key);
63
+ return VALID_SUFFIXES.some((suffix) => {
64
+ return base === suffix || base.endsWith(`.${suffix}`);
65
+ });
66
+ };
67
+
68
+ /**
69
+ * Check if a key has a suffix joined by hyphen instead of dot (e.g., 'plugin-name').
70
+ * Returns the corrected key if fixable, null otherwise.
71
+ */
72
+ const fixHyphenatedSuffix = (key) => {
73
+ const pluralSuffix = getPluralSuffix(key);
74
+ const base = stripPluralSuffix(key);
75
+ for (const suffix of VALID_SUFFIXES) {
76
+ if (base.endsWith(`-${suffix}`) && base.length > suffix.length + 1) {
77
+ const prefix = base.slice(0, -(suffix.length + 1));
78
+ return `${prefix}.${suffix}${pluralSuffix}`;
79
+ }
80
+ }
81
+ return null;
82
+ };
83
+
84
+ /**
85
+ * Check if a key uses dot.kebab-case format (no spaces).
86
+ */
87
+ const isDotNotation = (key) => {
88
+ const base = stripPluralSuffix(key);
89
+ return !base.includes(' ');
90
+ };
91
+
92
+ /**
93
+ * Convert a space-separated key to dot.kebab-case format.
94
+ */
95
+ const toDotNotation = (key) => {
96
+ const pluralSuffix = getPluralSuffix(key);
97
+ const base = stripPluralSuffix(key);
98
+ const words = base.split(' ');
99
+
100
+ if (words.length <= 1) {
101
+ return key;
102
+ }
103
+
104
+ const lastWord = words[words.length - 1];
105
+ const isLastWordSuffix = VALID_SUFFIXES.includes(lastWord);
106
+
107
+ if (isLastWordSuffix) {
108
+ const pathWords = words.slice(0, -1);
109
+ const kebab = pathWords.join('-');
110
+ return `${kebab}.${lastWord}${pluralSuffix}`;
111
+ }
112
+
113
+ // No valid suffix — kebab-case everything.
114
+ const kebab = words.join('-');
115
+ return `${kebab}${pluralSuffix}`;
116
+ };
117
+
118
+ // --- Key validation: resolve meta.id and load translations ---
119
+
120
+ /** Cache: pluginDir → { metaId, keys } */
121
+ const pluginCache = new Map();
122
+
123
+ /**
124
+ * Walk up from a file path to find the package root (directory containing src/translations.ts).
125
+ */
126
+ const findPackageDir = (filePath) => {
127
+ let dir = dirname(filePath);
128
+ for (let i = 0; i < 10; i++) {
129
+ if (existsSync(join(dir, 'src/translations.ts'))) {
130
+ return dir;
131
+ }
132
+ const parent = dirname(dir);
133
+ if (parent === dir) {
134
+ break;
135
+ }
136
+ dir = parent;
137
+ }
138
+ return null;
139
+ };
140
+
141
+ /**
142
+ * Resolve the namespace identifier for a package.
143
+ * Plugins use meta.id from src/meta.ts.
144
+ * UI packages use translationKey from src/translations.ts.
145
+ */
146
+ const resolveNamespace = (packageDir) => {
147
+ // Try plugin pattern: src/meta.ts with id field.
148
+ const metaPath = join(packageDir, 'src/meta.ts');
149
+ if (existsSync(metaPath)) {
150
+ const content = readFileSync(metaPath, 'utf-8');
151
+ const match = content.match(/id:\s*['"]([^'"]+)['"]/);
152
+ if (match) {
153
+ return { namespace: match[1], source: 'meta.id' };
154
+ }
155
+ }
156
+
157
+ // Try UI package pattern: export const translationKey = '...'.
158
+ const translationsPath = join(packageDir, 'src/translations.ts');
159
+ if (existsSync(translationsPath)) {
160
+ const content = readFileSync(translationsPath, 'utf-8');
161
+ const match = content.match(/export\s+const\s+translationKey\s*=\s*['"]([^'"]+)['"]/);
162
+ if (match) {
163
+ return { namespace: match[1], source: 'translationKey' };
164
+ }
165
+ }
166
+
167
+ return null;
168
+ };
169
+
170
+ /**
171
+ * Extract translation keys for the primary namespace from a translations.ts file.
172
+ * Handles both [meta.id]: { ... } and [translationKey]: { ... } patterns.
173
+ */
174
+ const readTranslationKeys = (packageDir, nsSource) => {
175
+ const keys = new Set();
176
+ const translationsPath = join(packageDir, 'src/translations.ts');
177
+ if (!existsSync(translationsPath)) {
178
+ return keys;
179
+ }
180
+
181
+ const content = readFileSync(translationsPath, 'utf-8');
182
+ const lines = content.split('\n');
183
+ let inBlock = false;
184
+ let braceDepth = 0;
185
+
186
+ // Match the namespace block header based on the source pattern.
187
+ const blockPattern = nsSource === 'meta.id' ? /\[meta\.id\]\s*:\s*\{/ : /\[translationKey\]\s*:\s*\{/;
188
+
189
+ for (const line of lines) {
190
+ const trimmed = line.trim();
191
+
192
+ if (!inBlock && trimmed.match(blockPattern)) {
193
+ inBlock = true;
194
+ braceDepth = 1;
195
+ continue;
196
+ }
197
+
198
+ if (inBlock) {
199
+ for (const ch of trimmed) {
200
+ if (ch === '{') {
201
+ braceDepth++;
202
+ }
203
+ if (ch === '}') {
204
+ braceDepth--;
205
+ }
206
+ }
207
+
208
+ const keyMatch = trimmed.match(/^['"]([^'"]+)['"]\s*:/);
209
+ if (keyMatch && braceDepth >= 1) {
210
+ keys.add(keyMatch[1]);
211
+ }
212
+
213
+ if (braceDepth <= 0) {
214
+ inBlock = false;
215
+ }
216
+ }
217
+ }
218
+
219
+ return keys;
220
+ };
221
+
222
+ /**
223
+ * Get package info (namespace + valid keys) for a file, with caching.
224
+ */
225
+ const getPackageInfo = (filePath) => {
226
+ const packageDir = findPackageDir(filePath);
227
+ if (!packageDir) {
228
+ return null;
229
+ }
230
+
231
+ if (pluginCache.has(packageDir)) {
232
+ return pluginCache.get(packageDir);
233
+ }
234
+
235
+ const nsInfo = resolveNamespace(packageDir);
236
+ if (!nsInfo) {
237
+ pluginCache.set(packageDir, null);
238
+ return null;
239
+ }
240
+
241
+ const keys = readTranslationKeys(packageDir, nsInfo.source);
242
+ const info = { namespace: nsInfo.namespace, nsSource: nsInfo.source, keys };
243
+ pluginCache.set(packageDir, info);
244
+ return info;
245
+ };
246
+
247
+ export default {
248
+ meta: {
249
+ type: 'suggestion',
250
+ fixable: 'code',
251
+ docs: {
252
+ description:
253
+ 'Enforce translation key format: dot.kebab-case with required type suffix. Validates keys exist in translations.',
254
+ },
255
+ messages: {
256
+ missingSuffix: 'Invalid translation key: "{{key}}"',
257
+ useDotsNotSpaces: 'Translation key "{{key}}" should use dot.kebab-case format. Suggested: "{{suggested}}".',
258
+ undefinedKey: 'Translation key "{{key}}" is not defined in translations for namespace "{{namespace}}".',
259
+ },
260
+ schema: [
261
+ {
262
+ type: 'object',
263
+ properties: {
264
+ suffixes: {
265
+ type: 'array',
266
+ items: { type: 'string' },
267
+ },
268
+ },
269
+ additionalProperties: false,
270
+ },
271
+ ],
272
+ },
273
+
274
+ create(context) {
275
+ const options = context.options[0] || {};
276
+ const suffixes = options.suffixes || VALID_SUFFIXES;
277
+ const filename =
278
+ context.filename || (context.getFilename && context.getFilename()) || context.physicalFilename || '';
279
+
280
+ // Check source text once to determine if this is a translation-aware file.
281
+ const sourceText = (context.sourceCode || context.getSourceCode()).text;
282
+ const usesTranslation = sourceText.includes('useTranslation');
283
+ const usesStaticNamespace =
284
+ sourceText.includes('useTranslation(meta.id)') || sourceText.includes('useTranslation(translationKey)');
285
+
286
+ // Resolve package info for this file (works for both plugins and UI packages).
287
+ const packageInfo = getPackageInfo(filename);
288
+
289
+ /**
290
+ * Check a string literal node that represents a translation key.
291
+ * @param {boolean} isDefinition - true for keys in translations.ts definitions.
292
+ */
293
+ const checkKeyFormat = (node, key, isDefinition = false) => {
294
+ // Check 1: Must use dot notation (no spaces).
295
+ if (!isDotNotation(key)) {
296
+ const suggested = toDotNotation(key);
297
+ context.report({
298
+ node,
299
+ messageId: 'useDotsNotSpaces',
300
+ data: { key, suggested },
301
+ fix: (fixer) => fixer.replaceText(node, `'${suggested}'`),
302
+ });
303
+ return;
304
+ }
305
+
306
+ // Check 2: Suffix joined by hyphen instead of dot (e.g., 'plugin-name' → 'plugin.name').
307
+ const fixedKey = fixHyphenatedSuffix(key);
308
+ if (fixedKey) {
309
+ context.report({
310
+ node,
311
+ messageId: 'useDotsNotSpaces',
312
+ data: { key, suggested: fixedKey },
313
+ fix: (fixer) => fixer.replaceText(node, `'${fixedKey}'`),
314
+ });
315
+ return;
316
+ }
317
+
318
+ // Check 3: Must end with a valid suffix (definitions only).
319
+ if (isDefinition && !hasValidSuffix(key)) {
320
+ context.report({
321
+ node,
322
+ messageId: 'missingSuffix',
323
+ data: { key, suffixes: suffixes.join(', ') },
324
+ });
325
+ }
326
+ };
327
+
328
+ /**
329
+ * Check that a key exists in the package's translations.
330
+ */
331
+ const checkKeyExists = (node, key, hasNsOverride) => {
332
+ // Only check if we resolved the package and the file uses a static namespace.
333
+ if (!packageInfo || !usesStaticNamespace || hasNsOverride) {
334
+ return;
335
+ }
336
+
337
+ // Check exact key, then check plural variants (i18next resolves 'foo.label' with { count } to 'foo.label_one'/'foo.label_other').
338
+ const hasPluralVariant = PLURAL_SUFFIXES.some((suffix) => packageInfo.keys.has(key + suffix));
339
+ if (!packageInfo.keys.has(key) && !hasPluralVariant) {
340
+ context.report({
341
+ node,
342
+ messageId: 'undefinedKey',
343
+ data: { key, namespace: packageInfo.namespace },
344
+ });
345
+ }
346
+ };
347
+
348
+ return {
349
+ // t('some key') or t('some key', { ns: ... }).
350
+ // Only fires in files that use useTranslation.
351
+ CallExpression(node) {
352
+ if (
353
+ !usesTranslation ||
354
+ node.callee.type !== 'Identifier' ||
355
+ node.callee.name !== 't' ||
356
+ node.arguments.length < 1 ||
357
+ node.arguments[0].type !== 'Literal' ||
358
+ typeof node.arguments[0].value !== 'string'
359
+ ) {
360
+ return;
361
+ }
362
+
363
+ const key = node.arguments[0].value;
364
+
365
+ // Check format.
366
+ checkKeyFormat(node.arguments[0], key);
367
+
368
+ // Check if there's an ns override in the second argument.
369
+ let hasNsOverride = false;
370
+ if (node.arguments.length >= 2 && node.arguments[1].type === 'ObjectExpression') {
371
+ hasNsOverride = node.arguments[1].properties.some(
372
+ (prop) =>
373
+ prop.key &&
374
+ ((prop.key.type === 'Identifier' && prop.key.name === 'ns') ||
375
+ (prop.key.type === 'Literal' && prop.key.value === 'ns')),
376
+ );
377
+ }
378
+
379
+ // Check existence.
380
+ checkKeyExists(node.arguments[0], key, hasNsOverride);
381
+ },
382
+
383
+ // Label tuples: ['key', { ns: meta.id, ... }]
384
+ // Matches ArrayExpression with a string literal first element and an object with an `ns` property.
385
+ ArrayExpression(node) {
386
+ if (
387
+ node.elements.length >= 2 &&
388
+ node.elements[0] &&
389
+ node.elements[0].type === 'Literal' &&
390
+ typeof node.elements[0].value === 'string' &&
391
+ node.elements[1] &&
392
+ node.elements[1].type === 'ObjectExpression' &&
393
+ node.elements[1].properties.some(
394
+ (prop) =>
395
+ prop.key &&
396
+ ((prop.key.type === 'Identifier' && prop.key.name === 'ns') ||
397
+ (prop.key.type === 'Literal' && prop.key.value === 'ns')),
398
+ )
399
+ ) {
400
+ const key = node.elements[0].value;
401
+
402
+ // Check format.
403
+ checkKeyFormat(node.elements[0], key);
404
+
405
+ // Check existence (determine if ns points to the package's own namespace).
406
+ const nsProp = node.elements[1].properties.find(
407
+ (prop) =>
408
+ prop.key &&
409
+ ((prop.key.type === 'Identifier' && prop.key.name === 'ns') ||
410
+ (prop.key.type === 'Literal' && prop.key.value === 'ns')),
411
+ );
412
+ const isOwnNamespace =
413
+ nsProp &&
414
+ nsProp.value &&
415
+ ((nsProp.value.type === 'MemberExpression' &&
416
+ nsProp.value.object.type === 'Identifier' &&
417
+ nsProp.value.object.name === 'meta' &&
418
+ nsProp.value.property.type === 'Identifier' &&
419
+ nsProp.value.property.name === 'id') ||
420
+ (nsProp.value.type === 'Identifier' && nsProp.value.name === 'translationKey'));
421
+ checkKeyExists(node.elements[0], key, !isOwnNamespace);
422
+ }
423
+ },
424
+
425
+ // Property keys in translations.ts files.
426
+ // Guard: only check files whose source contains the translations export pattern.
427
+ Property(node) {
428
+ if (!sourceText.includes('satisfies Resource[]')) {
429
+ return;
430
+ }
431
+
432
+ if (
433
+ node.key.type === 'Literal' &&
434
+ typeof node.key.value === 'string' &&
435
+ node.value.type === 'Literal' &&
436
+ typeof node.value.value === 'string'
437
+ ) {
438
+ checkKeyFormat(node.key, node.key.value, true);
439
+ }
440
+ },
441
+ };
442
+ },
443
+ };