text-encoding-identifier-case.js 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. 'use strict';
  2. const {replaceStringLiteral} = require('./fix/index.js');
  3. const MESSAGE_ID_ERROR = 'text-encoding-identifier/error';
  4. const MESSAGE_ID_SUGGESTION = 'text-encoding-identifier/suggestion';
  5. const messages = {
  6. [MESSAGE_ID_ERROR]: 'Prefer `{{replacement}}` over `{{value}}`.',
  7. [MESSAGE_ID_SUGGESTION]: 'Replace `{{value}}` with `{{replacement}}`.',
  8. };
  9. const getReplacement = encoding => {
  10. switch (encoding.toLowerCase()) {
  11. // eslint-disable-next-line unicorn/text-encoding-identifier-case
  12. case 'utf-8':
  13. case 'utf8':
  14. return 'utf8';
  15. case 'ascii':
  16. return 'ascii';
  17. // No default
  18. }
  19. };
  20. // `fs.{readFile,readFileSync}()`
  21. const isFsReadFileEncoding = node =>
  22. node.parent.type === 'CallExpression'
  23. && !node.parent.optional
  24. && node.parent.arguments[1] === node
  25. && node.parent.arguments[0]
  26. && node.parent.arguments[0].type !== 'SpreadElement'
  27. && node.parent.callee.type === 'MemberExpression'
  28. && !node.parent.callee.optional
  29. && !node.parent.callee.computed
  30. && node.parent.callee.property.type === 'Identifier'
  31. && (node.parent.callee.property.name === 'readFile' || node.parent.callee.property.name === 'readFileSync');
  32. /** @param {import('eslint').Rule.RuleContext} context */
  33. const create = () => ({
  34. Literal(node) {
  35. if (typeof node.value !== 'string') {
  36. return;
  37. }
  38. if (
  39. // eslint-disable-next-line unicorn/text-encoding-identifier-case
  40. node.value === 'utf-8'
  41. && node.parent.type === 'JSXAttribute'
  42. && node.parent.value === node
  43. && node.parent.name.type === 'JSXIdentifier'
  44. && node.parent.name.name.toLowerCase() === 'charset'
  45. && node.parent.parent.type === 'JSXOpeningElement'
  46. && node.parent.parent.attributes.includes(node.parent)
  47. && node.parent.parent.name.type === 'JSXIdentifier'
  48. && node.parent.parent.name.name.toLowerCase() === 'meta'
  49. ) {
  50. return;
  51. }
  52. const {raw} = node;
  53. const value = raw.slice(1, -1);
  54. const replacement = getReplacement(value);
  55. if (!replacement || replacement === value) {
  56. return;
  57. }
  58. const messageData = {
  59. value,
  60. replacement,
  61. };
  62. /** @param {import('eslint').Rule.RuleFixer} fixer */
  63. const fix = fixer => replaceStringLiteral(fixer, node, replacement);
  64. const problem = {
  65. node,
  66. messageId: MESSAGE_ID_ERROR,
  67. data: messageData,
  68. };
  69. if (isFsReadFileEncoding(node)) {
  70. problem.fix = fix;
  71. return problem;
  72. }
  73. problem.suggest = [
  74. {
  75. messageId: MESSAGE_ID_SUGGESTION,
  76. data: messageData,
  77. fix: fixer => replaceStringLiteral(fixer, node, replacement),
  78. },
  79. ];
  80. return problem;
  81. },
  82. });
  83. /** @type {import('eslint').Rule.RuleModule} */
  84. module.exports = {
  85. create,
  86. meta: {
  87. type: 'suggestion',
  88. docs: {
  89. description: 'Enforce consistent case for text encoding identifiers.',
  90. },
  91. fixable: 'code',
  92. hasSuggestions: true,
  93. messages,
  94. },
  95. };