mirror of https://github.com/lukechilds/node.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
520 lines
18 KiB
520 lines
18 KiB
/**
|
|
* @fileoverview Rule to enforce spacing before and after keywords.
|
|
* @author Toru Nagashima
|
|
* @copyright 2015 Toru Nagashima. All rights reserved.
|
|
* See LICENSE file in root directory for full license.
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
//------------------------------------------------------------------------------
|
|
// Requirements
|
|
//------------------------------------------------------------------------------
|
|
|
|
var astUtils = require("../ast-utils"),
|
|
keywords = require("../util/keywords");
|
|
|
|
//------------------------------------------------------------------------------
|
|
// Constants
|
|
//------------------------------------------------------------------------------
|
|
|
|
var PREV_TOKEN = /^[\)\]\}>]$/;
|
|
var NEXT_TOKEN = /^(?:[\(\[\{<~!]|\+\+?|--?)$/;
|
|
var PREV_TOKEN_M = /^[\)\]\}>*]$/;
|
|
var NEXT_TOKEN_M = /^[\{*]$/;
|
|
var TEMPLATE_OPEN_PAREN = /\$\{$/;
|
|
var TEMPLATE_CLOSE_PAREN = /^\}/;
|
|
var CHECK_TYPE = /^(?:JSXElement|RegularExpression|String|Template)$/;
|
|
var KEYS = keywords.concat(["as", "await", "from", "get", "let", "of", "set", "yield"]);
|
|
|
|
// check duplications.
|
|
(function() {
|
|
KEYS.sort();
|
|
for (var i = 1; i < KEYS.length; ++i) {
|
|
if (KEYS[i] === KEYS[i - 1]) {
|
|
throw new Error("Duplication was found in the keyword list: " + KEYS[i]);
|
|
}
|
|
}
|
|
}());
|
|
|
|
//------------------------------------------------------------------------------
|
|
// Helpers
|
|
//------------------------------------------------------------------------------
|
|
|
|
/**
|
|
* Checks whether or not a given token is a "Template" token ends with "${".
|
|
*
|
|
* @param {Token} token - A token to check.
|
|
* @returns {boolean} `true` if the token is a "Template" token ends with "${".
|
|
*/
|
|
function isOpenParenOfTemplate(token) {
|
|
return token.type === "Template" && TEMPLATE_OPEN_PAREN.test(token.value);
|
|
}
|
|
|
|
/**
|
|
* Checks whether or not a given token is a "Template" token starts with "}".
|
|
*
|
|
* @param {Token} token - A token to check.
|
|
* @returns {boolean} `true` if the token is a "Template" token starts with "}".
|
|
*/
|
|
function isCloseParenOfTemplate(token) {
|
|
return token.type === "Template" && TEMPLATE_CLOSE_PAREN.test(token.value);
|
|
}
|
|
|
|
//------------------------------------------------------------------------------
|
|
// Rule Definition
|
|
//------------------------------------------------------------------------------
|
|
|
|
module.exports = function(context) {
|
|
var sourceCode = context.getSourceCode();
|
|
|
|
/**
|
|
* Reports a given token if there are not space(s) before the token.
|
|
*
|
|
* @param {Token} token - A token to report.
|
|
* @param {RegExp|undefined} pattern - Optional. A pattern of the previous
|
|
* token to check.
|
|
* @returns {void}
|
|
*/
|
|
function expectSpaceBefore(token, pattern) {
|
|
pattern = pattern || PREV_TOKEN;
|
|
|
|
var prevToken = sourceCode.getTokenBefore(token);
|
|
if (prevToken &&
|
|
(CHECK_TYPE.test(prevToken.type) || pattern.test(prevToken.value)) &&
|
|
!isOpenParenOfTemplate(prevToken) &&
|
|
astUtils.isTokenOnSameLine(prevToken, token) &&
|
|
!sourceCode.isSpaceBetweenTokens(prevToken, token)
|
|
) {
|
|
context.report({
|
|
loc: token.loc.start,
|
|
message: "Expected space(s) before \"{{value}}\".",
|
|
data: token,
|
|
fix: function(fixer) {
|
|
return fixer.insertTextBefore(token, " ");
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reports a given token if there are space(s) before the token.
|
|
*
|
|
* @param {Token} token - A token to report.
|
|
* @param {RegExp|undefined} pattern - Optional. A pattern of the previous
|
|
* token to check.
|
|
* @returns {void}
|
|
*/
|
|
function unexpectSpaceBefore(token, pattern) {
|
|
pattern = pattern || PREV_TOKEN;
|
|
|
|
var prevToken = sourceCode.getTokenBefore(token);
|
|
if (prevToken &&
|
|
(CHECK_TYPE.test(prevToken.type) || pattern.test(prevToken.value)) &&
|
|
!isOpenParenOfTemplate(prevToken) &&
|
|
astUtils.isTokenOnSameLine(prevToken, token) &&
|
|
sourceCode.isSpaceBetweenTokens(prevToken, token)
|
|
) {
|
|
context.report({
|
|
loc: token.loc.start,
|
|
message: "Unexpected space(s) before \"{{value}}\".",
|
|
data: token,
|
|
fix: function(fixer) {
|
|
return fixer.removeRange([prevToken.range[1], token.range[0]]);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reports a given token if there are not space(s) after the token.
|
|
*
|
|
* @param {Token} token - A token to report.
|
|
* @param {RegExp|undefined} pattern - Optional. A pattern of the next
|
|
* token to check.
|
|
* @returns {void}
|
|
*/
|
|
function expectSpaceAfter(token, pattern) {
|
|
pattern = pattern || NEXT_TOKEN;
|
|
|
|
var nextToken = sourceCode.getTokenAfter(token);
|
|
if (nextToken &&
|
|
(CHECK_TYPE.test(nextToken.type) || pattern.test(nextToken.value)) &&
|
|
!isCloseParenOfTemplate(nextToken) &&
|
|
astUtils.isTokenOnSameLine(token, nextToken) &&
|
|
!sourceCode.isSpaceBetweenTokens(token, nextToken)
|
|
) {
|
|
context.report({
|
|
loc: token.loc.start,
|
|
message: "Expected space(s) after \"{{value}}\".",
|
|
data: token,
|
|
fix: function(fixer) {
|
|
return fixer.insertTextAfter(token, " ");
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reports a given token if there are space(s) after the token.
|
|
*
|
|
* @param {Token} token - A token to report.
|
|
* @param {RegExp|undefined} pattern - Optional. A pattern of the next
|
|
* token to check.
|
|
* @returns {void}
|
|
*/
|
|
function unexpectSpaceAfter(token, pattern) {
|
|
pattern = pattern || NEXT_TOKEN;
|
|
|
|
var nextToken = sourceCode.getTokenAfter(token);
|
|
if (nextToken &&
|
|
(CHECK_TYPE.test(nextToken.type) || pattern.test(nextToken.value)) &&
|
|
!isCloseParenOfTemplate(nextToken) &&
|
|
astUtils.isTokenOnSameLine(token, nextToken) &&
|
|
sourceCode.isSpaceBetweenTokens(token, nextToken)
|
|
) {
|
|
context.report({
|
|
loc: token.loc.start,
|
|
message: "Unexpected space(s) after \"{{value}}\".",
|
|
data: token,
|
|
fix: function(fixer) {
|
|
return fixer.removeRange([token.range[1], nextToken.range[0]]);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parses the option object and determines check methods for each keyword.
|
|
*
|
|
* @param {object|undefined} options - The option object to parse.
|
|
* @returns {object} - Normalized option object.
|
|
* Keys are keywords (there are for every keyword).
|
|
* Values are instances of `{"before": function, "after": function}`.
|
|
*/
|
|
function parseOptions(options) {
|
|
var before = !options || options.before !== false;
|
|
var after = !options || options.after !== false;
|
|
var defaultValue = {
|
|
before: before ? expectSpaceBefore : unexpectSpaceBefore,
|
|
after: after ? expectSpaceAfter : unexpectSpaceAfter
|
|
};
|
|
var overrides = (options && options.overrides) || {};
|
|
var retv = Object.create(null);
|
|
|
|
for (var i = 0; i < KEYS.length; ++i) {
|
|
var key = KEYS[i];
|
|
var override = overrides[key];
|
|
|
|
if (override) {
|
|
var thisBefore = ("before" in override) ? override.before : before;
|
|
var thisAfter = ("after" in override) ? override.after : after;
|
|
retv[key] = {
|
|
before: thisBefore ? expectSpaceBefore : unexpectSpaceBefore,
|
|
after: thisAfter ? expectSpaceAfter : unexpectSpaceAfter
|
|
};
|
|
} else {
|
|
retv[key] = defaultValue;
|
|
}
|
|
}
|
|
|
|
return retv;
|
|
}
|
|
|
|
var checkMethodMap = parseOptions(context.options[0]);
|
|
|
|
/**
|
|
* Reports a given token if usage of spacing followed by the token is
|
|
* invalid.
|
|
*
|
|
* @param {Token} token - A token to report.
|
|
* @param {RegExp|undefined} pattern - Optional. A pattern of the previous
|
|
* token to check.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingBefore(token, pattern) {
|
|
checkMethodMap[token.value].before(token, pattern);
|
|
}
|
|
|
|
/**
|
|
* Reports a given token if usage of spacing preceded by the token is
|
|
* invalid.
|
|
*
|
|
* @param {Token} token - A token to report.
|
|
* @param {RegExp|undefined} pattern - Optional. A pattern of the next
|
|
* token to check.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingAfter(token, pattern) {
|
|
checkMethodMap[token.value].after(token, pattern);
|
|
}
|
|
|
|
/**
|
|
* Reports a given token if usage of spacing around the token is invalid.
|
|
*
|
|
* @param {Token} token - A token to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingAround(token) {
|
|
checkSpacingBefore(token);
|
|
checkSpacingAfter(token);
|
|
}
|
|
|
|
/**
|
|
* Reports the first token of a given node if the first token is a keyword
|
|
* and usage of spacing around the token is invalid.
|
|
*
|
|
* @param {ASTNode|null} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingAroundFirstToken(node) {
|
|
var firstToken = node && sourceCode.getFirstToken(node);
|
|
if (firstToken && firstToken.type === "Keyword") {
|
|
checkSpacingAround(firstToken);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reports the first token of a given node if the first token is a keyword
|
|
* and usage of spacing followed by the token is invalid.
|
|
*
|
|
* This is used for unary operators (e.g. `typeof`), `function`, and `super`.
|
|
* Other rules are handling usage of spacing preceded by those keywords.
|
|
*
|
|
* @param {ASTNode|null} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingBeforeFirstToken(node) {
|
|
var firstToken = node && sourceCode.getFirstToken(node);
|
|
if (firstToken && firstToken.type === "Keyword") {
|
|
checkSpacingBefore(firstToken);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reports the previous token of a given node if the token is a keyword and
|
|
* usage of spacing around the token is invalid.
|
|
*
|
|
* @param {ASTNode|null} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingAroundTokenBefore(node) {
|
|
if (node) {
|
|
var token = sourceCode.getTokenBefore(node);
|
|
while (token.type !== "Keyword") {
|
|
token = sourceCode.getTokenBefore(token);
|
|
}
|
|
|
|
checkSpacingAround(token);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reports `class` and `extends` keywords of a given node if usage of
|
|
* spacing around those keywords is invalid.
|
|
*
|
|
* @param {ASTNode} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingForClass(node) {
|
|
checkSpacingAroundFirstToken(node);
|
|
checkSpacingAroundTokenBefore(node.superClass);
|
|
}
|
|
|
|
/**
|
|
* Reports `if` and `else` keywords of a given node if usage of spacing
|
|
* around those keywords is invalid.
|
|
*
|
|
* @param {ASTNode} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingForIfStatement(node) {
|
|
checkSpacingAroundFirstToken(node);
|
|
checkSpacingAroundTokenBefore(node.alternate);
|
|
}
|
|
|
|
/**
|
|
* Reports `try`, `catch`, and `finally` keywords of a given node if usage
|
|
* of spacing around those keywords is invalid.
|
|
*
|
|
* @param {ASTNode} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingForTryStatement(node) {
|
|
checkSpacingAroundFirstToken(node);
|
|
checkSpacingAroundFirstToken(node.handler);
|
|
checkSpacingAroundTokenBefore(node.finalizer);
|
|
}
|
|
|
|
/**
|
|
* Reports `do` and `while` keywords of a given node if usage of spacing
|
|
* around those keywords is invalid.
|
|
*
|
|
* @param {ASTNode} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingForDoWhileStatement(node) {
|
|
checkSpacingAroundFirstToken(node);
|
|
checkSpacingAroundTokenBefore(node.test);
|
|
}
|
|
|
|
/**
|
|
* Reports `for` and `in` keywords of a given node if usage of spacing
|
|
* around those keywords is invalid.
|
|
*
|
|
* @param {ASTNode} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingForForInStatement(node) {
|
|
checkSpacingAroundFirstToken(node);
|
|
checkSpacingAroundTokenBefore(node.right);
|
|
}
|
|
|
|
/**
|
|
* Reports `for` and `of` keywords of a given node if usage of spacing
|
|
* around those keywords is invalid.
|
|
*
|
|
* @param {ASTNode} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingForForOfStatement(node) {
|
|
checkSpacingAroundFirstToken(node);
|
|
|
|
// `of` is not a keyword token.
|
|
var token = sourceCode.getTokenBefore(node.right);
|
|
while (token.value !== "of") {
|
|
token = sourceCode.getTokenBefore(token);
|
|
}
|
|
checkSpacingAround(token);
|
|
}
|
|
|
|
/**
|
|
* Reports `import`, `export`, `as`, and `from` keywords of a given node if
|
|
* usage of spacing around those keywords is invalid.
|
|
*
|
|
* This rule handles the `*` token in module declarations.
|
|
*
|
|
* import*as A from "./a"; /*error Expected space(s) after "import".
|
|
* error Expected space(s) before "as".
|
|
*
|
|
* @param {ASTNode} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingForModuleDeclaration(node) {
|
|
var firstToken = sourceCode.getFirstToken(node);
|
|
checkSpacingBefore(firstToken, PREV_TOKEN_M);
|
|
checkSpacingAfter(firstToken, NEXT_TOKEN_M);
|
|
|
|
if (node.source) {
|
|
var fromToken = sourceCode.getTokenBefore(node.source);
|
|
checkSpacingBefore(fromToken, PREV_TOKEN_M);
|
|
checkSpacingAfter(fromToken, NEXT_TOKEN_M);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reports `as` keyword of a given node if usage of spacing around this
|
|
* keyword is invalid.
|
|
*
|
|
* @param {ASTNode} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingForImportNamespaceSpecifier(node) {
|
|
var asToken = sourceCode.getFirstToken(node, 1);
|
|
checkSpacingBefore(asToken, PREV_TOKEN_M);
|
|
}
|
|
|
|
/**
|
|
* Reports `static`, `get`, and `set` keywords of a given node if usage of
|
|
* spacing around those keywords is invalid.
|
|
*
|
|
* @param {ASTNode} node - A node to report.
|
|
* @returns {void}
|
|
*/
|
|
function checkSpacingForProperty(node) {
|
|
if (node.static) {
|
|
checkSpacingAroundFirstToken(node);
|
|
}
|
|
if (node.kind === "get" || node.kind === "set") {
|
|
var token = sourceCode.getFirstToken(
|
|
node,
|
|
node.static ? 1 : 0
|
|
);
|
|
checkSpacingAround(token);
|
|
}
|
|
}
|
|
|
|
return {
|
|
// Statements
|
|
DebuggerStatement: checkSpacingAroundFirstToken,
|
|
WithStatement: checkSpacingAroundFirstToken,
|
|
|
|
// Statements - Control flow
|
|
BreakStatement: checkSpacingAroundFirstToken,
|
|
ContinueStatement: checkSpacingAroundFirstToken,
|
|
ReturnStatement: checkSpacingAroundFirstToken,
|
|
ThrowStatement: checkSpacingAroundFirstToken,
|
|
TryStatement: checkSpacingForTryStatement,
|
|
|
|
// Statements - Choice
|
|
IfStatement: checkSpacingForIfStatement,
|
|
SwitchStatement: checkSpacingAroundFirstToken,
|
|
SwitchCase: checkSpacingAroundFirstToken,
|
|
|
|
// Statements - Loops
|
|
DoWhileStatement: checkSpacingForDoWhileStatement,
|
|
ForInStatement: checkSpacingForForInStatement,
|
|
ForOfStatement: checkSpacingForForOfStatement,
|
|
ForStatement: checkSpacingAroundFirstToken,
|
|
WhileStatement: checkSpacingAroundFirstToken,
|
|
|
|
// Statements - Declarations
|
|
ClassDeclaration: checkSpacingForClass,
|
|
ExportNamedDeclaration: checkSpacingForModuleDeclaration,
|
|
ExportDefaultDeclaration: checkSpacingAroundFirstToken,
|
|
ExportAllDeclaration: checkSpacingForModuleDeclaration,
|
|
FunctionDeclaration: checkSpacingBeforeFirstToken,
|
|
ImportDeclaration: checkSpacingForModuleDeclaration,
|
|
VariableDeclaration: checkSpacingAroundFirstToken,
|
|
|
|
// Expressions
|
|
ClassExpression: checkSpacingForClass,
|
|
FunctionExpression: checkSpacingBeforeFirstToken,
|
|
NewExpression: checkSpacingBeforeFirstToken,
|
|
Super: checkSpacingBeforeFirstToken,
|
|
ThisExpression: checkSpacingBeforeFirstToken,
|
|
UnaryExpression: checkSpacingBeforeFirstToken,
|
|
YieldExpression: checkSpacingBeforeFirstToken,
|
|
|
|
// Others
|
|
ImportNamespaceSpecifier: checkSpacingForImportNamespaceSpecifier,
|
|
MethodDefinition: checkSpacingForProperty,
|
|
Property: checkSpacingForProperty
|
|
};
|
|
};
|
|
|
|
module.exports.schema = [
|
|
{
|
|
"type": "object",
|
|
"properties": {
|
|
"before": {"type": "boolean"},
|
|
"after": {"type": "boolean"},
|
|
"overrides": {
|
|
"type": "object",
|
|
"properties": KEYS.reduce(function(retv, key) {
|
|
retv[key] = {
|
|
"type": "object",
|
|
"properties": {
|
|
"before": {"type": "boolean"},
|
|
"after": {"type": "boolean"}
|
|
},
|
|
"additionalProperties": false
|
|
};
|
|
return retv;
|
|
}, {}),
|
|
"additionalProperties": false
|
|
}
|
|
},
|
|
"additionalProperties": false
|
|
}
|
|
];
|
|
|