From 91ab6184a7a596ae37ab9a5a9b0129167e20bf10 Mon Sep 17 00:00:00 2001 From: Lev Chelyadinov Date: Thu, 9 Jan 2025 13:22:53 +0200 Subject: [PATCH 1/5] feat: extend the functionality of `inconsistent-naming` --- .changeset/rude-items-shout.md | 5 + .../inconsistent-naming/get-main-subject.ts | 90 ++++++++++++++++++ .../src/inconsistent-naming/index.spec.ts | 95 +++++++++++++++++-- .../src/inconsistent-naming/index.ts | 63 ++++++++---- 4 files changed, 226 insertions(+), 27 deletions(-) create mode 100644 .changeset/rude-items-shout.md create mode 100644 packages/steiger-plugin-fsd/src/inconsistent-naming/get-main-subject.ts diff --git a/.changeset/rude-items-shout.md b/.changeset/rude-items-shout.md new file mode 100644 index 00000000..580c128d --- /dev/null +++ b/.changeset/rude-items-shout.md @@ -0,0 +1,5 @@ +--- +'@feature-sliced/steiger-plugin': minor +--- + +Extend the functionality of `inconsistent-naming` to support multi-word names and handle rename collisions diff --git a/packages/steiger-plugin-fsd/src/inconsistent-naming/get-main-subject.ts b/packages/steiger-plugin-fsd/src/inconsistent-naming/get-main-subject.ts new file mode 100644 index 00000000..deadfbef --- /dev/null +++ b/packages/steiger-plugin-fsd/src/inconsistent-naming/get-main-subject.ts @@ -0,0 +1,90 @@ +import { wordPattern } from '../inconsistent-naming-scheme/detect-naming-scheme.js' + +/** + * Extract the main subject in a multi-word subject name. + * + * @example + * getMainSubject("a book with pages") // "book" + * getMainSubject("admin-users") // "users" + * getMainSubject("receiptsByOrder") // "receipts" + */ +export function getMainSubject(name: string) { + const words = [...name.matchAll(wordPattern)] + .map((match) => match[0]) + .filter((word) => !articles.includes(word.toLocaleLowerCase())) + + const prepositionIndex = words.findIndex((word) => prepositions.includes(word.toLocaleLowerCase())) + if (prepositionIndex === -1) { + return words[words.length - 1] + } + const mainPart = words.slice(0, prepositionIndex) + return mainPart[mainPart.length - 1] +} + +if (import.meta.vitest) { + const { test, expect } = import.meta.vitest + + test('getMainSubject', () => { + expect(getMainSubject('a book with pages')).toEqual('book') + expect(getMainSubject('admin-users')).toEqual('users') + expect(getMainSubject('receiptsByOrder')).toEqual('receipts') + }) +} + +const articles = ['a', 'an', 'the'] +const prepositions = [ + 'about', + 'above', + 'across', + 'after', + 'against', + 'along', + 'among', + 'around', + 'at', + 'before', + 'behind', + 'below', + 'beneath', + 'beside', + 'besides', + 'between', + 'beyond', + 'but', + 'by', + 'concerning', + 'despite', + 'down', + 'during', + 'except', + 'excepting', + 'for', + 'from', + 'in', + 'inside', + 'into', + 'like', + 'near', + 'of', + 'off', + 'on', + 'onto', + 'out', + 'outside', + 'over', + 'past', + 'regarding', + 'since', + 'through', + 'throughout', + 'to', + 'toward', + 'under', + 'underneath', + 'until', + 'up', + 'upon', + 'with', + 'within', + 'without', +] diff --git a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts index 5bfed103..87aa9a5e 100644 --- a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts +++ b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts @@ -1,10 +1,10 @@ import { expect, it } from 'vitest' -import { compareMessages, joinFromRoot, parseIntoFolder as parseIntoFsdRoot } from '@steiger/toolkit' +import { joinFromRoot, parseIntoFolder as parseIntoFsdRoot } from '@steiger/toolkit' import inconsistentNaming from './index.js' -it('reports no errors on slice names that are pluralized consistently', () => { - const root = parseIntoFsdRoot( +it('reports no errors on entity names that are pluralized consistently', () => { + const root1 = parseIntoFsdRoot( ` 📂 entities 📂 users @@ -16,11 +16,41 @@ it('reports no errors on slice names that are pluralized consistently', () => { `, joinFromRoot('users', 'user', 'project', 'src'), ) + const root2 = parseIntoFsdRoot( + ` + 📂 entities + 📂 user + 📂 ui + 📄 index.ts + 📂 post + 📂 ui + 📄 index.ts + `, + joinFromRoot('users', 'user', 'project', 'src'), + ) + + expect(inconsistentNaming.check(root1)).toEqual({ diagnostics: [] }) + expect(inconsistentNaming.check(root2)).toEqual({ diagnostics: [] }) +}) + +it('reports no errors on multi-word entity names that are pluralized consistently', () => { + const root = parseIntoFsdRoot( + ` + 📂 entities + 📂 admin-users + 📂 ui + 📄 index.ts + 📂 employers-of-record + 📂 ui + 📄 index.ts + `, + joinFromRoot('users', 'user', 'project', 'src'), + ) expect(inconsistentNaming.check(root)).toEqual({ diagnostics: [] }) }) -it('reports an error on slice names that are not pluralized consistently', () => { +it('reports an error on entity names that are not pluralized consistently', () => { const root = parseIntoFsdRoot( ` 📂 entities @@ -34,10 +64,10 @@ it('reports an error on slice names that are not pluralized consistently', () => joinFromRoot('users', 'user', 'project', 'src'), ) - const diagnostics = inconsistentNaming.check(root).diagnostics.sort(compareMessages) + const diagnostics = inconsistentNaming.check(root).diagnostics expect(diagnostics).toEqual([ { - message: 'Inconsistent pluralization of slice names. Prefer all plural names', + message: 'Inconsistent pluralization of entity names. Prefer all plural names.', fixes: [ { type: 'rename', @@ -54,10 +84,10 @@ it('prefers the singular form when there are more singular slices', () => { const root = parseIntoFsdRoot( ` 📂 entities - 📂 user + 📂 admin-user 📂 ui 📄 index.ts - 📂 post + 📂 news-post 📂 ui 📄 index.ts 📂 comments @@ -67,10 +97,10 @@ it('prefers the singular form when there are more singular slices', () => { joinFromRoot('users', 'user', 'project', 'src'), ) - const diagnostics = inconsistentNaming.check(root).diagnostics.sort(compareMessages) + const diagnostics = inconsistentNaming.check(root).diagnostics expect(diagnostics).toEqual([ { - message: 'Inconsistent pluralization of slice names. Prefer all singular names', + message: 'Inconsistent pluralization of entity names. Prefer all singular names.', fixes: [ { type: 'rename', @@ -82,3 +112,48 @@ it('prefers the singular form when there are more singular slices', () => { }, ]) }) + +it('recognizes the special case when there are two pluralizations of the same name', () => { + const root = parseIntoFsdRoot( + ` + 📂 entities + 📂 admin-user + 📂 ui + 📄 index.ts + 📂 admin-users + 📂 ui + 📄 index.ts + `, + joinFromRoot('users', 'user', 'project', 'src'), + ) + + const diagnostics = inconsistentNaming.check(root).diagnostics + expect(diagnostics).toEqual([ + { + message: 'Avoid having both "admin-user" and "admin-users" entities.', + location: { path: joinFromRoot('users', 'user', 'project', 'src', 'entities', 'admin-user') }, + }, + ]) +}) + +it('allows inconsistency between different slice groups', () => { + const root = parseIntoFsdRoot( + ` + 📂 entities + 📂 admin-user + 📂 ui + 📄 index.ts + 📂 group + 📄 index.ts + 📂 post-parts + 📂 posts + 📄 index.ts + 📂 authors + 📄 index.ts + `, + joinFromRoot('users', 'user', 'project', 'src'), + ) + + const diagnostics = inconsistentNaming.check(root).diagnostics + expect(diagnostics).toEqual([]) +}) diff --git a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts index 9fd3af17..5caa1705 100644 --- a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts +++ b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts @@ -7,8 +7,9 @@ import type { PartialDiagnostic, Rule } from '@steiger/toolkit' import { groupSlices } from '../_lib/group-slices.js' import { NAMESPACE } from '../constants.js' +import { getMainSubject } from './get-main-subject.js' -/** Detect inconsistent naming of slices on layers (singular vs plural) */ +/** Detect inconsistent pluralization of entities. */ const inconsistentNaming = { name: `${NAMESPACE}/inconsistent-naming` as const, check(root) { @@ -22,29 +23,57 @@ const inconsistentNaming = { const slices = getSlices(entities) const sliceNames = groupSlices(Object.keys(slices)) for (const [groupPrefix, group] of Object.entries(sliceNames)) { - const [pluralNames, singularNames] = partition(group, isPlural) + const [pluralNames, singularNames] = partition( + group.map((name) => [name, getMainSubject(name)] as const), + ([, mainSubject]) => isPlural(mainSubject), + ) + + /** Names that exist in both singular and plural forms for filtering later. */ + const duplicates = { + singular: [] as Array, + plural: [] as Array, + } if (pluralNames.length > 0 && singularNames.length > 0) { - const message = 'Inconsistent pluralization of slice names' + for (const [singularName, mainSubject] of singularNames) { + const pluralized = singularName.replace(mainSubject, plural(mainSubject)) + if (group.includes(pluralized)) { + duplicates.singular.push(singularName) + duplicates.plural.push(pluralized) + + diagnostics.push({ + message: `Avoid having both "${singularName}" and "${plural(singularName)}" entities${groupPrefix === '' ? '' : ' in the same slice group'}.`, + location: { path: join(entities.path, groupPrefix, singularName) }, + }) + } + } - if (pluralNames.length >= singularNames.length) { + const message = 'Inconsistent pluralization of entity names' + if ( + pluralNames.length >= singularNames.length && + singularNames.some(([name]) => !duplicates.singular.includes(name)) + ) { diagnostics.push({ - message: `${message}. Prefer all plural names`, - fixes: singularNames.map((name) => ({ - type: 'rename', - path: join(entities.path, groupPrefix, name), - newName: plural(name), - })), + message: `${message}. Prefer all plural names.`, + fixes: singularNames + .filter(([name]) => !duplicates.singular.includes(name)) + .map(([name, mainWord]) => ({ + type: 'rename', + path: join(entities.path, groupPrefix, name), + newName: name.replace(mainWord, plural(mainWord)), + })), location: { path: join(entities.path, groupPrefix) }, }) - } else { + } else if (pluralNames.some(([name]) => !duplicates.plural.includes(name))) { diagnostics.push({ - message: `${message}. Prefer all singular names`, - fixes: pluralNames.map((name) => ({ - type: 'rename', - path: join(entities.path, groupPrefix, name), - newName: singular(name), - })), + message: `${message}. Prefer all singular names.`, + fixes: pluralNames + .filter(([name]) => !duplicates.plural.includes(name)) + .map(([name, mainWord]) => ({ + type: 'rename', + path: join(entities.path, groupPrefix, name), + newName: name.replace(mainWord, singular(mainWord)), + })), location: { path: join(entities.path, groupPrefix) }, }) } From a9b25be803173ac31666371344bafa442162cf4d Mon Sep 17 00:00:00 2001 From: Lev Chelyadinov Date: Thu, 9 Jan 2025 15:38:32 +0200 Subject: [PATCH 2/5] Fix an import --- .../src/inconsistent-naming/get-main-subject.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/packages/steiger-plugin-fsd/src/inconsistent-naming/get-main-subject.ts b/packages/steiger-plugin-fsd/src/inconsistent-naming/get-main-subject.ts index deadfbef..5c8d9e53 100644 --- a/packages/steiger-plugin-fsd/src/inconsistent-naming/get-main-subject.ts +++ b/packages/steiger-plugin-fsd/src/inconsistent-naming/get-main-subject.ts @@ -1,4 +1,5 @@ -import { wordPattern } from '../inconsistent-naming-scheme/detect-naming-scheme.js' +/** Extracts individual words in every naming scheme. */ +export const wordPattern = /([A-Z0-9]{2,}(?![A-Z][a-z])|[A-Z]?[a-z0-9]+)/g /** * Extract the main subject in a multi-word subject name. From e9d01adcd580deb20461cc23cd2d2c9bba9d6f80 Mon Sep 17 00:00:00 2001 From: Lev Chelyadinov Date: Sat, 18 Jan 2025 12:08:05 +0100 Subject: [PATCH 3/5] Favor singular names when there's equal amounts --- .../src/inconsistent-naming/index.spec.ts | 6 +++--- .../steiger-plugin-fsd/src/inconsistent-naming/index.ts | 2 +- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts index 87aa9a5e..0cb1d6a9 100644 --- a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts +++ b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts @@ -67,12 +67,12 @@ it('reports an error on entity names that are not pluralized consistently', () = const diagnostics = inconsistentNaming.check(root).diagnostics expect(diagnostics).toEqual([ { - message: 'Inconsistent pluralization of entity names. Prefer all plural names.', + message: 'Inconsistent pluralization of entity names. Prefer all singular names.', fixes: [ { type: 'rename', - path: joinFromRoot('users', 'user', 'project', 'src', 'entities', 'user'), - newName: 'users', + path: joinFromRoot('users', 'user', 'project', 'src', 'entities', 'posts'), + newName: 'post', }, ], location: { path: joinFromRoot('users', 'user', 'project', 'src', 'entities') }, diff --git a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts index 5caa1705..6b897677 100644 --- a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts +++ b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts @@ -50,7 +50,7 @@ const inconsistentNaming = { const message = 'Inconsistent pluralization of entity names' if ( - pluralNames.length >= singularNames.length && + pluralNames.length > singularNames.length && singularNames.some(([name]) => !duplicates.singular.includes(name)) ) { diagnostics.push({ From b8daa1522fd570d4b295427f75b075bd73a3c11e Mon Sep 17 00:00:00 2001 From: Lev Chelyadinov Date: Sat, 18 Jan 2025 12:09:01 +0100 Subject: [PATCH 4/5] Improve the test phrasing --- .../steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts index 0cb1d6a9..1941fd02 100644 --- a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts +++ b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.spec.ts @@ -113,7 +113,7 @@ it('prefers the singular form when there are more singular slices', () => { ]) }) -it('recognizes the special case when there are two pluralizations of the same name', () => { +it('recognizes the special case when there is a plural and singular form of the same name', () => { const root = parseIntoFsdRoot( ` 📂 entities From 6645ffc320c5f7303a0a55b4346cc8c3e3ee62f9 Mon Sep 17 00:00:00 2001 From: Lev Chelyadinov Date: Sat, 18 Jan 2025 12:09:12 +0100 Subject: [PATCH 5/5] Simplify duplicate checks --- .../steiger-plugin-fsd/src/inconsistent-naming/index.ts | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts index 6b897677..8e002478 100644 --- a/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts +++ b/packages/steiger-plugin-fsd/src/inconsistent-naming/index.ts @@ -49,10 +49,7 @@ const inconsistentNaming = { } const message = 'Inconsistent pluralization of entity names' - if ( - pluralNames.length > singularNames.length && - singularNames.some(([name]) => !duplicates.singular.includes(name)) - ) { + if (pluralNames.length > singularNames.length && singularNames.length > duplicates.singular.length) { diagnostics.push({ message: `${message}. Prefer all plural names.`, fixes: singularNames @@ -64,7 +61,7 @@ const inconsistentNaming = { })), location: { path: join(entities.path, groupPrefix) }, }) - } else if (pluralNames.some(([name]) => !duplicates.plural.includes(name))) { + } else if (pluralNames.length > duplicates.plural.length) { diagnostics.push({ message: `${message}. Prefer all singular names.`, fixes: pluralNames