Skip to content

Commit 439c833

Browse files
authoredFeb 4, 2020
Update: array-callback-return checks Array.forEach (fixes #12551) (#12646)
array-callback-return rule now checks if Array.forEach returns a value, and, if it does, reports an error. This feature is being added disabled by default, and can be enabled by setting the option "checkForEach" to true. Signed-off-by: Gabriel R Sezefredo <g@briel.dev>
1 parent 33efd71 commit 439c833

File tree

3 files changed

+241
-70
lines changed

3 files changed

+241
-70
lines changed
 

‎docs/rules/array-callback-return.md

+61-4
Original file line numberDiff line numberDiff line change
@@ -10,10 +10,11 @@ var indexMap = myArray.reduce(function(memo, item, index) {
1010
}, {}); // Error: cannot set property 'b' of undefined
1111
```
1212

13-
This rule enforces usage of `return` statement in callbacks of array's methods.
14-
1513
## Rule Details
1614

15+
This rule enforces usage of `return` statement in callbacks of array's methods.
16+
Additionaly, it may also enforce the `forEach` array method callback to __not__ return a value by using the `checkForEach` option.
17+
1718
This rule finds callback functions of the following methods, then checks usage of `return` statement.
1819

1920
* [`Array.from`](https://www.ecma-international.org/ecma-262/6.0/#sec-array.from)
@@ -22,6 +23,7 @@ This rule finds callback functions of the following methods, then checks usage o
2223
* [`Array.prototype.find`](https://www.ecma-international.org/ecma-262/6.0/#sec-array.prototype.find)
2324
* [`Array.prototype.findIndex`](https://www.ecma-international.org/ecma-262/6.0/#sec-array.prototype.findindex)
2425
* [`Array.prototype.flatMap`](https://www.ecma-international.org/ecma-262/10.0/#sec-array.prototype.flatmap)
26+
* [`Array.prototype.forEach`](https://www.ecma-international.org/ecma-262/6.0/#sec-array.prototype.foreach) (optional, based on `checkForEach` parameter)
2527
* [`Array.prototype.map`](https://www.ecma-international.org/ecma-262/6.0/#sec-array.prototype.map)
2628
* [`Array.prototype.reduce`](https://www.ecma-international.org/ecma-262/6.0/#sec-array.prototype.reduce)
2729
* [`Array.prototype.reduceRight`](https://www.ecma-international.org/ecma-262/6.0/#sec-array.prototype.reduceright)
@@ -75,9 +77,12 @@ var bar = foo.map(node => node.getAttribute("id"));
7577

7678
## Options
7779

78-
This rule has an object option:
80+
This rule accepts a configuration object with two options:
81+
82+
* `"allowImplicit": false` (default) When set to `true`, allows callbacks of methods that require a return value to implicitly return `undefined` with a `return` statement containing no expression.
83+
* `"checkForEach": false` (default) When set to `true`, rule will also report `forEach` callbacks that return a value.
7984

80-
* `"allowImplicit": false` (default) When set to true, allows implicitly returning `undefined` with a `return` statement containing no expression.
85+
### allowImplicit
8186

8287
Examples of **correct** code for the `{ "allowImplicit": true }` option:
8388

@@ -88,6 +93,58 @@ var undefAllTheThings = myArray.map(function(item) {
8893
});
8994
```
9095

96+
### checkForEach
97+
98+
Examples of **incorrect** code for the `{ "checkForEach": true }` option:
99+
100+
```js
101+
/*eslint array-callback-return: ["error", { checkForEach: true }]*/
102+
103+
myArray.forEach(function(item) {
104+
return handleItem(item)
105+
});
106+
107+
myArray.forEach(function(item) {
108+
if (item < 0) {
109+
return x;
110+
}
111+
handleItem(item);
112+
});
113+
114+
myArray.forEach(item => handleItem(item));
115+
116+
myArray.forEach(item => {
117+
return handleItem(item);
118+
});
119+
```
120+
121+
Examples of **correct** code for the `{ "checkForEach": true }` option:
122+
123+
```js
124+
/*eslint array-callback-return: ["error", { checkForEach: true }]*/
125+
126+
myArray.forEach(function(item) {
127+
handleItem(item)
128+
});
129+
130+
myArray.forEach(function(item) {
131+
if (item < 0) {
132+
return;
133+
}
134+
handleItem(item);
135+
});
136+
137+
myArray.forEach(function(item) {
138+
handleItem(item);
139+
return;
140+
});
141+
142+
myArray.forEach(item => {
143+
handleItem(item);
144+
});
145+
```
146+
147+
91148
## Known Limitations
92149

93150
This rule checks callback functions of methods with the given names, *even if* the object which has the method is *not* an array.

‎lib/rules/array-callback-return.js

+84-35
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ const astUtils = require("./utils/ast-utils");
1818
//------------------------------------------------------------------------------
1919

2020
const TARGET_NODE_TYPE = /^(?:Arrow)?FunctionExpression$/u;
21-
const TARGET_METHODS = /^(?:every|filter|find(?:Index)?|flatMap|map|reduce(?:Right)?|some|sort)$/u;
21+
const TARGET_METHODS = /^(?:every|filter|find(?:Index)?|flatMap|forEach|map|reduce(?:Right)?|some|sort)$/u;
2222

2323
/**
2424
* Checks a given code path segment is reachable.
@@ -61,12 +61,13 @@ function isTargetMethod(node) {
6161

6262
/**
6363
* Checks whether or not a given node is a function expression which is the
64-
* callback of an array method.
64+
* callback of an array method, returning the method name.
6565
* @param {ASTNode} node A node to check. This is one of
6666
* FunctionExpression or ArrowFunctionExpression.
67-
* @returns {boolean} `true` if the node is the callback of an array method.
67+
* @returns {string} The method name if the node is a callback method,
68+
* null otherwise.
6869
*/
69-
function isCallbackOfArrayMethod(node) {
70+
function getArrayMethodName(node) {
7071
let currentNode = node;
7172

7273
while (currentNode) {
@@ -95,7 +96,7 @@ function isCallbackOfArrayMethod(node) {
9596
const func = astUtils.getUpperFunction(parent);
9697

9798
if (func === null || !astUtils.isCallee(func)) {
98-
return false;
99+
return null;
99100
}
100101
currentNode = func.parent;
101102
break;
@@ -108,27 +109,31 @@ function isCallbackOfArrayMethod(node) {
108109
*/
109110
case "CallExpression":
110111
if (astUtils.isArrayFromMethod(parent.callee)) {
111-
return (
112+
if (
112113
parent.arguments.length >= 2 &&
113114
parent.arguments[1] === currentNode
114-
);
115+
) {
116+
return "from";
117+
}
115118
}
116119
if (isTargetMethod(parent.callee)) {
117-
return (
120+
if (
118121
parent.arguments.length >= 1 &&
119122
parent.arguments[0] === currentNode
120-
);
123+
) {
124+
return astUtils.getStaticPropertyName(parent.callee);
125+
}
121126
}
122-
return false;
127+
return null;
123128

124129
// Otherwise this node is not target.
125130
default:
126-
return false;
131+
return null;
127132
}
128133
}
129134

130135
/* istanbul ignore next: unreachable */
131-
return false;
136+
return null;
132137
}
133138

134139
//------------------------------------------------------------------------------
@@ -153,6 +158,10 @@ module.exports = {
153158
allowImplicit: {
154159
type: "boolean",
155160
default: false
161+
},
162+
checkForEach: {
163+
type: "boolean",
164+
default: false
156165
}
157166
},
158167
additionalProperties: false
@@ -162,15 +171,17 @@ module.exports = {
162171
messages: {
163172
expectedAtEnd: "Expected to return a value at the end of {{name}}.",
164173
expectedInside: "Expected to return a value in {{name}}.",
165-
expectedReturnValue: "{{name}} expected a return value."
174+
expectedReturnValue: "{{name}} expected a return value.",
175+
expectedNoReturnValue: "{{name}} did not expect a return value."
166176
}
167177
},
168178

169179
create(context) {
170180

171-
const options = context.options[0] || { allowImplicit: false };
181+
const options = context.options[0] || { allowImplicit: false, checkForEach: false };
172182

173183
let funcInfo = {
184+
arrayMethodName: null,
174185
upper: null,
175186
codePath: null,
176187
hasReturn: false,
@@ -188,18 +199,32 @@ module.exports = {
188199
* @returns {void}
189200
*/
190201
function checkLastSegment(node) {
191-
if (funcInfo.shouldCheck &&
192-
funcInfo.codePath.currentSegments.some(isReachable)
193-
) {
202+
203+
if (!funcInfo.shouldCheck) {
204+
return;
205+
}
206+
207+
let messageId = null;
208+
209+
if (funcInfo.arrayMethodName === "forEach") {
210+
if (options.checkForEach && node.type === "ArrowFunctionExpression" && node.expression) {
211+
messageId = "expectedNoReturnValue";
212+
}
213+
} else {
214+
if (node.body.type === "BlockStatement" && funcInfo.codePath.currentSegments.some(isReachable)) {
215+
messageId = funcInfo.hasReturn ? "expectedAtEnd" : "expectedInside";
216+
}
217+
}
218+
219+
if (messageId) {
220+
let name = astUtils.getFunctionNameWithKind(funcInfo.node);
221+
222+
name = messageId === "expectedNoReturnValue" ? lodash.upperFirst(name) : name;
194223
context.report({
195224
node,
196225
loc: getLocation(node, context.getSourceCode()).loc.start,
197-
messageId: funcInfo.hasReturn
198-
? "expectedAtEnd"
199-
: "expectedInside",
200-
data: {
201-
name: astUtils.getFunctionNameWithKind(funcInfo.node)
202-
}
226+
messageId,
227+
data: { name }
203228
});
204229
}
205230
}
@@ -208,14 +233,20 @@ module.exports = {
208233

209234
// Stacks this function's information.
210235
onCodePathStart(codePath, node) {
236+
237+
let methodName = null;
238+
239+
if (TARGET_NODE_TYPE.test(node.type)) {
240+
methodName = getArrayMethodName(node);
241+
}
242+
211243
funcInfo = {
244+
arrayMethodName: methodName,
212245
upper: funcInfo,
213246
codePath,
214247
hasReturn: false,
215248
shouldCheck:
216-
TARGET_NODE_TYPE.test(node.type) &&
217-
node.body.type === "BlockStatement" &&
218-
isCallbackOfArrayMethod(node) &&
249+
methodName &&
219250
!node.async &&
220251
!node.generator,
221252
node
@@ -229,20 +260,38 @@ module.exports = {
229260

230261
// Checks the return statement is valid.
231262
ReturnStatement(node) {
232-
if (funcInfo.shouldCheck) {
233-
funcInfo.hasReturn = true;
263+
264+
if (!funcInfo.shouldCheck) {
265+
return;
266+
}
267+
268+
funcInfo.hasReturn = true;
269+
270+
let messageId = null;
271+
272+
if (funcInfo.arrayMethodName === "forEach") {
273+
274+
// if checkForEach: true, returning a value at any path inside a forEach is not allowed
275+
if (options.checkForEach && node.argument) {
276+
messageId = "expectedNoReturnValue";
277+
}
278+
} else {
234279

235280
// if allowImplicit: false, should also check node.argument
236281
if (!options.allowImplicit && !node.argument) {
237-
context.report({
238-
node,
239-
messageId: "expectedReturnValue",
240-
data: {
241-
name: lodash.upperFirst(astUtils.getFunctionNameWithKind(funcInfo.node))
242-
}
243-
});
282+
messageId = "expectedReturnValue";
244283
}
245284
}
285+
286+
if (messageId) {
287+
context.report({
288+
node,
289+
messageId,
290+
data: {
291+
name: lodash.upperFirst(astUtils.getFunctionNameWithKind(funcInfo.node))
292+
}
293+
});
294+
}
246295
},
247296

248297
// Reports a given function if the last path is reachable.

‎tests/lib/rules/array-callback-return.js

+96-31
Original file line numberDiff line numberDiff line change
@@ -16,74 +16,105 @@ const ruleTester = new RuleTester();
1616

1717
const allowImplicitOptions = [{ allowImplicit: true }];
1818

19+
const checkForEachOptions = [{ checkForEach: true }];
20+
21+
const allowImplicitCheckForEach = [{ allowImplicit: true, checkForEach: true }];
22+
1923
ruleTester.run("array-callback-return", rule, {
2024
valid: [
2125

22-
// options: { allowImplicit: false }
23-
"Array.from(x, function() { return true; })",
24-
"Int32Array.from(x, function() { return true; })",
25-
{ code: "Array.from(x, function() { return true; })", options: [{ allowImplicit: false }] },
26-
{ code: "Int32Array.from(x, function() { return true; })", options: [{ allowImplicit: false }] },
26+
"foo.every(function(){}())",
27+
"foo.every(function(){ return function() { return true; }; }())",
28+
"foo.every(function(){ return function() { return; }; })",
2729

28-
// options: { allowImplicit: true }
29-
{ code: "Array.from(x, function() { return; })", options: allowImplicitOptions },
30-
{ code: "Int32Array.from(x, function() { return; })", options: allowImplicitOptions },
30+
"foo.forEach(bar || function(x) { var a=0; })",
31+
"foo.forEach(bar || function(x) { return a; })",
32+
"foo.forEach(function() {return function() { var a = 0;}}())",
33+
"foo.forEach(function(x) { var a=0; })",
34+
"foo.forEach(function(x) { return a;})",
35+
"foo.forEach(function(x) { return; })",
36+
"foo.forEach(function(x) { if (a === b) { return;} var a=0; })",
37+
"foo.forEach(function(x) { if (a === b) { return x;} var a=0; })",
38+
"foo.bar().forEach(function(x) { return; })",
39+
"[\"foo\",\"bar\",\"baz\"].forEach(function(x) { return x; })",
40+
{ code: "foo.forEach(x => { var a=0; })", parserOptions: { ecmaVersion: 6 } },
41+
{ code: "foo.forEach(x => { if (a === b) { return;} var a=0; })", parserOptions: { ecmaVersion: 6 } },
42+
{ code: "foo.forEach(x => x)", parserOptions: { ecmaVersion: 6 } },
43+
{ code: "foo.forEach(val => y += val)", parserOptions: { ecmaVersion: 6 } },
3144

32-
"Arrow.from(x, function() {})",
45+
{ code: "foo.map(async function(){})", parserOptions: { ecmaVersion: 8 } },
46+
{ code: "foo.map(async () => {})", parserOptions: { ecmaVersion: 8 } },
47+
{ code: "foo.map(function* () {})", parserOptions: { ecmaVersion: 6 } },
3348

3449
// options: { allowImplicit: false }
50+
{ code: "Array.from(x, function() { return true; })", options: [{ allowImplicit: false }] },
51+
{ code: "Int32Array.from(x, function() { return true; })", options: [{ allowImplicit: false }] },
3552
"foo.every(function() { return true; })",
3653
"foo.filter(function() { return true; })",
3754
"foo.find(function() { return true; })",
3855
"foo.findIndex(function() { return true; })",
3956
"foo.flatMap(function() { return true; })",
57+
"foo.forEach(function() { return; })",
4058
"foo.map(function() { return true; })",
4159
"foo.reduce(function() { return true; })",
4260
"foo.reduceRight(function() { return true; })",
4361
"foo.some(function() { return true; })",
4462
"foo.sort(function() { return 0; })",
63+
{ code: "foo.every(() => { return true; })", parserOptions: { ecmaVersion: 6 } },
64+
"foo.every(function() { if (a) return true; else return false; })",
65+
"foo.every(function() { switch (a) { case 0: bar(); default: return true; } })",
66+
"foo.every(function() { try { bar(); return true; } catch (err) { return false; } })",
67+
"foo.every(function() { try { bar(); } finally { return true; } })",
4568

4669
// options: { allowImplicit: true }
70+
{ code: "Array.from(x, function() { return; })", options: allowImplicitOptions },
71+
{ code: "Int32Array.from(x, function() { return; })", options: allowImplicitOptions },
4772
{ code: "foo.every(function() { return; })", options: allowImplicitOptions },
4873
{ code: "foo.filter(function() { return; })", options: allowImplicitOptions },
4974
{ code: "foo.find(function() { return; })", options: allowImplicitOptions },
5075
{ code: "foo.findIndex(function() { return; })", options: allowImplicitOptions },
5176
{ code: "foo.flatMap(function() { return; })", options: allowImplicitOptions },
77+
{ code: "foo.forEach(function() { return; })", options: allowImplicitOptions },
5278
{ code: "foo.map(function() { return; })", options: allowImplicitOptions },
5379
{ code: "foo.reduce(function() { return; })", options: allowImplicitOptions },
5480
{ code: "foo.reduceRight(function() { return; })", options: allowImplicitOptions },
5581
{ code: "foo.some(function() { return; })", options: allowImplicitOptions },
5682
{ code: "foo.sort(function() { return; })", options: allowImplicitOptions },
83+
{ code: "foo.every(() => { return; })", options: allowImplicitOptions, parserOptions: { ecmaVersion: 6 } },
84+
{ code: "foo.every(function() { if (a) return; else return a; })", options: allowImplicitOptions },
85+
{ code: "foo.every(function() { switch (a) { case 0: bar(); default: return; } })", options: allowImplicitOptions },
86+
{ code: "foo.every(function() { try { bar(); return; } catch (err) { return; } })", options: allowImplicitOptions },
87+
{ code: "foo.every(function() { try { bar(); } finally { return; } })", options: allowImplicitOptions },
88+
89+
// options: { checkForEach: true }
90+
{ code: "foo.forEach(function(x) { return; })", options: checkForEachOptions },
91+
{ code: "foo.forEach(function(x) { var a=0; })", options: checkForEachOptions },
92+
{ code: "foo.forEach(function(x) { if (a === b) { return;} var a=0; })", options: checkForEachOptions },
93+
{ code: "foo.forEach(function() {return function() { if (a == b) { return; }}}())", options: checkForEachOptions },
94+
{ code: "foo.forEach(x => { var a=0; })", options: checkForEachOptions, parserOptions: { ecmaVersion: 6 } },
95+
{ code: "foo.forEach(x => { if (a === b) { return;} var a=0; })", options: checkForEachOptions, parserOptions: { ecmaVersion: 6 } },
96+
{ code: "foo.forEach(x => { x })", options: checkForEachOptions, parserOptions: { ecmaVersion: 6 } },
97+
{ code: "foo.forEach(bar || function(x) { return; })", options: checkForEachOptions },
98+
{ code: "Array.from(x, function() { return true; })", options: checkForEachOptions },
99+
{ code: "Int32Array.from(x, function() { return true; })", options: checkForEachOptions },
100+
{ code: "foo.every(() => { return true; })", options: checkForEachOptions, parserOptions: { ecmaVersion: 6 } },
101+
{ code: "foo.every(function() { if (a) return 1; else return a; })", options: checkForEachOptions },
102+
{ code: "foo.every(function() { switch (a) { case 0: return bar(); default: return a; } })", options: checkForEachOptions },
103+
{ code: "foo.every(function() { try { bar(); return 1; } catch (err) { return err; } })", options: checkForEachOptions },
104+
{ code: "foo.every(function() { try { bar(); } finally { return 1; } })", options: checkForEachOptions },
105+
{ code: "foo.every(function() { return; })", options: allowImplicitCheckForEach },
57106

107+
"Arrow.from(x, function() {})",
58108
"foo.abc(function() {})",
59109
"every(function() {})",
60110
"foo[every](function() {})",
61111
"var every = function() {}",
62112
{ code: "foo[`${every}`](function() {})", parserOptions: { ecmaVersion: 6 } },
63-
{ code: "foo.every(() => true)", parserOptions: { ecmaVersion: 6 } },
64-
65-
// options: { allowImplicit: false }
66-
{ code: "foo.every(() => { return true; })", parserOptions: { ecmaVersion: 6 } },
67-
"foo.every(function() { if (a) return true; else return false; })",
68-
"foo.every(function() { switch (a) { case 0: bar(); default: return true; } })",
69-
"foo.every(function() { try { bar(); return true; } catch (err) { return false; } })",
70-
"foo.every(function() { try { bar(); } finally { return true; } })",
113+
{ code: "foo.every(() => true)", parserOptions: { ecmaVersion: 6 } }
71114

72-
// options: { allowImplicit: true }
73-
{ code: "foo.every(() => { return; })", options: allowImplicitOptions, parserOptions: { ecmaVersion: 6 } },
74-
{ code: "foo.every(function() { if (a) return; else return a; })", options: allowImplicitOptions },
75-
{ code: "foo.every(function() { switch (a) { case 0: bar(); default: return; } })", options: allowImplicitOptions },
76-
{ code: "foo.every(function() { try { bar(); return; } catch (err) { return; } })", options: allowImplicitOptions },
77-
{ code: "foo.every(function() { try { bar(); } finally { return; } })", options: allowImplicitOptions },
78-
79-
"foo.every(function(){}())",
80-
"foo.every(function(){ return function() { return true; }; }())",
81-
"foo.every(function(){ return function() { return; }; })",
82-
{ code: "foo.map(async function(){})", parserOptions: { ecmaVersion: 8 } },
83-
{ code: "foo.map(async () => {})", parserOptions: { ecmaVersion: 8 } },
84-
{ code: "foo.map(function* () {})", parserOptions: { ecmaVersion: 6 } }
85115
],
86116
invalid: [
117+
87118
{ code: "Array.from(x, function() {})", errors: [{ messageId: "expectedInside", data: { name: "function" } }] },
88119
{ code: "Array.from(x, function foo() {})", errors: [{ messageId: "expectedInside", data: { name: "function 'foo'" } }] },
89120
{ code: "Int32Array.from(x, function() {})", errors: [{ messageId: "expectedInside", data: { name: "function" } }] },
@@ -134,6 +165,40 @@ ruleTester.run("array-callback-return", rule, {
134165
{ code: "foo.every(function(){ return function() {}; }())", errors: [{ message: "Expected to return a value in function.", column: 30 }] },
135166
{ code: "foo.every(function(){ return function foo() {}; }())", errors: [{ message: "Expected to return a value in function 'foo'.", column: 39 }] },
136167
{ code: "foo.every(() => {})", options: [{ allowImplicit: false }], parserOptions: { ecmaVersion: 6 }, errors: [{ message: "Expected to return a value in arrow function." }] },
137-
{ code: "foo.every(() => {})", options: [{ allowImplicit: true }], parserOptions: { ecmaVersion: 6 }, errors: [{ message: "Expected to return a value in arrow function." }] }
168+
{ code: "foo.every(() => {})", options: [{ allowImplicit: true }], parserOptions: { ecmaVersion: 6 }, errors: [{ message: "Expected to return a value in arrow function." }] },
169+
170+
// options: { allowImplicit: true }
171+
{ code: "Array.from(x, function() {})", options: allowImplicitOptions, errors: [{ messageId: "expectedInside", data: { name: "function" } }] },
172+
{ code: "foo.every(function() {})", options: allowImplicitOptions, errors: [{ messageId: "expectedInside", data: { name: "function" } }] },
173+
{ code: "foo.filter(function foo() {})", options: allowImplicitOptions, errors: [{ messageId: "expectedInside", data: { name: "function 'foo'" } }] },
174+
{ code: "foo.find(function foo() {})", options: allowImplicitOptions, errors: [{ messageId: "expectedInside", data: { name: "function 'foo'" } }] },
175+
{ code: "foo.map(function() {})", options: allowImplicitOptions, errors: [{ messageId: "expectedInside", data: { name: "function" } }] },
176+
{ code: "foo.reduce(function() {})", options: allowImplicitOptions, errors: [{ messageId: "expectedInside", data: { name: "function" } }] },
177+
{ code: "foo.reduceRight(function() {})", options: allowImplicitOptions, errors: [{ messageId: "expectedInside", data: { name: "function" } }] },
178+
{ code: "foo.bar.baz.every(function foo() {})", options: allowImplicitOptions, errors: [{ messageId: "expectedInside", data: { name: "function 'foo'" } }] },
179+
{ code: "foo.every(cb || function() {})", options: allowImplicitOptions, errors: ["Expected to return a value in function."] },
180+
{ code: "[\"foo\",\"bar\"].sort(function foo() {})", options: allowImplicitOptions, errors: [{ messageId: "expectedInside", data: { name: "function 'foo'" } }] },
181+
{ code: "foo.forEach(x => x)", options: allowImplicitCheckForEach, parserOptions: { ecmaVersion: 6 }, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Arrow function" } }] },
182+
{ code: "foo.forEach(function(x) { if (a == b) {return x;}})", options: allowImplicitCheckForEach, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Function" } }] },
183+
{ code: "foo.forEach(function bar(x) { return x;})", options: allowImplicitCheckForEach, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Function 'bar'" } }] },
184+
185+
// // options: { checkForEach: true }
186+
{ code: "foo.forEach(x => x)", options: checkForEachOptions, parserOptions: { ecmaVersion: 6 }, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Arrow function" } }] },
187+
{ code: "foo.forEach(val => y += val)", options: checkForEachOptions, parserOptions: { ecmaVersion: 6 }, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Arrow function" } }] },
188+
{ code: "[\"foo\",\"bar\"].forEach(x => ++x)", options: checkForEachOptions, parserOptions: { ecmaVersion: 6 }, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Arrow function" } }] },
189+
{ code: "foo.bar().forEach(x => x === y)", options: checkForEachOptions, parserOptions: { ecmaVersion: 6 }, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Arrow function" } }] },
190+
{ code: "foo.forEach(function() {return function() { if (a == b) { return a; }}}())", options: checkForEachOptions, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Function" } }] },
191+
{ code: "foo.forEach(function(x) { if (a == b) {return x;}})", options: checkForEachOptions, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Function" } }] },
192+
{ code: "foo.forEach(function(x) { if (a == b) {return undefined;}})", options: checkForEachOptions, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Function" } }] },
193+
{ code: "foo.forEach(function bar(x) { return x;})", options: checkForEachOptions, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Function 'bar'" } }] },
194+
{ code: "foo.bar().forEach(function bar(x) { return x;})", options: checkForEachOptions, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Function 'bar'" } }] },
195+
{ code: "[\"foo\",\"bar\"].forEach(function bar(x) { return x;})", options: checkForEachOptions, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Function 'bar'" } }] },
196+
{ code: "foo.forEach((x) => { return x;})", options: checkForEachOptions, parserOptions: { ecmaVersion: 6 }, errors: [{ messageId: "expectedNoReturnValue", data: { name: "Arrow function" } }] },
197+
{ code: "Array.from(x, function() {})", options: checkForEachOptions, errors: [{ messageId: "expectedInside", data: { name: "function" } }] },
198+
{ code: "foo.every(function() {})", options: checkForEachOptions, errors: [{ messageId: "expectedInside", data: { name: "function" } }] },
199+
{ code: "foo.filter(function foo() {})", options: checkForEachOptions, errors: [{ messageId: "expectedInside", data: { name: "function 'foo'" } }] },
200+
{ code: "foo.filter(function foo() { return; })", options: checkForEachOptions, errors: [{ messageId: "expectedReturnValue", data: { name: "Function 'foo'" } }] },
201+
{ code: "foo.every(cb || function() {})", options: checkForEachOptions, errors: ["Expected to return a value in function."] }
202+
138203
]
139204
});

0 commit comments

Comments
 (0)
Please sign in to comment.