|
| 1 | +import type { RuleContext } from '@typescript-eslint/utils/ts-eslint' |
| 2 | +import type { TSESTree } from '@typescript-eslint/types' |
| 3 | + |
| 4 | +import type { SortingNode } from '../typings' |
| 5 | + |
| 6 | +import { validateGroupsConfiguration } from '../utils/validate-groups-configuration' |
| 7 | +import { hasPartitionComment } from '../utils/is-partition-comment' |
| 8 | +import { sortNodesByGroups } from '../utils/sort-nodes-by-groups' |
| 9 | +import { getCommentsBefore } from '../utils/get-comments-before' |
| 10 | +import { createEslintRule } from '../utils/create-eslint-rule' |
| 11 | +import { getDecoratorName } from '../utils/get-decorator-name' |
| 12 | +import { getGroupNumber } from '../utils/get-group-number' |
| 13 | +import { getSourceCode } from '../utils/get-source-code' |
| 14 | +import { toSingleLine } from '../utils/to-single-line' |
| 15 | +import { rangeToDiff } from '../utils/range-to-diff' |
| 16 | +import { getSettings } from '../utils/get-settings' |
| 17 | +import { useGroups } from '../utils/use-groups' |
| 18 | +import { makeFixes } from '../utils/make-fixes' |
| 19 | +import { complete } from '../utils/complete' |
| 20 | +import { pairwise } from '../utils/pairwise' |
| 21 | + |
| 22 | +type MESSAGE_ID = 'unexpectedDecoratorsGroupOrder' | 'unexpectedDecoratorsOrder' |
| 23 | + |
| 24 | +type Group<T extends string[]> = 'unknown' | T[number] |
| 25 | + |
| 26 | +export type Options<T extends string[]> = [ |
| 27 | + Partial<{ |
| 28 | + customGroups: { [key in T[number]]: string[] | string } |
| 29 | + type: 'alphabetical' | 'line-length' | 'natural' |
| 30 | + partitionByComment: string[] | boolean | string |
| 31 | + specialCharacters: 'remove' | 'trim' | 'keep' |
| 32 | + groups: (Group<T>[] | Group<T>)[] |
| 33 | + matcher: 'minimatch' | 'regex' |
| 34 | + sortOnParameters: boolean |
| 35 | + sortOnProperties: boolean |
| 36 | + sortOnAccessors: boolean |
| 37 | + sortOnMethods: boolean |
| 38 | + sortOnClasses: boolean |
| 39 | + order: 'desc' | 'asc' |
| 40 | + ignoreCase: boolean |
| 41 | + }>, |
| 42 | +] |
| 43 | + |
| 44 | +type SortDecoratorsSortingNode = SortingNode<TSESTree.Decorator> |
| 45 | + |
| 46 | +export default createEslintRule<Options<string[]>, MESSAGE_ID>({ |
| 47 | + name: 'sort-decorators', |
| 48 | + meta: { |
| 49 | + type: 'suggestion', |
| 50 | + docs: { |
| 51 | + description: 'Enforce sorted decorators.', |
| 52 | + }, |
| 53 | + fixable: 'code', |
| 54 | + schema: [ |
| 55 | + { |
| 56 | + type: 'object', |
| 57 | + properties: { |
| 58 | + type: { |
| 59 | + description: 'Specifies the sorting method.', |
| 60 | + type: 'string', |
| 61 | + enum: ['alphabetical', 'natural', 'line-length'], |
| 62 | + }, |
| 63 | + order: { |
| 64 | + description: |
| 65 | + 'Determines whether the sorted items should be in ascending or descending order.', |
| 66 | + type: 'string', |
| 67 | + enum: ['asc', 'desc'], |
| 68 | + }, |
| 69 | + matcher: { |
| 70 | + description: 'Specifies the string matcher.', |
| 71 | + type: 'string', |
| 72 | + enum: ['minimatch', 'regex'], |
| 73 | + }, |
| 74 | + ignoreCase: { |
| 75 | + description: |
| 76 | + 'Controls whether sorting should be case-sensitive or not.', |
| 77 | + type: 'boolean', |
| 78 | + }, |
| 79 | + specialCharacters: { |
| 80 | + description: |
| 81 | + 'Controls how special characters should be handled before sorting.', |
| 82 | + type: 'string', |
| 83 | + enum: ['remove', 'trim', 'keep'], |
| 84 | + }, |
| 85 | + sortOnClasses: { |
| 86 | + description: |
| 87 | + 'Controls whether sorting should be enabled for class decorators.', |
| 88 | + type: 'boolean', |
| 89 | + }, |
| 90 | + sortOnMethods: { |
| 91 | + description: |
| 92 | + 'Controls whether sorting should be enabled for class method decorators.', |
| 93 | + type: 'boolean', |
| 94 | + }, |
| 95 | + sortOnParameters: { |
| 96 | + description: |
| 97 | + 'Controls whether sorting should be enabled for method parameter decorators.', |
| 98 | + type: 'boolean', |
| 99 | + }, |
| 100 | + sortOnProperties: { |
| 101 | + description: |
| 102 | + 'Controls whether sorting should be enabled for class property decorators.', |
| 103 | + type: 'boolean', |
| 104 | + }, |
| 105 | + sortOnAccessors: { |
| 106 | + description: |
| 107 | + 'Controls whether sorting should be enabled for class accessor decorators.', |
| 108 | + type: 'boolean', |
| 109 | + }, |
| 110 | + partitionByComment: { |
| 111 | + description: |
| 112 | + 'Allows you to use comments to separate the decorators into logical groups.', |
| 113 | + anyOf: [ |
| 114 | + { |
| 115 | + type: 'boolean', |
| 116 | + }, |
| 117 | + { |
| 118 | + type: 'string', |
| 119 | + }, |
| 120 | + { |
| 121 | + type: 'array', |
| 122 | + items: { |
| 123 | + type: 'string', |
| 124 | + }, |
| 125 | + }, |
| 126 | + ], |
| 127 | + }, |
| 128 | + groups: { |
| 129 | + description: 'Specifies the order of the groups.', |
| 130 | + type: 'array', |
| 131 | + items: { |
| 132 | + oneOf: [ |
| 133 | + { |
| 134 | + type: 'string', |
| 135 | + }, |
| 136 | + { |
| 137 | + type: 'array', |
| 138 | + items: { |
| 139 | + type: 'string', |
| 140 | + }, |
| 141 | + }, |
| 142 | + ], |
| 143 | + }, |
| 144 | + }, |
| 145 | + customGroups: { |
| 146 | + description: 'Specifies custom groups.', |
| 147 | + type: 'object', |
| 148 | + additionalProperties: { |
| 149 | + oneOf: [ |
| 150 | + { |
| 151 | + type: 'string', |
| 152 | + }, |
| 153 | + { |
| 154 | + type: 'array', |
| 155 | + items: { |
| 156 | + type: 'string', |
| 157 | + }, |
| 158 | + }, |
| 159 | + ], |
| 160 | + }, |
| 161 | + }, |
| 162 | + }, |
| 163 | + additionalProperties: false, |
| 164 | + }, |
| 165 | + ], |
| 166 | + messages: { |
| 167 | + unexpectedDecoratorsGroupOrder: |
| 168 | + 'Expected "{{right}}" ({{rightGroup}}) to come before "{{left}}" ({{leftGroup}}).', |
| 169 | + unexpectedDecoratorsOrder: |
| 170 | + 'Expected "{{right}}" to come before "{{left}}".', |
| 171 | + }, |
| 172 | + }, |
| 173 | + defaultOptions: [ |
| 174 | + { |
| 175 | + type: 'alphabetical', |
| 176 | + order: 'asc', |
| 177 | + ignoreCase: true, |
| 178 | + specialCharacters: 'keep', |
| 179 | + partitionByComment: false, |
| 180 | + matcher: 'minimatch', |
| 181 | + groups: [], |
| 182 | + customGroups: {}, |
| 183 | + sortOnClasses: true, |
| 184 | + sortOnMethods: true, |
| 185 | + sortOnAccessors: true, |
| 186 | + sortOnProperties: true, |
| 187 | + sortOnParameters: true, |
| 188 | + }, |
| 189 | + ], |
| 190 | + create: context => { |
| 191 | + let settings = getSettings(context.settings) |
| 192 | + |
| 193 | + let options = complete(context.options.at(0), settings, { |
| 194 | + type: 'alphabetical', |
| 195 | + matcher: 'minimatch', |
| 196 | + ignoreCase: true, |
| 197 | + specialCharacters: 'keep', |
| 198 | + partitionByComment: false, |
| 199 | + customGroups: {}, |
| 200 | + order: 'asc', |
| 201 | + groups: [], |
| 202 | + sortOnClasses: true, |
| 203 | + sortOnMethods: true, |
| 204 | + sortOnAccessors: true, |
| 205 | + sortOnProperties: true, |
| 206 | + sortOnParameters: true, |
| 207 | + } as const) |
| 208 | + |
| 209 | + validateGroupsConfiguration( |
| 210 | + options.groups, |
| 211 | + ['unknown'], |
| 212 | + Object.keys(options.customGroups), |
| 213 | + ) |
| 214 | + |
| 215 | + return { |
| 216 | + ClassDeclaration: Declaration => |
| 217 | + options.sortOnClasses |
| 218 | + ? sortDecorators(context, options, Declaration.decorators) |
| 219 | + : undefined, |
| 220 | + AccessorProperty: accessorDefinition => |
| 221 | + options.sortOnAccessors |
| 222 | + ? sortDecorators(context, options, accessorDefinition.decorators) |
| 223 | + : undefined, |
| 224 | + MethodDefinition: methodDefinition => |
| 225 | + options.sortOnMethods |
| 226 | + ? sortDecorators(context, options, methodDefinition.decorators) |
| 227 | + : undefined, |
| 228 | + PropertyDefinition: propertyDefinition => |
| 229 | + options.sortOnProperties |
| 230 | + ? sortDecorators(context, options, propertyDefinition.decorators) |
| 231 | + : undefined, |
| 232 | + Decorator: decorator => { |
| 233 | + if (!options.sortOnParameters) { |
| 234 | + return |
| 235 | + } |
| 236 | + if ( |
| 237 | + 'decorators' in decorator.parent && |
| 238 | + decorator.parent.type === 'Identifier' && |
| 239 | + decorator.parent.parent.type === 'FunctionExpression' |
| 240 | + ) { |
| 241 | + let { decorators } = decorator.parent |
| 242 | + if (decorator !== decorators[0]) { |
| 243 | + return |
| 244 | + } |
| 245 | + sortDecorators(context, options, decorators) |
| 246 | + } |
| 247 | + }, |
| 248 | + } |
| 249 | + }, |
| 250 | +}) |
| 251 | + |
| 252 | +let sortDecorators = ( |
| 253 | + context: Readonly<RuleContext<MESSAGE_ID, Options<string[]>>>, |
| 254 | + options: Required<Options<string[]>[0]>, |
| 255 | + decorators: TSESTree.Decorator[], |
| 256 | +) => { |
| 257 | + if (decorators.length < 2) { |
| 258 | + return |
| 259 | + } |
| 260 | + let sourceCode = getSourceCode(context) |
| 261 | + let partitionComment = options.partitionByComment |
| 262 | + |
| 263 | + let formattedMembers: SortDecoratorsSortingNode[][] = decorators.reduce( |
| 264 | + (accumulator: SortDecoratorsSortingNode[][], decorator) => { |
| 265 | + if ( |
| 266 | + partitionComment && |
| 267 | + hasPartitionComment( |
| 268 | + partitionComment, |
| 269 | + getCommentsBefore(decorator, sourceCode), |
| 270 | + options.matcher, |
| 271 | + ) |
| 272 | + ) { |
| 273 | + accumulator.push([]) |
| 274 | + } |
| 275 | + |
| 276 | + let { getGroup, setCustomGroups } = useGroups(options) |
| 277 | + let name = getDecoratorName(decorator) |
| 278 | + |
| 279 | + setCustomGroups(options.customGroups, name) |
| 280 | + |
| 281 | + let sortingNode: SortDecoratorsSortingNode = { |
| 282 | + size: rangeToDiff(decorator.range), |
| 283 | + node: decorator, |
| 284 | + group: getGroup(), |
| 285 | + name, |
| 286 | + } |
| 287 | + |
| 288 | + accumulator.at(-1)!.push(sortingNode) |
| 289 | + |
| 290 | + return accumulator |
| 291 | + }, |
| 292 | + [[]], |
| 293 | + ) |
| 294 | + |
| 295 | + let sortedNodes = formattedMembers.flatMap(nodes => |
| 296 | + sortNodesByGroups(nodes, options), |
| 297 | + ) |
| 298 | + |
| 299 | + let nodes = formattedMembers.flat() |
| 300 | + pairwise(nodes, (left, right) => { |
| 301 | + let indexOfLeft = sortedNodes.indexOf(left) |
| 302 | + let indexOfRight = sortedNodes.indexOf(right) |
| 303 | + if (indexOfLeft <= indexOfRight) { |
| 304 | + return |
| 305 | + } |
| 306 | + let leftNum = getGroupNumber(options.groups, left) |
| 307 | + let rightNum = getGroupNumber(options.groups, right) |
| 308 | + context.report({ |
| 309 | + messageId: |
| 310 | + leftNum !== rightNum |
| 311 | + ? 'unexpectedDecoratorsGroupOrder' |
| 312 | + : 'unexpectedDecoratorsOrder', |
| 313 | + data: { |
| 314 | + left: toSingleLine(left.name), |
| 315 | + leftGroup: left.group, |
| 316 | + right: toSingleLine(right.name), |
| 317 | + rightGroup: right.group, |
| 318 | + }, |
| 319 | + node: right.node, |
| 320 | + fix: fixer => makeFixes(fixer, nodes, sortedNodes, sourceCode, options), |
| 321 | + }) |
| 322 | + }) |
| 323 | +} |
0 commit comments