You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
141 lines
5.1 KiB
141 lines
5.1 KiB
"use strict";
|
|
|
|
Object.defineProperty(exports, "__esModule", {
|
|
value: true
|
|
});
|
|
exports.default = void 0;
|
|
|
|
var _experimentalUtils = require("@typescript-eslint/experimental-utils");
|
|
|
|
var _utils = require("./utils");
|
|
|
|
const isThenOrCatchCall = node => node.type === _experimentalUtils.AST_NODE_TYPES.CallExpression && node.callee.type === _experimentalUtils.AST_NODE_TYPES.MemberExpression && (0, _utils.isSupportedAccessor)(node.callee.property) && ['then', 'catch'].includes((0, _utils.getAccessorValue)(node.callee.property));
|
|
|
|
const isExpectCallPresentInFunction = body => {
|
|
if (body.type === _experimentalUtils.AST_NODE_TYPES.BlockStatement) {
|
|
return body.body.find(line => {
|
|
if (line.type === _experimentalUtils.AST_NODE_TYPES.ExpressionStatement) {
|
|
return isFullExpectCall(line.expression);
|
|
}
|
|
|
|
if (line.type === _experimentalUtils.AST_NODE_TYPES.ReturnStatement && line.argument) {
|
|
return isFullExpectCall(line.argument);
|
|
}
|
|
|
|
return false;
|
|
});
|
|
}
|
|
|
|
return isFullExpectCall(body);
|
|
};
|
|
|
|
const isFullExpectCall = expression => expression.type === _experimentalUtils.AST_NODE_TYPES.CallExpression && (0, _utils.isExpectMember)(expression.callee);
|
|
|
|
const reportReturnRequired = (context, node) => {
|
|
context.report({
|
|
loc: {
|
|
end: {
|
|
column: node.loc.end.column,
|
|
line: node.loc.end.line
|
|
},
|
|
start: node.loc.start
|
|
},
|
|
messageId: 'returnPromise',
|
|
node
|
|
});
|
|
};
|
|
|
|
const isPromiseReturnedLater = (node, testFunctionBody) => {
|
|
let promiseName;
|
|
|
|
if (node.type === _experimentalUtils.AST_NODE_TYPES.ExpressionStatement && node.expression.type === _experimentalUtils.AST_NODE_TYPES.CallExpression && node.expression.callee.type === _experimentalUtils.AST_NODE_TYPES.MemberExpression && (0, _utils.isSupportedAccessor)(node.expression.callee.object)) {
|
|
promiseName = (0, _utils.getAccessorValue)(node.expression.callee.object);
|
|
} else if (node.type === _experimentalUtils.AST_NODE_TYPES.VariableDeclarator && node.id.type === _experimentalUtils.AST_NODE_TYPES.Identifier) {
|
|
promiseName = node.id.name;
|
|
}
|
|
|
|
const lastLineInTestFunc = testFunctionBody[testFunctionBody.length - 1];
|
|
return lastLineInTestFunc.type === _experimentalUtils.AST_NODE_TYPES.ReturnStatement && lastLineInTestFunc.argument && ('name' in lastLineInTestFunc.argument && lastLineInTestFunc.argument.name === promiseName || !promiseName);
|
|
};
|
|
|
|
const isTestFunc = node => node.type === _experimentalUtils.AST_NODE_TYPES.CallExpression && (0, _utils.isSupportedAccessor)(node.callee) && [_utils.TestCaseName.it, _utils.TestCaseName.test].includes((0, _utils.getAccessorValue)(node.callee));
|
|
|
|
const findTestFunction = node => {
|
|
while (node) {
|
|
if ((0, _utils.isFunction)(node) && node.parent && isTestFunc(node.parent)) {
|
|
return node;
|
|
}
|
|
|
|
node = node.parent;
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
const isParentThenOrPromiseReturned = (node, testFunctionBody) => node.type === _experimentalUtils.AST_NODE_TYPES.ReturnStatement || isPromiseReturnedLater(node, testFunctionBody);
|
|
|
|
const verifyExpectWithReturn = (promiseCallbacks, node, context, testFunctionBody) => {
|
|
promiseCallbacks.some(promiseCallback => {
|
|
if (promiseCallback && (0, _utils.isFunction)(promiseCallback)) {
|
|
if (isExpectCallPresentInFunction(promiseCallback.body) && node.parent.parent && !isParentThenOrPromiseReturned(node.parent.parent, testFunctionBody)) {
|
|
reportReturnRequired(context, node.parent.parent);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
});
|
|
};
|
|
|
|
const isHavingAsyncCallBackParam = testFunction => testFunction.params[0] && testFunction.params[0].type === _experimentalUtils.AST_NODE_TYPES.Identifier;
|
|
|
|
var _default = (0, _utils.createRule)({
|
|
name: __filename,
|
|
meta: {
|
|
docs: {
|
|
category: 'Best Practices',
|
|
description: 'Enforce having return statement when testing with promises',
|
|
recommended: 'error'
|
|
},
|
|
messages: {
|
|
returnPromise: 'Promise should be returned to test its fulfillment or rejection'
|
|
},
|
|
type: 'suggestion',
|
|
schema: []
|
|
},
|
|
defaultOptions: [],
|
|
|
|
create(context) {
|
|
return {
|
|
CallExpression(node) {
|
|
if (!isThenOrCatchCall(node) || node.parent && node.parent.type === _experimentalUtils.AST_NODE_TYPES.AwaitExpression) {
|
|
return;
|
|
}
|
|
|
|
const testFunction = findTestFunction(node);
|
|
|
|
if (testFunction && !isHavingAsyncCallBackParam(testFunction)) {
|
|
const {
|
|
body
|
|
} = testFunction;
|
|
|
|
if (body.type !== _experimentalUtils.AST_NODE_TYPES.BlockStatement) {
|
|
return;
|
|
}
|
|
|
|
const testFunctionBody = body.body;
|
|
const [fulfillmentCallback, rejectionCallback] = node.arguments; // then block can have two args, fulfillment & rejection
|
|
// then block can have one args, fulfillment
|
|
// catch block can have one args, rejection
|
|
// ref: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise
|
|
|
|
verifyExpectWithReturn([fulfillmentCallback, rejectionCallback], node.callee, context, testFunctionBody);
|
|
}
|
|
}
|
|
|
|
};
|
|
}
|
|
|
|
});
|
|
|
|
exports.default = _default;
|