index.js 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. 'use strict';
  2. const fs = require('fs');
  3. const valueParser = require('postcss-value-parser');
  4. const functionsListPath = require('css-functions-list');
  5. const declarationValueIndex = require('../../utils/declarationValueIndex');
  6. const optionsMatches = require('../../utils/optionsMatches');
  7. const report = require('../../utils/report');
  8. const ruleMessages = require('../../utils/ruleMessages');
  9. const validateOptions = require('../../utils/validateOptions');
  10. const isStandardSyntaxFunction = require('../../utils/isStandardSyntaxFunction');
  11. const isCustomFunction = require('../../utils/isCustomFunction');
  12. const { isRegExp, isString } = require('../../utils/validateTypes');
  13. const isStandardSyntaxValue = require('../../utils/isStandardSyntaxValue');
  14. const ruleName = 'function-no-unknown';
  15. const messages = ruleMessages(ruleName, {
  16. rejected: (name) => `Unexpected unknown function "${name}"`,
  17. });
  18. const meta = {
  19. url: 'https://stylelint.io/user-guide/rules/function-no-unknown',
  20. };
  21. /** @type {import('stylelint').Rule} */
  22. const rule = (primary, secondaryOptions) => {
  23. return (root, result) => {
  24. const validOptions = validateOptions(
  25. result,
  26. ruleName,
  27. { actual: primary },
  28. {
  29. actual: secondaryOptions,
  30. possible: {
  31. ignoreFunctions: [isString, isRegExp],
  32. },
  33. optional: true,
  34. },
  35. );
  36. if (!validOptions) {
  37. return;
  38. }
  39. const functionsList = [
  40. ...JSON.parse(fs.readFileSync(functionsListPath.toString(), 'utf8')),
  41. // #5960
  42. '-webkit-gradient',
  43. 'color-stop',
  44. 'from',
  45. 'to',
  46. // #6537
  47. 'scroll',
  48. ];
  49. root.walkDecls((decl) => {
  50. const { value } = decl;
  51. if (!isStandardSyntaxValue(value)) return;
  52. valueParser(value).walk((node) => {
  53. const name = node.value;
  54. if (node.type !== 'function') {
  55. return;
  56. }
  57. if (!isStandardSyntaxFunction(node)) {
  58. return;
  59. }
  60. if (isCustomFunction(name)) {
  61. return;
  62. }
  63. if (optionsMatches(secondaryOptions, 'ignoreFunctions', name)) {
  64. return;
  65. }
  66. if (functionsList.includes(name.toLowerCase())) {
  67. return;
  68. }
  69. report({
  70. message: messages.rejected,
  71. messageArgs: [name],
  72. node: decl,
  73. index: declarationValueIndex(decl) + node.sourceIndex,
  74. result,
  75. ruleName,
  76. word: name,
  77. });
  78. });
  79. });
  80. };
  81. };
  82. rule.ruleName = ruleName;
  83. rule.messages = messages;
  84. rule.meta = meta;
  85. module.exports = rule;