no-restricted-block.js 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. /**
  2. * @author Yosuke Ota
  3. * See LICENSE file in root directory for full license.
  4. */
  5. 'use strict'
  6. const utils = require('../utils')
  7. const regexp = require('../utils/regexp')
  8. /**
  9. * @typedef {object} ParsedOption
  10. * @property { (block: VElement) => boolean } test
  11. * @property {string} [message]
  12. */
  13. /**
  14. * @param {string} str
  15. * @returns {(str: string) => boolean}
  16. */
  17. function buildMatcher(str) {
  18. if (regexp.isRegExp(str)) {
  19. const re = regexp.toRegExp(str)
  20. return (s) => {
  21. re.lastIndex = 0
  22. return re.test(s)
  23. }
  24. }
  25. return (s) => s === str
  26. }
  27. /**
  28. * @param {any} option
  29. * @returns {ParsedOption}
  30. */
  31. function parseOption(option) {
  32. if (typeof option === 'string') {
  33. const matcher = buildMatcher(option)
  34. return {
  35. test(block) {
  36. return matcher(block.rawName)
  37. }
  38. }
  39. }
  40. const parsed = parseOption(option.element)
  41. parsed.message = option.message
  42. return parsed
  43. }
  44. /**
  45. * @param {VElement} block
  46. */
  47. function defaultMessage(block) {
  48. return `Using \`<${block.rawName}>\` is not allowed.`
  49. }
  50. module.exports = {
  51. meta: {
  52. type: 'suggestion',
  53. docs: {
  54. description: 'disallow specific block',
  55. categories: undefined,
  56. url: 'https://eslint.vuejs.org/rules/no-restricted-block.html'
  57. },
  58. fixable: null,
  59. schema: {
  60. type: 'array',
  61. items: {
  62. oneOf: [
  63. { type: 'string' },
  64. {
  65. type: 'object',
  66. properties: {
  67. element: { type: 'string' },
  68. message: { type: 'string', minLength: 1 }
  69. },
  70. required: ['element'],
  71. additionalProperties: false
  72. }
  73. ]
  74. },
  75. uniqueItems: true,
  76. minItems: 0
  77. },
  78. messages: {
  79. // eslint-disable-next-line eslint-plugin/report-message-format
  80. restrictedBlock: '{{message}}'
  81. }
  82. },
  83. /** @param {RuleContext} context */
  84. create(context) {
  85. /** @type {ParsedOption[]} */
  86. const options = context.options.map(parseOption)
  87. const documentFragment =
  88. context.parserServices.getDocumentFragment &&
  89. context.parserServices.getDocumentFragment()
  90. function getTopLevelHTMLElements() {
  91. if (documentFragment) {
  92. return documentFragment.children.filter(utils.isVElement)
  93. }
  94. return []
  95. }
  96. return {
  97. /** @param {Program} node */
  98. Program(node) {
  99. if (utils.hasInvalidEOF(node)) {
  100. return
  101. }
  102. for (const block of getTopLevelHTMLElements()) {
  103. for (const option of options) {
  104. if (option.test(block)) {
  105. const message = option.message || defaultMessage(block)
  106. context.report({
  107. node: block.startTag,
  108. messageId: 'restrictedBlock',
  109. data: { message }
  110. })
  111. break
  112. }
  113. }
  114. }
  115. }
  116. }
  117. }
  118. }