no-expose-after-await.js 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. /**
  2. * @author Yosuke Ota
  3. * See LICENSE file in root directory for full license.
  4. */
  5. 'use strict'
  6. const { findVariable } = require('@eslint-community/eslint-utils')
  7. const utils = require('../utils')
  8. /**
  9. * Get the callee member node from the given CallExpression
  10. * @param {CallExpression} node CallExpression
  11. */
  12. function getCalleeMemberNode(node) {
  13. const callee = utils.skipChainExpression(node.callee)
  14. if (callee.type === 'MemberExpression') {
  15. const name = utils.getStaticPropertyName(callee)
  16. if (name) {
  17. return { name, member: callee }
  18. }
  19. }
  20. return null
  21. }
  22. module.exports = {
  23. meta: {
  24. type: 'problem',
  25. docs: {
  26. description: 'disallow asynchronously registered `expose`',
  27. categories: ['vue3-essential'],
  28. url: 'https://eslint.vuejs.org/rules/no-expose-after-await.html'
  29. },
  30. fixable: null,
  31. schema: [],
  32. messages: {
  33. forbidden: '`{{name}}` is forbidden after an `await` expression.'
  34. }
  35. },
  36. /** @param {RuleContext} context */
  37. create(context) {
  38. /**
  39. * @typedef {object} SetupScopeData
  40. * @property {boolean} afterAwait
  41. * @property {[number,number]} range
  42. * @property {(node: Identifier, callNode: CallExpression) => boolean} isExposeReferenceId
  43. * @property {(node: Identifier) => boolean} isContextReferenceId
  44. */
  45. /**
  46. * @typedef {object} ScopeStack
  47. * @property {ScopeStack | null} upper
  48. * @property {FunctionDeclaration | FunctionExpression | ArrowFunctionExpression | Program} scopeNode
  49. */
  50. /** @type {Map<FunctionDeclaration | FunctionExpression | ArrowFunctionExpression | Program, SetupScopeData>} */
  51. const setupScopes = new Map()
  52. /** @type {ScopeStack | null} */
  53. let scopeStack = null
  54. return utils.compositingVisitors(
  55. {
  56. /**
  57. * @param {Program} node
  58. */
  59. Program(node) {
  60. scopeStack = {
  61. upper: scopeStack,
  62. scopeNode: node
  63. }
  64. }
  65. },
  66. {
  67. /**
  68. * @param {FunctionExpression | FunctionDeclaration | ArrowFunctionExpression} node
  69. */
  70. ':function'(node) {
  71. scopeStack = {
  72. upper: scopeStack,
  73. scopeNode: node
  74. }
  75. },
  76. ':function:exit'() {
  77. scopeStack = scopeStack && scopeStack.upper
  78. },
  79. /** @param {AwaitExpression} node */
  80. AwaitExpression(node) {
  81. if (!scopeStack) {
  82. return
  83. }
  84. const setupScope = setupScopes.get(scopeStack.scopeNode)
  85. if (!setupScope || !utils.inRange(setupScope.range, node)) {
  86. return
  87. }
  88. setupScope.afterAwait = true
  89. },
  90. /** @param {CallExpression} node */
  91. CallExpression(node) {
  92. if (!scopeStack) {
  93. return
  94. }
  95. const setupScope = setupScopes.get(scopeStack.scopeNode)
  96. if (
  97. !setupScope ||
  98. !setupScope.afterAwait ||
  99. !utils.inRange(setupScope.range, node)
  100. ) {
  101. return
  102. }
  103. const { isContextReferenceId, isExposeReferenceId } = setupScope
  104. if (
  105. node.callee.type === 'Identifier' &&
  106. isExposeReferenceId(node.callee, node)
  107. ) {
  108. // setup(props,{expose}) {expose()}
  109. context.report({
  110. node,
  111. messageId: 'forbidden',
  112. data: {
  113. name: node.callee.name
  114. }
  115. })
  116. } else {
  117. const expose = getCalleeMemberNode(node)
  118. if (
  119. expose &&
  120. expose.name === 'expose' &&
  121. expose.member.object.type === 'Identifier' &&
  122. isContextReferenceId(expose.member.object)
  123. ) {
  124. // setup(props,context) {context.emit()}
  125. context.report({
  126. node,
  127. messageId: 'forbidden',
  128. data: {
  129. name: expose.name
  130. }
  131. })
  132. }
  133. }
  134. }
  135. },
  136. (() => {
  137. const scriptSetup = utils.getScriptSetupElement(context)
  138. if (!scriptSetup) {
  139. return {}
  140. }
  141. return {
  142. /**
  143. * @param {Program} node
  144. */
  145. Program(node) {
  146. setupScopes.set(node, {
  147. afterAwait: false,
  148. range: scriptSetup.range,
  149. isExposeReferenceId: (id, callNode) =>
  150. callNode.parent.type === 'ExpressionStatement' &&
  151. callNode.parent.parent === node &&
  152. id.name === 'defineExpose',
  153. isContextReferenceId: () => false
  154. })
  155. }
  156. }
  157. })(),
  158. utils.defineVueVisitor(context, {
  159. onSetupFunctionEnter(node) {
  160. const contextParam = node.params[1]
  161. if (!contextParam) {
  162. // no arguments
  163. return
  164. }
  165. if (contextParam.type === 'RestElement') {
  166. // cannot check
  167. return
  168. }
  169. if (contextParam.type === 'ArrayPattern') {
  170. // cannot check
  171. return
  172. }
  173. /** @type {Set<Identifier>} */
  174. const contextReferenceIds = new Set()
  175. /** @type {Set<Identifier>} */
  176. const exposeReferenceIds = new Set()
  177. if (contextParam.type === 'ObjectPattern') {
  178. const exposeProperty = utils.findAssignmentProperty(
  179. contextParam,
  180. 'expose'
  181. )
  182. if (!exposeProperty) {
  183. return
  184. }
  185. const exposeParam = exposeProperty.value
  186. // `setup(props, {emit})`
  187. const variable =
  188. exposeParam.type === 'Identifier'
  189. ? findVariable(context.getScope(), exposeParam)
  190. : null
  191. if (!variable) {
  192. return
  193. }
  194. for (const reference of variable.references) {
  195. if (!reference.isRead()) {
  196. continue
  197. }
  198. exposeReferenceIds.add(reference.identifier)
  199. }
  200. } else if (contextParam.type === 'Identifier') {
  201. // `setup(props, context)`
  202. const variable = findVariable(context.getScope(), contextParam)
  203. if (!variable) {
  204. return
  205. }
  206. for (const reference of variable.references) {
  207. if (!reference.isRead()) {
  208. continue
  209. }
  210. contextReferenceIds.add(reference.identifier)
  211. }
  212. }
  213. setupScopes.set(node, {
  214. afterAwait: false,
  215. range: node.range,
  216. isExposeReferenceId: (id) => exposeReferenceIds.has(id),
  217. isContextReferenceId: (id) => contextReferenceIds.has(id)
  218. })
  219. },
  220. onSetupFunctionExit(node) {
  221. setupScopes.delete(node)
  222. }
  223. })
  224. )
  225. }
  226. }