Skip to content

Commit d447ffb

Browse files
authoredOct 9, 2024··
feat: throw error when a group does not exist or duplicated in sort-classes
1 parent 4e7e5ad commit d447ffb

5 files changed

+168
-2
lines changed
 

‎rules/sort-classes-utils.ts

+47-1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,8 @@ import type {
99
} from './sort-classes.types'
1010
import type { CompareOptions } from '../utils/compare'
1111

12+
import { validateNoDuplicatedGroups } from '../utils/validate-groups-configuration'
13+
import { allModifiers, allSelectors } from './sort-classes.types'
1214
import { matches } from '../utils/matches'
1315

1416
interface CustomGroupMatchesProps {
@@ -72,7 +74,7 @@ export const generateOfficialGroups = (
7274
/**
7375
* Get possible combinations of n elements from an array
7476
*/
75-
const getCombinations = (array: string[], n: number): string[][] => {
77+
export const getCombinations = (array: string[], n: number): string[][] => {
7678
let result: string[][] = []
7779

7880
let backtrack = (start: number, comb: string[]) => {
@@ -253,3 +255,47 @@ export const getCompareOptions = (
253255
ignoreCase: options.ignoreCase,
254256
}
255257
}
258+
259+
export let validateGroupsConfiguration = (
260+
groups: Required<SortClassesOptions[0]>['groups'],
261+
customGroups: Required<SortClassesOptions[0]>['customGroups'],
262+
): void => {
263+
let availableCustomGroupNames = Array.isArray(customGroups)
264+
? customGroups.map(customGroup => customGroup.groupName)
265+
: Object.keys(customGroups)
266+
let invalidGroups = groups
267+
.flat()
268+
.filter(
269+
group =>
270+
!isPredefinedGroup(group) && !availableCustomGroupNames.includes(group),
271+
)
272+
if (invalidGroups.length) {
273+
throw new Error('Invalid group(s): ' + invalidGroups.join(', '))
274+
}
275+
validateNoDuplicatedGroups(groups)
276+
}
277+
278+
const isPredefinedGroup = (input: string): boolean => {
279+
if (input === 'unknown') {
280+
return true
281+
}
282+
let singleWordSelector = input.split('-').at(-1)
283+
if (!singleWordSelector) {
284+
return false
285+
}
286+
let twoWordsSelector = input.split('-').slice(-2).join('-')
287+
let isTwoWordSelectorValid = allSelectors.includes(
288+
twoWordsSelector as Selector,
289+
)
290+
if (
291+
!allSelectors.includes(singleWordSelector as Selector) &&
292+
!isTwoWordSelectorValid
293+
) {
294+
return false
295+
}
296+
let modifiers = input.split('-').slice(0, isTwoWordSelectorValid ? -2 : -1)
297+
return (
298+
new Set(modifiers).size === modifiers.length &&
299+
modifiers.every(modifier => allModifiers.includes(modifier as Modifier))
300+
)
301+
}

‎rules/sort-classes.ts

+3
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ import type {
99
import type { SortingNodeWithDependencies } from '../utils/sort-nodes-by-dependencies'
1010

1111
import {
12+
validateGroupsConfiguration,
1213
getOverloadSignatureGroups,
1314
generateOfficialGroups,
1415
customGroupMatches,
@@ -243,6 +244,8 @@ export default createEslintRule<SortClassesOptions, MESSAGE_ID>({
243244
order: 'asc',
244245
} as const)
245246

247+
validateGroupsConfiguration(options.groups, options.customGroups)
248+
246249
let sourceCode = getSourceCode(context)
247250
let className = node.parent.id?.name
248251

‎test/sort-classes-utils.test.ts

+86-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,11 @@
11
import { describe, expect, it } from 'vitest'
22

3-
import { generateOfficialGroups } from '../rules/sort-classes-utils'
3+
import {
4+
validateGroupsConfiguration,
5+
generateOfficialGroups,
6+
getCombinations,
7+
} from '../rules/sort-classes-utils'
8+
import { allModifiers, allSelectors } from '../rules/sort-classes.types'
49

510
describe('sort-classes-utils', () => {
611
it('sort-classes-utils: should generate official groups', () => {
@@ -44,4 +49,84 @@ describe('sort-classes-utils', () => {
4449
'method',
4550
])
4651
})
52+
53+
describe('validateGroupsConfiguration', () => {
54+
it('allows predefined groups', () => {
55+
let allModifierCombinationPermutations =
56+
getAllNonEmptyCombinations(allModifiers)
57+
let allPredefinedGroups = allSelectors
58+
.map(selector =>
59+
allModifierCombinationPermutations.map(
60+
modifiers => `${modifiers.join('-')}-${selector}`,
61+
),
62+
)
63+
.flat()
64+
.concat(allSelectors)
65+
expect(
66+
validateGroupsConfiguration(allPredefinedGroups, []),
67+
).toBeUndefined()
68+
})
69+
70+
it('allows custom groups with the new API', () => {
71+
expect(
72+
validateGroupsConfiguration(
73+
['static-property', 'myCustomGroup'],
74+
[
75+
{
76+
groupName: 'myCustomGroup',
77+
},
78+
],
79+
),
80+
).toBeUndefined()
81+
})
82+
83+
it('throws an error with predefined groups with duplicate modifiers', () => {
84+
expect(() =>
85+
validateGroupsConfiguration(['static-static-property'], []),
86+
).toThrow('Invalid group(s): static-static-property')
87+
})
88+
89+
it('throws an error if a duplicate group is provided', () => {
90+
expect(() =>
91+
validateGroupsConfiguration(['static-property', 'static-property'], []),
92+
).toThrow('Duplicated group(s): static-property')
93+
})
94+
95+
it('throws an error if invalid groups are provided with the new API', () => {
96+
expect(() =>
97+
validateGroupsConfiguration(
98+
['static-property', 'myCustomGroup', ''],
99+
[
100+
{
101+
groupName: 'myCustomGroupNotReferenced',
102+
},
103+
],
104+
),
105+
).toThrow('Invalid group(s): myCustomGroup')
106+
})
107+
108+
it('allows groups with the old API', () => {
109+
expect(
110+
validateGroupsConfiguration(['static-property', 'myCustomGroup'], {
111+
myCustomGroup: 'foo',
112+
}),
113+
).toBeUndefined()
114+
})
115+
116+
it('throws an error if invalid custom groups are provided with the old API', () => {
117+
expect(() =>
118+
validateGroupsConfiguration(['static-property', 'myCustomGroup'], {
119+
myCustomGroupNotReferenced: 'foo',
120+
}),
121+
).toThrow('Invalid group(s): myCustomGroup')
122+
})
123+
})
47124
})
125+
126+
const getAllNonEmptyCombinations = (array: string[]): string[][] => {
127+
let result: string[][] = []
128+
for (let i = 1; i < array.length; i++) {
129+
result = [...result, ...getCombinations(array, i)]
130+
}
131+
return result
132+
}

‎test/validate-groups-configuration.test.ts

+10
Original file line numberDiff line numberDiff line change
@@ -21,4 +21,14 @@ describe('validate-groups-configuration', () => {
2121
)
2222
}).toThrow('Invalid group(s): invalidGroup1, invalidGroup2')
2323
})
24+
25+
it('throws an error when a duplicate group is provided', () => {
26+
expect(() => {
27+
validateGroupsConfiguration(
28+
['predefinedGroup', 'predefinedGroup'],
29+
['predefinedGroup'],
30+
[],
31+
)
32+
}).toThrow('Duplicated group(s): predefinedGroup')
33+
})
2434
})

‎utils/validate-groups-configuration.ts

+22
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,9 @@
1+
/**
2+
* Throws an error if one of the following conditions is met:
3+
* - One or more groups specified in `groups` are not predefined nor specified
4+
* in `customGroups`
5+
* - A group is specified in `groups` more than once
6+
*/
17
export let validateGroupsConfiguration = (
28
groups: (string[] | string)[],
39
allowedPredefinedGroups: string[],
@@ -13,4 +19,20 @@ export let validateGroupsConfiguration = (
1319
if (invalidGroups.length) {
1420
throw new Error('Invalid group(s): ' + invalidGroups.join(', '))
1521
}
22+
validateNoDuplicatedGroups(groups)
23+
}
24+
25+
/**
26+
* Throws an error if a group is specified more than once
27+
*/
28+
export let validateNoDuplicatedGroups = (
29+
groups: (string[] | string)[],
30+
): void => {
31+
let flattenGroups = groups.flat()
32+
let duplicatedGroups = flattenGroups.filter(
33+
(group, index) => flattenGroups.indexOf(group) !== index,
34+
)
35+
if (duplicatedGroups.length) {
36+
throw new Error('Duplicated group(s): ' + duplicatedGroups.join(', '))
37+
}
1638
}

0 commit comments

Comments
 (0)
Please sign in to comment.