@checkdigit/eslint-plugin 7.6.0-PR.75-4751 → 7.6.0-PR.97-b19c

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (77) hide show
  1. package/dist-mjs/index.mjs +8 -156
  2. package/dist-mjs/no-status-code-assert.mjs +63 -0
  3. package/dist-mjs/require-resolve-full-response.mjs +5 -5
  4. package/dist-types/no-status-code-assert.d.ts +5 -0
  5. package/package.json +1 -1
  6. package/src/index.ts +4 -152
  7. package/src/no-status-code-assert.ts +85 -0
  8. package/src/require-resolve-full-response.ts +4 -4
  9. package/dist-mjs/agent/add-assert-import.mjs +0 -58
  10. package/dist-mjs/agent/add-base-path-const.mjs +0 -65
  11. package/dist-mjs/agent/add-base-path-import.mjs +0 -60
  12. package/dist-mjs/agent/add-url-domain.mjs +0 -61
  13. package/dist-mjs/agent/agent-test-wiring.mjs +0 -221
  14. package/dist-mjs/agent/fetch-response-body-json.mjs +0 -146
  15. package/dist-mjs/agent/fetch-response-header-getter.mjs +0 -117
  16. package/dist-mjs/agent/fetch-response-status.mjs +0 -76
  17. package/dist-mjs/agent/fetch-then.mjs +0 -205
  18. package/dist-mjs/agent/fetch.mjs +0 -48
  19. package/dist-mjs/agent/file.mjs +0 -43
  20. package/dist-mjs/agent/fix-function-call-arguments.mjs +0 -153
  21. package/dist-mjs/agent/no-fixture.mjs +0 -383
  22. package/dist-mjs/agent/no-mapped-response.mjs +0 -75
  23. package/dist-mjs/agent/no-service-wrapper.mjs +0 -185
  24. package/dist-mjs/agent/no-status-code.mjs +0 -59
  25. package/dist-mjs/agent/no-supertest.mjs +0 -332
  26. package/dist-mjs/agent/no-unused-function-argument.mjs +0 -79
  27. package/dist-mjs/agent/no-unused-imports.mjs +0 -81
  28. package/dist-mjs/agent/no-unused-service-variable.mjs +0 -74
  29. package/dist-mjs/agent/response-reference.mjs +0 -75
  30. package/dist-mjs/agent/supertest-then.mjs +0 -170
  31. package/dist-mjs/agent/url.mjs +0 -32
  32. package/dist-types/agent/add-assert-import.d.ts +0 -4
  33. package/dist-types/agent/add-base-path-const.d.ts +0 -4
  34. package/dist-types/agent/add-base-path-import.d.ts +0 -4
  35. package/dist-types/agent/add-url-domain.d.ts +0 -4
  36. package/dist-types/agent/agent-test-wiring.d.ts +0 -4
  37. package/dist-types/agent/fetch-response-body-json.d.ts +0 -4
  38. package/dist-types/agent/fetch-response-header-getter.d.ts +0 -4
  39. package/dist-types/agent/fetch-response-status.d.ts +0 -4
  40. package/dist-types/agent/fetch-then.d.ts +0 -4
  41. package/dist-types/agent/fetch.d.ts +0 -8
  42. package/dist-types/agent/file.d.ts +0 -7
  43. package/dist-types/agent/fix-function-call-arguments.d.ts +0 -9
  44. package/dist-types/agent/no-fixture.d.ts +0 -4
  45. package/dist-types/agent/no-mapped-response.d.ts +0 -4
  46. package/dist-types/agent/no-service-wrapper.d.ts +0 -4
  47. package/dist-types/agent/no-status-code.d.ts +0 -4
  48. package/dist-types/agent/no-supertest.d.ts +0 -4
  49. package/dist-types/agent/no-unused-function-argument.d.ts +0 -4
  50. package/dist-types/agent/no-unused-imports.d.ts +0 -4
  51. package/dist-types/agent/no-unused-service-variable.d.ts +0 -4
  52. package/dist-types/agent/response-reference.d.ts +0 -16
  53. package/dist-types/agent/supertest-then.d.ts +0 -4
  54. package/dist-types/agent/url.d.ts +0 -4
  55. package/src/agent/add-assert-import.ts +0 -74
  56. package/src/agent/add-base-path-const.ts +0 -81
  57. package/src/agent/add-base-path-import.ts +0 -69
  58. package/src/agent/add-url-domain.ts +0 -76
  59. package/src/agent/agent-test-wiring.ts +0 -273
  60. package/src/agent/fetch-response-body-json.ts +0 -194
  61. package/src/agent/fetch-response-header-getter.ts +0 -148
  62. package/src/agent/fetch-response-status.ts +0 -100
  63. package/src/agent/fetch-then.ts +0 -358
  64. package/src/agent/fetch.ts +0 -69
  65. package/src/agent/file.ts +0 -42
  66. package/src/agent/fix-function-call-arguments.ts +0 -200
  67. package/src/agent/no-fixture.ts +0 -581
  68. package/src/agent/no-mapped-response.ts +0 -84
  69. package/src/agent/no-service-wrapper.ts +0 -241
  70. package/src/agent/no-status-code.ts +0 -69
  71. package/src/agent/no-supertest.ts +0 -517
  72. package/src/agent/no-unused-function-argument.ts +0 -98
  73. package/src/agent/no-unused-imports.ts +0 -103
  74. package/src/agent/no-unused-service-variable.ts +0 -93
  75. package/src/agent/response-reference.ts +0 -153
  76. package/src/agent/supertest-then.ts +0 -230
  77. package/src/agent/url.ts +0 -32
