checkValues.js 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. "use strict";
  2. Object.defineProperty(exports, "__esModule", {
  3. value: true
  4. });
  5. exports.default = void 0;
  6. var _iterateJsdoc = _interopRequireDefault(require("../iterateJsdoc.js"));
  7. var _semver = _interopRequireDefault(require("semver"));
  8. var _spdxExpressionParse = _interopRequireDefault(require("spdx-expression-parse"));
  9. function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
  10. const allowedKinds = new Set(['class', 'constant', 'event', 'external', 'file', 'function', 'member', 'mixin', 'module', 'namespace', 'typedef']);
  11. var _default = exports.default = (0, _iterateJsdoc.default)(({
  12. utils,
  13. report,
  14. context
  15. }) => {
  16. const options = context.options[0] || {};
  17. const {
  18. allowedLicenses = null,
  19. allowedAuthors = null,
  20. numericOnlyVariation = false,
  21. licensePattern = '/([^\n\r]*)/gu'
  22. } = options;
  23. utils.forEachPreferredTag('version', (jsdocParameter, targetTagName) => {
  24. const version = /** @type {string} */utils.getTagDescription(jsdocParameter).trim();
  25. if (!version) {
  26. report(`Missing JSDoc @${targetTagName} value.`, null, jsdocParameter);
  27. } else if (!_semver.default.valid(version)) {
  28. report(`Invalid JSDoc @${targetTagName}: "${utils.getTagDescription(jsdocParameter)}".`, null, jsdocParameter);
  29. }
  30. });
  31. utils.forEachPreferredTag('kind', (jsdocParameter, targetTagName) => {
  32. const kind = /** @type {string} */utils.getTagDescription(jsdocParameter).trim();
  33. if (!kind) {
  34. report(`Missing JSDoc @${targetTagName} value.`, null, jsdocParameter);
  35. } else if (!allowedKinds.has(kind)) {
  36. report(`Invalid JSDoc @${targetTagName}: "${utils.getTagDescription(jsdocParameter)}"; ` + `must be one of: ${[...allowedKinds].join(', ')}.`, null, jsdocParameter);
  37. }
  38. });
  39. if (numericOnlyVariation) {
  40. utils.forEachPreferredTag('variation', (jsdocParameter, targetTagName) => {
  41. const variation = /** @type {string} */utils.getTagDescription(jsdocParameter).trim();
  42. if (!variation) {
  43. report(`Missing JSDoc @${targetTagName} value.`, null, jsdocParameter);
  44. } else if (!Number.isInteger(Number(variation)) || Number(variation) <= 0) {
  45. report(`Invalid JSDoc @${targetTagName}: "${utils.getTagDescription(jsdocParameter)}".`, null, jsdocParameter);
  46. }
  47. });
  48. }
  49. utils.forEachPreferredTag('since', (jsdocParameter, targetTagName) => {
  50. const version = /** @type {string} */utils.getTagDescription(jsdocParameter).trim();
  51. if (!version) {
  52. report(`Missing JSDoc @${targetTagName} value.`, null, jsdocParameter);
  53. } else if (!_semver.default.valid(version)) {
  54. report(`Invalid JSDoc @${targetTagName}: "${utils.getTagDescription(jsdocParameter)}".`, null, jsdocParameter);
  55. }
  56. });
  57. utils.forEachPreferredTag('license', (jsdocParameter, targetTagName) => {
  58. const licenseRegex = utils.getRegexFromString(licensePattern, 'g');
  59. const matches = /** @type {string} */utils.getTagDescription(jsdocParameter).matchAll(licenseRegex);
  60. let positiveMatch = false;
  61. for (const match of matches) {
  62. const license = match[1] || match[0];
  63. if (license) {
  64. positiveMatch = true;
  65. }
  66. if (!license.trim()) {
  67. // Avoid reporting again as empty match
  68. if (positiveMatch) {
  69. return;
  70. }
  71. report(`Missing JSDoc @${targetTagName} value.`, null, jsdocParameter);
  72. } else if (allowedLicenses) {
  73. if (allowedLicenses !== true && !allowedLicenses.includes(license)) {
  74. report(`Invalid JSDoc @${targetTagName}: "${license}"; expected one of ${allowedLicenses.join(', ')}.`, null, jsdocParameter);
  75. }
  76. } else {
  77. try {
  78. (0, _spdxExpressionParse.default)(license);
  79. } catch {
  80. report(`Invalid JSDoc @${targetTagName}: "${license}"; expected SPDX expression: https://spdx.org/licenses/.`, null, jsdocParameter);
  81. }
  82. }
  83. }
  84. });
  85. utils.forEachPreferredTag('author', (jsdocParameter, targetTagName) => {
  86. const author = /** @type {string} */utils.getTagDescription(jsdocParameter).trim();
  87. if (!author) {
  88. report(`Missing JSDoc @${targetTagName} value.`, null, jsdocParameter);
  89. } else if (allowedAuthors && !allowedAuthors.includes(author)) {
  90. report(`Invalid JSDoc @${targetTagName}: "${utils.getTagDescription(jsdocParameter)}"; expected one of ${allowedAuthors.join(', ')}.`, null, jsdocParameter);
  91. }
  92. });
  93. }, {
  94. iterateAllJsdocs: true,
  95. meta: {
  96. docs: {
  97. description: 'This rule checks the values for a handful of tags: `@version`, `@since`, `@license` and `@author`.',
  98. url: 'https://github.com/gajus/eslint-plugin-jsdoc/blob/main/docs/rules/check-values.md#repos-sticky-header'
  99. },
  100. schema: [{
  101. additionalProperties: false,
  102. properties: {
  103. allowedAuthors: {
  104. items: {
  105. type: 'string'
  106. },
  107. type: 'array'
  108. },
  109. allowedLicenses: {
  110. anyOf: [{
  111. items: {
  112. type: 'string'
  113. },
  114. type: 'array'
  115. }, {
  116. type: 'boolean'
  117. }]
  118. },
  119. licensePattern: {
  120. type: 'string'
  121. },
  122. numericOnlyVariation: {
  123. type: 'boolean'
  124. }
  125. },
  126. type: 'object'
  127. }],
  128. type: 'suggestion'
  129. }
  130. });
  131. module.exports = exports.default;
  132. //# sourceMappingURL=checkValues.js.map