UNPKG

eslint-plugin-jsdoc

Version:
109 lines (105 loc) 3.88 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 {import('../iterateJsdoc.js').Utils} utils * @param {import('../iterateJsdoc.js').Settings} settings * @returns {boolean} */ const canSkip = (utils, settings) => { const voidingTags = [ // An abstract function is by definition incomplete // so it is perfectly fine if a return is documented but // not present within the function. // A subclass may inherit the doc and implement the // missing return. 'abstract', 'virtual', // A constructor function returns `this` by default, so may be `@returns` // tag indicating this but no explicit return 'class', 'constructor', 'interface']; if (settings.mode === 'closure') { // Structural Interface in GCC terms, equivalent to @interface tag as far as this rule is concerned voidingTags.push('record'); } return utils.hasATag(voidingTags) || utils.isConstructor() || utils.classHasTag('interface') || settings.mode === 'closure' && utils.classHasTag('record'); }; // eslint-disable-next-line complexity -- Temporary var _default = exports.default = (0, _iterateJsdoc.default)(({ context, node, report, settings, utils }) => { const { exemptAsync = true, exemptGenerators = settings.mode === 'typescript', reportMissingReturnForUndefinedTypes = false } = context.options[0] || {}; if (canSkip(utils, settings)) { return; } if (exemptAsync && utils.isAsync()) { return; } const tagName = /** @type {string} */utils.getPreferredTagName({ tagName: 'returns' }); if (!tagName) { return; } const tags = utils.getTags(tagName); if (tags.length === 0) { return; } if (tags.length > 1) { report(`Found more than one @${tagName} declaration.`); return; } const [tag] = tags; const type = tag.type.trim(); // https://www.typescriptlang.org/docs/handbook/release-notes/typescript-3-7.html#assertion-functions if (/asserts\s/u.test(type)) { return; } const returnNever = type === 'never'; if (returnNever && utils.hasValueOrExecutorHasNonEmptyResolveValue(false)) { report(`JSDoc @${tagName} declaration set with "never" but return expression is present in function.`); return; } // In case a return value is declared in JSDoc, we also expect one in the code. if (!returnNever && (reportMissingReturnForUndefinedTypes || !utils.mayBeUndefinedTypeTag(tag)) && (tag.type === '' && !utils.hasValueOrExecutorHasNonEmptyResolveValue(exemptAsync) || tag.type !== '' && !utils.hasValueOrExecutorHasNonEmptyResolveValue(exemptAsync, true)) && Boolean(!exemptGenerators || !node || !('generator' in ( /** @type {import('../iterateJsdoc.js').Node} */node)) || !( /** @type {import('@typescript-eslint/types').TSESTree.FunctionDeclaration} */node).generator)) { report(`JSDoc @${tagName} declaration present but return expression not available in function.`); } }, { meta: { docs: { description: 'Requires a return statement in function body if a `@returns` tag is specified in jsdoc comment.', url: 'https://github.com/gajus/eslint-plugin-jsdoc/blob/main/docs/rules/require-returns-check.md#repos-sticky-header' }, schema: [{ additionalProperties: false, properties: { exemptAsync: { default: true, type: 'boolean' }, exemptGenerators: { type: 'boolean' }, reportMissingReturnForUndefinedTypes: { default: false, type: 'boolean' } }, type: 'object' }], type: 'suggestion' } }); module.exports = exports.default; //# sourceMappingURL=requireReturnsCheck.cjs.map