Skip to content

Commit 2539959

Browse files
committed
Add support for destructuring resolution.
Until now, the propTypeHandler would not be able to recognize that ``` var {PropTypes} = require('react'); ``` refers to the React module, and thus not recognize the types properly. This extends `resolveToValue` to return a MemberExpression if the path resolves to an ObjectPattern. I.e. `bar` in ``` var {foo: {bar}} = baz; bar; ``` would resolve to a path representing `baz.foo.bar`.
1 parent 648207d commit 2539959

File tree

5 files changed

+198
-4
lines changed

5 files changed

+198
-4
lines changed

src/handlers/__tests__/propTypeHandler-test.js

+5
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,7 @@ describe('propTypeHandler', () => {
6060
' propTypes: {',
6161
' foo: PropTypes.bool,',
6262
' bar: require("react").PropTypes.bool,',
63+
' baz: OtherPropTypes.bool,',
6364
' }',
6465
'})',
6566
].join('\n'));
@@ -74,6 +75,10 @@ describe('propTypeHandler', () => {
7475
bar: {
7576
type: {},
7677
required: false
78+
},
79+
baz: {
80+
type: {},
81+
required: false
7782
}
7883
});
7984
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
/*
2+
* Copyright (c) 2015, Facebook, Inc.
3+
* All rights reserved.
4+
*
5+
* This source code is licensed under the BSD-style license found in the
6+
* LICENSE file in the root directory of this source tree. An additional grant
7+
* of patent rights can be found in the PATENTS file in the same directory.
8+
*
9+
*/
10+
11+
/*global jest, describe, beforeEach, it, expect*/
12+
13+
"use strict";
14+
15+
jest.autoMockOff();
16+
17+
describe('resolveToModule', () => {
18+
var utils;
19+
var resolveToModule;
20+
21+
function parse(src) {
22+
var root = utils.parse(src);
23+
return root.get('body', root.node.body.length - 1, 'expression');
24+
}
25+
26+
beforeEach(() => {
27+
resolveToModule = require('../resolveToModule');
28+
utils = require('../../../tests/utils');
29+
});
30+
31+
it('resolves identifiers', () => {
32+
var path = parse([
33+
'var foo = require("Foo");',
34+
'foo;'
35+
].join('\n'));
36+
expect(resolveToModule(path)).toBe('Foo');
37+
});
38+
39+
it('resolves function calls', () => {
40+
var path = parse([
41+
'var foo = require("Foo");',
42+
'foo();'
43+
].join('\n'));
44+
expect(resolveToModule(path)).toBe('Foo');
45+
});
46+
47+
it('resolves member expressions', () => {
48+
var path = parse([
49+
'var foo = require("Foo");',
50+
'foo.bar().baz;'
51+
].join('\n'));
52+
expect(resolveToModule(path)).toBe('Foo');
53+
});
54+
55+
it('understands destructuring', () => {
56+
var path = parse([
57+
'var {foo} = require("Foo");',
58+
'foo;'
59+
].join('\n'));
60+
expect(resolveToModule(path)).toBe('Foo');
61+
});
62+
});
+89
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,89 @@
1+
/*
2+
* Copyright (c) 2015, Facebook, Inc.
3+
* All rights reserved.
4+
*
5+
* This source code is licensed under the BSD-style license found in the
6+
* LICENSE file in the root directory of this source tree. An additional grant
7+
* of patent rights can be found in the PATENTS file in the same directory.
8+
*
9+
*/
10+
11+
/*global jest, describe, beforeEach, it, expect*/
12+
13+
"use strict";
14+
15+
jest.autoMockOff();
16+
17+
describe('resolveToValue', () => {
18+
var astNodesAreEquivalent;
19+
var builders;
20+
var utils;
21+
var resolveToValue;
22+
23+
function parse(src) {
24+
var root = utils.parse(src);
25+
return root.get('body', root.node.body.length - 1, 'expression');
26+
}
27+
28+
beforeEach(() => {
29+
var recast = require('recast');
30+
astNodesAreEquivalent = recast.types.astNodesAreEquivalent;
31+
builders = recast.types.builders;
32+
resolveToValue = require('../resolveToValue');
33+
utils = require('../../../tests/utils');
34+
});
35+
36+
it('resolves simple variable declarations', () => {
37+
var path = parse([
38+
'var foo = 42;',
39+
'foo;'
40+
].join('\n'));
41+
expect(astNodesAreEquivalent(
42+
resolveToValue(path).node,
43+
builders.literal(42)
44+
)).toBe(true);
45+
});
46+
47+
it('resolves object destructuring', () => {
48+
var path = parse([
49+
'var {foo: {bar: baz}} = bar;',
50+
'baz;'
51+
].join('\n'));
52+
53+
// Node should be equal to bar.foo.bar
54+
expect(astNodesAreEquivalent(
55+
resolveToValue(path).node,
56+
builders.memberExpression(
57+
builders.memberExpression(
58+
builders.identifier('bar'),
59+
builders.identifier('foo')
60+
),
61+
builders.identifier('bar')
62+
)
63+
)).toBe(true);
64+
});
65+
66+
it('handles SpreadProperties properly', () => {
67+
var path = parse([
68+
'var {foo: {bar}, ...baz} = bar;',
69+
'baz;'
70+
].join('\n'));
71+
72+
expect(astNodesAreEquivalent(
73+
resolveToValue(path).node,
74+
path.node
75+
)).toBe(true);
76+
});
77+
78+
it('returns the original path if it cannot be resolved', () => {
79+
var path = parse([
80+
'function foo() {}',
81+
'foo()'
82+
].join('\n'));
83+
84+
expect(astNodesAreEquivalent(
85+
resolveToValue(path).node,
86+
path.node
87+
)).toBe(true);
88+
});
89+
});

