UNPKG

eslint-plugin-jsdoc

Version:
170 lines (159 loc) 5.81 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _iterateJsdoc = _interopRequireDefault(require("../iterateJsdoc.cjs")); function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; } /** * @param {string} str * @param {string[]} excludeTags * @returns {string} */ const maskExcludedContent = (str, excludeTags) => { const regContent = new RegExp(`([ \\t]+\\*)[ \\t]@(?:${excludeTags.join('|')})(?=[ \\n])([\\w\\|\\W]*?\\n)(?=[ \\t]*\\*(?:[ \\t]*@\\w+\\s|\\/))`, 'gv'); return str.replace(regContent, (_match, margin, code) => { return (margin + '\n').repeat(code.match(/\n/gv).length); }); }; /** * @param {string} str * @returns {string} */ const maskCodeBlocks = str => { const regContent = /([ \t]+\*)[ \t]```[^\n]*?([\w\|\W]*?\n)(?=[ \t]*\*(?:[ \t]*(?:```|@\w+\s)|\/))/gv; return str.replaceAll(regContent, (_match, margin, code) => { return (margin + '\n').repeat(code.match(/\n/gv).length); }); }; /** * @param {string[]} lines * @param {number} lineIndex * @returns {number} */ const getLineNumber = (lines, lineIndex) => { const precedingText = lines.slice(0, lineIndex).join('\n'); const lineBreaks = precedingText.match(/\n/gv) || []; return lineBreaks.length + 1; }; var _default = exports.default = (0, _iterateJsdoc.default)(({ context, jsdocNode, report, sourceCode }) => { const options = context.options[0] || {}; const /** @type {{excludeTags: string[], allowIndentedSections: boolean}} */{ allowIndentedSections = false, excludeTags = ['example'] } = options; const textWithoutCodeBlocks = maskCodeBlocks(sourceCode.getText(jsdocNode)); const text = excludeTags.length ? maskExcludedContent(textWithoutCodeBlocks, excludeTags) : textWithoutCodeBlocks; if (allowIndentedSections) { // When allowIndentedSections is enabled, only check for indentation on tag lines // and the very first line of the main description const lines = text.split('\n'); let hasSeenContent = false; let currentSectionIndent = null; for (const [lineIndex, line] of lines.entries()) { // Check for indentation (two or more spaces after *) const indentMatch = line.match(/^(?:\/?\**|[\t ]*)\*([\t ]{2,})/v); if (indentMatch) { // Check what comes after the indentation const afterIndent = line.slice(indentMatch[0].length); const indentAmount = indentMatch[1].length; // If this is a tag line with indentation, always report if (/^@\w+/v.test(afterIndent)) { report('There must be no indentation.', null, { line: getLineNumber(lines, lineIndex) }); return; } // If we haven't seen any content yet (main description first line) and there's content, report if (!hasSeenContent && afterIndent.trim().length > 0) { report('There must be no indentation.', null, { line: getLineNumber(lines, lineIndex) }); return; } // For continuation lines, check consistency if (hasSeenContent && afterIndent.trim().length > 0) { if (currentSectionIndent === null) { // First indented line in this section, set the indent level currentSectionIndent = indentAmount; } else if (indentAmount < currentSectionIndent) { // Indentation is less than the established level (inconsistent) report('There must be no indentation.', null, { line: getLineNumber(lines, lineIndex) }); return; } } } else if (/^\s*\*\s+\S/v.test(line)) { // No indentation on this line, reset section indent tracking // (unless it's just whitespace or a closing comment) currentSectionIndent = null; } // Track if we've seen any content (non-whitespace after the *) if (/^\s*\*\s+\S/v.test(line)) { hasSeenContent = true; } // Reset section indent when we encounter a tag if (/@\w+/v.test(line)) { currentSectionIndent = null; } } } else { const reg = /^(?:\/?\**|[ \t]*)\*[ \t]{2}/gmv; if (reg.test(text)) { const lineBreaks = text.slice(0, reg.lastIndex).match(/\n/gv) || []; report('There must be no indentation.', null, { line: lineBreaks.length }); } } }, { iterateAllJsdocs: true, meta: { docs: { description: 'Reports invalid padding inside JSDoc blocks.', url: 'https://github.com/gajus/eslint-plugin-jsdoc/blob/main/docs/rules/check-indentation.md#repos-sticky-header' }, schema: [{ additionalProperties: false, properties: { allowIndentedSections: { description: 'Allows indentation of nested sections on subsequent lines (like bullet lists)', type: 'boolean' }, excludeTags: { description: `Array of tags (e.g., \`['example', 'description']\`) whose content will be "hidden" from the \`check-indentation\` rule. Defaults to \`['example']\`. By default, the whole JSDoc block will be checked for invalid padding. That would include \`@example\` blocks too, which can get in the way of adding full, readable examples of code without ending up with multiple linting issues. When disabled (by passing \`excludeTags: []\` option), the following code *will* report a padding issue: \`\`\`js /** * @example * anArray.filter((a) => { * return a.b; * }); */ \`\`\``, items: { pattern: '^\\S+$', type: 'string' }, type: 'array' } }, type: 'object' }], type: 'layout' } }); module.exports = exports.default; //# sourceMappingURL=checkIndentation.cjs.map