123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217 |
- /**
- * @fileoverview Don't introduce side effects in computed properties
- * @author Michał Sajnóg
- */
- 'use strict'
- const {
- ReferenceTracker,
- findVariable
- } = require('@eslint-community/eslint-utils')
- const utils = require('../utils')
- /**
- * @typedef {import('../utils').VueObjectData} VueObjectData
- * @typedef {import('../utils').VueVisitor} VueVisitor
- * @typedef {import('../utils').ComponentComputedProperty} ComponentComputedProperty
- */
- module.exports = {
- meta: {
- type: 'problem',
- docs: {
- description: 'disallow side effects in computed properties',
- categories: ['vue3-essential', 'essential'],
- url: 'https://eslint.vuejs.org/rules/no-side-effects-in-computed-properties.html'
- },
- fixable: null,
- schema: [],
- messages: {
- unexpectedSideEffectInFunction:
- 'Unexpected side effect in computed function.',
- unexpectedSideEffectInProperty:
- 'Unexpected side effect in "{{key}}" computed property.'
- }
- },
- /** @param {RuleContext} context */
- create(context) {
- /** @type {Map<ObjectExpression, ComponentComputedProperty[]>} */
- const computedPropertiesMap = new Map()
- /** @type {Array<FunctionExpression | ArrowFunctionExpression>} */
- const computedCallNodes = []
- /** @type {[number, number][]} */
- const setupRanges = []
- /**
- * @typedef {object} ScopeStack
- * @property {ScopeStack | null} upper
- * @property {BlockStatement | Expression | null} body
- */
- /**
- * @type {ScopeStack | null}
- */
- let scopeStack = null
- /** @param {FunctionExpression | ArrowFunctionExpression | FunctionDeclaration} node */
- function onFunctionEnter(node) {
- scopeStack = {
- upper: scopeStack,
- body: node.body
- }
- }
- function onFunctionExit() {
- scopeStack = scopeStack && scopeStack.upper
- }
- const nodeVisitor = {
- ':function': onFunctionEnter,
- ':function:exit': onFunctionExit,
- /**
- * @param {(Identifier | ThisExpression) & {parent: MemberExpression}} node
- * @param {VueObjectData|undefined} [info]
- */
- 'MemberExpression > :matches(Identifier, ThisExpression)'(node, info) {
- if (!scopeStack) {
- return
- }
- const targetBody = scopeStack.body
- const computedProperty = (
- info ? computedPropertiesMap.get(info.node) || [] : []
- ).find(
- (cp) =>
- cp.value &&
- cp.value.range[0] <= node.range[0] &&
- node.range[1] <= cp.value.range[1] &&
- targetBody === cp.value
- )
- if (computedProperty) {
- const mem = node.parent
- if (mem.object !== node) {
- return
- }
- const isThis = utils.isThis(node, context)
- const isVue = node.type === 'Identifier' && node.name === 'Vue'
- const isVueSet =
- mem.parent.type === 'CallExpression' &&
- mem.property.type === 'Identifier' &&
- ((isThis && mem.property.name === '$set') ||
- (isVue && mem.property.name === 'set'))
- const invalid = isVueSet
- ? { node: mem.property }
- : isThis && utils.findMutating(mem)
- if (invalid) {
- context.report({
- node: invalid.node,
- messageId: 'unexpectedSideEffectInProperty',
- data: { key: computedProperty.key || 'Unknown' }
- })
- }
- return
- }
- // ignore `this` for computed functions
- if (node.type === 'ThisExpression') {
- return
- }
- const computedFunction = computedCallNodes.find(
- (c) =>
- c.range[0] <= node.range[0] &&
- node.range[1] <= c.range[1] &&
- targetBody === c.body
- )
- if (!computedFunction) {
- return
- }
- const mem = node.parent
- if (mem.object !== node) {
- return
- }
- const variable = findVariable(context.getScope(), node)
- if (!variable || variable.defs.length !== 1) {
- return
- }
- const def = variable.defs[0]
- if (
- def.type === 'ImplicitGlobalVariable' ||
- def.type === 'TDZ' ||
- def.type === 'ImportBinding'
- ) {
- return
- }
- const isDeclaredInsideSetup = setupRanges.some(
- ([start, end]) =>
- start <= def.node.range[0] && def.node.range[1] <= end
- )
- if (!isDeclaredInsideSetup) {
- return
- }
- if (
- computedFunction.range[0] <= def.node.range[0] &&
- def.node.range[1] <= computedFunction.range[1]
- ) {
- // mutating local variables are accepted
- return
- }
- const invalid = utils.findMutating(node)
- if (invalid) {
- context.report({
- node: invalid.node,
- messageId: 'unexpectedSideEffectInFunction'
- })
- }
- }
- }
- const scriptSetupNode = utils.getScriptSetupElement(context)
- if (scriptSetupNode) {
- setupRanges.push(scriptSetupNode.range)
- }
- return utils.compositingVisitors(
- {
- Program() {
- const tracker = new ReferenceTracker(context.getScope())
- const traceMap = utils.createCompositionApiTraceMap({
- [ReferenceTracker.ESM]: true,
- computed: {
- [ReferenceTracker.CALL]: true
- }
- })
- for (const { node } of tracker.iterateEsmReferences(traceMap)) {
- if (node.type !== 'CallExpression') {
- continue
- }
- const getterBody = utils.getGetterBodyFromComputedFunction(node)
- if (getterBody) {
- computedCallNodes.push(getterBody)
- }
- }
- }
- },
- scriptSetupNode
- ? utils.defineScriptSetupVisitor(context, nodeVisitor)
- : utils.defineVueVisitor(context, {
- onVueObjectEnter(node) {
- computedPropertiesMap.set(node, utils.getComputedProperties(node))
- },
- onSetupFunctionEnter(node) {
- setupRanges.push(node.body.range)
- },
- ...nodeVisitor
- })
- )
- }
- }
|