125 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			125 lines
		
	
	
		
			2.9 KiB
		
	
	
	
		
			JavaScript
		
	
	
		
			Executable File
		
	
	
	
	
/**
 | 
						|
 * @fileoverview Rule to disallow calls to the `Object` constructor without an argument
 | 
						|
 * @author Francesco Trotta
 | 
						|
 */
 | 
						|
 | 
						|
"use strict";
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// Requirements
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
 | 
						|
const {
 | 
						|
	getVariableByName,
 | 
						|
	isArrowToken,
 | 
						|
	isStartOfExpressionStatement,
 | 
						|
	needsPrecedingSemicolon,
 | 
						|
} = require("./utils/ast-utils");
 | 
						|
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
// Rule Definition
 | 
						|
//------------------------------------------------------------------------------
 | 
						|
 | 
						|
/** @type {import('../types').Rule.RuleModule} */
 | 
						|
module.exports = {
 | 
						|
	meta: {
 | 
						|
		type: "suggestion",
 | 
						|
 | 
						|
		docs: {
 | 
						|
			description:
 | 
						|
				"Disallow calls to the `Object` constructor without an argument",
 | 
						|
			recommended: false,
 | 
						|
			url: "https://eslint.org/docs/latest/rules/no-object-constructor",
 | 
						|
		},
 | 
						|
 | 
						|
		hasSuggestions: true,
 | 
						|
 | 
						|
		schema: [],
 | 
						|
 | 
						|
		messages: {
 | 
						|
			preferLiteral: "The object literal notation {} is preferable.",
 | 
						|
			useLiteral: "Replace with '{{replacement}}'.",
 | 
						|
			useLiteralAfterSemicolon:
 | 
						|
				"Replace with '{{replacement}}', add preceding semicolon.",
 | 
						|
		},
 | 
						|
	},
 | 
						|
 | 
						|
	create(context) {
 | 
						|
		const sourceCode = context.sourceCode;
 | 
						|
 | 
						|
		/**
 | 
						|
		 * Determines whether or not an object literal that replaces a specified node needs to be enclosed in parentheses.
 | 
						|
		 * @param {ASTNode} node The node to be replaced.
 | 
						|
		 * @returns {boolean} Whether or not parentheses around the object literal are required.
 | 
						|
		 */
 | 
						|
		function needsParentheses(node) {
 | 
						|
			if (isStartOfExpressionStatement(node)) {
 | 
						|
				return true;
 | 
						|
			}
 | 
						|
 | 
						|
			const prevToken = sourceCode.getTokenBefore(node);
 | 
						|
 | 
						|
			if (prevToken && isArrowToken(prevToken)) {
 | 
						|
				return true;
 | 
						|
			}
 | 
						|
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		/**
 | 
						|
		 * Reports on nodes where the `Object` constructor is called without arguments.
 | 
						|
		 * @param {ASTNode} node The node to evaluate.
 | 
						|
		 * @returns {void}
 | 
						|
		 */
 | 
						|
		function check(node) {
 | 
						|
			if (
 | 
						|
				node.callee.type !== "Identifier" ||
 | 
						|
				node.callee.name !== "Object" ||
 | 
						|
				node.arguments.length
 | 
						|
			) {
 | 
						|
				return;
 | 
						|
			}
 | 
						|
 | 
						|
			const variable = getVariableByName(
 | 
						|
				sourceCode.getScope(node),
 | 
						|
				"Object",
 | 
						|
			);
 | 
						|
 | 
						|
			if (variable && variable.identifiers.length === 0) {
 | 
						|
				let replacement;
 | 
						|
				let fixText;
 | 
						|
				let messageId = "useLiteral";
 | 
						|
 | 
						|
				if (needsParentheses(node)) {
 | 
						|
					replacement = "({})";
 | 
						|
					if (needsPrecedingSemicolon(sourceCode, node)) {
 | 
						|
						fixText = ";({})";
 | 
						|
						messageId = "useLiteralAfterSemicolon";
 | 
						|
					} else {
 | 
						|
						fixText = "({})";
 | 
						|
					}
 | 
						|
				} else {
 | 
						|
					replacement = fixText = "{}";
 | 
						|
				}
 | 
						|
 | 
						|
				context.report({
 | 
						|
					node,
 | 
						|
					messageId: "preferLiteral",
 | 
						|
					suggest: [
 | 
						|
						{
 | 
						|
							messageId,
 | 
						|
							data: { replacement },
 | 
						|
							fix: fixer => fixer.replaceText(node, fixText),
 | 
						|
						},
 | 
						|
					],
 | 
						|
				});
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		return {
 | 
						|
			CallExpression: check,
 | 
						|
			NewExpression: check,
 | 
						|
		};
 | 
						|
	},
 | 
						|
};
 |