123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 |
- /**
- * @fileoverview Require component name property to match its file name
- * @author Rodrigo Pedra Brum <rodrigo.pedra@gmail.com>
- */
- 'use strict'
- // ------------------------------------------------------------------------------
- // Requirements
- // ------------------------------------------------------------------------------
- const utils = require('../utils')
- const casing = require('../utils/casing')
- const path = require('path')
- // ------------------------------------------------------------------------------
- // Rule Definition
- // ------------------------------------------------------------------------------
- module.exports = {
- meta: {
- type: 'suggestion',
- docs: {
- description: 'require component name property to match its file name',
- categories: undefined,
- url: 'https://eslint.vuejs.org/rules/match-component-file-name.html'
- },
- fixable: null,
- schema: [
- {
- type: 'object',
- properties: {
- extensions: {
- type: 'array',
- items: {
- type: 'string'
- },
- uniqueItems: true,
- additionalItems: false
- },
- shouldMatchCase: {
- type: 'boolean'
- }
- },
- additionalProperties: false
- }
- ]
- },
- /** @param {RuleContext} context */
- create(context) {
- const options = context.options[0]
- const shouldMatchCase = (options && options.shouldMatchCase) || false
- const extensionsArray = options && options.extensions
- const allowedExtensions = Array.isArray(extensionsArray)
- ? extensionsArray
- : ['jsx']
- const extension = path.extname(context.getFilename())
- const filename = path.basename(context.getFilename(), extension)
- /** @type {Rule.ReportDescriptor[]} */
- const errors = []
- let componentCount = 0
- if (!allowedExtensions.includes(extension.replace(/^\./, ''))) {
- return {}
- }
- // ----------------------------------------------------------------------
- // Private
- // ----------------------------------------------------------------------
- /**
- * @param {string} name
- * @param {string} filename
- */
- function compareNames(name, filename) {
- if (shouldMatchCase) {
- return name === filename
- }
- return (
- casing.pascalCase(name) === filename ||
- casing.kebabCase(name) === filename
- )
- }
- /**
- * @param {Literal | TemplateLiteral} node
- */
- function verifyName(node) {
- let name
- if (node.type === 'TemplateLiteral') {
- const quasis = node.quasis[0]
- name = quasis.value.cooked
- } else {
- name = `${node.value}`
- }
- if (!compareNames(name, filename)) {
- errors.push({
- node,
- message:
- 'Component name `{{name}}` should match file name `{{filename}}`.',
- data: { filename, name }
- })
- }
- }
- /**
- * @param {Expression | SpreadElement} node
- * @returns {node is (Literal | TemplateLiteral)}
- */
- function canVerify(node) {
- return (
- node.type === 'Literal' ||
- (node.type === 'TemplateLiteral' &&
- node.expressions.length === 0 &&
- node.quasis.length === 1)
- )
- }
- return Object.assign(
- {},
- utils.executeOnCallVueComponent(context, (node) => {
- if (node.arguments.length === 2) {
- const argument = node.arguments[0]
- if (canVerify(argument)) {
- verifyName(argument)
- }
- }
- }),
- utils.executeOnVue(context, (object) => {
- const node = utils.findProperty(object, 'name')
- componentCount++
- if (!node) return
- if (!canVerify(node.value)) return
- verifyName(node.value)
- }),
- {
- 'Program:exit'() {
- if (componentCount > 1) return
- errors.forEach((error) => context.report(error))
- }
- }
- )
- }
- }
|