123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369 |
- /**
- * @fileoverview Require `expose` in Vue components
- * @author Yosuke Ota <https://github.com/ota-meshi>
- */
- 'use strict'
- const {
- findVariable,
- isOpeningBraceToken,
- isClosingBraceToken
- } = require('@eslint-community/eslint-utils')
- const utils = require('../utils')
- const { getVueComponentDefinitionType } = require('../utils')
- const FIX_EXPOSE_BEFORE_OPTIONS = new Set([
- 'name',
- 'components',
- 'directives',
- 'extends',
- 'mixins',
- 'provide',
- 'inject',
- 'inheritAttrs',
- 'props',
- 'emits'
- ])
- /**
- * @param {Property | SpreadElement} node
- * @returns {node is ObjectExpressionProperty}
- */
- function isExposeProperty(node) {
- return (
- node.type === 'Property' &&
- utils.getStaticPropertyName(node) === 'expose' &&
- !node.computed
- )
- }
- /**
- * Get the callee member node from the given CallExpression
- * @param {CallExpression} node CallExpression
- */
- function getCalleeMemberNode(node) {
- const callee = utils.skipChainExpression(node.callee)
- if (callee.type === 'MemberExpression') {
- const name = utils.getStaticPropertyName(callee)
- if (name) {
- return { name, member: callee }
- }
- }
- return null
- }
- module.exports = {
- meta: {
- type: 'suggestion',
- docs: {
- description: 'require declare public properties using `expose`',
- categories: undefined,
- url: 'https://eslint.vuejs.org/rules/require-expose.html'
- },
- fixable: null,
- hasSuggestions: true,
- schema: [],
- messages: {
- requireExpose:
- 'The public properties of the component must be explicitly declared using `expose`. If the component does not have public properties, declare it empty.',
- addExposeOptionForEmpty:
- 'Add the `expose` option to give an empty array.',
- addExposeOptionForAll:
- 'Add the `expose` option to declare all properties.'
- }
- },
- /** @param {RuleContext} context */
- create(context) {
- if (utils.isScriptSetup(context)) {
- return {}
- }
- /**
- * @typedef {object} SetupContext
- * @property {Set<Identifier>} exposeReferenceIds
- * @property {Set<Identifier>} contextReferenceIds
- */
- /** @type {Map<ObjectExpression, SetupContext>} */
- const setupContexts = new Map()
- /** @type {Set<ObjectExpression>} */
- const calledExpose = new Set()
- /**
- * @typedef {FunctionExpression | FunctionDeclaration | ArrowFunctionExpression} FunctionNode
- */
- /**
- * @typedef {object} ScopeStack
- * @property {ScopeStack | null} upper
- * @property {FunctionNode} functionNode
- * @property {boolean} returnFunction
- */
- /**
- * @type {ScopeStack | null}
- */
- let scopeStack = null
- /** @type {Map<FunctionNode, ObjectExpression>} */
- const setupFunctions = new Map()
- /** @type {Set<ObjectExpression>} */
- const setupRender = new Set()
- /**
- * @param {Expression} node
- * @returns {boolean}
- */
- function isFunction(node) {
- if (
- node.type === 'ArrowFunctionExpression' ||
- node.type === 'FunctionExpression'
- ) {
- return true
- }
- if (node.type === 'Identifier') {
- const variable = findVariable(context.getScope(), node)
- if (variable) {
- for (const def of variable.defs) {
- if (def.type === 'FunctionName') {
- return true
- }
- if (def.type === 'Variable' && def.node.init) {
- return isFunction(def.node.init)
- }
- }
- }
- }
- return false
- }
- return utils.defineVueVisitor(context, {
- onSetupFunctionEnter(node, { node: vueNode }) {
- setupFunctions.set(node, vueNode)
- const contextParam = node.params[1]
- if (!contextParam) {
- // no arguments
- return
- }
- if (contextParam.type === 'RestElement') {
- // cannot check
- return
- }
- if (contextParam.type === 'ArrayPattern') {
- // cannot check
- return
- }
- /** @type {Set<Identifier>} */
- const contextReferenceIds = new Set()
- /** @type {Set<Identifier>} */
- const exposeReferenceIds = new Set()
- if (contextParam.type === 'ObjectPattern') {
- const exposeProperty = utils.findAssignmentProperty(
- contextParam,
- 'expose'
- )
- if (!exposeProperty) {
- return
- }
- const exposeParam = exposeProperty.value
- // `setup(props, {emit})`
- const variable =
- exposeParam.type === 'Identifier'
- ? findVariable(context.getScope(), exposeParam)
- : null
- if (!variable) {
- return
- }
- for (const reference of variable.references) {
- if (!reference.isRead()) {
- continue
- }
- exposeReferenceIds.add(reference.identifier)
- }
- } else if (contextParam.type === 'Identifier') {
- // `setup(props, context)`
- const variable = findVariable(context.getScope(), contextParam)
- if (!variable) {
- return
- }
- for (const reference of variable.references) {
- if (!reference.isRead()) {
- continue
- }
- contextReferenceIds.add(reference.identifier)
- }
- }
- setupContexts.set(vueNode, {
- contextReferenceIds,
- exposeReferenceIds
- })
- },
- CallExpression(node, { node: vueNode }) {
- if (calledExpose.has(vueNode)) {
- // already called
- return
- }
- // find setup context
- const setupContext = setupContexts.get(vueNode)
- if (setupContext) {
- const { contextReferenceIds, exposeReferenceIds } = setupContext
- if (
- node.callee.type === 'Identifier' &&
- exposeReferenceIds.has(node.callee)
- ) {
- // setup(props,{expose}) {expose()}
- calledExpose.add(vueNode)
- } else {
- const expose = getCalleeMemberNode(node)
- if (
- expose &&
- expose.name === 'expose' &&
- expose.member.object.type === 'Identifier' &&
- contextReferenceIds.has(expose.member.object)
- ) {
- // setup(props,context) {context.emit()}
- calledExpose.add(vueNode)
- }
- }
- }
- },
- /** @param {FunctionNode} node */
- ':function'(node) {
- scopeStack = {
- upper: scopeStack,
- functionNode: node,
- returnFunction: false
- }
- if (
- node.type === 'ArrowFunctionExpression' &&
- node.expression &&
- isFunction(node.body)
- ) {
- scopeStack.returnFunction = true
- }
- },
- ReturnStatement(node) {
- if (!scopeStack) {
- return
- }
- if (
- !scopeStack.returnFunction &&
- node.argument &&
- isFunction(node.argument)
- ) {
- scopeStack.returnFunction = true
- }
- },
- ':function:exit'(node) {
- if (scopeStack && scopeStack.returnFunction) {
- const vueNode = setupFunctions.get(node)
- if (vueNode) {
- setupRender.add(vueNode)
- }
- }
- scopeStack = scopeStack && scopeStack.upper
- },
- onVueObjectExit(component, { type }) {
- if (calledExpose.has(component)) {
- // `expose` function is called
- return
- }
- if (setupRender.has(component)) {
- // `setup` function is render function
- return
- }
- if (type === 'definition') {
- const defType = getVueComponentDefinitionType(component)
- if (defType === 'mixin') {
- return
- }
- }
- if (component.properties.some(isExposeProperty)) {
- // has `expose`
- return
- }
- context.report({
- node: component,
- messageId: 'requireExpose',
- suggest: buildSuggest(component, context)
- })
- }
- })
- }
- }
- /**
- * @param {ObjectExpression} object
- * @param {RuleContext} context
- * @returns {Rule.SuggestionReportDescriptor[]}
- */
- function buildSuggest(object, context) {
- const propertyNodes = object.properties.filter(utils.isProperty)
- const sourceCode = context.getSourceCode()
- const beforeOptionNode = propertyNodes.find((p) =>
- FIX_EXPOSE_BEFORE_OPTIONS.has(utils.getStaticPropertyName(p) || '')
- )
- const allProps = [
- ...new Set(
- utils.iterateProperties(
- object,
- new Set(['props', 'data', 'computed', 'setup', 'methods', 'watch'])
- )
- )
- ]
- return [
- {
- messageId: 'addExposeOptionForEmpty',
- fix: buildFix('expose: []')
- },
- ...(allProps.length > 0
- ? [
- {
- messageId: 'addExposeOptionForAll',
- fix: buildFix(
- `expose: [${allProps
- .map((p) => JSON.stringify(p.name))
- .join(', ')}]`
- )
- }
- ]
- : [])
- ]
- /**
- * @param {string} text
- */
- function buildFix(text) {
- /**
- * @param {RuleFixer} fixer
- */
- return (fixer) => {
- if (beforeOptionNode) {
- return fixer.insertTextAfter(beforeOptionNode, `,\n${text}`)
- } else if (object.properties.length > 0) {
- const after = propertyNodes[0] || object.properties[0]
- return fixer.insertTextAfter(
- sourceCode.getTokenBefore(after),
- `\n${text},`
- )
- } else {
- const objectLeftBrace = /** @type {Token} */ (
- sourceCode.getFirstToken(object, isOpeningBraceToken)
- )
- const objectRightBrace = /** @type {Token} */ (
- sourceCode.getLastToken(object, isClosingBraceToken)
- )
- return fixer.insertTextAfter(
- objectLeftBrace,
- `\n${text}${
- objectLeftBrace.loc.end.line < objectRightBrace.loc.start.line
- ? ''
- : '\n'
- }`
- )
- }
- }
- }
- }
|