eslint-plugin-jsonc
Version:
ESLint plugin for JSON, JSONC and JSON5 files.
175 lines (174 loc) • 7.15 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
const utils_1 = require("../utils");
const eslint_ast_utils_1 = require("../utils/eslint-ast-utils");
const eslint_utils_1 = require("@eslint-community/eslint-utils");
exports.default = (0, utils_1.createRule)("array-bracket-newline", {
meta: {
docs: {
description: "enforce line breaks after opening and before closing array brackets",
recommended: null,
extensionRule: true,
layout: true,
},
type: "layout",
fixable: "whitespace",
schema: [
{
oneOf: [
{
type: "string",
enum: ["always", "never", "consistent"],
},
{
type: "object",
properties: {
multiline: {
type: "boolean",
},
minItems: {
type: ["integer", "null"],
minimum: 0,
},
},
additionalProperties: false,
},
],
},
],
messages: {
unexpectedOpeningLinebreak: "There should be no linebreak after '['.",
unexpectedClosingLinebreak: "There should be no linebreak before ']'.",
missingOpeningLinebreak: "A linebreak is required after '['.",
missingClosingLinebreak: "A linebreak is required before ']'.",
},
},
create(context) {
const sourceCode = context.sourceCode;
if (!sourceCode.parserServices.isJSON) {
return {};
}
function normalizeOptionValue(option) {
let consistent = false;
let multiline = false;
let minItems = 0;
if (option) {
if (option === "consistent") {
consistent = true;
minItems = Number.POSITIVE_INFINITY;
}
else if (option === "always" ||
(typeof option !== "string" && option.minItems === 0)) {
minItems = 0;
}
else if (option === "never") {
minItems = Number.POSITIVE_INFINITY;
}
else {
multiline = Boolean(option.multiline);
minItems = option.minItems || Number.POSITIVE_INFINITY;
}
}
else {
consistent = false;
multiline = true;
minItems = Number.POSITIVE_INFINITY;
}
return { consistent, multiline, minItems };
}
function normalizeOptions(options) {
const value = normalizeOptionValue(options);
return { JSONArrayExpression: value, JSONArrayPattern: value };
}
function reportNoBeginningLinebreak(node, token) {
context.report({
node: node,
loc: token.loc,
messageId: "unexpectedOpeningLinebreak",
fix(fixer) {
const nextToken = sourceCode.getTokenAfter(token, {
includeComments: true,
});
if (!nextToken || (0, eslint_utils_1.isCommentToken)(nextToken))
return null;
return fixer.removeRange([token.range[1], nextToken.range[0]]);
},
});
}
function reportNoEndingLinebreak(node, token) {
context.report({
node: node,
loc: token.loc,
messageId: "unexpectedClosingLinebreak",
fix(fixer) {
const previousToken = sourceCode.getTokenBefore(token, {
includeComments: true,
});
if (!previousToken || (0, eslint_utils_1.isCommentToken)(previousToken))
return null;
return fixer.removeRange([previousToken.range[1], token.range[0]]);
},
});
}
function reportRequiredBeginningLinebreak(node, token) {
context.report({
node: node,
loc: token.loc,
messageId: "missingOpeningLinebreak",
fix(fixer) {
return fixer.insertTextAfter(token, "\n");
},
});
}
function reportRequiredEndingLinebreak(node, token) {
context.report({
node: node,
loc: token.loc,
messageId: "missingClosingLinebreak",
fix(fixer) {
return fixer.insertTextBefore(token, "\n");
},
});
}
function check(node) {
const elements = node.elements;
const normalizedOptions = normalizeOptions(context.options[0]);
const options = normalizedOptions[node.type];
const openBracket = sourceCode.getFirstToken(node);
const closeBracket = sourceCode.getLastToken(node);
const firstIncComment = sourceCode.getTokenAfter(openBracket, {
includeComments: true,
});
const lastIncComment = sourceCode.getTokenBefore(closeBracket, {
includeComments: true,
});
const first = sourceCode.getTokenAfter(openBracket);
const last = sourceCode.getTokenBefore(closeBracket);
const needsLinebreaks = elements.length >= options.minItems ||
(options.multiline &&
elements.length > 0 &&
firstIncComment.loc.start.line !== lastIncComment.loc.end.line) ||
(elements.length === 0 &&
firstIncComment.type === "Block" &&
firstIncComment.loc.start.line !== lastIncComment.loc.end.line &&
firstIncComment === lastIncComment) ||
(options.consistent &&
openBracket.loc.end.line !== first.loc.start.line);
if (needsLinebreaks) {
if ((0, eslint_ast_utils_1.isTokenOnSameLine)(openBracket, first))
reportRequiredBeginningLinebreak(node, openBracket);
if ((0, eslint_ast_utils_1.isTokenOnSameLine)(last, closeBracket))
reportRequiredEndingLinebreak(node, closeBracket);
}
else {
if (!(0, eslint_ast_utils_1.isTokenOnSameLine)(openBracket, first))
reportNoBeginningLinebreak(node, openBracket);
if (!(0, eslint_ast_utils_1.isTokenOnSameLine)(last, closeBracket))
reportNoEndingLinebreak(node, closeBracket);
}
}
return {
JSONArrayExpression: check,
};
},
});
;