blob: d32d07d0653cbd4d01eecf8ee02191076e6c3945 [file] [log] [blame]
"use strict";
const _ = require("lodash");
const optionsMatches = require("../../utils/optionsMatches");
const report = require("../../utils/report");
const ruleMessages = require("../../utils/ruleMessages");
const styleSearch = require("style-search");
const validateOptions = require("../../utils/validateOptions");
const ruleName = "max-empty-lines";
const messages = ruleMessages(ruleName, {
expected: max =>
`Expected no more than ${max} empty ${max === 1 ? "line" : "lines"}`
});
const rule = function(max, options) {
let emptyLines = 0;
let lastIndex = -1;
return (root, result) => {
const validOptions = validateOptions(
result,
ruleName,
{
actual: max,
possible: _.isNumber
},
{
actual: options,
possible: {
ignore: ["comments"]
},
optional: true
}
);
if (!validOptions) {
return;
}
const ignoreComments = optionsMatches(options, "ignore", "comments");
emptyLines = 0;
lastIndex = -1;
const rootString = root.toString();
styleSearch(
{
source: rootString,
target: /\r\n/.test(rootString) ? "\r\n" : "\n",
comments: ignoreComments ? "skip" : "check"
},
match => {
checkMatch(rootString, match.startIndex, match.endIndex, root);
}
);
function checkMatch(source, matchStartIndex, matchEndIndex, node) {
const eof = matchEndIndex === source.length ? true : false;
let violation = false;
// Additional check for beginning of file
if (!matchStartIndex || lastIndex === matchStartIndex) {
emptyLines++;
} else {
emptyLines = 0;
}
lastIndex = matchEndIndex;
if (emptyLines > max) violation = true;
if (!eof && !violation) return;
if (violation) {
report({
message: messages.expected(max),
node,
index: matchStartIndex,
result,
ruleName
});
}
// Additional check for end of file
if (eof && max) {
emptyLines++;
if (emptyLines > max && isEofNode(result.root, node)) {
report({
message: messages.expected(max),
node,
index: matchEndIndex,
result,
ruleName
});
}
}
}
};
};
/**
* Checks whether the given node is the last node of file.
* @param {Document|null} document the document node with `postcss-html` and `postcss-jsx`.
* @param {Root} root the root node of css
*/
function isEofNode(document, root) {
if (!document || document.constructor.name !== "Document") {
return true;
}
// In the `postcss-html` and `postcss-jsx` syntax, checks that there is text after the given node.
let after;
if (root === document.last) {
after = _.get(document, "raws.afterEnd");
} else {
const rootIndex = document.index(root);
after = _.get(document.nodes[rootIndex + 1], "raws.beforeStart");
}
return !(after + "").trim();
}
rule.ruleName = ruleName;
rule.messages = messages;
module.exports = rule;