Skip to content

Commit 33ee58f

Browse files
committed
support long property paths and conjunctions in prefer-filter, prefer-reject and prop-shorthand
1 parent 88c5d61 commit 33ee58f

14 files changed

+64
-57
lines changed

README.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -37,8 +37,8 @@ Finally, enable all of the rules that you would like to use.
3737
"lodash3/prefer-chain": 1,
3838
"lodash3/preferred-alias": 1,
3939
"lodash3/no-single-chain": 1,
40-
"lodash3/prefer-reject": 1,
41-
"lodash3/prefer-filter": 1,
40+
"lodash3/prefer-reject": [1,3],
41+
"lodash3/prefer-filter": [1,3],
4242
"lodash3/no-unnecessary-bind": 1,
4343
"lodash3/unwrap": 1,
4444
"lodash3/prefer-compact": 1,

docs/rules/matches-shorthand.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ When using certain method in lodash such as filter, code can be shorter and more
44

55
## Rule Details
66

7-
This rule takes one argument, maximum path length.
7+
This rule takes one argument, maximum path length (default is 3).
88

99
The following patterns are considered warnings:
1010

docs/rules/prefer-filter.md

+10-9
Original file line numberDiff line numberDiff line change
@@ -4,25 +4,25 @@ When using _.forEach with a single `if` statement, you should probably use `_.fi
44

55
## Rule Details
66

7-
This rule takes no arguments.
7+
This rule takes one argument, maximum path length (default is 3).
88

99
The following patterns are considered warnings:
1010

1111
```js
12-
13-
_(arr).forEach(function(x) {
14-
if (x.a) {
12+
_(users).forEach(function(user) {
13+
if (user.name.familyName) {
1514
// ...
1615
}
1716
});
1817

19-
if (!x.a) {
18+
_(users).forEach(function(user) {
19+
if (!user.active) {
2020
// ...
2121
}
2222
});
2323

24-
_.forEach(arr, function(x) {
25-
if (x.a === b) {
24+
_.forEach(users, function(user) {
25+
if (user.name.givenName === 'Bob') {
2626
// ...
2727
}
2828
});
@@ -31,7 +31,8 @@ _.forEach(arr, function(x) {
3131
The following patterns are not considered warnings:
3232

3333
```js
34-
35-
var x = _.filter(arr, function(x) {return !x.a && p});
34+
var x = _.filter(users, function(user) {
35+
return !user.active && user.name.givenName === 'Bob'
36+
});
3637

3738
```

docs/rules/prefer-matches.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ When writing an expression like `a.foo === 1 && a.bar === 2 && a.baz === 3`, it
44

55
## Rule Details
66

7-
This rule takes one argument - the minimal length of the condition(default is 3).
7+
This rule takes one argument - the minimal length of the condition (default is 3).
88

99
The following patterns are considered warnings:
1010

docs/rules/prefer-reject.md

+13-7
Original file line numberDiff line numberDiff line change
@@ -4,24 +4,30 @@ When using _.filter with a negative condition, it could improve readability by s
44

55
## Rule Details
66

7-
This rule takes no arguments.
7+
This rule takes one argument, maximum path length (default is 3).
88

99
The following patterns are considered warnings:
1010

1111
```js
12+
_.filter(users, function(user) {
13+
return user.name.givenName !== 'Bob';
14+
});
1215

13-
_.filter(arr, function(x) { return x.a !== b});
14-
15-
_.filter(arr, function(x) {return !x.isSomething})
16+
_.filter(users, function(user) {
17+
return !user.isSomething;
18+
});
1619
```
1720

1821
The following patterns are not considered warnings:
1922

2023
```js
24+
_.filter(users, function(user) {
25+
return !user.active && isSomething;
26+
});
2127

22-
var x = _.filter(arr, function(x) {return !x.a && p});
23-
24-
var x = _.filter(arr, function(x) {return !f(x)}; // The function f could take multiple arguments, e.g. parseInt
28+
_.filter(users, function(user) {
29+
return !f(user); // The function f could take multiple arguments, e.g. parseInt
30+
});
2531
```
2632

2733

docs/rules/prop-shorthand.md

+4-2
Original file line numberDiff line numberDiff line change
@@ -9,13 +9,15 @@ This rule takes no arguments.
99
The following patterns are considered warnings:
1010

1111
```js
12-
var ids = _.map([], function (i) { return i.id; });
12+
var ids = _.map([], function (user) {
13+
return user.name.familyName;
14+
});
1315
```
1416

1517
The following patterns are not considered warnings:
1618

1719
```js
18-
var ids = _.map([], 'id');
20+
var ids = _.map([], 'name.familyName');
1921
```
2022

2123

lib/rules/matches-prop-shorthand.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ module.exports = function (context) {
1515
var SUPPORT_MATCHES_PROPERTY_STYLE_CB = ['find', 'detect', 'filter', 'select', 'reject', 'findIndex', 'findLastIndex', 'some', 'every'];
1616

1717
function shouldPreferMatches(func) {
18-
return astUtil.isEqEqEqToParamMember(1, astUtil.getValueReturnedInFirstLine(func), astUtil.getFirstParamName(func));
18+
return astUtil.isEqEqEqToParamMember(astUtil.getValueReturnedInFirstLine(func), astUtil.getFirstParamName(func), 1);
1919
}
2020

2121
function methodSupportsShorthand(node) {

lib/rules/prefer-filter.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,9 @@ module.exports = function (context) {
1919
}
2020

2121
function canBeShorthand(exp, paramName) {
22-
return astUtil.isIdentifierOfParam(exp, paramName) || astUtil.isMemberExpOfArg(exp, paramName) || astUtil.isNegationOfParamMember(exp, paramName) ||
23-
astUtil.isEqEqEqToParamMember(maxPropertyPathLength, exp, paramName) || astUtil.isNotEqEqToParamMember(exp, paramName);
22+
return astUtil.isIdentifierOfParam(exp, paramName) ||
23+
astUtil.isMemberExpOfArg(exp, paramName, maxPropertyPathLength) || astUtil.isNegationOfParamMember(exp, paramName, maxPropertyPathLength) ||
24+
astUtil.isEqEqEqToParamMember(exp, paramName, maxPropertyPathLength) || astUtil.isNotEqEqToParamMember(exp, paramName, maxPropertyPathLength);
2425
}
2526

2627
function onlyHasSimplifiableIf(func) {

lib/rules/prefer-reject.js

+5-2
Original file line numberDiff line numberDiff line change
@@ -11,11 +11,14 @@ module.exports = function (context) {
1111
var lodashUtil = require('../util/lodashUtil');
1212
var astUtil = require('../util/astUtil');
1313

14+
var DEFAULT_MAX_PROPERTY_PATH_LENGTH = 3;
15+
var maxPropertyPathLength = parseInt(context.options[0], 10) || DEFAULT_MAX_PROPERTY_PATH_LENGTH;
16+
1417
function isNegativeExpressionFunction(func) {
1518
var returnValue = astUtil.getValueReturnedInFirstLine(func);
1619
var firstParamName = astUtil.getFirstParamName(func);
17-
return astUtil.isNegationOfParamMember(returnValue, firstParamName) ||
18-
astUtil.isNotEqEqToParamMember(returnValue, firstParamName);
20+
return astUtil.isNegationOfParamMember(returnValue, firstParamName, maxPropertyPathLength) ||
21+
astUtil.isNotEqEqToParamMember(returnValue, firstParamName, maxPropertyPathLength);
1922
}
2023

2124
return {

lib/rules/prop-shorthand.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ module.exports = function (context) {
1414
var astUtil = require('../util/astUtil');
1515

1616
function canUseShorthand(func) {
17-
return astUtil.isMemberExpOfArg(astUtil.getValueReturnedInFirstLine(func), astUtil.getFirstParamName(func));
17+
return astUtil.isMemberExpOfArg(astUtil.getValueReturnedInFirstLine(func), astUtil.getFirstParamName(func), Number.MAX_VALUE);
1818
}
1919

2020
function methodSupportsShorthand(node) {

lib/util/astUtil.js

+8-12
Original file line numberDiff line numberDiff line change
@@ -35,13 +35,9 @@ function isPropAccess(node) {
3535
(node.computed === true && node.property.type === 'Literal' && _.isString(node.property.value));
3636
}
3737

38-
function isMemberExpOfArg(node, argName) {
39-
return argName && _.get(node, 'type') === 'MemberExpression' && _.get(node, 'object.name') === argName && isPropAccess(node);
40-
}
41-
42-
function isNestedMemberExpOfArg(node, argName, maxPropertyPathLength) {
38+
function isMemberExpOfArg(node, argName, maxPropertyPathLength) {
4339
if (maxPropertyPathLength > 0 && argName && node && node.object && node.type === 'MemberExpression' && isPropAccess(node)) {
44-
return node.object.name === argName || isNestedMemberExpOfArg(node.object, argName, maxPropertyPathLength - 1);
40+
return node.object.name === argName || isMemberExpOfArg(node.object, argName, maxPropertyPathLength - 1);
4541
}
4642
return false;
4743
}
@@ -62,25 +58,25 @@ function isObjectOfMethodCall(node) {
6258
return _.get(node, 'parent.object') === node && _.get(node, 'parent.parent.type') === 'CallExpression';
6359
}
6460

65-
function isBinaryExpWithParamMember(operator, maxPropertyPathLength, exp, paramName) {
61+
function isBinaryExpWithParamMember(operator, exp, paramName, maxPropertyPathLength) {
6662
return exp && exp.type === 'BinaryExpression' && exp.operator === operator &&
67-
(isNestedMemberExpOfArg(exp.left, paramName, maxPropertyPathLength) || isNestedMemberExpOfArg(exp.right, paramName, maxPropertyPathLength));
63+
(isMemberExpOfArg(exp.left, paramName, maxPropertyPathLength) || isMemberExpOfArg(exp.right, paramName, maxPropertyPathLength));
6864
}
6965

7066
function isConjunctionOfEqEqEqToParamMember(exp, paramName, maxPropertyPathLength) {
7167
if (exp && exp.type === 'LogicalExpression' && exp.operator === '&&') {
7268
return isConjunctionOfEqEqEqToParamMember(exp.left, paramName, maxPropertyPathLength) &&
7369
isConjunctionOfEqEqEqToParamMember(exp.right, paramName, maxPropertyPathLength);
7470
}
75-
return isBinaryExpWithParamMember('===', maxPropertyPathLength, exp, paramName);
71+
return isBinaryExpWithParamMember('===', exp, paramName, maxPropertyPathLength);
7672
}
7773

7874
function isNegationExpression(exp) {
7975
return exp && exp.type === 'UnaryExpression' && exp.operator === '!';
8076
}
8177

82-
function isNegationOfParamMember(exp, firstParamName) {
83-
return isNegationExpression(exp) && isMemberExpOfArg(exp.argument, firstParamName);
78+
function isNegationOfParamMember(exp, firstParamName, maxPropertyPathLength) {
79+
return isNegationExpression(exp) && isMemberExpOfArg(exp.argument, firstParamName, maxPropertyPathLength);
8480
}
8581

8682
function isIdentifierOfParam(exp, paramName) {
@@ -139,7 +135,7 @@ module.exports = {
139135
hasOnlyOneStatement: hasOnlyOneStatement,
140136
isObjectOfMethodCall: isObjectOfMethodCall,
141137
isEqEqEqToParamMember: isBinaryExpWithParamMember.bind(null, '==='),
142-
isNotEqEqToParamMember: isBinaryExpWithParamMember.bind(null, '!==', 1),
138+
isNotEqEqToParamMember: isBinaryExpWithParamMember.bind(null, '!=='),
143139
isNegationOfParamMember: isNegationOfParamMember,
144140
isIdentifierOfParam: isIdentifierOfParam,
145141
isNegationExpression: isNegationExpression,

tests/lib/rules/prefer-filter.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -21,19 +21,19 @@ ruleTester.run('prefer-filter', rule, {
2121
'_.forEach(arr, function(x, y) { if (x){} })'
2222
],
2323
invalid: [{
24-
code: '_(arr).forEach(function(x) { if (x.a) {}})',
24+
code: '_(arr).forEach(function(x) { if (x.a.b.c) {}})',
2525
errors: [ruleError]
2626
}, {
2727
code: '_(arr).forEach(function(x) { if (x) {}})',
2828
errors: [ruleError]
2929
}, {
30-
code: '_.forEach(arr, function(x) { if (x.a === b) {}})',
30+
code: '_.forEach(arr, function(x) { if (x.a.b.c === d) {}})',
3131
errors: [ruleError]
3232
}, {
33-
code: '_.forEach(arr, function(x) { if (x.a !== b) {}})',
33+
code: '_.forEach(arr, function(x) { if (x.a.b.c !== d) {}})',
3434
errors: [ruleError]
3535
}, {
36-
code: '_.forEach(arr, function(x) { if (!x.a) {}})',
36+
code: '_.forEach(arr, function(x) { if (!x.a.b.c) {}})',
3737
errors: [ruleError]
3838
}]
3939
});

tests/lib/rules/prefer-reject.js

+6-8
Original file line numberDiff line numberDiff line change
@@ -15,22 +15,20 @@ var ruleTester = new RuleTester();
1515
var ruleError = {message: 'Prefer _.reject over negative condition'};
1616
ruleTester.run('prefer-reject', rule, {
1717
valid: [
18-
'var x = _.filter(arr, function(x) {return !x.a && p})'
18+
'_.filter(users, function(user) {return !user.active && isSomething;});',
19+
'_.filter(users, function(user) {return !f(user);});'
1920
],
2021
invalid: [{
21-
code: '_(arr).map(f).filter(function(x) {return !x.isSomething})',
22+
code: '_(users).map(t).filter(function(user) {return !user.name.givenName})',
2223
errors: [ruleError]
2324
}, {
24-
code: '_.filter(arr, function(x) { return x.a !== b})',
25+
code: '_.filter(users, function(user) {return user.name.givenName !== "Bob";});',
2526
errors: [ruleError]
2627
}, {
27-
code: '_.filter(arr, function(x) { return b !== x.a})',
28+
code: '_.filter(users, function(user) {return !user.isSomething;});',
2829
errors: [ruleError]
2930
}, {
30-
code: '_.filter(arr, function(x) {return !x.isSomething})',
31-
errors: [ruleError]
32-
}, {
33-
code: '_.filter(arr, x => !x.isSomething)',
31+
code: '_.filter(arr, user => !user.active)',
3432
ecmaFeatures: {arrowFunctions: true},
3533
errors: [ruleError]
3634
}]

tests/lib/rules/prop-shorthand.js

+5-5
Original file line numberDiff line numberDiff line change
@@ -26,19 +26,19 @@ ruleTester.run('prop-shorthand', rule, {
2626
'var r = _.map([])'
2727
],
2828
invalid: [{
29-
code: 'var ids = _(users).map(function (i) { return i.id; });',
29+
code: 'var ids = _(arr).map(function (i) { return i.a.b.c; });',
3030
errors: errors
3131
}, {
32-
code: 'var ids = _.map([], function (i) { return i.id; });',
32+
code: 'var ids = _.map([], function (i) { return i.a; });',
3333
errors: errors
3434
}, {
35-
code: 'var ids = _(users).map("x").map("y").map(function (i) { return i.id; });',
35+
code: 'var ids = _(arr).map("x").map("y").map(function (i) { return i.a.b; });',
3636
errors: errors
3737
}, {
38-
code: 'var ids = _.map([], function (i) { return i["id"]; });',
38+
code: 'var ids = _.map([], function (i) { return i["a"]; });',
3939
errors: errors
4040
}, {
41-
code: 'var ids = _.map([], i => i.id);',
41+
code: 'var ids = _.map([], i => i.a.b.c);',
4242
ecmaFeatures: {arrowFunctions: true},
4343
errors: errors
4444
}]

0 commit comments

Comments
 (0)