From 9fe8b34105076bb8c02b23f049796e840de48d3c Mon Sep 17 00:00:00 2001 From: Yosuke Ota Date: Thu, 30 Nov 2023 18:01:39 +0900 Subject: [PATCH] Fix crash for TSFunctionType with `type-literal` option in `vue/define-emits-declaration` rule (#2336) --- lib/rules/define-emits-declaration.js | 51 ++++++++++++------- tests/lib/rules/define-emits-declaration.js | 19 +++++++ .../util-types/ast/es-ast.ts | 4 +- 3 files changed, 53 insertions(+), 21 deletions(-) diff --git a/lib/rules/define-emits-declaration.js b/lib/rules/define-emits-declaration.js index 3d12e0bc3..e3de57e4f 100644 --- a/lib/rules/define-emits-declaration.js +++ b/lib/rules/define-emits-declaration.js @@ -7,7 +7,7 @@ const utils = require('../utils') /** - * @typedef {import('@typescript-eslint/types').TSESTree.TSTypeLiteral} TSTypeLiteral + * @typedef {import('@typescript-eslint/types').TSESTree.TypeNode} TypeNode * */ @@ -54,24 +54,7 @@ module.exports = { } case 'type-literal': { - if (node.arguments.length > 0) { - context.report({ - node, - messageId: 'hasArg' - }) - return - } - - const typeArguments = node.typeArguments || node.typeParameters - const param = /** @type {TSTypeLiteral} */ (typeArguments.params[0]) - for (const memberNode of param.members) { - if (memberNode.type !== 'TSPropertySignature') { - context.report({ - node: memberNode, - messageId: 'hasTypeCallArg' - }) - } - } + verifyTypeLiteral(node) break } @@ -89,5 +72,35 @@ module.exports = { } } }) + + /** @param {CallExpression} node */ + function verifyTypeLiteral(node) { + if (node.arguments.length > 0) { + context.report({ + node, + messageId: 'hasArg' + }) + return + } + + const typeArguments = node.typeArguments || node.typeParameters + const param = /** @type {TypeNode|undefined} */ (typeArguments?.params[0]) + if (!param) return + if (param.type === 'TSTypeLiteral') { + for (const memberNode of param.members) { + if (memberNode.type !== 'TSPropertySignature') { + context.report({ + node: memberNode, + messageId: 'hasTypeCallArg' + }) + } + } + } else if (param.type === 'TSFunctionType') { + context.report({ + node: param, + messageId: 'hasTypeCallArg' + }) + } + } } } diff --git a/tests/lib/rules/define-emits-declaration.js b/tests/lib/rules/define-emits-declaration.js index 43ecb385b..6e42d1f4a 100644 --- a/tests/lib/rules/define-emits-declaration.js +++ b/tests/lib/rules/define-emits-declaration.js @@ -243,6 +243,25 @@ tester.run('define-emits-declaration', rule, { line: 5 } ] + }, + { + filename: 'test.vue', + code: ` + + `, + options: ['type-literal'], + parserOptions: { + parser: require.resolve('@typescript-eslint/parser') + }, + errors: [ + { + message: + 'Use new type literal declaration instead of the old call signature declaration.', + line: 3 + } + ] } ] }) diff --git a/typings/eslint-plugin-vue/util-types/ast/es-ast.ts b/typings/eslint-plugin-vue/util-types/ast/es-ast.ts index db3a7f39c..fc3676a56 100644 --- a/typings/eslint-plugin-vue/util-types/ast/es-ast.ts +++ b/typings/eslint-plugin-vue/util-types/ast/es-ast.ts @@ -522,7 +522,7 @@ export interface CallExpression extends HasParentNode { typeArguments?: TS.TSTypeParameterInstantiation /* @deprecated */ - typeParameters: never + typeParameters?: never } export interface Super extends HasParentNode { type: 'Super' @@ -534,7 +534,7 @@ export interface NewExpression extends HasParentNode { typeArguments?: TSTypeParameterInstantiation /* @deprecated */ - typeParameters: never + typeParameters?: never } interface BaseMemberExpression extends HasParentNode { type: 'MemberExpression'