src/utils/resolveToValue.js

+41-4
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,33 @@
1313
*/
1414
"use strict";
1515

16-
var types = require('recast').types.namedTypes;
16+
var {
17+
NodePath,
18+
builders,
19+
namedTypes: types
20+
} = require('recast').types;
21+
22+
function buildMemberExpressionFromPattern(path: NodePath): ?Node {
23+
var node = path.node;
24+
if (types.Property.check(node)) {
25+
var objPath = buildMemberExpressionFromPattern(path.parent);
26+
if (objPath) {
27+
return new NodePath(
28+
builders.memberExpression(
29+
objPath.node,
30+
node.key,
31+
types.Literal.check(node.key)
32+
),
33+
objPath
34+
);
35+
}
36+
} else if (types.ObjectPattern.check(node)) {
37+
return buildMemberExpressionFromPattern(path.parent);
38+
} else if (types.VariableDeclarator.check(node)) {
39+
return path.get('init');
40+
}
41+
return null;
42+
}
1743

1844
/**
1945
* If the path is an identifier, it is resolved in the scope chain.
@@ -32,11 +58,22 @@ function resolveToValue(path: NodePath): NodePath {
3258
if (scope) {
3359
var bindings = scope.getBindings()[node.name];
3460
if (bindings.length > 0) {
35-
var parentPath = scope.getBindings()[node.name][0].parent;
61+
var resultPath = scope.getBindings()[node.name][0];
62+
var parentPath = resultPath.parent;
3663
if (types.VariableDeclarator.check(parentPath.node)) {
37-
parentPath = parentPath.get('init');
64+
resultPath = parentPath.get('init');
65+
} else if (types.Property.check(parentPath.node)) {
66+
// must be inside a pattern
67+
var memberExpressionPath = buildMemberExpressionFromPattern(
68+
parentPath
69+
);
70+
if (memberExpressionPath) {
71+
return memberExpressionPath;
72+
}
3873
}
39-
return resolveToValue(parentPath);
74+
return resultPath.node !== path.node ?
75+
resolveToValue(resultPath) :
76+
path;
4077
}
4178
}
4279
}

tests/utils.js

+1
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@ function parseWithTemplate(src, template) {
3333
var REACT_TEMPLATE = [
3434
'var React = require("React");',
3535
'var PropTypes = React.PropTypes;',
36+
'var {PropTypes: OtherPropTypes} = require("React");',
3637
'%s;',
3738
].join('\n');
3839

0 commit comments

Comments
 (0)