@@ -1,146 +0,0 @@
1
- // src/agent/fetch-response-body-json.ts
2
- import { strict as assert } from "node:assert";
3
- import { AST_NODE_TYPES, ESLintUtils } from "@typescript-eslint/utils";
4
- import getDocumentationUrl from "../get-documentation-url.mjs";
5
- import { getAncestor } from "../library/ts-tree.mjs";
6
- import { isFetchResponse } from "./fetch.mjs";
7
- var ruleId = "fetch-response-body-json";
8
- var createRule = ESLintUtils.RuleCreator((name) => getDocumentationUrl(name));
9
- var rule = createRule({
10
- name: ruleId,
11
- meta: {
12
- type: "suggestion",
13
- docs: {
14
- description: 'Replace "response.body" with "await response.json()".'
15
- },
16
- messages: {
17
- refactorNeeded: "Please extract the fetch call and check its reponse status code before accessing its response body.",
18
- replaceBodyWithJson: 'Replace "response.body" with "await response.json()".',
19
- unknownError: 'Unknown error occurred in file "{{fileName}}": {{ error }}.'
20
- },
21
- fixable: "code",
22
- schema: []
23
- },
24
- defaultOptions: [],
25
- create(context) {
26
- const parserServices = ESLintUtils.getParserServices(context);
27
- const typeChecker = parserServices.program.getTypeChecker();
28
- const allChanges = /* @__PURE__ */ new Map();
29
- return {
30
- 'MemberExpression[property.name="body"]': (responseBodyNode) => {
31
- try {
32
- const responseNode = parserServices.esTreeNodeToTSNodeMap.get(responseBodyNode.object);
33
- const responseType = typeChecker.getTypeAtLocation(responseNode);
34
- if (isFetchResponse(responseType)) {
35
- if (responseBodyNode.object.type !== AST_NODE_TYPES.Identifier) {
36
- context.report({
37
- node: responseBodyNode,
38
- messageId: "refactorNeeded"
39
- });
40
- return;
41
- }
42
- const enclosingFunction = getAncestor(
43
- responseBodyNode,
44
- (node) => node.type === AST_NODE_TYPES.ArrowFunctionExpression || node.type === AST_NODE_TYPES.FunctionExpression || node.type === AST_NODE_TYPES.FunctionDeclaration
45
- );
46
- const enclosingStatement = getAncestor(
47
- responseBodyNode,
48
- (node) => (node.type === AST_NODE_TYPES.VariableDeclaration || node.type === AST_NODE_TYPES.ExpressionStatement || node.type === AST_NODE_TYPES.ReturnStatement) && node.parent.type === AST_NODE_TYPES.BlockStatement
49
- );
50
- const enclosingStatementIndex = enclosingFunction.body.body.indexOf(
51
- enclosingStatement
52
- );
53
- const responseVariableName = responseBodyNode.object.name;
54
- const isResponseBodyVariableDeclared = enclosingStatement.type === AST_NODE_TYPES.VariableDeclaration && enclosingStatement.declarations.some(
55
- (declaration) => declaration.init === responseBodyNode || declaration.init?.type === AST_NODE_TYPES.TSAsExpression && declaration.init.expression === responseBodyNode
56
- );
57
- const responseBodyVariableName = isResponseBodyVariableDeclared ? enclosingStatement.declarations.find(
58
- (declaration) => declaration.init === responseBodyNode || declaration.init?.type === AST_NODE_TYPES.TSAsExpression && declaration.init.expression === responseBodyNode
59
- )?.id : `${responseBodyNode.object.name}Body`;
60
- const change = {
61
- enclosingFunction,
62
- enclosingStatement,
63
- enclosingStatementIndex,
64
- responseVariableName,
65
- responseBodyNode,
66
- responseBodyVariableName,
67
- isResponseBodyVariableDeclared
68
- };
69
- const changesByFunction = allChanges.get(enclosingFunction) ?? /* @__PURE__ */ new Map();
70
- const changesByResponse = changesByFunction.get(responseVariableName) ?? [];
71
- changesByResponse.push(change);
72
- changesByFunction.set(responseVariableName, changesByResponse);
73
- allChanges.set(enclosingFunction, changesByFunction);
74
- }
75
- } catch (error) {
76
- console.error(`Failed to apply ${ruleId} rule for file "${context.filename}":`, error);
77
- context.report({
78
- node: responseBodyNode,
79
- messageId: "unknownError",
80
- data: {
81
- fileName: context.filename,
82
- error: error instanceof Error ? error.toString() : JSON.stringify(error)
83
- }
84
- });
85
- }
86
- },
87
- "Program:exit": () => {
88
- if (allChanges.size === 0) {
89
- return;
90
- }
91
- const fixes = [];
92
- for (const changesByFunction of allChanges.values()) {
93
- for (const changesByResponse of changesByFunction.values()) {
94
- const orderedChanges = changesByResponse.sort(
95
- (changeA, changeB) => changeA.enclosingStatementIndex - changeB.enclosingStatementIndex
96
- );
97
- const firstChange = orderedChanges[0];
98
- assert(firstChange);
99
- const {
100
- responseBodyNode,
101
- responseVariableName,
102
- responseBodyVariableName,
103
- isResponseBodyVariableDeclared,
104
- enclosingStatement
105
- } = firstChange;
106
- let remainingChanges;
107
- if (!isResponseBodyVariableDeclared) {
108
- fixes.push({
109
- node: context.sourceCode.getTokenBefore(enclosingStatement),
110
- text: `
111
- const ${responseBodyVariableName} = await ${responseVariableName}.json();`,
112
- insert: true
113
- });
114
- remainingChanges = orderedChanges;
115
- } else {
116
- fixes.push({
117
- node: responseBodyNode,
118
- text: `await ${responseVariableName}.json()`,
119
- insert: false
120
- });
121
- remainingChanges = orderedChanges.slice(1);
122
- }
123
- for (const change of remainingChanges) {
124
- fixes.push({ node: change.responseBodyNode, text: responseBodyVariableName, insert: false });
125
- }
126
- }
127
- }
128
- for (const fix of fixes.reverse()) {
129
- context.report({
130
- node: fix.node,
131
- messageId: "replaceBodyWithJson",
132
- fix(fixer) {
133
- return fix.insert ? fixer.insertTextAfter(fix.node, fix.text) : fixer.replaceText(fix.node, fix.text);
134
- }
135
- });
136
- }
137
- }
138
- };
139
- }
140
- });
141
- var fetch_response_body_json_default = rule;
142
- export {
143
- fetch_response_body_json_default as default,
144
- ruleId
145
- };
146
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FnZW50L2ZldGNoLXJlc3BvbnNlLWJvZHktanNvbi50cyJdLAogICJtYXBwaW5ncyI6ICI7QUFRQSxTQUFTLFVBQVUsY0FBYztBQUVqQyxTQUFTLGdCQUFnQixtQkFBNkI7QUFFdEQsT0FBTyx5QkFBeUI7QUFDaEMsU0FBUyxtQkFBbUI7QUFDNUIsU0FBUyx1QkFBdUI7QUFFekIsSUFBTSxTQUFTO0FBRXRCLElBQU0sYUFBYSxZQUFZLFlBQVksQ0FBQyxTQUFTLG9CQUFvQixJQUFJLENBQUM7QUFhOUUsSUFBTSxPQUEwRixXQUFXO0FBQUEsRUFDekcsTUFBTTtBQUFBLEVBQ04sTUFBTTtBQUFBLElBQ0osTUFBTTtBQUFBLElBQ04sTUFBTTtBQUFBLE1BQ0osYUFBYTtBQUFBLElBQ2Y7QUFBQSxJQUNBLFVBQVU7QUFBQSxNQUNSLGdCQUNFO0FBQUEsTUFDRixxQkFBcUI7QUFBQSxNQUNyQixjQUFjO0FBQUEsSUFDaEI7QUFBQSxJQUNBLFNBQVM7QUFBQSxJQUNULFFBQVEsQ0FBQztBQUFBLEVBQ1g7QUFBQSxFQUNBLGdCQUFnQixDQUFDO0FBQUEsRUFDakIsT0FBTyxTQUFTO0FBQ2QsVUFBTSxpQkFBaUIsWUFBWSxrQkFBa0IsT0FBTztBQUM1RCxVQUFNLGNBQWMsZUFBZSxRQUFRLGVBQWU7QUFDMUQsVUFBTSxhQUFhLG9CQUFJLElBQTBDO0FBRWpFLFdBQU87QUFBQSxNQUNMLDBDQUEwQyxDQUFDLHFCQUFnRDtBQUN6RixZQUFJO0FBQ0YsZ0JBQU0sZUFBZSxlQUFlLHNCQUFzQixJQUFJLGlCQUFpQixNQUFNO0FBQ3JGLGdCQUFNLGVBQWUsWUFBWSxrQkFBa0IsWUFBWTtBQUUvRCxjQUFJLGdCQUFnQixZQUFZLEdBQUc7QUFDakMsZ0JBQUksaUJBQWlCLE9BQU8sU0FBUyxlQUFlLFlBQVk7QUFDOUQsc0JBQVEsT0FBTztBQUFBLGdCQUNiLE1BQU07QUFBQSxnQkFDTixXQUFXO0FBQUEsY0FDYixDQUFDO0FBQ0Q7QUFBQSxZ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
@@ -1,117 +0,0 @@
1
- // src/agent/fetch-response-header-getter.ts
2
- import { AST_NODE_TYPES, ESLintUtils } from "@typescript-eslint/utils";
3
- import getDocumentationUrl from "../get-documentation-url.mjs";
4
- var ruleId = "fetch-response-header-getter-ts";
5
- var HEADER_BUILTIN_FUNCTIONS = Object.keys(Headers.prototype);
6
- var createRule = ESLintUtils.RuleCreator((name) => getDocumentationUrl(name));
7
- var rule = createRule({
8
- name: ruleId,
9
- meta: {
10
- type: "suggestion",
11
- docs: {
12
- description: 'Use "get()" method to get header value from the headers object of the fetch response.'
13
- },
14
- messages: {
15
- useGetter: 'Use "get()" method to get header value from the headers object of the fetch response.',
16
- unknownError: 'Unknown error occurred in file "{{fileName}}": {{ error }}.'
17
- },
18
- fixable: "code",
19
- schema: []
20
- },
21
- defaultOptions: [],
22
- create(context) {
23
- const parserServices = ESLintUtils.getParserServices(context);
24
- const typeChecker = parserServices.program.getTypeChecker();
25
- const sourceCode = context.sourceCode;
26
- return {
27
- MemberExpression: (responseHeadersAccess) => {
28
- try {
29
- if (responseHeadersAccess.property.type === AST_NODE_TYPES.Identifier && HEADER_BUILTIN_FUNCTIONS.includes(responseHeadersAccess.property.name)) {
30
- return;
31
- }
32
- const responseHeadersTsNode = parserServices.esTreeNodeToTSNodeMap.get(responseHeadersAccess.object);
33
- let responseHeadersType = typeChecker.getTypeAtLocation(responseHeadersTsNode);
34
- responseHeadersType = responseHeadersType.isUnion() ? responseHeadersType.types[0] : responseHeadersType;
35
- const responseHeadersTypeName = (
36
- // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
37
- (responseHeadersType.symbol ?? responseHeadersType.aliasSymbol)?.escapedName
38
- );
39
- if (responseHeadersTypeName !== "Headers" && responseHeadersTypeName !== "HeaderGetter") {
40
- return;
41
- }
42
- let replacementText;
43
- if (!responseHeadersAccess.computed) {
44
- replacementText = `${sourceCode.getText(responseHeadersAccess.object)}.get('${sourceCode.getText(responseHeadersAccess.property)}')`;
45
- } else if (responseHeadersAccess.property.type === AST_NODE_TYPES.Identifier || responseHeadersAccess.property.type === AST_NODE_TYPES.Literal || responseHeadersAccess.property.type === AST_NODE_TYPES.TemplateLiteral) {
46
- replacementText = `${sourceCode.getText(responseHeadersAccess.object)}.get(${sourceCode.getText(responseHeadersAccess.property)})`;
47
- } else {
48
- throw new Error(`Unexpected property type: ${responseHeadersAccess.property.type}`);
49
- }
50
- context.report({
51
- messageId: "useGetter",
52
- node: responseHeadersAccess.property,
53
- fix(fixer) {
54
- return fixer.replaceText(responseHeadersAccess, replacementText);
55
- }
56
- });
57
- } catch (error) {
58
- console.error(`Failed to apply ${ruleId} rule for file "${context.filename}":`, error);
59
- context.report({
60
- node: responseHeadersAccess,
61
- messageId: "unknownError",
62
- data: {
63
- fileName: context.filename,
64
- error: error instanceof Error ? error.toString() : JSON.stringify(error)
65
- }
66
- });
67
- }
68
- },
69
- // convert response.get() to response.headers.get()
70
- 'CallExpression[callee.property.name="get"]': (responseHeadersAccess) => {
71
- try {
72
- if (responseHeadersAccess.callee.type !== AST_NODE_TYPES.MemberExpression) {
73
- return;
74
- }
75
- if (responseHeadersAccess.callee.object.type !== AST_NODE_TYPES.Identifier || responseHeadersAccess.callee.object.name === "request") {
76
- return;
77
- }
78
- const responseNode = responseHeadersAccess.callee.object;
79
- const responseHeadersTsNode = parserServices.esTreeNodeToTSNodeMap.get(responseNode);
80
- const responseType = typeChecker.getTypeAtLocation(responseHeadersTsNode);
81
- const typeName = typeChecker.typeToString(responseType);
82
- if (typeName === "InboundContext" || typeName.endsWith("RequestType")) {
83
- return;
84
- }
85
- const hasHeadersProperty = responseType.getProperties().some((symbol) => symbol.name === "headers");
86
- if (!hasHeadersProperty) {
87
- return;
88
- }
89
- const replacementText = `${sourceCode.getText(responseNode)}.headers`;
90
- context.report({
91
- messageId: "useGetter",
92
- node: responseHeadersAccess,
93
- fix(fixer) {
94
- return fixer.replaceText(responseNode, replacementText);
95
- }
96
- });
97
- } catch (error) {
98
- console.error(`Failed to apply ${ruleId} rule for file "${context.filename}":`, error);
99
- context.report({
100
- node: responseHeadersAccess,
101
- messageId: "unknownError",
102
- data: {
103
- fileName: context.filename,
104
- error: error instanceof Error ? error.toString() : JSON.stringify(error)
105
- }
106
- });
107
- }
108
- }
109
- };
110
- }
111
- });
112
- var fetch_response_header_getter_default = rule;
113
- export {
114
- fetch_response_header_getter_default as default,
115
- ruleId
116
- };
117
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FnZW50L2ZldGNoLXJlc3BvbnNlLWhlYWRlci1nZXR0ZXIudHMiXSwKICAibWFwcGluZ3MiOiAiO0FBUUEsU0FBUyxnQkFBZ0IsbUJBQTZCO0FBRXRELE9BQU8seUJBQXlCO0FBRXpCLElBQU0sU0FBUztBQUN0QixJQUFNLDJCQUEyQixPQUFPLEtBQUssUUFBUSxTQUFTO0FBRTlELElBQU0sYUFBYSxZQUFZLFlBQVksQ0FBQyxTQUFTLG9CQUFvQixJQUFJLENBQUM7QUFFOUUsSUFBTSxPQUE2RCxXQUFXO0FBQUEsRUFDNUUsTUFBTTtBQUFBLEVBQ04sTUFBTTtBQUFBLElBQ0osTUFBTTtBQUFBLElBQ04sTUFBTTtBQUFBLE1BQ0osYUFBYTtBQUFBLElBQ2Y7QUFBQSxJQUNBLFVBQVU7QUFBQSxNQUNSLFdBQVc7QUFBQSxNQUNYLGNBQWM7QUFBQSxJQUNoQjtBQUFBLElBQ0EsU0FBUztBQUFBLElBQ1QsUUFBUSxDQUFDO0FBQUEsRUFDWDtBQUFBLEVBQ0EsZ0JBQWdCLENBQUM7QUFBQSxFQUNqQixPQUFPLFNBQVM7QUFDZCxVQUFNLGlCQUFpQixZQUFZLGtCQUFrQixPQUFPO0FBQzVELFVBQU0sY0FBYyxlQUFlLFFBQVEsZUFBZTtBQUMxRCxVQUFNLGFBQWEsUUFBUTtBQUUzQixXQUFPO0FBQUEsTUFDTCxrQkFBa0IsQ0FBQywwQkFBcUQ7QUFDdEUsWUFBSTtBQUNGLGNBQ0Usc0JBQXNCLFNBQVMsU0FBUyxlQUFlLGNBQ3ZELHlCQUF5QixTQUFTLHNCQUFzQixTQUFTLElBQUksR0FDckU7QUFFQTtBQUFBLFVBQ0Y7QUFFQSxnQkFBTSx3QkFBd0IsZUFBZSxzQkFBc0IsSUFBSSxzQkFBc0IsTUFBTTtBQUNuRyxjQUFJLHNCQUFzQixZQUFZLGtCQUFrQixxQkFBcUI7QUFFN0UsZ0NBQXNCLG9CQUFvQixRQUFRLElBQUksb0JBQW9CLE1BQU0sQ0FBQyxJQUFLO0FBQ3RGLGdCQUFNO0FBQUE7QUFBQSxhQUNILG9CQUFvQixVQUFV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
@@ -1,76 +0,0 @@
1
- // src/agent/fetch-response-status.ts
2
- import { AST_NODE_TYPES, ESLintUtils } from "@typescript-eslint/utils";
3
- import getDocumentationUrl from "../get-documentation-url.mjs";
4
- import { isFetchResponse } from "./fetch.mjs";
5
- var ruleId = "fetch-response-status";
6
- var createRule = ESLintUtils.RuleCreator((name) => getDocumentationUrl(name));
7
- var rule = createRule({
8
- name: ruleId,
9
- meta: {
10
- type: "problem",
11
- docs: {
12
- description: 'Replace "response.body" with "await response.json()".'
13
- },
14
- messages: {
15
- renameStatusCodeProperty: 'Rename "statusCode" with "status".',
16
- unknownError: 'Unknown error occurred in file "{{fileName}}": {{ error }}.'
17
- },
18
- fixable: "code",
19
- schema: []
20
- },
21
- defaultOptions: [],
22
- create(context) {
23
- const parserServices = ESLintUtils.getParserServices(context);
24
- const typeChecker = parserServices.program.getTypeChecker();
25
- return {
26
- VariableDeclaration: (variableDeclaration) => {
27
- const variableInit = variableDeclaration.declarations[0]?.init;
28
- if (!variableInit || variableInit.type !== AST_NODE_TYPES.AwaitExpression || variableInit.argument.type !== AST_NODE_TYPES.CallExpression) {
29
- return;
30
- }
31
- const variableId = variableDeclaration.declarations[0]?.id;
32
- if (variableId.type !== AST_NODE_TYPES.ObjectPattern) {
33
- return;
34
- }
35
- const statusCodeProperty = variableId.properties.find(
36
- (property) => property.type === AST_NODE_TYPES.Property && property.key.type === AST_NODE_TYPES.Identifier && property.key.name === "statusCode"
37
- );
38
- if (!statusCodeProperty) {
39
- return;
40
- }
41
- if (variableInit.argument.callee.type !== AST_NODE_TYPES.Identifier || variableInit.argument.callee.name !== "fetch") {
42
- const variableNode = parserServices.esTreeNodeToTSNodeMap.get(variableId);
43
- const variableType = typeChecker.getTypeAtLocation(variableNode);
44
- if (!isFetchResponse(variableType)) {
45
- return;
46
- }
47
- }
48
- try {
49
- context.report({
50
- node: statusCodeProperty,
51
- messageId: "renameStatusCodeProperty",
52
- fix(fixer) {
53
- return statusCodeProperty.shorthand ? fixer.replaceText(statusCodeProperty, "status: statusCode") : fixer.replaceText(statusCodeProperty.key, "status");
54
- }
55
- });
56
- } catch (error) {
57
- console.error(`Failed to apply ${ruleId} rule for file "${context.filename}":`, error);
58
- context.report({
59
- node: statusCodeProperty,
60
- messageId: "unknownError",
61
- data: {
62
- fileName: context.filename,
63
- error: error instanceof Error ? error.toString() : JSON.stringify(error)
64
- }
65
- });
66
- }
67
- }
68
- };
69
- }
70
- });
71
- var fetch_response_status_default = rule;
72
- export {
73
- fetch_response_status_default as default,
74
- ruleId
75
- };
76
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FnZW50L2ZldGNoLXJlc3BvbnNlLXN0YXR1cy50cyJdLAogICJtYXBwaW5ncyI6ICI7QUFRQSxTQUFTLGdCQUFnQixtQkFBNkI7QUFFdEQsT0FBTyx5QkFBeUI7QUFDaEMsU0FBUyx1QkFBdUI7QUFFekIsSUFBTSxTQUFTO0FBRXRCLElBQU0sYUFBYSxZQUFZLFlBQVksQ0FBQyxTQUFTLG9CQUFvQixJQUFJLENBQUM7QUFFOUUsSUFBTSxPQUE0RSxXQUFXO0FBQUEsRUFDM0YsTUFBTTtBQUFBLEVBQ04sTUFBTTtBQUFBLElBQ0osTUFBTTtBQUFBLElBQ04sTUFBTTtBQUFBLE1BQ0osYUFBYTtBQUFBLElBQ2Y7QUFBQSxJQUNBLFVBQVU7QUFBQSxNQUNSLDBCQUEwQjtBQUFBLE1BQzFCLGNBQWM7QUFBQSxJQUNoQjtBQUFBLElBQ0EsU0FBUztBQUFBLElBQ1QsUUFBUSxDQUFDO0FBQUEsRUFDWDtBQUFBLEVBQ0EsZ0JBQWdCLENBQUM7QUFBQSxFQUNqQixPQUFPLFNBQVM7QUFDZCxVQUFNLGlCQUFpQixZQUFZLGtCQUFrQixPQUFPO0FBQzVELFVBQU0sY0FBYyxlQUFlLFFBQVEsZUFBZTtBQUUxRCxXQUFPO0FBQUEsTUFDTCxxQkFBcUIsQ0FBQyx3QkFBc0Q7QUFDMUUsY0FBTSxlQUFlLG9CQUFvQixhQUFhLENBQUMsR0FBRztBQUMxRCxZQUNFLENBQUMsZ0JBQ0QsYUFBYSxTQUFTLGVBQWUsbUJBQ3JDLGFBQWEsU0FBUyxTQUFTLGVBQWUsZ0JBQzlDO0FBQ0E7QUFBQSxRQUNGO0FBRUEsY0FBTSxhQUFhLG9CQUFvQixhQUFhLENBQUMsR0FBRztBQUN4RCxZQUFJLFdBQVcsU0FBUyxlQUFlLGVBQWU7QUFDcEQ7QUFBQSxRQUNGO0FBQ0EsY0FBTSxxQkFBcUIsV0FBVyxXQUFXO0FBQUEsVUFDL0MsQ0FBQyxhQUNDLFNBQVMsU0FBUyxlQUFlLFlBQ2pDLFNBQVMsSUFBSSxTQUFTLGVBQWUsY0FDckMsU0FBUyxJQUFJLFNBQVM7QUFBQSxRQUMxQjtBQUNBLFlBQUksQ0FBQyxvQkFBb0I7QUFDdkI7QUFBQSxRQUNGO0FBRUEsWUFDRSxhQUFhLFNBQVMsT0FBTyxTQUFTLGVBQWUsY0FDckQsYUFBYSxTQUFTLE9BQU8sU0FBUyxTQUN0QztBQUNBLGdCQUFNLGVBQWUsZUFBZSxzQkFBc0IsSUFBSSxVQUFVO0FBQ3hFLGdCQUFNLGVBQWUsWUFBWSxrQkFBa0IsWUFBWTtBQUMvRCxjQUFJLENBQUMsZ0JBQWdCLFlBQVksR0FBRztBQUNsQztBQUFBLFVBQ0Y7QUFBQSxRQUNGO0FBRUEsWUFBSTtBQUNGLGtCQUFRLE9BQU87QUFBQSxZQUNiLE1BQU07QUFBQSxZQUNOLFdBQVc7QUFBQSxZQUNYLElBQUksT0FBTztBQUNULHFCQUFPLG1CQUFtQixZQUN0QixNQUFNLFlBQVksb0JBQW9CLG9CQUFvQixJQUMxRCxNQUFNLFlBQVksbUJBQW1CLEtBQUssUUFBUTtBQUFBLFlBQ3hEO0FBQUEsVUFDRixDQUFDO0FBQUEsUUFDSCxTQUFTLE9BQU87QUFFZCxrQkFBUSxNQUFNLG1CQUFtQixNQUFNLG1CQUFtQixRQUFRLFFBQVEsTUFBTSxLQUFLO0FBQ3JGLGtCQUFRLE9BQU87QUFBQSxZQUNiLE1BQU07QUFBQSxZQUNOLFdBQVc7QUFBQSxZQUNYLE1BQU07QUFBQSxjQUNKLFVBQVUsUUFBUTtBQUFBLGNBQ2xCLE9BQU8saUJBQWlCLFFBQVEsTUFBTSxTQUFTLElBQUksS0FBSyxVQUFVLEtBQUs7QUFBQSxZQUN6RTtBQUFBLFVBQ0YsQ0FBQztBQUFBLFFBQ0g7QUFBQSxNQUNGO0FBQUEsSUFDRjtBQUFBLEVBQ0Y7QUFDRixDQUFDO0FBRUQsSUFBTyxnQ0FBUTsiLAogICJuYW1lcyI6IFtdCn0K
@@ -1,205 +0,0 @@
1
- // src/agent/fetch-then.ts
2
- import { strict as assert } from "node:assert";
3
- import { AST_NODE_TYPES, ESLintUtils } from "@typescript-eslint/utils";
4
- import { getEnclosingFunction, getParent, isUsedInArrayOrAsArgument } from "../library/ts-tree.mjs";
5
- import getDocumentationUrl from "../get-documentation-url.mjs";
6
- import { getIndentation } from "../library/format.mjs";
7
- import { isValidPropertyName } from "../library/variable.mjs";
8
- import { hasAssertions } from "./fetch.mjs";
9
- import { replaceEndpointUrlPrefixWithBasePath } from "./url.mjs";
10
- var ruleId = "fetch-then";
11
- function analyzeFixtureCall(call, results, sourceCode) {
12
- const parent = getParent(call);
13
- if (!parent) {
14
- return;
15
- }
16
- let nextCall;
17
- if (parent.type !== AST_NODE_TYPES.MemberExpression) {
18
- results.fixtureNode = call;
19
- return;
20
- }
21
- if (parent.property.type === AST_NODE_TYPES.Identifier) {
22
- if (parent.property.name === "expect") {
23
- const assertionCall = getParent(parent);
24
- assert.ok(assertionCall && assertionCall.type === AST_NODE_TYPES.CallExpression);
25
- results.assertions = [...results.assertions ?? [], assertionCall.arguments];
26
- nextCall = assertionCall;
27
- } else if (parent.property.name === "send") {
28
- const sendRequestBodyCall = getParent(parent);
29
- assert.ok(sendRequestBodyCall && sendRequestBodyCall.type === AST_NODE_TYPES.CallExpression);
30
- results.requestBody = sendRequestBodyCall.arguments[0];
31
- nextCall = sendRequestBodyCall;
32
- } else if (parent.property.name === "set") {
33
- const setRequestHeaderCall = getParent(parent);
34
- assert.ok(setRequestHeaderCall && setRequestHeaderCall.type === AST_NODE_TYPES.CallExpression);
35
- const [name, value] = setRequestHeaderCall.arguments;
36
- results.requestHeaders = [...results.requestHeaders ?? [], { name, value }];
37
- nextCall = setRequestHeaderCall;
38
- }
39
- } else {
40
- throw new Error(`Unexpected TSESTree.Expression in fixture/supertest call ${sourceCode.getText(parent)}.`);
41
- }
42
- if (nextCall) {
43
- analyzeFixtureCall(nextCall, results, sourceCode);
44
- }
45
- }
46
- function createResponseAssertions(fixtureCallInformation, sourceCode, responseVariableName) {
47
- let statusAssertion;
48
- const nonStatusAssertions = [];
49
- for (const expectArguments of fixtureCallInformation.assertions ?? []) {
50
- if (expectArguments.length === 1) {
51
- const [assertionArgument] = expectArguments;
52
- assert.ok(assertionArgument);
53
- if (assertionArgument.type === AST_NODE_TYPES.MemberExpression && assertionArgument.object.type === AST_NODE_TYPES.Identifier && assertionArgument.object.name === "StatusCodes" || assertionArgument.type === AST_NODE_TYPES.Literal || sourceCode.getText(assertionArgument).includes("StatusCodes.")) {
54
- statusAssertion = `assert.equal(${responseVariableName}.status, ${sourceCode.getText(assertionArgument)})`;
55
- } else if (assertionArgument.type === AST_NODE_TYPES.ArrowFunctionExpression) {
56
- let functionBody = sourceCode.getText(assertionArgument.body);
57
- const [originalResponseArgument] = assertionArgument.params;
58
- assert.ok(originalResponseArgument?.type === AST_NODE_TYPES.Identifier);
59
- const originalResponseArgumentName = originalResponseArgument.name;
60
- if (originalResponseArgumentName !== responseVariableName) {
61
- functionBody = functionBody.replace(
62
- new RegExp(`\\b${originalResponseArgumentName}\\b`, "ug"),
63
- responseVariableName
64
- );
65
- }
66
- nonStatusAssertions.push(`assert.doesNotThrow(()=>${functionBody})`);
67
- } else if (assertionArgument.type === AST_NODE_TYPES.Identifier) {
68
- nonStatusAssertions.push(
69
- `assert.doesNotThrow(()=>${sourceCode.getText(assertionArgument)}(${responseVariableName}))`
70
- );
71
- } else if (assertionArgument.type === AST_NODE_TYPES.ObjectExpression || assertionArgument.type === AST_NODE_TYPES.CallExpression) {
72
- nonStatusAssertions.push(
73
- `assert.deepEqual(await ${responseVariableName}.json(), ${sourceCode.getText(assertionArgument)})`
74
- );
75
- } else {
76
- throw new Error(`Unexpected Supertest assertion argument: ".expect(${sourceCode.getText(assertionArgument)})`);
77
- }
78
- } else if (expectArguments.length === 2) {
79
- const [headerName, headerValue] = expectArguments;
80
- assert.ok(headerName && headerValue);
81
- const headersReference = `${responseVariableName}.headers`;
82
- if (headerValue.type === AST_NODE_TYPES.Literal && headerValue.value instanceof RegExp) {
83
- nonStatusAssertions.push(
84
- `assert.ok(${headersReference}.get(${sourceCode.getText(headerName)}).match(${sourceCode.getText(headerValue)}))`
85
- );
86
- } else {
87
- nonStatusAssertions.push(
88
- `assert.equal(${headersReference}.get(${sourceCode.getText(headerName)}), ${sourceCode.getText(headerValue)})`
89
- );
90
- }
91
- }
92
- }
93
- return {
94
- statusAssertion,
95
- nonStatusAssertions
96
- };
97
- }
98
- var createRule = ESLintUtils.RuleCreator((name) => getDocumentationUrl(name));
99
- var rule = createRule({
100
- name: ruleId,
101
- meta: {
102
- type: "suggestion",
103
- docs: {
104
- description: "Prefer native fetch API over customized fixture API.",
105
- url: getDocumentationUrl(ruleId)
106
- },
107
- messages: {
108
- preferNativeFetch: "Prefer native fetch API over customized fixture API.",
109
- // shouldUseHeaderGetter: 'Getter should be used to access response headers.',
110
- unknownError: 'Unknown error occurred in file "{{fileName}}": {{ error }}.'
111
- },
112
- fixable: "code",
113
- schema: []
114
- },
115
- defaultOptions: [],
116
- create(context) {
117
- const sourceCode = context.sourceCode;
118
- const scopeManager = sourceCode.scopeManager;
119
- assert.ok(scopeManager);
120
- return {
121
- 'CallExpression[callee.object.object.name="fixture"][callee.object.property.name="api"]': (fixtureCall) => {
122
- try {
123
- if (!hasAssertions(fixtureCall)) {
124
- return;
125
- }
126
- if (!(isUsedInArrayOrAsArgument(fixtureCall) || getEnclosingFunction(fixtureCall)?.async === false)) {
127
- return;
128
- }
129
- const fixtureFunction = fixtureCall.callee;
130
- assert.ok(fixtureFunction.type === AST_NODE_TYPES.MemberExpression);
131
- const indentation = getIndentation(fixtureCall, sourceCode);
132
- const [urlArgumentNode] = fixtureCall.arguments;
133
- assert.ok(urlArgumentNode !== void 0);
134
- const fixtureCallInformation = {};
135
- analyzeFixtureCall(fixtureCall, fixtureCallInformation, sourceCode);
136
- const originalUrlArgumentText = sourceCode.getText(urlArgumentNode);
137
- const fetchUrlArgumentText = replaceEndpointUrlPrefixWithBasePath(originalUrlArgumentText);
138
- const methodNode = fixtureFunction.property;
139
- assert.ok(methodNode.type === AST_NODE_TYPES.Identifier);
140
- const fetchRequestArgumentLines = [
141
- "{",
142
- ` method: '${methodNode.name.toUpperCase()}',`,
143
- ...fixtureCallInformation.requestBody ? [` body: JSON.stringify(${sourceCode.getText(fixtureCallInformation.requestBody)}),`] : [],
144
- ...fixtureCallInformation.requestHeaders ? [
145
- ` headers: {`,
146
- ...fixtureCallInformation.requestHeaders.map(
147
- ({ name, value }) => (
148
- // eslint-disable-next-line @typescript-eslint/restrict-template-expressions, no-nested-ternary, sonarjs/no-nested-template-literals
149
- ` ${name.type === AST_NODE_TYPES.Literal ? isValidPropertyName(name.value) ? name.value : `'${name.value}'` : `[${sourceCode.getText(name)}]`}: ${sourceCode.getText(value)},`
150
- )
151
- ),
152
- ` },`
153
- ] : [],
154
- "}"
155
- ].join(`
156
- ${indentation}`);
157
- const responseVariableNameToUse = "res";
158
- const { statusAssertion, nonStatusAssertions } = createResponseAssertions(
159
- fixtureCallInformation,
160
- sourceCode,
161
- responseVariableNameToUse
162
- );
163
- const disableLintComment = "// eslint-disable-next-line @checkdigit/no-promise-instance-method";
164
- const fetchCallText = `fetch(${fetchUrlArgumentText}, ${fetchRequestArgumentLines})`;
165
- const appendingAssignmentAndAssertionText = [
166
- ...statusAssertion !== void 0 ? [statusAssertion] : [],
167
- ...nonStatusAssertions
168
- ].join(`;
169
- ${indentation}`);
170
- const replacementText = fixtureCallInformation.assertions ? [
171
- disableLintComment,
172
- `${fetchCallText}.then((${responseVariableNameToUse}) => {`,
173
- appendingAssignmentAndAssertionText === "" ? "" : ` ${appendingAssignmentAndAssertionText};`,
174
- ` return ${responseVariableNameToUse};`,
175
- `})`
176
- ].join(`
177
- ${indentation}`) : fetchCallText;
178
- context.report({
179
- node: fixtureCall,
180
- messageId: "preferNativeFetch",
181
- fix(fixer) {
182
- return fixer.replaceText(fixtureCallInformation.fixtureNode, replacementText);
183
- }
184
- });
185
- } catch (error) {
186
- console.error(`Failed to apply ${ruleId} rule for file "${context.filename}":`, error);
187
- context.report({
188
- node: fixtureCall,
189
- messageId: "unknownError",
190
- data: {
191
- fileName: context.filename,
192
- error: error instanceof Error ? error.toString() : JSON.stringify(error)
193
- }
194
- });
195
- }
196
- }
197
- };
198
- }
199
- });
200
- var fetch_then_default = rule;
201
- export {
202
- fetch_then_default as default,
203
- ruleId
204
- };
205
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FnZW50L2ZldGNoLXRoZW4udHMiXSwKICAibWFwcGluZ3MiOiAiO0FBUUEsU0FBUyxVQUFVLGNBQWM7QUFHakMsU0FBUyxnQkFBZ0IsbUJBQTZCO0FBR3RELFNBQVMsc0JBQXNCLFdBQVcsaUNBQWlDO0FBQzNFLE9BQU8seUJBQXlCO0FBQ2hDLFNBQVMsc0JBQXNCO0FBQy9CLFNBQVMsMkJBQTJCO0FBQ3BDLFNBQVMscUJBQXFCO0FBQzlCLFNBQVMsNENBQTRDO0FBRTlDLElBQU0sU0FBUztBQVV0QixTQUFTLG1CQUFtQixNQUErQixTQUFpQyxZQUF3QjtBQUNsSCxRQUFNLFNBQVMsVUFBVSxJQUFJO0FBQzdCLE1BQUksQ0FBQyxRQUFRO0FBQ1g7QUFBQSxFQUNGO0FBRUEsTUFBSTtBQUNKLE1BQUksT0FBTyxTQUFTLGVBQWUsa0JBQWtCO0FBQ25ELFlBQVEsY0FBYztBQUN0QjtBQUFBLEVBQ0Y7QUFFQSxNQUFJLE9BQU8sU0FBUyxTQUFTLGVBQWUsWUFBWTtBQUN0RCxRQUFJLE9BQU8sU0FBUyxTQUFTLFVBQVU7QUFFckMsWUFBTSxnQkFBZ0IsVUFBVSxNQUFNO0FBQ3RDLGFBQU8sR0FBRyxpQkFBaUIsY0FBYyxTQUFTLGVBQWUsY0FBYztBQUMvRSxjQUFRLGFBQWEsQ0FBQyxHQUFJLFFBQVEsY0FBYyxDQUFDLEdBQUksY0FBYyxTQUFrQztBQUNyRyxpQkFBVztBQUFBLElBQ2IsV0FBVyxPQUFPLFNBQVMsU0FBUyxRQUFRO0FBRTFDLFlBQU0sc0JBQXNCLFVBQVUsTUFBTTtBQUM1QyxhQUFPLEdBQUcsdUJBQXVCLG9CQUFvQixTQUFTLGVBQWUsY0FBYztBQUMzRixjQUFRLGNBQWMsb0JBQW9CLFVBQVUsQ0FBQztBQUNyRCxpQkFBVztBQUFBLElBQ2IsV0FBVyxPQUFPLFNBQVMsU0FBUyxPQUFPO0FBRXpDLFlBQU0sdUJBQXVCLFVBQVUsTUFBTTtBQUM3QyxhQUFPLEdBQUcsd0JBQXdCLHFCQUFxQixTQUFTLGVBQWUsY0FB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
@@ -1,48 +0,0 @@
1
- // src/agent/fetch.ts
2
- import { AST_NODE_TYPES } from "@typescript-eslint/utils";
3
- import "typescript";
4
- import { getParent, isBlockStatement } from "../library/ts-tree.mjs";
5
- function getResponseBodyRetrievalText(responseVariableName) {
6
- return `await ${responseVariableName}.json()`;
7
- }
8
- function getResponseStatusRetrievalText(responseVariableName) {
9
- return `${responseVariableName}.status`;
10
- }
11
- function getResponseHeadersRetrievalText(responseVariableName) {
12
- return `${responseVariableName}.headers`;
13
- }
14
- function isInvalidResponseHeadersAccess(responseHeadersAccess) {
15
- const responseHeaderAccessParent = getParent(responseHeadersAccess);
16
- if (responseHeaderAccessParent?.type === AST_NODE_TYPES.VariableDeclarator) {
17
- return false;
18
- }
19
- if (responseHeaderAccessParent?.type === AST_NODE_TYPES.CallExpression && responseHeaderAccessParent.callee.type === AST_NODE_TYPES.MemberExpression && responseHeaderAccessParent.callee.property.type === AST_NODE_TYPES.Identifier && responseHeaderAccessParent.callee.property.name === "get") {
20
- return true;
21
- }
22
- return !(responseHeaderAccessParent?.type === AST_NODE_TYPES.MemberExpression && responseHeaderAccessParent.property.type === AST_NODE_TYPES.Identifier && responseHeaderAccessParent.property.name === "get");
23
- }
24
- function hasAssertions(fixtureCall) {
25
- if (isBlockStatement(fixtureCall)) {
26
- return false;
27
- }
28
- const parent = getParent(fixtureCall);
29
- if (!parent) {
30
- return false;
31
- }
32
- if (parent.type === AST_NODE_TYPES.MemberExpression && parent.property.type === AST_NODE_TYPES.Identifier && parent.property.name === "expect" && getParent(parent)?.type === AST_NODE_TYPES.CallExpression) {
33
- return true;
34
- }
35
- return hasAssertions(parent);
36
- }
37
- function isFetchResponse(type) {
38
- return type.getProperties().some((symbol) => symbol.name === "body") && type.getProperties().some((symbol) => symbol.name === "json");
39
- }
40
- export {
41
- getResponseBodyRetrievalText,
42
- getResponseHeadersRetrievalText,
43
- getResponseStatusRetrievalText,
44
- hasAssertions,
45
- isFetchResponse,
46
- isInvalidResponseHeadersAccess
47
- };
48
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FnZW50L2ZldGNoLnRzIl0sCiAgIm1hcHBpbmdzIjogIjtBQUVBLFNBQVMsc0JBQWdDO0FBQ3pDLE9BQWU7QUFFZixTQUFTLFdBQVcsd0JBQXdCO0FBRXJDLFNBQVMsNkJBQTZCLHNCQUE4QjtBQUN6RSxTQUFPLFNBQVMsb0JBQW9CO0FBQ3RDO0FBRU8sU0FBUywrQkFBK0Isc0JBQThCO0FBQzNFLFNBQU8sR0FBRyxvQkFBb0I7QUFDaEM7QUFFTyxTQUFTLGdDQUFnQyxzQkFBOEI7QUFDNUUsU0FBTyxHQUFHLG9CQUFvQjtBQUNoQztBQUVPLFNBQVMsK0JBQStCLHVCQUErQztBQUM1RixRQUFNLDZCQUE2QixVQUFVLHFCQUFxQjtBQUNsRSxNQUFJLDRCQUE0QixTQUFTLGVBQWUsb0JBQW9CO0FBQzFFLFdBQU87QUFBQSxFQUNUO0FBRUEsTUFDRSw0QkFBNEIsU0FBUyxlQUFlLGtCQUNwRCwyQkFBMkIsT0FBTyxTQUFTLGVBQWUsb0JBQzFELDJCQUEyQixPQUFPLFNBQVMsU0FBUyxlQUFlLGNBQ25FLDJCQUEyQixPQUFPLFNBQVMsU0FBUyxPQUNwRDtBQUNBLFdBQU87QUFBQSxFQUNUO0FBRUEsU0FBTyxFQUNMLDRCQUE0QixTQUFTLGVBQWUsb0JBQ3BELDJCQUEyQixTQUFTLFNBQVMsZUFBZSxjQUM1RCwyQkFBMkIsU0FBUyxTQUFTO0FBRWpEO0FBRU8sU0FBUyxjQUFjLGFBQXFDO0FBQ2pFLE1BQUksaUJBQWlCLFdBQVcsR0FBRztBQUNqQyxXQUFPO0FBQUEsRUFDVDtBQUVBLFFBQU0sU0FBUyxVQUFVLFdBQVc7QUFDcEMsTUFBSSxDQUFDLFFBQVE7QUFDWCxXQUFPO0FBQUEsRUFDVDtBQUVBLE1BQ0UsT0FBTyxTQUFTLGVBQWUsb0JBQy9CLE9BQU8sU0FBUyxTQUFTLGVBQWUsY0FDeEMsT0FBTyxTQUFTLFNBQVMsWUFDekIsVUFBVSxNQUFNLEdBQUcsU0FBUyxlQUFlLGdCQUMzQztBQUNBLFdBQU87QUFBQSxFQUNUO0FBRUEsU0FBTyxjQUFjLE1BQU07QUFDN0I7QUFFTyxTQUFTLGdCQUFnQixNQUF3QjtBQUN0RCxTQUNFLEtBQUssY0FBYyxFQUFFLEtBQUssQ0FBQyxXQUFXLE9BQU8sU0FBUyxNQUFNLEtBQzVELEtBQUssY0FBYyxFQUFFLEtBQUssQ0FBQyxXQUFXLE9BQU8sU0FBUyxNQUFNO0FBRWhFOyIsCiAgIm5hbWVzIjogW10KfQo=