[react-packager] Fix more node_modules resolution rules

Summary:
@public
Fixes #773
This fixes `.json` name resolution. And also reads `package.json` when doing a directory module resolution.
The algorithm can be found here: https://nodejs.org/api/modules.html
I'll probably start including the node (or browserify) modules test in later diffs to make sure we're fully compliant.

Test Plan:
* ./runJestTests.sh
* ./runJestTests.sh PackagerIntegration
* open playground and require a json file
* test redbox
This commit is contained in:
Amjad Masad 2015-06-04 15:09:13 -07:00
parent 53b2c39cc0
commit 30fc7389d1
2 changed files with 80 additions and 24 deletions

View File

@ -168,9 +168,11 @@ describe('DependencyGraph', function() {
'/**', '/**',
' * @providesModule index', ' * @providesModule index',
' */', ' */',
'require("./a.json")' 'require("./a.json")',
'require("./b")'
].join('\n'), ].join('\n'),
'a.json': JSON.stringify({}), 'a.json': JSON.stringify({}),
'b.json': JSON.stringify({}),
} }
}); });
@ -186,7 +188,7 @@ describe('DependencyGraph', function() {
id: 'index', id: 'index',
altId: 'package/index', altId: 'package/index',
path: '/root/index.js', path: '/root/index.js',
dependencies: ['./a.json'], dependencies: ['./a.json', './b'],
isAsset: false, isAsset: false,
isAsset_DEPRECATED: false, isAsset_DEPRECATED: false,
isJSON: undefined, isJSON: undefined,
@ -205,6 +207,17 @@ describe('DependencyGraph', function() {
resolution: undefined, resolution: undefined,
resolveDependency: undefined, resolveDependency: undefined,
}, },
{
id: 'package/b.json',
isJSON: true,
path: '/root/b.json',
dependencies: [],
isAsset: false,
isAsset_DEPRECATED: false,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]); ]);
}); });
}); });
@ -851,6 +864,58 @@ describe('DependencyGraph', function() {
}); });
}); });
pit('should resolve require to main if it is a dir w/ a package.json', function() {
var root = '/root';
fs.__setMockFilesystem({
'root': {
'package.json': JSON.stringify({
name: 'test',
}),
'index.js': 'require("./lib/")',
lib: {
'package.json': JSON.stringify({
'main': 'main.js',
}),
'index.js': 'lol',
'main.js': 'lol',
},
}
});
var dgraph = new DependencyGraph({
roots: [root],
fileWatcher: fileWatcher,
assetExts: ['png', 'jpg'],
});
return dgraph.getOrderedDependencies('/root/index.js').then(function(deps) {
expect(getDataFromModules(deps))
.toEqual([
{
id: 'test/index',
path: '/root/index.js',
dependencies: ['./lib/'],
isAsset: false,
isAsset_DEPRECATED: false,
isJSON: undefined,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
{
id: '/root/lib/main.js',
path: '/root/lib/main.js',
dependencies: [],
isAsset: false,
isAsset_DEPRECATED: false,
isJSON: undefined,
isPolyfill: false,
resolution: undefined,
resolveDependency: undefined,
},
]);
});
});
pit('should ignore malformed packages', function() { pit('should ignore malformed packages', function() {
var root = '/root'; var root = '/root';
fs.__setMockFilesystem({ fs.__setMockFilesystem({

View File

@ -1,6 +1,3 @@
// TODO
// Fix it to work with tests
/** /**
* Copyright (c) 2015-present, Facebook, Inc. * Copyright (c) 2015-present, Facebook, Inc.
* All rights reserved. * All rights reserved.
@ -296,17 +293,19 @@ DependecyGraph.prototype.resolveDependency = function(
modulePath = path.join(dir, depModuleId); modulePath = path.join(dir, depModuleId);
modulePath = browserFieldRedirect(packageJson, modulePath); modulePath = browserFieldRedirect(packageJson, modulePath);
// JS modules can be required without extensios. dep = this._graph[modulePath] ||
if (!this._isFileAsset(modulePath) && !modulePath.match(/\.json$/)) { this._graph[modulePath + '.js'] ||
modulePath = withExtJs(modulePath); this._graph[modulePath + '.json'];
}
dep = this._graph[modulePath]; // Maybe the dependency is a directory and there is a packageJson and/or index.js inside it.
// Maybe the dependency is a directory and there is an index.js inside it.
if (dep == null) { if (dep == null) {
var dirPackageJson = this._packageByRoot[path.join(dir, depModuleId).replace(/\/$/, '')];
if (dirPackageJson) {
dep = this._resolvePackageMain(dirPackageJson);
} else {
dep = this._graph[path.join(dir, depModuleId, 'index.js')]; dep = this._graph[path.join(dir, depModuleId, 'index.js')];
} }
}
// Maybe it's an asset with @n.nx resolution and the path doesn't map // Maybe it's an asset with @n.nx resolution and the path doesn't map
// to the id // to the id
@ -444,14 +443,6 @@ DependecyGraph.prototype._processPackage = function(packagePath) {
return Promise.resolve(); return Promise.resolve();
} }
if (packageJson.name == null) {
debug(
'WARNING: package.json `%s` is missing a name field',
packagePath
);
return Promise.resolve();
}
packageJson._root = packageRoot; packageJson._root = packageRoot;
self._addPackageToIndices(packageJson); self._addPackageToIndices(packageJson);
@ -461,14 +452,14 @@ DependecyGraph.prototype._processPackage = function(packagePath) {
DependecyGraph.prototype._addPackageToIndices = function(packageJson) { DependecyGraph.prototype._addPackageToIndices = function(packageJson) {
this._packageByRoot[packageJson._root] = packageJson; this._packageByRoot[packageJson._root] = packageJson;
if (!this._isInNodeModules(packageJson._root)) { if (!this._isInNodeModules(packageJson._root) && packageJson.name != null) {
this._packagesById[packageJson.name] = packageJson; this._packagesById[packageJson.name] = packageJson;
} }
}; };
DependecyGraph.prototype._removePackageFromIndices = function(packageJson) { DependecyGraph.prototype._removePackageFromIndices = function(packageJson) {
delete this._packageByRoot[packageJson._root]; delete this._packageByRoot[packageJson._root];
if (!this._isInNodeModules(packageJson._root)) { if (!this._isInNodeModules(packageJson._root) && packageJson.name != null) {
delete this._packagesById[packageJson.name]; delete this._packagesById[packageJson.name];
} }
}; };
@ -536,7 +527,7 @@ DependecyGraph.prototype._processModule = function(modulePath) {
*/ */
DependecyGraph.prototype._lookupName = function(modulePath) { DependecyGraph.prototype._lookupName = function(modulePath) {
var packageJson = this._lookupPackage(modulePath); var packageJson = this._lookupPackage(modulePath);
if (packageJson == null) { if (packageJson == null || packageJson.name == null) {
return path.resolve(modulePath); return path.resolve(modulePath);
} else { } else {
var relativePath = var relativePath =