diff --git a/README.md b/README.md index 9434c381a..461154450 100644 --- a/README.md +++ b/README.md @@ -1350,6 +1350,82 @@ function quux (foo) { } // Message: Missing JSDoc @param "foo" declaration. + +/** + * @implements + */ +function quux (foo) { + +} +// Message: Missing JSDoc @param "foo" declaration. + +/** + * @augments + */ +function quux (foo) { + +} +// Message: Missing JSDoc @param "foo" declaration. + +/** + * @extends + */ +function quux (foo) { + +} +// Message: Missing JSDoc @param "foo" declaration. + +/** + * @override + */ +class A { + /** + * + */ + quux (foo) { + + } +} +// Message: Missing JSDoc @param "foo" declaration. + +/** + * @implements + */ +class A { + /** + * + */ + quux (foo) { + + } +} +// Message: Missing JSDoc @param "foo" declaration. + +/** + * @augments + */ +class A { + /** + * + */ + quux (foo) { + + } +} +// Message: Missing JSDoc @param "foo" declaration. + +/** + * @extends + */ +class A { + /** + * + */ + quux (foo) { + + } +} +// Message: Missing JSDoc @param "foo" declaration. ``` The following patterns are not considered problems: @@ -1392,6 +1468,154 @@ function quux (foo) { } // Settings: {"jsdoc":{"allowOverrideWithoutParam":true}} + +/** + * @implements + */ +function quux (foo) { + +} +// Settings: {"jsdoc":{"allowImplementsWithoutParam":true}} + +/** + * @implements + * @param foo + */ +function quux (foo) { + +} + +/** + * @augments + */ +function quux (foo) { + +} +// Settings: {"jsdoc":{"allowAugmentsExtendsWithoutParam":true}} + +/** + * @augments + * @param foo + */ +function quux (foo) { + +} + +/** + * @extends + */ +function quux (foo) { + +} +// Settings: {"jsdoc":{"allowAugmentsExtendsWithoutParam":true}} + +/** + * @extends + * @param foo + */ +function quux (foo) { + +} + +/** + * @override + */ +class A { + /** + * @param foo + */ + quux (foo) { + + } +} + +/** + * @override + */ +class A { + /** + * + */ + quux (foo) { + + } +} +// Settings: {"jsdoc":{"allowOverrideWithoutParam":true}} + +/** + * @implements + */ +class A { + /** + * + */ + quux (foo) { + + } +} +// Settings: {"jsdoc":{"allowImplementsWithoutParam":true}} + +/** + * @implements + */ +class A { + /** + * @param foo + */ + quux (foo) { + + } +} + +/** + * @augments + */ +class A { + /** + * + */ + quux (foo) { + + } +} +// Settings: {"jsdoc":{"allowAugmentsExtendsWithoutParam":true}} + +/** + * @augments + */ +class A { + /** + * @param foo + */ + quux (foo) { + + } +} + +/** + * @extends + */ +class A { + /** + * + */ + quux (foo) { + + } +} +// Settings: {"jsdoc":{"allowAugmentsExtendsWithoutParam":true}} + +/** + * @extends + */ +class A { + /** + * @param foo + */ + quux (foo) { + + } +} ``` diff --git a/src/iterateJsdoc.js b/src/iterateJsdoc.js index acea30020..899fa7a33 100644 --- a/src/iterateJsdoc.js +++ b/src/iterateJsdoc.js @@ -2,7 +2,36 @@ import _ from 'lodash'; import commentParser from 'comment-parser'; import jsdocUtils from './jsdocUtils'; -const curryUtils = (functionNode, jsdoc, tagNamePreference, additionalTagNames, allowOverrideWithoutParam) => { +const parseComment = (commentNode, indent) => { + // Preserve JSDoc block start/end indentation. + return commentParser(indent + '/*' + commentNode.value + indent + '*/', { + // @see https://github.com/yavorskiy/comment-parser/issues/21 + parsers: [ + commentParser.PARSERS.parse_tag, + commentParser.PARSERS.parse_type, + (str, data) => { + if (_.includes(['return', 'returns'], data.tag)) { + return null; + } + + return commentParser.PARSERS.parse_name(str, data); + }, + commentParser.PARSERS.parse_description + ] + })[0] || {}; +}; + +const curryUtils = ( + functionNode, + jsdoc, + tagNamePreference, + additionalTagNames, + allowOverrideWithoutParam, + allowImplementsWithoutParam, + allowAugmentsExtendsWithoutParam, + ancestors, + sourceCode +) => { const utils = {}; utils.getFunctionParameterNames = () => { @@ -33,26 +62,36 @@ const curryUtils = (functionNode, jsdoc, tagNamePreference, additionalTagNames, return allowOverrideWithoutParam; }; + utils.isImplementsAllowedWithoutParam = () => { + return allowImplementsWithoutParam; + }; + + utils.isAugmentsExtendsAllowedWithoutParam = () => { + return allowAugmentsExtendsWithoutParam; + }; + + utils.classHasTag = (tagName) => { + const greatGrandParent = ancestors.slice(-3)[0]; + const greatGrandParentValue = greatGrandParent && sourceCode.getFirstToken(greatGrandParent).value; + + if (greatGrandParentValue === 'class') { + const classJsdocNode = sourceCode.getJSDocComment(greatGrandParent); + const indent = _.repeat(' ', classJsdocNode.loc.start.column); + const classJsdoc = parseComment(classJsdocNode, indent); + + if (jsdocUtils.hasTag(classJsdoc, tagName)) { + return true; + } + } + + return false; + }; + return utils; }; -export const parseComment = (commentNode, indent) => { - // Preserve JSDoc block start/end indentation. - return commentParser(indent + '/*' + commentNode.value + indent + '*/', { - // @see https://github.com/yavorskiy/comment-parser/issues/21 - parsers: [ - commentParser.PARSERS.parse_tag, - commentParser.PARSERS.parse_type, - (str, data) => { - if (_.includes(['return', 'returns'], data.tag)) { - return null; - } - - return commentParser.PARSERS.parse_name(str, data); - }, - commentParser.PARSERS.parse_description - ] - })[0] || {}; +export { + parseComment }; export default (iterator) => { @@ -61,6 +100,8 @@ export default (iterator) => { const tagNamePreference = _.get(context, 'settings.jsdoc.tagNamePreference') || {}; const additionalTagNames = _.get(context, 'settings.jsdoc.additionalTagNames') || {}; const allowOverrideWithoutParam = Boolean(_.get(context, 'settings.jsdoc.allowOverrideWithoutParam')); + const allowImplementsWithoutParam = Boolean(_.get(context, 'settings.jsdoc.allowImplementsWithoutParam')); + const allowAugmentsExtendsWithoutParam = Boolean(_.get(context, 'settings.jsdoc.allowAugmentsExtendsWithoutParam')); const checkJsdoc = (functionNode) => { const jsdocNode = sourceCode.getJSDocComment(functionNode); @@ -69,6 +110,8 @@ export default (iterator) => { return; } + const ancestors = context.getAncestors(); + const indent = _.repeat(' ', jsdocNode.loc.start.column); const jsdoc = parseComment(jsdocNode, indent); @@ -100,7 +143,17 @@ export default (iterator) => { } }; - const utils = curryUtils(functionNode, jsdoc, tagNamePreference, additionalTagNames, allowOverrideWithoutParam); + const utils = curryUtils( + functionNode, + jsdoc, + tagNamePreference, + additionalTagNames, + allowOverrideWithoutParam, + allowImplementsWithoutParam, + allowAugmentsExtendsWithoutParam, + ancestors, + sourceCode + ); iterator({ context, diff --git a/src/rules/requireParam.js b/src/rules/requireParam.js index d8f275bfa..b4e4f047a 100644 --- a/src/rules/requireParam.js +++ b/src/rules/requireParam.js @@ -1,6 +1,7 @@ import _ from 'lodash'; import iterateJsdoc from '../iterateJsdoc'; +// eslint-disable-next-line complexity export default iterateJsdoc(({ report, utils @@ -15,7 +16,20 @@ export default iterateJsdoc(({ // When settings.jsdoc.allowOverrideWithoutParam is true, override implies that all documentation is inherited. // See https://github.com/gajus/eslint-plugin-jsdoc/issues/73 - if (utils.hasTag('override') && utils.isOverrideAllowedWithoutParam()) { + if ((utils.hasTag('override') || utils.classHasTag('override')) && utils.isOverrideAllowedWithoutParam()) { + return; + } + + // When settings.jsdoc.allowImplementsWithoutParam is true, implements implies that all documentation is inherited. + // See https://github.com/gajus/eslint-plugin-jsdoc/issues/100 + if ((utils.hasTag('implements') || utils.classHasTag('implements')) && utils.isImplementsAllowedWithoutParam()) { + return; + } + + // When settings.jsdoc.allowAugmentsExtendsWithoutParam is true, augments or extends implies that all documentation is inherited. + // See https://github.com/gajus/eslint-plugin-jsdoc/issues/100 + if ((utils.hasTag('augments') || utils.hasTag('extends') || + utils.classHasTag('augments') || utils.classHasTag('extends')) && utils.isAugmentsExtendsAllowedWithoutParam()) { return; } diff --git a/test/rules/assertions/requireParam.js b/test/rules/assertions/requireParam.js index 79039aaf9..b7f4a1856 100644 --- a/test/rules/assertions/requireParam.js +++ b/test/rules/assertions/requireParam.js @@ -66,6 +66,131 @@ export default { message: 'Missing JSDoc @param "foo" declaration.' } ] + }, + { + code: ` + /** + * @implements + */ + function quux (foo) { + + } + `, + errors: [ + { + message: 'Missing JSDoc @param "foo" declaration.' + } + ] + }, + { + code: ` + /** + * @augments + */ + function quux (foo) { + + } + `, + errors: [ + { + message: 'Missing JSDoc @param "foo" declaration.' + } + ] + }, + { + code: ` + /** + * @extends + */ + function quux (foo) { + + } + `, + errors: [ + { + message: 'Missing JSDoc @param "foo" declaration.' + } + ] + }, + { + code: ` + /** + * @override + */ + class A { + /** + * + */ + quux (foo) { + + } + } + `, + errors: [ + { + message: 'Missing JSDoc @param "foo" declaration.' + } + ] + }, + { + code: ` + /** + * @implements + */ + class A { + /** + * + */ + quux (foo) { + + } + } + `, + errors: [ + { + message: 'Missing JSDoc @param "foo" declaration.' + } + ] + }, + { + code: ` + /** + * @augments + */ + class A { + /** + * + */ + quux (foo) { + + } + } + `, + errors: [ + { + message: 'Missing JSDoc @param "foo" declaration.' + } + ] + }, + { + code: ` + /** + * @extends + */ + class A { + /** + * + */ + quux (foo) { + + } + } + `, + errors: [ + { + message: 'Missing JSDoc @param "foo" declaration.' + } + ] } ], valid: [ @@ -131,6 +256,224 @@ export default { allowOverrideWithoutParam: true } } + }, + { + code: ` + /** + * @implements + */ + function quux (foo) { + + } + `, + settings: { + jsdoc: { + allowImplementsWithoutParam: true + } + } + }, + { + code: ` + /** + * @implements + * @param foo + */ + function quux (foo) { + + } + ` + }, + { + code: ` + /** + * @augments + */ + function quux (foo) { + + } + `, + settings: { + jsdoc: { + allowAugmentsExtendsWithoutParam: true + } + } + }, + { + code: ` + /** + * @augments + * @param foo + */ + function quux (foo) { + + } + ` + }, + { + code: ` + /** + * @extends + */ + function quux (foo) { + + } + `, + settings: { + jsdoc: { + allowAugmentsExtendsWithoutParam: true + } + } + }, + { + code: ` + /** + * @extends + * @param foo + */ + function quux (foo) { + + } + ` + }, + { + code: ` + /** + * @override + */ + class A { + /** + * @param foo + */ + quux (foo) { + + } + } + ` + }, + { + code: ` + /** + * @override + */ + class A { + /** + * + */ + quux (foo) { + + } + } + `, + settings: { + jsdoc: { + allowOverrideWithoutParam: true + } + } + }, + { + code: ` + /** + * @implements + */ + class A { + /** + * + */ + quux (foo) { + + } + } + `, + settings: { + jsdoc: { + allowImplementsWithoutParam: true + } + } + }, + { + code: ` + /** + * @implements + */ + class A { + /** + * @param foo + */ + quux (foo) { + + } + } + ` + }, + { + code: ` + /** + * @augments + */ + class A { + /** + * + */ + quux (foo) { + + } + } + `, + settings: { + jsdoc: { + allowAugmentsExtendsWithoutParam: true + } + } + }, + { + code: ` + /** + * @augments + */ + class A { + /** + * @param foo + */ + quux (foo) { + + } + } + ` + }, + { + code: ` + /** + * @extends + */ + class A { + /** + * + */ + quux (foo) { + + } + } + `, + settings: { + jsdoc: { + allowAugmentsExtendsWithoutParam: true + } + } + }, + { + code: ` + /** + * @extends + */ + class A { + /** + * @param foo + */ + quux (foo) { + + } + } + ` } ] };