eslint-plugin-jest
Version:
ESLint rules for Jest
86 lines (84 loc) • 2.88 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _utils = require("@typescript-eslint/utils");
var _utils2 = require("./utils");
const toThrowMatchers = ['toThrow', 'toThrowError', 'toThrowErrorMatchingSnapshot', 'toThrowErrorMatchingInlineSnapshot'];
const baseRule = (() => {
try {
// eslint-disable-next-line @typescript-eslint/no-require-imports
const TSESLintPlugin = require('@typescript-eslint/eslint-plugin');
return TSESLintPlugin.rules['unbound-method'];
} catch (e) {
const error = e;
if (error.code === 'MODULE_NOT_FOUND') {
return null;
}
throw error;
}
})();
const DEFAULT_MESSAGE = 'This rule requires `@typescript-eslint/eslint-plugin`';
var _default = exports.default = (0, _utils2.createRule)({
defaultOptions: [{
ignoreStatic: false
}],
...baseRule,
name: __filename,
meta: {
messages: {
unbound: DEFAULT_MESSAGE,
unboundWithoutThisAnnotation: DEFAULT_MESSAGE
},
schema: [],
type: 'problem',
...baseRule?.meta,
docs: {
description: 'Enforce unbound methods are called with their expected scope',
...baseRule?.meta.docs
}
},
create(context) {
const baseSelectors = baseRule?.create(context);
if (!baseSelectors) {
return {};
}
/**
* Checks if a MemberExpression is an argument to a `jest.mocked()` call.
* This handles cases like `jest.mocked(service.method)` where `service.method`
* should not be flagged as an unbound method.
*/
const isArgumentToJestMocked = node => {
// Check if the immediate parent is a CallExpression
if (node.parent?.type !== _utils.AST_NODE_TYPES.CallExpression) {
return false;
}
const parentCall = node.parent;
return parentCall.callee.type === _utils.AST_NODE_TYPES.MemberExpression && (0, _utils2.isSupportedAccessor)(parentCall.callee.object, 'jest') && (0, _utils2.isSupportedAccessor)(parentCall.callee.property, 'mocked');
};
return {
...baseSelectors,
MemberExpression(node) {
if (isArgumentToJestMocked(node)) {
return;
}
if (node.parent?.type === _utils.AST_NODE_TYPES.CallExpression) {
const jestFnCall = (0, _utils2.parseJestFnCall)((0, _utils2.findTopMostCallExpression)(node.parent), context);
if (jestFnCall?.type === 'jest' && jestFnCall.members.length >= 1 && (0, _utils2.isIdentifier)(jestFnCall.members[0], 'mocked')) {
return;
}
if (jestFnCall?.type === 'expect') {
const {
matcher
} = jestFnCall;
if (!toThrowMatchers.includes((0, _utils2.getAccessorValue)(matcher))) {
return;
}
}
}
baseSelectors.MemberExpression?.(node);
}
};
}
});