UNPKG

4.5 kBJavaScriptView Raw
1"use strict";
2var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3 if (k2 === undefined) k2 = k;
4 Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
5}) : (function(o, m, k, k2) {
6 if (k2 === undefined) k2 = k;
7 o[k2] = m[k];
8}));
9var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
10 Object.defineProperty(o, "default", { enumerable: true, value: v });
11}) : function(o, v) {
12 o["default"] = v;
13});
14var __importStar = (this && this.__importStar) || function (mod) {
15 if (mod && mod.__esModule) return mod;
16 var result = {};
17 if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
18 __setModuleDefault(result, mod);
19 return result;
20};
21Object.defineProperty(exports, "__esModule", { value: true });
22const utils_1 = require("@typescript-eslint/utils");
23const ts = __importStar(require("typescript"));
24const util = __importStar(require("../util"));
25exports.default = util.createRule({
26 name: 'restrict-template-expressions',
27 meta: {
28 type: 'problem',
29 docs: {
30 description: 'Enforce template literal expressions to be of string type',
31 recommended: 'error',
32 requiresTypeChecking: true,
33 },
34 messages: {
35 invalidType: 'Invalid type "{{type}}" of template literal expression.',
36 },
37 schema: [
38 {
39 type: 'object',
40 properties: {
41 allowNumber: { type: 'boolean' },
42 allowBoolean: { type: 'boolean' },
43 allowAny: { type: 'boolean' },
44 allowNullish: { type: 'boolean' },
45 allowRegExp: { type: 'boolean' },
46 },
47 },
48 ],
49 },
50 defaultOptions: [
51 {
52 allowNumber: true,
53 },
54 ],
55 create(context, [options]) {
56 const service = util.getParserServices(context);
57 const typeChecker = service.program.getTypeChecker();
58 function isUnderlyingTypePrimitive(type) {
59 if (util.isTypeFlagSet(type, ts.TypeFlags.StringLike)) {
60 return true;
61 }
62 if (options.allowNumber &&
63 util.isTypeFlagSet(type, ts.TypeFlags.NumberLike | ts.TypeFlags.BigIntLike)) {
64 return true;
65 }
66 if (options.allowBoolean &&
67 util.isTypeFlagSet(type, ts.TypeFlags.BooleanLike)) {
68 return true;
69 }
70 if (options.allowAny && util.isTypeAnyType(type)) {
71 return true;
72 }
73 if (options.allowRegExp &&
74 util.getTypeName(typeChecker, type) === 'RegExp') {
75 return true;
76 }
77 if (options.allowNullish &&
78 util.isTypeFlagSet(type, ts.TypeFlags.Null | ts.TypeFlags.Undefined)) {
79 return true;
80 }
81 return false;
82 }
83 return {
84 TemplateLiteral(node) {
85 // don't check tagged template literals
86 if (node.parent.type === utils_1.AST_NODE_TYPES.TaggedTemplateExpression) {
87 return;
88 }
89 for (const expression of node.expressions) {
90 const expressionType = util.getConstrainedTypeAtLocation(typeChecker, service.esTreeNodeToTSNodeMap.get(expression));
91 if (!isInnerUnionOrIntersectionConformingTo(expressionType, isUnderlyingTypePrimitive)) {
92 context.report({
93 node: expression,
94 messageId: 'invalidType',
95 data: { type: typeChecker.typeToString(expressionType) },
96 });
97 }
98 }
99 },
100 };
101 function isInnerUnionOrIntersectionConformingTo(type, predicate) {
102 return rec(type);
103 function rec(innerType) {
104 if (innerType.isUnion()) {
105 return innerType.types.every(rec);
106 }
107 if (innerType.isIntersection()) {
108 return innerType.types.some(rec);
109 }
110 return predicate(innerType);
111 }
112 }
113 },
114});
115//# sourceMappingURL=restrict-template-expressions.js.map
\No newline at end of file